diff --git a/.nojekyll b/.nojekyll new file mode 100644 index 00000000000..e69de29bb2d diff --git a/404.html b/404.html new file mode 100644 index 00000000000..56a9d1d9739 --- /dev/null +++ b/404.html @@ -0,0 +1,41 @@ + + + + + + 404 | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + +
Skip to content

404

PAGE NOT FOUND

But if you don't change your direction, and if you keep looking, you may end up where you are heading.
+ + + + \ No newline at end of file diff --git a/CNAME b/CNAME new file mode 100644 index 00000000000..439f8e04848 --- /dev/null +++ b/CNAME @@ -0,0 +1 @@ +docs.celestia.org diff --git a/Celestia-og.png b/Celestia-og.png new file mode 100644 index 00000000000..1ecc736c8b8 Binary files /dev/null and b/Celestia-og.png differ diff --git a/README.html b/README.html new file mode 100644 index 00000000000..1683a797742 --- /dev/null +++ b/README.html @@ -0,0 +1,45 @@ + + + + + + Celestia Documentation Site | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Skip to content

Celestia Documentation Site

Welcome to the official documentation repository for Celestia.

Here you'll find comprehensive guides, tutorials, and reference materials to help you make the most out of Celestia.

Building the site

To get started, clone the repository and run the following:

bash
yarn && yarn dev
yarn && yarn dev

This documentation site is built with VitePress

Contribution Guidelines

We love contributions from the community! Whether you're fixing typos, improving content clarity, or adding new topics, every contribution helps.

  • Fork & Clone: Fork this repository and clone it to your local machine.
  • Branch: Always create a new branch for your changes. Naming it relevantly.
  • Commit Changes: Make your changes and commit them with a clear and concise commit message.
  • Push & Create PR: Push your changes to your fork and create a pull request to the main branch of this repository.

Please ensure to review the detailed Contribution Guidelines above before making a pull request.

Directory Structure

  • /learn: A category for learning about Celestia.
  • /developers: A category with tutorials and guides for deploying rollups and interacting with Celestia nodes.
  • /nodes: Technical reference materials for all node types.
  • /community: A category for the Celestia community.
  • /public: Images, diagrams, and other media files used in the documentation.

Feedback & Suggestions

We value feedback from the community. If you have suggestions for improvements or find any discrepancies in the documentation, please raise an issue in this repository.

+ + + + \ No newline at end of file diff --git a/arbitrum/add-custom-chain-to-bridge.png b/arbitrum/add-custom-chain-to-bridge.png new file mode 100644 index 00000000000..ceddb937678 Binary files /dev/null and b/arbitrum/add-custom-chain-to-bridge.png differ diff --git a/arbitrum/blockscout.png b/arbitrum/blockscout.png new file mode 100644 index 00000000000..30920eacc63 Binary files /dev/null and b/arbitrum/blockscout.png differ diff --git a/arbitrum/bridge-in-explorer-rollup-tx.png b/arbitrum/bridge-in-explorer-rollup-tx.png new file mode 100644 index 00000000000..2acd573ce31 Binary files /dev/null and b/arbitrum/bridge-in-explorer-rollup-tx.png differ diff --git a/arbitrum/bridge-in-pending-txs.png b/arbitrum/bridge-in-pending-txs.png new file mode 100644 index 00000000000..8feda115a65 Binary files /dev/null and b/arbitrum/bridge-in-pending-txs.png differ diff --git a/arbitrum/bridge-in-sepolia-tx-explorer.png b/arbitrum/bridge-in-sepolia-tx-explorer.png new file mode 100644 index 00000000000..f15ce456cf5 Binary files /dev/null and b/arbitrum/bridge-in-sepolia-tx-explorer.png differ diff --git a/arbitrum/bridge-in-settled-txs.png b/arbitrum/bridge-in-settled-txs.png new file mode 100644 index 00000000000..61f21731e1c Binary files /dev/null and b/arbitrum/bridge-in-settled-txs.png differ diff --git a/arbitrum/bridge-in-start.png b/arbitrum/bridge-in-start.png new file mode 100644 index 00000000000..6a960e4a1af Binary files /dev/null and b/arbitrum/bridge-in-start.png differ diff --git a/arbitrum/bridge-in-success.png b/arbitrum/bridge-in-success.png new file mode 100644 index 00000000000..ce679715af2 Binary files /dev/null and b/arbitrum/bridge-in-success.png differ diff --git a/arbitrum/bridge-out-begin-overview.png b/arbitrum/bridge-out-begin-overview.png new file mode 100644 index 00000000000..74e1763c02f Binary files /dev/null and b/arbitrum/bridge-out-begin-overview.png differ diff --git a/arbitrum/bridge-out-begin.png b/arbitrum/bridge-out-begin.png new file mode 100644 index 00000000000..da365b6c735 Binary files /dev/null and b/arbitrum/bridge-out-begin.png differ diff --git a/arbitrum/bridge-out-claim-success-withdrawal.png b/arbitrum/bridge-out-claim-success-withdrawal.png new file mode 100644 index 00000000000..63291a49313 Binary files /dev/null and b/arbitrum/bridge-out-claim-success-withdrawal.png differ diff --git a/arbitrum/bridge-out-claim-withdrawal.png b/arbitrum/bridge-out-claim-withdrawal.png new file mode 100644 index 00000000000..3a7f7b0ac93 Binary files /dev/null and b/arbitrum/bridge-out-claim-withdrawal.png differ diff --git a/arbitrum/bridge-out-logs-details-1.png b/arbitrum/bridge-out-logs-details-1.png new file mode 100644 index 00000000000..a07a7434ef5 Binary files /dev/null and b/arbitrum/bridge-out-logs-details-1.png differ diff --git a/arbitrum/bridge-out-logs-explorer-2.png b/arbitrum/bridge-out-logs-explorer-2.png new file mode 100644 index 00000000000..77950e6735c Binary files /dev/null and b/arbitrum/bridge-out-logs-explorer-2.png differ diff --git a/arbitrum/bridge-out-pending.png b/arbitrum/bridge-out-pending.png new file mode 100644 index 00000000000..78b0639af17 Binary files /dev/null and b/arbitrum/bridge-out-pending.png differ diff --git a/arbitrum/bridge-out-rollup-tx-details.png b/arbitrum/bridge-out-rollup-tx-details.png new file mode 100644 index 00000000000..109161bd218 Binary files /dev/null and b/arbitrum/bridge-out-rollup-tx-details.png differ diff --git a/arbitrum/bridge-out-small-screenshot.png b/arbitrum/bridge-out-small-screenshot.png new file mode 100644 index 00000000000..00cf76ce4ab Binary files /dev/null and b/arbitrum/bridge-out-small-screenshot.png differ diff --git a/arbitrum/bridge-overview-deposit-and-withdrawal-l3.png b/arbitrum/bridge-overview-deposit-and-withdrawal-l3.png new file mode 100644 index 00000000000..b175e6448bb Binary files /dev/null and b/arbitrum/bridge-overview-deposit-and-withdrawal-l3.png differ diff --git a/arbitrum/bridge-settings.png b/arbitrum/bridge-settings.png new file mode 100644 index 00000000000..d99a161e30a Binary files /dev/null and b/arbitrum/bridge-settings.png differ diff --git a/arbitrum/choose_da.png b/arbitrum/choose_da.png new file mode 100644 index 00000000000..c2ba7c9b667 Binary files /dev/null and b/arbitrum/choose_da.png differ diff --git a/arbitrum/configuration.png b/arbitrum/configuration.png new file mode 100644 index 00000000000..c664b6f6f52 Binary files /dev/null and b/arbitrum/configuration.png differ diff --git a/arbitrum/download-config.png b/arbitrum/download-config.png new file mode 100644 index 00000000000..4066d300976 Binary files /dev/null and b/arbitrum/download-config.png differ diff --git a/arbitrum/explorer-view.png b/arbitrum/explorer-view.png new file mode 100644 index 00000000000..b1bd445d379 Binary files /dev/null and b/arbitrum/explorer-view.png differ diff --git a/arbitrum/live-orbit-chains.png b/arbitrum/live-orbit-chains.png new file mode 100644 index 00000000000..c1f7e87af6a Binary files /dev/null and b/arbitrum/live-orbit-chains.png differ diff --git a/assets/README.md.73690789.js b/assets/README.md.73690789.js new file mode 100644 index 00000000000..f5e603406cc --- /dev/null +++ b/assets/README.md.73690789.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"Celestia Documentation Site","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Celestia Documentation Site | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"README.md","filePath":"README.md","lastUpdated":1698682354000}'),o={name:"README.md"},n=i('

Celestia Documentation Site

Welcome to the official documentation repository for Celestia.

Here you'll find comprehensive guides, tutorials, and reference materials to help you make the most out of Celestia.

Building the site

To get started, clone the repository and run the following:

bash
yarn && yarn dev
yarn && yarn dev

This documentation site is built with VitePress

Contribution Guidelines

We love contributions from the community! Whether you're fixing typos, improving content clarity, or adding new topics, every contribution helps.

Please ensure to review the detailed Contribution Guidelines above before making a pull request.

Directory Structure

Feedback & Suggestions

We value feedback from the community. If you have suggestions for improvements or find any discrepancies in the documentation, please raise an issue in this repository.

',16),s=[n];function r(l,c,u,d,p,h){return t(),a("div",null,s)}const y=e(o,[["render",r]]);export{g as __pageData,y as default}; diff --git a/assets/README.md.73690789.lean.js b/assets/README.md.73690789.lean.js new file mode 100644 index 00000000000..2cdc64917bf --- /dev/null +++ b/assets/README.md.73690789.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"Celestia Documentation Site","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Celestia Documentation Site | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"README.md","filePath":"README.md","lastUpdated":1698682354000}'),o={name:"README.md"},n=i("",16),s=[n];function r(l,c,u,d,p,h){return t(),a("div",null,s)}const y=e(o,[["render",r]]);export{g as __pageData,y as default}; diff --git a/assets/app.a7ebcf51.js b/assets/app.a7ebcf51.js new file mode 100644 index 00000000000..ca5486b140c --- /dev/null +++ b/assets/app.a7ebcf51.js @@ -0,0 +1 @@ +import{a3 as i,s,a4 as c,a5 as l,a6 as d,a7 as f,a8 as m,a9 as h,aa as A,ab as g,ac as y,ad as P,X as v,d as w,u as C,j as R,y as _,ae as b,af as D,ag as E}from"./chunks/framework.51d6c45b.js";import{t as p}from"./chunks/theme.6bee4cb8.js";const L={extends:p,Layout:()=>i(p.Layout,null,{}),enhanceApp({app:e,router:a,siteData:t}){}};function u(e){if(e.extends){const a=u(e.extends);return{...a,...e,async enhanceApp(t){a.enhanceApp&&await a.enhanceApp(t),e.enhanceApp&&await e.enhanceApp(t)}}}return e}const o=u(L),T=w({name:"VitePressApp",setup(){const{site:e}=C();return R(()=>{_(()=>{document.documentElement.lang=e.value.lang,document.documentElement.dir=e.value.dir})}),b(),D(),E(),o.setup&&o.setup(),()=>i(o.Layout)}});async function j(){const e=O(),a=x();a.provide(l,e);const t=d(e.route);return a.provide(f,t),a.component("Content",m),a.component("ClientOnly",h),Object.defineProperties(a.config.globalProperties,{$frontmatter:{get(){return t.frontmatter.value}},$params:{get(){return t.page.value.params}}}),o.enhanceApp&&await o.enhanceApp({app:a,router:e,siteData:A}),{app:a,router:e,data:t}}function x(){return g(T)}function O(){let e=s,a;return y(t=>{let n=P(t),r=null;return n&&(e&&(a=n),(e||a===n)&&(n=n.replace(/\.js$/,".lean.js")),r=v(()=>import(n),[])),s&&(e=!1),r},o.NotFound)}s&&j().then(({app:e,router:a,data:t})=>{a.go().then(()=>{c(a.route,t.site),e.mount("#app")})});export{j as createApp}; diff --git a/assets/chunks/@localSearchIndexroot.00d83d1a.js b/assets/chunks/@localSearchIndexroot.00d83d1a.js new file mode 100644 index 00000000000..aff11484a9f --- /dev/null +++ b/assets/chunks/@localSearchIndexroot.00d83d1a.js @@ -0,0 +1 @@ +const e='{"documentCount":731,"nextId":731,"documentIds":{"0":"/README#celestia-documentation-site","1":"/README#building-the-site","2":"/README#contribution-guidelines","3":"/README#directory-structure","4":"/README#feedback-suggestions","5":"/community/calendar#community-calendar","6":"/community/coc#celestia-org-code-of-conduct","7":"/community/coc#our-pledge","8":"/community/coc#our-standards","9":"/community/coc#enforcement-responsibilities","10":"/community/coc#scope","11":"/community/coc#enforcement","12":"/community/coc#enforcement-guidelines","13":"/community/coc#_1-correction","14":"/community/coc#_2-warning","15":"/community/coc#_3-temporary-ban","16":"/community/coc#_4-permanent-ban","17":"/community/coc#attribution","18":"/community/foundation-delegation-program#the-celestia-foundation-delegation-program","19":"/community/foundation-delegation-program#objectives-of-the-program","20":"/community/foundation-delegation-program#foundation-delegation-process","21":"/community/foundation-delegation-program#program-launch","22":"/community/foundation-delegation-program#cohort-process","23":"/community/foundation-delegation-program#key-points","24":"/community/foundation-delegation-program#eligibility-criteria","25":"/community/foundation-delegation-program#undelegation-criteria","26":"/community/foundation-delegation-program#application","27":"/community/foundation-delegation-program#application-details","28":"/community/foundation-delegation-program#cohort-information","29":"/community/foundation-delegation-program#feedback-process","30":"/community/itn-tos#supplemental-incentivized-testnet-terms","31":"/community/itn-tos#_1-general-terms","32":"/community/itn-tos#_2-taxes","33":"/community/itn-tos#_3-supplemental-privacy-information","34":"/community/itn-tos#_4-certain-additional-representations","35":"/community/modular-meetup-guide#modular-meetup-guide","36":"/community/modular-meetup-guide#before-the-meetup","37":"/community/modular-meetup-guide#during-the-meetup","38":"/community/modular-meetup-guide#after-the-meetup","39":"/community/modular-meetup-guide#logistics-and-guidance","40":"/community/modular-meetup-guide#venue","41":"/community/modular-meetup-guide#catering-and-refreshments","42":"/community/modular-meetup-guide#audience","43":"/community/modular-meetup-guide#speakers","44":"/community/modular-meetup-guide#sponsors","45":"/community/modular-meetup-guide#communications-and-marketing","46":"/community/modular-meetup-guide#recording","47":"/community/modular-meetup-guide#utilizing-meetup-com-platform","48":"/community/modular-meetup-guide#onboarding-questions-for-community-members-joining-a-modular-meetup","49":"/community/modular-meetup-intro#celestia-modular-meetup-program","50":"/community/modular-meetup-intro#program-description","51":"/community/modular-meetup-intro#important-info","52":"/community/modular-meetup-intro#celestia-org-community-code-of-conduct","53":"/community/modular-meetup-intro#signup-form","54":"/community/modular-meetup-intro#emails","55":"/community/modular-meetup-intro#discord","56":"/community/modular-meetup-intro#materials","57":"/community/modular-meetup-toolkit#modular-meetup-toolkit","58":"/community/modular-meetup-toolkit#celestia-branding-guidelines","59":"/community/modular-meetup-toolkit#sample-introduction-to-modularity-workshop-presentation","60":"/community/modular-meetup-toolkit#sample-run-a-celestia-light-node-workshop-presentation","61":"/community/modular-meetup-toolkit#sample-deploy-a-sovereign-rollup-workshop-presentation","62":"/community/modular-meetup-toolkit#sample-modular-meetup-introduction-workshop-presentation","63":"/community/modular-meetup-toolkit#swag-logistics","64":"/community/overview#community-overview","65":"/community/speaker-list#speaker-list","66":"/developers/arbitrum-bridge#bridging-in-and-out-of-your-orbit-rollup","67":"/developers/arbitrum-bridge#bridge-in-deposit-to-your-rollup","68":"/developers/arbitrum-bridge#step-1-add-your-custom-chain-config","69":"/developers/arbitrum-bridge#step-2-deposit-to-your-orbit-rollup","70":"/developers/arbitrum-bridge#bridge-out-withdrawal-from-your-rollup","71":"/developers/arbitrum-bridge#step-1-choose-an-amount-to-withdraw-from-your-rollup","72":"/developers/arbitrum-bridge#step-2-claim-your-withdrawal","73":"/developers/arbitrum-dapp-deploy#deploy-a-dapp-on-your-arbitrum-rollup-devnet","74":"/developers/arbitrum-dapp-deploy#dependencies","75":"/developers/arbitrum-dapp-deploy#setup-and-contract-deployment","76":"/developers/arbitrum-dapp-deploy#interact-with-the-contract","77":"/developers/arbitrum-dapp-deploy#update-the-frontend","78":"/developers/arbitrum-dapp-deploy#interact-with-the-frontend","79":"/developers/arbitrum-deploy#quickstart-deploy-an-arbitrum-orbit-rollup","80":"/developers/arbitrum-deploy#prerequisites","81":"/developers/arbitrum-deploy#setup","82":"/developers/arbitrum-deploy#step-1-acquire-arbitrum-sepolia-eth","83":"/developers/arbitrum-deploy#step-2-pick-your-deployment-type","84":"/developers/arbitrum-deploy#step-3-configure-your-orbit-chain-s-deployment","85":"/developers/arbitrum-deploy#step-3-review-deploy-your-orbit-chain","86":"/developers/arbitrum-deploy#step-4-download-your-chain-s-configuration-files-and-launch-your-chain","87":"/developers/arbitrum-deploy#step-5-clone-the-setup-script-repository-and-add-your-configuration-files","88":"/developers/arbitrum-deploy#step-6-pick-an-l2-rpc-url-for-the-batch-poster","89":"/developers/arbitrum-deploy#step-7-run-your-light-node-for-mocha-testnet","90":"/developers/arbitrum-deploy#step-8-run-your-chain-s-node-and-block-explorer","91":"/developers/arbitrum-deploy#step-9-finish-setting-up-your-chain","92":"/developers/arbitrum-deploy#congratulations-with-celestia-underneath","93":"/developers/arbitrum-deploy#appendix","94":"/developers/arbitrum-deploy#compatibility-matrix","95":"/developers/arbitrum-deploy#blobstream-x-contract-deployments","96":"/developers/arbitrum-deploy#arbitrum-sepolia","97":"/developers/arbitrum-deploy#base-sepolia","98":"/developers/arbitrum-deploy#arbitrum-sepolia-additional-deployments","99":"/developers/arbitrum-deploy#base-sepolia-additional-deployments","100":"/developers/arbitrum-full-node#running-a-full-node-and-or-validator","101":"/developers/arbitrum-full-node#prerequisites","102":"/developers/arbitrum-full-node#running-a-full-node","103":"/developers/arbitrum-full-node#running-a-full-node-with-validation","104":"/developers/arbitrum-integration#introduction-to-arbitrum-rollups-with-celestia-as-da","105":"/developers/arbitrum-integration#overview","106":"/developers/arbitrum-integration#key-components","107":"/developers/arbitrum-integration#da-provider-implementation","108":"/developers/arbitrum-integration#preimage-oracle-implementation","109":"/developers/arbitrum-integration#blobstream-x-implementation","110":"/developers/arbitrum-integration#ethereum-fallback-mechanism-in-nitro","111":"/developers/arbitrum-integration#next-steps","112":"/developers/arbitrum-smart-contract#deploy-a-smart-contract-to-your-arbitrum-rollup","113":"/developers/arbitrum-smart-contract#overview","114":"/developers/arbitrum-smart-contract#prerequisites","115":"/developers/arbitrum-smart-contract#setup","116":"/developers/arbitrum-smart-contract#create-your-smart-contract","117":"/developers/arbitrum-smart-contract#test-your-smart-contract","118":"/developers/arbitrum-smart-contract#deploying-your-smart-contract","119":"/developers/arbitrum-smart-contract#funded-accounts","120":"/developers/arbitrum-smart-contract#using-our-arbitrum-devnet","121":"/developers/arbitrum-smart-contract#interacting-with-your-smart-contract","122":"/developers/arbitrum-smart-contract#next-steps","123":"/developers/blobstream-contracts#integrate-with-blobstream-contracts","124":"/developers/blobstream-contracts#getting-started","125":"/developers/blobstream-contracts#prerequisites","126":"/developers/blobstream-contracts#installing-blobstream-x-contracts","127":"/developers/blobstream-contracts#example-usage","128":"/developers/blobstream-contracts#data-structures","129":"/developers/blobstream-contracts#interface","130":"/developers/blobstream-contracts#querying-the-proof","131":"/developers/blobstream-contracts#verifying-data-inclusion-for-fraud-proofs","132":"/developers/blobstream-offchain#integrate-with-blobstream-client","133":"/developers/blobstream-offchain#blobstream-demo-rollup","134":"/developers/blobstream-offchain#defining-a-chain","135":"/developers/blobstream-offchain#rollup-sequencer","136":"/developers/blobstream-offchain#committing-to-data","137":"/developers/blobstream-offchain#creating-blocks","138":"/developers/blobstream-offchain#rollup-full-node","139":"/developers/blobstream-offchain#downloading-the-block","140":"/developers/blobstream-offchain#more-documentation","141":"/developers/blobstream-offchain#proving-inclusion-via-blobstream","142":"/developers/blobstream-offchain#submitting-block-data-to-celestia-via-light-node","143":"/developers/blobstream-offchain#posting-headers-to-ethereum","144":"/developers/blobstream-proof-queries#blobstream-proofs-queries","145":"/developers/blobstream-proof-queries#prerequisites","146":"/developers/blobstream-proof-queries#overview-of-the-proof-queries","147":"/developers/blobstream-proof-queries#the-celestia-square","148":"/developers/blobstream-proof-queries#the-commitment-scheme","149":"/developers/blobstream-proof-queries#hands-on-demonstration","150":"/developers/blobstream-proof-queries#_1-data-root-inclusion-proof","151":"/developers/blobstream-proof-queries#full-example-of-proving-that-a-celestia-block-was-committed-to-by-blobstream-x-contract","152":"/developers/blobstream-proof-queries#_2-transaction-inclusion-proof","153":"/developers/blobstream-proof-queries#converting-the-proofs-to-be-usable-in-the-daverifier-library","154":"/developers/blobstream-proof-queries#data","155":"/developers/blobstream-proof-queries#shareproofs","156":"/developers/blobstream-proof-queries#namespace","157":"/developers/blobstream-proof-queries#rowroots","158":"/developers/blobstream-proof-queries#rowproofs","159":"/developers/blobstream-proof-queries#attestationproof","160":"/developers/blobstream-proof-queries#listening-for-new-data-commitments","161":"/developers/blobstream-proof-queries#example-rollup-that-uses-the-daverifier","162":"/developers/blobstream-proof-queries#conclusion","163":"/developers/blobstream-rollups#introduction-to-blobstream-rollups","164":"/developers/blobstream-rollups#concepts","165":"/developers/blobstream-rollups#share-commitment","166":"/developers/blobstream-rollups#share-commitment-proof-details","167":"/developers/blobstream-rollups#share-commitment-compact-proofs","168":"/developers/blobstream-rollups#share-commitment-pros","169":"/developers/blobstream-rollups#share-commitment-cons","170":"/developers/blobstream-rollups#sequence-of-spans","171":"/developers/blobstream-rollups#sequence-of-spans-proof-details","172":"/developers/blobstream-rollups#sequence-of-spans-proving-unavailable-data","173":"/developers/blobstream-rollups#sequence-of-spans-proving-inclusion-of-some-data","174":"/developers/blobstream-rollups#sequence-of-spans-pros","175":"/developers/blobstream-rollups#sequence-of-spans-cons","176":"/developers/blobstream-rollups#optimistic-rollups","177":"/developers/blobstream-rollups#optimistic-rollups-that-use-a-sequence-of-spans","178":"/developers/blobstream-rollups#optimistic-rollups-that-use-a-sequence-of-spans-pros","179":"/developers/blobstream-rollups#optimistic-rollups-that-use-a-sequence-of-spans-cons","180":"/developers/blobstream-rollups#optimistic-rollups-that-use-a-sequence-of-spans-example","181":"/developers/blobstream-rollups#optimistic-rollups-that-use-share-commitments","182":"/developers/blobstream-rollups#optimistic-rollups-that-use-share-commitments-pros","183":"/developers/blobstream-rollups#optimistic-rollups-that-use-share-commitments-cons","184":"/developers/blobstream-rollups#zk-rollups","185":"/developers/blobstream-rollups#zk-rollups-that-use-sequence-of-spans","186":"/developers/blobstream-rollups#zk-rollups-that-use-sequence-of-spans-pros","187":"/developers/blobstream-rollups#zk-rollups-that-use-sequence-of-spans-cons","188":"/developers/blobstream-rollups#zk-rollups-that-use-share-commitments","189":"/developers/blobstream-rollups#protobuf-deserialization-inside-a-zk-circuit","190":"/developers/blobstream-rollups#zk-rollups-that-use-share-commitments-pros","191":"/developers/blobstream-rollups#zk-rollups-that-use-share-commitments-cons","192":"/developers/blobstream-rollups#heavy-merkle-proofs-usage","193":"/developers/blobstream-rollups#heavy-merkle-proofs-usage-pros","194":"/developers/blobstream-rollups#heavy-merkle-proofs-usage-cons","195":"/developers/blobstream-rollups#conclusion","196":"/developers/blobstream-x-deploy#new-blobstream-x-deployments","197":"/developers/blobstream-x-deploy#deploy-a-new-succinctgateway-contract","198":"/developers/blobstream-x-deploy#deploy-a-blobstreamx-contract-to-the-new-chain","199":"/developers/blobstream-x-deploy#run-a-local-prover","200":"/developers/blobstream#blobstream-streaming-modular-da-to-ethereum","201":"/developers/blobstream#what-is-blobstream","202":"/developers/blobstream#blobstream-vs-data-availability-committees-dacs","203":"/developers/blobstream#decentralization-and-security","204":"/developers/blobstream#mechanism-of-verification","205":"/developers/blobstream#flexibility-and-scalability","206":"/developers/blobstream#what-is-blobstream-x","207":"/developers/blobstream#integrate-with-blobstream-x","208":"/developers/blobstream#how-blobstream-x-works","209":"/developers/blobstream#how-to-integrate-with-blobstream-x","210":"/developers/blobstream#deployed-contracts","211":"/developers/blobstream#blobstream-rollups","212":"/developers/bubs-testnet#bubs-testnet","213":"/developers/bubs-testnet#built-with-the-op-stack-and-celestia","214":"/developers/bubs-testnet#building-on-bubs","215":"/developers/bubs-testnet#rpc-urls","216":"/developers/bubs-testnet#https","217":"/developers/bubs-testnet#wss","218":"/developers/bubs-testnet#bridge","219":"/developers/bubs-testnet#faucet","220":"/developers/bubs-testnet#explorer","221":"/developers/bubs-testnet#status","222":"/developers/bubs-testnet#next-steps","223":"/developers/build-whatever#build-whatever","224":"/developers/build-whatever#quickstart-building-on-celestia","225":"/developers/build-whatever#choose-a-framework","226":"/developers/build-whatever#rollups-as-a-service","227":"/developers/build-whatever#smart-contracts","228":"/developers/build-whatever#what-is-a-rollup","229":"/developers/build-whatever#what-is-a-modular-blockchain","230":"/developers/build-whatever#benefits-of-modular-blockchains","231":"/developers/build-whatever#ease-of-deploying-a-chain","232":"/developers/build-whatever#scaling","233":"/developers/build-whatever#customizability","234":"/developers/celestia-node-key#create-a-wallet-with-celestia-node","235":"/developers/celestia-node-key#using-the-cel-key-utility","236":"/developers/celestia-node-key#installation","237":"/developers/celestia-node-key#steps-for-generating-node-keys","238":"/developers/celestia-node-key#steps-for-exporting-node-keys","239":"/developers/celestia-node-key#steps-for-importing-node-keys","240":"/developers/celestia-node-key#view-all-options-for-cel-key","241":"/developers/celestia-node-key#docker-and-cel-key","242":"/developers/celestia-node-key#prerequisites","243":"/developers/celestia-node-key#running-your-node","244":"/developers/celestia-node-key#mounting-existing-keys-to-container","245":"/developers/deploy-on-bubs#deploy-a-smart-contract-on-bubs-testnet","246":"/developers/deploy-on-bubs#dependencies","247":"/developers/deploy-on-bubs#setup","248":"/developers/deploy-on-bubs#create-your-smart-contract","249":"/developers/deploy-on-bubs#test-your-smart-contract","250":"/developers/deploy-on-bubs#deploying-your-smart-contract","251":"/developers/deploy-on-bubs#using-anvil","252":"/developers/deploy-on-bubs#using-bubs","253":"/developers/deploy-on-bubs#interacting-with-your-smart-contract","254":"/developers/deploy-on-bubs#next-steps","255":"/developers/ethereum-fallback#ethereum-fallback","256":"/developers/full-stack-modular-development-guide#full-stack-modular-blockchain-development-guide","257":"/developers/full-stack-modular-development-guide#getting-started","258":"/developers/full-stack-modular-development-guide#pre-requisites","259":"/developers/full-stack-modular-development-guide#project-setup","260":"/developers/full-stack-modular-development-guide#updating-the-contract-and-tests","261":"/developers/full-stack-modular-development-guide#running-the-test","262":"/developers/full-stack-modular-development-guide#updating-the-deployment-script","263":"/developers/full-stack-modular-development-guide#deploying-locally","264":"/developers/full-stack-modular-development-guide#deploying-to-the-ethermint-sovereign-rollup","265":"/developers/full-stack-modular-development-guide#building-the-frontend","266":"/developers/full-stack-modular-development-guide#configuring-environment-variables","267":"/developers/full-stack-modular-development-guide#configuring-the-entrypoint","268":"/developers/full-stack-modular-development-guide#creating-and-reading-posts","269":"/developers/full-stack-modular-development-guide#adding-ethermint-chain-to-metamask","270":"/developers/full-stack-modular-development-guide#testing-it-out-on-ethermint","271":"/developers/full-stack-modular-development-guide#now-give-it-a-spin-🌀","272":"/developers/gm-portal-bubs#deploying-a-dapp-on-bubs-testnet","273":"/developers/gm-portal-bubs#update-the-frontend","274":"/developers/gm-portal-bubs#interact-with-the-frontend","275":"/developers/gm-portal-bubs#next-steps","276":"/developers/integrate-celestia#integrate-celestia-for-service-providers","277":"/developers/integrate-celestia#getting-started","278":"/developers/integrate-celestia#celestia-service-provider-notes","279":"/developers/integrate-celestia#custody-and-key-management","280":"/developers/integrate-celestia#rpc-and-querying","281":"/developers/integrate-celestia#compatibility","282":"/developers/integrate-celestia#syncing","283":"/developers/integrate-celestia#notable-exceptions-relative-to-other-blockchains","284":"/developers/intro-to-op-stack#introduction-to-op-stack-integration","285":"/developers/intro-to-op-stack#about-the-integration","286":"/developers/intro-to-op-stack#table-of-contents-of-the-category","287":"/developers/intro-to-op-stack#celestia-and-op-stack-repository","288":"/developers/intro-to-op-stack#what-are-optimism-and-the-op-stack","289":"/developers/intro-to-op-stack#what-is-celestia","290":"/developers/intro-to-op-stack#op-stack-and-celestia","291":"/developers/intro-to-op-stack#next-steps","292":"/developers/node-api#node-api","293":"/developers/node-api#rpc-api","294":"/developers/node-api#library","295":"/developers/node-api#rpc","296":"/developers/node-api#rpc-api-tutorial","297":"/developers/node-api#gateway-api","298":"/developers/node-api#gateway-api-tutorial","299":"/developers/node-tutorial#celestia-node-rpc-cli-tutorial","300":"/developers/node-tutorial#introduction","301":"/developers/node-tutorial#blobs","302":"/developers/node-tutorial#namespaces","303":"/developers/node-tutorial#hardware-requirements","304":"/developers/node-tutorial#setting-up-dependencies","305":"/developers/node-tutorial#install-golang","306":"/developers/node-tutorial#celestia-node","307":"/developers/node-tutorial#install-celestia-node","308":"/developers/node-tutorial#instantiate-a-celestia-light-node","309":"/developers/node-tutorial#connect-to-a-public-core-endpoint","310":"/developers/node-tutorial#keys-and-wallets","311":"/developers/node-tutorial#rpc-cli-guide","312":"/developers/node-tutorial#command-formatting","313":"/developers/node-tutorial#basic-flags","314":"/developers/node-tutorial#auth-token","315":"/developers/node-tutorial#node-store","316":"/developers/node-tutorial#auth-token-on-custom-or-private-network","317":"/developers/node-tutorial#submitting-data","318":"/developers/node-tutorial#optional-submit-with-curl","319":"/developers/node-tutorial#retrieving-data","320":"/developers/node-tutorial#setting-the-gas-price","321":"/developers/node-tutorial#examples","322":"/developers/node-tutorial#check-your-balance","323":"/developers/node-tutorial#check-the-balance-of-another-address","324":"/developers/node-tutorial#get-your-node-id","325":"/developers/node-tutorial#get-your-account-address","326":"/developers/node-tutorial#get-block-header-by-height","327":"/developers/node-tutorial#combined-commands","328":"/developers/node-tutorial#get-data-availability-sampler-stats","329":"/developers/node-tutorial#transfer-balance-of-utia-to-another-account","330":"/developers/node-tutorial#api-version","331":"/developers/node-tutorial#help","332":"/developers/node-tutorial#advanced-example","333":"/developers/node-tutorial#additional-resources","334":"/developers/node-tutorial#submitting-a-blob-using-curl","335":"/developers/node-tutorial#post-an-svg-as-a-pfb","336":"/developers/node-tutorial#golang-guide","337":"/developers/node-tutorial#troubleshooting","338":"/developers/optimism-devnet#deploy-an-op-stack-devnet","339":"/developers/optimism-devnet#dependency-setup","340":"/developers/optimism-devnet#environment-setup-and-golang-installation","341":"/developers/optimism-devnet#clone-repository","342":"/developers/optimism-devnet#build-devnet","343":"/developers/optimism-devnet#start-devnet","344":"/developers/optimism-devnet#view-the-logs-of-the-devnet","345":"/developers/optimism-devnet#stop-devnet","346":"/developers/optimism-devnet#viewing-containers","347":"/developers/optimism-devnet#find-a-transaction","348":"/developers/optimism-devnet#read-the-transaction-call-data","349":"/developers/optimism-devnet#find-the-data-on-celestia","350":"/developers/optimism-devnet#ethereum-fallback-mechanism-in-op-stack","351":"/developers/optimism-devnet#implementation-of-fallback","352":"/developers/optimism-devnet#testing-the-fallback","353":"/developers/optimism-devnet#span-batches","354":"/developers/optimism#deploy-an-op-stack-devnet-to-celestia","355":"/developers/optimism#pick-your-deployment-type","356":"/developers/optimism#using-a-local-devnet","357":"/developers/optimism#using-a-light-node","358":"/developers/optimism#using-a-raas-provider","359":"/developers/optimism#build-the-devnet","360":"/developers/optimism#start-the-devnet","361":"/developers/optimism#view-the-logs-of-the-devnet","362":"/developers/optimism#stop-the-devnet","363":"/developers/optimism#clean-the-devnet","364":"/developers/optimism#deploying-to-an-l1-or-l2","365":"/developers/prompt-scavenger#prompt-scavenger","366":"/developers/prompt-scavenger#dependencies","367":"/developers/prompt-scavenger#install-celestia-node-and-run-a-light-node","368":"/developers/prompt-scavenger#node-api-authentication-key","369":"/developers/prompt-scavenger#openai-key","370":"/developers/prompt-scavenger#building-the-prompt-scavenger","371":"/developers/prompt-scavenger#copy-over-go-files","372":"/developers/prompt-scavenger#build-your-import-statements","373":"/developers/prompt-scavenger#helpful-functions","374":"/developers/prompt-scavenger#wrapping-up-the-functions","375":"/developers/prompt-scavenger#next-steps","376":"/developers/requesting-data-commitment-ranges#requesting-data-commitment-ranges","377":"/developers/requesting-data-commitment-ranges#recommended-setup","378":"/developers/requesting-data-commitment-ranges#local-proving","379":"/developers/requesting-data-commitment-ranges#request-proof-onchain","380":"/developers/rollkit#rollkit","381":"/developers/rollkit#tutorials","382":"/developers/submit-data#submitting-data-blobs-to-celestia","383":"/developers/submit-data#fee-market-and-mempool","384":"/developers/submit-data#fees-and-gas-limits","385":"/developers/submit-data#estimating-pfb-gas","386":"/developers/submit-data#gas-fee-calculation","387":"/developers/submit-data#estimating-gas-programmatically","388":"/developers/submit-data#submitting-multiple-transactions-in-one-block-from-the-same-account","389":"/developers/submit-data#api","390":"/developers/submit-data#the-celestia-app-consensus-node-cli","391":"/developers/submit-data#the-celestia-node-light-node-cli","392":"/developers/submit-data#grpc-to-a-consensus-node-via-the-user-package","393":"/developers/submit-data#rpc-to-a-celestia-node","394":"/developers/transaction-resubmission#transaction-resubmission","395":"/developers/transaction-resubmission#monitoring-and-resubmission","396":"/developers/transaction-resubmission#notes","397":"/developers/wallets#wallet-integrations-with-celestia","398":"/developers/wallets#add-celestia-network-parameters-to-keplr-with-react","399":"/developers/wallets#adding-a-custom-chain-to-leap","400":"/developers/wallets#adding-a-custom-chain-to-cosmostation","401":"/learn/how-celestia-works/data-availability-faq#data-availability-faq","402":"/learn/how-celestia-works/data-availability-faq#what-is-data-availability","403":"/learn/how-celestia-works/data-availability-faq#what-is-the-data-availability-problem","404":"/learn/how-celestia-works/data-availability-faq#how-do-nodes-verify-data-availability-in-celestia","405":"/learn/how-celestia-works/data-availability-faq#what-is-data-availability-sampling","406":"/learn/how-celestia-works/data-availability-faq#what-are-some-of-the-security-assumptions-that-celestia-makes-for-data-availability-sampling","407":"/learn/how-celestia-works/data-availability-faq#why-is-block-reconstruction-necessary-for-security","408":"/learn/how-celestia-works/data-availability-faq#what-is-data-storage","409":"/learn/how-celestia-works/data-availability-faq#what-is-the-problem-around-data-storage","410":"/learn/how-celestia-works/data-availability-faq#what-is-the-difference-between-data-availability-and-data-storage","411":"/learn/how-celestia-works/data-availability-faq#where-does-blockchain-state-fit-into-this","412":"/learn/how-celestia-works/data-availability-faq#why-doesn-t-celestia-incentivize-storage-of-historical-data","413":"/learn/how-celestia-works/data-availability-faq#who-may-store-historical-data-if-there-is-no-reward","414":"/learn/how-celestia-works/data-availability-faq#what-are-some-things-blockchains-can-do-to-provide-stronger-assurances-of-data-retrievability","415":"/learn/how-celestia-works/data-availability-layer#celestia-s-data-availability-layer","416":"/learn/how-celestia-works/data-availability-layer#data-availability-sampling-das","417":"/learn/how-celestia-works/data-availability-layer#scalability","418":"/learn/how-celestia-works/data-availability-layer#fraud-proofs-of-incorrectly-extended-data","419":"/learn/how-celestia-works/data-availability-layer#namespaced-merkle-trees-nmts","420":"/learn/how-celestia-works/data-availability-layer#building-a-pos-blockchain-for-da","421":"/learn/how-celestia-works/data-availability-layer#providing-data-availability","422":"/learn/how-celestia-works/monolithic-vs-modular#monolithic-vs-modular-blockchains","423":"/learn/how-celestia-works/overview#introduction","424":"/learn/how-celestia-works/transaction-lifecycle#the-lifecycle-of-a-celestia-app-transaction","425":"/learn/how-celestia-works/transaction-lifecycle#checking-data-availability","426":"/learn/how-to-stake-tia#how-to-stake-tia","427":"/learn/how-to-stake-tia#select-your-preferred-wallet","428":"/learn/how-to-stake-tia#stake-tia-with-keplr-wallet","429":"/learn/how-to-stake-tia#open-your-keplr-browser-extension","430":"/learn/how-to-stake-tia#select-celestia-network-and-search-for-a-validator","431":"/learn/how-to-stake-tia#stake-your-tia-tokens","432":"/learn/how-to-stake-tia#confirm-and-manage-your-tia","433":"/learn/how-to-stake-tia#stake-tia-with-leap-wallet","434":"/learn/how-to-stake-tia#open-your-leap-browser-extension","435":"/learn/how-to-stake-tia#select-a-validator-and-stake-tia","436":"/learn/how-to-stake-tia#confirm-and-manage-your-tia-1","437":"/learn/paying-for-blobspace#paying-for-blobspace","438":"/learn/paying-for-blobspace#payforblobs-transactions","439":"/learn/paying-for-blobspace#fee-market-overview","440":"/learn/retrievability#data-retrievability-and-pruning","441":"/learn/retrievability#data-retrievability-and-pruning-in-celestia-node","442":"/learn/retrievability#suggested-practices-for-rollups","443":"/learn/staking-governance-supply#staking-governance-supply","444":"/learn/staking-governance-supply#proof-of-stake-on-celestia","445":"/learn/staking-governance-supply#inflation","446":"/learn/staking-governance-supply#decentralised-governance","447":"/learn/staking-governance-supply#network-parameters","448":"/learn/staking-governance-supply#community-pool","449":"/learn/staking-governance-supply#tia-allocation-at-genesis","450":"/learn/staking-governance-supply#unlocks","451":"/learn/staking#staking-on-celestia","452":"/learn/staking#mainnet-beta","453":"/learn/staking#mocha-testnet","454":"/learn/tia#overview-of-tia","455":"/learn/tia#tia-at-a-glance","456":"/learn/tia#role-of-tia","457":"/learn/tia#paying-for-blobspace","458":"/learn/tia#bootstrapping-new-rollups","459":"/learn/tia#proof-of-stake","460":"/learn/tia#decentralised-governance","461":"/learn/tia#denominations","462":"/learn/tia#tia-display-token","463":"/learn/tia#utia-staking-denomination","464":"/learn/tia#microtia-staking-denomination-alias","465":"/nodes/arabica-devnet#arabica-devnet","466":"/nodes/arabica-devnet#network-stability-and-upgrades","467":"/nodes/arabica-devnet#network-details","468":"/nodes/arabica-devnet#software-version-numbers","469":"/nodes/arabica-devnet#integrations","470":"/nodes/arabica-devnet#rpc-endpoints","471":"/nodes/arabica-devnet#using-consensus-endpoints-with-da-nodes","472":"/nodes/arabica-devnet#arabica-devnet-faucet","473":"/nodes/arabica-devnet#discord","474":"/nodes/arabica-devnet#web","475":"/nodes/arabica-devnet#explorers","476":"/nodes/arabica-devnet#network-upgrades","477":"/nodes/bridge-node#setting-up-a-celestia-bridge-node","478":"/nodes/bridge-node#overview-of-bridge-nodes","479":"/nodes/bridge-node#hardware-requirements","480":"/nodes/bridge-node#setting-up-your-bridge-node","481":"/nodes/bridge-node#setup-the-dependencies","482":"/nodes/bridge-node#deploy-the-celestia-bridge-node","483":"/nodes/bridge-node#install-celestia-node","484":"/nodes/bridge-node#initialize-the-bridge-node","485":"/nodes/bridge-node#run-the-bridge-node","486":"/nodes/bridge-node#optional-run-the-bridge-node-with-a-custom-key","487":"/nodes/bridge-node#optional-start-the-bridge-node-with-systemd","488":"/nodes/celestia-app-commands#helpful-cli-commands","489":"/nodes/celestia-app-commands#creating-a-wallet","490":"/nodes/celestia-app-commands#key-management","491":"/nodes/celestia-app-commands#importing-and-exporting-keys","492":"/nodes/celestia-app-commands#querying-subcommands","493":"/nodes/celestia-app-commands#token-management","494":"/nodes/celestia-app-commands#governance","495":"/nodes/celestia-app-commands#community-pool","496":"/nodes/celestia-app-commands#claim-validator-rewards","497":"/nodes/celestia-app-commands#delegate-undelegate-tokens","498":"/nodes/celestia-app-commands#unjailing-the-validator","499":"/nodes/celestia-app-commands#how-to-export-logs-with-systemd","500":"/nodes/celestia-app-commands#signing-genesis-for-a-new-network","501":"/nodes/celestia-app-metrics#metrics","502":"/nodes/celestia-app-metrics#setup","503":"/nodes/celestia-app-metrics#visualization","504":"/nodes/celestia-app-metrics#node-exporter","505":"/nodes/celestia-app-metrics#alerts","506":"/nodes/celestia-app-multisig#multisig","507":"/nodes/celestia-app-multisig#command-line","508":"/nodes/celestia-app-multisig#resources","509":"/nodes/celestia-app-slashing#slashing-on-celestia","510":"/nodes/celestia-app-upgrade-monitor#upgrade-monitor","511":"/nodes/celestia-app-vesting#how-to-create-a-vesting-account-with-celestia-app","512":"/nodes/celestia-app-vesting#local-devnet","513":"/nodes/celestia-app-vesting#setting-up-the-local-devnet","514":"/nodes/celestia-app-vesting#run-the-devnet","515":"/nodes/celestia-app-vesting#save-the-home-directory-path","516":"/nodes/celestia-app-vesting#check-the-version-of-the-devnet","517":"/nodes/celestia-app-vesting#next-steps","518":"/nodes/celestia-app-vesting#setting-up-vesting-account-on-devnet","519":"/nodes/celestia-app-vesting#create-a-new-key","520":"/nodes/celestia-app-vesting#list-your-keys","521":"/nodes/celestia-app-vesting#set-variables","522":"/nodes/celestia-app-vesting#create-your-devnet-vesting-account","523":"/nodes/celestia-app-vesting#query-the-devnet-vesting-account-details","524":"/nodes/celestia-app-vesting#query-the-devnet-base-account-details","525":"/nodes/celestia-app-vesting#query-the-balances-of-the-devnet-accounts","526":"/nodes/celestia-app-vesting#mocha","527":"/nodes/celestia-app-vesting#create-a-wallet","528":"/nodes/celestia-app-vesting#fund-your-account","529":"/nodes/celestia-app-vesting#create-a-vesting-account-on-mocha","530":"/nodes/celestia-app-vesting#optional-set-up-a-full-consensus-node-or-validator","531":"/nodes/celestia-app-vesting#optional-change-your-client-toml","532":"/nodes/celestia-app-vesting#notes","533":"/nodes/celestia-app-vesting#conclusion","534":"/nodes/celestia-app-wallet#create-a-wallet-with-celestia-app","535":"/nodes/celestia-app-wallet#prerequisites","536":"/nodes/celestia-app-wallet#create-a-wallet","537":"/nodes/celestia-app-wallet#fund-a-wallet","538":"/nodes/celestia-app#install-celestia-app","539":"/nodes/celestia-app#building-binary-from-source","540":"/nodes/celestia-app#installing-a-pre-built-binary","541":"/nodes/celestia-app#ports","542":"/nodes/celestia-node-custom-networks#custom-networks-and-values","543":"/nodes/celestia-node-metrics#celestia-node-metrics","544":"/nodes/celestia-node-metrics#running-metrics-flags","545":"/nodes/celestia-node-metrics#mainnet-beta","546":"/nodes/celestia-node-metrics#mocha-testnet","547":"/nodes/celestia-node-metrics#tls-connections","548":"/nodes/celestia-node-metrics#metrics-endpoint-design-considerations","549":"/nodes/celestia-node-troubleshooting#troubleshooting","550":"/nodes/celestia-node-troubleshooting#network-selection","551":"/nodes/celestia-node-troubleshooting#chain-id","552":"/nodes/celestia-node-troubleshooting#ports","553":"/nodes/celestia-node-troubleshooting#changing-the-location-of-your-node-store","554":"/nodes/celestia-node-troubleshooting#resetting-your-config","555":"/nodes/celestia-node-troubleshooting#clearing-the-data-store","556":"/nodes/celestia-node-troubleshooting#fatal-headers-given-to-the-heightsub-are-in-the-wrong-order","557":"/nodes/celestia-node-troubleshooting#error-too-many-open-files","558":"/nodes/celestia-node#install-celestia-node","559":"/nodes/celestia-node#installing-from-source","560":"/nodes/celestia-node#installing-a-pre-built-binary","561":"/nodes/celestia-node#next-steps","562":"/nodes/celestia-node#upgrading-your-binary","563":"/nodes/config-toml#config-toml-guide","564":"/nodes/config-toml#pre-requisites","565":"/nodes/config-toml#understanding-config-toml","566":"/nodes/config-toml#core","567":"/nodes/config-toml#p2p","568":"/nodes/config-toml#bootstrap","569":"/nodes/config-toml#mutual-peers","570":"/nodes/config-toml#services","571":"/nodes/config-toml#trustedhash-and-trustedpeer","572":"/nodes/decide-node#deciding-which-node-to-run","573":"/nodes/decide-node#beginner","574":"/nodes/decide-node#advanced","575":"/nodes/docker-images#🐳-docker-setup","576":"/nodes/docker-images#prerequisites","577":"/nodes/docker-images#quick-start","578":"/nodes/docker-images#light-node-setup-with-persistent-storage","579":"/nodes/docker-images#initialize-the-node-store-and-key","580":"/nodes/docker-images#start-the-node","581":"/nodes/docker-images#video-walkthrough","582":"/nodes/docker-images#_2-5-minute-version","583":"/nodes/docker-images#troubleshooting","584":"/nodes/environment#development-environment","585":"/nodes/environment#install-dependencies","586":"/nodes/environment#install-golang","587":"/nodes/full-consensus-node#setting-up-a-celestia-full-consensus-node","588":"/nodes/full-consensus-node#hardware-requirements","589":"/nodes/full-consensus-node#setting-up-a-full-consensus-node","590":"/nodes/full-consensus-node#set-up-the-dependencies","591":"/nodes/full-consensus-node#install-celestia-app","592":"/nodes/full-consensus-node#set-up-the-p2p-networks","593":"/nodes/full-consensus-node#storage-and-pruning-configurations","594":"/nodes/full-consensus-node#connecting-a-consensus-node-to-a-bridge-node","595":"/nodes/full-consensus-node#enable-transaction-indexing","596":"/nodes/full-consensus-node#retain-all-block-data","597":"/nodes/full-consensus-node#querying-transactions-by-hash","598":"/nodes/full-consensus-node#accessing-historical-state","599":"/nodes/full-consensus-node#saving-on-storage-requirements","600":"/nodes/full-consensus-node#syncing","601":"/nodes/full-consensus-node#state-sync","602":"/nodes/full-consensus-node#quick-sync","603":"/nodes/full-consensus-node#start-the-consensus-node","604":"/nodes/full-consensus-node#extra-resources-for-consensus-nodes","605":"/nodes/full-consensus-node#optional-reset-network","606":"/nodes/full-consensus-node#optional-configuring-an-rpc-endpoint","607":"/nodes/full-consensus-node#expose-rpc","608":"/nodes/full-consensus-node#note-on-external-address","609":"/nodes/full-consensus-node#restart-the-node","610":"/nodes/full-consensus-node#optional-transaction-indexer-configuration-options","611":"/nodes/full-consensus-node#optional-discard-abci-responses-configuration","612":"/nodes/full-storage-node#setting-up-a-celestia-full-storage-node","613":"/nodes/full-storage-node#overview-of-full-storage-nodes","614":"/nodes/full-storage-node#hardware-requirements","615":"/nodes/full-storage-node#setting-up-your-full-storage-node","616":"/nodes/full-storage-node#setup-the-dependencies","617":"/nodes/full-storage-node#install-celestia-node","618":"/nodes/full-storage-node#run-the-full-storage-node","619":"/nodes/full-storage-node#initialize-the-full-storage-node","620":"/nodes/full-storage-node#start-the-full-storage-node","621":"/nodes/full-storage-node#optional-run-the-full-storage-node-with-a-custom-key","622":"/nodes/full-storage-node#optional-start-the-full-storage-node-with-systemd","623":"/nodes/full-storage-node#stop-the-full-storage-node","624":"/nodes/hardfork-process#celestia-hardfork-process","625":"/nodes/hardfork-process#general-process","626":"/nodes/hardfork-process#mocha-hardfork","627":"/nodes/ibc-relayer#ibc-relaying-guide","628":"/nodes/ibc-relayer#hermes","629":"/nodes/ibc-relayer#configuration","630":"/nodes/ibc-relayer#add-relayer-wallets","631":"/nodes/ibc-relayer#verify-configuration-files","632":"/nodes/ibc-relayer#create-a-connection-between-2-chains","633":"/nodes/ibc-relayer#create-clients","634":"/nodes/ibc-relayer#open-connection-over-new-clients","635":"/nodes/ibc-relayer#configure-channels-in-hermes","636":"/nodes/ibc-relayer#start-the-relayer","637":"/nodes/ibc-relayer#transfer","638":"/nodes/ibc-relayer#token-filter","639":"/nodes/instantiate-testnet#celestia-app-network-instantiation-guide","640":"/nodes/instantiate-testnet#hardware-requirements","641":"/nodes/instantiate-testnet#setup-dependencies","642":"/nodes/instantiate-testnet#celestia-app-installation","643":"/nodes/instantiate-testnet#spin-up-a-celestia-testnet","644":"/nodes/instantiate-testnet#optional-reset-working-directory","645":"/nodes/instantiate-testnet#initialize-a-working-directory","646":"/nodes/instantiate-testnet#create-a-new-key","647":"/nodes/instantiate-testnet#add-genesis-account-keyname","648":"/nodes/instantiate-testnet#optional-adding-other-validators","649":"/nodes/instantiate-testnet#create-the-genesis-transaction-for-new-chain","650":"/nodes/instantiate-testnet#creating-the-genesis-json-file","651":"/nodes/instantiate-testnet#modify-your-config-file","652":"/nodes/instantiate-testnet#add-your-node-as-a-persistent-peer","653":"/nodes/instantiate-testnet#instantiate-the-network","654":"/nodes/light-node#setting-up-a-celestia-light-node","655":"/nodes/light-node#overview-of-light-nodes","656":"/nodes/light-node#hardware-requirements","657":"/nodes/light-node#setting-up-your-light-node","658":"/nodes/light-node#install-celestia-node","659":"/nodes/light-node#initialize-the-light-node","660":"/nodes/light-node#start-the-light-node","661":"/nodes/light-node#keys-and-wallets","662":"/nodes/light-node#testnet-tokens","663":"/nodes/light-node#optional-run-the-light-node-with-a-custom-key","664":"/nodes/light-node#optional-start-light-node-with-systemd","665":"/nodes/light-node#data-availability-sampling","666":"/nodes/mainnet#mainnet-beta","667":"/nodes/mainnet#network-stability-and-upgrades","668":"/nodes/mainnet#network-details","669":"/nodes/mainnet#software-version-numbers","670":"/nodes/mainnet#network-parameters","671":"/nodes/mainnet#maximum-bytes","672":"/nodes/mainnet#integrations","673":"/nodes/mainnet#consensus-nodes","674":"/nodes/mainnet#consensus-rpc-endpoints","675":"/nodes/mainnet#api-endpoints","676":"/nodes/mainnet#grpc-endpoints","677":"/nodes/mainnet#websocket-endpoints","678":"/nodes/mainnet#data-availability-nodes","679":"/nodes/mainnet#data-availability-da-rpc-endpoints","680":"/nodes/mainnet#archival-da-rpc-endpoints","681":"/nodes/mainnet#grove-archival-endpoints","682":"/nodes/mainnet#explorers","683":"/nodes/mainnet#analytics","684":"/nodes/mainnet#network-upgrades","685":"/nodes/mocha-testnet#mocha-testnet","686":"/nodes/mocha-testnet#network-details","687":"/nodes/mocha-testnet#software-version-numbers","688":"/nodes/mocha-testnet#rpc-for-da-bridge-full-and-light-nodes","689":"/nodes/mocha-testnet#rpc-endpoints","690":"/nodes/mocha-testnet#api-endpoints","691":"/nodes/mocha-testnet#grpc-endpoints","692":"/nodes/mocha-testnet#bridge-and-full-node-endpoints","693":"/nodes/mocha-testnet#mocha-testnet-faucet","694":"/nodes/mocha-testnet#explorers","695":"/nodes/mocha-testnet#network-upgrades","696":"/nodes/overview#overview-to-running-nodes-on-celestia","697":"/nodes/participate#participate-in-the-celestia-networks","698":"/nodes/participate#mainnet-beta","699":"/nodes/participate#compatible-software-versions-for-mainnet-beta","700":"/nodes/participate#testnets","701":"/nodes/participate#arabica-devnet","702":"/nodes/participate#compatible-software-versions-for-arabica-devnet","703":"/nodes/participate#mocha-testnet","704":"/nodes/participate#compatible-software-versions-for-mocha-testnet","705":"/nodes/participate#network-upgrades","706":"/nodes/quick-start#quick-start-guide","707":"/nodes/quick-start#celestia-node","708":"/nodes/quick-start#celestia-app","709":"/nodes/quick-start#getting-started","710":"/nodes/systemd#setting-up-your-node-as-a-background-process-with-systemd","711":"/nodes/systemd#consensus-nodes","712":"/nodes/systemd#start-the-celestia-app-with-systemd","713":"/nodes/systemd#data-availability-nodes","714":"/nodes/systemd#celestia-full-storage-node","715":"/nodes/systemd#celestia-bridge-node","716":"/nodes/systemd#celestia-light-node","717":"/nodes/validator-node#setting-up-a-celestia-validator-node","718":"/nodes/validator-node#hardware-requirements","719":"/nodes/validator-node#setting-up-a-validator-node","720":"/nodes/validator-node#wallet","721":"/nodes/validator-node#delegate-stake-to-a-validator","722":"/nodes/validator-node#optional-deploy-the-celestia-node","723":"/nodes/validator-node#install-celestia-node","724":"/nodes/validator-node#initialize-the-bridge-node","725":"/nodes/validator-node#run-the-bridge-node","726":"/nodes/validator-node#optional-start-the-bridge-node-with-systemd","727":"/nodes/validator-node#run-the-validator-node","728":"/nodes/validator-node#submit-your-validator-information","729":"/nodes/validator-node#optional-transaction-indexer-configuration-options","730":"/nodes/validator-node#additional-resources"},"fieldIds":{"title":0,"titles":1,"text":2},"fieldLength":{"0":[3,1,24],"1":[3,3,22],"2":[2,3,69],"3":[2,3,34],"4":[3,3,25],"5":[2,1,32],"6":[5,1,1],"7":[2,5,62],"8":[2,5,114],"9":[2,5,57],"10":[1,5,43],"11":[1,5,39],"12":[2,5,22],"13":[2,7,38],"14":[2,7,57],"15":[3,7,50],"16":[3,7,34],"17":[1,5,47],"18":[5,1,1],"19":[4,5,57],"20":[3,5,1],"21":[2,7,53],"22":[2,7,41],"23":[2,7,110],"24":[2,7,162],"25":[2,7,62],"26":[1,5,45],"27":[2,6,147],"28":[2,5,16],"29":[2,5,17],"30":[4,1,150],"31":[3,4,229],"32":[2,4,69],"33":[4,4,137],"34":[4,4,460],"35":[3,1,31],"36":[3,3,129],"37":[3,3,69],"38":[3,3,72],"39":[3,3,39],"40":[1,6,164],"41":[3,6,124],"42":[1,6,150],"43":[1,6,190],"44":[1,6,170],"45":[3,6,159],"46":[1,6,132],"47":[4,6,145],"48":[9,6,167],"49":[4,1,28],"50":[2,4,64],"51":[2,4,1],"52":[6,6,71],"53":[2,6,70],"54":[1,6,35],"55":[1,6,28],"56":[1,6,48],"57":[3,1,37],"58":[3,3,12],"59":[6,3,44],"60":[8,3,58],"61":[7,3,48],"62":[6,3,23],"63":[2,3,37],"64":[2,1,26],"65":[2,1,143],"66":[8,1,29],"67":[6,8,1],"68":[7,11,47],"69":[7,11,68],"70":[6,8,1],"71":[10,11,81],"72":[5,11,40],"73":[8,1,18],"74":[1,8,14],"75":[4,8,85],"76":[4,12,94],"77":[3,8,51],"78":[4,8,15],"79":[6,1,77],"80":[1,6,52],"81":[1,6,10],"82":[6,7,63],"83":[6,7,58],"84":[8,7,238],"85":[8,7,115],"86":[10,7,74],"87":[12,7,39],"88":[11,7,129],"89":[9,7,238],"90":[10,7,56],"91":[7,7,231],"92":[4,7,49],"93":[1,6,17],"94":[2,7,75],"95":[4,7,32],"96":[2,11,13],"97":[2,11,13],"98":[4,7,48],"99":[4,7,48],"100":[7,1,1],"101":[1,7,20],"102":[4,7,71],"103":[6,7,34],"104":[8,1,1],"105":[1,8,51],"106":[2,8,33],"107":[3,10,175],"108":[3,10,128],"109":[3,10,77],"110":[5,10,86],"111":[2,8,17],"112":[8,1,1],"113":[1,8,33],"114":[1,8,18],"115":[1,8,35],"116":[4,8,103],"117":[4,8,139],"118":[4,8,1],"119":[2,9,51],"120":[4,9,87],"121":[5,8,91],"122":[2,8,28],"123":[4,1,1],"124":[2,4,1],"125":[1,6,9],"126":[4,6,58],"127":[2,6,95],"128":[2,4,43],"129":[1,4,50],"130":[3,4,31],"131":[6,4,124],"132":[4,1,1],"133":[3,4,75],"134":[3,4,149],"135":[2,4,168],"136":[3,6,101],"137":[2,6,115],"138":[3,4,1],"139":[3,7,164],"140":[2,7,1],"141":[4,9,10],"142":[8,9,13],"143":[4,9,39],"144":[3,1,1],"145":[1,3,26],"146":[5,3,101],"147":[3,7,1],"148":[3,7,134],"149":[3,3,72],"150":[5,6,100],"151":[15,6,228],"152":[4,6,86],"153":[9,3,87],"154":[1,11,46],"155":[1,11,194],"156":[1,11,94],"157":[1,11,47],"158":[1,11,113],"159":[1,11,272],"160":[5,11,72],"161":[6,11,347],"162":[1,3,58],"163":[4,1,47],"164":[1,4,1],"165":[2,5,22],"166":[4,7,91],"167":[4,7,82],"168":[3,7,40],"169":[3,7,63],"170":[3,5,85],"171":[5,8,30],"172":[6,8,119],"173":[7,8,118],"174":[4,8,14],"175":[4,8,2],"176":[2,4,57],"177":[8,5,70],"178":[9,11,50],"179":[9,11,2],"180":[9,11,42],"181":[6,5,106],"182":[7,9,17],"183":[7,9,27],"184":[2,4,65],"185":[7,5,88],"186":[8,10,35],"187":[8,10,2],"188":[6,5,29],"189":[6,9,81],"190":[7,9,2],"191":[7,9,26],"192":[4,5,78],"193":[5,9,2],"194":[5,9,13],"195":[1,4,48],"196":[4,1,36],"197":[5,4,9],"198":[8,4,15],"199":[4,4,13],"200":[6,1,1],"201":[4,6,106],"202":[7,6,1],"203":[3,12,40],"204":[3,12,42],"205":[3,12,69],"206":[5,6,111],"207":[4,6,106],"208":[4,9,58],"209":[6,9,31],"210":[2,9,40],"211":[2,9,16],"212":[2,1,30],"213":[7,2,116],"214":[3,2,49],"215":[2,4,34],"216":[1,6,8],"217":[1,6,40],"218":[1,4,22],"219":[1,4,20],"220":[1,4,12],"221":[1,4,12],"222":[2,2,30],"223":[2,1,43],"224":[4,2,1],"225":[3,6,22],"226":[4,6,8],"227":[2,6,10],"228":[5,2,41],"229":[6,2,100],"230":[4,2,1],"231":[5,6,81],"232":[1,6,123],"233":[1,6,107],"234":[6,1,42],"235":[5,6,42],"236":[1,11,56],"237":[5,11,111],"238":[5,11,32],"239":[5,11,33],"240":[6,11,5],"241":[4,6,1],"242":[1,10,16],"243":[3,10,144],"244":[5,10,147],"245":[7,1,14],"246":[1,7,26],"247":[1,7,35],"248":[4,7,103],"249":[4,7,139],"250":[4,7,1],"251":[2,9,62],"252":[2,9,90],"253":[5,7,90],"254":[2,7,27],"255":[2,1,97],"256":[6,1,88],"257":[2,6,34],"258":[2,8,34],"259":[2,8,26],"260":[5,10,193],"261":[3,10,18],"262":[4,10,70],"263":[2,10,120],"264":[6,8,142],"265":[3,8,97],"266":[3,11,75],"267":[3,8,121],"268":[4,8,270],"269":[5,8,60],"270":[5,8,98],"271":[6,8,32],"272":[6,1,156],"273":[3,6,51],"274":[4,6,15],"275":[2,6,34],"276":[5,1,18],"277":[2,5,28],"278":[4,5,84],"279":[4,7,27],"280":[3,7,62],"281":[1,7,36],"282":[1,7,28],"283":[6,7,81],"284":[5,1,31],"285":[3,5,65],"286":[5,7,16],"287":[5,5,13],"288":[8,8,77],"289":[4,8,49],"290":[4,5,116],"291":[2,5,35],"292":[2,1,28],"293":[2,2,46],"294":[1,3,16],"295":[1,3,48],"296":[3,3,19],"297":[2,2,66],"298":[3,3,12],"299":[5,1,26],"300":[1,5,1],"301":[1,6,19],"302":[1,6,66],"303":[2,5,30],"304":[3,5,105],"305":[2,5,97],"306":[2,5,1],"307":[3,5,122],"308":[5,5,55],"309":[6,5,120],"310":[3,5,107],"311":[3,5,39],"312":[2,6,104],"313":[2,6,71],"314":[3,8,67],"315":[2,8,56],"316":[7,10,69],"317":[2,6,124],"318":[4,8,10],"319":[2,6,113],"320":[4,6,78],"321":[1,6,1],"322":[3,7,49],"323":[6,7,45],"324":[4,7,52],"325":[4,7,31],"326":[5,7,112],"327":[2,7,24],"328":[5,7,10],"329":[7,7,75],"330":[2,7,20],"331":[1,7,25],"332":[2,6,39],"333":[2,5,1],"334":[5,7,124],"335":[6,7,21],"336":[2,7,21],"337":[1,7,46],"338":[5,1,14],"339":[2,5,1],"340":[5,7,10],"341":[2,7,44],"342":[2,5,28],"343":[2,6,30],"344":[5,6,23],"345":[2,6,28],"346":[2,6,52],"347":[3,5,83],"348":[5,5,84],"349":[5,5,50],"350":[6,5,24],"351":[3,9,167],"352":[3,9,80],"353":[2,5,57],"354":[7,1,29],"355":[4,7,31],"356":[4,11,28],"357":[4,11,108],"358":[4,11,22],"359":[3,7,28],"360":[3,7,135],"361":[5,7,20],"362":[3,7,12],"363":[3,7,14],"364":[7,7,18],"365":[2,1,84],"366":[1,2,28],"367":[8,3,174],"368":[4,3,41],"369":[2,3,24],"370":[4,2,73],"371":[4,5,38],"372":[4,5,60],"373":[2,5,244],"374":[4,5,119],"375":[2,2,52],"376":[4,1,48],"377":[2,4,81],"378":[2,4,33],"379":[3,4,63],"380":[1,1,98],"381":[1,1,44],"382":[5,1,27],"383":[4,5,26],"384":[4,9,103],"385":[3,12,141],"386":[3,12,152],"387":[3,12,78],"388":[10,9,118],"389":[1,5,11],"390":[6,6,14],"391":[5,6,19],"392":[9,6,199],"393":[5,6,21],"394":[2,1,53],"395":[3,2,42],"396":[1,2,37],"397":[4,1,21],"398":[8,4,181],"399":[6,4,82],"400":[6,4,100],"401":[3,1,1],"402":[5,3,82],"403":[7,3,73],"404":[9,3,41],"405":[6,3,62],"406":[15,3,80],"407":[8,3,65],"408":[5,3,33],"409":[8,3,70],"410":[10,3,26],"411":[8,3,51],"412":[10,3,73],"413":[11,3,42],"414":[15,3,34],"415":[5,1,69],"416":[5,5,122],"417":[1,9,86],"418":[6,9,122],"419":[5,5,130],"420":[6,5,1],"421":[3,11,120],"422":[4,1,172],"423":[1,1,90],"424":[7,1,158],"425":[3,7,163],"426":[4,1,43],"427":[4,4,1],"428":[5,4,1],"429":[6,8,20],"430":[9,8,15],"431":[5,8,25],"432":[6,8,22],"433":[5,4,1],"434":[6,8,17],"435":[7,8,29],"436":[6,8,22],"437":[3,1,1],"438":[2,3,70],"439":[3,3,44],"440":[4,1,60],"441":[7,4,93],"442":[4,4,143],"443":[4,1,1],"444":[5,4,68],"445":[1,4,73],"446":[2,4,1],"447":[2,5,35],"448":[2,5,37],"449":[4,4,82],"450":[1,8,109],"451":[3,1,30],"452":[2,3,34],"453":[2,3,31],"454":[3,1,1],"455":[4,3,33],"456":[3,3,1],"457":[3,4,33],"458":[3,4,68],"459":[3,4,41],"460":[2,4,41],"461":[1,4,1],"462":[3,5,15],"463":[3,5,27],"464":[4,5,9],"465":[2,1,45],"466":[4,2,64],"467":[2,2,1],"468":[3,4,1],"469":[1,2,49],"470":[2,3,74],"471":[6,3,116],"472":[3,2,24],"473":[1,3,41],"474":[1,3,12],"475":[1,2,19],"476":[2,2,9],"477":[6,1,23],"478":[4,6,108],"479":[2,6,30],"480":[5,6,16],"481":[3,7,7],"482":[5,6,1],"483":[3,8,17],"484":[4,8,85],"485":[4,8,112],"486":[9,9,49],"487":[7,8,23],"488":[3,1,102],"489":[3,3,27],"490":[2,3,19],"491":[4,5,52],"492":[2,3,19],"493":[2,3,46],"494":[1,3,175],"495":[2,4,89],"496":[3,3,28],"497":[4,3,28],"498":[3,3,24],"499":[6,3,38],"500":[6,3,93],"501":[1,1,41],"502":[1,1,80],"503":[1,1,243],"504":[2,1,52],"505":[1,1,63],"506":[1,1,40],"507":[2,1,97],"508":[1,1,29],"509":[3,1,135],"510":[2,1,24],"511":[9,1,42],"512":[2,9,18],"513":[5,11,1],"514":[3,15,20],"515":[5,15,79],"516":[5,15,22],"517":[2,15,53],"518":[6,11,17],"519":[4,14,89],"520":[3,14,32],"521":[2,14,20],"522":[5,14,144],"523":[6,14,62],"524":[6,14,42],"525":[6,14,62],"526":[1,9,42],"527":[3,10,50],"528":[3,10,10],"529":[6,10,91],"530":[9,10,40],"531":[5,10,88],"532":[1,9,25],"533":[1,9,20],"534":[6,1,18],"535":[1,6,20],"536":[3,6,57],"537":[3,6,75],"538":[3,1,19],"539":[4,3,111],"540":[5,3,108],"541":[1,3,90],"542":[4,1,82],"543":[3,1,28],"544":[3,3,58],"545":[2,5,23],"546":[2,5,27],"547":[2,5,78],"548":[4,3,80],"549":[1,1,1],"550":[2,1,64],"551":[2,3,48],"552":[1,1,106],"553":[7,1,81],"554":[3,1,81],"555":[4,1,26],"556":[10,1,71],"557":[7,1,109],"558":[3,1,1],"559":[3,3,126],"560":[5,3,102],"561":[2,3,39],"562":[3,3,26],"563":[3,1,1],"564":[2,3,12],"565":[3,3,34],"566":[1,4,35],"567":[1,4,1],"568":[1,5,45],"569":[2,5,31],"570":[1,4,1],"571":[3,5,61],"572":[5,1,16],"573":[1,5,38],"574":[1,5,29],"575":[3,1,71],"576":[1,3,14],"577":[2,3,105],"578":[6,3,77],"579":[6,8,79],"580":[3,8,64],"581":[2,3,1],"582":[4,5,1],"583":[1,3,54],"584":[2,1,24],"585":[2,2,80],"586":[2,2,97],"587":[7,1,24],"588":[2,7,72],"589":[6,7,16],"590":[4,7,7],"591":[3,7,8],"592":[5,7,126],"593":[4,7,1],"594":[6,11,34],"595":[3,14,6],"596":[4,14,25],"597":[4,11,25],"598":[3,11,49],"599":[4,11,54],"600":[1,7,41],"601":[2,8,106],"602":[2,8,72],"603":[4,7,46],"604":[5,7,1],"605":[3,11,22],"606":[5,11,29],"607":[2,16,65],"608":[4,16,56],"609":[3,16,13],"610":[5,11,94],"611":[5,11,90],"612":[7,1,31],"613":[5,7,26],"614":[2,7,31],"615":[6,7,16],"616":[3,9,11],"617":[3,7,10],"618":[5,10,1],"619":[5,12,17],"620":[5,12,131],"621":[10,10,50],"622":[8,10,24],"623":[5,10,32],"624":[3,1,65],"625":[2,3,84],"626":[2,4,97],"627":[3,1,93],"628":[1,3,85],"629":[1,4,180],"630":[3,4,84],"631":[3,4,49],"632":[6,3,52],"633":[2,3,31],"634":[5,5,85],"635":[4,5,44],"636":[3,3,7],"637":[1,3,41],"638":[2,4,37],"639":[5,1,42],"640":[2,5,8],"641":[2,5,11],"642":[3,5,12],"643":[5,5,32],"644":[4,9,38],"645":[4,9,44],"646":[4,9,49],"647":[4,9,29],"648":[4,9,48],"649":[7,9,76],"650":[5,9,74],"651":[4,9,38],"652":[7,9,66],"653":[3,9,42],"654":[6,1,25],"655":[4,6,42],"656":[2,6,30],"657":[5,6,22],"658":[3,7,10],"659":[4,6,37],"660":[4,8,106],"661":[3,8,80],"662":[2,11,38],"663":[9,8,49],"664":[6,8,15],"665":[3,6,16],"666":[2,1,49],"667":[4,2,66],"668":[2,2,1],"669":[3,2,1],"670":[2,2,31],"671":[2,4,104],"672":[1,2,56],"673":[2,3,5],"674":[3,5,50],"675":[2,5,51],"676":[2,5,55],"677":[2,5,8],"678":[3,3,6],"679":[5,6,123],"680":[4,6,37],"681":[3,10,31],"682":[1,2,26],"683":[1,2,21],"684":[2,2,10],"685":[2,1,107],"686":[2,2,1],"687":[3,2,1],"688":[8,2,87],"689":[2,2,58],"690":[2,2,72],"691":[2,2,74],"692":[5,2,33],"693":[3,2,63],"694":[1,2,23],"695":[2,2,9],"696":[6,1,89],"697":[5,1,1],"698":[2,5,40],"699":[6,7,1],"700":[1,5,12],"701":[2,6,45],"702":[6,8,1],"703":[2,6,48],"704":[6,6,1],"705":[2,5,9],"706":[3,1,25],"707":[2,3,48],"708":[2,3,29],"709":[2,3,76],"710":[10,1,13],"711":[2,10,22],"712":[6,12,113],"713":[3,10,1],"714":[4,13,72],"715":[3,13,106],"716":[3,13,93],"717":[6,1,23],"718":[2,6,29],"719":[5,6,26],"720":[1,6,8],"721":[5,6,113],"722":[5,6,48],"723":[3,10,10],"724":[4,10,47],"725":[4,10,8],"726":[7,12,23],"727":[4,6,152],"728":[4,6,16],"729":[5,6,19],"730":[2,6,15]},"averageFieldLength":[3.7181942544459643,5.659370725034197,56.12175102599179],"storedFields":{"0":{"title":"Celestia Documentation Site","titles":[]},"1":{"title":"Building the site","titles":["Celestia Documentation Site"]},"2":{"title":"Contribution Guidelines","titles":["Celestia Documentation Site"]},"3":{"title":"Directory Structure","titles":["Celestia Documentation Site"]},"4":{"title":"Feedback & Suggestions","titles":["Celestia Documentation Site"]},"5":{"title":"Community calendar","titles":[]},"6":{"title":"Celestia.org Code of Conduct","titles":[]},"7":{"title":"Our Pledge","titles":["Celestia.org Code of Conduct"]},"8":{"title":"Our Standards","titles":["Celestia.org Code of Conduct"]},"9":{"title":"Enforcement Responsibilities","titles":["Celestia.org Code of Conduct"]},"10":{"title":"Scope","titles":["Celestia.org Code of Conduct"]},"11":{"title":"Enforcement","titles":["Celestia.org Code of Conduct"]},"12":{"title":"Enforcement Guidelines","titles":["Celestia.org Code of Conduct"]},"13":{"title":"1. Correction","titles":["Celestia.org Code of Conduct","Enforcement Guidelines"]},"14":{"title":"2. Warning","titles":["Celestia.org Code of Conduct","Enforcement Guidelines"]},"15":{"title":"3. Temporary Ban","titles":["Celestia.org Code of Conduct","Enforcement Guidelines"]},"16":{"title":"4. Permanent Ban","titles":["Celestia.org Code of Conduct","Enforcement Guidelines"]},"17":{"title":"Attribution","titles":["Celestia.org Code of Conduct"]},"18":{"title":"The Celestia Foundation Delegation Program","titles":[]},"19":{"title":"Objectives of the program","titles":["The Celestia Foundation Delegation Program"]},"20":{"title":"Foundation delegation process","titles":["The Celestia Foundation Delegation Program"]},"21":{"title":"Program launch","titles":["The Celestia Foundation Delegation Program","Foundation delegation process"]},"22":{"title":"Cohort process","titles":["The Celestia Foundation Delegation Program","Foundation delegation process"]},"23":{"title":"Key Points","titles":["The Celestia Foundation Delegation Program","Foundation delegation process"]},"24":{"title":"Eligibility criteria","titles":["The Celestia Foundation Delegation Program","Foundation delegation process"]},"25":{"title":"Undelegation criteria","titles":["The Celestia Foundation Delegation Program","Foundation delegation process"]},"26":{"title":"Application","titles":["The Celestia Foundation Delegation Program"]},"27":{"title":"Application details","titles":["The Celestia Foundation Delegation Program","Application"]},"28":{"title":"Cohort information","titles":["The Celestia Foundation Delegation Program"]},"29":{"title":"Feedback process","titles":["The Celestia Foundation Delegation Program"]},"30":{"title":"SUPPLEMENTAL INCENTIVIZED TESTNET TERMS","titles":[]},"31":{"title":"1. General Terms","titles":["SUPPLEMENTAL INCENTIVIZED TESTNET TERMS"]},"32":{"title":"2. Taxes","titles":["SUPPLEMENTAL INCENTIVIZED TESTNET TERMS"]},"33":{"title":"3. Supplemental Privacy Information","titles":["SUPPLEMENTAL INCENTIVIZED TESTNET TERMS"]},"34":{"title":"4. Certain Additional Representations","titles":["SUPPLEMENTAL INCENTIVIZED TESTNET TERMS"]},"35":{"title":"Modular Meetup guide","titles":[]},"36":{"title":"Before the Meetup","titles":["Modular Meetup guide"]},"37":{"title":"During the Meetup","titles":["Modular Meetup guide"]},"38":{"title":"After the Meetup","titles":["Modular Meetup guide"]},"39":{"title":"Logistics and guidance","titles":["Modular Meetup guide"]},"40":{"title":"Venue","titles":["Modular Meetup guide","Logistics and guidance"]},"41":{"title":"Catering and refreshments","titles":["Modular Meetup guide","Logistics and guidance"]},"42":{"title":"Audience","titles":["Modular Meetup guide","Logistics and guidance"]},"43":{"title":"Speakers","titles":["Modular Meetup guide","Logistics and guidance"]},"44":{"title":"Sponsors","titles":["Modular Meetup guide","Logistics and guidance"]},"45":{"title":"Communications and marketing","titles":["Modular Meetup guide","Logistics and guidance"]},"46":{"title":"Recording","titles":["Modular Meetup guide","Logistics and guidance"]},"47":{"title":"Utilizing Meetup.com platform","titles":["Modular Meetup guide","Logistics and guidance"]},"48":{"title":"Onboarding questions for community members joining a Modular Meetup","titles":["Modular Meetup guide","Logistics and guidance"]},"49":{"title":"Celestia Modular Meetup program","titles":[]},"50":{"title":"Program description","titles":["Celestia Modular Meetup program"]},"51":{"title":"Important info","titles":["Celestia Modular Meetup program"]},"52":{"title":"Celestia.org Community Code of Conduct","titles":["Celestia Modular Meetup program","Important info"]},"53":{"title":"Signup form","titles":["Celestia Modular Meetup program","Important info"]},"54":{"title":"Emails","titles":["Celestia Modular Meetup program","Important info"]},"55":{"title":"Discord","titles":["Celestia Modular Meetup program","Important info"]},"56":{"title":"Materials","titles":["Celestia Modular Meetup program","Important info"]},"57":{"title":"Modular Meetup Toolkit","titles":[]},"58":{"title":"Celestia branding guidelines","titles":["Modular Meetup Toolkit"]},"59":{"title":"Sample “Introduction to Modularity” workshop presentation","titles":["Modular Meetup Toolkit"]},"60":{"title":"Sample “Run a Celestia light node” workshop presentation","titles":["Modular Meetup Toolkit"]},"61":{"title":"Sample “Deploy a Sovereign Rollup” workshop presentation","titles":["Modular Meetup Toolkit"]},"62":{"title":"Sample “Modular Meetup Introduction” workshop presentation","titles":["Modular Meetup Toolkit"]},"63":{"title":"Swag logistics","titles":["Modular Meetup Toolkit"]},"64":{"title":"Community overview","titles":[]},"65":{"title":"Speaker list","titles":[]},"66":{"title":"Bridging in and out of your Orbit rollup","titles":[]},"67":{"title":"Bridge in (deposit) to your rollup","titles":["Bridging in and out of your Orbit rollup"]},"68":{"title":"Step 1: Add your custom chain config","titles":["Bridging in and out of your Orbit rollup","Bridge in (deposit) to your rollup"]},"69":{"title":"Step 2: Deposit to your Orbit rollup","titles":["Bridging in and out of your Orbit rollup","Bridge in (deposit) to your rollup"]},"70":{"title":"Bridge out (withdrawal) from your rollup","titles":["Bridging in and out of your Orbit rollup"]},"71":{"title":"Step 1: Choose an amount to withdraw from your rollup","titles":["Bridging in and out of your Orbit rollup","Bridge out (withdrawal) from your rollup"]},"72":{"title":"Step 2: Claim your withdrawal","titles":["Bridging in and out of your Orbit rollup","Bridge out (withdrawal) from your rollup"]},"73":{"title":"Deploy a dapp on your Arbitrum rollup devnet","titles":[]},"74":{"title":"Dependencies","titles":["Deploy a dapp on your Arbitrum rollup devnet"]},"75":{"title":"Setup and contract deployment","titles":["Deploy a dapp on your Arbitrum rollup devnet"]},"76":{"title":"Interact with the contract","titles":["Deploy a dapp on your Arbitrum rollup devnet","Setup and contract deployment"]},"77":{"title":"Update the frontend","titles":["Deploy a dapp on your Arbitrum rollup devnet"]},"78":{"title":"Interact with the frontend","titles":["Deploy a dapp on your Arbitrum rollup devnet"]},"79":{"title":"Quickstart: Deploy an Arbitrum Orbit rollup","titles":[]},"80":{"title":"Prerequisites","titles":["Quickstart: Deploy an Arbitrum Orbit rollup"]},"81":{"title":"Setup","titles":["Quickstart: Deploy an Arbitrum Orbit rollup"]},"82":{"title":"Step 1: Acquire Arbitrum Sepolia ETH","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"83":{"title":"Step 2: Pick your deployment type","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"84":{"title":"Step 3: Configure your Orbit chain\'s deployment","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"85":{"title":"Step 3: Review & Deploy your Orbit chain","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"86":{"title":"Step 4: Download your chain\'s configuration files and launch your chain","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"87":{"title":"Step 5: Clone the setup script repository and add your configuration files","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"88":{"title":"Step 6: Pick an L2 RPC URL for the Batch Poster","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"89":{"title":"Step 7: Run your light node for Mocha testnet","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"90":{"title":"Step 8: Run your chain\'s node and block explorer","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"91":{"title":"Step 9: Finish setting up your chain","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"92":{"title":"Congratulations with Celestia underneath","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"93":{"title":"Appendix","titles":["Quickstart: Deploy an Arbitrum Orbit rollup"]},"94":{"title":"Compatibility matrix","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Appendix"]},"95":{"title":"Blobstream X contract deployments","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Appendix"]},"96":{"title":"Arbitrum Sepolia","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Appendix","Blobstream X contract deployments"]},"97":{"title":"Base Sepolia","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Appendix","Blobstream X contract deployments"]},"98":{"title":"Arbitrum Sepolia additional deployments","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Appendix"]},"99":{"title":"Base Sepolia additional deployments","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Appendix"]},"100":{"title":"Running a full node and/or validator","titles":[]},"101":{"title":"Prerequisites","titles":["Running a full node and/or validator"]},"102":{"title":"Running a full node","titles":["Running a full node and/or validator"]},"103":{"title":"Running a full node with validation","titles":["Running a full node and/or validator"]},"104":{"title":"Introduction to Arbitrum rollups with Celestia as DA","titles":[]},"105":{"title":"Overview","titles":["Introduction to Arbitrum rollups with Celestia as DA"]},"106":{"title":"Key components","titles":["Introduction to Arbitrum rollups with Celestia as DA"]},"107":{"title":"DA provider implementation","titles":["Introduction to Arbitrum rollups with Celestia as DA","Key components"]},"108":{"title":"Preimage Oracle Implementation","titles":["Introduction to Arbitrum rollups with Celestia as DA","Key components"]},"109":{"title":"Blobstream X implementation","titles":["Introduction to Arbitrum rollups with Celestia as DA","Key components"]},"110":{"title":"Ethereum fallback mechanism in Nitro","titles":["Introduction to Arbitrum rollups with Celestia as DA","Key components"]},"111":{"title":"Next steps","titles":["Introduction to Arbitrum rollups with Celestia as DA"]},"112":{"title":"Deploy a smart contract to your Arbitrum rollup","titles":[]},"113":{"title":"Overview","titles":["Deploy a smart contract to your Arbitrum rollup"]},"114":{"title":"Prerequisites","titles":["Deploy a smart contract to your Arbitrum rollup"]},"115":{"title":"Setup","titles":["Deploy a smart contract to your Arbitrum rollup"]},"116":{"title":"Create your smart contract","titles":["Deploy a smart contract to your Arbitrum rollup"]},"117":{"title":"Test your smart contract","titles":["Deploy a smart contract to your Arbitrum rollup"]},"118":{"title":"Deploying your smart contract","titles":["Deploy a smart contract to your Arbitrum rollup"]},"119":{"title":"Funded accounts","titles":["Deploy a smart contract to your Arbitrum rollup","Deploying your smart contract"]},"120":{"title":"Using our Arbitrum devnet","titles":["Deploy a smart contract to your Arbitrum rollup","Deploying your smart contract"]},"121":{"title":"Interacting with your smart contract","titles":["Deploy a smart contract to your Arbitrum rollup"]},"122":{"title":"Next steps","titles":["Deploy a smart contract to your Arbitrum rollup"]},"123":{"title":"Integrate with Blobstream contracts","titles":[]},"124":{"title":"Getting started","titles":["Integrate with Blobstream contracts"]},"125":{"title":"Prerequisites","titles":["Integrate with Blobstream contracts","Getting started"]},"126":{"title":"Installing Blobstream X contracts","titles":["Integrate with Blobstream contracts","Getting started"]},"127":{"title":"Example usage","titles":["Integrate with Blobstream contracts","Getting started"]},"128":{"title":"Data structures","titles":["Integrate with Blobstream contracts"]},"129":{"title":"Interface","titles":["Integrate with Blobstream contracts"]},"130":{"title":"Querying the proof","titles":["Integrate with Blobstream contracts"]},"131":{"title":"Verifying data inclusion for fraud proofs","titles":["Integrate with Blobstream contracts"]},"132":{"title":"Integrate with Blobstream client","titles":[]},"133":{"title":"Blobstream demo rollup","titles":["Integrate with Blobstream client"]},"134":{"title":"Defining a chain","titles":["Integrate with Blobstream client"]},"135":{"title":"Rollup sequencer","titles":["Integrate with Blobstream client"]},"136":{"title":"Committing to data","titles":["Integrate with Blobstream client","Rollup sequencer"]},"137":{"title":"Creating blocks","titles":["Integrate with Blobstream client","Rollup sequencer"]},"138":{"title":"Rollup full node","titles":["Integrate with Blobstream client"]},"139":{"title":"Downloading the block","titles":["Integrate with Blobstream client","Rollup full node"]},"140":{"title":"More documentation","titles":["Integrate with Blobstream client","Rollup full node"]},"141":{"title":"Proving inclusion via Blobstream","titles":["Integrate with Blobstream client","Rollup full node","More documentation"]},"142":{"title":"Submitting block data to Celestia via light node","titles":["Integrate with Blobstream client","Rollup full node","More documentation"]},"143":{"title":"Posting headers to Ethereum","titles":["Integrate with Blobstream client","Rollup full node","More documentation"]},"144":{"title":"Blobstream proofs queries","titles":[]},"145":{"title":"Prerequisites","titles":["Blobstream proofs queries"]},"146":{"title":"Overview of the proof queries","titles":["Blobstream proofs queries"]},"147":{"title":"The Celestia square","titles":["Blobstream proofs queries","Overview of the proof queries"]},"148":{"title":"The commitment scheme","titles":["Blobstream proofs queries","Overview of the proof queries"]},"149":{"title":"Hands-on demonstration","titles":["Blobstream proofs queries"]},"150":{"title":"1. Data root inclusion proof","titles":["Blobstream proofs queries","Hands-on demonstration"]},"151":{"title":"Full example of proving that a Celestia block was committed to by Blobstream X contract","titles":["Blobstream proofs queries","Hands-on demonstration"]},"152":{"title":"2. Transaction inclusion proof","titles":["Blobstream proofs queries","Hands-on demonstration"]},"153":{"title":"Converting the proofs to be usable in the DAVerifier library","titles":["Blobstream proofs queries"]},"154":{"title":"data","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"155":{"title":"shareProofs","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"156":{"title":"namespace","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"157":{"title":"rowRoots","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"158":{"title":"rowProofs","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"159":{"title":"attestationProof","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"160":{"title":"Listening for new data commitments","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"161":{"title":"Example rollup that uses the DAVerifier","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"162":{"title":"Conclusion","titles":["Blobstream proofs queries"]},"163":{"title":"Introduction to Blobstream rollups","titles":[]},"164":{"title":"Concepts","titles":["Introduction to Blobstream rollups"]},"165":{"title":"Share commitment","titles":["Introduction to Blobstream rollups","Concepts"]},"166":{"title":"Share commitment: Proof details","titles":["Introduction to Blobstream rollups","Concepts","Share commitment"]},"167":{"title":"Share commitment: Compact proofs","titles":["Introduction to Blobstream rollups","Concepts","Share commitment"]},"168":{"title":"Share commitment: Pros","titles":["Introduction to Blobstream rollups","Concepts","Share commitment"]},"169":{"title":"Share commitment: Cons","titles":["Introduction to Blobstream rollups","Concepts","Share commitment"]},"170":{"title":"Sequence of spans","titles":["Introduction to Blobstream rollups","Concepts"]},"171":{"title":"Sequence of spans: Proof details","titles":["Introduction to Blobstream rollups","Concepts","Sequence of spans"]},"172":{"title":"Sequence of spans: Proving unavailable data","titles":["Introduction to Blobstream rollups","Concepts","Sequence of spans"]},"173":{"title":"Sequence of spans: Proving inclusion of some data","titles":["Introduction to Blobstream rollups","Concepts","Sequence of spans"]},"174":{"title":"Sequence of spans: Pros","titles":["Introduction to Blobstream rollups","Concepts","Sequence of spans"]},"175":{"title":"Sequence of spans: Cons","titles":["Introduction to Blobstream rollups","Concepts","Sequence of spans"]},"176":{"title":"Optimistic rollups","titles":["Introduction to Blobstream rollups"]},"177":{"title":"Optimistic rollups that use a sequence of spans","titles":["Introduction to Blobstream rollups","Optimistic rollups"]},"178":{"title":"Optimistic rollups that use a sequence of spans: Pros","titles":["Introduction to Blobstream rollups","Optimistic rollups","Optimistic rollups that use a sequence of spans"]},"179":{"title":"Optimistic rollups that use a sequence of spans: Cons","titles":["Introduction to Blobstream rollups","Optimistic rollups","Optimistic rollups that use a sequence of spans"]},"180":{"title":"Optimistic rollups that use a sequence of spans: Example","titles":["Introduction to Blobstream rollups","Optimistic rollups","Optimistic rollups that use a sequence of spans"]},"181":{"title":"Optimistic rollups that use share commitments","titles":["Introduction to Blobstream rollups","Optimistic rollups"]},"182":{"title":"Optimistic rollups that use share commitments: Pros","titles":["Introduction to Blobstream rollups","Optimistic rollups","Optimistic rollups that use share commitments"]},"183":{"title":"Optimistic rollups that use share commitments: Cons","titles":["Introduction to Blobstream rollups","Optimistic rollups","Optimistic rollups that use share commitments"]},"184":{"title":"Zk-rollups","titles":["Introduction to Blobstream rollups"]},"185":{"title":"Zk-rollups that use sequence of spans","titles":["Introduction to Blobstream rollups","Zk-rollups"]},"186":{"title":"Zk-rollups that use sequence of spans: Pros","titles":["Introduction to Blobstream rollups","Zk-rollups","Zk-rollups that use sequence of spans"]},"187":{"title":"Zk-rollups that use sequence of spans: Cons","titles":["Introduction to Blobstream rollups","Zk-rollups","Zk-rollups that use sequence of spans"]},"188":{"title":"Zk-rollups that use share commitments","titles":["Introduction to Blobstream rollups","Zk-rollups"]},"189":{"title":"Protobuf deserialization inside a zk-circuit","titles":["Introduction to Blobstream rollups","Zk-rollups","Zk-rollups that use share commitments"]},"190":{"title":"Zk-rollups that use share commitments: Pros","titles":["Introduction to Blobstream rollups","Zk-rollups","Zk-rollups that use share commitments"]},"191":{"title":"Zk-rollups that use share commitments: Cons","titles":["Introduction to Blobstream rollups","Zk-rollups","Zk-rollups that use share commitments"]},"192":{"title":"Heavy merkle proofs usage","titles":["Introduction to Blobstream rollups","Zk-rollups"]},"193":{"title":"heavy merkle proofs usage: Pros","titles":["Introduction to Blobstream rollups","Zk-rollups","Heavy merkle proofs usage"]},"194":{"title":"heavy merkle proofs usage: Cons","titles":["Introduction to Blobstream rollups","Zk-rollups","Heavy merkle proofs usage"]},"195":{"title":"Conclusion","titles":["Introduction to Blobstream rollups"]},"196":{"title":"New Blobstream X deployments","titles":[]},"197":{"title":"Deploy a new SuccinctGateway contract","titles":["New Blobstream X deployments"]},"198":{"title":"Deploy a BlobstreamX contract to the new chain","titles":["New Blobstream X deployments"]},"199":{"title":"Run a local prover","titles":["New Blobstream X deployments"]},"200":{"title":"Blobstream: Streaming modular DA to Ethereum","titles":[]},"201":{"title":"What is Blobstream?","titles":["Blobstream: Streaming modular DA to Ethereum"]},"202":{"title":"Blobstream vs. data availability committees (DACs)","titles":["Blobstream: Streaming modular DA to Ethereum"]},"203":{"title":"Decentralization and security","titles":["Blobstream: Streaming modular DA to Ethereum","Blobstream vs. data availability committees (DACs)"]},"204":{"title":"Mechanism of verification","titles":["Blobstream: Streaming modular DA to Ethereum","Blobstream vs. data availability committees (DACs)"]},"205":{"title":"Flexibility and scalability","titles":["Blobstream: Streaming modular DA to Ethereum","Blobstream vs. data availability committees (DACs)"]},"206":{"title":"What is Blobstream X?","titles":["Blobstream: Streaming modular DA to Ethereum"]},"207":{"title":"Integrate with Blobstream X","titles":["Blobstream: Streaming modular DA to Ethereum"]},"208":{"title":"How Blobstream X works","titles":["Blobstream: Streaming modular DA to Ethereum","Integrate with Blobstream X"]},"209":{"title":"How to integrate with Blobstream X","titles":["Blobstream: Streaming modular DA to Ethereum","Integrate with Blobstream X"]},"210":{"title":"Deployed contracts","titles":["Blobstream: Streaming modular DA to Ethereum","Integrate with Blobstream X"]},"211":{"title":"Blobstream rollups","titles":["Blobstream: Streaming modular DA to Ethereum","Integrate with Blobstream X"]},"212":{"title":"Bubs testnet","titles":[]},"213":{"title":"Built with the OP Stack and Celestia","titles":["Bubs testnet"]},"214":{"title":"Building on Bubs","titles":["Bubs testnet"]},"215":{"title":"RPC URLs","titles":["Bubs testnet","Building on Bubs"]},"216":{"title":"HTTPS","titles":["Bubs testnet","Building on Bubs","RPC URLs"]},"217":{"title":"WSS","titles":["Bubs testnet","Building on Bubs","RPC URLs"]},"218":{"title":"Bridge","titles":["Bubs testnet","Building on Bubs"]},"219":{"title":"Faucet","titles":["Bubs testnet","Building on Bubs"]},"220":{"title":"Explorer","titles":["Bubs testnet","Building on Bubs"]},"221":{"title":"Status","titles":["Bubs testnet","Building on Bubs"]},"222":{"title":"Next steps","titles":["Bubs testnet"]},"223":{"title":"Build whatever","titles":[]},"224":{"title":"Quickstart - Building on Celestia","titles":["Build whatever"]},"225":{"title":"Choose a framework","titles":["Build whatever","Quickstart - Building on Celestia"]},"226":{"title":"Rollups-as-a-Service","titles":["Build whatever","Quickstart - Building on Celestia"]},"227":{"title":"Smart contracts","titles":["Build whatever","Quickstart - Building on Celestia"]},"228":{"title":"What is a rollup?","titles":["Build whatever"]},"229":{"title":"What is a modular blockchain?","titles":["Build whatever"]},"230":{"title":"Benefits of modular blockchains","titles":["Build whatever"]},"231":{"title":"Ease of deploying a chain","titles":["Build whatever","Benefits of modular blockchains"]},"232":{"title":"Scaling","titles":["Build whatever","Benefits of modular blockchains"]},"233":{"title":"Customizability","titles":["Build whatever","Benefits of modular blockchains"]},"234":{"title":"Create a wallet with celestia-node","titles":[]},"235":{"title":"Using the cel-key utility","titles":["Create a wallet with celestia-node"]},"236":{"title":"Installation","titles":["Create a wallet with celestia-node","Using the cel-key utility"]},"237":{"title":"Steps for generating node keys","titles":["Create a wallet with celestia-node","Using the cel-key utility"]},"238":{"title":"Steps for exporting node keys","titles":["Create a wallet with celestia-node","Using the cel-key utility"]},"239":{"title":"Steps for importing node keys","titles":["Create a wallet with celestia-node","Using the cel-key utility"]},"240":{"title":"View all options for cel-key","titles":["Create a wallet with celestia-node","Using the cel-key utility"]},"241":{"title":"Docker and cel-key","titles":["Create a wallet with celestia-node"]},"242":{"title":"Prerequisites","titles":["Create a wallet with celestia-node","Docker and cel-key"]},"243":{"title":"Running your node","titles":["Create a wallet with celestia-node","Docker and cel-key"]},"244":{"title":"Mounting existing keys to container","titles":["Create a wallet with celestia-node","Docker and cel-key"]},"245":{"title":"Deploy a smart contract on Bubs testnet","titles":[]},"246":{"title":"Dependencies","titles":["Deploy a smart contract on Bubs testnet"]},"247":{"title":"Setup","titles":["Deploy a smart contract on Bubs testnet"]},"248":{"title":"Create your smart contract","titles":["Deploy a smart contract on Bubs testnet"]},"249":{"title":"Test your smart contract","titles":["Deploy a smart contract on Bubs testnet"]},"250":{"title":"Deploying your smart contract","titles":["Deploy a smart contract on Bubs testnet"]},"251":{"title":"Using Anvil","titles":["Deploy a smart contract on Bubs testnet","Deploying your smart contract"]},"252":{"title":"Using Bubs","titles":["Deploy a smart contract on Bubs testnet","Deploying your smart contract"]},"253":{"title":"Interacting with your smart contract","titles":["Deploy a smart contract on Bubs testnet"]},"254":{"title":"Next steps","titles":["Deploy a smart contract on Bubs testnet"]},"255":{"title":"Ethereum fallback","titles":[]},"256":{"title":"Full stack modular blockchain development guide","titles":[]},"257":{"title":"Getting started","titles":["Full stack modular blockchain development guide"]},"258":{"title":"Pre-requisites","titles":["Full stack modular blockchain development guide","Getting started"]},"259":{"title":"Project setup","titles":["Full stack modular blockchain development guide","Getting started"]},"260":{"title":"Updating the contract and tests","titles":["Full stack modular blockchain development guide","Getting started","Project setup"]},"261":{"title":"Running the test","titles":["Full stack modular blockchain development guide","Getting started","Project setup"]},"262":{"title":"Updating the deployment script","titles":["Full stack modular blockchain development guide","Getting started","Project setup"]},"263":{"title":"Deploying locally","titles":["Full stack modular blockchain development guide","Getting started","Project setup"]},"264":{"title":"Deploying to the Ethermint Sovereign Rollup","titles":["Full stack modular blockchain development guide","Getting started"]},"265":{"title":"Building the frontend","titles":["Full stack modular blockchain development guide","Getting started"]},"266":{"title":"Configuring environment variables","titles":["Full stack modular blockchain development guide","Getting started","Building the frontend"]},"267":{"title":"Configuring the entrypoint","titles":["Full stack modular blockchain development guide","Getting started"]},"268":{"title":"Creating and reading posts","titles":["Full stack modular blockchain development guide","Getting started"]},"269":{"title":"Adding Ethermint Chain to MetaMask","titles":["Full stack modular blockchain development guide","Getting started"]},"270":{"title":"Testing it out on Ethermint","titles":["Full stack modular blockchain development guide","Getting started"]},"271":{"title":"Now give it a spin 🌀","titles":["Full stack modular blockchain development guide","Getting started"]},"272":{"title":"Deploying a dapp on Bubs testnet","titles":[]},"273":{"title":"Update the frontend","titles":["Deploying a dapp on Bubs testnet"]},"274":{"title":"Interact with the frontend","titles":["Deploying a dapp on Bubs testnet"]},"275":{"title":"Next steps","titles":["Deploying a dapp on Bubs testnet"]},"276":{"title":"Integrate Celestia for service providers","titles":[]},"277":{"title":"Getting started","titles":["Integrate Celestia for service providers"]},"278":{"title":"Celestia service provider notes","titles":["Integrate Celestia for service providers"]},"279":{"title":"Custody and key management","titles":["Integrate Celestia for service providers","Celestia service provider notes"]},"280":{"title":"RPC and querying","titles":["Integrate Celestia for service providers","Celestia service provider notes"]},"281":{"title":"Compatibility","titles":["Integrate Celestia for service providers","Celestia service provider notes"]},"282":{"title":"Syncing","titles":["Integrate Celestia for service providers","Celestia service provider notes"]},"283":{"title":"Notable exceptions relative to other blockchains","titles":["Integrate Celestia for service providers","Celestia service provider notes"]},"284":{"title":"Introduction to OP Stack integration","titles":[]},"285":{"title":"About the integration","titles":["Introduction to OP Stack integration"]},"286":{"title":"Table of contents of the category","titles":["Introduction to OP Stack integration","About the integration"]},"287":{"title":"Celestia and OP Stack repository","titles":["Introduction to OP Stack integration"]},"288":{"title":"What are Optimism and the OP Stack?","titles":["Introduction to OP Stack integration","Celestia and OP Stack repository"]},"289":{"title":"What is Celestia?","titles":["Introduction to OP Stack integration","Celestia and OP Stack repository"]},"290":{"title":"OP Stack and Celestia","titles":["Introduction to OP Stack integration"]},"291":{"title":"Next steps","titles":["Introduction to OP Stack integration"]},"292":{"title":"Node API","titles":[]},"293":{"title":"RPC API","titles":["Node API"]},"294":{"title":"Library","titles":["Node API","RPC API"]},"295":{"title":"RPC","titles":["Node API","RPC API"]},"296":{"title":"RPC API tutorial","titles":["Node API","RPC API"]},"297":{"title":"Gateway API","titles":["Node API"]},"298":{"title":"Gateway API tutorial","titles":["Node API","Gateway API"]},"299":{"title":"Celestia-node RPC CLI tutorial","titles":[]},"300":{"title":"Introduction","titles":["Celestia-node RPC CLI tutorial"]},"301":{"title":"Blobs","titles":["Celestia-node RPC CLI tutorial","Introduction"]},"302":{"title":"Namespaces","titles":["Celestia-node RPC CLI tutorial","Introduction"]},"303":{"title":"Hardware requirements","titles":["Celestia-node RPC CLI tutorial"]},"304":{"title":"Setting up dependencies","titles":["Celestia-node RPC CLI tutorial"]},"305":{"title":"Install Golang","titles":["Celestia-node RPC CLI tutorial"]},"306":{"title":"Celestia-node","titles":["Celestia-node RPC CLI tutorial"]},"307":{"title":"Install celestia-node","titles":["Celestia-node RPC CLI tutorial","Celestia-node"]},"308":{"title":"Instantiate a Celestia light node","titles":["Celestia-node RPC CLI tutorial","Celestia-node"]},"309":{"title":"Connect to a public core endpoint","titles":["Celestia-node RPC CLI tutorial","Celestia-node"]},"310":{"title":"Keys and wallets","titles":["Celestia-node RPC CLI tutorial","Celestia-node"]},"311":{"title":"RPC CLI guide","titles":["Celestia-node RPC CLI tutorial"]},"312":{"title":"Command formatting","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide"]},"313":{"title":"Basic flags","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide"]},"314":{"title":"Auth token 🔐","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Basic flags"]},"315":{"title":"Node store","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Basic flags"]},"316":{"title":"Auth token on custom or private network","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Basic flags","Node store"]},"317":{"title":"Submitting data","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide"]},"318":{"title":"Optional: Submit with curl","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Submitting data"]},"319":{"title":"Retrieving data","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide"]},"320":{"title":"Setting the gas price","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide"]},"321":{"title":"Examples","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide"]},"322":{"title":"Check your balance","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"323":{"title":"Check the balance of another address","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"324":{"title":"Get your node ID","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"325":{"title":"Get your account address","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"326":{"title":"Get block header by height","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"327":{"title":"Combined commands","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"328":{"title":"Get data availability sampler stats","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"329":{"title":"Transfer balance of utia to another account","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"330":{"title":"API version","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"331":{"title":"Help","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"332":{"title":"Advanced example","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide"]},"333":{"title":"Additional resources","titles":["Celestia-node RPC CLI tutorial"]},"334":{"title":"Submitting a blob using curl","titles":["Celestia-node RPC CLI tutorial","Additional resources"]},"335":{"title":"Post an SVG as a PFB","titles":["Celestia-node RPC CLI tutorial","Additional resources"]},"336":{"title":"Golang guide","titles":["Celestia-node RPC CLI tutorial","Additional resources"]},"337":{"title":"Troubleshooting","titles":["Celestia-node RPC CLI tutorial","Additional resources"]},"338":{"title":"Deploy an OP Stack devnet","titles":[]},"339":{"title":"Dependency setup","titles":["Deploy an OP Stack devnet"]},"340":{"title":"Environment setup and Golang installation","titles":["Deploy an OP Stack devnet","Dependency setup"]},"341":{"title":"Clone repository","titles":["Deploy an OP Stack devnet","Dependency setup"]},"342":{"title":"Build devnet","titles":["Deploy an OP Stack devnet"]},"343":{"title":"Start devnet","titles":["Deploy an OP Stack devnet","Build devnet"]},"344":{"title":"View the logs of the devnet","titles":["Deploy an OP Stack devnet","Build devnet"]},"345":{"title":"Stop devnet","titles":["Deploy an OP Stack devnet","Build devnet"]},"346":{"title":"Viewing containers","titles":["Deploy an OP Stack devnet","Build devnet"]},"347":{"title":"Find a transaction","titles":["Deploy an OP Stack devnet"]},"348":{"title":"Read the transaction call data","titles":["Deploy an OP Stack devnet"]},"349":{"title":"Find the data on Celestia","titles":["Deploy an OP Stack devnet"]},"350":{"title":"Ethereum fallback mechanism in OP Stack","titles":["Deploy an OP Stack devnet"]},"351":{"title":"Implementation of fallback","titles":["Deploy an OP Stack devnet","Ethereum fallback mechanism in OP Stack"]},"352":{"title":"Testing the fallback","titles":["Deploy an OP Stack devnet","Ethereum fallback mechanism in OP Stack"]},"353":{"title":"Span batches","titles":["Deploy an OP Stack devnet"]},"354":{"title":"Deploy an OP Stack devnet to Celestia","titles":[]},"355":{"title":"Pick your deployment type","titles":["Deploy an OP Stack devnet to Celestia"]},"356":{"title":"Using a local devnet","titles":["Deploy an OP Stack devnet to Celestia","Pick your deployment type"]},"357":{"title":"Using a light node","titles":["Deploy an OP Stack devnet to Celestia","Pick your deployment type"]},"358":{"title":"Using a RaaS provider","titles":["Deploy an OP Stack devnet to Celestia","Pick your deployment type"]},"359":{"title":"Build the devnet","titles":["Deploy an OP Stack devnet to Celestia"]},"360":{"title":"Start the devnet","titles":["Deploy an OP Stack devnet to Celestia"]},"361":{"title":"View the logs of the devnet","titles":["Deploy an OP Stack devnet to Celestia"]},"362":{"title":"Stop the devnet","titles":["Deploy an OP Stack devnet to Celestia"]},"363":{"title":"Clean the devnet","titles":["Deploy an OP Stack devnet to Celestia"]},"364":{"title":"Deploying to an L1 (or L2)","titles":["Deploy an OP Stack devnet to Celestia"]},"365":{"title":"Prompt scavenger","titles":[]},"366":{"title":"Dependencies","titles":["Prompt scavenger"]},"367":{"title":"Install Celestia Node and run a light node","titles":["Prompt scavenger","Dependencies"]},"368":{"title":"Node API authentication key","titles":["Prompt scavenger","Dependencies"]},"369":{"title":"OpenAI key","titles":["Prompt scavenger","Dependencies"]},"370":{"title":"Building the Prompt Scavenger","titles":["Prompt scavenger"]},"371":{"title":"Copy over Go files","titles":["Prompt scavenger","Building the Prompt Scavenger"]},"372":{"title":"Build your import statements","titles":["Prompt scavenger","Building the Prompt Scavenger"]},"373":{"title":"Helpful functions","titles":["Prompt scavenger","Building the Prompt Scavenger"]},"374":{"title":"Wrapping up the functions","titles":["Prompt scavenger","Building the Prompt Scavenger"]},"375":{"title":"Next steps","titles":["Prompt scavenger"]},"376":{"title":"Requesting data commitment ranges","titles":[]},"377":{"title":"Recommended setup","titles":["Requesting data commitment ranges"]},"378":{"title":"Local proving","titles":["Requesting data commitment ranges"]},"379":{"title":"Request proof onchain","titles":["Requesting data commitment ranges"]},"380":{"title":"Rollkit","titles":[]},"381":{"title":"Tutorials","titles":["Rollkit"]},"382":{"title":"Submitting data blobs to Celestia","titles":[]},"383":{"title":"Fee market and mempool","titles":["Submitting data blobs to Celestia"]},"384":{"title":"Fees and gas limits","titles":["Submitting data blobs to Celestia","Fee market and mempool"]},"385":{"title":"Estimating PFB gas","titles":["Submitting data blobs to Celestia","Fee market and mempool","Fees and gas limits"]},"386":{"title":"Gas fee calculation","titles":["Submitting data blobs to Celestia","Fee market and mempool","Fees and gas limits"]},"387":{"title":"Estimating gas programmatically","titles":["Submitting data blobs to Celestia","Fee market and mempool","Fees and gas limits"]},"388":{"title":"Submitting multiple transactions in one block from the same account","titles":["Submitting data blobs to Celestia","Fee market and mempool"]},"389":{"title":"API","titles":["Submitting data blobs to Celestia"]},"390":{"title":"The celestia-app consensus node CLI","titles":["Submitting data blobs to Celestia","API"]},"391":{"title":"The celestia-node light node CLI","titles":["Submitting data blobs to Celestia","API"]},"392":{"title":"GRPC to a consensus node via the user package","titles":["Submitting data blobs to Celestia","API"]},"393":{"title":"RPC to a celestia-node","titles":["Submitting data blobs to Celestia","API"]},"394":{"title":"Transaction resubmission","titles":[]},"395":{"title":"Monitoring and resubmission","titles":["Transaction resubmission"]},"396":{"title":"Notes","titles":["Transaction resubmission"]},"397":{"title":"Wallet integrations with Celestia","titles":[]},"398":{"title":"Add Celestia network parameters to Keplr with React","titles":["Wallet integrations with Celestia"]},"399":{"title":"Adding a custom chain to Leap","titles":["Wallet integrations with Celestia"]},"400":{"title":"Adding a custom chain to Cosmostation","titles":["Wallet integrations with Celestia"]},"401":{"title":"Data availability FAQ","titles":[]},"402":{"title":"What is data availability?","titles":["Data availability FAQ"]},"403":{"title":"What is the data availability problem?","titles":["Data availability FAQ"]},"404":{"title":"How do nodes verify data availability in Celestia?","titles":["Data availability FAQ"]},"405":{"title":"What is data availability sampling?","titles":["Data availability FAQ"]},"406":{"title":"What are some of the security assumptions that Celestia makes for data availability sampling?","titles":["Data availability FAQ"]},"407":{"title":"Why is block reconstruction necessary for security?","titles":["Data availability FAQ"]},"408":{"title":"What is data storage?","titles":["Data availability FAQ"]},"409":{"title":"What is the problem around data storage?","titles":["Data availability FAQ"]},"410":{"title":"What is the difference between data availability and data storage?","titles":["Data availability FAQ"]},"411":{"title":"Where does blockchain state fit into this?","titles":["Data availability FAQ"]},"412":{"title":"Why doesn’t Celestia incentivize storage of historical data?","titles":["Data availability FAQ"]},"413":{"title":"Who may store historical data if there is no reward?","titles":["Data availability FAQ"]},"414":{"title":"What are some things blockchains can do to provide stronger assurances of data retrievability?","titles":["Data availability FAQ"]},"415":{"title":"Celestia\'s data availability layer","titles":[]},"416":{"title":"Data availability sampling (DAS)","titles":["Celestia\'s data availability layer"]},"417":{"title":"Scalability","titles":["Celestia\'s data availability layer","Data availability sampling (DAS)"]},"418":{"title":"Fraud proofs of incorrectly extended data","titles":["Celestia\'s data availability layer","Data availability sampling (DAS)"]},"419":{"title":"Namespaced Merkle trees (NMTs)","titles":["Celestia\'s data availability layer"]},"420":{"title":"Building a PoS blockchain for DA","titles":["Celestia\'s data availability layer"]},"421":{"title":"Providing data availability","titles":["Celestia\'s data availability layer","Building a PoS blockchain for DA"]},"422":{"title":"Monolithic vs. modular blockchains","titles":[]},"423":{"title":"Introduction","titles":[]},"424":{"title":"The lifecycle of a celestia-app transaction","titles":[]},"425":{"title":"Checking data availability","titles":["The lifecycle of a celestia-app transaction"]},"426":{"title":"How to stake TIA","titles":[]},"427":{"title":"Select your preferred wallet","titles":["How to stake TIA"]},"428":{"title":"Stake TIA with Keplr wallet","titles":["How to stake TIA"]},"429":{"title":"1️⃣ Open your Keplr browser extension","titles":["How to stake TIA","Stake TIA with Keplr wallet"]},"430":{"title":"2️⃣ Select Celestia network and search for a validator","titles":["How to stake TIA","Stake TIA with Keplr wallet"]},"431":{"title":"3️⃣ Stake your TIA tokens","titles":["How to stake TIA","Stake TIA with Keplr wallet"]},"432":{"title":"4️⃣ Confirm and manage your TIA","titles":["How to stake TIA","Stake TIA with Keplr wallet"]},"433":{"title":"Stake TIA with Leap wallet","titles":["How to stake TIA"]},"434":{"title":"1️⃣ Open your Leap browser extension","titles":["How to stake TIA","Stake TIA with Leap wallet"]},"435":{"title":"2️⃣ Select a validator and stake TIA","titles":["How to stake TIA","Stake TIA with Leap wallet"]},"436":{"title":"3️⃣ Confirm and manage your TIA","titles":["How to stake TIA","Stake TIA with Leap wallet"]},"437":{"title":"Paying for blobspace","titles":[]},"438":{"title":"PayForBlobs transactions","titles":["Paying for blobspace"]},"439":{"title":"Fee market overview","titles":["Paying for blobspace"]},"440":{"title":"Data retrievability and pruning","titles":[]},"441":{"title":"Data retrievability and pruning in celestia-node","titles":["Data retrievability and pruning"]},"442":{"title":"Suggested practices for rollups","titles":["Data retrievability and pruning"]},"443":{"title":"Staking, governance, & supply","titles":[]},"444":{"title":"Proof-of-stake on Celestia","titles":["Staking, governance, & supply"]},"445":{"title":"Inflation","titles":["Staking, governance, & supply"]},"446":{"title":"Decentralised governance","titles":["Staking, governance, & supply"]},"447":{"title":"Network parameters","titles":["Staking, governance, & supply","Decentralised governance"]},"448":{"title":"Community pool","titles":["Staking, governance, & supply","Decentralised governance"]},"449":{"title":"TIA allocation at genesis","titles":["Staking, governance, & supply"]},"450":{"title":"Unlocks","titles":["Staking, governance, & supply","TIA allocation at genesis"]},"451":{"title":"Staking on Celestia","titles":[]},"452":{"title":"Mainnet Beta","titles":["Staking on Celestia"]},"453":{"title":"Mocha testnet","titles":["Staking on Celestia"]},"454":{"title":"Overview of TIA","titles":[]},"455":{"title":"TIA at a glance","titles":["Overview of TIA"]},"456":{"title":"Role of TIA","titles":["Overview of TIA"]},"457":{"title":"Paying for blobspace","titles":["Overview of TIA","Role of TIA"]},"458":{"title":"Bootstrapping new rollups","titles":["Overview of TIA","Role of TIA"]},"459":{"title":"Proof-of-stake","titles":["Overview of TIA","Role of TIA"]},"460":{"title":"Decentralised governance","titles":["Overview of TIA","Role of TIA"]},"461":{"title":"Denominations","titles":["Overview of TIA","Role of TIA"]},"462":{"title":"TIA: display token","titles":["Overview of TIA","Role of TIA","Denominations"]},"463":{"title":"utia: staking denomination","titles":["Overview of TIA","Role of TIA","Denominations"]},"464":{"title":"microtia: staking denomination alias","titles":["Overview of TIA","Role of TIA","Denominations"]},"465":{"title":"Arabica devnet","titles":[]},"466":{"title":"Network stability and upgrades","titles":["Arabica devnet"]},"467":{"title":"Network details","titles":["Arabica devnet"]},"468":{"title":"Software version numbers","titles":["Arabica devnet","Network details"]},"469":{"title":"Integrations","titles":["Arabica devnet"]},"470":{"title":"RPC endpoints","titles":["Arabica devnet","Integrations"]},"471":{"title":"Using consensus endpoints with DA nodes","titles":["Arabica devnet","Integrations"]},"472":{"title":"Arabica devnet faucet","titles":["Arabica devnet"]},"473":{"title":"Discord","titles":["Arabica devnet","Arabica devnet faucet"]},"474":{"title":"Web","titles":["Arabica devnet","Arabica devnet faucet"]},"475":{"title":"Explorers","titles":["Arabica devnet"]},"476":{"title":"Network upgrades","titles":["Arabica devnet"]},"477":{"title":"Setting up a Celestia bridge node","titles":[]},"478":{"title":"Overview of bridge nodes","titles":["Setting up a Celestia bridge node"]},"479":{"title":"Hardware requirements","titles":["Setting up a Celestia bridge node"]},"480":{"title":"Setting up your bridge node","titles":["Setting up a Celestia bridge node"]},"481":{"title":"Setup the dependencies","titles":["Setting up a Celestia bridge node","Setting up your bridge node"]},"482":{"title":"Deploy the Celestia bridge node","titles":["Setting up a Celestia bridge node"]},"483":{"title":"Install Celestia Node","titles":["Setting up a Celestia bridge node","Deploy the Celestia bridge node"]},"484":{"title":"Initialize the bridge node","titles":["Setting up a Celestia bridge node","Deploy the Celestia bridge node"]},"485":{"title":"Run the bridge node","titles":["Setting up a Celestia bridge node","Deploy the Celestia bridge node"]},"486":{"title":"Optional: run the bridge node with a custom key","titles":["Setting up a Celestia bridge node","Deploy the Celestia bridge node","Run the bridge node"]},"487":{"title":"Optional: start the bridge node with SystemD","titles":["Setting up a Celestia bridge node","Deploy the Celestia bridge node"]},"488":{"title":"Helpful CLI commands","titles":[]},"489":{"title":"Creating a wallet","titles":["Helpful CLI commands"]},"490":{"title":"Key management","titles":["Helpful CLI commands"]},"491":{"title":"Importing and exporting keys","titles":["Helpful CLI commands","Key management"]},"492":{"title":"Querying subcommands","titles":["Helpful CLI commands"]},"493":{"title":"Token management","titles":["Helpful CLI commands"]},"494":{"title":"Governance","titles":["Helpful CLI commands"]},"495":{"title":"Community Pool","titles":["Helpful CLI commands","Governance"]},"496":{"title":"Claim validator rewards","titles":["Helpful CLI commands"]},"497":{"title":"Delegate & undelegate tokens","titles":["Helpful CLI commands"]},"498":{"title":"Unjailing the validator","titles":["Helpful CLI commands"]},"499":{"title":"How to export logs with SystemD","titles":["Helpful CLI commands"]},"500":{"title":"Signing genesis for a new network","titles":["Helpful CLI commands"]},"501":{"title":"Metrics","titles":[]},"502":{"title":"Setup","titles":["Metrics"]},"503":{"title":"Visualization","titles":["Metrics"]},"504":{"title":"Node exporter","titles":["Metrics"]},"505":{"title":"Alerts","titles":["Metrics"]},"506":{"title":"Multisig","titles":[]},"507":{"title":"Command line","titles":["Multisig"]},"508":{"title":"Resources","titles":["Multisig"]},"509":{"title":"Slashing on Celestia","titles":[]},"510":{"title":"Upgrade Monitor","titles":[]},"511":{"title":"How to create a vesting account with celestia-app","titles":[]},"512":{"title":"Local devnet","titles":["How to create a vesting account with celestia-app"]},"513":{"title":"Setting up the local devnet","titles":["How to create a vesting account with celestia-app","Local devnet"]},"514":{"title":"Run the devnet","titles":["How to create a vesting account with celestia-app","Local devnet","Setting up the local devnet"]},"515":{"title":"Save the home directory path","titles":["How to create a vesting account with celestia-app","Local devnet","Setting up the local devnet"]},"516":{"title":"Check the version of the devnet","titles":["How to create a vesting account with celestia-app","Local devnet","Setting up the local devnet"]},"517":{"title":"Next steps","titles":["How to create a vesting account with celestia-app","Local devnet","Setting up the local devnet"]},"518":{"title":"Setting up vesting account on devnet","titles":["How to create a vesting account with celestia-app","Local devnet"]},"519":{"title":"Create a new key","titles":["How to create a vesting account with celestia-app","Local devnet","Setting up vesting account on devnet"]},"520":{"title":"List your keys","titles":["How to create a vesting account with celestia-app","Local devnet","Setting up vesting account on devnet"]},"521":{"title":"Set variables","titles":["How to create a vesting account with celestia-app","Local devnet","Setting up vesting account on devnet"]},"522":{"title":"Create your devnet vesting account","titles":["How to create a vesting account with celestia-app","Local devnet","Setting up vesting account on devnet"]},"523":{"title":"Query the devnet vesting account details","titles":["How to create a vesting account with celestia-app","Local devnet","Setting up vesting account on devnet"]},"524":{"title":"Query the devnet base account details","titles":["How to create a vesting account with celestia-app","Local devnet","Setting up vesting account on devnet"]},"525":{"title":"Query the balances of the devnet accounts","titles":["How to create a vesting account with celestia-app","Local devnet","Setting up vesting account on devnet"]},"526":{"title":"Mocha","titles":["How to create a vesting account with celestia-app"]},"527":{"title":"Create a wallet","titles":["How to create a vesting account with celestia-app","Mocha"]},"528":{"title":"Fund your account","titles":["How to create a vesting account with celestia-app","Mocha"]},"529":{"title":"Create a vesting account on Mocha","titles":["How to create a vesting account with celestia-app","Mocha"]},"530":{"title":"Optional: Set up a full consensus node or validator","titles":["How to create a vesting account with celestia-app","Mocha"]},"531":{"title":"Optional: Change your client.toml","titles":["How to create a vesting account with celestia-app","Mocha"]},"532":{"title":"Notes","titles":["How to create a vesting account with celestia-app"]},"533":{"title":"Conclusion","titles":["How to create a vesting account with celestia-app"]},"534":{"title":"Create a wallet with celestia-app","titles":[]},"535":{"title":"Prerequisites","titles":["Create a wallet with celestia-app"]},"536":{"title":"Create a wallet","titles":["Create a wallet with celestia-app"]},"537":{"title":"Fund a wallet","titles":["Create a wallet with celestia-app"]},"538":{"title":"Install celestia-app","titles":[]},"539":{"title":"Building binary from source","titles":["Install celestia-app"]},"540":{"title":"Installing a pre-built binary","titles":["Install celestia-app"]},"541":{"title":"Ports","titles":["Install celestia-app"]},"542":{"title":"Custom networks and values","titles":[]},"543":{"title":"celestia-node metrics","titles":[]},"544":{"title":"Running metrics flags","titles":["celestia-node metrics"]},"545":{"title":"Mainnet Beta","titles":["celestia-node metrics","Running metrics flags"]},"546":{"title":"Mocha testnet","titles":["celestia-node metrics","Running metrics flags"]},"547":{"title":"TLS connections","titles":["celestia-node metrics","Running metrics flags"]},"548":{"title":"Metrics endpoint design considerations","titles":["celestia-node metrics"]},"549":{"title":"Troubleshooting","titles":[]},"550":{"title":"Network selection","titles":["Troubleshooting"]},"551":{"title":"Chain ID","titles":["Troubleshooting","Network selection"]},"552":{"title":"Ports","titles":["Troubleshooting"]},"553":{"title":"Changing the location of your node store","titles":["Troubleshooting"]},"554":{"title":"Resetting your config","titles":["Troubleshooting"]},"555":{"title":"Clearing the data store","titles":["Troubleshooting"]},"556":{"title":"FATAL headers given to the heightSub are in the wrong order","titles":["Troubleshooting"]},"557":{"title":"Error: "too many open files"","titles":["Troubleshooting"]},"558":{"title":"Install celestia-node","titles":[]},"559":{"title":"Installing from source","titles":["Install celestia-node"]},"560":{"title":"Installing a pre-built binary","titles":["Install celestia-node"]},"561":{"title":"Next steps","titles":["Install celestia-node"]},"562":{"title":"Upgrading your binary","titles":["Install celestia-node"]},"563":{"title":"config.toml guide","titles":[]},"564":{"title":"Pre-requisites","titles":["config.toml guide"]},"565":{"title":"Understanding config.toml","titles":["config.toml guide"]},"566":{"title":"Core","titles":["config.toml guide","Understanding config.toml"]},"567":{"title":"P2P","titles":["config.toml guide","Understanding config.toml"]},"568":{"title":"Bootstrap","titles":["config.toml guide","Understanding config.toml","P2P"]},"569":{"title":"Mutual peers","titles":["config.toml guide","Understanding config.toml","P2P"]},"570":{"title":"Services","titles":["config.toml guide","Understanding config.toml"]},"571":{"title":"TrustedHash and TrustedPeer","titles":["config.toml guide","Understanding config.toml","Services"]},"572":{"title":"Deciding which node to run","titles":[]},"573":{"title":"Beginner","titles":["Deciding which node to run"]},"574":{"title":"Advanced","titles":["Deciding which node to run"]},"575":{"title":"🐳 Docker setup","titles":[]},"576":{"title":"Prerequisites","titles":["🐳 Docker setup"]},"577":{"title":"Quick start","titles":["🐳 Docker setup"]},"578":{"title":"Light node setup with persistent storage","titles":["🐳 Docker setup"]},"579":{"title":"Initialize the node store and key","titles":["🐳 Docker setup","Light node setup with persistent storage"]},"580":{"title":"Start the node","titles":["🐳 Docker setup","Light node setup with persistent storage"]},"581":{"title":"Video walkthrough","titles":["🐳 Docker setup"]},"582":{"title":"2.5 minute version","titles":["🐳 Docker setup","Video walkthrough"]},"583":{"title":"Troubleshooting","titles":["🐳 Docker setup"]},"584":{"title":"Development environment","titles":[]},"585":{"title":"Install dependencies","titles":["Development environment"]},"586":{"title":"Install Golang","titles":["Development environment"]},"587":{"title":"Setting up a Celestia full consensus node","titles":[]},"588":{"title":"Hardware requirements","titles":["Setting up a Celestia full consensus node"]},"589":{"title":"Setting up a full consensus node","titles":["Setting up a Celestia full consensus node"]},"590":{"title":"Set up the dependencies","titles":["Setting up a Celestia full consensus node","Setting up a full consensus node"]},"591":{"title":"Install celestia-app","titles":["Setting up a Celestia full consensus node","Setting up a full consensus node"]},"592":{"title":"Set up the P2P networks","titles":["Setting up a Celestia full consensus node","Setting up a full consensus node"]},"593":{"title":"Storage and pruning configurations","titles":["Setting up a Celestia full consensus node"]},"594":{"title":"Connecting a consensus node to a bridge node","titles":["Setting up a Celestia full consensus node","Storage and pruning configurations"]},"595":{"title":"Enable transaction indexing","titles":["Setting up a Celestia full consensus node","Storage and pruning configurations","Connecting a consensus node to a bridge node"]},"596":{"title":"Retain all block data","titles":["Setting up a Celestia full consensus node","Storage and pruning configurations","Connecting a consensus node to a bridge node"]},"597":{"title":"Querying transactions by hash","titles":["Setting up a Celestia full consensus node","Storage and pruning configurations"]},"598":{"title":"Accessing historical state","titles":["Setting up a Celestia full consensus node","Storage and pruning configurations"]},"599":{"title":"Saving on storage requirements","titles":["Setting up a Celestia full consensus node","Storage and pruning configurations"]},"600":{"title":"Syncing","titles":["Setting up a Celestia full consensus node"]},"601":{"title":"State sync","titles":["Setting up a Celestia full consensus node","Syncing"]},"602":{"title":"Quick sync","titles":["Setting up a Celestia full consensus node","Syncing"]},"603":{"title":"Start the consensus node","titles":["Setting up a Celestia full consensus node"]},"604":{"title":"Extra resources for consensus nodes","titles":["Setting up a Celestia full consensus node"]},"605":{"title":"Optional: Reset network","titles":["Setting up a Celestia full consensus node","Extra resources for consensus nodes"]},"606":{"title":"Optional: Configuring an RPC endpoint","titles":["Setting up a Celestia full consensus node","Extra resources for consensus nodes"]},"607":{"title":"Expose RPC","titles":["Setting up a Celestia full consensus node","Extra resources for consensus nodes","Optional: Configuring an RPC endpoint"]},"608":{"title":"Note on external-address","titles":["Setting up a Celestia full consensus node","Extra resources for consensus nodes","Optional: Configuring an RPC endpoint"]},"609":{"title":"Restart the node","titles":["Setting up a Celestia full consensus node","Extra resources for consensus nodes","Optional: Configuring an RPC endpoint"]},"610":{"title":"Optional: Transaction indexer configuration options","titles":["Setting up a Celestia full consensus node","Extra resources for consensus nodes"]},"611":{"title":"Optional: Discard ABCI responses configuration","titles":["Setting up a Celestia full consensus node","Extra resources for consensus nodes"]},"612":{"title":"Setting up a Celestia full storage Node","titles":[]},"613":{"title":"Overview of full storage nodes","titles":["Setting up a Celestia full storage Node"]},"614":{"title":"Hardware requirements","titles":["Setting up a Celestia full storage Node"]},"615":{"title":"Setting up your full storage node","titles":["Setting up a Celestia full storage Node"]},"616":{"title":"Setup the dependencies","titles":["Setting up a Celestia full storage Node","Setting up your full storage node"]},"617":{"title":"Install celestia-node","titles":["Setting up a Celestia full storage Node"]},"618":{"title":"Run the full storage node","titles":["Setting up a Celestia full storage Node","Install celestia-node"]},"619":{"title":"Initialize the full storage node","titles":["Setting up a Celestia full storage Node","Install celestia-node","Run the full storage node"]},"620":{"title":"Start the full storage node","titles":["Setting up a Celestia full storage Node","Install celestia-node","Run the full storage node"]},"621":{"title":"Optional: run the full storage node with a custom key","titles":["Setting up a Celestia full storage Node","Install celestia-node"]},"622":{"title":"Optional: start the full storage node with SystemD","titles":["Setting up a Celestia full storage Node","Install celestia-node"]},"623":{"title":"Stop the full storage node","titles":["Setting up a Celestia full storage Node","Install celestia-node"]},"624":{"title":"Celestia hardfork process","titles":[]},"625":{"title":"General process","titles":["Celestia hardfork process"]},"626":{"title":"Mocha hardfork","titles":["Celestia hardfork process","General process"]},"627":{"title":"IBC relaying guide","titles":[]},"628":{"title":"Hermes","titles":["IBC relaying guide"]},"629":{"title":"Configuration","titles":["IBC relaying guide","Hermes"]},"630":{"title":"Add relayer wallets","titles":["IBC relaying guide","Hermes"]},"631":{"title":"Verify configuration files","titles":["IBC relaying guide","Hermes"]},"632":{"title":"Create a connection between 2 chains","titles":["IBC relaying guide"]},"633":{"title":"Create clients","titles":["IBC relaying guide"]},"634":{"title":"Open connection over new clients","titles":["IBC relaying guide","Create clients"]},"635":{"title":"Configure channels in Hermes","titles":["IBC relaying guide","Create clients"]},"636":{"title":"Start the relayer","titles":["IBC relaying guide"]},"637":{"title":"Transfer","titles":["IBC relaying guide"]},"638":{"title":"Token filter","titles":["IBC relaying guide","Transfer"]},"639":{"title":"Celestia App network instantiation guide","titles":[]},"640":{"title":"Hardware requirements","titles":["Celestia App network instantiation guide"]},"641":{"title":"Setup dependencies","titles":["Celestia App network instantiation guide"]},"642":{"title":"celestia-app installation","titles":["Celestia App network instantiation guide"]},"643":{"title":"Spin up a Celestia testnet","titles":["Celestia App network instantiation guide"]},"644":{"title":"Optional: Reset working directory","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"645":{"title":"Initialize a working directory","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"646":{"title":"Create a new key","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"647":{"title":"Add genesis account KeyName","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"648":{"title":"Optional: Adding other validators","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"649":{"title":"Create the genesis transaction for new chain","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"650":{"title":"Creating the genesis JSON file","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"651":{"title":"Modify your config file","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"652":{"title":"Add your node as a persistent peer","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"653":{"title":"Instantiate the network","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"654":{"title":"Setting up a Celestia light node","titles":[]},"655":{"title":"Overview of light nodes","titles":["Setting up a Celestia light node"]},"656":{"title":"Hardware requirements","titles":["Setting up a Celestia light node"]},"657":{"title":"Setting up your light node","titles":["Setting up a Celestia light node"]},"658":{"title":"Install celestia-node","titles":["Setting up a Celestia light node","Setting up your light node"]},"659":{"title":"Initialize the light node","titles":["Setting up a Celestia light node"]},"660":{"title":"Start the light node","titles":["Setting up a Celestia light node","Initialize the light node"]},"661":{"title":"Keys and wallets","titles":["Setting up a Celestia light node","Initialize the light node"]},"662":{"title":"Testnet tokens","titles":["Setting up a Celestia light node","Initialize the light node","Keys and wallets"]},"663":{"title":"Optional: run the light node with a custom key","titles":["Setting up a Celestia light node","Initialize the light node"]},"664":{"title":"Optional: start light node with SystemD","titles":["Setting up a Celestia light node","Initialize the light node"]},"665":{"title":"Data availability sampling","titles":["Setting up a Celestia light node"]},"666":{"title":"Mainnet Beta","titles":[]},"667":{"title":"Network stability and upgrades","titles":["Mainnet Beta"]},"668":{"title":"Network details","titles":["Mainnet Beta"]},"669":{"title":"Software version numbers","titles":["Mainnet Beta"]},"670":{"title":"Network parameters","titles":["Mainnet Beta"]},"671":{"title":"Maximum bytes","titles":["Mainnet Beta","Network parameters"]},"672":{"title":"Integrations","titles":["Mainnet Beta"]},"673":{"title":"Consensus nodes","titles":["Mainnet Beta","Integrations"]},"674":{"title":"Consensus RPC endpoints","titles":["Mainnet Beta","Integrations","Consensus nodes"]},"675":{"title":"API endpoints","titles":["Mainnet Beta","Integrations","Consensus nodes"]},"676":{"title":"gRPC endpoints","titles":["Mainnet Beta","Integrations","Consensus nodes"]},"677":{"title":"WebSocket endpoints","titles":["Mainnet Beta","Integrations","Consensus nodes"]},"678":{"title":"Data availability nodes","titles":["Mainnet Beta","Integrations"]},"679":{"title":"Data availability (DA) RPC endpoints","titles":["Mainnet Beta","Integrations","Data availability nodes"]},"680":{"title":"Archival DA RPC endpoints","titles":["Mainnet Beta","Integrations","Data availability nodes"]},"681":{"title":"Grove archival endpoints","titles":["Mainnet Beta","Integrations","Data availability nodes","Archival DA RPC endpoints"]},"682":{"title":"Explorers","titles":["Mainnet Beta"]},"683":{"title":"Analytics","titles":["Mainnet Beta"]},"684":{"title":"Network upgrades","titles":["Mainnet Beta"]},"685":{"title":"Mocha testnet","titles":[]},"686":{"title":"Network details","titles":["Mocha testnet"]},"687":{"title":"Software version numbers","titles":["Mocha testnet"]},"688":{"title":"RPC for DA bridge, full, and light nodes","titles":["Mocha testnet"]},"689":{"title":"RPC endpoints","titles":["Mocha testnet"]},"690":{"title":"API endpoints","titles":["Mocha testnet"]},"691":{"title":"gRPC endpoints","titles":["Mocha testnet"]},"692":{"title":"Bridge and full node endpoints","titles":["Mocha testnet"]},"693":{"title":"Mocha testnet faucet","titles":["Mocha testnet"]},"694":{"title":"Explorers","titles":["Mocha testnet"]},"695":{"title":"Network upgrades","titles":["Mocha testnet"]},"696":{"title":"Overview to running nodes on Celestia","titles":[]},"697":{"title":"Participate in the Celestia networks","titles":[]},"698":{"title":"Mainnet Beta","titles":["Participate in the Celestia networks"]},"699":{"title":"Compatible software versions for Mainnet Beta","titles":["Participate in the Celestia networks","Mainnet Beta"]},"700":{"title":"Testnets","titles":["Participate in the Celestia networks"]},"701":{"title":"Arabica Devnet","titles":["Participate in the Celestia networks","Testnets"]},"702":{"title":"Compatible software versions for Arabica devnet","titles":["Participate in the Celestia networks","Testnets","Arabica Devnet"]},"703":{"title":"Mocha testnet","titles":["Participate in the Celestia networks","Testnets"]},"704":{"title":"Compatible software versions for Mocha testnet","titles":["Participate in the Celestia networks","Testnets"]},"705":{"title":"Network upgrades","titles":["Participate in the Celestia networks"]},"706":{"title":"Quick start guide","titles":[]},"707":{"title":"Celestia Node","titles":["Quick start guide"]},"708":{"title":"Celestia App","titles":["Quick start guide"]},"709":{"title":"Getting started","titles":["Quick start guide"]},"710":{"title":"Setting up your node as a background process with SystemD","titles":[]},"711":{"title":"Consensus nodes","titles":["Setting up your node as a background process with SystemD"]},"712":{"title":"Start the celestia-app with SystemD","titles":["Setting up your node as a background process with SystemD","Consensus nodes"]},"713":{"title":"Data availability nodes","titles":["Setting up your node as a background process with SystemD"]},"714":{"title":"Celestia full storage node","titles":["Setting up your node as a background process with SystemD","Data availability nodes"]},"715":{"title":"Celestia bridge node","titles":["Setting up your node as a background process with SystemD","Data availability nodes"]},"716":{"title":"Celestia light node","titles":["Setting up your node as a background process with SystemD","Data availability nodes"]},"717":{"title":"Setting up a Celestia validator node","titles":[]},"718":{"title":"Hardware requirements","titles":["Setting up a Celestia validator node"]},"719":{"title":"Setting up a validator node","titles":["Setting up a Celestia validator node"]},"720":{"title":"Wallet","titles":["Setting up a Celestia validator node","Setting up a validator node"]},"721":{"title":"Delegate stake to a validator","titles":["Setting up a Celestia validator node","Setting up a validator node"]},"722":{"title":"Optional: Deploy the celestia-node","titles":["Setting up a Celestia validator node"]},"723":{"title":"Install celestia-node","titles":["Setting up a Celestia validator node","Optional: Deploy the celestia-node"]},"724":{"title":"Initialize the bridge node","titles":["Setting up a Celestia validator node","Optional: Deploy the celestia-node"]},"725":{"title":"Run the bridge node","titles":["Setting up a Celestia validator node","Optional: Deploy the celestia-node"]},"726":{"title":"Optional: start the bridge node with SystemD","titles":["Setting up a Celestia validator node","Optional: Deploy the celestia-node","Run the bridge node"]},"727":{"title":"Run the validator node","titles":["Setting up a Celestia validator node"]},"728":{"title":"Submit your validator information","titles":["Setting up a Celestia validator node"]},"729":{"title":"Optional: Transaction indexer configuration options","titles":["Setting up a Celestia validator node"]},"730":{"title":"Additional resources","titles":["Setting up a Celestia validator node"]}},"dirtCount":0,"index":[["^external",{"2":{"608":2}}],["^persistent",{"2":{"592":6}}],["^seeds",{"2":{"592":4}}],["^0",{"2":{"116":2,"117":2,"127":2,"161":2,"248":2,"249":2,"260":4,"262":2}}],["∑i=1nssn",{"2":{"386":1}}],["×ss×gcpbb",{"2":{"386":1}}],["🌀",{"0":{"271":1}}],["🌉🌉🌉🌉🌉",{"2":{"91":2}}],["|",{"2":{"270":2,"317":4,"319":2,"327":2,"329":2,"332":2,"348":4,"353":1,"492":4,"592":10,"602":12,"712":4}}],["||",{"2":{"235":2}}],["\\trowproofs",{"2":{"158":2,"161":2}}],["\\trowroots",{"2":{"157":2,"161":2}}],["\\treturn",{"2":{"151":2,"155":8,"156":2,"157":2,"158":2,"159":2,"161":18}}],["\\tmaxns",{"2":{"155":2,"161":2}}],["\\tminns",{"2":{"155":2,"161":2}}],["\\tversion",{"2":{"155":4,"161":6}}],["\\tvalid",{"2":{"151":2}}],["\\tvar",{"2":{"151":2,"155":6,"156":2,"159":2,"161":10}}],["\\tsimplerollup",{"2":{"161":2}}],["\\tsimplerollupwrapper",{"2":{"161":2}}],["\\tsidenodes",{"2":{"151":2,"159":2,"161":2}}],["\\tshareproofs",{"2":{"155":2,"161":2}}],["\\tsharesproof",{"2":{"152":1,"161":4}}],["\\twrappedproof",{"2":{"151":2}}],["\\twrapper",{"2":{"151":2,"159":2}}],["\\tproof",{"2":{"151":2}}],["\\theight",{"2":{"151":2,"159":2,"161":4}}],["\\tnonce",{"2":{"151":2,"159":2,"161":4}}],["\\tcopy",{"2":{"155":6,"156":2,"161":2}}],["\\tcommitted",{"2":{"151":2}}],["\\tctx",{"2":{"150":2,"151":2,"161":4}}],["\\tfor",{"2":{"151":4,"155":2,"157":2,"158":2,"159":4,"161":16}}],["\\tfmt",{"2":{"150":2}}],["\\tlatestblocknumber",{"2":{"151":2,"159":2}}],["\\tdatarootinclusionproof",{"2":{"159":2,"161":4}}],["\\tdataroot",{"2":{"151":2,"161":2}}],["\\tdefer",{"2":{"151":2,"159":2,"161":2}}],["\\tdcproof",{"2":{"150":2,"151":2}}],["\\tblockdataroot",{"2":{"159":2,"161":2}}],["\\tblockres",{"2":{"151":2}}],["\\tblobstreamxwrapper",{"2":{"151":4,"161":2}}],["\\ttmproto",{"2":{"161":2}}],["\\ttuple",{"2":{"151":2}}],["\\ttx",{"2":{"151":2,"161":2}}],["\\ttrpc",{"2":{"150":2,"151":2}}],["\\teventsiterator",{"2":{"151":2,"159":2}}],["\\tethclient",{"2":{"151":2,"159":2,"161":2}}],["\\tethcmn",{"2":{"151":2,"161":2}}],["\\terr",{"2":{"150":2,"151":6,"159":2,"161":4}}],["\\t\\tblockdataroot",{"2":{"161":2}}],["\\t\\tblockres",{"2":{"161":2}}],["\\t\\tbig",{"2":{"151":2}}],["\\t\\tuint64",{"2":{"161":2}}],["\\t\\tclient",{"2":{"161":2}}],["\\t\\tctx",{"2":{"161":2}}],["\\t\\tcase",{"2":{"160":4}}],["\\t\\tcopy",{"2":{"159":2}}],["\\t\\tproof",{"2":{"159":2,"161":2}}],["\\t\\tvar",{"2":{"159":2,"161":2}}],["\\t\\tversion",{"2":{"155":4,"156":2,"161":6}}],["\\t\\trowproofs",{"2":{"158":2,"161":2}}],["\\t\\trowroots",{"2":{"157":2,"161":2}}],["\\t\\treturn",{"2":{"151":30,"159":14,"161":8}}],["\\t\\tdcproof",{"2":{"161":2}}],["\\t\\tdigest",{"2":{"155":2,"161":4}}],["\\t\\tdataroot",{"2":{"151":2}}],["\\t\\tmax",{"2":{"155":2,"161":2}}],["\\t\\tmin",{"2":{"155":2,"161":2}}],["\\t\\tid",{"2":{"155":4,"156":2,"161":12}}],["\\t\\tif",{"2":{"151":2,"159":2}}],["\\t\\tsharesproof",{"2":{"161":2}}],["\\t\\tshareproofs",{"2":{"155":2,"161":2}}],["\\t\\tsimplerollupwrapper",{"2":{"161":2}}],["\\t\\tsidenodes",{"2":{"151":4,"155":2,"158":2,"159":2,"161":6}}],["\\t\\tfor",{"2":{"155":2,"158":2,"161":6}}],["\\t\\tfmt",{"2":{"150":6,"151":6,"161":2}}],["\\t\\twrappedproof",{"2":{"151":2}}],["\\t\\ttuplerootnonce",{"2":{"159":2,"161":2}}],["\\t\\ttuple",{"2":{"151":2,"159":2,"161":2}}],["\\t\\tnumleaves",{"2":{"151":2}}],["\\t\\tnil",{"2":{"151":6,"159":6}}],["\\t\\tkey",{"2":{"151":2}}],["\\t\\theight",{"2":{"151":2}}],["\\t\\tevent",{"2":{"161":2}}],["\\t\\te",{"2":{"151":2,"159":2}}],["\\t\\t\\tfor",{"2":{"161":2}}],["\\t\\t\\tattestationproof",{"2":{"161":2}}],["\\t\\t\\trowproofs",{"2":{"161":2}}],["\\t\\t\\trowroots",{"2":{"161":2}}],["\\t\\t\\treturn",{"2":{"160":2}}],["\\t\\t\\tnamespace",{"2":{"161":2}}],["\\t\\t\\tnumleaves",{"2":{"158":2,"159":2,"161":4}}],["\\t\\t\\tdata",{"2":{"161":2}}],["\\t\\t\\tdataroot",{"2":{"159":2,"161":2}}],["\\t\\t\\theight",{"2":{"159":2,"161":2}}],["\\t\\t\\tkey",{"2":{"158":2,"159":2,"161":4}}],["\\t\\t\\tcopy",{"2":{"158":2}}],["\\t\\t\\tcontext",{"2":{"151":2,"159":2,"160":2,"161":2}}],["\\t\\t\\tvar",{"2":{"158":2,"161":2}}],["\\t\\t\\tbzsidenode",{"2":{"161":2}}],["\\t\\t\\tbeginkey",{"2":{"155":2,"161":2}}],["\\t\\t\\tbreak",{"2":{"151":2,"159":2}}],["\\t\\t\\tshareproofs",{"2":{"161":2}}],["\\t\\t\\tsidenodes",{"2":{"155":4,"158":4,"159":2,"161":10}}],["\\t\\t\\tstart",{"2":{"151":2,"159":2}}],["\\t\\t\\t",{"2":{"151":2,"159":2,"160":2,"161":2}}],["\\t\\t\\t\\tbzsidenode",{"2":{"161":2}}],["\\t\\t\\t\\tdatacommitment",{"2":{"151":2,"159":2}}],["\\t\\t\\t\\tendblock",{"2":{"151":2,"159":2}}],["\\t\\t\\t\\tstartblock",{"2":{"151":2,"159":2}}],["\\t\\t\\t\\tproofnonce",{"2":{"151":2,"159":2}}],["\\t\\t\\tevent",{"2":{"151":2,"159":2}}],["\\t\\t\\tendkey",{"2":{"155":2,"161":2}}],["\\t\\t\\tend",{"2":{"151":2,"159":2}}],["\\t\\t",{"2":{"151":8,"152":2,"155":4,"158":4,"159":10,"160":4,"161":20}}],["\\t\\tos",{"2":{"150":6,"151":2,"161":2}}],["\\tif",{"2":{"150":6,"151":32,"152":2,"159":14,"161":10}}],["\\t",{"2":{"150":14,"151":90,"152":2,"155":8,"156":2,"157":2,"158":2,"159":31,"160":24,"161":94}}],["🤪",{"2":{"135":1}}],["`$home",{"2":{"592":2}}],["`job=",{"2":{"503":2}}],["`json",{"2":{"134":22}}],["```json",{"2":{"326":2}}],["`https",{"2":{"268":4}}],["`celestia",{"2":{"392":4}}],["`cel",{"2":{"236":2}}],["`availabledataroot`",{"2":{"159":2}}],["`",{"2":{"134":22,"236":2,"268":4,"398":12,"503":2}}],["z20=",{"2":{"319":2}}],["zshrcecho",{"2":{"305":1,"586":1}}],["zshrc",{"2":{"305":3,"586":3}}],["zero",{"2":{"127":2,"133":1,"201":1}}],["zkp",{"2":{"127":2}}],["zk",{"0":{"184":1,"185":1,"186":1,"187":1,"188":1,"189":1,"190":1,"191":1},"1":{"185":1,"186":2,"187":2,"188":1,"189":2,"190":2,"191":2,"192":1,"193":1,"194":1},"2":{"127":7,"136":2,"148":1,"163":1,"167":1,"184":1,"185":2,"186":1,"188":2,"189":4,"191":1,"192":5,"194":1,"195":1,"201":1,"206":4}}],["zip",{"2":{"86":2}}],["~2",{"2":{"671":1}}],["~30",{"2":{"378":1}}],["~",{"2":{"117":2,"244":2,"249":2,"486":1,"556":4,"602":18,"607":2,"621":1,"663":1}}],["μ",{"2":{"117":2,"249":2}}],["⠆",{"2":{"117":2,"120":2,"249":2,"252":2}}],["⠑",{"2":{"116":2,"248":2}}],["⠔",{"2":{"116":2,"248":2}}],["⠢",{"2":{"116":2,"248":2}}],["=1",{"2":{"671":1}}],["===",{"2":{"268":6}}],["==",{"2":{"151":2,"155":6,"158":3,"159":2,"260":2,"385":1,"579":2,"580":2}}],["=",{"2":{"116":2,"117":2,"127":2,"137":28,"139":14,"149":8,"150":14,"151":74,"152":4,"155":20,"157":6,"158":12,"159":40,"160":18,"161":96,"173":4,"207":1,"248":2,"249":2,"260":62,"267":10,"268":80,"270":2,"337":4,"373":52,"374":20,"377":1,"384":1,"386":4,"387":2,"392":44,"398":8,"463":1,"502":8,"531":10,"566":1,"568":2,"592":22,"595":2,"596":2,"597":2,"598":3,"599":6,"601":6,"608":4,"610":2,"611":2,"629":170,"635":8,"651":2}}],["⏰⏰⏰⏰⏰⏰",{"2":{"91":2}}],["😎",{"2":{"578":2}}],["🐳",{"0":{"575":1},"1":{"576":1,"577":1,"578":1,"579":1,"580":1,"581":1,"582":1,"583":1}}],["🔐",{"0":{"314":1}}],["🙂",{"2":{"133":1}}],["🔃🔃🔃",{"2":{"91":2}}],["🚀🚀🚀🚀",{"2":{"91":2}}],["📝📝📝📝📝",{"2":{"91":2}}],["💰💰💰💰💰💰",{"2":{"91":2}}],["😁",{"2":{"89":1}}],["xf",{"2":{"602":6}}],["x86",{"2":{"540":1,"560":1}}],["xv92a3qxexport",{"2":{"515":1}}],["xv92a3qx",{"2":{"515":3}}],["xxxxxxxxxxxxx",{"2":{"515":4}}],["xxd",{"2":{"348":2}}],["x64",{"2":{"480":1,"589":1,"615":1,"657":1,"719":1}}],["xzf",{"2":{"305":8,"586":8}}],["xyz",{"2":{"214":1,"216":1,"217":1,"219":1,"220":1,"252":2,"272":2,"377":1,"629":6,"674":3,"675":3,"676":3,"679":2,"688":2,"689":2,"691":2}}],["x",{"0":{"95":1,"109":1,"126":1,"151":1,"196":1,"206":1,"207":1,"208":1,"209":1},"1":{"96":1,"97":1,"197":1,"198":1,"199":1,"208":1,"209":1,"210":1,"211":1},"2":{"89":4,"94":1,"95":1,"106":1,"117":9,"126":2,"130":1,"131":7,"133":1,"148":6,"150":2,"159":2,"161":4,"162":1,"196":1,"198":1,"201":1,"206":5,"207":9,"208":3,"209":1,"210":8,"249":9,"334":2,"376":3,"377":4,"378":1,"379":2,"387":2,"392":2,"494":1,"509":1,"602":6}}],["✨",{"2":{"83":1,"91":2,"268":2}}],["07",{"2":{"634":8}}],["0ns",{"2":{"634":4}}],["0+",{"2":{"628":1}}],["010",{"2":{"548":1}}],["01",{"2":{"398":2,"500":2,"629":6,"674":1,"675":1,"676":1,"689":1,"690":1,"691":1,"727":2}}],["011",{"2":{"167":1}}],["0blockhash",{"2":{"348":1}}],["08",{"2":{"329":1}}],["025",{"2":{"629":2}}],["02",{"2":{"326":2,"398":2}}],["06",{"2":{"326":4}}],["001hermes",{"2":{"633":2,"634":1}}],["001",{"2":{"629":3,"630":2,"632":1,"633":4,"634":7,"635":3}}],["003442",{"2":{"557":4}}],["000",{"2":{"385":2,"386":1,"449":3,"455":3,"463":2,"509":1}}],["00000000ce1e571a",{"2":{"370":2}}],["008",{"2":{"329":1}}],["004",{"2":{"320":4}}],["002",{"2":{"320":1,"334":2}}],["0mfhykqui2bu+u1jxpzg7qy2bvv1lb3kiu+zak7nuiy=",{"2":{"317":2}}],["048091bc7ddc283f77bfbf91d73c44da58c3df8a9cbc867405d8b7f3daada22f",{"2":{"326":2}}],["04",{"2":{"281":1,"480":1,"589":1,"615":1,"657":1,"719":1}}],["0px",{"2":{"268":2}}],["0x",{"2":{"317":2,"626":1}}],["0xb6b15c8cb491557369f3c7d2c287b053eb229daa9c22138887752191c9520659",{"2":{"119":1}}],["0xb6052122545aacd2bdda0ca9fa56416bd968cdbc",{"2":{"99":1}}],["0x02",{"2":{"374":2}}],["0x00",{"2":{"374":2}}],["0x0000000000000000",{"2":{"347":2}}],["0x0000000000000000000000000000000000000000",{"2":{"347":2}}],["0x0000000000000000000000000000000000000000000000000000000000000000",{"2":{"347":2}}],["0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000",{"2":{"121":2,"253":2,"347":2}}],["0x000000000000000000000000000000000000000000000000000000000000000a",{"2":{"121":2,"253":2}}],["0x0",{"2":{"353":1}}],["0x0c",{"2":{"107":1}}],["0x04cae899fc0b7ef45c529f8bf075d54f6fb70ed9",{"2":{"99":1}}],["0x03b43f7b61fa100611191f481ef48aa1fc98f434",{"2":{"99":1}}],["0x01",{"2":{"374":2}}],["0x01b5905b154f21a393f5b5a0c6d15b53a493c05e",{"2":{"98":1}}],["0x019850732270d8c436585c7921219252422228b5d0f559da0da219f0fa2b7216",{"2":{"91":2}}],["0x8f15d6004598f0662dd673a9898dceef77be8cc28408cecc284b28d7be32307d",{"2":{"121":2,"253":2}}],["0x8f97cb7c643acd7f79f3b13841b24a243da51242",{"2":{"99":1}}],["0x89b7c7970c13bb587893a70697ad6d2a335b6a15",{"2":{"99":1}}],["0x8dee6e88d3b62b258c1574cbb7005e1c3cf193b60a99b5c2fcfae00819b7ed82",{"2":{"91":2}}],["0xd4b998a35d20d98ed3488221f0c161a0a9572d3de66399482553c8e3d2fae751",{"2":{"347":2}}],["0xd883010d04846765746888676f312e32312e33856c696e7578000000000000006b3afa42dce1f87f1f07a1ef569c4d43e41738ef93c865098bfa1458645f384e2e4498bcfe4ad9353ff1913a2e16162f496fafe5b0939a6c78fb5b503248d6da01",{"2":{"347":2}}],["0xd2d353916b34a877793628049c99858f04123ee1",{"2":{"98":1}}],["0xd0a6699fc7519966685181c80bf98d35afa1fc95",{"2":{"98":1}}],["0xe2662ff9b41f39e63a850e50e013ea66e60a4f37",{"2":{"98":1}}],["0xe24a60b758b51b0a3da5e8f4f6ddf1cd0aff646c",{"2":{"98":1}}],["0xe371afcb8437bf61bd831ef57be7a2496d88488b",{"2":{"98":1}}],["0x383ed2debf9f9055920cd7340418dda7e2bca6b989eb6992d83d123d4e322f2a",{"2":{"348":2}}],["0x3c44cdddb6a900fa2b585dd299e03d12fa4293bc",{"2":{"348":2}}],["0x3f1eae7d46d88f08fc2f8ed27fcb2ab183eb2d0e",{"2":{"119":1}}],["0x3b6e845fb9f0c8ee4e9f6d44781f6547d9c6359a",{"2":{"99":1}}],["0x3bc040eaca40b91fa06cf55ea91842fac88b1af4",{"2":{"98":1}}],["0x3681cbb0e95ab50b63f2fc524fbbcc78adefbd33",{"2":{"98":1}}],["0xaf5c1505c7dfcebca94d9a6a8c0caf99b6c87a8ed6d6c0b3161c9026f270a84f",{"2":{"348":2}}],["0xaf8ff6af1180c8be9e4e8f3a5f882b3b227233f4abbefa479836d3721682a389",{"2":{"347":2}}],["0xadd3a5dc0b8c605aeac891098e87cbaff43bb642896ebbf74f964c0690e46df2",{"2":{"348":2}}],["0xa83ca7775bc2889825bcdedffa5b758cf69e8794",{"2":{"210":2}}],["0xac3427e621c6f10dc2abdab00188d92690503914",{"2":{"98":1}}],["0xa773e19dc9e822933a7e72df9c87ed1578701d29",{"2":{"98":1}}],["0xaa3b8b63ccca3c98b948fd1d6ed875d378de2c6c",{"2":{"97":1}}],["0xaae3a04931345df5ac6e784bb6bdeb29b1ff0286",{"2":{"96":1}}],["0x20d8153aacc4e6d29558fa3916bff422bede9b5e",{"2":{"99":1}}],["0x2588867f19e2de51f90f0ab852c7ad11228e3d83",{"2":{"98":1}}],["0x22a6580faeca49cf86cbb2f18f2b7f98031fc6ad",{"2":{"96":1}}],["0x5fbdb2315678afecb367f032d93f642f64180aa3",{"2":{"120":2,"252":2}}],["0x55527d53fda37dbf1924482b40acf8625e1caa5b",{"2":{"99":1}}],["0x5a48adf22f526ebd06e3e8856cfea2490923cc55",{"2":{"99":1}}],["0x51d196e07a27dba0f4461dd6cc26108424f196f7",{"2":{"99":1}}],["0x5187a92539bb4a2befe1fc078745c84ab6d37171",{"2":{"98":1}}],["0x53dea3a90fd6c82840a1f7224f799d622f142df4",{"2":{"98":1}}],["0x5810f0916bae1067ca1efcc00aaaf30301af001c",{"2":{"98":1}}],["0x59d2db6c5095b9e329c80211b7a761d20064379e3382d156b69e5cf3b5fe2fc7",{"2":{"91":2}}],["0x664bf4bb4a57dd5768a0a98991d77c58fb7a4e164c2581c79fb33ce9c3d4c250",{"2":{"347":2}}],["0x676d",{"2":{"317":3}}],["0x61254e43e5c1e9e801f9c56b47a9ac3eadf6d1e9",{"2":{"99":1}}],["0x61e154128b6a1400ea8090b4431b4aa1dbb80cc4",{"2":{"98":1}}],["0x630093954cbf19fe4532a2edd0bd3b10deca7a4d",{"2":{"98":1}}],["0x6cb49605f10831749c6090ad09918bc61439bace",{"2":{"98":1}}],["0x6c7360a96165c570dcb7ce609d748d612c5fa5b76e229cd81ba5f5c93c00f805",{"2":{"91":2}}],["0x9f4dfae061b5ddd86f95a81be5daa0d7fe32e7f7f770f86dc375e0007d249bd2",{"2":{"348":2}}],["0x99e9d2f04352b42c18f1da5dd93a970f82c08afe",{"2":{"99":1}}],["0x9abc41fefae7e7543a01fa837aec909f96147280",{"2":{"99":1}}],["0x95cbda89325db5529eaf1813e181f66b83a7d65a",{"2":{"98":1}}],["0x95fea00e689e8d1cba909836e1ef1b941d5f21b1",{"2":{"98":1}}],["0x97b50f60b60d0e658fdbf185969db0a0327bd0ae9e57cd65af2a7f9be0eeb5b0",{"2":{"91":2}}],["0x42690c204d39600fddd3",{"2":{"312":4,"317":5,"319":8,"320":2,"327":2,"332":2}}],["0x4270889adcb82338c5ff5e64b45c0a3d31cfd08c",{"2":{"97":1}}],["0x40f8c63e0a20b399bcd9631a22e57bb988a9400e",{"2":{"99":1}}],["0x44b412b291fef00398501b2ca353ea912ad0fe13",{"2":{"99":1}}],["0x4888fdf44251d456bbfca92bfc6e180cfe0b096ffbea2f6da2a203a16902214f",{"2":{"91":2}}],["0x1dcc4de8dec75d7aab85b567b6ccd41ad312451b948a7413f0a142fd40d49347",{"2":{"347":2}}],["0x1cb54d2369752ef73511c202ff9cdfd0eadf3a77b7aef0092bea63f2b5d57659",{"2":{"347":2}}],["0x1cc4551922c069a9ade06756bf14bf0410ea44ff",{"2":{"99":1}}],["0x1",{"2":{"260":2}}],["0x131822bef6eb59656d7e1387c19b75be667e587006710365ec5cf58030786c42",{"2":{"121":2,"253":2}}],["0x1ae3a8dc1e7efd37f418b2987d3df74c5a917a8b",{"2":{"99":1}}],["0x1bb8add5e878b12fa37756392642eb94c53a1cf4",{"2":{"97":1,"99":1}}],["0x10c65b27d5031ce2351c719072e58f3153228887f027f9f6d65300d2b5b30152",{"2":{"94":1}}],["0x79a0a7a1b4936aafe7a37dbfb07a6a9e55c145a4ed6fd54f962649b4b7db8de7",{"2":{"347":2}}],["0x7973d0b475e898082df25c1617cbce1917cfed17",{"2":{"98":1}}],["0x79751b011bcc20f413a2c4e3af019b6e2a9738b9",{"2":{"96":1,"98":1}}],["0x78f8b2941dde5a8a312814ebd29c2e2a36f25e91",{"2":{"99":1}}],["0x7fc4d9a24949680fad666feee7cd6a100e39c4f0",{"2":{"98":1}}],["0x7fbeb5bc73a11b438891022786feb2c624f275f0",{"2":{"91":2}}],["0xff00000000000000000000000000000000000901",{"2":{"348":3}}],["0xf0c6429ebab2e7dc6e05dafb61128be21f13cb1e",{"2":{"210":1}}],["0xf1a793a793cd9fc588f5132d99008565ea361eb3535d66499575e9e1908200b2",{"2":{"120":2,"252":2}}],["0xf39fd6e51aad88f6f4ce6ab8827279cfffb92266",{"2":{"120":2,"252":2}}],["0xf31fd34f8a9d9057198d8b13e755e583766bd528459733d948d9ffbc980c9506",{"2":{"91":2}}],["0xf889a3174fddd9f78e6cd250ebf4c16f1bdd1b6a",{"2":{"99":1}}],["0xfb612fb83959b8acd3e49540b29c93c5a67e05f1",{"2":{"98":1}}],["0xf47dc66514fd78e4666e35abd12df7d1ae2c79f69f7dfedb8d98e4106142ab7c",{"2":{"91":2}}],["0xf2ddc2dad90e7e2b20a772bf89f989224165659d50824b98d7340e12265abf01",{"2":{"91":2}}],["0xcryptovestor",{"2":{"674":1}}],["0xce",{"2":{"348":2,"351":3}}],["0xce3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f",{"2":{"348":2}}],["0xc7535f078cb3880a0fd5e54fa7a3b4eaf09b3924",{"2":{"99":1}}],["0xc99eea0b8e67d5b2226ab6d37882daaf6dd7593b",{"2":{"99":1}}],["0xcd9fca5015b5ce2b06a2266e4a5dd54d9ca39f1a",{"2":{"99":1}}],["0xc61382d5609ab0ece36b2776349c8bdceeafdd13dde9624cdf3d746fb4cf7d79",{"2":{"91":2}}],["0xc3e209eb245fd59c8586777b499d6a665df3abd2",{"2":{"89":2,"95":1,"210":2}}],["0",{"2":{"80":2,"82":2,"89":6,"91":6,"94":1,"116":4,"117":5,"121":2,"126":1,"127":2,"137":2,"148":1,"149":6,"152":2,"155":2,"159":2,"161":4,"172":1,"173":6,"248":4,"249":5,"251":2,"253":2,"260":4,"266":6,"267":2,"268":4,"295":1,"319":8,"320":5,"322":2,"323":2,"324":10,"325":2,"326":10,"329":3,"334":14,"341":6,"348":7,"349":4,"351":1,"352":4,"360":6,"367":4,"373":2,"384":4,"385":1,"388":2,"392":8,"398":6,"400":3,"441":2,"502":2,"503":4,"522":10,"523":2,"524":2,"525":4,"540":1,"541":4,"596":3,"599":4,"601":2,"607":20,"626":1,"629":14,"634":14,"635":2,"721":8,"727":8}}],["$ip",{"2":{"652":2}}],["$input",{"2":{"348":2}}],["$external",{"2":{"608":2}}],["$encoded",{"2":{"349":2}}],["$seeds",{"2":{"592":8}}],["$staking",{"2":{"500":2,"649":3}}],["$rpc",{"2":{"529":2,"577":6,"580":6}}],["$request",{"2":{"310":1,"473":1,"662":1,"693":1}}],["$receiver",{"2":{"76":2,"272":2}}],["$from",{"2":{"522":6,"524":2,"525":2,"529":2}}],["$tia",{"2":{"647":2}}],["$to",{"2":{"522":6,"523":2,"525":2,"529":2}}],["$tx",{"2":{"348":2}}],["$broadcast",{"2":{"507":2}}],["$bubs",{"2":{"252":4,"253":6,"272":10}}],["$multisig",{"2":{"507":10}}],["$keyring",{"2":{"507":4}}],["$key",{"2":{"500":6,"646":3,"647":3,"649":3}}],["$keynamecelestia",{"2":{"367":1}}],["$keyname",{"2":{"367":3}}],["$validator",{"2":{"500":2,"507":6,"645":3,"721":2}}],["$networkdocker",{"2":{"577":3,"579":3}}],["$network",{"2":{"367":4,"577":3,"579":3}}],["$nodetype",{"2":{"367":2,"368":2}}],["$node",{"2":{"312":2,"315":3,"317":2,"319":8,"320":2,"322":2,"323":4,"324":2,"325":2,"326":4,"327":4,"328":2,"329":6,"330":2,"331":2,"332":4,"577":6,"579":2,"652":2,"715":2}}],["$celes",{"2":{"500":2}}],["$celestia",{"2":{"334":2,"519":2,"520":2,"522":6,"523":2,"524":2,"525":4}}],["$chain",{"2":{"500":4,"507":10,"645":4,"649":2}}],["$core",{"2":{"367":2}}],["$contract",{"2":{"76":6,"121":4,"253":4,"263":4,"264":4,"272":6}}],["$peer",{"2":{"652":1}}],["$peerpeer=",{"2":{"652":1}}],["$persistent",{"2":{"592":12}}],["$port",{"2":{"652":2}}],["$path",{"2":{"305":1,"586":1}}],["$private",{"2":{"75":2,"76":4,"251":2,"263":4,"264":4,"270":2,"272":6}}],["$address",{"2":{"329":6}}],["$authtype",{"2":{"368":2}}],["$auth",{"2":{"312":2,"368":2}}],["$anvil",{"2":{"251":2}}],["$arb",{"2":{"75":2,"76":8,"120":2,"121":4}}],["$",{"2":{"121":2,"244":2,"253":2,"268":4,"304":2,"327":2,"332":2,"540":2,"542":12,"560":2,"585":2,"602":6}}],["$l2",{"2":{"120":2,"121":2}}],["$home",{"2":{"75":4,"77":2,"102":2,"115":3,"116":1,"247":3,"248":1,"272":4,"273":2,"305":20,"307":2,"316":4,"317":2,"341":2,"342":2,"349":2,"357":2,"359":2,"360":3,"503":3,"514":3,"515":1,"516":2,"519":2,"539":3,"540":1,"559":2,"560":1,"565":3,"578":4,"579":8,"580":8,"586":20,"592":13,"601":1,"602":6,"605":2,"608":2,"629":2,"644":2,"648":1,"649":1,"650":4,"651":1}}],["qo",{"2":{"608":2}}],["qmaezrnbtgexcryc8pcvgrbs+umukniborae4qye7osgwcrwbvys",{"2":{"326":2}}],["qgb",{"2":{"161":2,"201":1}}],["q5",{"2":{"48":1}}],["q4",{"2":{"48":1}}],["q3",{"2":{"48":1}}],["q2",{"2":{"48":1}}],["q1",{"2":{"48":1}}],["q",{"2":{"44":1,"59":1,"60":1,"61":1,"65":1,"492":6,"493":4,"494":4,"495":2}}],["qubelabs",{"2":{"602":12}}],["quorum",{"2":{"494":1}}],["quot",{"0":{"557":2},"2":{"30":18,"31":6,"34":6,"47":2,"53":2,"76":20,"85":2,"88":2,"89":72,"91":12,"102":4,"108":4,"116":6,"117":40,"121":8,"127":12,"134":44,"139":8,"150":24,"151":76,"159":12,"160":8,"161":80,"219":2,"243":4,"244":24,"248":6,"249":40,"251":2,"253":8,"260":68,"262":12,"263":16,"264":16,"267":76,"268":232,"269":6,"270":4,"272":20,"304":4,"305":68,"317":36,"319":128,"320":16,"322":32,"323":32,"324":52,"325":20,"326":364,"327":4,"332":4,"334":76,"337":4,"348":4,"349":28,"350":2,"352":12,"353":4,"360":28,"367":20,"370":16,"372":48,"373":56,"374":4,"380":2,"387":4,"392":64,"398":96,"399":2,"400":4,"424":2,"442":2,"463":2,"470":2,"471":2,"494":156,"495":40,"500":4,"502":8,"503":36,"507":16,"515":2,"519":16,"520":32,"522":66,"523":20,"524":8,"525":16,"531":20,"536":2,"540":4,"542":16,"556":2,"560":4,"585":4,"586":68,"592":58,"595":4,"597":6,"598":6,"599":10,"601":8,"602":36,"607":8,"608":12,"610":4,"629":188,"631":4,"634":64,"647":4,"651":4,"652":4,"712":2,"721":28,"727":36}}],["quic",{"2":{"324":6}}],["quicker",{"2":{"600":1}}],["quickly",{"2":{"458":1}}],["quick",{"0":{"577":1,"602":1,"706":1},"1":{"707":1,"708":1,"709":1},"2":{"282":1,"535":1,"579":1,"602":2,"628":1,"643":1,"709":1}}],["quickstart",{"0":{"79":1,"224":1},"1":{"80":1,"81":1,"82":1,"83":1,"84":1,"85":1,"86":1,"87":1,"88":1,"89":1,"90":1,"91":1,"92":1,"93":1,"94":1,"95":1,"96":1,"97":1,"98":1,"99":1,"225":1,"226":1,"227":1},"2":{"73":1,"81":1,"101":1}}],["quite",{"2":{"270":1}}],["quests",{"2":{"375":1}}],["question",{"2":{"48":3,"402":1,"632":1,"721":1}}],["questions",{"0":{"48":1},"2":{"17":1,"47":2}}],["queue",{"2":{"351":4}}],["queried",{"2":{"145":1,"150":1,"152":1,"153":1,"159":2,"170":1,"323":1}}],["queries",{"0":{"144":1,"146":1},"1":{"145":1,"146":1,"147":2,"148":2,"149":1,"150":1,"151":1,"152":1,"153":1,"154":1,"155":1,"156":1,"157":1,"158":1,"159":1,"160":1,"161":1,"162":1},"2":{"130":1,"149":1,"180":1,"413":1,"416":1,"425":3,"610":2,"611":3,"691":1}}],["queryable",{"2":{"159":2}}],["querying",{"0":{"130":1,"280":1,"492":1,"597":1},"2":{"135":2,"150":1,"152":1,"159":1,"180":1,"209":1,"215":1,"280":1,"471":1,"488":2,"523":1,"601":1,"679":1,"688":1,"689":1}}],["query",{"0":{"523":1,"524":1,"525":1},"2":{"76":1,"129":1,"130":1,"146":1,"148":1,"150":1,"151":2,"152":1,"154":1,"155":4,"156":1,"158":4,"159":3,"161":2,"170":1,"173":2,"181":1,"272":1,"322":1,"323":1,"330":1,"392":2,"416":1,"441":1,"488":6,"492":2,"494":1,"523":2,"524":2,"525":4,"537":2,"542":1,"597":1,"598":1,"610":1,"620":1,"633":2,"660":1}}],["quad",{"2":{"588":1}}],["quadratic",{"2":{"417":1}}],["quantum",{"2":{"201":1}}],["qualify",{"2":{"34":1}}],["qualification",{"2":{"34":3}}],["qualified",{"2":{"34":1}}],["quality",{"2":{"24":1,"46":1,"47":1,"65":1}}],["quarterly",{"2":{"42":1}}],["quarter",{"2":{"29":1}}],["9390",{"2":{"676":1,"691":1}}],["973",{"2":{"671":2}}],["96112",{"2":{"522":4}}],["96ms",{"2":{"117":2,"249":2}}],["999994999800000",{"2":{"525":2}}],["99",{"2":{"405":1}}],["9anbaxno1b4x5lr2qy5qwqwrmnozejkctxwzq9bexsg=",{"2":{"326":4}}],["91e04695cf9cf531bc0891e7b1d602b3e8022c86",{"2":{"326":8}}],["9545",{"2":{"263":4}}],["9545anvil",{"2":{"263":1}}],["9099",{"2":{"691":1}}],["9095",{"2":{"676":1}}],["9090",{"2":{"309":1,"334":1,"367":1,"470":1,"471":1,"484":1,"485":1,"503":2,"541":2,"552":1,"620":1,"660":2,"676":3,"679":6,"688":1,"691":5}}],["90",{"2":{"268":2}}],["90vw",{"2":{"268":2}}],["900px",{"2":{"267":2}}],["9000",{"2":{"267":2,"269":1}}],["9000000utia",{"2":{"649":1}}],["90000",{"2":{"151":2,"159":4}}],["902",{"2":{"31":1,"34":5}}],["9",{"0":{"91":1},"2":{"33":1,"34":1,"360":2,"449":1}}],["786bytes",{"2":{"671":1}}],["786",{"2":{"671":1}}],["75",{"2":{"394":1,"395":1,"396":1,"450":1}}],["767",{"2":{"347":2}}],["7a5fabb19713d732d967b1da84fa0df5e87a7b62302d783f78743e216c1a3550",{"2":{"326":2}}],["741743z",{"2":{"326":2}}],["71",{"2":{"324":4}}],["7",{"0":{"89":1},"2":{"31":1,"34":1,"326":4,"341":3,"347":2,"449":2,"523":2,"628":1}}],["5s",{"2":{"360":2,"503":2,"629":4}}],["5iieerohbmff+ser3jpvroieejzjby+tre0ntadqll3",{"2":{"326":4}}],["56",{"2":{"303":2,"656":2}}],["57",{"2":{"161":2}}],["58",{"2":{"91":2,"155":4,"161":2}}],["5158076079",{"2":{"91":2}}],["5",{"0":{"87":1,"582":1},"2":{"31":1,"34":1,"161":2,"268":4,"360":2,"365":2,"366":1,"373":1,"375":1,"388":2,"400":1,"445":1,"455":1,"503":2,"509":1,"629":4}}],["500ms",{"2":{"629":4}}],["5000100000000utia",{"2":{"500":2}}],["500000000",{"2":{"326":4}}],["500",{"2":{"303":1,"588":1,"629":2,"656":1,"718":1}}],["5001",{"2":{"268":2}}],["50px",{"2":{"268":2}}],["50",{"2":{"21":1,"23":2,"303":1,"656":1}}],["+stake",{"2":{"434":1}}],["+qdel2aq==",{"2":{"349":2}}],["+1",{"2":{"137":2,"139":4}}],["+",{"2":{"27":2,"102":1,"137":2,"139":2,"172":1,"173":2,"260":4,"268":6,"295":1,"356":1,"398":10,"553":1,"623":1,"671":1}}],["87",{"2":{"556":2}}],["80000",{"2":{"329":4}}],["8000",{"2":{"329":4,"503":1}}],["800px",{"2":{"268":4}}],["8gb",{"2":{"285":1}}],["8548",{"2":{"119":1}}],["8545cast",{"2":{"264":1,"347":1,"348":1}}],["8545export",{"2":{"251":1}}],["8545",{"2":{"119":1,"251":2,"263":1,"264":5,"267":2,"269":1,"270":1,"347":1,"348":1}}],["8547",{"2":{"75":1,"119":1,"120":2}}],["8547export",{"2":{"75":1,"120":1}}],["8449",{"2":{"91":3}}],["8",{"0":{"90":1},"2":{"23":2,"34":1,"94":2,"116":6,"117":4,"126":1,"127":2,"161":2,"244":2,"248":6,"249":4,"260":4,"262":2,"266":4,"351":2,"373":2,"445":1,"449":1,"455":1,"515":4,"588":1,"718":1}}],["kv",{"2":{"595":2,"597":3,"610":5}}],["kjnodes",{"2":{"453":1,"674":1,"675":1,"676":1,"683":1}}],["k×k",{"2":{"416":3,"418":1,"424":1}}],["kr",{"2":{"392":6}}],["kbps",{"2":{"303":2,"656":2}}],["known",{"2":{"201":1,"206":1}}],["knowing",{"2":{"48":1}}],["know",{"2":{"44":1,"45":1,"47":2,"107":1,"223":1,"229":1,"379":1,"440":1,"598":1}}],["knowledgeable",{"2":{"42":1,"43":1}}],["knowledge",{"2":{"34":1,"43":1,"46":1,"65":1,"127":2,"133":1,"201":1}}],["kept",{"2":{"403":1}}],["keplrbutton",{"2":{"398":2}}],["keplr",{"0":{"398":1,"428":1,"429":1},"1":{"429":1,"430":1,"431":1,"432":1},"2":{"397":1,"398":18,"426":1,"429":2,"430":1,"431":1,"452":1,"453":1,"506":1}}],["keeping",{"2":{"417":1}}],["keeps",{"2":{"108":1,"148":1}}],["keep",{"2":{"41":1,"42":1,"136":1,"237":2,"466":1,"504":1,"592":1,"599":3,"651":2,"727":1}}],["keyname",{"0":{"647":1}}],["keyname=",{"2":{"367":1}}],["keyfile",{"2":{"491":2}}],["keybase",{"2":{"491":1}}],["keymake",{"2":{"307":1,"559":1}}],["keyring",{"2":{"237":6,"238":7,"239":6,"243":2,"244":4,"264":2,"270":2,"293":1,"310":10,"367":4,"392":10,"485":2,"486":7,"489":4,"491":1,"507":6,"527":3,"531":4,"536":2,"553":3,"620":2,"621":7,"646":2,"647":2,"649":2,"661":10,"663":7,"721":2,"727":2}}],["key`",{"2":{"236":4}}],["keycast",{"2":{"121":1,"253":1,"263":1,"264":1}}],["key=ac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80",{"2":{"272":2}}],["key=$",{"2":{"264":2,"270":2}}],["key=0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80",{"2":{"251":2,"252":2}}],["key=0xe887f7d17d07cc7b8004053fb8826f6657084e88904bb61590e498ca04704cf2",{"2":{"120":2}}],["key=0xb6b15c8cb491557369f3c7d2c287b053eb229daa9c22138887752191c9520659",{"2":{"75":2}}],["key=",{"2":{"91":2,"268":2,"370":2}}],["keys`",{"2":{"392":4}}],["keys",{"0":{"237":1,"238":1,"239":1,"244":1,"310":1,"491":1,"520":1,"661":1},"1":{"662":1},"2":{"27":1,"31":1,"86":1,"113":1,"235":2,"237":2,"243":1,"244":8,"263":2,"264":2,"270":2,"278":1,"308":1,"486":1,"488":4,"489":1,"490":16,"491":6,"500":2,"507":10,"519":2,"520":2,"521":1,"527":8,"531":2,"536":4,"553":3,"583":1,"621":1,"626":5,"630":6,"631":1,"646":2,"648":1,"663":1,"721":2}}],["key",{"0":{"23":1,"106":1,"235":1,"240":1,"241":1,"279":1,"368":1,"369":1,"486":1,"490":1,"519":1,"579":1,"621":1,"646":1,"663":1},"1":{"107":1,"108":1,"109":1,"110":1,"236":1,"237":1,"238":1,"239":1,"240":1,"242":1,"243":1,"244":1,"491":1},"2":{"38":2,"43":1,"55":1,"75":5,"76":9,"84":3,"88":1,"91":2,"106":1,"119":3,"120":5,"121":3,"137":2,"155":4,"158":5,"173":3,"234":2,"235":2,"236":5,"237":18,"238":13,"239":12,"240":2,"243":5,"244":11,"251":5,"252":5,"253":3,"263":7,"264":12,"266":4,"270":6,"272":14,"279":1,"307":2,"310":15,"322":1,"326":4,"357":1,"366":1,"367":5,"369":2,"370":2,"373":2,"377":1,"392":2,"415":1,"460":1,"485":5,"486":9,"490":8,"491":13,"494":4,"507":2,"517":1,"518":1,"519":8,"520":6,"521":1,"523":2,"524":4,"525":4,"527":1,"553":3,"559":2,"575":2,"579":1,"610":1,"620":5,"621":9,"626":2,"629":12,"646":2,"647":1,"661":16,"663":9}}],["kit",{"2":{"57":2,"58":1,"63":1}}],["kickstarting",{"2":{"39":1}}],["kinds",{"2":{"180":1}}],["kindly",{"2":{"52":1}}],["kind",{"2":{"8":1,"34":1}}],["kindness",{"2":{"8":1}}],["k",{"2":{"31":1,"34":3,"161":8,"418":3}}],["64x64",{"2":{"671":1}}],["64",{"2":{"540":1,"560":1,"671":1}}],["6093df76dba90f04ff63d197fc1569f04ed3dbe64081a0bba9bad4e69aa570d2",{"2":{"522":1}}],["6093df76dba90f04ff63d197fc1569f04ed3dbe64081a0bba9bad4e69aa570d2gas",{"2":{"522":1}}],["6000000",{"2":{"374":2}}],["67",{"2":{"450":3}}],["6363c68770c200fd794445668f9b18f5b1dd1125180d6e8d5ab004f7dd7a0f48",{"2":{"326":4}}],["65",{"2":{"324":4,"385":1,"386":1}}],["62562",{"2":{"320":2}}],["6962",{"2":{"128":1}}],["6",{"0":{"88":1},"2":{"21":1,"24":1,"27":1,"31":1,"34":1,"80":1,"82":1,"398":6,"399":1,"400":1,"449":2,"455":1,"479":1,"614":1,"718":1}}],["jwttoken",{"2":{"373":4}}],["jwt",{"2":{"370":3,"373":3}}],["jemalloc",{"2":{"307":2,"559":2}}],["jemallocmake",{"2":{"307":1,"559":1}}],["jq",{"2":{"304":10,"327":2,"329":3,"332":3,"585":10,"712":4}}],["jail",{"2":{"509":1}}],["jailed",{"2":{"24":2,"25":1,"27":1,"509":1}}],["javascriptyarn",{"2":{"265":1}}],["javascript",{"2":{"265":2}}],["j",{"2":{"155":4,"158":4,"161":8}}],["jsimport",{"2":{"398":3}}],["jsximport",{"2":{"267":1,"268":1}}],["jsx",{"2":{"267":1,"268":1,"270":3,"398":1}}],["jsxnpm",{"2":{"266":1}}],["jsxyarn",{"2":{"265":1}}],["js",{"2":{"114":2,"246":2,"258":1,"265":1,"398":2}}],["jsonrpc",{"2":{"322":2,"323":2,"324":2,"325":2,"326":2,"334":4}}],["jsondocker",{"2":{"102":2}}],["json",{"0":{"650":1},"2":{"68":1,"77":4,"84":2,"86":2,"87":2,"88":1,"89":1,"91":2,"92":1,"102":6,"134":2,"137":2,"265":3,"268":2,"273":4,"280":1,"295":1,"317":2,"319":4,"320":1,"322":1,"323":1,"324":1,"325":1,"334":2,"353":1,"393":1,"488":6,"494":9,"495":5,"500":1,"507":18,"592":1,"648":3,"649":3,"650":6}}],["jcstein",{"2":{"75":2,"272":2}}],["journalctl",{"2":{"499":6,"712":2,"714":2,"715":2,"716":2}}],["journey",{"2":{"35":1,"50":1}}],["joho",{"2":{"372":2}}],["job",{"2":{"48":1,"228":1,"412":1,"503":8}}],["joining",{"0":{"48":1},"2":{"42":1,"47":1,"53":1,"65":1}}],["join",{"2":{"19":1,"50":1,"476":1,"684":1,"695":1,"705":1}}],["jurisdiction",{"2":{"24":1,"34":5}}],["justifycontent",{"2":{"268":6}}],["just",{"2":{"8":1,"85":1,"91":2,"102":1,"126":1,"135":1,"235":1,"263":1,"329":1,"360":1,"367":1,"373":1,"425":1,"447":1,"452":1,"453":1,"466":1,"503":1,"511":1,"620":1,"703":1,"709":1}}],["482",{"2":{"671":1}}],["478",{"2":{"671":1}}],["4hermes",{"2":{"633":1}}],["443export",{"2":{"529":1}}],["443",{"2":{"470":1,"529":1,"531":2,"629":2,"676":6,"691":3}}],["4️⃣",{"0":{"432":1}}],["4k",{"2":{"416":1,"417":1,"418":1,"424":1,"425":1}}],["4celestia",{"2":{"341":1}}],["46",{"2":{"149":2}}],["43494",{"2":{"121":4,"253":4}}],["49s",{"2":{"91":2}}],["4989526079",{"2":{"91":2}}],["4094×482bytes",{"2":{"671":1}}],["4095",{"2":{"671":1}}],["4096",{"2":{"671":1}}],["400000",{"2":{"629":4}}],["400px",{"2":{"268":2}}],["40px",{"2":{"268":2}}],["40",{"2":{"43":1,"326":2}}],["4",{"0":{"16":1,"34":1,"86":1},"2":{"22":1,"23":3,"26":1,"27":1,"31":1,"34":13,"80":1,"82":1,"85":1,"89":2,"91":2,"95":1,"161":2,"172":1,"207":1,"244":4,"285":1,"317":2,"341":2,"348":2,"376":1,"441":1,"449":1,"450":1,"453":1,"470":2,"479":1,"614":1,"629":3,"630":2,"632":1,"633":5,"634":10,"635":3,"688":3,"689":2,"690":1,"691":4,"692":8}}],["3152",{"2":{"634":2}}],["3108",{"2":{"629":2,"635":2}}],["33373",{"2":{"674":1,"677":1}}],["33554432",{"2":{"629":4}}],["33",{"2":{"450":3,"542":2}}],["3️⃣",{"0":{"431":1,"436":1}}],["3d96b7d238e7e0456f6af8e7cdf0a67bd6cf9c2089ecb559c659dcaa1f880353",{"2":{"326":2}}],["34",{"2":{"149":2,"628":1}}],["32",{"2":{"149":2,"151":6,"155":3,"158":4,"159":6,"161":14,"351":2}}],["39",{"2":{"139":2,"151":2,"159":2,"244":4,"312":8,"317":8,"320":4,"326":2,"327":4,"332":4,"334":4,"352":4,"392":8,"398":62,"488":2,"503":2,"519":4,"520":8,"522":8,"523":4,"524":8,"531":2,"578":2,"592":40,"607":4,"635":24,"721":4,"727":4}}],["3767182372",{"2":{"121":2,"253":2}}],["3001",{"2":{"629":2}}],["30000",{"2":{"629":2}}],["30000000",{"2":{"347":2}}],["3000",{"2":{"503":1,"629":2}}],["30s",{"2":{"360":1,"629":4}}],["30sda",{"2":{"360":1}}],["30t08",{"2":{"326":2}}],["30",{"2":{"91":2,"155":2,"161":2,"441":4,"629":4}}],["3663",{"2":{"628":1}}],["3623",{"2":{"628":1}}],["362101",{"2":{"334":2}}],["36",{"2":{"23":1}}],["35",{"2":{"23":1}}],["3",{"0":{"15":1,"33":1,"84":1,"85":1},"2":{"23":1,"30":1,"31":1,"34":1,"43":1,"89":2,"91":4,"94":1,"102":2,"106":1,"121":2,"148":1,"149":6,"161":2,"244":2,"253":2,"365":2,"366":1,"373":3,"375":1,"450":1,"470":2,"494":1,"502":2,"507":4,"540":1,"560":1,"629":4}}],["2382",{"2":{"634":4}}],["26660",{"2":{"502":4,"503":2}}],["26656",{"2":{"470":1,"608":2,"652":1}}],["26650",{"2":{"349":2,"352":2,"360":7}}],["26659version",{"2":{"244":1}}],["26659",{"2":{"243":4,"244":3,"360":6,"367":2,"552":1}}],["26657",{"2":{"89":2,"150":2,"151":2,"334":1,"470":1,"471":1,"485":2,"541":2,"552":1,"607":4,"629":4,"679":6,"688":1,"689":3,"712":2}}],["26658curl",{"2":{"334":1}}],["26658",{"2":{"89":3,"313":1,"334":1,"360":4,"370":2,"552":1}}],["26",{"2":{"449":1}}],["2️⃣",{"0":{"430":1,"435":1}}],["2k",{"2":{"418":1}}],["2k×2k",{"2":{"416":2,"424":1}}],["2727",{"2":{"634":6}}],["27t13",{"2":{"326":2}}],["27709",{"2":{"117":2,"249":2}}],["250",{"2":{"588":2}}],["25",{"2":{"400":1,"450":1,"509":1}}],["252614",{"2":{"317":3,"319":9,"332":2}}],["252607",{"2":{"317":2}}],["255",{"2":{"268":6}}],["25630",{"2":{"324":4}}],["256",{"2":{"117":2,"249":2}}],["2s",{"2":{"229":1}}],["2283",{"2":{"347":2}}],["22",{"2":{"161":2,"243":2,"244":2,"285":1,"316":4}}],["299137127z",{"2":{"326":2}}],["29",{"2":{"149":2,"155":7,"156":1,"161":2}}],["28",{"2":{"155":5,"156":3,"161":8}}],["28409",{"2":{"117":2,"249":2}}],["28334",{"2":{"117":2,"249":2}}],["28217653",{"2":{"91":2}}],["28217647",{"2":{"91":2}}],["2f",{"2":{"69":1}}],["2e",{"2":{"69":1}}],["2d",{"2":{"69":1}}],["2c",{"2":{"69":1,"72":1}}],["2blockhash",{"2":{"121":1,"253":1}}],["2b",{"2":{"69":1,"72":1}}],["2ae1ddf74ef7",{"2":{"149":2}}],["2a",{"2":{"69":1,"72":1}}],["24s",{"2":{"116":2,"248":2}}],["24",{"2":{"25":1,"348":2}}],["21090",{"2":{"691":1}}],["21572",{"2":{"348":2}}],["21568",{"2":{"347":2}}],["2121",{"2":{"243":2,"244":2,"324":8,"470":4,"541":2,"542":4,"552":2,"715":2}}],["21",{"2":{"23":1,"116":2,"243":2,"244":2,"248":2,"509":1}}],["2019",{"2":{"666":1}}],["20000",{"2":{"629":2}}],["200",{"2":{"324":4}}],["20+",{"2":{"265":1}}],["20210819022825",{"2":{"149":2}}],["2023",{"2":{"30":1,"326":4}}],["2024",{"2":{"21":1}}],["20",{"2":{"23":1,"43":1,"150":2,"281":1,"449":1,"480":1,"589":1,"615":1,"657":1,"719":1}}],["2",{"0":{"14":1,"32":1,"69":1,"72":1,"83":1,"152":1,"582":1,"632":1},"2":{"17":2,"23":2,"24":1,"27":1,"31":1,"32":1,"34":1,"41":1,"43":1,"48":1,"71":1,"72":1,"80":1,"91":2,"94":4,"117":4,"121":1,"127":2,"148":1,"152":1,"161":3,"173":1,"185":1,"249":4,"253":1,"268":2,"295":1,"322":2,"323":2,"324":2,"325":2,"326":4,"334":4,"341":3,"343":1,"347":2,"416":1,"418":3,"419":3,"421":1,"424":1,"448":1,"450":2,"460":1,"470":2,"471":2,"500":2,"507":4,"509":1,"524":2,"626":2,"679":2,"688":2,"689":1,"690":1,"691":1,"692":6,"722":1,"727":2}}],["1×478bytes",{"2":{"671":1}}],["1️⃣",{"0":{"429":1,"434":1}}],["1utia",{"2":{"392":2,"507":2}}],["17",{"2":{"449":1}}],["171",{"2":{"324":4}}],["172118057",{"2":{"322":2}}],["1px",{"2":{"268":2}}],["1s",{"2":{"228":1,"229":1}}],["19000000utia",{"2":{"493":2}}],["19",{"2":{"116":4,"126":1,"127":2,"248":4,"326":2,"449":1}}],["1933",{"2":{"31":2,"34":3}}],["151",{"2":{"542":2}}],["15s",{"2":{"503":2}}],["1559",{"2":{"384":1}}],["15",{"2":{"91":2,"150":2,"152":2,"159":1,"268":4,"449":1,"503":2}}],["15002",{"2":{"676":1}}],["1500",{"2":{"27":1}}],["1443",{"2":{"691":1}}],["1400000",{"2":{"557":1}}],["147105",{"2":{"327":2}}],["14",{"2":{"91":2,"149":2,"542":2}}],["1317",{"2":{"470":1,"690":1}}],["13",{"2":{"80":1,"91":2,"94":2,"116":2,"117":2,"248":2,"249":2,"260":4,"262":2,"441":2,"560":1,"670":1}}],["1g",{"2":{"71":1}}],["1f",{"2":{"71":1}}],["1em",{"2":{"268":2}}],["1ether",{"2":{"76":2,"272":2}}],["1e",{"2":{"71":1}}],["1d",{"2":{"71":1}}],["1container",{"2":{"243":1,"244":1}}],["1c",{"2":{"68":1,"71":1}}],["1b",{"2":{"68":1,"71":1}}],["1a",{"2":{"68":1,"71":1}}],["11090",{"2":{"676":1}}],["11036",{"2":{"503":1}}],["115",{"2":{"542":2}}],["118",{"2":{"398":2,"399":1,"400":1}}],["1141",{"2":{"347":2}}],["11",{"2":{"34":1,"309":2,"326":4,"399":2,"470":11,"471":2,"474":1,"475":1,"485":2,"660":2,"661":2}}],["180000",{"2":{"629":4}}],["18",{"2":{"31":1,"84":1,"149":2,"267":2}}],["1687908352",{"2":{"523":2}}],["1686748051",{"2":{"522":8,"523":2,"529":2}}],["16xlarge",{"2":{"378":1}}],["1699638350",{"2":{"347":2}}],["16002",{"2":{"691":1}}],["160",{"2":{"285":1}}],["16px",{"2":{"268":2}}],["16",{"2":{"30":1,"91":2,"161":2,"602":12}}],["103",{"2":{"494":2}}],["105",{"2":{"494":3}}],["1089",{"2":{"395":1}}],["10s",{"2":{"360":2,"629":4}}],["1040676758",{"2":{"348":2}}],["106",{"2":{"324":4}}],["10px",{"2":{"268":6}}],["10+",{"2":{"40":1}}],["10",{"2":{"24":3,"34":1,"89":4,"121":3,"150":2,"253":3,"266":2,"285":1,"317":1,"324":4,"385":1,"399":1,"445":1,"455":1,"473":1,"479":1,"614":1,"629":4,"693":1}}],["1000",{"2":{"629":2}}],["10001",{"2":{"578":4}}],["1000utia",{"2":{"507":4}}],["10000",{"2":{"374":2}}],["100000utia",{"2":{"507":2,"522":12,"529":4}}],["100000",{"2":{"329":4,"522":4,"523":2,"525":2,"529":2,"629":4}}],["1000000utia",{"2":{"721":2}}],["10000000utia",{"2":{"647":2}}],["100000000000utia",{"2":{"495":2}}],["1000000000",{"2":{"494":2}}],["1000000000utia",{"2":{"494":2,"495":2}}],["1000000",{"2":{"323":2,"499":2}}],["100vh",{"2":{"268":2}}],["100",{"2":{"21":1,"89":2,"160":2,"268":2,"444":2,"629":2}}],["12d3koowd5wcbjxkqudjhxfjtfmrzoysgvltvht5hmovbslcbv22",{"2":{"715":1}}],["12d3koowd5wcbjxkqudjhxfjtfmrzoysgvltvht5hmovbslcbv22node",{"2":{"715":1}}],["12d3koowkeertzvmpudxyszo2edqps6ms67n6lt5mpdulskpsxbq",{"2":{"542":2}}],["12d3koowlt1yssrd7xwsbjh7tu1hqanf5m64dhv6aum6cyejxmpk",{"2":{"470":1}}],["12d3koowewuqrjulanpukdfgvohw3roeuu53ec9t9v5cww3mkvdq",{"2":{"470":1}}],["12d3koowcmgm5ezwvfcn9zlavigfluwafxp5pcm78nfkb9jpbtua",{"2":{"470":1}}],["12d3koowgqwzdeqm54dce6lxzffr97bnhvm6rn7km7mfwdomfm4s",{"2":{"470":1}}],["12d3koowffhcaaqy56oeqy3plzudlsv4ryafvwkatzrepupdoslp",{"2":{"324":2}}],["1234567890",{"2":{"392":2}}],["12345",{"2":{"260":14,"263":2,"264":2}}],["127",{"2":{"251":1,"334":3,"349":2,"352":2,"367":2,"607":2,"629":4}}],["12",{"2":{"23":2,"283":1,"341":3,"360":4,"449":1}}],["1",{"0":{"13":1,"31":1,"68":1,"71":1,"82":1,"150":1},"2":{"17":2,"22":1,"23":7,"24":1,"25":1,"26":1,"30":1,"31":8,"32":1,"34":2,"41":1,"48":1,"80":1,"82":1,"83":1,"91":4,"94":6,"102":4,"116":2,"117":3,"121":2,"137":4,"139":2,"148":1,"149":4,"150":6,"151":2,"152":3,"155":6,"156":2,"159":2,"161":14,"207":2,"228":2,"229":1,"243":1,"244":1,"248":2,"249":3,"251":1,"253":2,"260":12,"309":2,"322":2,"323":2,"324":6,"325":2,"326":13,"329":1,"334":7,"343":1,"346":1,"349":2,"351":1,"352":2,"353":1,"360":4,"367":2,"376":1,"388":1,"392":2,"398":2,"400":1,"409":1,"418":1,"445":1,"449":1,"450":8,"455":3,"463":2,"470":2,"471":2,"479":2,"485":2,"494":1,"499":2,"500":2,"509":1,"588":2,"607":2,"614":2,"626":2,"629":14,"660":2,"661":2,"671":1,"692":4,"718":2,"727":2}}],["uatom",{"2":{"629":2}}],["udp",{"2":{"324":6,"541":1,"552":1}}],["uri",{"2":{"309":6,"310":6,"471":1,"484":8,"485":2,"486":6,"493":4,"544":4,"545":2,"546":2,"547":8,"550":2,"620":2,"621":6,"660":2,"663":6,"715":2,"716":2,"724":2}}],["urldocker",{"2":{"580":3}}],["url=this",{"2":{"577":2}}],["url=",{"2":{"91":4}}],["url=https",{"2":{"252":2,"272":2,"529":2}}],["url=http",{"2":{"75":2,"120":2,"251":2}}],["urlexport",{"2":{"76":1,"272":1}}],["urlcast",{"2":{"76":3,"121":1,"253":1,"272":3}}],["url",{"0":{"88":1},"2":{"47":2,"75":5,"76":12,"88":5,"89":4,"91":1,"120":4,"121":7,"217":1,"246":1,"251":4,"252":5,"253":7,"263":2,"264":6,"269":1,"272":17,"312":1,"313":1,"347":2,"348":2,"360":3,"377":2,"399":3,"400":4,"471":2,"503":2,"529":5,"531":1,"577":9,"580":3,"629":4,"679":3,"688":1}}],["urls",{"0":{"215":1},"1":{"216":1,"217":1},"2":{"47":2,"88":1,"215":2}}],["ubuntu",{"2":{"281":1,"285":1,"304":1,"305":2,"480":1,"575":1,"585":1,"586":2,"589":1,"615":1,"657":1,"719":1}}],["utility",{"0":{"235":1},"1":{"236":1,"237":1,"238":1,"239":1,"240":1},"2":{"234":1,"235":3,"237":1,"307":1,"559":1,"661":1}}],["utilizing",{"0":{"47":1}}],["utilizes",{"2":{"206":1}}],["utilize",{"2":{"43":1,"45":2,"46":1,"282":1}}],["utia=tia×10−6",{"2":{"455":1}}],["utia",{"0":{"329":1,"463":1},"2":{"89":1,"320":1,"322":2,"323":2,"329":6,"398":6,"399":1,"400":1,"463":3,"464":1,"494":2,"522":4,"523":2,"525":4,"629":2}}],["uint32",{"2":{"387":4,"392":4}}],["uint",{"2":{"260":18}}],["uint8",{"2":{"156":2}}],["uint64",{"2":{"134":8,"135":4,"137":4,"139":4,"151":10,"159":4,"161":14,"373":8,"374":2,"379":3}}],["uint256",{"2":{"116":4,"117":4,"121":2,"127":2,"153":2,"155":4,"158":4,"159":4,"173":3,"248":4,"249":4,"253":2,"379":1}}],["ui",{"2":{"68":1,"71":1,"72":2,"84":1}}],["ultimate",{"2":{"49":1}}],["u",{"2":{"31":2,"34":5,"499":6,"712":2,"714":2,"715":2,"716":2}}],["upper",{"2":{"671":1}}],["upgrading",{"0":{"562":1}}],["upgradeexecutor",{"2":{"91":2}}],["upgrade",{"0":{"510":1},"2":{"25":1,"91":4,"94":1,"278":1,"304":3,"510":1,"562":1,"585":3,"624":3,"625":4,"626":2}}],["upgrades",{"0":{"466":1,"476":1,"667":1,"684":1,"695":1,"705":1},"2":{"24":1,"465":1,"466":1,"476":1,"494":1,"510":1,"624":2,"625":1,"666":1,"667":1,"684":1,"695":1,"703":1,"705":1}}],["upmake",{"2":{"343":1,"360":1}}],["upload",{"2":{"268":1,"283":1,"303":1,"479":1,"588":1,"614":1,"630":1,"656":1,"718":1}}],["uploaded",{"2":{"264":1}}],["uploading",{"2":{"258":1}}],["updocker",{"2":{"244":1}}],["updating",{"0":{"260":1,"262":1},"2":{"169":1,"376":2,"515":2}}],["updatepost",{"2":{"260":6}}],["updatename",{"2":{"260":2}}],["updateheaders",{"2":{"137":2}}],["updatesudo",{"2":{"304":1,"585":1}}],["updates",{"2":{"117":1,"208":1,"249":1,"260":4,"466":1,"667":1,"698":1,"701":1}}],["update",{"0":{"77":1,"273":1},"2":{"77":2,"84":1,"149":1,"169":1,"206":3,"207":1,"208":1,"229":1,"260":2,"262":1,"267":1,"268":1,"270":1,"273":2,"304":4,"341":4,"374":1,"376":1,"388":1,"403":1,"422":1,"441":1,"494":2,"554":4,"585":4,"650":1}}],["updatedpost",{"2":{"260":4}}],["updated",{"2":{"5":1,"91":2,"137":1,"207":1,"256":1,"365":1,"376":1,"466":1,"701":1}}],["upcoming",{"2":{"53":1,"166":1,"201":1,"624":1}}],["upstream",{"2":{"341":1}}],["ups",{"2":{"39":1}}],["upon",{"2":{"34":3,"56":1,"137":1,"143":1,"319":1,"334":1,"450":1,"486":1,"621":1,"663":1}}],["up",{"0":{"91":1,"304":1,"374":1,"477":1,"480":1,"513":1,"518":1,"530":1,"587":1,"589":1,"590":1,"592":1,"612":1,"615":1,"643":1,"654":1,"657":1,"710":1,"717":1,"719":1},"1":{"478":1,"479":1,"480":1,"481":2,"482":1,"483":1,"484":1,"485":1,"486":1,"487":1,"514":1,"515":1,"516":1,"517":1,"519":1,"520":1,"521":1,"522":1,"523":1,"524":1,"525":1,"588":1,"589":1,"590":2,"591":2,"592":2,"593":1,"594":1,"595":1,"596":1,"597":1,"598":1,"599":1,"600":1,"601":1,"602":1,"603":1,"604":1,"605":1,"606":1,"607":1,"608":1,"609":1,"610":1,"611":1,"613":1,"614":1,"615":1,"616":2,"617":1,"618":1,"619":1,"620":1,"621":1,"622":1,"623":1,"644":1,"645":1,"646":1,"647":1,"648":1,"649":1,"650":1,"651":1,"652":1,"653":1,"655":1,"656":1,"657":1,"658":2,"659":1,"660":1,"661":1,"662":1,"663":1,"664":1,"665":1,"711":1,"712":1,"713":1,"714":1,"715":1,"716":1,"718":1,"719":1,"720":2,"721":2,"722":1,"723":1,"724":1,"725":1,"726":1,"727":1,"728":1,"729":1,"730":1},"2":{"21":1,"23":1,"26":1,"36":1,"37":1,"38":1,"47":3,"48":1,"54":1,"60":1,"61":1,"88":2,"90":1,"115":1,"166":2,"213":1,"229":1,"234":1,"237":1,"243":2,"244":3,"247":1,"268":1,"285":1,"290":1,"304":2,"308":1,"316":1,"343":2,"360":1,"366":1,"367":1,"369":1,"373":2,"380":1,"394":1,"411":1,"412":1,"477":1,"487":2,"501":1,"503":1,"504":1,"517":1,"526":2,"529":1,"530":1,"539":1,"547":1,"559":1,"569":1,"584":1,"587":1,"592":1,"600":2,"612":1,"616":1,"631":1,"643":1,"644":1,"654":1,"657":2,"664":1,"696":2,"711":1,"712":1,"717":1,"719":1,"726":2,"727":1}}],["uptime",{"2":{"21":1,"23":1,"27":2,"33":1,"221":1,"544":1,"679":1}}],["unexplored",{"2":{"667":1}}],["unexpectedly",{"2":{"466":1}}],["unordered",{"2":{"634":2}}],["unjail",{"2":{"498":3,"509":2}}],["unjailing",{"0":{"498":1}}],["unbonded",{"2":{"509":1}}],["unbonding",{"2":{"509":1}}],["unbond",{"2":{"497":3}}],["unused",{"2":{"384":1,"386":1}}],["untested",{"2":{"281":1}}],["until",{"2":{"107":1,"139":1,"388":1,"392":2,"411":1,"445":1,"455":1,"466":1,"628":1,"712":1}}],["unknown",{"2":{"159":1}}],["unstable",{"2":{"701":1}}],["unstake",{"2":{"432":1,"436":1}}],["unsafe",{"2":{"264":2,"270":2,"555":4,"605":2,"644":2}}],["unsubscribe",{"2":{"160":2}}],["unsignedtx",{"2":{"507":8}}],["unsigned",{"2":{"116":1,"117":1,"248":1,"249":1}}],["unsolicited",{"2":{"14":1,"15":1}}],["unpacked",{"2":{"109":1}}],["unpealed",{"2":{"108":2}}],["unprofessional",{"2":{"13":1}}],["unlocked",{"2":{"450":9}}],["unlock",{"2":{"398":2,"450":3}}],["unlocks",{"0":{"450":1},"2":{"289":1,"450":4}}],["unlocking",{"2":{"55":1}}],["unlimited",{"2":{"502":2}}],["unlike",{"2":{"136":1,"167":1,"293":1,"379":1}}],["unlicensed",{"2":{"116":2,"117":2,"248":2,"249":2}}],["unless",{"2":{"24":1,"31":2,"34":4,"334":1,"643":1}}],["unified",{"2":{"148":1}}],["unit",{"2":{"117":1,"249":1,"392":2,"712":2,"714":2,"715":2,"716":2}}],["united",{"2":{"34":9}}],["university",{"2":{"40":1}}],["universities",{"2":{"40":2,"44":1}}],["unique",{"2":{"33":1,"47":1,"84":2,"231":1,"416":2,"418":1,"515":1,"626":1}}],["unavailability",{"2":{"255":1}}],["unavailable",{"0":{"172":1},"2":{"139":1,"177":1}}],["unaffected",{"2":{"232":1}}],["unaffiliated",{"2":{"31":1}}],["unable",{"2":{"31":3,"388":1,"409":1,"509":1}}],["unacceptable",{"2":{"8":1,"11":1}}],["undelegate",{"0":{"497":1},"2":{"497":1}}],["undelegation",{"0":{"25":1}}],["undergoes",{"2":{"466":1}}],["underneath",{"0":{"92":1},"2":{"107":2,"108":1}}],["underlying",{"2":{"84":1,"91":1,"107":1,"110":1,"351":1}}],["underwriter",{"2":{"34":1}}],["undertake",{"2":{"34":1}}],["undertaking",{"2":{"34":1}}],["understanding",{"0":{"565":1},"1":{"566":1,"567":1,"568":1,"569":1,"570":1,"571":1},"2":{"42":1,"65":1,"114":2,"222":1,"229":1,"242":1,"246":2,"445":1,"576":2}}],["understand",{"2":{"30":1,"33":1,"34":4,"47":1,"48":3,"65":1,"89":1,"130":1,"291":1,"439":1,"529":1}}],["under",{"2":{"24":1,"27":1,"30":1,"31":1,"34":10,"68":2,"72":1,"88":1,"94":1,"129":1,"192":1,"235":1,"237":1,"367":1,"384":1,"502":2,"557":1,"601":1,"729":1}}],["unwelcome",{"2":{"13":1}}],["usr",{"2":{"305":21,"540":1,"560":1,"586":21}}],["usual",{"2":{"173":1}}],["usually",{"2":{"84":1,"485":1,"569":1,"620":1,"660":2}}],["usable",{"0":{"153":1},"1":{"154":1,"155":1,"156":1,"157":1,"158":1,"159":1,"160":1,"161":1},"2":{"130":1}}],["usage",{"0":{"127":1,"192":1,"193":1,"194":1},"1":{"193":1,"194":1},"2":{"194":1,"307":1,"488":2,"491":1,"492":1,"493":2,"504":1,"559":1}}],["using",{"0":{"120":1,"235":1,"251":1,"252":1,"334":1,"356":1,"357":1,"358":1,"471":1},"1":{"236":1,"237":1,"238":1,"239":1,"240":1},"2":{"10":1,"33":1,"36":1,"37":1,"48":1,"75":2,"79":1,"82":1,"84":1,"88":1,"89":6,"91":1,"108":1,"110":1,"111":1,"113":1,"119":1,"121":1,"126":3,"131":3,"135":1,"136":2,"139":1,"146":1,"148":3,"150":2,"152":3,"154":1,"159":2,"160":1,"161":1,"166":1,"167":1,"168":2,"170":3,"171":2,"173":6,"174":1,"176":1,"177":2,"182":1,"184":1,"186":1,"189":3,"192":1,"195":1,"201":2,"207":1,"213":2,"233":1,"234":1,"236":1,"243":1,"244":3,"253":1,"255":1,"260":1,"262":1,"263":1,"265":2,"266":1,"268":1,"270":3,"272":2,"278":1,"290":2,"301":1,"310":1,"311":1,"314":1,"315":1,"316":2,"318":1,"332":1,"334":2,"355":1,"360":2,"365":3,"373":2,"377":1,"387":2,"391":1,"392":4,"393":2,"398":2,"418":1,"419":2,"421":3,"424":2,"438":1,"442":2,"458":1,"471":1,"472":1,"484":1,"486":1,"494":1,"495":1,"505":1,"506":1,"511":1,"521":1,"529":1,"534":1,"539":1,"542":1,"553":3,"554":1,"575":3,"577":1,"579":2,"583":1,"588":2,"597":1,"599":1,"600":1,"620":1,"621":1,"627":1,"629":1,"660":1,"661":1,"662":1,"663":1,"690":2,"691":1,"693":1,"721":1}}],["useaccount",{"2":{"268":4}}],["useeffect",{"2":{"268":6}}],["useful",{"2":{"48":1,"57":1,"90":1,"466":1,"690":1,"710":1,"712":1}}],["user=$user",{"2":{"712":2,"714":2,"715":2,"716":2}}],["user",{"0":{"392":1},"2":{"42":1,"47":2,"48":1,"131":2,"133":1,"228":1,"255":1,"260":2,"265":1,"267":1,"278":1,"292":1,"293":1,"313":2,"317":1,"384":2,"386":3,"388":1,"392":8,"396":1,"398":4,"459":1,"462":1,"503":1,"508":1,"553":8,"571":1,"578":1,"638":2,"712":2,"714":2,"715":2,"716":2}}],["users",{"2":{"19":1,"30":1,"84":1,"163":1,"181":1,"201":2,"232":1,"255":1,"268":3,"295":1,"297":1,"316":1,"375":1,"382":1,"384":3,"386":3,"387":1,"388":2,"389":1,"402":2,"409":2,"412":2,"413":1,"423":1,"424":1,"442":1,"451":1,"575":1,"583":1,"667":1,"679":1,"688":1,"689":1,"690":1,"691":1,"698":1}}],["useshareexchange",{"2":{"357":1}}],["usestate",{"2":{"268":10}}],["uses",{"0":{"161":1},"2":{"40":1,"94":1,"105":1,"110":1,"121":1,"135":4,"139":1,"161":3,"176":1,"180":1,"186":1,"195":1,"203":1,"204":1,"213":1,"235":1,"253":1,"260":1,"285":1,"290":1,"296":1,"351":1,"384":1,"386":1,"387":1,"388":1,"407":1,"416":1,"439":1,"445":1,"459":1,"502":1,"509":1,"601":1,"627":1,"638":1}}],["use",{"0":{"177":1,"178":1,"179":1,"180":1,"181":1,"182":1,"183":1,"185":1,"186":1,"187":1,"188":1,"190":1,"191":1},"1":{"178":1,"179":1,"180":1,"182":1,"183":1,"186":1,"187":1,"189":1,"190":1,"191":1},"2":{"8":1,"13":1,"19":1,"31":3,"33":3,"34":1,"45":1,"47":1,"48":2,"82":1,"88":2,"89":3,"102":2,"107":1,"119":1,"120":1,"121":1,"133":1,"134":2,"136":2,"137":1,"142":1,"146":1,"148":1,"151":2,"153":1,"155":2,"159":2,"160":1,"161":2,"163":1,"172":2,"173":2,"184":1,"188":1,"195":1,"198":1,"201":1,"206":2,"217":1,"232":1,"236":1,"237":2,"239":1,"253":1,"260":2,"262":1,"263":4,"264":1,"270":1,"278":2,"280":1,"297":1,"299":1,"302":1,"307":1,"309":1,"312":4,"313":1,"314":1,"315":2,"317":4,"319":5,"320":2,"329":1,"330":1,"331":2,"334":1,"336":1,"345":1,"356":1,"357":1,"358":1,"360":2,"367":1,"370":1,"371":1,"372":2,"383":1,"384":1,"385":1,"392":6,"397":1,"398":1,"421":1,"422":1,"442":1,"457":1,"475":1,"484":1,"494":2,"495":1,"503":2,"510":1,"512":1,"515":1,"516":1,"522":1,"527":1,"529":2,"530":1,"531":1,"539":1,"547":1,"552":1,"559":1,"566":1,"575":1,"620":1,"623":1,"628":1,"630":1,"632":1,"633":2,"634":1,"635":1,"645":2,"650":1,"682":1,"694":1,"709":1}}],["used",{"2":{"3":1,"30":1,"84":2,"88":1,"89":4,"91":1,"107":3,"108":1,"134":2,"136":1,"137":2,"139":1,"155":1,"156":1,"172":1,"185":2,"201":1,"206":1,"208":1,"235":1,"278":1,"281":1,"283":1,"294":1,"304":1,"309":1,"313":2,"320":1,"332":1,"351":1,"368":1,"384":1,"385":2,"386":1,"387":2,"416":1,"421":1,"424":1,"465":1,"471":1,"483":1,"485":1,"494":1,"501":1,"509":1,"522":2,"536":1,"539":1,"541":1,"552":1,"565":1,"578":1,"584":1,"608":1,"611":1,"620":2,"625":1,"627":1,"630":1,"691":1,"692":1,"721":2,"727":2}}],["us",{"2":{"8":1,"24":1,"27":1,"30":1,"31":3,"33":2,"34":1,"48":6,"109":3,"232":1,"332":1,"373":3,"398":1,"671":2}}],["huzwottdmd36n1f75a9bshxnlrascnnpqiwqihdvhcu",{"2":{"326":4}}],["hub",{"2":{"214":1,"219":1,"627":1,"629":1,"630":1}}],["h2",{"2":{"268":8}}],["h3",{"2":{"268":8}}],["h1",{"2":{"268":8}}],["historic",{"2":{"441":1,"442":3}}],["historical",{"0":{"412":1,"413":1,"598":1},"2":{"409":2,"412":2,"413":2,"414":1,"440":1,"442":9,"598":1}}],["history",{"2":{"409":1,"413":1,"471":1,"587":1,"588":1,"679":1,"680":1,"696":1}}],["hidden",{"2":{"135":1,"365":1}}],["hiring",{"2":{"44":1}}],["highly",{"2":{"485":1,"573":1,"620":1}}],["highlighted",{"2":{"146":1}}],["highlight",{"2":{"44":1,"64":1}}],["highlights",{"2":{"38":1}}],["higher",{"2":{"84":1,"232":1,"320":1,"383":1,"388":1,"439":1,"445":1}}],["high",{"2":{"23":1,"46":1,"65":1,"84":1,"108":1,"109":2,"131":1,"133":1,"201":1,"205":1,"206":1,"398":2,"416":1,"557":1,"691":1}}],["h",{"2":{"34":1,"334":4}}],["hollow",{"2":{"519":2}}],["holds",{"2":{"86":1}}],["holders",{"2":{"447":1,"495":1}}],["holder",{"2":{"34":1}}],["hold",{"2":{"34":1,"42":1}}],["holding",{"2":{"34":2}}],["homecd",{"2":{"519":1}}],["home=",{"2":{"515":2}}],["home",{"0":{"515":1},"2":{"313":1,"360":2,"515":4,"519":3,"520":4,"522":12,"523":4,"524":4,"525":8,"541":1,"552":1,"553":8,"579":8,"580":8,"605":2,"644":2}}],["homebrew",{"2":{"304":4,"585":4}}],["hooks",{"2":{"265":1}}],["hood",{"2":{"94":1,"235":1,"384":1}}],["hosted",{"2":{"213":1,"377":1,"547":1}}],["hosts",{"2":{"89":2}}],["host",{"2":{"89":8,"102":2,"228":1,"229":1,"268":2,"370":1,"531":2,"578":3,"629":4,"633":6}}],["hosting",{"2":{"27":1,"43":1,"46":1,"48":2,"79":1}}],["honesty",{"2":{"409":1}}],["honest",{"2":{"84":1,"406":2,"416":1,"418":1,"425":2}}],["hour",{"2":{"48":1,"207":2,"376":1}}],["hours",{"2":{"25":1,"71":1,"72":1,"169":1,"207":1,"376":1}}],["hope",{"2":{"48":2}}],["however",{"2":{"44":1,"47":1,"135":1,"139":1,"173":1,"184":1,"195":1,"384":1,"385":1,"407":1,"417":1,"419":1,"421":1,"494":1,"547":1,"571":1,"611":1}}],["how",{"0":{"208":1,"209":1,"404":1,"426":1,"499":1,"511":1},"1":{"427":1,"428":1,"429":1,"430":1,"431":1,"432":1,"433":1,"434":1,"435":1,"436":1,"512":1,"513":1,"514":1,"515":1,"516":1,"517":1,"518":1,"519":1,"520":1,"521":1,"522":1,"523":1,"524":1,"525":1,"526":1,"527":1,"528":1,"529":1,"530":1,"531":1,"532":1,"533":1},"2":{"22":1,"41":1,"42":1,"44":2,"47":1,"48":1,"60":1,"66":1,"84":1,"89":1,"92":1,"109":1,"111":1,"113":1,"122":2,"130":1,"131":1,"133":1,"135":1,"136":1,"143":2,"146":1,"148":2,"170":1,"180":1,"207":1,"233":1,"242":1,"243":1,"254":1,"256":2,"280":1,"285":1,"290":1,"299":1,"311":1,"312":2,"314":1,"338":1,"365":1,"373":1,"374":1,"376":1,"385":2,"386":1,"387":2,"392":2,"397":2,"398":2,"405":1,"426":1,"438":1,"439":1,"441":1,"444":2,"445":1,"447":1,"448":1,"457":1,"459":1,"469":2,"501":2,"508":1,"510":1,"511":1,"525":1,"526":2,"530":1,"533":2,"534":1,"547":1,"587":1,"610":1,"611":1,"627":2,"672":2,"685":3,"696":1,"706":1}}],["http2",{"2":{"690":1}}],["httpexport",{"2":{"252":1,"272":1}}],["http",{"2":{"88":1,"89":4,"90":1,"91":3,"119":2,"120":1,"150":4,"151":4,"161":2,"216":1,"252":1,"263":2,"264":6,"266":2,"267":4,"268":2,"269":1,"272":1,"297":1,"313":1,"360":2,"370":2,"541":1,"552":2,"577":1,"690":1}}],["https",{"0":{"216":1},"2":{"17":3,"75":2,"87":2,"91":2,"155":2,"156":2,"159":2,"214":1,"216":1,"219":1,"220":1,"236":2,"268":2,"269":1,"272":2,"287":1,"304":2,"305":8,"307":2,"341":2,"349":2,"371":2,"377":1,"399":3,"400":4,"452":3,"453":3,"470":2,"474":1,"475":3,"493":4,"508":4,"531":2,"539":2,"540":2,"559":2,"560":2,"577":1,"585":2,"586":8,"592":12,"602":12,"629":8,"630":2,"674":2,"675":3,"676":1,"682":7,"683":3,"690":12,"694":5}}],["html",{"2":{"17":1,"155":2,"156":2}}],["halo",{"2":{"478":1}}],["halting",{"2":{"403":1}}],["hamburger",{"2":{"400":1}}],["handpicking",{"2":{"503":1}}],["handful",{"2":{"263":1}}],["hand",{"2":{"205":1}}],["hands",{"0":{"149":1},"1":{"150":1,"151":1,"152":1}}],["handling",{"2":{"110":1,"213":2,"290":2}}],["handled",{"2":{"384":1}}],["handles",{"2":{"91":1,"213":1,"290":1,"313":1,"379":1}}],["handle",{"2":{"31":1,"351":1,"392":4,"423":1}}],["happen",{"2":{"425":1,"625":1,"626":1}}],["happens",{"2":{"107":1,"625":1}}],["happened",{"2":{"85":1}}],["happening",{"2":{"5":1}}],["happy",{"2":{"47":1,"63":1}}],["having",{"2":{"40":2,"41":1,"84":1,"168":1,"169":1,"181":1,"191":1,"405":2,"575":1}}],["haven",{"2":{"398":2,"561":1}}],["have",{"2":{"4":1,"9":1,"24":3,"27":1,"30":1,"31":5,"32":3,"34":8,"40":4,"41":1,"42":1,"43":1,"45":1,"46":2,"47":1,"63":1,"72":1,"76":1,"79":1,"85":1,"89":1,"90":1,"91":2,"108":2,"119":1,"125":1,"133":1,"135":1,"149":1,"153":4,"167":1,"181":2,"191":1,"199":1,"206":1,"222":1,"233":1,"237":1,"256":1,"265":1,"267":2,"271":1,"272":1,"283":2,"297":1,"302":2,"305":1,"311":1,"313":1,"337":1,"354":1,"355":1,"357":2,"367":1,"374":1,"376":1,"388":1,"403":1,"404":2,"407":1,"413":1,"419":1,"425":2,"442":1,"449":1,"478":1,"487":1,"509":3,"517":1,"518":1,"526":1,"537":2,"540":1,"542":1,"543":1,"552":1,"560":1,"564":1,"572":1,"577":1,"580":1,"586":1,"588":2,"626":1,"629":2,"630":1,"632":1,"634":1,"635":1,"644":1,"649":1,"650":1,"653":1,"655":1,"662":1,"671":1,"679":1,"685":1,"712":1,"726":1,"727":1}}],["hash=580b3dff8a7c716968161d91116a1e171f486298d582874e93714e489c9e6e88",{"2":{"542":2}}],["hash=",{"2":{"542":2}}],["hash=0x79a0a7a1b4936aafe7a37dbfb07a6a9e55c145a4ed6fd54f962649b4b7db8de7",{"2":{"347":1}}],["hash=0x79a0a7a1b4936aafe7a37dbfb07a6a9e55c145a4ed6fd54f962649b4b7db8de7export",{"2":{"347":1}}],["hashtopost",{"2":{"260":8}}],["hashtags",{"2":{"45":1}}],["hashing",{"2":{"108":1}}],["hashes",{"2":{"108":2,"127":4,"206":1}}],["hash",{"0":{"597":1},"2":{"91":6,"108":2,"120":2,"127":10,"134":2,"137":2,"151":2,"159":1,"170":1,"252":2,"260":30,"268":1,"307":1,"326":24,"329":1,"347":4,"348":4,"419":1,"542":6,"559":1,"571":2,"597":1,"601":3,"610":2,"721":3,"727":2}}],["has",{"2":{"32":1,"34":2,"47":1,"50":1,"84":1,"89":1,"91":3,"94":1,"107":2,"108":1,"110":2,"127":3,"166":1,"167":1,"168":1,"195":1,"255":1,"259":1,"263":1,"264":1,"290":1,"297":1,"337":1,"351":1,"352":1,"373":1,"379":1,"385":2,"402":1,"412":1,"417":1,"419":1,"441":1,"442":1,"466":1,"473":1,"478":1,"503":1,"505":1,"523":1,"575":1,"602":1,"620":1,"634":1,"659":1,"670":1,"671":1,"693":1,"700":1,"712":1,"716":1}}],["had",{"2":{"32":1,"161":2,"257":1}}],["hard",{"2":{"671":2}}],["hardforks",{"2":{"624":1,"625":1,"703":1}}],["hardfork",{"0":{"624":1,"626":1},"1":{"625":1,"626":1},"2":{"466":1,"624":2,"625":2,"626":3}}],["hardhat",{"2":{"91":1}}],["hardware",{"0":{"303":1,"479":1,"588":1,"614":1,"640":1,"656":1,"718":1},"2":{"27":1,"60":1,"61":1,"303":1,"479":1,"588":1,"614":1,"640":1,"656":1,"718":1}}],["harassing",{"2":{"11":1}}],["harassment",{"2":{"7":1,"8":1,"16":1}}],["harmfully",{"2":{"509":1}}],["harmful",{"2":{"9":1,"25":1}}],["hermes",{"0":{"628":1,"635":1},"1":{"629":1,"630":1,"631":1},"2":{"627":2,"628":9,"629":4,"630":4,"631":2,"632":1,"636":1}}],["hereby",{"2":{"32":1,"34":4}}],["herein",{"2":{"30":1}}],["here",{"2":{"0":1,"22":1,"52":1,"60":1,"61":1,"64":1,"107":1,"108":1,"109":1,"117":1,"134":1,"135":1,"137":1,"139":1,"225":1,"249":1,"260":3,"264":1,"267":1,"314":1,"317":2,"319":3,"323":1,"326":2,"329":1,"367":1,"372":1,"373":4,"377":1,"392":4,"484":1,"485":1,"494":1,"503":2,"529":1,"545":1,"546":1,"547":1,"548":1,"601":1,"646":2,"647":1,"649":1,"672":1,"711":1,"727":1}}],["hello",{"2":{"317":1}}],["helpful",{"0":{"373":1,"488":1},"1":{"489":1,"490":1,"491":1,"492":1,"493":1,"494":1,"495":1,"496":1,"497":1,"498":1,"499":1,"500":1},"2":{"373":4,"539":1,"540":1}}],["helpcelestia",{"2":{"331":2,"492":1,"493":1,"529":1,"539":1,"637":1}}],["helper",{"2":{"141":1,"155":1,"158":1,"159":1,"373":2}}],["helping",{"2":{"47":1,"255":1,"488":2,"639":1}}],["helps",{"2":{"2":1,"43":1,"46":1,"48":4,"84":1}}],["help",{"0":{"331":1},"2":{"0":1,"33":1,"35":2,"36":1,"39":1,"40":1,"41":1,"42":3,"43":2,"44":1,"45":2,"46":2,"47":3,"48":2,"57":2,"131":1,"232":1,"240":2,"331":4,"375":1,"381":1,"459":1,"488":6,"492":1,"493":1,"529":2,"539":2,"548":1,"568":1,"637":2,"685":1,"706":1}}],["hence",{"2":{"213":1,"290":1,"385":1,"612":1}}],["heightsub",{"0":{"556":1},"2":{"556":4}}],["height=15",{"2":{"159":1}}],["height",{"0":{"326":1},"2":{"107":2,"128":1,"134":8,"135":4,"137":2,"139":6,"148":3,"150":2,"151":16,"152":1,"159":25,"161":8,"166":1,"170":2,"181":2,"185":1,"207":1,"268":4,"317":5,"319":6,"320":3,"326":10,"327":2,"332":3,"334":1,"351":2,"373":20,"374":6,"419":1,"445":1,"488":4,"522":4,"598":1,"600":1,"601":3,"610":2,"721":2,"727":2}}],["hexbytes",{"2":{"157":2,"161":2}}],["hextoaddress",{"2":{"151":2,"159":2,"160":2,"161":2}}],["hexadecimal",{"2":{"121":1,"253":1,"317":1}}],["hex",{"2":{"76":1,"89":1,"155":1,"156":1,"272":1,"317":8,"319":2,"372":2,"373":5,"390":4,"391":4}}],["hea",{"2":{"373":2}}],["health",{"2":{"501":1,"631":4,"679":1}}],["healthcheck",{"2":{"360":2}}],["healthy",{"2":{"7":1,"631":1}}],["heavy",{"0":{"192":1,"193":1,"194":1},"1":{"193":1,"194":1},"2":{"194":1}}],["headerrangefunctionid",{"2":{"377":1}}],["headerparam",{"2":{"373":2}}],["header",{"0":{"326":1},"2":{"89":1,"107":2,"108":1,"127":2,"128":1,"129":1,"134":11,"135":9,"136":1,"137":18,"139":19,"159":2,"172":1,"206":2,"326":8,"327":2,"332":4,"360":2,"372":2,"373":2,"377":2,"378":1,"416":1,"419":1,"424":1,"425":4,"438":1,"556":2,"601":1}}],["headers",{"0":{"143":1,"556":1},"2":{"89":1,"95":1,"107":1,"133":1,"137":1,"139":5,"143":1,"148":1,"201":1,"268":2,"416":1,"417":1,"418":1,"425":5,"478":2,"556":3,"613":2,"655":3,"715":1,"716":1}}],["head",{"2":{"72":1,"131":1,"135":1,"229":1,"304":2,"310":1,"356":1,"367":1,"505":1,"528":1,"585":2,"600":1}}],["heard",{"2":{"229":1}}],["hear",{"2":{"48":2}}],["hesitate",{"2":{"40":1,"45":1}}],["hedging",{"2":{"34":1}}],["vlog",{"2":{"557":4}}],["vghpcybpcybhbiblegftcgxlig9mihnvbwugymxvyibkyxrh",{"2":{"334":2}}],["vv",{"2":{"261":1}}],["vvforge",{"2":{"261":1}}],["vms",{"2":{"233":1}}],["vm",{"2":{"232":1,"233":4,"260":4,"262":4}}],["vs",{"0":{"202":1,"422":1},"1":{"203":1,"204":1,"205":1},"2":{"680":1}}],["v",{"2":{"102":2,"277":1,"348":2,"373":14,"579":11,"580":10}}],["v1beta1",{"2":{"494":4,"522":2,"523":2,"524":2}}],["v1",{"2":{"94":2,"149":8,"324":6,"341":6,"351":1,"384":2,"388":1,"540":1,"628":1}}],["v2",{"2":{"94":1,"102":2}}],["vulnerability",{"2":{"84":1}}],["v0",{"2":{"80":1,"94":3,"149":4,"341":3,"360":4,"441":2,"560":1,"628":1}}],["void",{"2":{"519":2}}],["voice",{"2":{"40":1}}],["vote",{"2":{"426":1,"447":2,"448":1,"494":5,"495":1}}],["voting",{"2":{"326":4,"460":1,"494":1,"696":1}}],["volume",{"2":{"360":1,"578":2,"579":6,"580":2,"583":1}}],["volumes",{"2":{"244":2,"360":2}}],["volunteer",{"2":{"36":1}}],["volunteers",{"2":{"36":1,"38":1,"52":1}}],["vouch",{"2":{"203":1}}],["vest",{"2":{"522":1}}],["vestingcelestia",{"2":{"527":1}}],["vesting",{"0":{"511":1,"518":1,"522":1,"523":1,"529":1},"1":{"512":1,"513":1,"514":1,"515":1,"516":1,"517":1,"518":1,"519":2,"520":2,"521":2,"522":2,"523":2,"524":2,"525":2,"526":1,"527":1,"528":1,"529":1,"530":1,"531":1,"532":1,"533":1},"2":{"278":1,"385":1,"511":4,"517":1,"518":1,"519":5,"520":2,"521":1,"522":19,"523":8,"525":3,"526":2,"527":4,"529":10,"532":1,"533":2}}],["veto",{"2":{"494":2}}],["ve",{"2":{"45":1,"65":1,"76":1,"120":1,"122":1,"223":1,"229":1,"252":1,"254":1,"257":1,"262":1,"264":1,"272":1,"365":1,"525":1,"533":1,"561":1}}],["venue",{"0":{"40":1},"2":{"36":2,"37":1,"40":4,"44":2,"45":2}}],["vendors",{"2":{"31":1}}],["ver=",{"2":{"305":3,"586":3}}],["very",{"2":{"133":1,"283":1}}],["verifiable",{"2":{"206":1}}],["verified",{"2":{"146":1,"161":2,"166":1,"178":1,"181":1,"184":1,"188":1,"192":1,"402":1,"403":1,"404":1,"412":1}}],["verifier",{"2":{"130":1,"131":1,"141":1,"161":2,"189":1,"192":1,"208":1}}],["verifies",{"2":{"117":2,"129":2,"131":3,"206":2,"208":1,"249":2}}],["verification",{"0":{"204":1},"2":{"31":1,"89":1,"107":1,"131":1,"148":1,"159":1,"185":2,"189":1,"192":1,"385":1,"601":1}}],["verifydatarootinclusion",{"2":{"151":4}}],["verifymultirowrootstodataroottupleroot",{"2":{"131":1}}],["verifyrowroottodataroottupleroot",{"2":{"131":2}}],["verifysharestodataroottupleroot",{"2":{"131":2,"161":2}}],["verifying",{"0":{"131":1},"2":{"128":2,"135":2,"136":1,"148":1,"162":1,"166":2,"177":1,"278":1,"279":1,"404":1,"410":1,"418":1}}],["verifyzkp",{"2":{"127":4}}],["verifyattestation",{"2":{"110":1,"127":2,"129":1,"151":2}}],["verify",{"0":{"404":1,"631":1},"2":{"89":2,"127":4,"131":4,"136":1,"151":6,"153":2,"155":6,"158":4,"159":2,"161":10,"162":1,"170":1,"173":1,"176":1,"177":1,"178":1,"180":1,"181":1,"184":1,"185":2,"189":2,"192":3,"232":2,"305":1,"307":1,"402":5,"404":2,"405":1,"406":1,"407":1,"412":1,"415":1,"416":1,"418":1,"419":1,"422":1,"423":2,"519":2,"522":1,"559":1,"586":1,"601":1,"628":1,"632":1}}],["versed",{"2":{"65":1}}],["versioned",{"2":{"671":1}}],["versioncd",{"2":{"516":1}}],["versioncelestia",{"2":{"307":1,"559":1}}],["versiongo",{"2":{"305":1,"586":1}}],["versions",{"0":{"699":1,"702":1,"704":1},"2":{"89":1,"94":1,"149":2,"624":1}}],["version",{"0":{"330":1,"468":1,"516":1,"582":1,"669":1,"687":1},"2":{"17":2,"94":2,"126":1,"155":11,"156":11,"161":6,"213":1,"278":1,"290":1,"305":4,"307":7,"317":1,"319":8,"326":2,"330":1,"334":2,"341":3,"351":1,"354":2,"367":1,"374":1,"384":2,"421":1,"441":1,"442":1,"488":6,"512":1,"516":2,"526":1,"529":1,"539":1,"559":6,"562":1,"586":4,"627":1,"628":1,"634":12}}],["vpn",{"2":{"31":1}}],["vanilla",{"2":{"421":1}}],["vastly",{"2":{"232":1}}],["vars",{"2":{"367":1,"373":3}}],["var",{"2":{"137":2,"368":1,"373":2,"374":6,"515":4}}],["vary",{"2":{"41":1,"385":1,"445":1,"671":1}}],["variant",{"2":{"265":2}}],["variables",{"0":{"266":1,"521":1},"2":{"89":1,"266":1,"342":1,"359":1,"373":2,"377":1,"386":1,"521":1,"527":1,"579":1}}],["variable",{"2":{"75":2,"91":1,"116":2,"117":2,"120":1,"121":1,"154":1,"248":2,"249":2,"252":2,"253":1,"263":1,"264":1,"270":1,"272":2,"315":1,"329":1,"347":1,"348":1,"360":1,"386":1,"439":1,"515":1,"721":1}}],["variety",{"2":{"41":1,"233":1}}],["varies",{"2":{"23":1}}],["various",{"2":{"34":1,"36":1,"47":1,"65":1,"223":1}}],["val",{"2":{"721":2}}],["valconspub",{"2":{"398":2}}],["valcons",{"2":{"398":2}}],["valoperpub",{"2":{"398":2}}],["valoper",{"2":{"398":2,"496":2,"497":4}}],["validao",{"2":{"674":1,"675":1,"676":1}}],["validating",{"2":{"92":1,"103":1,"145":1,"624":1}}],["validation",{"0":{"103":1},"2":{"84":2,"139":2}}],["validatehermes",{"2":{"631":1}}],["validates",{"2":{"421":1,"488":2}}],["validated",{"2":{"27":1}}],["validate",{"2":{"27":1,"181":1,"417":1,"418":1,"478":1,"488":2,"600":1,"631":1,"701":1}}],["validator1",{"2":{"645":1}}],["validatorwalletcreator",{"2":{"98":1,"99":1}}],["validatorutils",{"2":{"98":1,"99":1}}],["validators",{"0":{"648":1},"2":{"19":1,"21":1,"22":1,"24":1,"26":1,"27":1,"29":1,"84":9,"85":1,"108":1,"133":1,"134":1,"206":2,"231":1,"283":1,"326":6,"383":1,"407":1,"439":1,"444":1,"466":1,"494":3,"624":3,"625":2,"626":2,"648":3,"649":1,"652":1,"685":1,"701":2,"703":1}}],["validator",{"0":{"100":1,"430":1,"435":1,"496":1,"498":1,"530":1,"717":1,"719":1,"721":1,"727":1,"728":1},"1":{"101":1,"102":1,"103":1,"718":1,"719":1,"720":2,"721":2,"722":1,"723":1,"724":1,"725":1,"726":1,"727":1,"728":1,"729":1,"730":1},"2":{"19":2,"21":2,"23":2,"24":4,"27":7,"84":2,"86":1,"91":1,"103":1,"108":1,"137":1,"231":1,"309":2,"326":4,"380":1,"411":1,"430":1,"435":1,"444":3,"451":1,"459":2,"465":3,"470":4,"471":2,"478":1,"485":5,"488":2,"496":5,"497":6,"498":3,"500":2,"507":6,"509":15,"517":2,"520":2,"521":1,"525":1,"529":1,"530":2,"536":4,"541":1,"552":1,"574":1,"620":3,"646":1,"651":4,"652":2,"660":5,"661":2,"673":1,"685":1,"696":1,"703":1,"708":1,"709":1,"711":1,"717":2,"718":1,"721":7,"722":2,"727":17}}],["validiums",{"2":{"403":1}}],["validity",{"2":{"184":1,"201":1,"232":3,"380":1}}],["valid",{"2":{"131":2,"151":2,"166":2,"176":1,"177":1,"181":1,"185":2,"189":2,"192":2,"232":1,"416":1,"422":2,"425":1,"631":2}}],["valuable",{"2":{"40":1,"43":2,"48":1,"65":1}}],["valuelog",{"2":{"557":2}}],["values",{"0":{"542":1},"2":{"24":1,"85":1,"155":1,"237":1,"317":1,"377":1,"447":1,"542":2,"554":3}}],["value",{"2":{"4":1,"44":1,"76":3,"89":2,"116":2,"121":1,"155":6,"156":1,"185":1,"248":2,"253":1,"268":15,"272":3,"317":2,"320":1,"324":1,"326":4,"348":2,"349":4,"352":2,"384":1,"385":1,"386":1,"494":4,"547":1,"557":3,"599":2,"610":2,"611":1,"645":2,"646":1}}],["visualize",{"2":{"503":1}}],["visualizer",{"2":{"503":1}}],["visualization",{"0":{"503":1}}],["vision",{"2":{"458":1}}],["visited",{"2":{"398":2}}],["visit",{"2":{"79":1,"83":1,"84":1,"88":1,"90":1,"218":1,"219":1,"220":1,"221":1,"357":2,"575":1}}],["visibility",{"2":{"45":1}}],["visible",{"2":{"7":1}}],["vite",{"2":{"256":1,"265":4,"266":2,"268":4}}],["vitepress",{"2":{"1":1}}],["virtual",{"2":{"214":1}}],["virtually",{"2":{"65":1}}],["vibrant",{"2":{"55":1}}],["video",{"0":{"581":1},"1":{"582":1},"2":{"40":1,"46":1,"60":1,"61":1}}],["videos",{"2":{"37":1,"46":1}}],["viewed",{"2":{"548":1}}],["viewing",{"0":{"346":1},"2":{"494":1}}],["viewstate",{"2":{"268":6}}],["view",{"0":{"240":1,"344":1,"361":1},"2":{"34":2,"69":1,"71":1,"90":1,"121":1,"213":1,"253":1,"260":5,"268":15,"292":1,"331":2,"344":1,"346":1,"361":1,"399":1,"400":1,"488":1,"495":1,"529":1,"540":1,"560":1}}],["viewpoints",{"2":{"8":1}}],["violate",{"2":{"23":1,"34":1}}],["violating",{"2":{"14":1,"15":1,"24":1,"25":1}}],["violation",{"2":{"12":1,"13":1,"14":1,"15":1,"16":1,"31":1}}],["via",{"0":{"141":1,"142":1,"392":1},"2":{"10":1,"45":2,"134":1,"136":1,"139":1,"146":1,"148":1,"152":1,"155":1,"156":1,"159":1,"172":1,"173":1,"189":1,"236":4,"237":1,"379":1,"381":1,"392":2,"424":1,"426":1,"494":1,"537":1,"550":1,"630":1,"636":1,"637":1,"703":1}}],["eofsudo",{"2":{"712":1,"714":1,"715":1,"716":1}}],["eof",{"2":{"712":3,"714":3,"715":3,"716":3}}],["egrep",{"2":{"602":6}}],["egg",{"2":{"519":2}}],["eye",{"2":{"504":1}}],["e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855",{"2":{"326":8}}],["ephemeral",{"2":{"232":1}}],["eclipse",{"2":{"406":1}}],["ecfg",{"2":{"392":4}}],["ec2",{"2":{"378":1}}],["echo",{"2":{"121":1,"253":1,"270":2,"348":2,"592":10,"652":2}}],["ecosystem",{"2":{"8":1,"21":1,"27":3,"43":3,"48":1,"50":1,"53":1,"60":1,"61":1,"65":3,"201":1,"256":1,"288":3,"448":1,"449":2,"450":2,"503":1,"666":1}}],["economic",{"2":{"7":1,"24":1}}],["equally",{"2":{"423":1}}],["equality",{"2":{"260":1}}],["equal",{"2":{"117":1,"249":1}}],["equipments",{"2":{"40":1}}],["equipment",{"2":{"36":2,"37":1,"40":2,"46":1}}],["equivalent",{"2":{"34":2}}],["era",{"2":{"458":1}}],["erasure",{"2":{"146":1,"281":1,"406":1,"407":3,"421":1,"424":1,"438":1,"478":2}}],["err",{"2":{"137":18,"139":12,"150":16,"151":78,"152":4,"159":34,"160":26,"161":30,"373":40,"392":50}}],["errorcodes",{"2":{"161":2}}],["errorf",{"2":{"139":4,"151":2,"159":2}}],["error",{"0":{"557":1},"2":{"93":1,"135":10,"137":4,"139":4,"151":6,"159":2,"161":4,"255":3,"337":3,"351":2,"352":1,"373":16,"392":2,"554":3,"557":6,"649":1}}],["errors",{"2":{"89":1,"350":1,"357":2,"630":1}}],["erc20outbox",{"2":{"98":1,"99":1}}],["erc20rollupeventinbox",{"2":{"98":1,"99":1}}],["erc20inbox",{"2":{"98":1,"99":1}}],["erc20bridge",{"2":{"98":1,"99":1}}],["erc20tokenbridge",{"2":{"91":2}}],["erc20",{"2":{"91":2}}],["eight",{"2":{"419":1}}],["eighteen",{"2":{"31":1}}],["eip",{"2":{"384":1}}],["eip4844",{"2":{"107":1}}],["either",{"2":{"65":1,"102":1,"167":1,"184":1,"236":1,"262":1,"266":1,"310":1,"313":1,"334":1,"341":1,"355":1,"360":1,"384":1,"425":1,"450":1,"503":3,"537":1,"540":1,"560":1,"577":1}}],["es",{"2":{"674":1,"675":1,"676":1}}],["estimates",{"2":{"386":1}}],["estimategas",{"2":{"386":2}}],["estimated",{"2":{"386":1,"392":2}}],["estimate",{"2":{"42":1,"91":8,"386":1,"387":1,"522":2}}],["estimating",{"0":{"385":1,"387":1},"2":{"42":2,"386":1,"392":2}}],["established",{"2":{"634":1}}],["establish",{"2":{"42":1,"88":2,"627":1}}],["especially",{"2":{"40":1,"50":1,"394":1}}],["essentially",{"2":{"548":1}}],["essential",{"2":{"40":1,"45":1,"47":1,"53":1,"86":1,"88":1,"215":1,"304":5,"451":1,"457":1,"541":1,"552":1,"585":5}}],["efficient",{"2":{"232":1,"387":1,"423":1}}],["efficiently",{"2":{"165":1,"402":1}}],["effect",{"2":{"417":1}}],["effects",{"2":{"161":2}}],["effectivegasprice",{"2":{"121":2,"253":2}}],["effectively",{"2":{"45":1,"57":1,"602":1}}],["effective",{"2":{"44":1,"48":1,"671":1}}],["effected",{"2":{"34":1}}],["efforts",{"2":{"34":2,"47":1}}],["eu",{"2":{"34":2}}],["eth0",{"2":{"608":2}}],["ethers",{"2":{"266":2,"268":12}}],["ethermintd",{"2":{"264":2,"270":2}}],["ethermint",{"0":{"264":1,"269":1,"270":1},"2":{"257":1,"263":1,"264":4,"267":15,"268":4,"269":1,"270":7,"271":1}}],["ethereumclient",{"2":{"135":6,"137":4,"139":2}}],["ethereum",{"0":{"110":1,"143":1,"200":1,"255":1,"350":1},"1":{"201":1,"202":1,"203":1,"204":1,"205":1,"206":1,"207":1,"208":1,"209":1,"210":1,"211":1,"351":1,"352":1},"2":{"39":1,"76":1,"80":3,"82":1,"83":1,"89":1,"101":2,"105":1,"106":1,"109":2,"110":4,"114":1,"121":1,"129":1,"133":2,"134":4,"135":12,"137":3,"139":4,"143":3,"151":12,"161":12,"201":3,"204":1,"205":2,"206":8,"207":2,"210":2,"213":3,"214":1,"218":1,"246":1,"252":1,"253":1,"255":6,"256":1,"268":8,"269":1,"272":1,"284":1,"285":3,"288":1,"290":3,"350":2,"351":6,"352":3,"376":1,"384":1,"458":1}}],["ethcmn",{"2":{"151":2,"159":2,"160":2,"161":2}}],["ethclient",{"2":{"151":10,"159":8,"160":8,"161":8}}],["eths",{"2":{"91":2}}],["eth",{"0":{"82":1},"2":{"69":1,"76":2,"80":2,"82":5,"84":1,"85":2,"88":1,"89":6,"91":7,"93":1,"246":1,"264":2,"270":2,"272":2,"343":1,"351":2,"458":1}}],["ethnicity",{"2":{"7":1}}],["etc",{"2":{"33":1,"36":2,"38":1,"48":1,"146":1,"172":1,"265":1,"312":2,"471":1,"557":2,"679":1,"688":1,"712":4,"714":4,"715":4,"716":4}}],["else",{"2":{"151":2,"398":2,"630":1}}],["elastic",{"2":{"39":1}}],["electronic",{"2":{"34":1}}],["element",{"2":{"30":1}}],["eligible",{"2":{"30":1,"31":1}}],["eligibility",{"0":{"24":1},"2":{"21":1,"23":1,"31":1,"33":1,"34":2}}],["earlier",{"2":{"370":1}}],["early",{"2":{"201":1,"449":4,"450":2,"519":2}}],["earn",{"2":{"30":1,"31":1,"444":1,"509":1}}],["easiest",{"2":{"575":2}}],["easier",{"2":{"48":1}}],["easily",{"2":{"284":1,"289":1}}],["ease",{"0":{"231":1}}],["easy2stake",{"2":{"674":1,"675":1}}],["easy",{"2":{"47":1,"213":1,"231":1,"265":1,"290":1,"423":1,"458":1}}],["eager",{"2":{"48":1}}],["each",{"2":{"26":1,"28":1,"42":1,"52":1,"65":1,"89":1,"108":1,"116":1,"128":1,"134":1,"136":1,"139":1,"159":2,"169":1,"248":1,"278":1,"384":1,"385":3,"386":2,"387":2,"388":2,"416":1,"417":1,"423":1,"424":2,"425":1,"438":1,"439":1,"442":1,"445":1,"469":1,"509":1,"531":1,"553":1,"630":2,"671":1,"672":1,"685":2,"696":2}}],["evolution",{"2":{"421":1,"666":1}}],["evolves",{"2":{"288":1}}],["evolved",{"2":{"256":1}}],["evidence",{"2":{"278":1,"326":2}}],["evm",{"2":{"79":1,"105":1,"151":4,"159":6,"160":2,"161":4,"163":1,"201":1,"210":1,"212":2,"214":2,"227":1,"233":1,"257":1,"258":2,"264":3,"350":1,"364":1,"381":1,"500":6,"626":3}}],["evaluated",{"2":{"26":1}}],["evaluate",{"2":{"24":1,"34":1,"38":1}}],["ever",{"2":{"34":1,"519":2}}],["everything",{"2":{"63":1,"127":2,"184":1,"599":5,"631":1}}],["everyone",{"2":{"7":1,"41":1,"52":2,"643":1,"645":1,"648":1,"650":1,"685":1}}],["every",{"2":{"2":1,"22":1,"26":1,"29":1,"107":1,"207":2,"232":2,"302":1,"376":2,"416":3,"417":1,"419":3,"422":1,"424":5,"425":1,"445":1,"503":4,"515":1,"527":1,"531":1,"600":1,"643":1}}],["even",{"2":{"31":1,"34":1,"40":2,"351":1,"394":1,"418":1,"422":1,"578":1,"625":1}}],["eventually",{"2":{"388":1,"425":1}}],["event",{"2":{"10":1,"24":1,"31":1,"36":5,"37":2,"38":3,"40":2,"41":2,"42":1,"43":5,"44":11,"45":9,"46":5,"52":2,"56":1,"89":3,"110":1,"151":12,"159":6,"160":6,"255":2,"260":4,"379":1,"611":1,"629":4}}],["eventschan",{"2":{"160":6}}],["eventsiterator",{"2":{"151":8,"159":8}}],["events",{"2":{"5":2,"42":6,"52":1,"54":1,"56":1,"65":1,"88":3,"89":2,"151":4,"159":8,"160":2,"161":4,"260":4,"522":2,"611":2,"624":1}}],["e",{"2":{"10":1,"23":1,"31":1,"33":1,"38":1,"48":2,"128":1,"151":12,"159":13,"172":2,"173":1,"185":1,"243":4,"260":2,"268":8,"290":1,"312":1,"380":1,"388":1,"405":1,"416":3,"417":1,"418":4,"419":4,"421":2,"422":6,"424":4,"425":9,"503":1,"551":1,"577":12,"579":12,"580":12,"592":10,"608":2,"655":1}}],["ed25519",{"2":{"278":1}}],["editing",{"2":{"557":1,"607":1,"631":1}}],["edits",{"2":{"9":1}}],["edit",{"2":{"9":1,"357":1,"531":1,"557":1,"629":2,"635":1}}],["education",{"2":{"7":1,"50":1}}],["embedded",{"2":{"478":1}}],["embark",{"2":{"365":1}}],["emvpw0p8nijmvnocp4bv6k+v6gjmwdxuku=",{"2":{"334":2}}],["emitted",{"2":{"379":1}}],["emit",{"2":{"260":4}}],["emits",{"2":{"151":2,"159":2,"161":2}}],["empty",{"2":{"568":1}}],["employed",{"2":{"509":1}}],["employees",{"2":{"47":1}}],["employ",{"2":{"402":1}}],["empowering",{"2":{"52":1}}],["empower",{"2":{"50":1}}],["emphasizing",{"2":{"40":1}}],["empathy",{"2":{"8":1}}],["emergency",{"2":{"24":1}}],["emails",{"0":{"54":1},"2":{"54":2}}],["email",{"2":{"8":1,"24":1,"27":1,"36":1,"45":2,"47":2,"53":1,"54":1,"63":1}}],["engine",{"2":{"576":1,"578":1}}],["engages",{"2":{"509":1}}],["engaged",{"2":{"42":1}}],["engagement",{"2":{"38":1,"43":1}}],["engage",{"2":{"34":1,"44":1,"45":1,"46":1,"50":1,"451":1}}],["engaging",{"2":{"25":1,"41":1,"43":1,"47":1,"55":1,"63":1,"65":1}}],["enabling",{"2":{"398":2,"423":1,"547":1,"703":1}}],["enabled",{"2":{"353":1,"502":1,"541":1,"552":1,"629":12,"637":1}}],["enables",{"2":{"110":1,"201":1,"218":1,"232":1,"233":1,"255":1,"307":1,"415":1,"417":1,"421":2,"422":1,"423":1,"444":1,"544":1,"547":1,"559":1,"722":1}}],["enablement",{"2":{"31":1}}],["enable",{"0":{"595":1},"2":{"19":1,"89":4,"284":1,"289":1,"297":1,"353":1,"380":1,"398":2,"415":1,"418":1,"421":1,"544":1,"547":2,"594":1,"627":1,"712":3,"714":3,"715":3,"716":3}}],["env",{"2":{"266":1,"268":4,"367":1,"368":1,"370":1,"373":11,"377":1}}],["environment",{"0":{"266":1,"340":1,"584":1},"1":{"585":1,"586":1},"2":{"8":1,"30":1,"52":1,"214":1,"217":1,"234":1,"244":2,"257":1,"263":1,"266":1,"291":1,"304":2,"342":1,"354":1,"359":1,"360":2,"366":1,"373":2,"422":1,"539":1,"547":1,"559":1,"575":1,"584":2,"657":1,"666":1,"721":1}}],["ens",{"2":{"265":1}}],["ensuring",{"2":{"19":1,"45":1,"65":1}}],["ensures",{"2":{"255":1,"402":1,"406":1,"422":1,"611":1}}],["ensure",{"2":{"2":1,"37":1,"42":1,"43":1,"45":1,"46":2,"47":1,"84":1,"86":1,"88":1,"205":1,"255":1,"337":1,"351":1,"422":1,"440":2,"557":1,"588":2,"625":1,"631":1,"650":1,"655":1,"667":1}}],["enough",{"2":{"145":1,"206":1,"416":1}}],["encrypt",{"2":{"491":2}}],["encrypted",{"2":{"238":1,"491":5}}],["encapsulating",{"2":{"86":1}}],["encounter",{"2":{"337":1,"554":1,"557":1,"649":1,"667":1}}],["encouraging",{"2":{"50":1}}],["encouraged",{"2":{"309":1,"554":1,"685":1}}],["encourages",{"2":{"41":1,"44":1,"47":1,"84":1}}],["encourage",{"2":{"37":1,"43":1,"44":1,"45":1}}],["encodetostring",{"2":{"373":2}}],["encode",{"2":{"154":1,"348":1,"373":3,"392":2,"416":1}}],["encoded",{"2":{"134":1,"136":1,"146":1,"167":2,"181":1,"188":1,"189":1,"192":1,"317":9,"319":2,"348":2,"351":1,"390":4,"391":4,"494":1}}],["encoding",{"2":{"134":1,"169":1,"372":4,"392":6,"407":2,"416":2,"418":8,"421":1,"424":1}}],["encompasses",{"2":{"478":1}}],["encompass",{"2":{"47":1}}],["enjoyed",{"2":{"271":1}}],["enjoy",{"2":{"52":1,"65":1,"91":2}}],["enjoyable",{"2":{"41":1}}],["enlightening",{"2":{"50":1}}],["endian",{"2":{"351":1}}],["ending",{"2":{"155":2,"422":1}}],["endkey",{"2":{"155":3}}],["endblock",{"2":{"151":6,"159":4}}],["endpoint",{"0":{"309":1,"548":1,"606":1},"1":{"607":1,"608":1,"609":1},"2":{"89":3,"107":2,"120":1,"145":1,"150":2,"151":4,"152":2,"159":5,"161":4,"251":1,"263":1,"268":4,"309":3,"317":1,"334":1,"469":1,"470":2,"471":1,"478":1,"485":1,"503":4,"544":6,"545":2,"546":2,"547":4,"548":1,"577":1,"601":1,"606":1,"608":1,"620":2,"660":3,"672":1,"679":1,"681":1,"689":1,"690":1,"691":1}}],["endpoints",{"0":{"470":1,"471":1,"674":1,"675":1,"676":1,"677":1,"679":1,"680":1,"681":1,"689":1,"690":1,"691":1,"692":1},"1":{"681":1},"2":{"33":1,"215":1,"280":1,"297":1,"302":1,"308":1,"367":3,"470":6,"471":3,"529":1,"552":1,"601":2,"679":3,"688":2,"690":2,"692":1,"708":1,"724":1}}],["end",{"2":{"44":1,"134":2,"150":1,"152":1,"155":3,"161":4,"170":1,"173":1,"379":1,"419":1,"424":1,"522":2,"523":2,"629":4}}],["enhanced",{"2":{"465":1}}],["enhances",{"2":{"41":1,"44":1}}],["enhance",{"2":{"38":1,"425":1}}],["entails",{"2":{"418":1,"422":4,"423":1}}],["entering",{"2":{"721":1}}],["enter",{"2":{"239":1,"269":1,"431":1,"435":1,"491":2,"503":1,"509":1,"519":2}}],["enters",{"2":{"24":1}}],["entrypoint",{"0":{"267":1},"2":{"208":2,"243":2,"244":2,"267":1}}],["entry",{"2":{"84":2,"217":1}}],["enthusiastic",{"2":{"49":1}}],["enthusiasts",{"2":{"43":1,"50":1}}],["entire",{"2":{"35":1,"108":1,"244":1,"351":1,"406":1,"415":1,"425":1,"588":1,"602":1,"680":1}}],["entirely",{"2":{"34":1,"143":1}}],["entitle",{"2":{"472":1,"693":1}}],["entitles",{"2":{"31":1}}],["entities",{"2":{"203":1,"412":1}}],["entity",{"2":{"27":2,"31":5,"33":1,"34":2,"547":1}}],["enforced",{"2":{"384":1,"583":1}}],["enforce",{"2":{"52":1}}],["enforcement",{"0":{"9":1,"11":1,"12":1},"1":{"13":1,"14":1,"15":1,"16":1},"2":{"11":1,"17":1}}],["enforcing",{"2":{"9":1,"14":1,"15":1}}],["exit",{"2":{"150":6,"151":2,"161":2}}],["existent",{"2":{"136":1}}],["exist",{"2":{"34":1,"196":1,"441":1,"442":1,"452":1,"453":1,"486":1,"621":1,"663":1}}],["exists",{"2":{"34":1,"165":1,"168":1,"196":1,"633":1}}],["existing",{"0":{"244":1},"2":{"23":1,"26":1,"42":1,"47":1,"60":1,"61":1,"89":1,"182":1,"213":1,"233":2,"243":1,"244":2,"260":2,"279":1,"285":1,"290":1,"307":1,"357":1,"388":1,"503":1,"539":1,"554":1,"556":1,"559":1,"626":1,"632":1,"634":1,"635":1,"650":1,"700":1}}],["exhaustive",{"2":{"107":2}}],["extradata",{"2":{"347":2}}],["extra",{"0":{"604":1},"1":{"605":1,"606":1,"607":1,"608":1,"609":1,"610":1,"611":1},"2":{"89":2,"93":1,"173":1,"386":1,"730":1}}],["extends",{"2":{"424":1}}],["extendedheaders",{"2":{"655":1}}],["extended",{"0":{"418":1},"2":{"172":1,"173":1,"332":1,"416":4,"418":6,"424":4,"425":2,"438":1}}],["extension",{"0":{"429":1,"434":1},"2":{"398":2,"522":4}}],["extent",{"2":{"34":1}}],["external",{"0":{"608":1},"2":{"14":1,"76":1,"105":1,"231":1,"272":1,"297":1,"503":4,"608":3}}],["execstart=$",{"2":{"712":2,"714":2,"715":2,"716":2}}],["exec",{"2":{"243":2,"244":2}}],["executable",{"2":{"424":4,"438":2}}],["execution",{"2":{"139":1,"212":1,"213":1,"214":1,"229":2,"232":3,"233":3,"257":1,"283":1,"289":1,"290":1,"415":2,"421":1,"422":7,"423":1,"458":1}}],["executing",{"2":{"91":2,"422":1,"611":1}}],["executor",{"2":{"91":6}}],["executed",{"2":{"422":1,"424":1}}],["execute",{"2":{"57":1,"63":1,"229":1,"232":1,"289":1,"304":1,"380":1,"421":1,"522":1,"553":1,"585":1,"600":1}}],["exercises",{"2":{"135":1}}],["exemptions",{"2":{"34":1}}],["exemption",{"2":{"34":8}}],["exact",{"2":{"387":1,"445":1,"671":1}}],["exactly",{"2":{"26":1,"278":1,"503":2}}],["example",{"0":{"127":1,"151":1,"161":1,"180":1,"332":1},"2":{"66":1,"88":1,"91":1,"127":1,"154":1,"155":1,"156":1,"158":1,"159":4,"161":3,"170":1,"178":1,"180":1,"229":1,"243":1,"244":1,"259":1,"260":1,"271":1,"309":2,"312":1,"314":1,"316":2,"317":4,"319":3,"323":1,"326":1,"329":1,"332":1,"334":2,"360":3,"377":1,"385":1,"419":2,"422":1,"425":1,"442":1,"471":2,"484":1,"485":1,"491":1,"493":2,"522":1,"529":1,"536":1,"541":1,"545":1,"546":1,"552":1,"553":1,"578":1,"579":1,"598":1,"610":1,"611":1,"632":1,"660":1,"715":1,"721":1}}],["examples",{"0":{"321":1},"1":{"322":1,"323":1,"324":1,"325":1,"326":1,"327":1,"328":1,"329":1,"330":1,"331":1},"2":{"8":2,"10":1,"143":1,"320":1,"334":1,"542":1,"547":1}}],["exciting",{"2":{"626":1}}],["except",{"2":{"394":1}}],["exceptions",{"0":{"283":1}}],["exceptional",{"2":{"19":1,"43":1}}],["exclusively",{"2":{"465":1}}],["exclusive",{"2":{"53":1,"65":1}}],["exchange",{"2":{"24":1,"34":3,"85":1}}],["exposes",{"2":{"690":1}}],["expose",{"0":{"607":1}}],["exposed",{"2":{"135":1,"280":1,"295":1,"308":1,"485":1,"620":1,"660":2}}],["exporter",{"0":{"504":1},"2":{"504":1}}],["exported",{"2":{"491":2}}],["exporting",{"0":{"238":1,"491":1}}],["export",{"0":{"499":1},"2":{"75":3,"91":1,"120":2,"238":7,"243":1,"251":2,"252":2,"263":1,"264":4,"268":2,"270":2,"272":3,"305":4,"314":2,"315":1,"316":2,"334":1,"342":2,"348":3,"359":2,"368":1,"398":10,"488":4,"491":3,"499":1,"503":1,"521":2,"527":2,"542":15,"586":4}}],["expand",{"2":{"46":1,"52":1}}],["expiration",{"2":{"34":1}}],["express",{"2":{"44":1}}],["expressed",{"2":{"34":1}}],["expression",{"2":{"7":1}}],["expensive",{"2":{"169":1,"181":1,"183":1,"186":1}}],["expenses",{"2":{"44":1}}],["experiment",{"2":{"639":1}}],["experimentation",{"2":{"233":1}}],["experimentalsuggestchain",{"2":{"398":4}}],["experimental",{"2":{"201":1,"307":2,"559":2,"667":1,"698":1}}],["experimenting",{"2":{"225":1}}],["experiences",{"2":{"8":1}}],["experience",{"2":{"7":2,"8":1,"27":1,"34":1,"38":1,"41":1,"44":1,"52":1,"293":1,"365":1,"698":1}}],["experts",{"2":{"43":2,"65":1}}],["expertise",{"2":{"43":1,"48":1,"65":1}}],["expectrevert",{"2":{"260":2}}],["expects",{"2":{"237":1,"544":1,"583":1}}],["expect",{"2":{"54":1,"65":1,"133":1,"244":1}}],["expectations",{"2":{"42":1}}],["expected",{"2":{"24":1,"36":1,"139":2,"442":2,"523":1}}],["exploring",{"2":{"222":1,"572":1}}],["exploreme",{"2":{"682":1}}],["explored",{"2":{"195":1}}],["explorers",{"0":{"475":1,"682":1,"694":1},"2":{"276":1,"413":1,"475":1,"682":2,"694":1,"727":1}}],["explorer",{"0":{"90":1,"220":1},"2":{"71":1,"90":2,"220":2,"367":2,"399":2,"400":1,"453":1,"475":1,"682":2,"694":1,"721":1,"727":1}}],["explore",{"2":{"5":1,"40":1,"56":1,"57":1,"291":1,"346":1}}],["explains",{"2":{"627":1}}],["explain",{"2":{"256":1}}],["explained",{"2":{"181":1,"186":1,"195":1}}],["explanation",{"2":{"13":1,"405":1}}],["explicit",{"2":{"8":1}}],["pbcopy",{"2":{"270":1}}],["pbcopyprivate",{"2":{"270":1}}],["pnpm",{"2":{"266":1}}],["pwd",{"2":{"244":2}}],["psql",{"2":{"610":2}}],["ps",{"2":{"243":1,"244":1,"346":1}}],["psdocker",{"2":{"243":1,"244":1,"346":1}}],["p",{"2":{"243":2,"268":12,"348":2,"602":6}}],["pkg",{"2":{"151":2,"161":2,"304":4,"392":6,"585":4}}],["pfbs",{"2":{"167":1,"181":1,"183":1,"258":1,"387":1}}],["pfb",{"0":{"335":1,"385":1},"2":{"91":1,"146":5,"148":1,"167":4,"168":1,"169":1,"189":2,"319":3,"375":1,"385":4,"439":1,"671":2}}],["p2p",{"0":{"567":1,"592":1},"1":{"568":1,"569":1},"2":{"89":4,"135":1,"237":9,"238":6,"239":6,"243":4,"244":3,"308":4,"309":8,"310":8,"312":2,"314":4,"316":4,"324":2,"334":2,"360":5,"367":6,"421":1,"470":5,"471":2,"484":4,"485":6,"486":4,"488":2,"541":2,"542":4,"544":2,"546":2,"547":4,"550":4,"551":5,"552":2,"553":6,"554":4,"555":4,"556":4,"577":12,"579":12,"580":6,"592":1,"619":4,"620":2,"621":4,"659":4,"660":6,"661":8,"663":4,"692":1,"715":2}}],["pinnacle",{"2":{"666":1}}],["pinning",{"2":{"442":1}}],["pin",{"2":{"442":1}}],["pings",{"2":{"377":1}}],["pink",{"2":{"146":1}}],["pieces",{"2":{"134":1}}],["pick",{"0":{"83":1,"88":1,"355":1},"1":{"356":1,"357":1,"358":1},"2":{"237":1,"430":1,"536":1,"561":1,"592":1,"727":1}}],["pitches",{"2":{"44":1}}],["pitch",{"2":{"44":2}}],["pizza",{"2":{"41":1}}],["phase",{"2":{"571":1}}],["phrase",{"2":{"519":2,"630":1}}],["photos",{"2":{"37":1,"38":1}}],["physical",{"2":{"8":1}}],["plugin",{"2":{"505":1}}],["plus",{"2":{"82":1,"372":1}}],["plain",{"2":{"317":1,"319":1,"452":1}}],["plaintext",{"2":{"317":1,"349":2,"352":2}}],["play",{"2":{"460":1,"573":1,"653":1}}],["playbook",{"2":{"50":1}}],["playlist",{"2":{"46":1}}],["platform",{"0":{"47":1},"2":{"55":1,"214":1,"377":2,"379":1,"575":1,"666":1}}],["platforms",{"2":{"36":1,"45":1,"46":1,"256":2,"281":1}}],["plastic",{"2":{"41":1}}],["planned",{"2":{"42":1,"45":1}}],["planning",{"2":{"38":1,"48":1,"561":1,"626":1,"709":1}}],["plans",{"2":{"41":1}}],["plan",{"2":{"36":1,"41":1,"42":1,"45":1,"46":1,"47":1,"57":1,"63":1,"709":1}}],["placeholder=",{"2":{"268":4}}],["placement",{"2":{"23":2,"27":1}}],["place",{"2":{"23":10,"44":1,"223":1,"367":1,"503":1,"519":2,"632":1}}],["plethora",{"2":{"504":1}}],["pledge",{"0":{"7":1},"2":{"7":2,"34":1}}],["please",{"2":{"2":1,"4":1,"27":4,"30":3,"36":1,"53":2,"63":1,"89":1,"133":1,"143":1,"201":1,"237":1,"398":2,"556":2,"564":1,"574":1,"575":1,"588":1,"628":2,"630":1,"671":1,"696":1,"728":1}}],["polypore",{"2":{"629":6}}],["polyfills",{"2":{"267":2}}],["polaris",{"2":{"381":1}}],["policies",{"2":{"31":1,"33":1}}],["policy",{"2":{"30":2,"33":3,"629":4,"635":4}}],["political",{"2":{"8":1}}],["popup",{"2":{"431":1}}],["populating",{"2":{"608":1}}],["populates",{"2":{"471":1,"679":1,"688":1}}],["populate",{"2":{"108":1}}],["popularity",{"2":{"42":1}}],["popular",{"2":{"42":1,"45":1}}],["pops",{"2":{"243":2,"244":2,"309":2,"400":2,"485":2,"493":4,"529":2,"531":2,"553":2,"629":2,"660":2,"661":2,"674":1,"675":1,"676":1,"679":1,"688":1,"690":1,"691":1}}],["point",{"2":{"217":1,"237":1,"370":1,"388":1,"422":2,"503":2,"519":2,"674":1,"675":1,"676":1,"689":1,"690":1,"691":1}}],["pointing",{"2":{"170":1}}],["pointer",{"2":{"107":1,"110":1,"134":2,"136":1,"170":1}}],["points",{"0":{"23":1}}],["portid",{"2":{"634":4}}],["portions",{"2":{"405":1,"406":1,"407":1}}],["portion",{"2":{"22":1,"146":1,"304":1,"417":1,"450":1,"459":1,"526":1,"530":1}}],["ports",{"0":{"541":1,"552":1},"2":{"243":4,"244":4,"309":2,"360":2,"367":2,"470":3,"471":4,"484":2,"537":2,"541":7,"550":2,"552":7,"577":2,"603":2,"620":2,"653":2,"660":2,"679":2,"688":2,"724":2}}],["portrays",{"2":{"180":1}}],["port",{"2":{"89":1,"263":3,"268":2,"269":1,"309":5,"334":2,"360":1,"367":7,"471":8,"484":5,"485":1,"503":3,"504":1,"531":2,"541":2,"552":4,"620":1,"629":4,"634":8,"651":4,"652":3,"660":2,"679":18,"689":1,"690":1,"691":1}}],["portal",{"2":{"75":8,"77":8,"78":1,"79":1,"83":3,"84":1,"222":1,"254":1,"272":8,"273":8,"274":1,"286":1}}],["poolcelestia",{"2":{"495":1}}],["pool",{"0":{"448":1,"495":1},"2":{"42":1,"448":2,"460":1,"495":9}}],["potentially",{"2":{"42":1,"46":1,"281":1,"385":1,"386":1}}],["potential",{"2":{"36":1,"42":1,"43":2,"44":2,"55":1,"89":1,"205":1}}],["powerful",{"2":{"501":1}}],["powered",{"2":{"288":1}}],["power",{"2":{"24":1,"326":4,"365":1,"424":1}}],["pos",{"0":{"420":1},"1":{"421":1},"2":{"421":2}}],["possibly",{"2":{"396":1}}],["possible",{"2":{"45":1,"79":1,"106":1,"139":2,"180":1,"232":1,"278":1,"337":1,"416":1,"418":1,"425":1,"440":1,"442":1,"556":1}}],["possibilities",{"2":{"275":1,"289":1,"291":1,"422":1}}],["postgresql",{"2":{"610":1}}],["postdataandgetheight",{"2":{"373":5,"374":2}}],["postcontainerstyle",{"2":{"268":4}}],["postcontent",{"2":{"268":4}}],["postcreated",{"2":{"260":4}}],["postupdated",{"2":{"260":4}}],["postid",{"2":{"260":20}}],["postings",{"2":{"86":1}}],["posting",{"0":{"143":1},"2":{"10":1,"84":3,"85":1,"88":2,"89":1,"107":1,"133":1,"137":1,"213":3,"255":2,"290":3,"352":1,"573":1}}],["poster",{"0":{"88":1},"2":{"84":4,"86":1,"88":2,"89":7,"91":3,"107":3}}],["posted",{"2":{"76":1,"83":3,"84":1,"89":1,"91":2,"107":2,"110":1,"127":1,"131":1,"133":1,"134":2,"135":1,"139":3,"143":1,"146":1,"148":1,"153":4,"166":2,"178":1,"184":1,"185":1,"189":1,"192":1,"213":1,"272":1,"290":1,"301":1}}],["post",{"0":{"335":1},"2":{"40":1,"43":1,"45":1,"78":1,"79":1,"84":1,"89":1,"107":2,"133":1,"137":2,"163":2,"176":1,"177":1,"207":1,"258":1,"260":72,"263":2,"264":3,"268":28,"271":1,"274":1,"334":4,"335":1,"357":1,"374":1}}],["posts",{"0":{"268":1},"2":{"36":1,"38":1,"83":1,"213":1,"260":24,"268":25,"285":1,"290":1,"380":1}}],["positions",{"2":{"173":1}}],["position",{"2":{"34":3,"107":1}}],["positiveerror",{"2":{"554":1}}],["positives",{"2":{"425":1}}],["positive",{"2":{"8":1,"52":1,"288":1,"425":1,"554":3}}],["peerexchange",{"2":{"569":1}}],["peers=$",{"2":{"592":6}}],["peers",{"0":{"569":1},"2":{"568":2,"571":3,"592":33,"601":1,"608":1,"651":3,"692":1}}],["peer",{"0":{"652":1},"2":{"394":1,"442":6,"571":3,"652":3,"728":1}}],["peerid",{"2":{"324":1}}],["pessimistic",{"2":{"380":1}}],["pending",{"2":{"69":1,"71":1}}],["percentage",{"2":{"444":1,"495":1,"509":3}}],["persisted",{"2":{"583":1,"611":1}}],["persistent",{"0":{"578":1,"652":1},"1":{"579":1,"580":1},"2":{"580":1,"592":9,"651":5,"652":1}}],["persist",{"2":{"578":1}}],["perspective",{"2":{"425":1,"478":1}}],["persons",{"2":{"34":1}}],["person",{"2":{"31":3,"34":7,"46":2,"65":1}}],["personal",{"2":{"5":1,"7":1,"8":1,"27":1,"43":1,"44":1}}],["per",{"2":{"43":1,"385":2,"386":2,"387":1,"392":2,"439":1,"445":1,"455":1,"473":2,"693":2}}],["perfect",{"2":{"40":1}}],["perform",{"2":{"139":2,"263":1,"264":1,"422":2,"425":1,"441":1,"654":1,"655":1}}],["performing",{"2":{"121":1,"215":1,"253":1,"417":1,"425":2}}],["performs",{"2":{"117":1,"249":1,"312":1,"424":1}}],["performed",{"2":{"33":1,"282":1,"417":1,"631":2,"657":1}}],["performance",{"2":{"19":1,"21":1,"24":1,"33":1,"38":1,"233":1,"307":1,"465":1,"501":1,"557":1,"559":1,"667":1,"691":1,"698":1}}],["pertain",{"2":{"34":1}}],["permit",{"2":{"34":1}}],["permissionless",{"2":{"415":1,"422":1,"459":1}}],["permissions",{"2":{"295":1,"578":1,"583":2}}],["permissioned",{"2":{"204":1}}],["permission",{"2":{"8":1}}],["permanently",{"2":{"31":1,"440":1,"509":1}}],["permanent",{"0":{"16":1},"2":{"14":1,"15":1,"16":1}}],["periods",{"2":{"24":1,"351":1}}],["period",{"2":{"14":1,"15":2,"23":1,"25":1,"34":5,"84":2,"360":2,"509":3,"634":2}}],["people",{"2":{"8":1,"14":1,"15":1,"40":1,"42":1,"47":1,"48":1}}],["packet",{"2":{"629":8,"635":4}}],["packets",{"2":{"627":3,"628":1,"629":2}}],["packages",{"2":{"304":1,"585":1,"706":1}}],["package",{"0":{"392":1},"2":{"102":1,"150":1,"151":1,"161":1,"372":1,"392":2,"489":1,"503":1}}],["pagination",{"2":{"525":4}}],["pages",{"2":{"73":1,"357":1}}],["page",{"2":{"27":1,"69":1,"85":1,"111":1,"159":1,"221":1,"223":1,"237":1,"243":2,"246":1,"256":1,"272":1,"309":3,"310":1,"356":1,"367":2,"397":1,"424":1,"429":1,"484":3,"485":1,"503":2,"505":1,"509":1,"529":1,"537":1,"539":1,"540":1,"550":1,"575":2,"577":2,"584":1,"603":1,"620":4,"626":1,"653":1,"657":1,"660":4,"685":2,"724":3,"730":1}}],["paired",{"2":{"503":1}}],["paid",{"2":{"85":1,"442":1}}],["padded",{"2":{"424":1}}],["padding",{"2":{"268":6}}],["path=$path",{"2":{"305":4,"586":4}}],["path",{"0":{"515":1},"2":{"236":2,"237":1,"268":2,"313":1,"351":2,"360":1,"486":1,"494":4,"495":2,"515":1,"565":1,"579":6,"580":4,"583":1,"621":1,"663":1}}],["pattern",{"2":{"16":1}}],["payee",{"2":{"629":2}}],["payer",{"2":{"522":2}}],["paying",{"0":{"437":1,"457":1},"1":{"438":1,"439":1},"2":{"458":1}}],["payload",{"2":{"373":4}}],["payforblob",{"2":{"151":6,"159":2,"161":2,"374":1,"485":1,"620":2,"661":1,"665":1}}],["payforblobs",{"0":{"438":1},"2":{"146":1,"255":1,"310":1,"337":1,"357":1,"386":1,"390":2,"424":3,"425":1,"438":3,"457":1,"660":1}}],["pay",{"2":{"85":1,"89":1,"310":1,"373":2,"386":1,"485":1,"620":1,"661":1}}],["payments",{"2":{"84":1}}],["payment",{"2":{"32":1,"278":1,"438":2}}],["paradigm",{"2":{"422":2}}],["parallel",{"2":{"233":1}}],["param",{"2":{"494":2}}],["paramfilter",{"2":{"494":1}}],["params=",{"2":{"398":6}}],["params",{"2":{"89":1,"278":1,"334":2,"398":27,"494":3,"571":1}}],["parameterchangeproposal",{"2":{"494":2}}],["parameters",{"0":{"398":1,"447":1,"670":1},"1":{"671":1},"2":{"91":1,"153":4,"159":2,"312":1,"385":2,"386":1,"397":1,"398":3,"399":1,"400":1,"425":1,"447":2,"460":1,"494":1,"509":1,"670":2}}],["parameter",{"2":{"84":2,"131":1,"385":1,"386":1,"387":1,"494":3,"557":2,"671":2,"727":1}}],["parity",{"2":{"172":1,"416":1,"418":1}}],["parsing",{"2":{"167":1,"168":2,"177":1,"181":1,"192":1}}],["parse",{"2":{"161":2,"167":2,"169":1,"181":1,"183":1,"332":1}}],["parenthash",{"2":{"347":2}}],["parent",{"2":{"84":1,"88":2,"91":18}}],["parts",{"2":{"326":4,"438":1,"460":1}}],["partnering",{"2":{"40":1,"44":1}}],["part",{"2":{"34":1,"53":1,"146":1,"148":2,"149":1,"152":1,"161":6,"172":1,"173":5,"177":1,"181":3,"185":1,"206":1,"231":1,"351":1,"419":2,"422":1,"450":1,"457":1,"458":1,"628":1,"670":1,"709":1,"722":1}}],["partitioned",{"2":{"419":1,"424":1}}],["partitions",{"2":{"302":1,"419":1}}],["particularly",{"2":{"281":1,"402":1}}],["particular",{"2":{"206":1,"373":1}}],["participating",{"2":{"30":1,"31":1,"34":1,"48":1,"52":1,"65":1,"469":1,"500":1,"645":1,"672":1,"685":1,"707":1}}],["participations",{"2":{"34":1}}],["participation",{"2":{"7":1,"19":1,"24":1,"30":2,"31":2,"32":4,"34":4,"38":1,"55":1,"84":1}}],["participates",{"2":{"696":1}}],["participate",{"0":{"697":1},"1":{"698":1,"699":1,"700":1,"701":1,"702":1,"703":1,"704":1,"705":1},"2":{"30":3,"31":5,"33":1,"34":2,"44":1,"53":1,"65":1,"84":1,"380":1,"423":1,"470":1,"643":1,"672":1,"685":1,"696":1,"700":1,"701":1,"717":1,"727":1}}],["participant",{"2":{"24":1,"54":1}}],["participants",{"2":{"23":1,"31":1,"46":1,"48":3,"52":1,"65":1,"648":1,"650":4,"651":2,"652":1}}],["parties",{"2":{"33":1,"176":1,"412":1}}],["party",{"2":{"31":10,"33":4,"276":1,"412":1,"602":1}}],["paper",{"2":{"34":1,"416":1,"419":1,"425":1}}],["passphrase",{"2":{"491":2,"721":3}}],["passport",{"2":{"33":1}}],["password",{"2":{"491":3,"503":1,"519":2}}],["passes",{"2":{"313":1,"424":1}}],["passed",{"2":{"117":2,"249":2,"320":1,"373":1,"486":1,"488":2,"620":1,"621":1,"660":1,"663":1}}],["pass",{"2":{"117":4,"131":1,"249":4,"313":1,"398":2,"494":1}}],["passionate",{"2":{"65":1}}],["pasted",{"2":{"370":1}}],["paste",{"2":{"89":1,"133":1,"367":1}}],["past",{"2":{"5":1,"44":1,"408":1,"409":3,"410":1,"412":1,"413":2,"522":1,"598":1,"644":1}}],["punish",{"2":{"509":1}}],["put",{"2":{"438":1}}],["pub",{"2":{"326":4,"398":2,"523":2,"524":2}}],["pubkey=$",{"2":{"500":2,"727":2}}],["pubkeyed25519",{"2":{"326":4}}],["pubkey",{"2":{"244":4,"519":4,"520":8,"524":2}}],["pubs",{"2":{"42":1}}],["publish",{"2":{"414":1,"438":1,"502":1}}],["publishes",{"2":{"289":1}}],["published",{"2":{"107":1,"130":1,"148":1,"151":2,"162":1,"204":1,"228":1,"229":1,"260":14,"268":4,"351":2,"402":2,"440":2,"442":1}}],["publishing",{"2":{"8":1,"503":3}}],["publicprovider",{"2":{"267":4}}],["publicly",{"2":{"34":1,"65":1,"607":1}}],["public",{"0":{"309":1},"2":{"3":1,"8":1,"10":1,"13":1,"15":2,"16":1,"33":1,"34":4,"43":1,"79":1,"85":1,"113":1,"116":8,"117":9,"119":2,"127":6,"161":2,"248":8,"249":9,"260":26,"262":4,"267":2,"288":1,"297":1,"309":1,"367":1,"377":2,"410":1,"441":1,"442":2,"449":1,"450":1,"472":1,"537":2,"606":1,"607":1,"608":1,"648":1,"674":1,"675":1,"676":1,"679":1,"688":1,"689":1,"691":1,"693":1,"727":1}}],["pure",{"2":{"127":2}}],["purchasers",{"2":{"34":1}}],["purchased",{"2":{"8":1}}],["purposes",{"2":{"113":1,"135":1,"309":1,"408":1,"583":1,"685":1}}],["purpose",{"2":{"34":1,"52":1,"213":1,"236":1,"290":1,"412":1,"440":1,"569":1}}],["pursuant",{"2":{"30":1,"31":2,"34":3}}],["pulled",{"2":{"374":1}}],["pull",{"2":{"2":2,"236":1,"554":1,"650":1,"696":1}}],["push",{"2":{"2":2,"127":2,"629":4}}],["prune",{"2":{"441":1,"599":2,"680":1}}],["pruned",{"2":{"396":1,"441":1}}],["pruning",{"0":{"440":1,"441":1,"593":1},"1":{"441":1,"442":1,"594":1,"595":1,"596":1,"597":1,"598":1,"599":1},"2":{"440":1,"441":4,"471":1,"588":2,"598":2,"599":1,"679":1,"680":1}}],["pragma",{"2":{"116":2,"117":2,"127":2,"161":1,"248":2,"249":2,"260":4,"262":2}}],["practices",{"0":{"442":1},"2":{"31":1,"33":1,"60":1}}],["precise",{"2":{"671":1}}],["precompiles",{"2":{"233":1}}],["preserved",{"2":{"554":1}}],["presence",{"2":{"394":1}}],["presenters",{"2":{"43":1}}],["presented",{"2":{"34":1}}],["presenting",{"2":{"43":1}}],["presentation",{"0":{"59":1,"60":1,"61":1,"62":1},"2":{"36":1,"43":1,"44":1,"59":3,"60":3,"61":3,"62":2}}],["presentations",{"2":{"36":1,"37":1,"43":1,"60":1,"61":1,"65":1}}],["presently",{"2":{"34":1}}],["present",{"2":{"34":1,"204":1,"550":1,"554":1}}],["presumes",{"2":{"539":1}}],["predetermined",{"2":{"405":1}}],["prefix",{"2":{"348":2,"351":4,"399":1,"400":1,"629":12}}],["prefixed",{"2":{"317":1,"424":1}}],["preferences",{"2":{"540":1,"560":1}}],["preferred",{"0":{"427":1}}],["prefer",{"2":{"309":1,"367":1,"484":1}}],["preimage",{"0":{"108":1},"2":{"106":1,"108":3}}],["pre",{"0":{"258":1,"540":1,"560":1,"564":1},"2":{"84":1,"264":1,"538":1,"540":3,"560":3}}],["prerequisites",{"0":{"80":1,"101":1,"114":1,"125":1,"145":1,"242":1,"535":1,"576":1},"2":{"234":1}}],["prerequisite",{"2":{"53":1,"234":1,"507":2}}],["preparing",{"2":{"37":1}}],["prepare",{"2":{"36":2,"45":1}}],["previoushash",{"2":{"134":2,"137":2}}],["previous",{"2":{"38":1,"85":1,"110":1,"117":1,"134":2,"206":1,"249":1,"356":1,"408":1,"434":1,"522":1,"526":1,"530":1,"556":1,"649":1,"709":1}}],["previously",{"2":{"31":1,"213":1,"290":1,"537":1,"727":1}}],["prevents",{"2":{"288":1}}],["preventing",{"2":{"255":1}}],["prevent",{"2":{"33":1,"530":1,"531":1,"638":1}}],["pro",{"2":{"682":1}}],["probabilistically",{"2":{"425":1}}],["probability",{"2":{"416":1}}],["probably",{"2":{"229":1}}],["problems",{"2":{"409":1,"411":1}}],["problem",{"0":{"403":1,"409":1},"2":{"403":2,"412":2,"415":1,"422":1,"423":1}}],["problematic",{"2":{"402":1}}],["proxy",{"2":{"377":1}}],["proposal",{"2":{"448":1,"494":14,"495":5}}],["proposalscelestia",{"2":{"494":1}}],["proposals",{"2":{"426":1,"447":2,"460":1,"494":6,"495":3}}],["proposing",{"2":{"424":1}}],["propose",{"2":{"447":1,"509":1}}],["proposer",{"2":{"213":1,"290":1,"326":8,"343":1,"494":4}}],["proposed",{"2":{"34":1,"288":1,"403":1}}],["property",{"2":{"425":1,"455":1}}],["properties",{"2":{"232":1,"478":1}}],["properly",{"2":{"261":1,"571":1}}],["pros",{"0":{"168":1,"174":1,"178":1,"182":1,"186":1,"190":1,"193":1},"2":{"168":1}}],["prospectus",{"2":{"34":1}}],["prospective",{"2":{"21":1}}],["producers",{"2":{"418":2,"425":1}}],["producer",{"2":{"403":2,"424":4}}],["produceblock",{"2":{"137":2}}],["producing",{"2":{"135":2,"505":1,"696":1}}],["product",{"2":{"385":1,"386":2}}],["production",{"2":{"84":2,"231":1,"478":1,"666":1,"698":1}}],["products",{"2":{"31":2,"466":1}}],["provably",{"2":{"440":1}}],["prover",{"0":{"199":1},"2":{"199":1,"207":1}}],["proven",{"2":{"178":1,"185":1,"207":1}}],["proves",{"2":{"155":2,"201":1}}],["proveshares",{"2":{"152":3,"161":2,"173":1}}],["proved",{"2":{"136":1}}],["prove",{"2":{"130":1,"134":3,"137":1,"139":1,"146":1,"148":5,"150":1,"152":1,"153":2,"156":1,"158":1,"162":1,"163":1,"166":5,"172":2,"173":6,"176":1,"181":1,"186":1,"189":1,"206":1,"229":1,"419":1,"424":1}}],["provision",{"2":{"681":1}}],["provisions",{"2":{"34":2,"445":1}}],["proving",{"0":{"141":1,"151":1,"172":1,"173":1,"378":1},"2":{"133":1,"152":1,"162":1,"163":1,"165":1,"167":2,"173":2,"177":4,"377":1,"378":1,"402":1}}],["provided",{"2":{"30":1,"32":1,"33":1,"34":2,"39":1,"43":1,"50":1,"79":1,"91":1,"113":1,"156":1,"172":1,"181":1,"189":2,"235":1,"278":1,"373":3,"419":3,"568":1,"571":1,"601":2,"625":1}}],["provides",{"2":{"30":1,"47":1,"131":1,"214":1,"223":1,"293":1,"295":1,"415":1,"419":1,"421":1,"422":1,"423":1,"442":1,"501":1,"671":1}}],["providers",{"0":{"276":1},"1":{"277":1,"278":1,"279":1,"280":1,"281":1,"282":1,"283":1},"2":{"31":1,"40":1,"231":1,"267":3,"268":4,"276":1,"297":1,"442":2}}],["provider",{"0":{"107":1,"278":1,"358":1},"1":{"279":1,"280":1,"281":1,"282":1,"283":1},"2":{"27":1,"31":1,"40":1,"88":1,"106":1,"107":1,"226":1,"267":4,"268":8,"358":1,"602":1}}],["provide",{"0":{"414":1},"2":{"19":1,"27":1,"30":2,"31":1,"32":1,"33":1,"37":1,"41":1,"46":1,"47":1,"102":1,"109":1,"148":1,"149":1,"150":1,"152":1,"172":1,"173":2,"260":1,"317":1,"360":1,"407":1,"412":2,"413":2,"415":1,"419":2,"442":1,"471":2,"501":1,"522":1,"568":1,"649":1,"679":2,"683":1,"688":2,"696":1,"708":1,"727":1}}],["providing",{"0":{"421":1},"2":{"13":1,"41":1,"50":1,"55":1,"110":1,"212":1,"233":1,"295":1,"334":1,"465":1}}],["procedure",{"2":{"215":1,"311":1}}],["proceeding",{"2":{"85":1,"139":1,"628":1}}],["proceed",{"2":{"71":1,"84":2,"162":1,"189":1,"192":1,"234":1,"573":1,"648":1,"709":1}}],["processing",{"2":{"233":1}}],["processed",{"2":{"228":1,"394":1}}],["processes",{"2":{"31":1,"373":2,"395":1,"413":1,"478":1,"627":1,"710":1,"712":1}}],["process",{"0":{"20":1,"22":1,"29":1,"624":1,"625":1,"710":1},"1":{"21":1,"22":1,"23":1,"24":1,"25":1,"625":1,"626":2,"711":1,"712":1,"713":1,"714":1,"715":1,"716":1},"2":{"22":1,"31":1,"36":1,"47":1,"79":1,"84":1,"110":1,"136":1,"139":3,"160":2,"184":1,"185":1,"206":1,"213":1,"218":1,"228":1,"234":1,"290":1,"365":1,"407":1,"422":1,"478":3,"487":1,"548":1,"553":1,"603":1,"622":1,"624":1,"625":1,"664":1,"670":1,"711":1,"716":1,"726":1}}],["project",{"0":{"259":1},"1":{"260":1,"261":1,"262":1,"263":1},"2":{"115":7,"116":2,"167":1,"247":7,"248":2,"259":1,"265":5,"266":7,"495":2}}],["projector",{"2":{"40":1}}],["projectors",{"2":{"36":1,"40":1}}],["projects",{"2":{"24":1,"44":1,"265":1,"275":1,"293":1}}],["proofnonce",{"2":{"151":4,"159":2,"161":2}}],["proofs",{"0":{"131":1,"144":1,"153":1,"167":1,"192":1,"193":1,"194":1,"418":1},"1":{"145":1,"146":1,"147":1,"148":1,"149":1,"150":1,"151":1,"152":1,"153":1,"154":2,"155":2,"156":2,"157":2,"158":2,"159":2,"160":2,"161":2,"162":1,"193":1,"194":1},"2":{"108":1,"130":1,"131":2,"136":1,"145":1,"146":1,"148":1,"149":1,"152":5,"153":4,"155":8,"158":10,"160":1,"161":15,"162":1,"166":9,"167":5,"169":1,"172":1,"173":5,"174":1,"176":1,"177":3,"178":1,"180":2,"181":4,"183":1,"184":1,"188":1,"189":1,"192":3,"194":1,"201":1,"206":2,"207":1,"208":1,"209":1,"232":1,"376":1,"380":1,"406":1,"407":1,"416":1,"418":2,"419":1,"422":1,"425":1,"613":2,"627":1}}],["proof",{"0":{"130":1,"146":1,"150":1,"152":1,"166":1,"171":1,"379":1,"444":1,"459":1},"1":{"147":1,"148":1},"2":{"33":1,"108":1,"127":12,"128":2,"129":1,"130":1,"131":5,"133":1,"136":5,"139":2,"141":1,"148":2,"149":1,"150":6,"151":14,"152":7,"153":10,"154":1,"155":12,"157":1,"158":14,"159":11,"161":39,"162":2,"166":6,"167":1,"168":1,"171":1,"172":6,"173":11,"177":1,"180":1,"181":3,"185":3,"186":2,"189":3,"192":5,"195":1,"203":1,"206":4,"232":3,"377":1,"378":1,"379":3,"407":2,"416":1,"419":1,"425":1,"426":1,"444":3,"451":1,"459":2,"478":1,"509":1}}],["proto",{"2":{"161":2,"349":7,"352":5,"392":2}}],["protobuf",{"0":{"189":1},"2":{"149":4,"167":2,"168":1,"169":1,"181":2,"188":1,"189":2,"191":1,"192":2}}],["protocol",{"2":{"24":1,"30":2,"43":1,"105":1,"135":1,"206":2,"268":2,"384":1,"444":1,"449":1,"541":1,"547":1,"552":1,"627":1}}],["protection",{"2":{"295":1}}],["protect",{"2":{"25":1,"31":1,"89":1,"295":1}}],["prohibited",{"2":{"24":1,"31":1}}],["prometheuslistenaddr",{"2":{"502":2}}],["prometheus",{"2":{"502":9,"503":17,"504":1,"548":1,"628":1}}],["promptstring",{"2":{"374":4}}],["promptscavenger",{"2":{"371":6}}],["prompt",{"0":{"365":1,"370":1},"1":{"366":1,"367":1,"368":1,"369":1,"370":1,"371":2,"372":2,"373":2,"374":2,"375":1},"2":{"85":1,"298":1,"365":1,"373":1,"374":13}}],["prompted",{"2":{"83":1,"491":2,"727":1}}],["promptly",{"2":{"11":1,"107":1}}],["promise",{"2":{"268":2}}],["promised",{"2":{"31":1}}],["prominent",{"2":{"65":1}}],["promote",{"2":{"42":1,"45":2,"46":3,"65":1,"233":1}}],["promotions",{"2":{"37":1}}],["promotional",{"2":{"36":1,"37":1}}],["promoting",{"2":{"19":1}}],["profileecho",{"2":{"305":1,"586":1}}],["profile",{"2":{"305":3,"586":3}}],["proficient",{"2":{"19":1}}],["professionals",{"2":{"46":1}}],["professional",{"2":{"8":1,"34":1,"442":2}}],["programs",{"2":{"449":1}}],["programmatic",{"2":{"294":1}}],["programmatically",{"0":{"387":1},"2":{"143":1,"392":2}}],["programming",{"2":{"116":1,"162":1,"248":1}}],["program",{"0":{"18":1,"19":1,"21":1,"49":1,"50":1},"1":{"19":1,"20":1,"21":1,"22":1,"23":1,"24":1,"25":1,"26":1,"27":1,"28":1,"29":1,"50":1,"51":1,"52":1,"53":1,"54":1,"55":1,"56":1},"2":{"19":1,"21":2,"23":2,"24":3,"26":1,"27":3,"29":2,"30":11,"31":9,"32":2,"33":4,"34":10,"39":1,"42":1,"43":1,"50":1,"53":2,"54":1,"56":1,"62":3,"65":3}}],["primitive",{"2":{"404":1,"423":1}}],["primitives",{"2":{"402":1}}],["primarily",{"2":{"84":2,"293":1}}],["primary",{"2":{"19":1,"55":1,"110":1}}],["principle",{"2":{"288":1}}],["prints",{"2":{"373":2}}],["printf",{"2":{"373":6}}],["println",{"2":{"150":8,"151":6,"160":2,"161":2,"373":2,"392":6}}],["print",{"2":{"36":1,"488":2}}],["pricing",{"2":{"205":1}}],["priced",{"2":{"383":1}}],["price",{"0":{"320":1},"2":{"89":3,"320":10,"351":1,"384":4,"386":3,"392":2,"439":1,"629":8}}],["prices",{"2":{"8":1,"383":1}}],["prioritised",{"2":{"439":2}}],["prioritized",{"2":{"383":2}}],["priority",{"2":{"326":4}}],["prior",{"2":{"26":1,"34":3,"36":1,"45":1,"507":2,"624":1,"625":1}}],["privacy",{"0":{"33":1},"2":{"11":1,"30":2,"31":1,"33":3,"65":1}}],["privatecelestia",{"2":{"316":1}}],["private",{"0":{"316":1},"2":{"8":2,"13":1,"15":1,"27":1,"40":1,"75":5,"76":5,"84":2,"86":1,"91":2,"113":1,"119":2,"120":7,"121":4,"127":4,"233":1,"238":1,"251":5,"252":7,"253":4,"260":6,"263":6,"264":6,"270":2,"272":10,"316":7,"488":2,"491":4,"507":2,"517":1,"522":6,"556":4}}],["pr",{"2":{"2":1}}],["ntfs",{"2":{"583":1}}],["nw",{"2":{"508":1}}],["n11",{"2":{"419":2}}],["n12",{"2":{"419":2}}],["n14",{"2":{"419":1}}],["n7",{"2":{"419":1}}],["n8",{"2":{"419":1}}],["n2",{"2":{"417":1,"418":1,"419":1}}],["ns",{"2":{"392":6}}],["nsoy0rgl7hqt4vwlg441gqkjsz2fbunzxipgns8ov",{"2":{"244":2}}],["n",{"2":{"373":2,"409":1,"417":1,"418":2,"499":2,"522":2,"541":1,"552":1,"592":10,"727":2}}],["ncdu",{"2":{"304":4,"585":4}}],["nft",{"2":{"275":1,"408":1}}],["nfts",{"2":{"43":1}}],["nmts",{"0":{"419":1},"2":{"302":1,"415":2,"419":2,"424":1}}],["nmtproof",{"2":{"155":2,"161":2}}],["nmt",{"2":{"148":2,"152":2,"153":2,"372":2,"419":2}}],["npm",{"2":{"115":3,"247":3,"266":1}}],["nidstring",{"2":{"373":4}}],["nil",{"2":{"137":10,"139":6,"150":6,"151":36,"152":2,"159":14,"160":12,"161":12,"373":14,"392":16}}],["nitro",{"0":{"110":1},"2":{"90":1,"94":4,"102":4,"105":1,"106":1,"107":3,"108":2,"110":2,"113":1,"114":1,"119":2,"255":1}}],["nice",{"2":{"40":1,"267":1}}],["null",{"2":{"268":4,"522":4,"523":2,"525":4,"610":1,"712":2,"714":2,"715":2,"716":2,"721":2,"727":2}}],["numia",{"2":{"674":1,"675":1,"676":1,"679":1,"688":1,"689":1,"691":1}}],["numerator",{"2":{"629":4}}],["numerous",{"2":{"403":1,"422":1}}],["num",{"2":{"629":4}}],["numleaves",{"2":{"158":3,"173":2}}],["number=26656",{"2":{"652":1}}],["number=26656ip",{"2":{"652":1}}],["numbers",{"0":{"468":1,"669":1,"687":1}}],["number++",{"2":{"116":2,"248":2}}],["number",{"2":{"36":1,"42":1,"76":1,"84":2,"91":8,"116":7,"117":9,"121":3,"158":2,"163":1,"170":1,"172":2,"201":1,"232":1,"248":7,"249":9,"253":3,"272":1,"347":2,"385":3,"386":1,"388":1,"392":2,"406":1,"417":1,"423":2,"425":2,"502":4,"504":1,"523":2,"524":2,"557":1,"625":2,"652":2,"671":1}}],["nutshell",{"2":{"148":1}}],["nurturing",{"2":{"53":1}}],["necessitate",{"2":{"548":1}}],["necessity",{"2":{"547":1}}],["necessary",{"0":{"407":1},"2":{"25":1,"33":1,"36":1,"40":3,"86":1,"88":1,"107":2,"108":1,"153":4,"159":2,"172":1,"181":1,"204":1,"232":1,"304":1,"373":1,"394":1,"406":1,"421":1,"425":1,"585":1,"608":1,"629":1,"727":2}}],["net",{"2":{"309":1,"519":2,"661":1,"674":1,"675":1,"676":1,"679":1}}],["netcelestia",{"2":{"309":1,"661":1}}],["network=$network",{"2":{"577":6,"579":6,"580":6}}],["network=arabica",{"2":{"577":1}}],["network=arabicaexport",{"2":{"577":1}}],["network=celestia",{"2":{"577":1}}],["network=celestiaexport",{"2":{"577":1}}],["network=custom",{"2":{"542":2}}],["network=",{"2":{"360":4,"367":2,"542":2}}],["network=mochaexport",{"2":{"577":1}}],["network=mocha",{"2":{"243":2,"577":1}}],["networking",{"2":{"37":1,"41":1,"48":2}}],["networks",{"0":{"542":1,"592":1,"697":1},"1":{"698":1,"699":1,"700":1,"701":1,"702":1,"703":1,"704":1,"705":1},"2":{"27":1,"84":1,"135":1,"256":1,"397":1,"415":1,"459":1,"500":1,"550":1,"551":1,"592":17,"601":1,"624":1,"627":1,"632":1,"634":1,"652":1,"655":1,"662":1,"679":1,"685":1,"696":1,"709":1,"727":1,"728":1}}],["network",{"0":{"316":1,"398":1,"430":1,"447":1,"466":1,"467":1,"476":1,"500":1,"550":1,"605":1,"639":1,"653":1,"667":1,"668":1,"670":1,"684":1,"686":1,"695":1,"705":1},"1":{"468":1,"551":1,"640":1,"641":1,"642":1,"643":1,"644":1,"645":1,"646":1,"647":1,"648":1,"649":1,"650":1,"651":1,"652":1,"653":1,"671":1},"2":{"19":2,"25":1,"31":1,"43":1,"44":1,"79":1,"83":2,"84":2,"88":3,"89":4,"91":6,"102":2,"135":1,"139":2,"149":2,"153":2,"206":1,"210":1,"229":1,"232":1,"237":20,"238":12,"239":12,"243":3,"244":3,"262":1,"267":4,"268":2,"269":3,"276":1,"284":1,"289":1,"297":1,"301":1,"305":1,"307":2,"308":4,"309":8,"310":13,"314":8,"316":8,"317":2,"334":3,"342":1,"343":1,"345":1,"357":1,"359":1,"360":12,"367":7,"380":1,"394":2,"397":1,"398":4,"402":1,"411":1,"416":1,"417":1,"421":1,"422":3,"423":2,"425":4,"426":1,"430":1,"434":1,"441":2,"442":3,"444":1,"447":1,"451":3,"457":2,"459":2,"460":1,"469":1,"471":3,"476":1,"478":5,"484":4,"485":9,"486":4,"487":1,"500":1,"508":1,"509":1,"510":1,"512":1,"517":1,"531":2,"532":1,"539":2,"542":8,"544":4,"546":2,"547":8,"550":10,"551":6,"553":6,"554":8,"555":6,"556":10,"559":1,"568":1,"577":7,"579":6,"586":1,"592":1,"601":1,"607":2,"619":4,"620":5,"621":4,"624":4,"625":3,"626":1,"639":1,"645":1,"649":1,"654":1,"659":4,"660":9,"661":12,"663":4,"666":2,"667":2,"670":1,"672":1,"679":2,"684":1,"685":2,"688":1,"689":1,"692":1,"695":1,"696":4,"698":2,"703":1,"705":1,"717":1,"722":2,"726":1,"727":1}}],["nearby",{"2":{"45":1}}],["negatives",{"2":{"425":1}}],["negatively",{"2":{"27":1,"44":1}}],["negotiate",{"2":{"40":1}}],["nextheaderfunctionid",{"2":{"377":1}}],["nextheight",{"2":{"139":4}}],["next",{"0":{"111":1,"122":1,"222":1,"254":1,"275":1,"291":1,"375":1,"517":1,"561":1},"2":{"38":1,"76":2,"77":1,"83":2,"84":1,"88":1,"92":1,"107":1,"111":1,"115":1,"122":2,"131":1,"139":2,"151":2,"159":2,"169":1,"209":1,"243":1,"247":1,"254":1,"260":1,"263":2,"265":1,"266":1,"267":1,"270":2,"272":3,"273":1,"310":1,"317":1,"326":2,"341":1,"367":2,"373":2,"375":1,"377":1,"423":1,"424":2,"500":1,"503":1,"514":1,"525":6,"553":1,"646":1,"709":1}}],["needing",{"2":{"178":1,"263":1,"404":1,"415":1,"442":1,"522":1,"530":1,"531":1}}],["needs",{"2":{"44":1,"47":1,"48":1,"166":1,"181":1,"208":1,"232":1,"256":1,"365":1,"409":1,"419":1}}],["needed",{"2":{"36":1,"37":1,"40":1,"43":1,"86":1,"89":1,"107":1,"134":1,"153":2,"159":2,"173":2,"304":1,"366":1,"369":1,"373":2,"385":1,"386":1,"406":1,"407":1,"408":1,"421":1,"503":1,"506":1,"566":1,"571":1,"626":2,"706":1,"715":1}}],["need",{"2":{"34":1,"47":1,"63":1,"76":2,"77":1,"82":1,"84":1,"86":1,"89":2,"107":2,"108":1,"135":5,"136":1,"145":1,"150":1,"152":1,"153":1,"166":1,"168":1,"173":2,"178":1,"181":2,"183":1,"185":1,"188":2,"191":1,"192":1,"196":1,"231":2,"232":1,"236":1,"237":1,"243":1,"263":1,"264":2,"266":1,"268":4,"269":1,"270":2,"272":3,"273":1,"310":2,"311":1,"312":1,"313":1,"314":1,"316":1,"334":1,"354":1,"357":1,"360":2,"370":1,"372":1,"373":3,"375":1,"380":1,"398":2,"407":1,"417":1,"418":1,"442":1,"458":1,"471":1,"485":2,"503":4,"507":2,"518":1,"527":1,"529":2,"532":1,"535":1,"541":2,"544":1,"547":1,"551":1,"552":2,"553":1,"557":1,"568":1,"569":1,"578":2,"579":1,"594":1,"607":1,"608":1,"610":1,"620":2,"624":2,"626":3,"630":2,"634":1,"635":1,"640":1,"641":1,"642":1,"646":1,"648":1,"649":1,"660":1,"661":1,"679":1,"688":1,"703":1,"709":1,"721":1,"724":1}}],["neither",{"2":{"31":2,"421":1}}],["newmetric",{"2":{"674":1,"675":1,"676":1,"679":1,"688":1,"689":1,"691":1}}],["newblob",{"2":{"392":2}}],["newblobstreamxfilterer",{"2":{"160":2}}],["newblobstreamx",{"2":{"151":2,"159":2}}],["newcredentials",{"2":{"392":2}}],["newclient",{"2":{"373":4}}],["newowner",{"2":{"260":4}}],["newer",{"2":{"256":1,"494":1}}],["newwrappers",{"2":{"161":2}}],["newint",{"2":{"151":8,"155":4,"158":4,"159":8,"161":16,"374":2}}],["newnumber",{"2":{"116":4,"248":4}}],["newly",{"2":{"91":1,"403":1,"635":1}}],["newsletters",{"2":{"38":1}}],["newspaper",{"2":{"34":1}}],["new",{"0":{"160":1,"196":1,"197":1,"198":1,"458":1,"500":1,"519":1,"634":1,"646":1,"649":1},"1":{"197":1,"198":1,"199":1},"2":{"2":2,"23":2,"24":2,"26":1,"42":1,"43":1,"47":3,"48":1,"75":1,"88":1,"115":1,"116":1,"117":3,"134":1,"150":2,"151":4,"159":2,"160":3,"161":2,"196":1,"197":2,"198":1,"207":1,"231":2,"232":2,"233":1,"247":1,"248":1,"249":3,"256":1,"259":1,"260":7,"262":2,"264":2,"265":1,"266":1,"267":1,"268":8,"269":1,"271":1,"272":1,"288":1,"289":1,"346":1,"357":1,"365":1,"371":1,"373":4,"388":1,"394":1,"399":3,"400":1,"402":2,"403":1,"404":2,"410":1,"423":1,"429":1,"440":1,"442":2,"490":2,"503":1,"509":1,"517":1,"527":1,"553":1,"554":2,"568":1,"592":1,"609":1,"610":1,"611":1,"624":1,"625":2,"626":5,"627":1,"632":3,"633":1,"634":3,"635":1,"639":2,"644":1,"646":1,"649":1,"650":1,"653":1,"655":1,"701":1,"707":1}}],["navigate",{"2":{"88":1,"429":1,"434":1}}],["name=$",{"2":{"602":6}}],["name=",{"2":{"507":2}}],["name=validator",{"2":{"500":2,"646":2}}],["name=validator1",{"2":{"500":2,"645":2}}],["namekey",{"2":{"500":1}}],["namevalidator",{"2":{"500":1}}],["names",{"2":{"243":2,"244":2}}],["namespa",{"2":{"373":2}}],["namespace=000008e5f679bf7116cb",{"2":{"360":2}}],["namespaced",{"0":{"419":1},"2":{"302":1,"415":1,"419":1,"421":1,"424":3}}],["namespaces",{"0":{"302":1},"2":{"172":1,"302":1,"419":3,"424":1,"441":1,"442":1}}],["namespaceid",{"2":{"156":5,"161":8,"373":10,"374":6}}],["namespacenode",{"2":{"153":2,"155":12,"157":4,"161":10}}],["namespacemerklemultiproof",{"2":{"153":2,"155":10,"161":6}}],["namespace",{"0":{"156":1},"2":{"89":8,"91":1,"107":2,"134":8,"137":2,"148":2,"152":2,"153":6,"155":31,"156":21,"161":16,"173":2,"213":1,"299":1,"317":7,"319":12,"327":2,"332":3,"334":2,"370":3,"372":2,"373":11,"374":3,"390":2,"391":2,"392":6,"419":6,"424":5,"438":3,"442":1,"502":4}}],["namespce",{"2":{"137":2}}],["named",{"2":{"116":1,"248":1,"260":1,"265":1,"266":1,"357":1,"539":1,"540":1,"560":1}}],["name",{"2":{"27":1,"36":1,"69":1,"84":2,"232":1,"237":7,"238":6,"239":6,"243":2,"244":2,"260":18,"264":2,"265":2,"267":4,"269":1,"310":8,"399":1,"400":1,"486":7,"490":8,"491":4,"500":8,"503":6,"519":2,"520":4,"536":2,"542":2,"578":2,"592":7,"602":6,"621":7,"629":4,"645":4,"646":4,"647":4,"649":3,"661":2,"663":7,"721":2,"727":2}}],["naming",{"2":{"2":1,"47":3}}],["natural",{"2":{"412":1}}],["nature",{"2":{"13":1,"34":1,"205":1,"411":1,"415":1,"495":1}}],["nativecurrency",{"2":{"267":2}}],["natively",{"2":{"84":1}}],["native",{"2":{"80":1,"82":1,"91":6,"93":1,"106":1,"399":1,"426":1,"457":1,"463":1,"630":1,"637":1,"638":1}}],["nationality",{"2":{"7":1}}],["natnet",{"2":{"63":1}}],["nat",{"2":{"36":1,"608":1}}],["novel",{"2":{"415":1}}],["none",{"2":{"175":1,"179":1,"187":1,"190":1,"193":1,"634":4}}],["nonce",{"2":{"127":4,"129":1,"151":4,"153":4,"159":12,"161":6,"255":1,"347":2,"348":2,"388":2,"392":2}}],["non",{"2":{"34":1,"107":2,"136":1,"447":1,"522":2,"638":1}}],["now",{"0":{"271":1},"2":{"34":1,"76":1,"78":1,"85":1,"89":2,"92":1,"107":1,"117":2,"120":2,"121":1,"161":2,"199":1,"213":1,"222":1,"243":1,"244":1,"249":2,"251":2,"252":1,"253":1,"254":1,"257":1,"261":1,"262":2,"263":1,"264":2,"265":1,"266":1,"268":1,"270":4,"271":1,"272":2,"274":1,"290":1,"291":1,"308":1,"309":1,"326":1,"347":1,"348":2,"349":1,"360":1,"367":1,"368":1,"370":1,"373":2,"374":2,"398":1,"399":1,"400":1,"411":1,"441":1,"503":1,"517":2,"525":1,"561":1,"572":1,"577":1,"578":1,"580":1,"592":1,"622":1,"629":1,"630":1,"634":1,"635":1,"650":1,"652":1,"653":1,"679":1,"715":1,"716":1,"727":3}}],["nominee",{"2":{"34":1}}],["normal",{"2":{"139":3,"162":1,"385":1,"402":1,"601":1}}],["normally",{"2":{"40":1}}],["nor",{"2":{"31":2,"34":1,"421":1}}],["no",{"0":{"413":1},"2":{"14":1,"15":1,"23":1,"30":1,"31":5,"34":8,"40":1,"117":2,"120":2,"126":2,"169":1,"184":1,"213":1,"231":1,"249":2,"252":2,"268":2,"297":1,"320":1,"384":1,"402":1,"418":1,"425":1,"458":1,"463":1,"471":1,"472":1,"494":5,"505":1,"551":2,"571":1,"679":1,"693":1}}],["notably",{"2":{"385":1}}],["notable",{"0":{"283":1},"2":{"406":1}}],["notfound",{"2":{"337":2}}],["notify",{"2":{"655":1}}],["notification",{"2":{"72":1}}],["notice",{"2":{"153":4,"155":6,"156":2,"158":2,"159":4,"317":1,"421":1,"523":1,"524":1,"696":1}}],["nothing",{"2":{"34":1,"598":3}}],["noted",{"2":{"283":1,"643":1}}],["note",{"0":{"608":1},"2":{"27":1,"30":1,"53":1,"89":2,"102":1,"103":1,"108":1,"126":1,"134":1,"135":3,"137":1,"139":2,"149":1,"152":1,"162":1,"166":1,"170":1,"172":1,"201":1,"207":1,"208":1,"237":1,"256":1,"263":2,"264":2,"309":1,"353":1,"365":1,"367":1,"378":1,"385":2,"392":2,"419":1,"422":1,"424":2,"425":1,"473":1,"485":1,"494":1,"503":1,"507":2,"511":1,"515":1,"522":1,"530":1,"535":1,"544":1,"550":2,"557":1,"574":1,"598":1,"620":1,"626":1,"630":1,"649":1,"652":1,"671":1,"693":1,"709":1,"715":1}}],["notes",{"0":{"278":1,"396":1,"532":1},"1":{"279":1,"280":1,"281":1,"282":1,"283":1},"2":{"5":1,"54":1,"441":1}}],["not",{"2":{"8":1,"9":1,"23":1,"24":6,"27":1,"30":6,"31":11,"32":3,"33":3,"34":24,"40":1,"41":2,"65":1,"84":2,"88":2,"89":1,"91":2,"94":1,"108":1,"110":1,"126":1,"133":2,"136":1,"137":1,"151":2,"161":2,"166":1,"171":1,"172":1,"176":1,"177":1,"178":1,"181":2,"191":1,"196":1,"206":1,"208":1,"210":1,"237":2,"243":1,"255":1,"256":1,"264":1,"280":1,"281":1,"283":1,"289":1,"302":1,"305":1,"309":1,"310":1,"311":1,"314":3,"320":1,"337":2,"351":1,"357":1,"360":1,"367":1,"380":1,"384":3,"385":1,"386":1,"388":3,"392":2,"394":3,"395":1,"396":1,"406":1,"407":1,"412":1,"418":2,"424":1,"425":2,"440":1,"441":2,"442":2,"447":1,"465":1,"471":2,"472":1,"484":1,"485":1,"494":1,"503":1,"509":1,"515":1,"532":1,"535":1,"542":1,"550":1,"551":1,"554":1,"571":2,"583":1,"586":1,"588":1,"599":1,"608":1,"611":2,"612":1,"620":1,"623":1,"624":1,"625":1,"626":1,"630":1,"660":1,"679":1,"688":1,"690":1,"693":1,"696":1,"701":1,"722":2}}],["noders",{"2":{"674":1,"675":1,"676":1,"689":1,"690":1,"691":1}}],["noderpcip",{"2":{"373":4}}],["nodebuilder",{"2":{"554":6}}],["nodeheader",{"2":{"372":2}}],["nodetype=",{"2":{"367":2}}],["nodeconfig",{"2":{"86":1,"87":1,"88":1,"89":1,"102":2}}],["node",{"0":{"60":1,"89":1,"90":1,"100":1,"102":1,"103":1,"138":1,"142":1,"234":1,"237":1,"238":1,"239":1,"243":1,"292":1,"299":1,"306":1,"307":1,"308":1,"315":1,"324":1,"357":1,"367":2,"368":1,"390":1,"391":2,"392":1,"393":1,"441":1,"477":1,"480":1,"482":1,"483":1,"484":1,"485":1,"486":1,"487":1,"504":1,"530":1,"543":1,"553":1,"558":1,"572":1,"578":1,"579":1,"580":1,"587":1,"589":1,"594":2,"603":1,"609":1,"612":1,"615":1,"617":1,"618":1,"619":1,"620":1,"621":1,"622":1,"623":1,"652":1,"654":1,"657":1,"658":1,"659":1,"660":1,"663":1,"664":1,"692":1,"707":1,"710":1,"714":1,"715":1,"716":1,"717":1,"719":1,"722":1,"723":1,"724":1,"725":1,"726":1,"727":1},"1":{"101":1,"102":1,"103":1,"139":1,"140":1,"141":1,"142":1,"143":1,"235":1,"236":1,"237":1,"238":1,"239":1,"240":1,"241":1,"242":1,"243":1,"244":1,"293":1,"294":1,"295":1,"296":1,"297":1,"298":1,"300":1,"301":1,"302":1,"303":1,"304":1,"305":1,"306":1,"307":2,"308":2,"309":2,"310":2,"311":1,"312":1,"313":1,"314":1,"315":1,"316":2,"317":1,"318":1,"319":1,"320":1,"321":1,"322":1,"323":1,"324":1,"325":1,"326":1,"327":1,"328":1,"329":1,"330":1,"331":1,"332":1,"333":1,"334":1,"335":1,"336":1,"337":1,"478":1,"479":1,"480":1,"481":2,"482":1,"483":2,"484":2,"485":2,"486":3,"487":2,"544":1,"545":1,"546":1,"547":1,"548":1,"559":1,"560":1,"561":1,"562":1,"573":1,"574":1,"579":1,"580":1,"588":1,"589":1,"590":2,"591":2,"592":2,"593":1,"594":1,"595":3,"596":3,"597":1,"598":1,"599":1,"600":1,"601":1,"602":1,"603":1,"604":1,"605":1,"606":1,"607":1,"608":1,"609":1,"610":1,"611":1,"613":1,"614":1,"615":1,"616":2,"617":1,"618":2,"619":3,"620":3,"621":2,"622":2,"623":2,"655":1,"656":1,"657":1,"658":2,"659":1,"660":2,"661":2,"662":2,"663":2,"664":2,"665":1,"711":1,"712":1,"713":1,"714":1,"715":1,"716":1,"718":1,"719":1,"720":2,"721":2,"722":1,"723":2,"724":2,"725":2,"726":3,"727":1,"728":1,"729":1,"730":1},"2":{"3":1,"24":1,"25":1,"27":2,"48":1,"60":6,"80":1,"85":1,"86":1,"88":4,"89":10,"90":1,"91":1,"92":1,"94":3,"102":3,"103":3,"107":6,"108":1,"110":1,"114":2,"135":13,"139":1,"142":1,"145":5,"146":1,"155":16,"161":12,"166":1,"213":2,"232":1,"234":1,"235":2,"236":5,"237":14,"238":6,"239":6,"242":1,"243":15,"244":9,"246":2,"258":2,"265":1,"280":2,"283":1,"290":2,"292":2,"293":1,"294":1,"295":1,"296":2,"299":1,"302":1,"303":1,"304":2,"305":2,"307":17,"308":4,"309":2,"310":9,"311":2,"312":7,"313":6,"314":6,"315":9,"316":7,"317":11,"319":11,"320":2,"322":4,"323":5,"324":4,"325":3,"326":4,"327":4,"328":2,"329":6,"330":5,"331":2,"332":4,"334":7,"336":1,"343":1,"351":2,"355":1,"357":7,"360":4,"365":2,"366":1,"367":10,"368":2,"370":6,"371":1,"372":4,"373":12,"380":1,"384":3,"387":1,"391":1,"392":6,"393":1,"394":1,"402":3,"405":2,"406":4,"408":1,"416":1,"417":2,"419":1,"425":8,"441":3,"442":3,"449":1,"469":3,"470":2,"471":7,"477":1,"478":4,"479":1,"483":3,"484":3,"485":8,"486":2,"487":2,"488":4,"493":12,"499":6,"502":1,"503":1,"504":1,"505":1,"507":4,"508":1,"514":3,"515":2,"526":1,"527":1,"529":3,"530":2,"531":3,"535":1,"537":1,"539":1,"540":1,"541":5,"542":6,"543":4,"544":6,"545":2,"546":2,"547":4,"548":3,"550":5,"551":1,"552":5,"553":26,"554":6,"555":2,"556":6,"557":1,"559":17,"560":5,"561":6,"562":1,"564":1,"565":4,"566":1,"568":1,"571":7,"573":3,"574":3,"575":7,"577":30,"578":5,"579":27,"580":30,"583":1,"585":1,"586":2,"587":1,"588":5,"592":9,"594":3,"598":1,"600":1,"601":1,"602":2,"603":2,"606":1,"608":1,"610":1,"612":3,"614":1,"617":1,"620":11,"621":2,"622":2,"623":2,"629":4,"645":1,"649":1,"652":7,"653":2,"654":1,"656":1,"658":1,"659":2,"660":8,"661":9,"663":2,"664":1,"665":1,"667":1,"672":3,"673":2,"678":3,"679":5,"685":10,"688":4,"689":1,"690":1,"691":1,"692":4,"696":11,"706":1,"707":3,"708":2,"709":4,"711":1,"712":1,"714":2,"715":4,"716":6,"717":1,"718":1,"719":1,"722":5,"723":1,"724":1,"726":2,"727":1,"728":2,"730":1}}],["nodessection",{"2":{"730":1}}],["nodestake",{"2":{"674":1,"675":1,"676":1,"682":1,"694":1}}],["nodes",{"0":{"404":1,"471":1,"478":1,"604":1,"613":1,"655":1,"673":1,"678":1,"688":1,"696":1,"711":1,"713":1},"1":{"605":1,"606":1,"607":1,"608":1,"609":1,"610":1,"611":1,"674":1,"675":1,"676":1,"677":1,"679":1,"680":1,"681":1,"712":1,"714":1,"715":1,"716":1},"2":{"3":2,"60":1,"85":1,"130":1,"134":2,"135":1,"155":8,"158":2,"161":4,"177":1,"181":1,"232":2,"277":1,"293":1,"297":1,"304":1,"308":1,"380":2,"388":1,"402":1,"403":1,"404":2,"405":2,"406":5,"407":5,"409":1,"414":1,"415":1,"416":6,"417":6,"418":3,"419":3,"421":1,"422":3,"423":2,"425":5,"440":1,"441":8,"442":12,"470":3,"471":2,"477":1,"478":4,"503":2,"541":3,"548":1,"552":3,"555":1,"568":1,"569":1,"571":2,"572":1,"584":1,"587":1,"592":2,"606":1,"613":5,"625":1,"626":1,"651":2,"655":3,"672":1,"674":1,"675":1,"676":1,"679":4,"680":3,"685":1,"688":2,"689":1,"692":1,"703":1,"707":4,"708":1,"715":1,"717":1,"722":1}}],["ignore",{"2":{"419":1}}],["icon",{"2":{"400":1}}],["icons",{"2":{"58":1}}],["ixg+08hv5rspf3lle8ph+b2tugsgusbiseflxh6wb5e=",{"2":{"317":2,"319":12,"320":2}}],["ibc",{"0":{"627":1},"1":{"628":1,"629":1,"630":1,"631":1,"632":1,"633":1,"634":1,"635":1,"636":1,"637":1,"638":1},"2":{"627":3,"628":2,"629":4,"634":1,"637":4}}],["ibctransfer",{"2":{"278":1}}],["ibchost",{"2":{"278":1}}],["i++",{"2":{"137":2,"260":2}}],["illustrate",{"2":{"133":1}}],["illustrations",{"2":{"58":1}}],["i",{"2":{"128":1,"137":6,"151":4,"155":4,"157":4,"158":4,"159":5,"161":32,"172":2,"173":1,"185":1,"260":10,"290":1,"380":1,"388":1,"416":3,"417":1,"418":4,"419":3,"421":2,"422":6,"424":3,"425":8,"551":1,"592":10,"607":2,"608":2,"655":1}}],["io",{"2":{"91":2,"155":2,"156":2,"243":2,"268":6,"346":1,"360":4,"372":2,"400":1,"452":1,"475":1,"508":1,"577":6,"579":6,"580":6,"602":12,"674":1,"675":1,"676":1,"682":3,"683":1,"689":1,"690":1,"691":1,"694":3}}],["ip=",{"2":{"367":2,"370":2,"715":2}}],["ip6",{"2":{"324":4}}],["ip4",{"2":{"324":8,"542":4,"715":2}}],["ipfs",{"2":{"258":1,"260":2,"264":1,"266":3,"268":22}}],["ip",{"2":{"89":2,"243":2,"244":2,"269":1,"309":14,"310":6,"334":2,"360":3,"367":7,"370":1,"373":3,"470":1,"471":6,"484":10,"485":6,"486":6,"542":2,"544":2,"545":2,"546":2,"547":4,"550":2,"553":2,"554":2,"577":6,"580":6,"607":2,"620":2,"621":6,"651":4,"660":6,"661":6,"663":6,"679":4,"688":2,"715":2,"716":2,"724":2}}],["irrevocably",{"2":{"31":1}}],["idvalidator",{"2":{"645":1}}],["id=$",{"2":{"652":2}}],["id=testnet",{"2":{"500":2,"645":2}}],["id=",{"2":{"370":2,"507":2,"721":2,"727":2}}],["id=your",{"2":{"266":2}}],["ids",{"2":{"349":2,"542":1,"551":1}}],["idtopost",{"2":{"260":10}}],["idaoracle",{"2":{"127":6,"129":1,"131":1,"161":6}}],["idea",{"2":{"229":1}}],["ideas",{"2":{"41":1}}],["ideally",{"2":{"43":1}}],["ideal",{"2":{"40":1,"214":1}}],["identically",{"2":{"133":1}}],["identify",{"2":{"41":1,"48":1,"107":1,"134":2,"419":1}}],["identifying",{"2":{"31":1,"38":1}}],["identification",{"2":{"33":2}}],["identifiers",{"2":{"419":1}}],["identifier",{"2":{"33":1,"84":1,"116":2,"117":2,"127":2,"248":2,"249":2,"260":4,"262":2,"351":3}}],["identity",{"2":{"7":2,"31":1,"424":1,"438":1}}],["id",{"0":{"324":1,"551":1},"2":{"27":3,"84":2,"89":4,"134":1,"155":19,"156":13,"161":12,"237":1,"243":9,"244":10,"260":12,"266":1,"267":2,"268":8,"269":1,"310":1,"319":1,"322":2,"323":2,"324":6,"325":2,"326":10,"334":4,"346":6,"370":1,"373":9,"374":3,"377":4,"399":1,"424":1,"473":1,"485":1,"493":6,"494":14,"496":4,"497":8,"498":4,"500":8,"503":1,"507":20,"522":6,"529":3,"531":5,"542":5,"551":2,"592":6,"620":1,"626":1,"629":4,"634":40,"645":5,"649":4,"652":4,"660":1,"693":1,"721":1}}],["ifconfig",{"2":{"652":2}}],["if",{"0":{"413":1},"2":{"4":1,"24":2,"25":1,"26":1,"27":5,"30":1,"31":4,"33":1,"34":3,"36":2,"37":1,"40":3,"41":1,"42":2,"43":1,"44":2,"45":1,"46":2,"47":3,"48":3,"76":1,"77":1,"79":2,"89":4,"103":1,"107":3,"110":2,"117":2,"126":1,"134":1,"135":1,"136":1,"137":8,"139":10,"148":2,"152":1,"153":4,"154":1,"155":1,"159":1,"160":6,"161":2,"166":1,"167":1,"168":1,"170":1,"172":3,"176":1,"177":2,"178":1,"181":1,"183":1,"189":1,"195":1,"196":2,"204":1,"208":1,"223":1,"229":1,"232":2,"237":2,"249":2,"256":1,"268":2,"269":1,"270":1,"271":1,"272":1,"273":1,"290":1,"297":1,"302":2,"304":1,"305":1,"307":1,"309":2,"310":1,"311":1,"313":2,"314":2,"316":1,"317":2,"320":2,"329":1,"335":1,"336":1,"337":1,"344":1,"345":1,"351":4,"356":1,"357":3,"358":1,"361":1,"364":1,"365":1,"367":2,"373":15,"376":1,"384":1,"385":2,"387":1,"388":4,"392":26,"395":1,"396":1,"398":8,"399":1,"402":1,"404":1,"406":1,"407":1,"416":1,"418":2,"419":1,"425":3,"463":1,"465":1,"466":1,"484":2,"485":1,"494":1,"499":1,"502":3,"503":3,"505":1,"509":3,"511":1,"516":1,"519":2,"522":2,"531":1,"537":2,"539":1,"541":2,"542":1,"547":1,"550":1,"552":3,"553":1,"554":1,"556":1,"559":1,"561":2,"562":1,"568":2,"571":2,"573":1,"574":1,"575":2,"577":1,"578":1,"585":1,"586":1,"588":2,"592":2,"594":1,"598":1,"599":2,"603":1,"610":1,"620":1,"622":1,"625":1,"626":1,"631":1,"632":2,"633":1,"638":1,"639":2,"643":1,"644":1,"648":1,"649":2,"688":1,"690":1,"696":1,"707":1,"709":3,"711":1,"712":3,"714":1,"715":1,"716":2,"721":3,"722":1,"724":1,"727":1}}],["ini",{"2":{"557":1}}],["initcelestia",{"2":{"308":1,"316":1,"619":1,"659":1}}],["initgit",{"2":{"126":1}}],["init",{"2":{"115":5,"126":1,"247":5,"259":2,"264":3,"308":5,"316":5,"341":4,"367":2,"484":8,"488":2,"500":2,"550":3,"553":3,"556":4,"579":9,"592":6,"619":5,"645":2,"659":5,"724":2}}],["initiatives",{"2":{"448":1,"449":2,"450":1}}],["initiated",{"2":{"33":1}}],["initially",{"2":{"707":1}}],["initialisation",{"2":{"550":1,"568":1}}],["initialise",{"2":{"471":1,"679":1}}],["initialization",{"2":{"565":1,"571":1}}],["initializing",{"2":{"308":1,"484":1}}],["initializes",{"2":{"117":1,"249":1,"373":2}}],["initialize",{"0":{"484":1,"579":1,"619":1,"645":1,"659":1,"724":1},"1":{"660":1,"661":1,"662":1,"663":1,"664":1},"2":{"115":1,"126":1,"231":1,"247":1,"316":2,"357":1,"367":1,"488":2,"554":1,"556":1,"571":1,"579":1,"592":1}}],["initialized",{"2":{"85":1,"126":1,"564":1,"637":1,"644":1,"659":1}}],["initial",{"2":{"23":1,"84":1,"422":2,"444":1,"449":1,"450":1,"494":2}}],["inherently",{"2":{"440":1}}],["inherits",{"2":{"506":1}}],["inherit",{"2":{"231":1}}],["inheriting",{"2":{"201":1}}],["inability",{"2":{"409":1}}],["inappropriate",{"2":{"8":1,"9":1,"13":2,"15":1,"16":1}}],["injectedwallet",{"2":{"267":4}}],["innercontainerstyle",{"2":{"268":4}}],["innernode",{"2":{"155":12,"161":12}}],["innovative",{"2":{"232":1}}],["inputting",{"2":{"721":1,"727":1}}],["input=$",{"2":{"348":2}}],["input=ce3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f",{"2":{"348":2}}],["inputstyle",{"2":{"268":6}}],["inputs",{"2":{"127":2}}],["input",{"2":{"127":2,"185":3,"189":2,"192":2,"268":2,"348":4,"349":2,"544":1}}],["inbox",{"2":{"98":1,"99":1,"108":1,"342":2,"351":2,"359":2}}],["indeed",{"2":{"415":1,"424":1,"425":1}}],["independently",{"2":{"205":1,"422":1}}],["indexing",{"0":{"595":1},"2":{"408":1,"594":1,"597":1,"610":1}}],["index",{"2":{"134":2,"151":2,"152":2,"158":3,"159":2,"161":4,"170":1,"172":1,"173":3,"267":2,"268":4,"519":2,"610":2,"729":1}}],["indexer",{"0":{"610":1,"729":1},"2":{"595":1,"597":2,"610":3,"729":1}}],["indexers",{"2":{"413":1}}],["indexed",{"2":{"129":1,"610":2}}],["indexes",{"2":{"84":1}}],["induced",{"2":{"139":1}}],["industry",{"2":{"44":1}}],["indicates",{"2":{"334":1,"351":1,"557":1}}],["indicators",{"2":{"38":1}}],["individual",{"2":{"10":1,"16":1,"27":1,"65":1,"131":1,"260":2}}],["individuals",{"2":{"8":1,"16":1,"203":1,"288":1}}],["inflation",{"0":{"445":1},"2":{"445":3,"455":2}}],["infra",{"2":{"297":1}}],["infrastructure",{"2":{"24":1,"27":1,"48":1,"91":4,"288":2,"442":1,"449":1,"548":1,"685":1,"703":1}}],["infura",{"2":{"258":1,"266":6,"268":12}}],["infocurl",{"2":{"712":1}}],["infos",{"2":{"522":2}}],["inform",{"2":{"107":1}}],["informative",{"2":{"48":1,"63":1,"65":1}}],["information$",{"2":{"488":1}}],["information",{"0":{"28":1,"33":1,"728":1},"2":{"8":1,"30":4,"31":1,"32":1,"33":18,"34":1,"42":1,"48":2,"79":2,"85":1,"86":1,"92":1,"103":1,"107":1,"109":1,"110":1,"134":2,"135":2,"136":2,"148":2,"161":1,"170":1,"172":1,"207":1,"208":1,"221":1,"237":1,"243":2,"309":1,"310":1,"324":1,"326":1,"367":1,"387":1,"392":2,"408":2,"409":1,"411":1,"484":1,"485":1,"488":1,"494":1,"503":2,"505":1,"531":2,"537":1,"550":1,"577":1,"603":1,"620":3,"637":1,"653":1,"660":3,"724":1}}],["informalsystems",{"2":{"508":1}}],["informal",{"2":{"37":1}}],["info",{"0":{"51":1},"1":{"52":1,"53":1,"54":1,"55":1,"56":1},"2":{"27":1,"77":1,"273":1,"312":1,"324":2,"330":2,"411":1,"522":4,"629":2,"712":1,"721":2,"727":2}}],["int",{"2":{"373":2}}],["int64",{"2":{"151":8,"155":4,"159":8,"161":8}}],["intake",{"2":{"47":2}}],["introducing",{"2":{"423":1}}],["introduction",{"0":{"59":1,"62":1,"104":1,"163":1,"284":1,"300":1,"423":1},"1":{"105":1,"106":1,"107":1,"108":1,"109":1,"110":1,"111":1,"164":1,"165":1,"166":1,"167":1,"168":1,"169":1,"170":1,"171":1,"172":1,"173":1,"174":1,"175":1,"176":1,"177":1,"178":1,"179":1,"180":1,"181":1,"182":1,"183":1,"184":1,"185":1,"186":1,"187":1,"188":1,"189":1,"190":1,"191":1,"192":1,"193":1,"194":1,"195":1,"285":1,"286":1,"287":1,"288":1,"289":1,"290":1,"291":1,"301":1,"302":1},"2":{"43":1,"59":2,"61":1,"62":3,"79":1,"80":1,"101":1,"105":1,"277":1}}],["introduces",{"2":{"441":1}}],["introduced",{"2":{"395":1}}],["introduce",{"2":{"37":1,"256":1}}],["intel",{"2":{"305":1,"586":1}}],["integrating",{"2":{"209":1,"276":1}}],["integrations",{"0":{"397":1,"469":1,"672":1},"1":{"398":1,"399":1,"400":1,"470":1,"471":1,"673":1,"674":1,"675":1,"676":1,"677":1,"678":1,"679":1,"680":1,"681":1},"2":{"31":1,"255":1,"505":1}}],["integration",{"0":{"284":1,"285":1},"1":{"285":1,"286":2,"287":1,"288":1,"289":1,"290":1,"291":1},"2":{"31":1,"73":1,"79":2,"80":1,"94":1,"101":1,"105":2,"106":2,"107":2,"108":1,"109":2,"110":1,"201":1,"213":1,"222":1,"285":1,"287":1,"291":2,"357":1,"505":1,"667":1}}],["integrate",{"0":{"123":1,"132":1,"207":1,"209":1,"276":1},"1":{"124":1,"125":1,"126":1,"127":1,"128":1,"129":1,"130":1,"131":1,"133":1,"134":1,"135":1,"136":1,"137":1,"138":1,"139":1,"140":1,"141":1,"142":1,"143":1,"208":1,"209":1,"210":1,"211":1,"277":1,"278":1,"279":1,"280":1,"281":1,"282":1,"283":1},"2":{"207":1,"209":2,"233":1,"285":1,"365":1,"505":1}}],["integrated",{"2":{"109":1}}],["integrates",{"2":{"94":1}}],["integrity",{"2":{"84":2}}],["integer",{"2":{"84":1,"116":1,"117":1,"248":1,"249":1}}],["interruptions",{"2":{"667":1}}],["inter",{"2":{"627":1}}],["intermediate",{"2":{"417":1,"418":1,"424":1,"425":1}}],["interchain",{"2":{"367":1}}],["interval",{"2":{"360":2,"503":4,"554":4,"629":2}}],["interpreted",{"2":{"351":1}}],["interfaces",{"2":{"135":2,"260":2,"265":1,"452":1,"453":1,"462":1,"607":2}}],["interface",{"0":{"129":1},"2":{"107":2,"129":3,"131":1,"135":4,"311":1,"379":1,"380":1,"421":1,"506":1,"531":2}}],["internally",{"2":{"478":1}}],["internal",{"2":{"89":5,"293":1}}],["interestingly",{"2":{"423":1}}],["interested",{"2":{"44":2,"45":3,"48":3,"256":1,"336":1}}],["interest",{"2":{"42":2,"43":1,"53":1}}],["interests",{"2":{"42":1,"47":1,"48":1}}],["interactivecelestia",{"2":{"536":1}}],["interactive",{"2":{"65":1,"536":1}}],["interactions",{"2":{"14":1,"135":1,"215":1}}],["interaction",{"2":{"14":2,"15":3,"16":1,"31":1,"37":1,"88":1,"478":1}}],["interacting",{"0":{"121":1,"253":1},"2":{"3":1,"143":1,"292":1,"296":1,"312":2,"336":1,"541":1,"551":1,"552":1}}],["interact",{"0":{"76":1,"78":1,"274":1},"2":{"7":1,"76":2,"77":1,"120":1,"131":1,"210":1,"215":1,"217":1,"243":2,"244":2,"252":1,"272":2,"273":1,"280":1,"292":1,"311":1,"312":2,"313":1,"314":1,"315":2,"583":1,"655":1,"689":1,"690":1,"691":1}}],["intensive",{"2":{"598":1}}],["intend",{"2":{"34":1}}],["intent",{"2":{"34":1}}],["intention",{"2":{"34":1}}],["into",{"0":{"411":1},"2":{"23":1,"26":1,"59":1,"69":1,"75":1,"87":1,"91":1,"94":1,"107":1,"108":1,"109":1,"146":2,"148":1,"201":1,"237":1,"265":3,"268":2,"272":1,"302":1,"307":1,"357":1,"380":1,"384":1,"416":2,"418":1,"419":2,"422":1,"424":5,"438":3,"491":1,"501":1,"505":1,"514":1,"539":1,"544":1,"551":1,"554":1,"559":1,"562":1,"625":1,"630":1,"667":1}}],["inception",{"2":{"666":1}}],["incentive",{"2":{"425":1}}],["incentives",{"2":{"43":1,"412":1}}],["incentivizes",{"2":{"414":1}}],["incentivize",{"0":{"412":1},"2":{"412":1}}],["incentivized",{"0":{"30":1},"1":{"31":1,"32":1,"33":1,"34":1},"2":{"30":1,"449":1}}],["increment",{"2":{"116":3,"117":4,"248":3,"249":4}}],["incredible",{"2":{"50":1}}],["increased",{"2":{"91":2}}],["increases",{"2":{"45":1,"84":1,"116":1,"117":1,"248":1,"249":1,"405":1,"423":1}}],["increase",{"2":{"44":1,"45":1,"422":1,"423":1,"502":2,"557":1}}],["increasing",{"2":{"42":1,"417":1,"423":1,"557":1}}],["incubators",{"2":{"40":1}}],["incomplete",{"2":{"419":1}}],["income",{"2":{"34":1}}],["incorrectly",{"0":{"418":1},"2":{"406":1,"407":1,"418":2}}],["incorrect",{"2":{"89":1,"255":1,"352":1,"407":1}}],["incorporate",{"2":{"30":1}}],["incident",{"2":{"11":1,"14":1}}],["inclusion",{"0":{"131":1,"141":1,"150":1,"152":1,"173":1},"2":{"46":1,"128":2,"129":1,"131":4,"134":1,"136":3,"141":1,"146":1,"148":1,"150":1,"151":2,"154":1,"155":1,"159":1,"161":1,"162":1,"166":3,"167":2,"172":2,"173":7,"186":1,"189":1,"206":2,"395":1}}],["inclusive",{"2":{"7":1,"52":1}}],["including",{"2":{"14":1,"15":2,"16":1,"27":1,"31":1,"33":2,"34":4,"36":3,"37":1,"47":1,"52":1,"65":1,"84":1,"86":1,"92":1,"386":1,"449":1,"721":1}}],["included",{"2":{"129":2,"136":2,"137":3,"146":1,"159":2,"170":1,"172":1,"181":1,"206":1,"260":1,"317":1,"319":1,"320":1,"388":3,"394":1,"395":1,"396":1,"438":1,"626":1,"671":1}}],["includes",{"2":{"14":1,"57":1,"58":1,"85":1,"94":1,"109":1,"117":1,"249":1,"282":1,"411":1,"419":1,"438":1,"450":1,"628":1,"650":1}}],["include",{"2":{"8":2,"10":1,"30":3,"44":1,"170":1,"207":1,"213":1,"233":1,"260":1,"288":1,"313":1,"353":1,"374":1,"413":1,"438":1,"504":1,"651":1}}],["invoke",{"2":{"260":2}}],["involves",{"2":{"385":1,"410":1}}],["involve",{"2":{"135":2,"136":1}}],["involvement",{"2":{"43":1}}],["involved",{"2":{"14":1,"15":1,"188":1,"213":1,"290":1,"385":1,"386":1}}],["involving",{"2":{"34":1}}],["investigations",{"2":{"183":1}}],["investigate",{"2":{"42":1,"44":1}}],["investigated",{"2":{"11":1,"168":1,"181":1,"191":1}}],["invalid",{"2":{"136":2,"139":2,"161":4,"172":1,"177":2,"181":2,"418":2,"425":1,"554":2}}],["invites",{"2":{"54":1}}],["invite",{"2":{"43":2,"45":2}}],["invitations",{"2":{"36":1}}],["invitation",{"2":{"34":2,"53":1}}],["invisible",{"2":{"7":1}}],["inspect",{"2":{"402":1}}],["inspired",{"2":{"17":1,"176":1}}],["insecure",{"2":{"392":4}}],["inside",{"0":{"189":1},"2":{"170":2,"185":1,"186":1,"189":2,"191":1,"192":3,"194":1,"235":1,"237":1,"372":1,"486":1,"539":1,"621":1,"649":1,"650":1,"651":1,"663":1}}],["insight",{"2":{"501":1}}],["insightful",{"2":{"50":1}}],["insights",{"2":{"5":1,"38":1,"43":1,"65":1,"508":1}}],["instrumentation",{"2":{"502":6}}],["instructions",{"2":{"304":2,"377":1,"485":1,"510":1,"511":1,"522":1,"540":1,"560":1,"562":1,"575":3,"583":1,"585":2,"590":1,"620":1,"685":2,"719":1,"727":1,"729":1}}],["instead",{"2":{"102":1,"108":1,"135":1,"139":3,"174":1,"192":1,"255":1,"319":1,"360":1,"384":3,"388":1,"441":1,"458":1}}],["instability",{"2":{"667":1,"698":1}}],["instantaneous",{"2":{"623":1}}],["instantiation",{"0":{"639":1},"1":{"640":1,"641":1,"642":1,"643":1,"644":1,"645":1,"646":1,"647":1,"648":1,"649":1,"650":1,"651":1,"652":1,"653":1}}],["instantiating",{"2":{"103":1,"264":1,"308":1}}],["instantiate",{"0":{"308":1,"653":1},"2":{"308":1,"373":1,"374":1,"422":1,"639":1}}],["instance",{"2":{"69":1,"75":1,"90":2,"117":1,"249":1,"272":1,"373":1,"480":1,"503":1,"543":1,"556":1,"589":1,"615":1,"657":1,"719":1}}],["instances",{"2":{"11":1}}],["installer",{"2":{"575":1}}],["installed",{"2":{"114":1,"125":1,"242":1,"246":1,"305":2,"366":1,"398":1,"515":1,"526":1,"535":1,"539":1,"561":1,"564":1,"572":1,"586":2,"628":1,"629":1}}],["installmake",{"2":{"307":1,"539":1,"559":1}}],["installs",{"2":{"236":2,"307":1,"559":1}}],["installation",{"0":{"236":1,"340":1,"642":1},"2":{"234":1,"264":1,"540":2,"560":2}}],["installing",{"0":{"126":1,"540":1,"559":1,"560":1},"2":{"304":2,"307":2,"481":1,"483":1,"538":1,"540":1,"559":1,"560":1,"575":1,"583":1,"585":2,"590":1,"591":1,"617":1,"658":1,"706":1,"709":1,"723":1}}],["install",{"0":{"305":1,"307":1,"367":1,"483":1,"538":1,"558":1,"585":1,"586":1,"591":1,"617":1,"658":1,"723":1},"1":{"539":1,"540":1,"541":1,"559":1,"560":1,"561":1,"562":1,"618":1,"619":1,"620":1,"621":1,"622":1,"623":1},"2":{"87":2,"126":4,"236":2,"265":1,"266":3,"270":1,"304":16,"305":2,"307":3,"311":1,"340":1,"354":1,"367":1,"398":2,"483":1,"503":1,"505":1,"510":1,"512":1,"535":1,"539":3,"540":1,"559":3,"560":1,"562":1,"575":1,"585":15,"586":2,"628":1,"642":1,"658":1,"707":1,"708":1,"709":1,"712":2,"714":2,"715":2,"716":2}}],["insulting",{"2":{"8":1}}],["in",{"0":{"66":1,"67":1,"110":1,"153":1,"350":1,"388":1,"404":1,"441":1,"556":1,"635":1,"697":1},"1":{"67":1,"68":2,"69":2,"70":1,"71":1,"72":1,"154":1,"155":1,"156":1,"157":1,"158":1,"159":1,"160":1,"161":1,"351":1,"352":1,"698":1,"699":1,"700":1,"701":1,"702":1,"703":1,"704":1,"705":1},"2":{"3":1,"4":2,"5":1,"7":2,"8":1,"9":1,"10":1,"12":2,"13":1,"14":1,"19":1,"21":1,"23":6,"24":5,"25":3,"27":2,"29":1,"30":10,"31":17,"32":3,"33":3,"34":41,"35":1,"37":1,"38":1,"39":1,"40":2,"41":4,"42":3,"43":2,"44":6,"45":7,"46":4,"48":7,"50":2,"52":2,"53":2,"54":1,"56":1,"60":1,"61":1,"65":4,"66":2,"68":3,"69":4,"71":3,"72":3,"75":3,"76":4,"77":1,"82":1,"83":2,"84":12,"85":4,"86":1,"87":1,"88":5,"89":10,"91":10,"92":3,"93":1,"94":1,"95":1,"102":4,"105":1,"106":1,"107":7,"108":10,"109":1,"110":3,"111":1,"113":1,"115":1,"116":4,"117":8,"122":1,"128":2,"129":2,"130":1,"131":2,"134":13,"135":3,"136":12,"137":4,"139":1,"145":1,"146":8,"148":10,"149":1,"150":1,"154":3,"155":12,"156":1,"157":1,"158":9,"159":10,"160":1,"162":2,"163":1,"165":1,"166":3,"167":4,"169":3,"170":4,"171":1,"172":9,"173":9,"177":2,"178":1,"180":2,"181":5,"182":1,"183":1,"184":1,"185":3,"186":1,"188":1,"189":2,"192":1,"195":2,"201":1,"203":1,"205":1,"206":4,"207":3,"208":1,"211":1,"213":5,"214":1,"217":1,"229":1,"231":1,"232":2,"233":2,"235":1,"236":4,"237":2,"238":1,"243":3,"244":3,"245":1,"247":1,"248":4,"249":8,"255":4,"256":2,"260":6,"263":2,"264":3,"265":2,"266":1,"271":1,"272":7,"273":1,"278":1,"279":1,"280":4,"283":2,"285":1,"290":3,"291":1,"292":1,"297":1,"299":1,"302":1,"304":1,"305":1,"308":1,"309":2,"310":2,"312":1,"313":1,"314":1,"315":1,"316":1,"317":4,"319":4,"320":1,"324":2,"325":1,"329":6,"332":1,"334":3,"336":2,"346":1,"350":1,"351":4,"352":1,"353":2,"354":1,"356":1,"357":4,"358":1,"365":1,"367":4,"369":1,"370":1,"373":5,"378":1,"379":1,"380":5,"383":1,"384":3,"385":4,"386":4,"387":2,"388":2,"389":1,"391":1,"393":1,"394":2,"395":1,"396":1,"398":2,"399":2,"400":2,"404":2,"407":1,"409":1,"410":1,"411":1,"412":1,"415":1,"416":4,"417":3,"419":1,"422":9,"423":1,"424":3,"425":3,"426":1,"429":1,"430":1,"434":1,"438":1,"439":2,"440":3,"441":3,"442":4,"444":1,"445":2,"448":1,"449":1,"450":2,"455":1,"457":1,"458":3,"459":2,"460":1,"462":1,"463":1,"466":2,"469":2,"470":2,"471":2,"478":3,"484":1,"486":1,"491":1,"494":4,"495":2,"500":2,"503":3,"504":1,"509":3,"511":1,"515":1,"517":1,"519":3,"522":1,"523":1,"524":1,"526":2,"529":1,"531":1,"532":1,"536":1,"539":1,"540":2,"542":1,"543":1,"544":1,"547":2,"548":3,"552":1,"553":2,"554":1,"556":2,"557":4,"559":1,"560":2,"565":1,"568":1,"569":1,"571":1,"573":2,"574":1,"575":2,"579":3,"583":1,"586":1,"587":1,"592":5,"594":1,"596":1,"597":1,"598":2,"599":2,"601":3,"603":1,"608":1,"610":2,"611":6,"620":1,"621":1,"623":2,"624":2,"625":3,"626":3,"627":2,"628":1,"631":2,"632":5,"643":1,"644":1,"645":1,"648":3,"649":2,"650":2,"652":1,"659":1,"660":1,"661":2,"662":1,"663":1,"666":2,"670":1,"671":5,"672":4,"681":1,"685":3,"688":1,"690":1,"691":1,"692":1,"696":2,"700":1,"703":1,"706":2,"707":2,"709":4,"712":3,"716":2,"717":2,"721":2,"727":4}}],["imagine",{"2":{"233":1}}],["image",{"2":{"102":1,"243":3,"244":6,"360":4,"400":1,"577":1}}],["imagery",{"2":{"8":1}}],["images",{"2":{"3":1,"400":1}}],["immutable",{"2":{"84":2,"127":2}}],["improperly",{"2":{"406":1}}],["improve",{"2":{"47":1}}],["improvement",{"2":{"38":1,"417":1}}],["improvements",{"2":{"4":1,"667":1}}],["improved",{"2":{"29":1,"501":1}}],["improves",{"2":{"24":1}}],["improving",{"2":{"2":1}}],["impractical",{"2":{"402":1}}],["imported",{"2":{"270":1}}],["importing",{"0":{"239":1,"491":1},"2":{"542":1}}],["import",{"0":{"372":1},"2":{"117":4,"127":6,"150":2,"151":2,"161":6,"239":1,"249":4,"260":4,"262":4,"267":19,"268":17,"270":2,"371":1,"372":3,"387":1,"392":1,"398":7,"478":1,"491":5,"503":2,"542":1,"630":1}}],["importance",{"2":{"65":1}}],["important",{"0":{"51":1},"1":{"52":1,"53":1,"54":1,"55":1,"56":1},"2":{"24":1,"30":1,"108":1,"134":2,"386":1,"394":1,"421":1,"503":1,"504":1,"519":2,"551":1,"624":1}}],["impose",{"2":{"34":1}}],["implement",{"2":{"373":2,"422":1,"442":2}}],["implements",{"2":{"107":1}}],["implementations",{"2":{"135":2,"627":1}}],["implementation",{"0":{"107":1,"108":1,"109":1,"351":1},"2":{"33":1,"106":3,"126":1,"131":1,"133":1,"148":1,"161":4,"201":2,"205":1,"206":1,"380":1,"478":1,"625":1,"628":1}}],["implemented",{"2":{"31":1,"162":1,"351":1,"441":4,"690":1}}],["impact=profit",{"2":{"288":1}}],["impactful",{"2":{"43":1,"65":1}}],["impact",{"2":{"12":1,"13":1,"14":1,"15":1,"16":1,"17":1,"27":1,"44":1}}],["ith",{"2":{"386":2}}],["itemcount",{"2":{"260":6}}],["itn",{"2":{"30":7,"31":11,"32":3,"33":1,"34":38}}],["itself",{"2":{"30":1,"94":1,"503":2}}],["its",{"2":{"19":2,"27":2,"30":3,"34":1,"84":4,"85":2,"108":1,"126":1,"148":1,"155":2,"156":2,"161":2,"166":1,"170":1,"172":1,"176":1,"180":1,"181":1,"185":1,"204":1,"205":1,"222":2,"322":1,"375":1,"384":1,"385":1,"404":1,"405":1,"419":2,"422":1,"451":1,"459":1,"503":1,"557":1,"666":1,"712":1,"714":1,"715":1,"716":1}}],["it",{"0":{"270":1,"271":1},"2":{"2":2,"24":1,"30":1,"34":1,"35":1,"41":1,"42":1,"47":2,"48":1,"53":1,"69":1,"75":1,"76":1,"84":5,"86":2,"88":1,"89":3,"90":1,"102":1,"107":6,"108":1,"110":1,"116":1,"117":5,"120":2,"128":1,"129":1,"131":6,"133":1,"135":2,"139":2,"146":2,"148":2,"150":2,"152":1,"153":1,"154":1,"156":2,"159":1,"161":2,"163":1,"166":1,"168":1,"170":1,"172":2,"177":1,"180":1,"181":1,"185":2,"192":1,"205":1,"206":2,"208":3,"213":2,"214":2,"217":1,"228":1,"231":1,"234":1,"236":1,"248":1,"249":5,"252":2,"260":3,"262":1,"263":3,"264":2,"265":2,"268":4,"270":1,"272":2,"283":1,"290":2,"295":1,"297":1,"309":2,"312":1,"317":1,"326":2,"335":2,"337":1,"347":1,"348":1,"351":5,"357":1,"360":1,"365":1,"367":4,"370":2,"373":9,"374":3,"375":2,"379":1,"380":3,"385":2,"386":4,"388":4,"392":2,"395":1,"396":1,"398":4,"402":5,"405":2,"406":1,"411":1,"412":1,"418":1,"419":1,"422":1,"423":1,"424":5,"425":1,"442":2,"445":1,"466":5,"484":2,"485":1,"501":1,"502":2,"503":6,"504":1,"505":1,"509":1,"515":1,"519":2,"522":1,"533":1,"536":1,"540":1,"541":1,"544":1,"547":2,"550":1,"551":1,"552":1,"554":2,"556":2,"557":1,"560":1,"561":1,"573":1,"578":2,"588":2,"599":1,"602":1,"606":1,"607":2,"611":1,"620":2,"624":1,"626":1,"627":1,"628":2,"630":1,"631":1,"638":1,"648":1,"650":2,"651":1,"659":1,"666":1,"667":1,"671":1,"685":1,"688":1,"698":1,"701":1,"703":1,"709":1,"712":2,"721":1,"722":2}}],["issuing",{"2":{"458":1}}],["issuance",{"2":{"445":2}}],["issued",{"2":{"34":1}}],["issues",{"2":{"9":1,"60":1,"89":1,"285":1,"357":1,"562":1}}],["issue",{"2":{"4":1,"91":1,"388":1,"409":1,"445":1,"458":1,"628":1,"696":2}}],["isn",{"2":{"139":1,"671":1}}],["isvalid",{"2":{"139":2}}],["isolated",{"2":{"575":1}}],["isolates",{"2":{"232":1}}],["isolate",{"2":{"135":1}}],["is",{"0":{"201":1,"206":1,"228":1,"229":1,"289":1,"402":1,"403":1,"405":1,"407":1,"408":1,"409":1,"410":1,"413":1},"2":{"1":1,"5":1,"8":1,"10":1,"15":1,"17":1,"21":1,"22":1,"24":1,"26":1,"27":2,"30":5,"31":3,"33":1,"34":7,"35":1,"37":1,"40":1,"43":1,"44":1,"45":2,"47":3,"48":1,"49":1,"52":1,"53":2,"55":2,"56":1,"57":1,"59":1,"60":2,"61":2,"65":3,"71":1,"82":1,"83":3,"84":8,"85":1,"86":3,"88":3,"89":7,"91":10,"92":1,"94":2,"106":3,"107":7,"108":2,"109":1,"110":2,"117":7,"126":2,"127":2,"128":3,"129":2,"131":6,"133":3,"134":30,"135":5,"136":6,"137":6,"139":6,"143":1,"145":1,"146":3,"148":5,"150":2,"152":1,"154":1,"155":5,"156":3,"158":2,"159":12,"161":4,"162":1,"163":1,"165":1,"166":1,"167":1,"170":3,"171":1,"172":9,"173":12,"176":3,"177":6,"178":1,"181":9,"183":1,"184":1,"185":6,"186":1,"189":6,"191":1,"192":4,"195":1,"201":2,"203":1,"204":2,"205":2,"206":3,"208":3,"212":2,"213":8,"217":1,"218":1,"228":1,"229":1,"231":2,"232":1,"234":1,"235":1,"249":7,"255":3,"257":2,"260":4,"261":1,"262":2,"263":1,"264":3,"266":1,"268":3,"270":1,"276":1,"278":2,"280":1,"281":1,"282":1,"283":3,"284":2,"285":1,"288":2,"289":2,"290":7,"291":1,"292":1,"295":1,"296":1,"297":2,"301":1,"305":1,"307":2,"308":1,"309":1,"310":1,"312":5,"313":2,"314":2,"316":3,"317":3,"319":3,"320":2,"322":1,"323":1,"324":2,"325":1,"326":1,"329":2,"337":1,"351":5,"356":1,"357":4,"360":3,"367":1,"368":1,"370":4,"373":3,"377":1,"379":1,"380":4,"384":4,"385":4,"386":14,"387":2,"388":2,"392":8,"394":2,"395":1,"396":1,"398":5,"402":2,"403":3,"404":1,"405":3,"406":6,"407":3,"408":2,"409":2,"410":2,"411":3,"412":3,"414":1,"415":2,"416":6,"418":7,"419":8,"421":7,"422":3,"423":3,"424":5,"425":9,"426":1,"432":1,"436":1,"438":2,"440":5,"441":3,"442":4,"444":1,"450":3,"457":1,"458":1,"462":1,"463":4,"464":1,"465":4,"466":1,"469":1,"473":1,"474":1,"478":5,"480":1,"484":1,"485":4,"487":1,"494":3,"501":1,"503":5,"505":2,"506":1,"509":6,"510":1,"517":3,"519":2,"522":1,"523":1,"524":1,"531":2,"536":1,"540":2,"541":2,"542":1,"543":1,"545":1,"546":1,"547":2,"550":2,"551":4,"552":2,"553":1,"554":3,"556":2,"557":2,"559":2,"560":2,"566":1,"568":1,"569":2,"571":4,"573":1,"574":1,"575":3,"577":2,"578":1,"583":2,"586":1,"588":2,"589":1,"592":2,"594":1,"596":2,"598":1,"599":2,"600":2,"608":3,"610":7,"611":1,"612":1,"615":1,"620":3,"623":1,"624":3,"626":3,"627":1,"628":4,"631":3,"637":1,"638":1,"639":1,"645":2,"647":1,"648":1,"652":2,"655":1,"660":2,"662":1,"666":2,"667":2,"671":7,"672":2,"680":1,"681":1,"685":4,"689":2,"690":3,"691":3,"693":1,"698":2,"701":2,"703":1,"708":1,"709":2,"710":1,"712":3,"715":1,"716":1,"719":1,"722":3,"726":1,"727":1}}],["lcd",{"2":{"675":2}}],["l5dhzjldczo",{"2":{"519":2,"520":2}}],["ljj6hspn0kn09qf9fy8kdyh40000gn",{"2":{"515":4}}],["luckily",{"2":{"409":1,"412":1}}],["luck",{"2":{"356":1}}],["lunaroasis",{"2":{"309":2,"661":2,"674":1,"675":1,"676":1,"679":1}}],["l",{"2":{"351":2}}],["lfg",{"2":{"91":2}}],["l1",{"0":{"364":1},"2":{"82":2,"84":1,"91":8,"119":3,"347":1,"348":1,"352":1,"364":1,"403":1}}],["l2genesisdeltatimeoffset",{"2":{"353":1}}],["l2oo",{"2":{"342":2,"359":2}}],["l2s",{"2":{"110":1,"201":2,"205":1,"213":1,"255":2,"290":1}}],["l2",{"0":{"88":1,"364":1},"2":{"77":1,"82":1,"84":2,"85":1,"86":1,"91":6,"109":1,"113":1,"119":4,"120":2,"129":1,"131":1,"201":1,"204":2,"209":1,"255":1,"273":1,"284":1,"288":3,"364":1,"403":2}}],["lts",{"2":{"281":1,"480":1,"589":1,"615":1,"657":1,"719":1}}],["lt",{"2":{"69":1,"75":2,"76":2,"89":8,"137":2,"151":4,"159":4,"160":6,"207":1,"237":13,"238":12,"239":12,"243":5,"244":8,"260":2,"263":2,"264":2,"267":14,"268":100,"272":4,"309":6,"310":21,"312":6,"314":6,"315":4,"317":8,"319":8,"323":2,"326":4,"329":6,"331":4,"346":4,"347":1,"360":11,"390":4,"391":4,"398":14,"471":8,"473":3,"484":8,"485":4,"486":12,"490":8,"491":6,"492":8,"493":18,"494":20,"495":4,"496":6,"497":16,"498":4,"499":6,"500":2,"503":2,"531":4,"542":10,"544":10,"545":4,"546":4,"547":16,"550":10,"553":10,"554":8,"555":4,"556":8,"580":2,"620":4,"621":12,"630":4,"660":4,"661":6,"662":3,"663":12,"679":8,"693":3,"712":4,"714":4,"715":6,"716":6,"721":4,"724":2,"727":2}}],["l3s",{"2":{"110":1,"201":1,"255":1}}],["l3configuration",{"2":{"91":2}}],["l3",{"2":{"66":1,"69":1,"71":1,"86":2,"91":6}}],["living",{"2":{"670":1}}],["lives",{"2":{"48":1,"108":1,"517":1}}],["live",{"2":{"46":4,"68":1,"262":1,"263":1,"291":1,"517":1,"667":1}}],["life",{"2":{"438":1,"666":1}}],["lifecycle",{"0":{"424":1},"1":{"425":1}}],["libp2p",{"2":{"425":1,"478":1}}],["libssl",{"2":{"304":4,"585":4}}],["libs",{"2":{"161":2}}],["library",{"0":{"153":1,"294":1},"1":{"154":1,"155":1,"156":1,"157":1,"158":1,"159":1,"160":1,"161":1},"2":{"131":3,"153":1,"161":2,"172":1,"181":1,"189":1,"260":1,"265":2,"294":1,"295":1,"371":1}}],["libraries",{"2":{"40":2,"186":1,"265":1,"279":1,"281":1,"282":1,"372":1,"706":1}}],["lib",{"2":{"127":2,"131":1,"161":2}}],["lightdsudo",{"2":{"716":2}}],["lightd",{"2":{"716":13}}],["lightbridgefull",{"2":{"577":1}}],["light$home",{"2":{"357":1}}],["light",{"0":{"60":1,"89":1,"142":1,"308":1,"357":1,"367":1,"391":1,"578":1,"654":1,"655":1,"657":1,"659":1,"660":1,"663":1,"664":1,"688":1,"716":1},"1":{"579":1,"580":1,"655":1,"656":1,"657":1,"658":2,"659":1,"660":2,"661":2,"662":2,"663":2,"664":2,"665":1},"2":{"60":7,"80":1,"89":7,"109":1,"135":9,"142":1,"143":1,"201":2,"206":2,"232":1,"235":1,"237":2,"238":2,"239":2,"243":5,"244":10,"258":1,"283":1,"297":1,"303":1,"308":8,"309":13,"310":12,"314":3,"315":3,"316":12,"317":3,"334":3,"355":1,"357":13,"360":9,"366":1,"367":9,"368":1,"370":1,"387":1,"404":1,"405":4,"406":6,"407":2,"415":1,"416":5,"417":8,"418":3,"423":2,"425":9,"441":6,"470":1,"471":3,"478":1,"543":2,"548":1,"555":3,"561":1,"565":2,"571":6,"573":3,"579":6,"580":6,"601":2,"613":2,"654":1,"655":2,"656":1,"659":6,"660":8,"661":12,"663":9,"664":1,"665":1,"678":1,"679":1,"680":2,"685":1,"696":2,"707":2,"709":2,"715":1,"716":7}}],["lightning",{"2":{"48":1,"284":1}}],["lisbon",{"2":{"50":1}}],["listcelestia",{"2":{"527":1,"536":1}}],["listing",{"2":{"490":2}}],["listens",{"2":{"607":1}}],["listen",{"2":{"379":1,"425":1,"502":4,"503":1,"607":1,"655":1}}],["listen=0",{"2":{"360":2}}],["listeners",{"2":{"260":2}}],["listening",{"0":{"160":1},"2":{"160":1,"607":1}}],["list",{"0":{"65":1,"520":1},"2":{"27":2,"31":1,"36":1,"42":1,"43":2,"56":2,"65":6,"84":1,"107":3,"127":2,"155":2,"158":2,"196":1,"243":3,"244":3,"310":2,"367":1,"377":1,"416":1,"447":1,"484":2,"485":2,"490":2,"494":1,"502":2,"520":2,"527":2,"536":1,"553":2,"592":2,"620":4,"629":4,"635":4,"651":2,"660":2,"661":2,"672":1,"685":1,"724":3,"727":1}}],["lines",{"2":{"499":2}}],["linear",{"2":{"386":1}}],["line",{"0":{"507":1},"2":{"309":1,"311":1,"367":1,"484":1,"506":1,"531":1,"556":1,"566":1,"577":1,"611":1,"651":1}}],["lineup",{"2":{"43":1}}],["linuxdocker",{"2":{"578":1}}],["linux",{"2":{"89":1,"281":1,"305":12,"480":1,"540":1,"560":1,"576":1,"583":1,"586":12,"589":1,"615":1,"657":1,"719":1}}],["linked",{"2":{"142":1}}],["link",{"2":{"33":1,"469":1,"503":2,"504":1,"672":1,"685":1}}],["links",{"2":{"27":2,"54":1,"64":1}}],["little",{"2":{"40":1,"85":1,"283":1,"351":1,"649":1}}],["liechtenstein",{"2":{"34":3}}],["license",{"2":{"33":1,"116":2,"117":2,"127":2,"248":2,"249":2,"260":4,"262":2}}],["limitnofile=65535",{"2":{"712":2}}],["limitnofile=1400000",{"2":{"557":2,"714":2,"715":2}}],["limitnofile",{"2":{"557":2}}],["limit=fc+∑i=1nssn",{"2":{"386":1}}],["limit×gas",{"2":{"386":1}}],["limits",{"0":{"384":1},"1":{"385":1,"386":1,"387":1},"2":{"320":1,"422":1,"502":2,"557":1}}],["limit",{"2":{"317":1,"320":1,"329":1,"374":1,"384":2,"386":4,"387":1,"392":2,"473":1,"522":2,"557":1,"671":1,"693":1}}],["limited",{"2":{"33":1,"34":2,"278":1,"417":2,"423":1,"494":1}}],["limitations",{"2":{"169":1}}],["limitation",{"2":{"31":1}}],["liable",{"2":{"31":2,"33":1}}],["liability",{"2":{"30":1,"31":2}}],["likely",{"2":{"135":5,"283":2,"413":1,"554":1}}],["likelihood",{"2":{"45":1,"425":1}}],["like",{"2":{"14":1,"36":1,"40":2,"41":1,"46":1,"47":2,"48":1,"80":1,"82":1,"84":2,"89":1,"90":1,"102":1,"133":1,"134":1,"139":1,"223":1,"228":1,"229":2,"232":2,"243":1,"244":1,"256":3,"285":1,"288":1,"290":1,"297":2,"302":1,"304":1,"309":1,"310":1,"316":1,"319":1,"326":2,"329":3,"335":1,"337":1,"344":1,"345":1,"347":1,"356":1,"357":1,"358":1,"361":1,"364":1,"398":1,"402":1,"403":1,"405":1,"411":1,"414":1,"444":1,"459":1,"469":1,"486":1,"503":1,"511":1,"512":1,"516":1,"522":3,"542":2,"547":1,"552":1,"553":1,"556":1,"557":1,"575":1,"577":2,"585":1,"603":1,"621":1,"622":1,"631":1,"660":1,"663":1,"667":1,"672":1,"685":2,"707":1}}],["laddr",{"2":{"541":1}}],["ladder",{"2":{"17":1}}],["lang",{"2":{"531":2}}],["languages",{"2":{"162":1}}],["language",{"2":{"8":1,"13":1}}],["lag",{"2":{"466":1}}],["laid",{"2":{"189":1,"192":1}}],["latter",{"2":{"185":1,"423":1}}],["latency",{"2":{"629":4}}],["latestblock",{"2":{"207":6}}],["latestblocknumber",{"2":{"151":4,"159":4}}],["latestrollupheight",{"2":{"135":2,"137":6}}],["latest",{"2":{"93":1,"94":1,"102":1,"135":2,"149":1,"206":3,"207":1,"243":2,"278":1,"307":6,"347":2,"403":1,"442":2,"465":1,"466":2,"526":1,"539":12,"540":1,"559":6,"560":1,"562":1,"577":6,"579":6,"580":6,"588":1,"620":1,"627":1,"701":1}}],["later",{"2":{"40":1,"263":1,"265":1,"351":1,"368":1,"373":1,"409":1,"424":1,"425":1,"539":1,"646":1,"707":1,"709":1}}],["large",{"2":{"169":1,"270":1,"283":2,"378":1}}],["larger",{"2":{"40":1,"42":1,"283":1,"417":1,"502":2,"509":1}}],["layout",{"2":{"136":1,"146":2,"166":1}}],["layers",{"2":{"289":1,"415":2,"421":1,"422":7,"440":2}}],["layer",{"0":{"415":1},"1":{"416":1,"417":1,"418":1,"419":1,"420":1,"421":1},"2":{"89":1,"105":2,"109":1,"176":2,"201":1,"206":1,"212":1,"213":5,"214":1,"228":3,"229":3,"255":1,"277":1,"280":1,"285":1,"290":5,"299":1,"301":1,"343":3,"375":1,"380":2,"381":1,"415":3,"417":2,"419":4,"421":4,"422":6,"424":1,"425":1,"438":1,"458":1,"477":2,"478":1,"587":1,"708":2}}],["label",{"2":{"503":2,"637":1}}],["labels",{"2":{"503":4}}],["labeled",{"2":{"53":1}}],["labs",{"2":{"24":1,"30":1,"43":2,"44":2,"45":2,"46":5,"47":10,"65":1,"79":2,"91":1,"212":1,"449":1,"465":1,"624":1}}],["lastly",{"2":{"503":1}}],["lastblock",{"2":{"137":6}}],["last",{"2":{"27":1,"30":1,"41":1,"43":1,"326":6,"499":2,"509":1}}],["laws",{"2":{"34":11}}],["law",{"2":{"25":1,"34":1}}],["launching",{"2":{"42":1}}],["launch",{"0":{"21":1,"86":1},"2":{"85":1,"86":1,"213":1,"423":1,"450":2,"458":1,"626":1}}],["ledger",{"2":{"402":1}}],["len",{"2":{"137":8,"139":6,"151":2,"155":4,"157":2,"158":4,"159":2,"161":12,"392":4}}],["length",{"2":{"134":2,"260":6}}],["left",{"2":{"135":1,"177":2,"185":1,"189":1,"192":1,"412":1}}],["lets",{"2":{"232":1}}],["let",{"2":{"47":3,"85":1,"120":1,"131":1,"251":1,"257":1,"260":2,"262":1,"266":1,"268":2,"270":1,"308":1,"309":1,"312":1,"322":1,"323":1,"326":1,"365":1,"367":3,"370":1,"373":2,"565":1}}],["leveraging",{"2":{"257":1,"634":1}}],["leverages",{"2":{"127":1}}],["leverage",{"2":{"43":1,"44":1,"45":1,"681":1}}],["leveldb",{"2":{"610":1}}],["levels",{"2":{"295":1}}],["level",{"2":{"7":1,"131":1,"133":1,"146":1,"295":1,"405":1,"451":1,"465":1,"504":1,"629":2}}],["lemonade",{"2":{"41":1}}],["legacy",{"2":{"494":3,"495":2}}],["legal",{"2":{"30":1,"33":1,"34":1}}],["legended",{"2":{"34":1}}],["legend",{"2":{"34":3}}],["legends",{"2":{"34":2}}],["lessons",{"2":{"38":1}}],["less",{"2":{"24":1,"25":1,"167":1,"181":1,"229":1,"442":1,"671":1}}],["leave",{"2":{"712":1}}],["leaves",{"2":{"127":2,"155":6,"158":2,"424":1}}],["leaving",{"2":{"671":1}}],["leapwallet",{"2":{"452":1}}],["leap",{"0":{"399":1,"433":1,"434":1},"1":{"434":1,"435":1,"436":1},"2":{"397":1,"399":3,"426":1}}],["leafs",{"2":{"419":1}}],["leaf",{"2":{"158":2}}],["least",{"2":{"24":1,"41":1,"45":1,"80":1,"82":1,"406":1,"416":1,"418":1,"425":2,"507":2,"588":1,"601":1,"649":1}}],["lead",{"2":{"14":1,"15":1,"43":1}}],["leaders",{"2":{"7":1,"9":2,"11":2,"12":1,"13":1}}],["learned",{"2":{"38":1,"122":1,"254":1,"526":1,"533":1}}],["learning",{"2":{"3":1,"8":1,"46":1,"48":2,"256":1,"291":1}}],["learn",{"2":{"3":1,"46":1,"52":1,"56":1,"71":1,"79":2,"84":1,"92":1,"105":1,"111":1,"113":1,"116":1,"134":1,"146":1,"214":1,"229":1,"243":1,"248":1,"277":1,"279":1,"280":1,"288":1,"289":1,"307":1,"320":1,"365":2,"391":1,"393":1,"426":1,"438":1,"444":2,"447":2,"448":1,"459":1,"460":1,"469":1,"489":1,"511":1,"525":1,"532":1,"539":1,"540":2,"559":1,"560":1,"575":1,"633":1,"672":1,"681":1,"696":1}}],["locked",{"2":{"398":2,"450":1}}],["lock",{"2":{"233":1}}],["located",{"2":{"136":1,"259":1}}],["locate",{"2":{"134":1}}],["location",{"0":{"553":1},"2":{"24":1,"27":1,"41":2,"65":1,"134":2,"136":1,"137":1,"170":1,"171":1,"172":3,"185":1,"316":1,"357":1,"488":4,"515":1,"553":10,"565":1,"650":1,"659":1}}],["localroot",{"2":{"244":1}}],["localised",{"2":{"157":1}}],["locally",{"0":{"263":1},"2":{"79":1,"257":1,"262":1,"263":2,"264":1,"384":1,"517":1,"547":1}}],["localhost",{"2":{"75":2,"90":1,"91":3,"119":3,"120":3,"150":2,"151":2,"251":2,"263":2,"264":6,"267":6,"269":1,"270":1,"313":1,"347":2,"348":2,"360":2,"367":1,"370":2,"502":2,"503":4,"541":3,"552":3,"607":2,"712":2}}],["local",{"0":{"199":1,"356":1,"378":1,"512":1,"513":1},"1":{"513":1,"514":2,"515":2,"516":2,"517":2,"518":1,"519":1,"520":1,"521":1,"522":1,"523":1,"524":1,"525":1},"2":{"2":1,"40":1,"42":1,"43":2,"44":2,"45":3,"46":2,"53":1,"69":1,"71":1,"79":1,"85":1,"91":1,"92":1,"120":1,"199":1,"238":1,"244":1,"251":3,"263":4,"266":1,"305":21,"346":1,"352":1,"355":2,"356":1,"360":2,"370":1,"378":1,"491":2,"511":1,"516":1,"519":2,"520":4,"525":1,"526":1,"533":1,"540":1,"547":1,"560":1,"579":2,"580":2,"586":21,"688":1,"721":1}}],["loading",{"2":{"373":2}}],["loadenv",{"2":{"373":7,"374":2}}],["loads",{"2":{"268":2,"373":2}}],["load",{"2":{"69":1,"237":1,"373":6,"392":2,"609":1,"610":1,"611":1}}],["lookups",{"2":{"260":2}}],["looks",{"2":{"84":1,"319":1}}],["looking",{"2":{"48":1,"79":2,"91":1,"159":1,"214":1,"348":1,"574":1,"575":1}}],["look",{"2":{"46":1,"90":1,"102":1,"116":2,"121":2,"131":1,"133":1,"151":2,"159":4,"161":2,"243":1,"244":1,"248":2,"253":2,"309":1,"316":1,"319":1,"322":1,"329":1,"347":1,"348":1,"349":1,"367":1,"416":1,"421":1,"425":1,"522":1,"542":2,"557":1,"579":1,"580":1,"650":1,"660":1}}],["loop",{"2":{"30":1}}],["log",{"2":{"263":1,"351":4,"372":2,"373":12,"503":1,"522":2,"556":1,"557":2,"629":2,"721":2,"727":2}}],["logger",{"2":{"351":2}}],["logged",{"2":{"88":1}}],["logging",{"2":{"93":1,"260":1}}],["logsmake",{"2":{"344":1,"361":1}}],["logsbloom",{"2":{"121":2,"253":2,"347":2}}],["logs",{"0":{"344":1,"361":1,"499":1},"2":{"71":1,"91":1,"121":2,"253":2,"344":2,"346":6,"352":1,"353":1,"361":2,"499":7,"522":2,"557":1,"601":1,"626":1,"712":1,"714":1,"715":1,"716":1,"721":2,"727":2}}],["logos",{"2":{"47":1}}],["logo",{"2":{"44":1,"47":1,"58":1,"399":1}}],["logic",{"2":{"107":2,"110":1,"135":1,"177":1,"209":2,"233":1,"374":1,"421":1,"478":1}}],["logical",{"2":{"43":1}}],["login",{"2":{"78":1,"274":1,"503":1}}],["logistics",{"0":{"39":1,"63":1},"1":{"40":1,"41":1,"42":1,"43":1,"44":1,"45":1,"46":1,"47":1,"48":1},"2":{"36":1,"39":1,"43":1}}],["lost",{"2":{"554":1,"578":1}}],["lose",{"2":{"509":1,"536":1}}],["loses",{"2":{"24":1}}],["loss",{"2":{"31":2,"509":1}}],["lowers",{"2":{"84":1}}],["lower",{"2":{"84":2}}],["low",{"2":{"27":1,"146":1,"255":1,"284":1,"394":1,"398":2,"400":1}}],["lot",{"2":{"27":1,"39":1,"40":1,"57":1}}],["longer",{"2":{"84":1,"283":1,"402":1,"418":1,"458":1,"505":1}}],["long",{"2":{"23":1,"34":1,"85":1,"416":1,"441":1,"442":1,"445":1}}],["love",{"2":{"2":1,"48":1}}],["ll",{"2":{"0":1,"41":1,"56":1,"63":1,"65":1,"68":1,"69":1,"71":1,"76":1,"82":1,"84":1,"85":1,"92":1,"107":1,"120":1,"121":1,"243":1,"244":1,"251":2,"252":1,"253":1,"257":1,"263":2,"265":1,"267":1,"268":2,"269":1,"270":2,"272":1,"312":1,"347":1,"360":1,"365":4,"374":1,"399":1,"400":1,"402":1,"526":1,"553":1,"579":1,"646":1}}],["wget",{"2":{"304":10,"305":8,"585":10,"586":8,"608":2}}],["wrong",{"0":{"556":1},"2":{"556":2}}],["wrap",{"2":{"373":1}}],["wrapped",{"2":{"655":1}}],["wrappers",{"2":{"161":2}}],["wrapper",{"2":{"151":4,"159":2}}],["wrapping",{"0":{"374":1},"2":{"135":1}}],["wraps",{"2":{"135":2}}],["writing",{"2":{"135":2,"143":1}}],["write",{"2":{"39":1,"121":1,"244":1,"253":1,"351":1,"365":1,"519":2,"696":1}}],["written",{"2":{"13":1,"31":2,"213":2,"290":2,"305":1,"586":1}}],["wsl",{"2":{"583":2}}],["ws",{"2":{"119":1,"217":1,"629":4,"677":1}}],["wss",{"0":{"217":1},"2":{"88":6,"89":6,"217":1,"677":1}}],["won",{"2":{"88":1,"89":1,"181":1,"701":1}}],["wonderful",{"2":{"39":1}}],["worry",{"2":{"107":1,"575":1}}],["world",{"2":{"49":1,"365":1,"381":2,"666":1}}],["wordle",{"2":{"381":1}}],["words",{"2":{"129":1,"172":1,"409":1,"422":1}}],["word",{"2":{"42":1,"45":1}}],["workflow",{"2":{"373":1}}],["workshop",{"0":{"59":1,"60":1,"61":1,"62":1}}],["workshops",{"2":{"40":1,"43":1}}],["works",{"0":{"208":1},"2":{"41":2,"94":1,"385":1,"387":1,"388":1,"405":1,"444":1,"459":1,"523":1,"548":1}}],["working",{"0":{"644":1,"645":1},"2":{"36":1,"40":2,"236":2,"261":1,"265":1,"285":1,"307":1,"357":1,"502":1,"559":1,"644":1}}],["work",{"2":{"22":1,"24":1,"41":1,"131":1,"143":1,"167":1,"228":1,"281":1,"351":1,"357":1,"360":1,"419":1,"471":1,"624":1}}],["wouldn",{"2":{"407":1}}],["would",{"2":{"40":1,"45":1,"48":5,"102":1,"131":1,"133":1,"135":9,"136":1,"139":3,"181":2,"183":1,"189":2,"192":1,"204":1,"205":1,"275":1,"297":1,"302":1,"310":1,"316":1,"329":1,"380":1,"398":1,"469":1,"494":1,"512":1,"522":1,"542":1,"552":1,"575":1,"577":2,"603":1,"622":1,"672":1,"685":2}}],["www",{"2":{"17":3}}],["walkthrough",{"0":{"581":1},"1":{"582":1}}],["wallet=",{"2":{"721":2,"727":2}}],["wallets",{"0":{"310":1,"630":1,"661":1},"1":{"662":1},"2":{"265":1,"267":4,"397":1,"426":1,"462":1,"536":1,"630":3}}],["wallet",{"0":{"234":1,"397":1,"427":1,"428":1,"433":1,"489":1,"527":1,"534":1,"536":1,"537":1,"720":1},"1":{"235":1,"236":1,"237":1,"238":1,"239":1,"240":1,"241":1,"242":1,"243":1,"244":1,"398":1,"399":1,"400":1,"429":1,"430":1,"431":1,"432":1,"434":1,"435":1,"436":1,"535":1,"536":1,"537":1},"2":{"31":2,"33":4,"69":1,"71":1,"72":1,"76":1,"78":1,"80":1,"83":3,"85":2,"234":1,"252":1,"258":1,"265":1,"267":1,"268":4,"269":1,"272":1,"274":1,"293":1,"295":1,"297":1,"310":5,"367":1,"398":5,"399":2,"400":2,"452":2,"453":1,"485":1,"493":1,"494":6,"495":2,"496":2,"497":4,"498":2,"500":1,"534":1,"536":3,"537":2,"598":1,"620":1,"631":1,"661":1,"662":2,"720":1,"721":8,"727":4}}],["wagmiclient",{"2":{"267":4}}],["wagmiconfig",{"2":{"267":6}}],["wagmi",{"2":{"265":1,"266":2,"267":4,"268":2}}],["wants",{"2":{"643":1}}],["wantedby=multi",{"2":{"712":2,"714":2,"715":2,"716":2}}],["wanted",{"2":{"522":2,"721":2,"727":2}}],["wanting",{"2":{"295":1}}],["want",{"2":{"196":1,"223":1,"229":1,"232":1,"237":1,"264":1,"267":2,"297":1,"312":2,"379":1,"388":1,"392":4,"399":1,"405":1,"413":1,"494":1,"502":2,"536":1,"548":1,"568":2,"598":2,"599":2,"629":1,"639":2,"643":1,"648":1,"685":1,"701":1,"709":1,"721":2,"727":1}}],["waiting",{"2":{"91":6,"135":1,"137":1,"139":3,"169":2,"392":2}}],["wait",{"2":{"89":1,"107":1,"139":1,"161":2,"268":2,"388":1,"392":2,"435":1,"537":1}}],["waive",{"2":{"31":1}}],["watchopts",{"2":{"160":2}}],["watchdatacommitmentstored",{"2":{"160":3}}],["watch",{"2":{"46":1}}],["watchlist",{"2":{"31":1}}],["water",{"2":{"41":2}}],["way",{"2":{"34":1,"44":1,"82":1,"108":1,"149":1,"167":2,"168":1,"169":1,"170":1,"181":2,"189":1,"267":1,"283":1,"291":1,"296":1,"365":1,"412":1,"415":1,"466":1,"494":1,"503":1,"519":2,"536":1,"540":1,"560":1,"575":2,"655":1}}],["ways",{"2":{"7":1,"163":1,"211":1,"213":1,"231":1,"290":1,"292":1,"389":1,"696":1}}],["warn",{"2":{"505":1}}],["warning",{"0":{"14":1},"2":{"13":1,"14":1,"89":1,"297":1,"302":1,"466":1,"472":1,"552":1,"693":1}}],["warm",{"2":{"45":1}}],["warranties",{"2":{"34":1}}],["wasm",{"2":{"94":1}}],["was",{"0":{"151":1},"2":{"13":1,"34":1,"79":1,"81":1,"91":5,"109":1,"130":1,"131":5,"133":1,"137":1,"139":1,"146":1,"148":2,"151":6,"153":6,"154":1,"159":7,"161":2,"162":1,"166":2,"172":1,"178":1,"184":1,"185":1,"189":1,"192":1,"207":1,"213":1,"229":1,"243":1,"265":1,"290":1,"305":1,"317":1,"319":2,"320":1,"351":1,"373":3,"392":4,"402":1,"419":1,"424":1,"425":1,"515":1,"517":1,"539":1,"586":1,"602":1,"631":1,"657":1,"712":1,"714":1,"715":1,"716":1,"721":1}}],["width",{"2":{"267":2,"268":8}}],["windows",{"2":{"576":1,"583":2}}],["window",{"2":{"243":1,"244":1,"268":6,"394":1,"395":1,"396":1,"398":8,"441":5,"509":1,"623":1}}],["wise",{"2":{"424":1}}],["wish",{"2":{"206":1,"376":1,"384":1,"388":1,"442":1}}],["wisdom",{"2":{"56":1}}],["wiki",{"2":{"9":1}}],["willingness",{"2":{"388":1}}],["willing",{"2":{"293":1,"386":1}}],["will",{"2":{"9":2,"11":1,"12":1,"21":2,"22":2,"23":3,"24":2,"26":2,"27":1,"28":1,"29":1,"31":6,"33":1,"34":8,"40":1,"41":1,"42":2,"43":1,"44":1,"47":1,"48":3,"50":1,"52":1,"53":2,"64":2,"66":1,"71":1,"72":2,"76":1,"77":1,"79":2,"82":1,"83":2,"84":1,"85":2,"86":1,"88":3,"89":3,"90":3,"91":1,"102":1,"103":2,"107":5,"113":1,"119":3,"120":2,"121":2,"122":2,"126":1,"133":1,"134":1,"135":3,"139":1,"148":2,"149":1,"150":1,"151":2,"152":1,"153":5,"155":1,"159":2,"161":4,"163":1,"166":1,"167":4,"169":1,"170":1,"172":4,"173":2,"181":1,"184":1,"185":1,"186":1,"188":1,"189":2,"191":1,"192":3,"201":1,"207":4,"223":1,"231":2,"234":2,"236":1,"237":4,"243":1,"244":2,"245":1,"252":2,"253":2,"254":1,"256":2,"257":1,"263":3,"264":3,"268":4,"270":1,"272":2,"273":1,"283":3,"288":1,"290":1,"297":1,"299":1,"302":1,"304":1,"305":1,"307":2,"308":1,"309":1,"310":2,"311":2,"312":1,"313":1,"314":3,"315":1,"316":1,"317":5,"319":6,"320":3,"322":1,"323":1,"326":3,"329":1,"334":3,"338":1,"347":1,"348":1,"349":1,"352":1,"353":1,"354":1,"355":1,"357":3,"360":3,"367":5,"368":1,"369":1,"370":1,"371":2,"372":1,"373":3,"374":1,"375":2,"376":1,"379":2,"381":1,"383":1,"388":3,"392":4,"394":1,"398":4,"399":1,"400":1,"402":2,"403":1,"405":1,"406":1,"412":1,"413":1,"425":1,"426":1,"429":1,"431":1,"432":1,"436":1,"439":1,"440":1,"441":5,"442":2,"444":1,"449":1,"450":2,"462":1,"466":2,"469":1,"471":2,"477":1,"483":1,"484":1,"485":2,"491":3,"494":1,"503":6,"507":2,"509":8,"511":1,"515":3,"518":1,"519":1,"522":2,"523":1,"524":1,"525":2,"529":3,"530":1,"531":1,"534":1,"538":1,"539":4,"540":3,"542":3,"543":1,"544":3,"548":3,"550":2,"552":1,"553":1,"554":1,"557":1,"559":2,"560":3,"565":1,"571":1,"578":4,"579":1,"580":1,"584":1,"586":1,"592":1,"594":1,"598":1,"599":3,"600":1,"605":1,"610":2,"611":2,"612":1,"620":2,"623":1,"624":1,"625":1,"626":7,"629":1,"630":1,"638":1,"640":1,"641":1,"642":1,"645":2,"646":1,"649":3,"650":3,"654":2,"659":2,"661":2,"667":2,"672":1,"679":1,"685":1,"688":1,"701":1,"703":1,"707":1,"709":2,"712":1,"714":1,"715":2,"716":3,"717":1,"721":1,"727":2}}],["withheld",{"2":{"403":1}}],["withhold",{"2":{"403":1,"425":1}}],["withholding",{"2":{"403":1}}],["withtransportcredentials",{"2":{"392":2}}],["withcancel",{"2":{"373":2,"374":2}}],["withdraw",{"0":{"71":1},"2":{"496":2}}],["withdrawalsroot",{"2":{"347":2}}],["withdrawals",{"2":{"84":1}}],["withdrawal",{"0":{"70":1,"72":1},"1":{"71":1,"72":1},"2":{"24":1,"66":2,"71":2,"72":2}}],["within",{"2":{"10":1,"16":1,"21":1,"24":4,"33":2,"34":1,"40":1,"43":1,"50":1,"65":1,"84":1,"288":1,"312":1,"394":1,"395":1,"438":1,"441":3,"583":1}}],["without",{"2":{"8":1,"31":1,"34":1,"88":1,"107":1,"146":1,"168":1,"231":1,"268":2,"315":1,"351":1,"357":1,"404":1,"405":1,"415":1,"423":1,"442":1,"450":1,"522":1,"575":1,"577":1,"680":1}}],["with",{"0":{"76":1,"78":1,"92":1,"103":1,"104":1,"121":1,"123":1,"132":1,"207":1,"209":1,"213":1,"234":1,"253":1,"274":1,"318":1,"397":1,"398":1,"428":1,"433":1,"471":1,"486":1,"487":1,"499":1,"511":1,"534":1,"578":1,"621":1,"622":1,"663":1,"664":1,"710":1,"712":1,"726":1},"1":{"105":1,"106":1,"107":1,"108":1,"109":1,"110":1,"111":1,"124":1,"125":1,"126":1,"127":1,"128":1,"129":1,"130":1,"131":1,"133":1,"134":1,"135":1,"136":1,"137":1,"138":1,"139":1,"140":1,"141":1,"142":1,"143":1,"208":1,"209":1,"210":1,"211":1,"235":1,"236":1,"237":1,"238":1,"239":1,"240":1,"241":1,"242":1,"243":1,"244":1,"398":1,"399":1,"400":1,"429":1,"430":1,"431":1,"432":1,"434":1,"435":1,"436":1,"512":1,"513":1,"514":1,"515":1,"516":1,"517":1,"518":1,"519":1,"520":1,"521":1,"522":1,"523":1,"524":1,"525":1,"526":1,"527":1,"528":1,"529":1,"530":1,"531":1,"532":1,"533":1,"535":1,"536":1,"537":1,"579":1,"580":1,"711":1,"712":1,"713":1,"714":1,"715":1,"716":1},"2":{"1":1,"2":1,"3":2,"5":1,"14":3,"15":3,"24":2,"25":1,"26":1,"27":2,"30":1,"31":5,"32":6,"33":4,"34":18,"36":2,"37":1,"38":2,"40":5,"42":5,"43":7,"44":2,"45":2,"46":5,"47":3,"48":1,"49":1,"50":1,"53":1,"54":2,"55":1,"61":1,"63":1,"65":1,"76":4,"77":3,"78":1,"79":1,"80":1,"82":1,"84":2,"85":1,"88":2,"89":2,"91":6,"94":5,"101":1,"102":2,"105":1,"106":1,"107":3,"108":2,"109":2,"110":1,"115":2,"116":2,"117":1,"120":2,"121":1,"131":3,"135":3,"139":1,"143":1,"148":1,"155":2,"156":1,"157":1,"158":1,"159":8,"161":2,"162":1,"163":1,"170":1,"173":1,"176":1,"181":1,"201":2,"204":1,"206":4,"207":2,"209":3,"210":1,"212":3,"213":1,"215":2,"217":1,"226":1,"229":1,"232":1,"233":2,"234":1,"237":1,"242":1,"243":3,"244":5,"247":2,"248":2,"249":1,"251":1,"252":2,"253":1,"255":2,"256":1,"260":2,"262":1,"263":3,"264":1,"265":1,"266":1,"267":1,"268":1,"270":2,"272":4,"273":3,"274":1,"275":1,"278":1,"280":1,"281":1,"284":2,"285":2,"289":1,"290":1,"291":1,"292":2,"295":2,"296":1,"307":3,"309":2,"310":4,"311":1,"312":5,"313":1,"314":3,"315":2,"316":1,"317":2,"319":1,"322":1,"329":1,"334":2,"336":1,"337":1,"347":1,"351":4,"352":2,"355":1,"357":2,"360":1,"365":2,"367":1,"369":1,"370":1,"373":3,"374":1,"375":2,"377":2,"378":1,"379":1,"380":2,"381":1,"383":1,"386":1,"388":2,"394":1,"403":1,"407":1,"408":1,"409":1,"414":1,"416":2,"417":1,"418":6,"419":2,"421":1,"422":1,"423":1,"424":4,"425":3,"429":1,"438":1,"439":1,"444":1,"451":1,"459":1,"463":1,"465":1,"466":1,"470":2,"471":5,"473":1,"478":3,"485":2,"487":2,"488":2,"494":6,"496":1,"497":2,"498":1,"499":1,"503":5,"505":1,"506":1,"507":2,"509":1,"515":1,"522":2,"532":1,"537":1,"538":1,"539":1,"540":2,"541":1,"542":2,"544":1,"551":1,"552":1,"553":3,"554":1,"555":1,"559":2,"560":2,"571":1,"573":3,"577":1,"580":1,"583":1,"592":1,"594":1,"598":1,"607":1,"620":2,"622":1,"624":3,"625":1,"628":1,"630":1,"637":1,"639":2,"643":1,"646":1,"648":3,"649":2,"650":2,"652":4,"653":1,"655":1,"660":2,"661":3,"664":1,"665":1,"667":2,"671":1,"679":2,"689":1,"690":2,"691":1,"693":1,"703":1,"706":1,"707":1,"722":1,"726":2}}],["whole",{"2":{"416":1}}],["who",{"0":{"413":1},"2":{"22":1,"24":1,"31":1,"34":1,"43":1,"45":2,"46":3,"65":1,"213":1,"293":1,"297":1,"316":1,"643":1,"650":1,"701":1}}],["whiteboard",{"2":{"40":1}}],["whiteboards",{"2":{"36":1,"40":1}}],["white",{"2":{"34":1}}],["while",{"2":{"19":1,"41":1,"91":1,"116":1,"155":1,"163":1,"206":1,"213":1,"234":1,"248":1,"278":1,"283":1,"290":2,"329":1,"380":1,"417":1,"422":1,"425":1,"442":1,"557":4,"626":1,"667":1,"698":1,"708":1}}],["which",{"0":{"572":1},"1":{"573":1,"574":1},"2":{"8":1,"27":1,"30":3,"31":1,"34":3,"41":1,"47":1,"48":1,"76":1,"83":1,"84":3,"86":1,"88":2,"89":1,"90":1,"95":1,"102":1,"106":1,"107":1,"108":3,"109":3,"110":2,"131":1,"134":1,"135":2,"137":1,"146":2,"148":4,"150":1,"152":1,"153":2,"155":2,"156":1,"157":1,"159":5,"166":1,"172":1,"177":1,"178":1,"182":1,"185":1,"189":1,"191":1,"192":1,"201":1,"203":1,"204":1,"208":1,"213":2,"237":1,"243":1,"267":2,"272":1,"278":1,"288":1,"290":1,"292":1,"296":1,"297":1,"309":1,"317":4,"319":4,"320":1,"322":1,"351":1,"352":1,"353":1,"355":1,"360":1,"367":1,"373":3,"377":1,"379":1,"380":1,"385":1,"403":1,"411":1,"412":1,"422":1,"438":2,"442":1,"444":1,"460":1,"469":1,"483":1,"484":1,"485":1,"494":1,"503":2,"537":1,"539":1,"542":1,"548":1,"550":1,"572":1,"574":1,"577":1,"603":1,"607":2,"610":2,"612":1,"620":2,"624":2,"627":1,"628":1,"638":1,"650":2,"652":1,"653":1,"654":1,"660":3,"671":4,"672":1,"685":1,"690":3,"707":1,"709":1,"712":2,"714":2,"715":2,"716":2,"724":1,"729":1}}],["why",{"0":{"407":1,"412":1},"2":{"13":1,"27":1,"48":5,"65":1,"624":1}}],["whenever",{"2":{"45":1,"255":1,"685":1}}],["when",{"2":{"9":1,"10":1,"42":1,"48":1,"50":1,"65":1,"88":1,"89":2,"107":2,"146":1,"148":1,"151":2,"156":1,"159":2,"161":2,"167":1,"173":1,"177":1,"184":1,"185":2,"213":1,"243":1,"260":2,"263":1,"268":3,"270":1,"277":1,"290":1,"310":1,"312":1,"313":1,"357":1,"373":2,"374":2,"379":1,"386":1,"387":1,"402":2,"403":1,"419":1,"450":1,"494":1,"502":2,"503":2,"541":1,"547":1,"551":1,"552":1,"553":1,"557":2,"579":1,"608":1,"610":2,"625":1,"649":1,"662":1}}],["whereas",{"2":{"84":1}}],["where",{"0":{"411":1},"2":{"8":1,"107":2,"108":1,"136":1,"153":2,"157":1,"196":1,"308":1,"310":1,"312":1,"365":1,"367":1,"368":1,"374":1,"375":1,"380":1,"384":1,"385":1,"386":1,"394":1,"407":1,"418":1,"424":1,"432":1,"436":1,"441":1,"442":3,"473":1,"478":1,"489":1,"531":2,"623":1,"662":1,"666":1,"693":1}}],["whether",{"2":{"2":1,"34":1,"42":1,"394":1,"398":3,"403":1,"409":1,"415":1,"611":1}}],["whatever",{"0":{"223":1},"1":{"224":1,"225":1,"226":1,"227":1,"228":1,"229":1,"230":1,"231":1,"232":1,"233":1},"2":{"256":1,"277":1,"536":1}}],["what",{"0":{"201":1,"206":1,"228":1,"229":1,"288":1,"289":1,"402":1,"403":1,"405":1,"406":1,"408":1,"409":1,"410":1,"414":1},"2":{"8":1,"22":1,"48":4,"59":1,"60":1,"61":1,"71":1,"85":1,"117":1,"122":1,"133":1,"136":1,"139":1,"172":1,"223":1,"249":1,"254":1,"257":1,"275":1,"316":1,"319":1,"329":1,"374":2,"440":1,"465":1,"540":1,"544":1,"560":1,"624":1}}],["went",{"2":{"721":1}}],["we+s5gft6g944xbkvvygqb9oy+u",{"2":{"519":2,"520":2}}],["week",{"2":{"473":1,"693":1}}],["weeks",{"2":{"45":1}}],["weak",{"2":{"409":1,"601":1}}],["weth",{"2":{"91":2}}],["webhook",{"2":{"505":1}}],["web",{"0":{"474":1},"2":{"265":1,"474":1,"503":1}}],["websocket",{"0":{"677":1},"2":{"88":5,"150":2,"151":2,"629":4,"677":1}}],["websites",{"2":{"683":1}}],["website",{"2":{"27":1,"46":1,"88":1,"398":2}}],["web3provider",{"2":{"268":4}}],["web3",{"2":{"48":1,"50":1,"256":1}}],["were",{"2":{"17":1,"31":1,"34":2,"131":5,"153":6,"154":1,"206":1,"375":1,"450":1,"625":1}}],["well",{"2":{"14":1,"30":1,"40":1,"41":2,"65":2,"107":1,"108":1,"233":1,"263":1,"281":1,"293":1,"440":1,"620":1,"628":1}}],["welcoming",{"2":{"7":1,"40":1,"52":1}}],["welcome",{"2":{"0":1,"21":1,"30":1,"37":1,"47":2,"48":1,"49":1,"54":1,"57":1,"113":1,"365":1,"666":1}}],["we",{"2":{"2":1,"4":1,"7":2,"24":1,"30":2,"31":5,"33":9,"40":1,"48":8,"52":2,"65":2,"83":2,"84":1,"88":1,"103":1,"107":1,"117":1,"120":2,"121":2,"122":1,"126":2,"131":1,"134":4,"135":3,"136":1,"139":4,"146":1,"148":6,"150":1,"151":2,"152":1,"153":6,"154":2,"155":6,"158":1,"159":2,"161":4,"162":1,"167":1,"170":1,"171":1,"172":4,"173":5,"184":1,"232":1,"236":1,"243":1,"244":2,"245":1,"249":1,"251":1,"252":3,"253":2,"256":1,"257":1,"260":5,"261":1,"262":2,"263":7,"264":7,"265":2,"266":1,"267":4,"268":8,"269":3,"270":2,"277":1,"278":2,"279":1,"280":1,"282":1,"283":1,"285":1,"299":1,"305":1,"312":1,"317":6,"319":2,"322":1,"326":1,"334":1,"347":1,"357":1,"365":1,"367":2,"368":1,"370":2,"371":1,"372":3,"373":12,"374":3,"375":1,"381":1,"388":1,"392":12,"398":5,"440":1,"465":1,"494":2,"503":1,"505":2,"511":1,"515":1,"525":1,"526":2,"529":4,"534":1,"536":1,"544":1,"548":1,"553":1,"561":2,"566":1,"579":2,"586":1,"592":1,"605":1,"629":1,"630":2,"632":1,"634":1,"635":2,"645":2,"646":1,"667":1,"685":1,"688":1,"706":1,"709":1,"715":1}}],["dbbackend",{"2":{"610":1}}],["d6",{"2":{"419":1}}],["d5",{"2":{"419":2}}],["d4",{"2":{"419":2}}],["d3",{"2":{"419":1}}],["ddc4gfalv4klkv5zh4e16mmo5fpnos1f5wzpds8yk3s0rvs4ulljj13euw+ovdv6q23zuv1shroevk5aptit7bgmzunvc1oikwjtxvbn0bigm6k2znwq78cnst2ez3+nzqq84ds28or",{"2":{"349":2}}],["dns4",{"2":{"470":4}}],["dns",{"2":{"334":1}}],["d85c907ce660878a8203ac74baa147ccc1f87114b45b568b72ad207b62afe45e",{"2":{"326":2}}],["dl",{"2":{"305":8,"586":8}}],["dynamic",{"2":{"232":1,"385":1,"386":1}}],["dcproof",{"2":{"150":2,"151":2}}],["dfcelestia",{"2":{"102":1}}],["drafted",{"2":{"670":1}}],["drawing",{"2":{"56":1}}],["dropped",{"2":{"388":1}}],["drop",{"2":{"380":1,"388":1,"449":1}}],["dropdown",{"2":{"68":1,"269":1}}],["drift",{"2":{"629":4}}],["drinks",{"2":{"41":4}}],["driver",{"2":{"33":1,"351":2}}],["d",{"2":{"31":1,"34":1,"47":2,"89":1,"90":1,"139":4,"268":18,"290":1,"329":1,"335":1,"344":1,"345":1,"349":2,"352":2,"356":1,"357":1,"358":1,"361":1,"364":1,"449":1,"450":2,"511":1,"516":1,"522":2,"602":6}}],["daemon",{"2":{"557":3,"710":1,"712":6,"714":3,"715":3,"716":4,"722":1}}],["daring",{"2":{"519":2}}],["darwin",{"2":{"305":12,"540":1,"560":1,"586":12}}],["dah",{"2":{"326":2,"327":2,"332":2,"373":2}}],["dacd",{"2":{"349":1}}],["dac",{"2":{"204":1,"205":1}}],["dacs",{"0":{"202":1},"1":{"203":1,"204":1,"205":1},"2":{"203":1,"205":2}}],["daverifier",{"0":{"153":1,"161":1},"1":{"154":1,"155":1,"156":1,"157":1,"158":1,"159":1,"160":1,"161":1},"2":{"131":3,"153":1,"161":11,"172":1}}],["daservice",{"2":{"349":2,"352":2}}],["das",{"0":{"416":1},"1":{"417":1,"418":1},"2":{"108":1,"201":1,"328":2,"405":1,"415":2,"416":2,"417":4,"418":1,"425":4,"654":1,"655":1,"716":1}}],["dashboard",{"2":{"33":1,"429":2,"430":1,"432":1,"436":1,"471":1,"503":5,"548":1,"679":2}}],["dashboards",{"2":{"24":1,"444":1,"503":1,"727":1}}],["da",{"0":{"104":1,"107":1,"200":1,"420":1,"471":1,"679":1,"680":1,"688":1},"1":{"105":1,"106":1,"107":1,"108":1,"109":1,"110":1,"111":1,"201":1,"202":1,"203":1,"204":1,"205":1,"206":1,"207":1,"208":1,"209":1,"210":1,"211":1,"421":1,"681":1},"2":{"89":3,"94":1,"103":1,"106":1,"107":1,"110":1,"111":1,"133":1,"135":4,"136":1,"139":1,"176":2,"184":1,"201":2,"206":3,"213":5,"235":1,"255":1,"277":1,"285":2,"290":5,"293":2,"295":2,"297":1,"301":1,"336":1,"349":12,"352":6,"360":11,"415":3,"417":2,"419":4,"421":3,"422":6,"425":5,"470":5,"471":5,"478":1,"565":1,"654":1,"655":1,"679":7,"680":1,"688":2,"692":8,"707":2,"708":1}}],["dao",{"2":{"84":1}}],["dappforge",{"2":{"259":1}}],["dappsys",{"2":{"260":1}}],["dapps",{"2":{"256":1}}],["dapp",{"0":{"73":1,"272":1},"1":{"74":1,"75":1,"76":1,"77":1,"78":1,"273":1,"274":1,"275":1},"2":{"122":1,"222":1,"256":1,"259":3,"264":1,"269":1,"271":1,"286":1}}],["day",{"2":{"45":1,"441":2}}],["days",{"2":{"41":1,"441":2,"509":1,"600":1}}],["damage",{"2":{"31":1}}],["datastring",{"2":{"373":6}}],["datasharelen",{"2":{"134":4}}],["datasharestart",{"2":{"134":4}}],["datafromevmtransactions",{"2":{"351":3}}],["datalen",{"2":{"170":2,"172":1}}],["datahash",{"2":{"151":2,"161":2}}],["datacommitmentstored",{"2":{"379":1}}],["datacommitment",{"2":{"151":2,"159":2}}],["datarootinclusionproof",{"2":{"150":2,"151":2,"159":9,"161":10}}],["dataroot",{"2":{"127":4,"148":2,"151":2,"159":5,"161":2}}],["dataroottupleroot",{"2":{"150":1}}],["dataroottuples",{"2":{"128":5,"129":1}}],["dataroottuple",{"2":{"127":4,"128":2,"129":1,"131":1,"150":1,"151":2,"153":2,"159":7,"161":2}}],["dataavailabilityprovider",{"2":{"107":2}}],["data",{"0":{"128":1,"131":1,"136":1,"142":1,"150":1,"154":1,"160":1,"172":1,"173":1,"202":1,"317":1,"319":1,"328":1,"348":1,"349":1,"376":1,"382":1,"401":1,"402":1,"403":1,"404":1,"405":1,"406":1,"408":1,"409":1,"410":2,"412":1,"413":1,"414":1,"415":1,"416":1,"418":1,"421":1,"425":1,"440":1,"441":1,"555":1,"596":1,"665":1,"678":1,"679":1,"713":1},"1":{"203":1,"204":1,"205":1,"318":1,"377":1,"378":1,"379":1,"383":1,"384":1,"385":1,"386":1,"387":1,"388":1,"389":1,"390":1,"391":1,"392":1,"393":1,"402":1,"403":1,"404":1,"405":1,"406":1,"407":1,"408":1,"409":1,"410":1,"411":1,"412":1,"413":1,"414":1,"416":1,"417":2,"418":2,"419":1,"420":1,"421":1,"441":1,"442":1,"679":1,"680":1,"681":1,"714":1,"715":1,"716":1},"2":{"27":1,"31":2,"43":1,"48":1,"59":1,"79":1,"83":5,"88":3,"89":1,"105":2,"107":10,"108":8,"109":5,"110":6,"127":5,"128":1,"129":1,"130":1,"131":8,"133":2,"134":31,"135":7,"136":7,"137":16,"139":10,"142":1,"146":5,"148":9,"150":6,"151":10,"152":4,"153":14,"154":1,"156":1,"157":1,"158":1,"159":30,"160":1,"161":14,"162":3,"163":4,"165":2,"166":8,"167":3,"168":2,"170":9,"171":1,"172":7,"173":19,"176":2,"177":7,"178":1,"180":2,"181":3,"184":2,"185":8,"188":1,"189":5,"192":5,"201":3,"203":2,"204":4,"205":2,"206":5,"207":2,"210":1,"213":13,"215":1,"229":3,"232":2,"237":1,"255":2,"268":16,"280":3,"281":1,"284":1,"285":1,"289":1,"290":13,"299":2,"301":1,"302":3,"308":1,"317":9,"319":10,"326":2,"334":4,"343":1,"348":1,"350":1,"351":6,"352":1,"357":2,"363":1,"365":3,"373":11,"374":1,"375":1,"380":2,"381":1,"382":2,"390":2,"391":2,"392":8,"393":1,"402":11,"403":7,"404":7,"405":9,"406":2,"407":6,"408":4,"409":6,"410":4,"411":3,"412":9,"413":4,"414":6,"415":5,"416":11,"417":3,"418":11,"419":14,"421":5,"422":2,"423":6,"424":15,"425":9,"438":9,"440":7,"441":7,"442":12,"457":1,"458":1,"470":1,"477":1,"478":3,"503":4,"522":2,"543":1,"544":1,"548":2,"555":1,"556":1,"557":4,"560":1,"573":3,"578":3,"594":1,"596":2,"599":3,"602":19,"605":1,"606":2,"612":1,"613":1,"654":2,"655":1,"667":1,"680":5,"685":1,"696":5,"707":2,"716":1,"721":2,"722":2,"727":2}}],["date",{"2":{"26":1,"34":3,"36":1,"307":1,"559":1}}],["duty",{"2":{"519":2}}],["dubbing",{"2":{"421":1}}],["dumps",{"2":{"236":2}}],["dues",{"2":{"47":4}}],["due",{"2":{"24":1,"31":1,"65":1,"89":1,"255":1,"385":1,"388":1,"415":1,"422":1,"425":1,"550":1}}],["duration",{"2":{"23":3,"28":1}}],["during",{"0":{"37":1},"2":{"15":1,"23":1,"25":1,"34":1,"37":1,"41":1,"44":1,"159":1,"206":1,"351":1,"394":1,"406":1,"509":1,"540":1,"560":1,"568":1,"571":1}}],["double",{"2":{"509":2,"554":1}}],["dos",{"2":{"295":1}}],["domain",{"2":{"334":1}}],["dom",{"2":{"267":2}}],["doing",{"2":{"117":1,"139":1,"155":1,"229":1,"249":1,"267":1,"540":1,"560":1}}],["downside",{"2":{"418":1}}],["downmake",{"2":{"345":1,"362":1}}],["down",{"2":{"236":1,"345":2,"362":1,"387":1,"399":1,"400":1,"565":1,"625":1}}],["downtime",{"2":{"110":1,"255":2,"350":1,"351":1,"509":1}}],["downloads",{"2":{"307":1,"503":1,"559":1,"602":1}}],["downloading",{"0":{"139":1},"2":{"139":1,"304":1,"404":1,"418":2,"585":1}}],["downloaded",{"2":{"86":1,"87":1,"403":1,"442":1}}],["download",{"0":{"86":1},"2":{"85":1,"86":3,"135":1,"139":10,"283":1,"302":1,"303":1,"305":1,"402":3,"404":2,"405":1,"415":2,"416":1,"417":3,"418":1,"419":1,"421":1,"442":1,"479":1,"512":1,"540":1,"560":1,"586":1,"588":1,"592":7,"614":1,"656":1,"688":2,"718":1}}],["do",{"0":{"404":1,"414":1},"2":{"30":3,"31":1,"32":1,"34":2,"48":1,"82":1,"133":1,"158":1,"196":1,"229":1,"237":2,"243":1,"255":1,"262":1,"264":1,"267":1,"270":1,"280":1,"309":1,"310":1,"346":1,"358":1,"367":1,"373":1,"374":1,"376":1,"384":1,"399":1,"404":1,"407":1,"440":1,"441":1,"442":1,"466":1,"471":1,"484":1,"485":1,"503":2,"535":1,"548":1,"550":1,"583":1,"599":1,"607":1,"620":1,"623":1,"624":1,"630":1,"632":1,"639":1,"644":1,"660":1,"688":1,"707":1,"716":1}}],["done",{"2":{"77":1,"91":10,"133":1,"136":1,"152":1,"154":1,"160":2,"173":2,"183":1,"273":1,"352":1,"388":1,"395":1,"480":1,"494":1,"589":1,"615":1,"643":1,"703":1,"719":1}}],["don",{"2":{"27":1,"40":1,"45":1,"46":1,"71":1,"107":2,"231":1,"232":1,"233":1,"297":1,"392":2,"407":1,"412":1,"527":1,"610":1,"632":1,"709":1}}],["docker",{"0":{"241":1,"575":1},"1":{"242":1,"243":1,"244":1,"576":1,"577":1,"578":1,"579":1,"580":1,"581":1,"582":1,"583":1},"2":{"80":2,"89":7,"90":2,"242":2,"243":3,"244":4,"344":1,"346":2,"353":2,"357":1,"360":1,"575":7,"576":4,"578":1,"579":2,"580":2}}],["docs",{"2":{"24":1,"88":1,"102":2,"141":1,"166":1,"207":1,"393":1,"505":1,"508":1,"540":2,"548":1,"560":2,"601":1}}],["documented",{"2":{"160":1}}],["documents",{"2":{"102":2}}],["document",{"2":{"23":1,"38":1,"148":1,"163":1,"267":2,"276":1,"440":1,"507":6,"670":1}}],["documentation",{"0":{"0":1,"140":1},"1":{"1":1,"2":1,"3":1,"4":1,"141":1,"142":1,"143":1},"2":{"0":1,"1":1,"3":1,"4":1,"24":1,"31":3,"37":1,"79":1,"93":1,"130":1,"131":1,"135":2,"136":3,"161":1,"173":2,"180":1,"208":1,"211":1,"279":1,"292":1,"469":1,"489":2,"494":1,"502":2,"532":1,"672":1}}],["doesn",{"0":{"412":1},"2":{"145":1,"172":1,"233":1,"313":1,"337":1,"612":1,"628":1}}],["does",{"0":{"411":1},"2":{"23":1,"27":1,"30":1,"34":3,"108":1,"196":1,"289":1,"351":1,"380":1,"384":1,"386":1,"388":1,"418":1,"424":1,"465":1,"472":1,"515":1,"583":1,"625":1,"690":1,"693":1,"696":1}}],["dir",{"2":{"553":3}}],["direct",{"2":{"470":2,"471":1,"505":1}}],["directories",{"2":{"442":1}}],["directory",{"0":{"3":1,"515":1,"644":1,"645":1},"2":{"68":1,"75":1,"87":1,"92":1,"115":1,"116":2,"126":1,"236":2,"237":6,"244":5,"247":1,"248":2,"260":1,"264":1,"265":3,"266":1,"272":1,"305":1,"307":1,"310":2,"313":1,"344":1,"352":1,"357":1,"361":1,"371":1,"372":1,"486":1,"500":1,"514":1,"515":4,"517":1,"539":1,"556":1,"559":1,"578":4,"586":1,"602":1,"621":1,"644":2,"650":3,"661":2,"663":1}}],["directly",{"2":{"135":1,"139":2,"204":1,"280":1,"315":1,"379":1,"380":1,"384":1,"386":1,"392":2,"398":1}}],["directed",{"2":{"34":1,"620":1,"660":1}}],["dimensional",{"2":{"416":1,"418":4,"421":1,"424":1}}],["digest",{"2":{"155":9,"161":4}}],["digital",{"2":{"8":1,"278":1}}],["dial",{"2":{"151":2,"159":2,"160":2,"161":2,"392":2,"608":1}}],["diagram",{"2":{"148":1,"208":1,"445":1}}],["diagrams",{"2":{"3":1,"146":1}}],["did",{"2":{"48":2}}],["difficulty",{"2":{"40":1,"42":1,"347":2}}],["difference",{"0":{"410":1},"2":{"102":1,"139":1,"173":1,"181":2,"192":1}}],["different",{"2":{"5":1,"39":1,"64":1,"76":1,"88":1,"89":1,"131":1,"139":1,"162":1,"171":2,"177":1,"180":1,"184":1,"195":1,"207":1,"211":1,"218":1,"223":1,"237":1,"272":1,"295":1,"376":3,"388":1,"411":2,"422":1,"450":1,"469":1,"494":1,"503":1,"529":1,"551":2,"553":2,"601":1,"608":1,"672":1,"696":1,"709":1}}],["differing",{"2":{"8":1}}],["dividing",{"2":{"384":1}}],["divided",{"2":{"23":1,"26":1}}],["div",{"2":{"267":4,"268":36,"398":4}}],["dives",{"2":{"48":1,"59":1}}],["diverse",{"2":{"7":1}}],["disables",{"2":{"547":1,"610":1}}],["disable",{"2":{"326":2,"547":2}}],["disability",{"2":{"7":1}}],["disk",{"2":{"303":1,"479":1,"504":1,"588":1,"611":2,"614":1,"656":1,"718":1}}],["disruptions",{"2":{"255":1}}],["disruptive",{"2":{"19":1}}],["distinguish",{"2":{"84":1}}],["distributor",{"2":{"34":1}}],["distributors",{"2":{"34":1}}],["distributing",{"2":{"34":1}}],["distribution",{"2":{"34":2,"278":1,"472":1,"495":2,"496":2,"693":1}}],["distributed",{"2":{"422":1}}],["distribute",{"2":{"22":1}}],["disputes",{"2":{"84":1,"422":1}}],["dispute",{"2":{"84":1,"229":1}}],["displayed",{"2":{"491":1}}],["displaydenom",{"2":{"462":1}}],["displaying",{"2":{"37":1}}],["display",{"0":{"462":1},"2":{"33":1,"84":1,"268":8,"319":1}}],["disparagement",{"2":{"16":1}}],["disqualify",{"2":{"24":1,"31":1}}],["discard",{"0":{"611":1},"2":{"107":1,"109":1,"611":1}}],["disclaimer",{"2":{"71":1}}],["disclosure",{"2":{"34":1}}],["discussed",{"2":{"169":1}}],["discussion",{"2":{"43":2}}],["discussions",{"2":{"37":1,"50":1}}],["discuss",{"2":{"43":1,"46":1,"57":1,"184":1,"440":1}}],["discover",{"2":{"48":1,"285":1}}],["discovering",{"2":{"43":1,"601":1}}],["discounts",{"2":{"40":1}}],["discord",{"0":{"55":1,"473":1},"2":{"11":1,"27":1,"45":2,"53":3,"54":1,"55":2,"271":1,"309":1,"310":2,"367":1,"473":2,"537":1,"630":3,"662":1,"693":2}}],["discretionary",{"2":{"30":1}}],["discretion",{"2":{"25":1,"30":1,"31":1,"34":1}}],["discrepancies",{"2":{"4":1}}],["debug",{"2":{"488":2}}],["debugging",{"2":{"90":1,"488":2}}],["denoted",{"2":{"424":2}}],["denominator",{"2":{"629":4}}],["denomination",{"0":{"463":1,"464":1},"2":{"463":2}}],["denominations",{"0":{"461":1},"1":{"462":1,"463":1,"464":1}}],["denominated",{"2":{"457":1}}],["denominate",{"2":{"85":1}}],["denom",{"2":{"322":2,"323":2,"399":1,"400":1,"494":2,"522":4,"523":2,"525":4,"629":4}}],["deducting",{"2":{"384":1}}],["deducted",{"2":{"384":1}}],["dedicated",{"2":{"24":1,"37":1,"53":1,"212":1,"227":1}}],["demosubmitdata",{"2":{"392":2}}],["demo",{"0":{"133":1},"2":{"131":1,"133":1,"134":1,"135":4,"392":2}}],["demonstration",{"0":{"149":1},"1":{"150":1,"151":1,"152":1},"2":{"309":1,"646":1}}],["demonstrating",{"2":{"8":1,"16":1}}],["demonstrate",{"2":{"44":1,"398":1}}],["degree",{"2":{"108":1,"109":1}}],["defer",{"2":{"160":4,"373":2,"374":2,"392":2}}],["defaultestimategas",{"2":{"386":1,"387":2,"392":2}}],["default",{"2":{"110":2,"232":1,"237":2,"267":2,"268":2,"278":1,"309":1,"313":1,"320":2,"322":1,"334":1,"356":1,"357":2,"367":1,"376":1,"383":1,"385":1,"388":1,"398":2,"441":1,"471":1,"484":1,"486":1,"488":2,"502":2,"503":6,"541":1,"542":1,"550":1,"551":1,"552":1,"565":1,"566":1,"568":2,"571":1,"596":3,"599":4,"600":1,"607":1,"610":1,"611":1,"621":1,"629":4,"652":1,"663":1,"680":1,"689":1,"690":1,"691":1}}],["defaults",{"2":{"84":1,"309":1,"367":1,"386":1,"484":1,"494":1,"610":1}}],["definitions",{"2":{"342":1,"359":1,"450":2}}],["defining",{"0":{"134":1},"2":{"135":1}}],["defines",{"2":{"351":1,"727":1}}],["define",{"2":{"43":1,"134":2,"152":1,"173":1,"177":2,"185":1,"189":1,"192":1,"232":1,"691":1}}],["defined",{"2":{"30":3,"31":1,"34":6,"84":1,"135":1,"155":1,"167":1,"169":1,"170":1,"192":1,"450":2,"625":1,"727":1}}],["defi",{"2":{"43":1}}],["deeper",{"2":{"451":1}}],["deepen",{"2":{"65":1}}],["deep",{"2":{"48":1,"59":1}}],["deemed",{"2":{"13":1,"34":3}}],["deem",{"2":{"9":1,"12":1,"32":1}}],["decline",{"2":{"519":2}}],["declare",{"2":{"237":1,"243":1,"310":1,"485":1,"529":1,"551":1,"620":1,"660":1}}],["decreasing",{"2":{"455":1}}],["decreases",{"2":{"445":1}}],["decentralised",{"0":{"446":1,"460":1},"1":{"447":1,"448":1},"2":{"460":2}}],["decentralized",{"2":{"205":1,"275":1,"289":1}}],["decentralization",{"0":{"203":1}}],["decouple",{"2":{"422":1}}],["decoupling",{"2":{"232":1,"233":1,"289":1,"423":1}}],["decodestring",{"2":{"373":4}}],["decode",{"2":{"365":1,"373":2,"375":1,"392":2}}],["decoder",{"2":{"191":1}}],["decoded",{"2":{"155":4,"319":1,"351":1}}],["decoding",{"2":{"156":1,"373":4,"629":4}}],["decide",{"2":{"610":1}}],["deciding",{"0":{"572":1},"1":{"573":1,"574":1}}],["decimals",{"2":{"84":1,"267":2,"399":1,"400":1,"455":1}}],["decisions",{"2":{"9":1}}],["deck",{"2":{"44":2}}],["deal",{"2":{"425":1}}],["dealing",{"2":{"418":1}}],["dealer",{"2":{"34":1}}],["deadline",{"2":{"24":2,"26":1}}],["derparam",{"2":{"373":2}}],["derivation",{"2":{"351":2,"629":4}}],["derivative",{"2":{"34":2}}],["derive",{"2":{"351":1}}],["derogatory",{"2":{"8":1}}],["descendants",{"2":{"419":2}}],["desc",{"2":{"337":2}}],["descriptor",{"2":{"557":1}}],["descriptors",{"2":{"557":1}}],["description=celestia",{"2":{"712":2,"714":2,"715":2,"716":2}}],["descriptions",{"2":{"84":2}}],["description",{"0":{"50":1},"2":{"42":1,"44":1,"47":3,"84":1,"385":1,"449":1,"494":6,"495":2,"541":1,"552":1,"625":1}}],["describing",{"2":{"421":1}}],["describes",{"2":{"134":2,"592":1,"627":1,"722":1}}],["described",{"2":{"21":1,"30":1,"31":1,"34":1,"156":1,"185":1,"424":3,"449":1,"450":1}}],["deserialization",{"0":{"189":1},"2":{"189":1,"192":1}}],["deserialize",{"2":{"188":1}}],["deserialized",{"2":{"107":1,"189":2}}],["desktop",{"2":{"89":1,"575":1,"576":1,"578":1}}],["destination",{"2":{"71":1,"537":1,"627":1}}],["desired",{"2":{"82":1,"208":1,"305":1,"307":1,"334":1,"435":1,"539":1,"559":1,"586":1}}],["desire",{"2":{"44":1}}],["designs",{"2":{"195":1}}],["design",{"0":{"548":1},"2":{"169":1,"205":1,"233":1,"380":1,"503":1,"544":1,"548":1,"571":1}}],["designate",{"2":{"37":1}}],["designated",{"2":{"30":2}}],["designed",{"2":{"21":1,"49":1,"50":1,"57":1,"205":1,"294":1,"418":1,"685":2,"701":1}}],["delay",{"2":{"629":4,"634":4}}],["delayedcelestia",{"2":{"529":1}}],["delayed",{"2":{"511":2,"522":7,"529":1}}],["delays",{"2":{"84":1}}],["delta",{"2":{"353":1}}],["deleting",{"2":{"490":2}}],["deleted",{"2":{"578":1}}],["delete",{"2":{"235":1,"490":2,"556":1,"578":1,"605":1}}],["delegate",{"0":{"497":1,"721":1},"2":{"497":3,"509":1,"721":5}}],["delegated",{"2":{"26":2,"231":1,"509":1,"523":4,"727":1}}],["delegating",{"2":{"451":1,"459":1}}],["delegation=1000000",{"2":{"727":3}}],["delegation=1",{"2":{"500":2}}],["delegations",{"2":{"21":1,"28":1}}],["delegation",{"0":{"18":1,"20":1},"1":{"19":1,"20":1,"21":2,"22":2,"23":2,"24":2,"25":2,"26":1,"27":1,"28":1,"29":1},"2":{"19":1,"23":4,"24":4,"25":1,"27":2,"444":2,"488":2}}],["delegators",{"2":{"444":1,"509":2}}],["delegator",{"2":{"426":1,"444":1}}],["delivering",{"2":{"65":1}}],["delivery",{"2":{"41":1}}],["delivers",{"2":{"19":1}}],["deprecated",{"2":{"297":1,"302":1,"367":2,"552":1}}],["depth",{"2":{"84":1,"445":1}}],["depositing",{"2":{"93":1}}],["deposited",{"2":{"91":2}}],["deposit",{"0":{"67":1,"69":1},"1":{"68":1,"69":1},"2":{"66":2,"69":2,"91":5,"494":5,"495":2}}],["dependent",{"2":{"143":1,"386":1}}],["dependency",{"0":{"339":1},"1":{"340":1,"341":1},"2":{"126":2,"264":1}}],["dependencies",{"0":{"74":1,"246":1,"304":1,"366":1,"481":1,"585":1,"590":1,"616":1,"641":1},"1":{"367":1,"368":1,"369":1},"2":{"87":1,"266":1,"304":1,"311":1,"340":1,"342":1,"354":1,"359":1,"366":1,"481":1,"572":1,"575":1,"590":1,"616":1,"641":1,"657":1}}],["depend",{"2":{"89":1,"95":1,"205":1,"403":1,"406":1}}],["depending",{"2":{"40":1,"41":1,"65":1,"169":1,"267":2,"385":1,"469":1,"530":1,"540":1,"560":1,"600":1,"601":1,"610":1,"672":1,"685":1}}],["depends",{"2":{"34":1,"671":1}}],["dependable",{"2":{"19":1}}],["deployer",{"2":{"120":2,"252":2}}],["deployed",{"0":{"210":1},"2":{"48":1,"84":2,"85":1,"89":1,"91":3,"120":3,"196":1,"198":1,"199":1,"206":1,"208":3,"210":2,"252":3,"260":2,"263":2,"264":2,"270":1,"297":1,"377":1,"466":1,"592":1,"625":1}}],["deployhelper",{"2":{"98":1,"99":1}}],["deploy",{"0":{"61":1,"73":1,"79":1,"85":1,"112":1,"197":1,"198":1,"245":1,"338":1,"354":1,"482":1,"722":1},"1":{"74":1,"75":1,"76":1,"77":1,"78":1,"80":1,"81":1,"82":1,"83":1,"84":1,"85":1,"86":1,"87":1,"88":1,"89":1,"90":1,"91":1,"92":1,"93":1,"94":1,"95":1,"96":1,"97":1,"98":1,"99":1,"113":1,"114":1,"115":1,"116":1,"117":1,"118":1,"119":1,"120":1,"121":1,"122":1,"246":1,"247":1,"248":1,"249":1,"250":1,"251":1,"252":1,"253":1,"254":1,"339":1,"340":1,"341":1,"342":1,"343":1,"344":1,"345":1,"346":1,"347":1,"348":1,"349":1,"350":1,"351":1,"352":1,"353":1,"355":1,"356":1,"357":1,"358":1,"359":1,"360":1,"361":1,"362":1,"363":1,"364":1,"483":1,"484":1,"485":1,"486":1,"487":1,"723":1,"724":1,"725":1,"726":1},"2":{"61":1,"73":2,"74":1,"75":1,"83":1,"84":2,"85":3,"91":4,"101":1,"105":1,"111":1,"113":1,"120":2,"122":2,"196":1,"197":1,"198":1,"201":1,"212":1,"217":1,"222":2,"226":1,"227":1,"231":2,"245":1,"251":2,"252":1,"254":1,"262":1,"263":1,"264":1,"272":3,"284":1,"286":4,"289":1,"354":1,"364":1,"380":2,"466":1,"685":1,"701":1,"703":1}}],["deployments",{"0":{"95":1,"98":1,"99":1,"196":1},"1":{"96":1,"97":1,"197":1,"198":1,"199":1},"2":{"89":1,"95":2,"96":1,"97":1,"201":1,"207":1,"376":1}}],["deployment",{"0":{"75":1,"83":1,"84":1,"262":1,"355":1},"1":{"76":1,"356":1,"357":1,"358":1},"2":{"30":2,"75":1,"79":2,"83":3,"84":4,"85":1,"120":1,"208":1,"252":1,"262":1,"272":1,"364":1}}],["deploying",{"0":{"118":1,"231":1,"250":1,"263":1,"264":1,"272":1,"364":1},"1":{"119":1,"120":1,"251":1,"252":1,"273":1,"274":1,"275":1},"2":{"3":1,"61":1,"79":1,"82":1,"113":1,"223":1,"231":2,"258":1,"262":1,"264":1,"269":1,"458":2,"666":1,"698":1}}],["deter",{"2":{"509":1}}],["deterministic",{"2":{"422":1}}],["determining",{"2":{"12":1,"385":1}}],["determined",{"2":{"137":1,"386":1,"671":3}}],["determines",{"2":{"84":1}}],["determine",{"2":{"33":1,"36":1,"41":2,"43":1,"48":1,"450":1,"469":1,"672":1,"685":1}}],["detecting",{"2":{"601":1}}],["detects",{"2":{"398":1}}],["detected",{"2":{"139":1}}],["detect",{"2":{"33":1,"135":1}}],["details",{"0":{"27":1,"166":1,"171":1,"467":1,"523":1,"524":1,"668":1,"686":1},"1":{"468":1},"2":{"21":1,"47":1,"69":2,"71":1,"72":2,"85":1,"94":1,"146":1,"148":1,"149":1,"166":1,"167":1,"173":1,"180":1,"181":1,"185":1,"189":1,"192":4,"195":1,"208":1,"269":1,"297":1,"298":1,"302":1,"386":1,"416":1,"419":1,"421":2,"425":1,"435":1,"455":1,"509":1,"519":1,"523":1,"524":1,"552":1}}],["detailed",{"2":{"2":1,"22":1,"47":1,"79":1,"438":1,"494":1}}],["devs",{"2":{"449":1}}],["devnpm",{"2":{"270":1}}],["devnetl1",{"2":{"353":1}}],["devnet",{"0":{"73":1,"120":1,"338":1,"342":1,"343":1,"344":1,"345":1,"354":1,"356":1,"359":1,"360":1,"361":1,"362":1,"363":1,"465":1,"472":1,"512":1,"513":1,"514":1,"516":1,"518":1,"522":1,"523":1,"524":1,"525":1,"701":1,"702":1},"1":{"74":1,"75":1,"76":1,"77":1,"78":1,"339":1,"340":1,"341":1,"342":1,"343":2,"344":2,"345":2,"346":2,"347":1,"348":1,"349":1,"350":1,"351":1,"352":1,"353":1,"355":1,"356":1,"357":1,"358":1,"359":1,"360":1,"361":1,"362":1,"363":1,"364":1,"466":1,"467":1,"468":1,"469":1,"470":1,"471":1,"472":1,"473":2,"474":2,"475":1,"476":1,"513":1,"514":2,"515":2,"516":2,"517":2,"518":1,"519":2,"520":2,"521":2,"522":2,"523":2,"524":2,"525":2,"702":1},"2":{"74":1,"111":1,"113":1,"114":1,"119":3,"122":1,"251":1,"286":2,"290":1,"307":1,"309":1,"310":1,"338":1,"343":2,"344":3,"345":4,"346":1,"352":2,"353":1,"354":1,"355":2,"356":1,"360":5,"361":3,"362":3,"363":3,"398":2,"399":2,"465":1,"466":1,"469":1,"470":1,"471":1,"473":1,"484":1,"507":2,"511":1,"514":1,"515":1,"516":1,"517":4,"525":1,"526":1,"529":1,"533":1,"566":1,"620":2,"625":1,"660":1,"662":1,"685":1,"701":2,"709":1,"724":1}}],["devcd",{"2":{"75":1,"272":1}}],["devrel",{"2":{"41":1,"43":1}}],["developing",{"2":{"288":1,"620":1}}],["development",{"0":{"256":1,"584":1},"1":{"257":1,"258":1,"259":1,"260":1,"261":1,"262":1,"263":1,"264":1,"265":1,"266":1,"267":1,"268":1,"269":1,"270":1,"271":1,"585":1,"586":1},"2":{"79":1,"84":2,"231":1,"265":1,"288":1,"304":2,"449":2,"559":1,"584":2,"685":1,"698":1}}],["developer",{"2":{"24":1,"45":1,"68":1,"223":1,"292":1,"396":1,"548":1,"639":1}}],["developers",{"2":{"3":1,"84":1,"105":2,"201":2,"207":1,"212":1,"214":2,"215":1,"223":1,"225":1,"232":1,"233":3,"293":1,"397":1,"438":1,"440":1,"442":1,"449":1,"457":2,"458":3,"465":1,"466":1,"471":1,"685":1,"701":1,"703":1}}],["develop",{"2":{"24":1,"44":1,"341":3}}],["devops",{"2":{"24":1,"547":1}}],["dev",{"2":{"1":1,"75":1,"77":4,"102":1,"159":2,"270":1,"272":1,"273":4,"304":4,"585":4,"712":2,"714":2,"715":2,"716":2}}],["devyarn",{"2":{"1":1}}],["yconfirm",{"2":{"727":1}}],["ycelestia",{"2":{"496":1,"498":1}}],["ycd",{"2":{"115":1,"247":1}}],["yparity",{"2":{"348":2}}],["yum",{"2":{"304":4,"585":4}}],["ysudo",{"2":{"304":3,"585":3}}],["y",{"2":{"115":1,"247":1,"304":3,"496":1,"498":1,"522":9,"585":3,"727":4}}],["yamlda",{"2":{"360":1}}],["yamlversion",{"2":{"244":1}}],["yamlextra",{"2":{"89":1}}],["yarncd",{"2":{"265":1}}],["yarn",{"2":{"1":2,"75":4,"87":1,"91":2,"265":1,"266":1,"272":4}}],["ymlglobal",{"2":{"503":1}}],["yml",{"2":{"89":1,"244":2,"353":1,"357":1,"360":1,"503":3,"504":1}}],["yet",{"2":{"91":2,"137":1,"210":1,"337":1,"380":1,"561":1,"600":1,"626":1}}],["yesterday",{"2":{"499":2}}],["yes",{"2":{"48":1,"444":1,"494":1,"507":2,"522":1}}],["years",{"2":{"31":1,"698":1}}],["year",{"2":{"25":1,"26":1,"34":4,"445":2,"450":11,"455":2}}],["youth",{"2":{"519":2}}],["youtube",{"2":{"40":1,"46":2}}],["yours",{"2":{"515":1}}],["yourself",{"2":{"34":1,"503":2}}],["yourprivatekey",{"2":{"91":3}}],["your",{"0":{"66":1,"67":1,"68":1,"69":1,"70":1,"71":1,"72":1,"73":1,"83":1,"84":1,"85":1,"86":2,"87":1,"89":1,"90":1,"91":1,"112":1,"116":1,"117":1,"118":1,"121":1,"243":1,"248":1,"249":1,"250":1,"253":1,"322":1,"324":1,"325":1,"355":1,"372":1,"427":1,"429":1,"431":1,"432":1,"434":1,"436":1,"480":1,"520":1,"522":1,"528":1,"531":1,"553":1,"554":1,"562":1,"615":1,"651":1,"652":1,"657":1,"710":1,"728":1},"1":{"67":1,"68":2,"69":2,"70":1,"71":2,"72":2,"74":1,"75":1,"76":1,"77":1,"78":1,"113":1,"114":1,"115":1,"116":1,"117":1,"118":1,"119":2,"120":2,"121":1,"122":1,"251":1,"252":1,"356":1,"357":1,"358":1,"481":1,"616":1,"658":1,"711":1,"712":1,"713":1,"714":1,"715":1,"716":1},"2":{"2":5,"5":1,"24":1,"25":1,"27":5,"30":5,"31":5,"32":7,"33":9,"34":15,"35":1,"40":4,"41":7,"42":6,"43":11,"44":14,"45":15,"46":7,"47":8,"48":1,"50":1,"53":1,"55":1,"56":2,"57":1,"63":1,"65":3,"66":2,"68":2,"69":7,"71":7,"72":4,"73":1,"74":1,"75":3,"76":2,"77":2,"78":2,"80":1,"82":2,"83":3,"84":11,"85":13,"86":8,"87":1,"88":6,"89":16,"90":3,"91":20,"92":5,"113":1,"114":1,"115":1,"116":2,"117":2,"119":1,"121":1,"122":2,"149":1,"198":1,"208":1,"209":4,"217":1,"226":1,"227":1,"229":1,"234":1,"237":4,"239":1,"242":1,"243":2,"244":3,"246":1,"247":1,"248":2,"249":2,"252":2,"253":1,"258":4,"264":2,"266":1,"268":6,"269":2,"270":3,"271":3,"272":6,"273":2,"274":2,"290":1,"291":1,"296":1,"304":3,"305":3,"308":2,"309":2,"310":6,"312":3,"313":1,"314":3,"315":4,"316":1,"319":3,"324":1,"325":1,"329":1,"330":1,"334":2,"335":1,"337":1,"338":1,"349":1,"353":2,"354":1,"355":1,"357":4,"360":3,"367":5,"368":2,"370":1,"395":1,"399":1,"400":1,"430":1,"431":1,"435":1,"444":1,"452":1,"453":1,"465":1,"469":1,"471":3,"477":1,"484":3,"485":3,"488":4,"491":1,"496":1,"497":1,"498":1,"499":7,"500":2,"501":1,"502":3,"503":3,"505":2,"511":1,"516":1,"517":2,"519":5,"522":1,"525":1,"527":1,"528":1,"529":1,"531":1,"536":2,"537":2,"539":1,"540":3,"541":3,"542":2,"543":2,"544":4,"548":1,"550":2,"552":3,"553":6,"554":5,"557":1,"559":1,"560":3,"561":2,"562":2,"568":2,"575":2,"577":2,"578":2,"579":1,"583":1,"584":1,"585":1,"586":3,"588":1,"592":2,"594":2,"596":1,"597":1,"598":1,"599":2,"603":2,"606":1,"608":1,"610":1,"611":1,"616":1,"620":7,"630":1,"639":1,"643":1,"645":1,"646":1,"648":2,"649":3,"652":3,"653":2,"659":2,"660":4,"661":3,"662":1,"665":1,"672":1,"679":3,"681":2,"685":1,"709":2,"712":1,"721":2,"724":1,"727":9,"728":2,"729":1}}],["you",{"2":{"0":2,"2":1,"4":1,"24":3,"26":1,"27":6,"30":6,"31":23,"32":9,"33":4,"34":29,"35":4,"36":1,"39":1,"40":4,"41":7,"42":4,"43":3,"45":1,"46":3,"47":11,"48":8,"53":2,"54":1,"56":2,"57":2,"60":1,"61":1,"63":2,"64":1,"65":4,"68":1,"69":3,"71":3,"72":3,"75":1,"76":8,"77":3,"78":1,"79":4,"82":1,"83":1,"84":2,"85":5,"86":1,"87":1,"89":9,"90":2,"91":4,"92":1,"102":4,"103":1,"107":5,"108":2,"109":1,"113":1,"116":1,"119":1,"120":2,"122":2,"126":1,"134":1,"135":2,"148":2,"161":1,"166":1,"196":4,"199":2,"207":1,"210":1,"215":1,"217":2,"222":2,"223":5,"225":1,"229":3,"234":3,"236":1,"237":7,"238":1,"243":4,"244":2,"248":1,"251":1,"252":2,"254":3,"256":4,"257":2,"260":1,"263":2,"264":3,"267":2,"268":2,"270":6,"271":2,"272":10,"273":3,"274":1,"280":1,"290":2,"291":2,"297":2,"302":4,"304":1,"305":1,"307":3,"309":4,"310":11,"311":3,"312":6,"313":1,"314":3,"315":3,"316":2,"317":1,"319":3,"320":3,"323":1,"329":3,"330":1,"334":1,"335":2,"336":2,"337":2,"338":1,"344":1,"345":1,"346":2,"348":1,"352":1,"353":1,"354":1,"355":2,"356":2,"357":9,"358":2,"360":4,"361":1,"364":1,"365":5,"367":7,"370":4,"373":6,"374":6,"375":1,"376":3,"379":1,"380":2,"381":2,"387":1,"398":2,"399":6,"400":4,"402":1,"426":3,"432":2,"436":2,"444":1,"462":1,"465":1,"469":3,"470":2,"472":1,"473":1,"475":1,"484":3,"485":8,"487":1,"489":1,"491":3,"494":2,"496":1,"497":2,"498":1,"499":2,"500":3,"502":6,"503":10,"505":3,"511":1,"512":1,"515":1,"516":2,"517":2,"518":2,"519":3,"522":4,"523":1,"524":1,"525":3,"526":1,"527":1,"529":2,"530":3,"531":4,"532":1,"533":1,"534":1,"535":1,"536":4,"537":2,"538":1,"539":4,"540":3,"541":3,"542":3,"543":1,"544":1,"547":1,"548":2,"550":1,"552":5,"553":5,"554":2,"556":2,"557":2,"559":4,"560":3,"561":3,"562":2,"564":1,"565":1,"568":3,"569":1,"572":2,"573":3,"574":2,"575":5,"577":4,"578":10,"579":3,"580":1,"583":1,"585":1,"586":1,"587":1,"588":3,"592":5,"594":1,"598":3,"599":3,"601":2,"603":2,"606":1,"607":2,"608":1,"610":3,"611":2,"612":1,"616":1,"617":1,"620":9,"622":2,"626":2,"629":3,"630":1,"631":1,"632":2,"633":1,"634":3,"639":3,"640":1,"641":1,"642":1,"643":2,"644":3,"645":1,"646":1,"648":2,"649":5,"650":4,"652":6,"653":2,"654":2,"660":1,"661":6,"662":3,"665":1,"672":4,"679":2,"680":1,"681":2,"682":1,"685":8,"688":3,"693":2,"694":1,"696":3,"700":1,"706":2,"707":3,"708":2,"709":5,"711":1,"712":2,"714":2,"715":2,"716":1,"717":2,"721":8,"722":2,"723":1,"724":2,"726":1,"727":6}}],["ahzu6yr9xmpixlquhgbhj9xl3wiaoz6pe3cvml",{"2":{"520":2,"524":2}}],["ahead",{"2":{"41":1,"139":1,"271":1}}],["a5jf",{"2":{"519":2,"520":2}}],["akz",{"2":{"349":2}}],["akuumhj8fnuyvrbs38fdkeiaaaaaazb42trhw",{"2":{"349":2}}],["aka",{"2":{"184":1,"388":1}}],["aaaaaaaaaaaaaaaaaaaaaaaaaaecawqfbgcicra=",{"2":{"334":2}}],["aaaaaaaaaaaaaaaaaaaaaaaaaejpdcbnowap3dm=",{"2":{"319":8}}],["away",{"2":{"458":1,"625":1}}],["await",{"2":{"268":14,"398":4}}],["awareness",{"2":{"36":1}}],["awards",{"2":{"30":1,"34":1}}],["award",{"2":{"30":4}}],["a1",{"2":{"244":2}}],["airdrop",{"2":{"472":1,"693":1}}],["ai",{"2":{"365":1,"370":1,"452":1,"453":1,"674":1,"675":1,"676":1,"679":1,"683":1}}],["aiming",{"2":{"205":1}}],["aims",{"2":{"50":1,"278":1,"288":1}}],["aim",{"2":{"43":1,"205":1}}],["aid",{"2":{"35":1,"407":1}}],["aura",{"2":{"508":2}}],["augments",{"2":{"425":1,"478":1}}],["aunt",{"2":{"151":4}}],["aunts",{"2":{"151":4,"158":5,"159":4,"161":8}}],["audible",{"2":{"40":1}}],["audience",{"0":{"42":1},"2":{"40":1,"42":3,"43":1,"44":1,"45":1,"46":1}}],["authtype=",{"2":{"367":2}}],["authentication",{"0":{"368":1},"2":{"295":1,"313":1,"314":5,"315":1,"373":2}}],["authenticated",{"2":{"173":1}}],["authorize",{"2":{"506":1}}],["authorization",{"2":{"268":4,"313":1,"334":2}}],["authority",{"2":{"84":1,"268":2}}],["authorities",{"2":{"34":1}}],["auth",{"0":{"314":1,"316":1},"2":{"89":10,"268":4,"278":1,"312":2,"313":5,"314":9,"315":2,"316":7,"317":4,"319":2,"322":1,"334":7,"368":5,"370":1,"385":1,"494":2,"522":2,"524":2}}],["automated",{"2":{"395":1}}],["automatically",{"2":{"24":1,"84":1,"320":1,"387":1,"392":2,"509":1}}],["autoconnect",{"2":{"267":2}}],["autogenerated",{"2":{"243":1}}],["auto",{"2":{"84":2,"267":2,"268":4,"496":2,"498":2,"522":6,"629":2}}],["amanda",{"2":{"491":2}}],["amd64",{"2":{"305":12,"540":1,"560":1,"586":12}}],["amd",{"2":{"285":1,"305":1,"586":1}}],["amountceles",{"2":{"500":1}}],["amount=1000000utia",{"2":{"727":2}}],["amount=9000000utia",{"2":{"649":2}}],["amount=5000000000000utia",{"2":{"500":2}}],["amount=",{"2":{"500":2,"647":2}}],["amount",{"0":{"71":1},"2":{"41":1,"69":1,"84":1,"322":2,"323":2,"329":3,"378":1,"385":1,"386":1,"406":1,"414":1,"423":1,"431":1,"435":1,"450":2,"493":2,"494":2,"495":2,"497":4,"500":3,"522":8,"523":2,"525":4,"611":2,"647":2,"648":1,"649":3,"727":1}}],["amounts",{"2":{"34":1}}],["among",{"2":{"34":1,"37":1,"41":1,"421":1,"422":1,"478":1,"624":1}}],["amended",{"2":{"31":1}}],["ample",{"2":{"45":1}}],["amp",{"0":{"4":1,"85":1,"443":1,"497":1},"1":{"444":1,"445":1,"446":1,"447":1,"448":1,"449":1,"450":1},"2":{"1":4,"2":2,"44":1,"59":1,"60":1,"61":1,"65":1,"75":12,"85":1,"115":8,"151":14,"155":6,"156":2,"159":10,"160":2,"161":10,"247":8,"258":1,"268":12,"270":4,"272":4,"304":10,"377":1,"449":3,"450":5,"478":1,"503":1,"527":4,"585":10,"714":4,"715":4}}],["affect",{"2":{"395":1,"557":1}}],["affected",{"2":{"8":1}}],["affordable",{"2":{"46":1}}],["affiliated",{"2":{"44":1}}],["affirm",{"2":{"34":1}}],["after=network",{"2":{"712":2,"714":2,"715":2,"716":2}}],["afterwards",{"2":{"25":1}}],["after",{"0":{"38":1},"2":{"23":2,"24":2,"34":1,"40":1,"45":1,"46":1,"69":1,"71":2,"72":2,"79":1,"84":2,"86":1,"88":1,"90":1,"94":1,"117":1,"162":1,"170":1,"249":1,"304":2,"309":1,"319":1,"367":1,"373":2,"374":1,"394":1,"432":1,"436":1,"441":1,"484":1,"491":1,"509":2,"540":1,"554":1,"560":1,"565":1,"578":1,"585":2,"609":1,"610":1,"611":1,"624":1,"626":1,"629":1,"631":1,"650":1,"716":1,"721":1,"727":1,"728":1}}],["agnostic",{"2":{"421":1}}],["ago",{"2":{"243":2,"244":2}}],["again",{"2":{"76":1,"272":1,"337":1,"388":1,"425":1,"553":1,"554":1,"635":1}}],["against",{"2":{"31":3,"89":1,"128":1,"131":2,"206":1,"208":1,"556":1}}],["agreement",{"2":{"31":2,"34":3,"229":1}}],["agree",{"2":{"30":1,"31":5,"32":1,"33":1,"34":1,"422":1}}],["agreeing",{"2":{"30":1,"33":1,"34":2,"422":1}}],["ag",{"2":{"30":1}}],["aggression",{"2":{"16":1}}],["agenda",{"2":{"36":1}}],["agendas",{"2":{"5":1,"36":1}}],["agent",{"2":{"34":1}}],["age",{"2":{"7":1,"31":1}}],["avril14th",{"2":{"674":1,"675":1,"676":1,"689":1,"690":1,"691":1}}],["average",{"2":{"398":2,"400":1}}],["avoid",{"2":{"41":1,"44":1,"45":1,"89":1,"386":1,"503":1,"578":1,"630":1}}],["avoiding",{"2":{"14":1,"19":1}}],["availability",{"0":{"202":1,"328":1,"401":1,"402":1,"403":1,"404":1,"405":1,"406":1,"410":1,"415":1,"416":1,"421":1,"425":1,"665":1,"678":1,"679":1,"713":1},"1":{"203":1,"204":1,"205":1,"402":1,"403":1,"404":1,"405":1,"406":1,"407":1,"408":1,"409":1,"410":1,"411":1,"412":1,"413":1,"414":1,"416":1,"417":2,"418":2,"419":1,"420":1,"421":1,"679":1,"680":1,"681":1,"714":1,"715":1,"716":1},"2":{"34":1,"36":1,"42":1,"43":1,"59":1,"83":1,"105":2,"109":1,"110":2,"129":1,"131":1,"163":3,"177":1,"180":1,"192":1,"201":2,"203":2,"204":1,"205":2,"213":2,"229":1,"232":2,"255":2,"280":2,"285":1,"290":2,"299":1,"343":1,"375":1,"380":2,"381":1,"402":7,"403":3,"404":5,"405":4,"406":1,"407":1,"410":1,"411":1,"412":1,"415":4,"418":1,"422":1,"423":5,"424":1,"438":1,"440":2,"441":1,"457":1,"458":1,"470":1,"477":1,"478":2,"543":1,"548":1,"560":1,"573":2,"606":2,"654":2,"655":1,"667":1,"680":1,"685":1,"696":4,"707":2,"716":1,"722":2}}],["available",{"2":{"5":1,"17":2,"22":1,"30":2,"33":1,"34":6,"35":1,"37":1,"46":2,"56":1,"109":1,"131":1,"172":1,"176":1,"177":1,"206":2,"225":1,"228":1,"260":1,"367":1,"405":2,"410":1,"412":1,"415":1,"416":2,"421":1,"422":1,"424":3,"425":4,"438":2,"442":2,"450":2,"474":1,"488":2,"494":1,"502":2,"540":3,"560":3,"607":3,"610":1,"611":2,"671":1,"680":1}}],["aria2c",{"2":{"602":6}}],["aria2",{"2":{"585":4}}],["arise",{"2":{"411":1}}],["arises",{"2":{"403":1}}],["arrived",{"2":{"537":1}}],["array",{"2":{"267":1,"373":3,"571":1}}],["arranging",{"2":{"37":1}}],["arranges",{"2":{"424":1}}],["arranged",{"2":{"416":1,"422":1}}],["arrange",{"2":{"36":2,"42":1,"43":1}}],["arrangement",{"2":{"34":1,"422":1}}],["architectnodes",{"2":{"629":2,"689":1,"690":1,"691":1}}],["architecture",{"2":{"548":1}}],["architectures",{"2":{"256":1,"423":1,"540":1,"560":1}}],["archives",{"2":{"674":1,"675":1,"676":1}}],["archive",{"2":{"471":1,"588":1,"598":1,"674":1,"675":1,"676":1,"679":1}}],["archival",{"0":{"680":1,"681":1},"1":{"681":1},"2":{"441":3,"442":5,"680":2,"681":1}}],["arg",{"2":{"488":2}}],["args",{"2":{"312":3,"315":2,"579":4,"580":4}}],["arguments",{"2":{"185":1,"312":1}}],["argument",{"2":{"117":1,"189":1,"192":1,"249":1}}],["arm64",{"2":{"305":12,"540":1,"560":1,"586":12}}],["arm",{"2":{"305":1,"586":1}}],["armored",{"2":{"238":1,"491":2}}],["arabicaresturl",{"2":{"398":2}}],["arabicarpcurl",{"2":{"398":2}}],["arabicachainid",{"2":{"319":1,"357":4,"398":2,"399":1,"551":1,"592":6,"602":2}}],["arabicacelestia",{"2":{"308":1,"309":2,"310":1,"484":1,"485":1,"486":1,"619":1,"621":1,"659":1,"660":1,"661":1,"663":1}}],["arabicaversions",{"2":{"307":2,"539":4,"559":2,"577":2,"579":2,"580":2}}],["arabica",{"0":{"465":1,"472":1,"701":1,"702":1},"1":{"466":1,"467":1,"468":1,"469":1,"470":1,"471":1,"472":1,"473":2,"474":2,"475":1,"476":1,"702":1},"2":{"237":1,"307":1,"308":1,"309":5,"310":3,"314":3,"326":2,"357":1,"360":1,"398":6,"399":6,"465":2,"466":3,"469":1,"470":12,"471":5,"473":2,"474":1,"475":3,"484":2,"485":5,"486":1,"537":1,"551":2,"577":1,"592":2,"619":1,"620":4,"621":1,"625":1,"659":1,"660":5,"661":3,"662":1,"663":1,"685":1,"701":3,"703":1,"709":2,"724":1}}],["arbitrary",{"2":{"135":2,"495":1}}],["arbitrum",{"0":{"73":1,"79":1,"82":1,"96":1,"98":1,"104":1,"112":1,"120":1},"1":{"74":1,"75":1,"76":1,"77":1,"78":1,"80":1,"81":1,"82":1,"83":1,"84":1,"85":1,"86":1,"87":1,"88":1,"89":1,"90":1,"91":1,"92":1,"93":1,"94":1,"95":1,"96":1,"97":1,"98":1,"99":1,"105":1,"106":1,"107":1,"108":1,"109":1,"110":1,"111":1,"113":1,"114":1,"115":1,"116":1,"117":1,"118":1,"119":1,"120":1,"121":1,"122":1},"2":{"66":1,"68":1,"69":3,"71":4,"73":3,"74":1,"75":2,"79":4,"80":3,"81":1,"82":3,"83":1,"85":5,"88":6,"89":1,"91":2,"93":1,"95":1,"96":1,"101":3,"102":2,"105":5,"106":1,"107":1,"109":1,"110":2,"111":1,"113":1,"122":2,"207":1,"210":2,"255":1,"376":1}}],["arb",{"2":{"75":2,"120":2}}],["around",{"0":{"409":1},"2":{"13":1,"49":1,"256":1,"403":1,"409":1,"411":1,"548":1,"573":1,"653":1}}],["areas",{"2":{"38":1,"43":1,"48":1}}],["area",{"2":{"37":1,"42":1,"43":1,"44":1,"45":1,"48":1}}],["aren",{"2":{"35":1,"529":1}}],["are",{"0":{"288":1,"406":1,"414":1,"556":1},"2":{"9":2,"11":1,"17":1,"19":1,"21":2,"23":2,"24":3,"26":1,"30":2,"31":7,"32":2,"33":2,"34":8,"35":1,"36":1,"39":2,"41":1,"42":2,"48":3,"50":1,"52":1,"59":1,"65":1,"66":1,"76":1,"84":3,"85":1,"88":1,"89":4,"91":2,"107":1,"108":1,"116":1,"117":1,"127":2,"128":2,"134":1,"135":3,"136":1,"137":1,"139":3,"143":1,"146":3,"148":3,"151":2,"152":1,"155":1,"157":2,"158":2,"159":2,"161":2,"162":1,"166":2,"171":1,"172":1,"176":1,"177":1,"178":1,"183":1,"184":1,"185":2,"186":1,"189":1,"192":1,"195":1,"203":1,"204":1,"207":1,"210":1,"215":2,"223":1,"225":2,"228":2,"229":1,"231":1,"232":1,"233":1,"237":3,"243":2,"248":1,"249":1,"256":2,"260":1,"272":1,"275":1,"278":1,"280":1,"281":2,"283":2,"285":2,"290":1,"292":1,"293":1,"303":1,"304":4,"308":1,"309":3,"312":1,"314":1,"316":2,"337":1,"348":1,"351":2,"357":1,"360":1,"366":1,"367":1,"373":3,"374":4,"377":2,"384":2,"385":3,"386":1,"388":1,"394":1,"396":2,"404":1,"406":4,"409":1,"411":1,"412":1,"413":1,"415":3,"416":3,"417":1,"418":2,"421":1,"422":3,"424":4,"425":1,"438":1,"439":2,"441":1,"442":2,"445":1,"450":4,"465":1,"469":1,"471":1,"472":1,"475":1,"479":1,"484":1,"485":1,"489":2,"494":5,"495":2,"499":1,"500":1,"501":1,"502":2,"503":3,"509":1,"511":1,"517":1,"522":1,"529":1,"531":2,"537":1,"540":1,"541":3,"547":1,"550":1,"552":3,"554":2,"556":3,"560":1,"566":1,"568":1,"573":1,"574":1,"575":1,"577":1,"585":4,"588":4,"592":1,"600":1,"601":2,"603":1,"610":1,"611":6,"613":1,"614":1,"620":2,"622":1,"624":1,"625":2,"626":1,"627":1,"632":1,"648":1,"653":1,"656":1,"660":1,"672":1,"679":1,"682":1,"685":3,"688":2,"692":1,"693":1,"694":1,"696":1,"703":1,"707":2,"709":1,"711":2,"718":1,"722":1,"724":1,"727":4}}],["abstain",{"2":{"494":1}}],["abbreviation",{"2":{"455":1}}],["abci++",{"2":{"421":1,"424":1}}],["abci",{"0":{"611":1},"2":{"380":1,"421":1,"611":8}}],["abi",{"2":{"77":1,"151":2,"154":1,"161":2,"265":1,"268":4,"273":1}}],["ability",{"2":{"30":1,"110":1,"167":1,"403":1,"407":1,"408":1,"509":1,"620":2,"660":2}}],["able",{"2":{"34":3,"40":2,"41":1,"65":1,"71":2,"72":1,"88":1,"89":1,"134":2,"148":1,"167":1,"181":1,"183":1,"188":1,"231":1,"357":1,"375":1,"388":1,"399":1,"400":1,"404":1,"407":1,"419":2,"441":1,"505":1,"548":1,"701":1,"712":1,"714":1,"715":1,"716":1,"727":1}}],["abusive",{"2":{"11":1}}],["about",{"0":{"285":1},"1":{"286":1},"2":{"3":1,"8":1,"17":1,"21":1,"30":2,"33":1,"36":1,"42":1,"43":1,"48":5,"49":1,"65":1,"71":1,"79":2,"84":1,"92":1,"105":1,"107":1,"108":1,"116":1,"134":2,"146":1,"208":1,"229":1,"248":1,"256":1,"288":1,"291":1,"301":1,"302":1,"307":1,"319":1,"320":1,"387":1,"409":1,"410":1,"411":2,"460":1,"469":1,"488":2,"519":1,"540":1,"559":1,"560":1,"575":2,"672":1,"680":1,"681":1,"696":1}}],["above",{"2":{"2":1,"24":1,"31":1,"34":1,"75":2,"83":1,"89":1,"103":1,"120":1,"139":1,"142":1,"148":3,"153":1,"154":1,"156":1,"160":1,"162":1,"167":1,"170":1,"184":1,"189":1,"195":1,"272":2,"316":1,"317":1,"319":3,"322":1,"385":1,"392":2,"421":1,"424":2,"470":1,"471":1,"478":1,"495":1,"522":1,"562":1,"578":1,"648":1,"649":1,"661":1}}],["atomic",{"2":{"423":1}}],["atlas",{"2":{"94":1}}],["atmosphere",{"2":{"40":1,"41":1}}],["at",{"0":{"449":1,"455":1},"1":{"450":1},"2":{"10":1,"11":1,"17":3,"24":1,"27":1,"30":2,"33":1,"34":2,"40":2,"41":2,"42":2,"43":1,"44":3,"45":1,"46":1,"48":1,"80":1,"82":1,"91":1,"95":1,"116":1,"119":1,"131":2,"133":1,"153":2,"159":4,"178":1,"201":1,"214":1,"228":1,"248":1,"259":1,"262":1,"267":1,"287":1,"289":1,"319":1,"373":3,"376":2,"377":1,"388":1,"406":1,"409":1,"416":3,"418":1,"421":1,"424":1,"425":3,"444":1,"445":2,"448":1,"449":1,"450":6,"451":1,"455":1,"471":1,"474":1,"486":1,"488":4,"503":2,"505":1,"507":2,"515":1,"532":1,"540":1,"541":1,"548":1,"552":1,"560":1,"588":1,"598":1,"601":1,"621":1,"625":1,"628":1,"630":1,"648":1,"649":1,"663":1,"667":1,"679":1,"688":1,"701":1,"715":1}}],["attach",{"2":{"503":2}}],["attack",{"2":{"403":3,"406":1}}],["attacks",{"2":{"8":1,"84":1}}],["attempting",{"2":{"632":1}}],["attempt",{"2":{"402":1}}],["attesting",{"2":{"159":1}}],["attestations",{"2":{"204":2,"255":1}}],["attestation",{"2":{"153":2,"159":4}}],["attestationproof",{"0":{"159":1},"2":{"153":6,"159":6,"161":4}}],["attests",{"2":{"148":1}}],["attested",{"2":{"127":2,"210":1}}],["attended",{"2":{"45":1}}],["attendees",{"2":{"36":3,"37":4,"38":3,"40":2,"41":2,"42":2,"43":2,"45":1,"46":1,"48":1,"52":1,"65":2}}],["attending",{"2":{"45":2}}],["attend",{"2":{"42":1,"46":3}}],["attendance",{"2":{"36":1,"38":1,"42":3,"45":2}}],["attention",{"2":{"8":1}}],["attracting",{"2":{"45":1}}],["attract",{"2":{"42":1,"43":1}}],["attribution",{"0":{"17":1}}],["ad5ezbg0",{"2":{"334":2}}],["adoption",{"2":{"256":1}}],["adopted",{"2":{"34":1}}],["adjusted",{"2":{"385":1,"386":1}}],["adjust",{"2":{"233":1}}],["adr019",{"2":{"445":1}}],["adrs",{"2":{"421":1,"478":1}}],["adr",{"2":{"167":1,"548":2}}],["adapt",{"2":{"148":1,"288":1}}],["adapted",{"2":{"17":1,"81":1,"450":1}}],["admin",{"2":{"89":2,"314":4,"316":2,"334":2,"367":2,"503":1}}],["administrative",{"2":{"84":1,"293":1}}],["adhere",{"2":{"52":1}}],["adhering",{"2":{"24":1}}],["adventure",{"2":{"365":1}}],["advertise",{"2":{"608":1}}],["advertising",{"2":{"34":1}}],["adverse",{"2":{"32":1}}],["advanced",{"0":{"332":1,"574":1}}],["advance",{"2":{"41":1,"45":1}}],["advances",{"2":{"8":1,"206":1}}],["advice",{"2":{"32":3}}],["advise",{"2":{"688":1}}],["advised",{"2":{"550":1,"592":1}}],["adviser",{"2":{"32":1}}],["advisable",{"2":{"32":1,"588":1}}],["addnetworkkeplr",{"2":{"398":17}}],["adds",{"2":{"385":1,"424":1}}],["addblock",{"2":{"137":2,"139":4}}],["addrs",{"2":{"324":2}}],["addr",{"2":{"89":1,"367":2,"392":4,"502":2,"554":2,"629":8}}],["addressed",{"2":{"256":1}}],["addresses",{"2":{"84":2,"89":3,"92":1,"119":1,"288":1}}],["address=$",{"2":{"608":2,"652":2}}],["address=$evm",{"2":{"500":2}}],["address=address",{"2":{"527":4}}],["address=celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5",{"2":{"521":1}}],["address=celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5export",{"2":{"521":1}}],["address=celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j",{"2":{"521":2}}],["address=celestia1c425ckmve2489atttx022qpc02gxspa29wmh0d",{"2":{"329":1}}],["address=celestia1c425ckmve2489atttx022qpc02gxspa29wmh0dexport",{"2":{"329":1}}],["address=0x70997970c51812dc3a010c7d01b50e0d17dc79c8",{"2":{"342":1,"359":1}}],["address=0x70997970c51812dc3a010c7d01b50e0d17dc79c8export",{"2":{"342":1,"359":1}}],["address=0xff00000000000000000000000000000000000000",{"2":{"342":2,"359":2}}],["address=0x5fbdb2315678afecb367f032d93f642f64180aa3",{"2":{"120":1,"252":1}}],["address=0x5fbdb2315678afecb367f032d93f642f64180aa3export",{"2":{"120":1,"252":1}}],["address=",{"2":{"75":2,"263":2,"264":2,"272":2,"500":2,"503":1}}],["address",{"0":{"323":1,"325":1,"608":1},"2":{"8":1,"10":1,"24":1,"27":3,"33":2,"44":1,"75":3,"76":8,"77":2,"84":8,"89":6,"91":4,"98":1,"99":1,"121":4,"134":1,"151":2,"159":2,"160":2,"161":2,"210":1,"244":4,"253":4,"260":10,"263":10,"264":8,"268":10,"269":1,"270":4,"272":11,"273":2,"309":1,"310":7,"313":1,"323":9,"325":3,"326":8,"329":2,"348":1,"351":5,"367":3,"377":2,"392":2,"399":1,"400":1,"473":5,"484":1,"485":2,"493":10,"495":1,"500":5,"502":2,"519":3,"520":4,"521":3,"522":16,"523":5,"524":5,"525":4,"528":1,"529":4,"537":2,"541":2,"542":2,"552":1,"607":1,"608":8,"620":2,"626":2,"629":4,"646":1,"651":8,"652":4,"661":2,"662":5,"693":5,"721":2}}],["added",{"2":{"84":1,"91":2,"110":1,"268":4,"375":1,"402":1,"503":2,"506":1,"626":1,"648":1,"650":1}}],["addition",{"2":{"34":1,"40":1,"412":1,"425":1,"458":1,"569":1}}],["additionally",{"2":{"33":1,"45":1,"89":1,"106":1,"107":2,"110":1,"135":1,"416":1,"447":1}}],["additional",{"0":{"34":1,"98":1,"99":1,"333":1,"730":1},"1":{"334":1,"335":1,"336":1,"337":1},"2":{"32":1,"34":1,"38":1,"44":1,"46":1,"96":1,"97":1,"105":1,"134":1,"186":1,"266":1,"295":1,"312":1,"385":1,"418":1,"432":1,"436":1,"730":1}}],["adding",{"0":{"269":1,"399":1,"400":1,"648":1},"2":{"2":1,"91":2,"233":1,"490":2,"504":1,"630":1,"631":1}}],["add",{"0":{"68":1,"87":1,"398":1,"630":1,"647":1,"652":1},"2":{"5":1,"45":1,"47":1,"68":1,"83":1,"89":2,"135":2,"139":4,"235":1,"237":6,"239":6,"266":1,"268":4,"269":2,"280":1,"305":1,"309":1,"310":2,"367":3,"397":2,"398":6,"399":4,"400":3,"442":1,"450":1,"484":1,"488":4,"490":2,"500":5,"503":1,"507":8,"511":1,"519":2,"527":5,"536":2,"544":1,"553":1,"568":1,"586":1,"625":1,"629":1,"630":6,"635":3,"646":2,"647":2,"650":1,"651":1,"661":2}}],["apt",{"2":{"304":6,"585":6}}],["aptyummac",{"2":{"304":1,"585":1}}],["aptyum",{"2":{"304":1,"585":1}}],["apache",{"2":{"127":2}}],["api",{"0":{"292":1,"293":1,"296":1,"297":1,"298":1,"330":1,"368":1,"389":1,"675":1,"690":1},"1":{"293":1,"294":2,"295":2,"296":2,"297":1,"298":2,"390":1,"391":1,"392":1,"393":1},"2":{"48":1,"88":1,"266":4,"280":2,"292":5,"293":4,"294":1,"295":1,"297":2,"298":1,"299":1,"309":1,"311":1,"313":1,"314":1,"315":2,"330":1,"336":1,"365":3,"366":1,"369":1,"370":1,"371":1,"372":2,"373":1,"377":2,"393":2,"399":1,"400":1,"413":1,"442":2,"470":3,"494":1,"543":1,"573":1,"606":1,"628":1,"675":18,"690":14}}],["apology",{"2":{"13":1}}],["apologizing",{"2":{"8":1}}],["appcelestia",{"2":{"605":1,"644":1}}],["appcd",{"2":{"539":1}}],["appconsts",{"2":{"392":4}}],["appdsudo",{"2":{"712":2}}],["appd",{"2":{"390":2,"392":2,"488":4,"489":2,"490":8,"491":6,"492":6,"493":10,"494":10,"495":4,"496":2,"497":4,"498":2,"500":10,"507":18,"515":2,"516":2,"519":2,"520":2,"522":6,"523":2,"524":2,"525":4,"527":8,"529":4,"536":6,"537":4,"539":5,"540":5,"592":12,"603":2,"605":2,"609":1,"610":1,"611":1,"637":3,"644":3,"645":2,"646":2,"647":2,"649":2,"650":3,"652":2,"653":2,"711":1,"712":15,"721":4,"727":6}}],["append",{"2":{"127":2,"139":2,"374":2}}],["appendix",{"0":{"93":1},"1":{"94":1,"95":1,"96":1,"97":1,"98":1,"99":1},"2":{"89":1}}],["appear",{"2":{"91":1,"431":1,"503":1}}],["appearance",{"2":{"7":1}}],["appeal",{"2":{"42":1}}],["app",{"0":{"390":1,"424":1,"511":1,"534":1,"538":1,"591":1,"639":1,"642":1,"708":1,"712":1},"1":{"425":1,"512":1,"513":1,"514":1,"515":1,"516":1,"517":1,"518":1,"519":1,"520":1,"521":1,"522":1,"523":1,"524":1,"525":1,"526":1,"527":1,"528":1,"529":1,"530":1,"531":1,"532":1,"533":1,"535":1,"536":1,"537":1,"539":1,"540":1,"541":1,"640":1,"641":1,"642":1,"643":1,"644":1,"645":1,"646":1,"647":1,"648":1,"649":1,"650":1,"651":1,"652":1,"653":1},"2":{"77":1,"88":2,"151":2,"155":4,"156":4,"161":4,"267":6,"268":5,"270":5,"273":1,"280":1,"326":4,"381":1,"384":2,"387":2,"388":1,"392":20,"394":1,"421":6,"424":1,"425":2,"452":1,"453":1,"478":3,"488":2,"503":4,"509":1,"511":1,"512":1,"514":3,"515":8,"516":2,"519":4,"520":2,"522":6,"523":2,"524":2,"525":4,"526":1,"534":1,"535":1,"538":1,"539":18,"540":4,"541":1,"591":1,"592":13,"596":1,"598":1,"599":2,"601":1,"602":18,"603":1,"605":1,"607":2,"608":2,"610":1,"611":1,"612":1,"626":2,"627":1,"639":1,"642":1,"644":1,"648":1,"649":1,"650":3,"651":1,"670":1,"696":1,"708":3,"709":2,"712":1,"715":1,"722":1}}],["appreciate",{"2":{"48":1,"52":1}}],["appreciation",{"2":{"43":2}}],["approximation",{"2":{"385":1,"671":1}}],["approximately",{"2":{"71":1,"72":1}}],["approving",{"2":{"71":1}}],["approve",{"2":{"69":1,"72":1,"83":1,"431":1}}],["approval",{"2":{"53":1,"431":1}}],["approaches",{"2":{"162":1}}],["approach",{"2":{"42":1,"43":1,"44":1,"184":1,"191":1,"469":1,"672":1,"685":1}}],["appropriate",{"2":{"9":2,"135":2,"438":1,"557":1,"629":1}}],["apple",{"2":{"305":1,"540":1,"560":1,"586":1}}],["applicable",{"2":{"25":2,"27":1,"31":1,"33":1,"34":9,"36":1}}],["applicant",{"2":{"23":1}}],["applicants",{"2":{"23":8,"24":2,"26":2}}],["applications",{"2":{"24":1,"26":1,"212":1,"214":2,"217":1,"228":1,"232":2,"233":2,"291":1,"302":2,"381":1,"413":1,"419":1,"440":1,"442":1,"465":1,"666":1,"698":1,"710":1,"712":1}}],["application",{"0":{"26":1,"27":1},"1":{"27":1},"2":{"21":3,"24":3,"27":1,"232":2,"233":1,"256":1,"263":1,"265":1,"270":1,"289":2,"302":1,"334":2,"380":1,"384":2,"388":1,"419":7,"421":3,"458":1,"478":1,"488":10,"504":1,"536":1,"557":1,"602":1,"610":1}}],["applies",{"2":{"10":2,"103":1}}],["applying",{"2":{"27":1,"38":1,"416":1}}],["apply",{"2":{"21":1,"25":1,"26":1,"34":1,"422":1,"544":1,"601":1}}],["appointed",{"2":{"10":1}}],["acelestia",{"2":{"721":1}}],["acquire",{"0":{"82":1},"2":{"82":1}}],["acquaintances",{"2":{"44":1,"45":1}}],["achieved",{"2":{"50":1,"504":1,"594":1}}],["across",{"2":{"47":2,"107":1,"155":1,"232":1,"449":1,"638":1}}],["acknowledging",{"2":{"44":1}}],["acknowledge",{"2":{"31":4,"34":4}}],["accurately",{"2":{"386":1}}],["accurate",{"2":{"384":1}}],["accuracy",{"2":{"33":1,"34":1}}],["accname",{"2":{"244":2,"310":6,"367":2,"486":6,"621":6,"661":6,"663":6}}],["accomplish",{"2":{"244":1}}],["accomplished",{"2":{"213":1,"290":1}}],["accommodate",{"2":{"36":1}}],["accordingly",{"2":{"48":1,"89":1}}],["according",{"2":{"41":1,"206":1}}],["accordance",{"2":{"33":1,"34":3}}],["accountexport",{"2":{"527":1}}],["accounted",{"2":{"385":1}}],["accountname",{"2":{"392":2}}],["accountaddress",{"2":{"312":1}}],["accounts",{"0":{"119":1,"525":1},"2":{"91":5,"120":1,"151":2,"161":2,"251":1,"263":1,"385":1,"506":4,"507":2,"525":1,"532":1}}],["account",{"0":{"325":1,"329":1,"388":1,"511":1,"518":1,"522":1,"523":1,"524":1,"528":1,"529":1,"647":1},"1":{"512":1,"513":1,"514":1,"515":1,"516":1,"517":1,"518":1,"519":2,"520":2,"521":2,"522":2,"523":2,"524":2,"525":2,"526":1,"527":1,"528":1,"529":1,"530":1,"531":1,"532":1,"533":1},"2":{"10":1,"34":4,"74":1,"76":2,"84":2,"88":3,"91":9,"258":1,"266":1,"270":2,"272":3,"322":2,"325":3,"337":4,"352":1,"357":1,"369":1,"385":3,"388":1,"392":6,"399":1,"400":1,"411":1,"488":4,"494":4,"500":3,"507":10,"511":3,"517":1,"518":1,"519":2,"522":12,"523":11,"524":6,"525":3,"526":2,"527":4,"529":4,"533":2,"551":1,"620":1,"629":4,"630":1,"647":2,"660":1}}],["accessed",{"2":{"607":1}}],["accessing",{"0":{"598":1},"2":{"409":1,"410":1}}],["accessibility",{"2":{"256":1}}],["accessible",{"2":{"36":1,"65":1,"206":1,"236":4,"409":1,"541":2,"552":2}}],["access",{"2":{"31":4,"42":1,"43":1,"53":2,"56":1,"65":1,"90":1,"145":1,"168":1,"182":1,"191":1,"201":1,"293":1,"294":1,"334":1,"385":1,"398":2,"404":1,"408":1,"409":2,"413":2,"442":4,"471":2,"620":1,"660":1,"679":1,"688":1}}],["accepts",{"2":{"109":1,"425":2}}],["accept",{"2":{"30":1,"185":1,"384":1,"388":1,"425":1,"502":2,"606":1}}],["accepted",{"2":{"23":1,"108":1}}],["acceptable",{"2":{"9":1}}],["accepting",{"2":{"8":2}}],["actors",{"2":{"413":1}}],["actually",{"2":{"283":1,"402":1}}],["actual",{"2":{"135":3,"137":1,"146":1,"171":1,"385":1,"445":1}}],["activating",{"2":{"624":1,"625":1}}],["activation",{"2":{"353":1}}],["activate",{"2":{"568":1}}],["activity",{"2":{"90":1}}],["activities",{"2":{"25":1,"33":3,"36":1,"37":1,"64":1}}],["active",{"2":{"21":1,"24":3,"27":2,"55":1,"243":1,"244":1,"478":1}}],["acting",{"2":{"10":1,"34":1}}],["actions",{"2":{"14":1,"31":2,"135":4}}],["action",{"2":{"9":1,"12":1,"31":1,"312":1}}],["act",{"2":{"7":1,"31":3,"34":13,"509":1}}],["aside",{"2":{"450":1}}],["async",{"2":{"268":6,"398":2}}],["ascii",{"2":{"238":1,"491":2}}],["asked",{"2":{"52":1,"685":1}}],["ask",{"2":{"40":1,"43":1,"44":1,"398":2,"425":1}}],["aspects",{"2":{"47":1,"65":1}}],["aspect",{"2":{"31":1}}],["assumption",{"2":{"386":1,"406":2,"409":1}}],["assumptions",{"0":{"406":1},"2":{"203":1,"409":1}}],["assumed",{"2":{"370":1,"406":1,"463":1}}],["assumes",{"2":{"334":1,"386":1,"406":1,"543":1,"559":1}}],["assume",{"2":{"135":1,"232":1,"367":1}}],["assurances",{"0":{"414":1},"2":{"34":2}}],["assign",{"2":{"84":1,"500":1}}],["assist",{"2":{"46":1,"47":1}}],["assistance",{"2":{"45":1,"47":1}}],["associated",{"2":{"24":1,"32":1,"45":1,"159":2,"322":1,"424":2}}],["asset",{"2":{"457":1}}],["assets",{"2":{"8":1,"218":1}}],["assert",{"2":{"189":1,"260":1}}],["asserted",{"2":{"185":2}}],["asserteq",{"2":{"117":4,"249":4,"260":9}}],["asserts",{"2":{"117":1,"249":1}}],["assertion",{"2":{"260":2}}],["assertions",{"2":{"84":3}}],["asserting",{"2":{"117":1,"249":1}}],["assessing",{"2":{"38":1}}],["assess",{"2":{"21":1,"42":1}}],["as",{"0":{"104":1,"226":1,"335":1,"652":1,"710":1},"1":{"105":1,"106":1,"107":1,"108":1,"109":1,"110":1,"111":1,"711":1,"712":1,"713":1,"714":1,"715":1,"716":1},"2":{"7":1,"8":2,"10":1,"14":2,"23":1,"24":2,"30":5,"31":3,"32":1,"34":10,"36":1,"40":2,"42":1,"43":3,"44":2,"45":2,"47":1,"50":1,"54":1,"56":1,"57":1,"65":3,"71":1,"75":2,"76":1,"84":1,"85":2,"86":2,"89":1,"91":1,"107":2,"108":2,"111":1,"119":1,"120":1,"126":2,"128":1,"131":1,"134":5,"135":2,"136":2,"139":1,"142":1,"148":1,"153":1,"154":1,"155":2,"156":3,"157":1,"158":1,"160":2,"161":3,"167":1,"173":1,"176":1,"178":2,"181":1,"182":1,"184":1,"185":3,"186":1,"189":6,"192":2,"195":1,"201":1,"203":1,"205":1,"206":2,"207":2,"208":1,"213":4,"217":1,"231":6,"252":1,"255":2,"260":4,"263":5,"264":1,"272":3,"276":1,"278":2,"279":1,"285":1,"288":1,"290":4,"293":2,"294":1,"295":2,"308":1,"312":3,"316":1,"317":2,"329":1,"334":1,"346":1,"347":1,"348":2,"351":4,"358":1,"373":2,"374":2,"384":2,"385":1,"386":2,"388":1,"392":2,"394":1,"402":2,"405":2,"406":1,"408":1,"409":1,"412":2,"416":4,"418":3,"419":4,"421":1,"422":5,"424":3,"425":3,"426":1,"440":3,"441":4,"442":4,"444":1,"448":1,"450":2,"458":3,"459":1,"460":1,"466":1,"471":1,"478":1,"485":1,"487":1,"488":2,"494":1,"495":1,"501":1,"503":5,"504":1,"506":1,"515":1,"521":3,"522":1,"523":1,"527":1,"529":1,"536":2,"541":1,"547":1,"548":1,"551":1,"552":1,"571":1,"583":1,"588":1,"596":1,"601":1,"602":1,"603":1,"620":4,"622":1,"623":1,"626":1,"628":3,"637":1,"639":1,"647":1,"651":1,"652":1,"660":2,"664":1,"666":1,"667":1,"670":2,"671":1,"685":1,"690":1,"701":1,"703":1,"709":1,"710":1,"711":1,"712":1,"716":1,"721":2,"722":1,"726":1,"728":1}}],["anger",{"2":{"519":2}}],["anchoring",{"2":{"268":1}}],["analogously",{"2":{"129":1}}],["analogous",{"2":{"128":1}}],["analog",{"2":{"127":2}}],["analytics",{"0":{"683":1},"2":{"683":2}}],["analysis",{"2":{"181":1}}],["analyse",{"2":{"33":1}}],["analyze",{"2":{"38":1}}],["anvil",{"0":{"251":1},"2":{"120":1,"251":4,"263":3}}],["another",{"0":{"323":1,"329":1},"2":{"84":1,"91":2,"168":1,"181":1,"243":1,"244":1,"255":1,"260":2,"309":1,"323":1,"350":1,"367":1,"384":1,"424":1,"484":1,"493":1,"509":1,"571":1,"613":1,"638":1,"722":1}}],["answer",{"2":{"48":1,"522":1}}],["answers",{"2":{"17":1,"47":1,"402":1}}],["annual",{"2":{"445":2}}],["annually",{"2":{"445":1,"455":1}}],["announced",{"2":{"45":1}}],["announce",{"2":{"45":1}}],["announcement",{"2":{"45":1,"476":1,"684":1,"685":1,"695":1,"705":1}}],["announcing",{"2":{"45":2}}],["anniversary",{"2":{"34":3}}],["anticipate",{"2":{"42":1}}],["anticipation",{"2":{"31":1}}],["an",{"0":{"71":1,"79":1,"88":1,"335":1,"338":1,"354":1,"364":1,"606":1},"1":{"80":1,"81":1,"82":1,"83":1,"84":1,"85":1,"86":1,"87":1,"88":1,"89":1,"90":1,"91":1,"92":1,"93":1,"94":1,"95":1,"96":1,"97":1,"98":1,"99":1,"339":1,"340":1,"341":1,"342":1,"343":1,"344":1,"345":1,"346":1,"347":1,"348":1,"349":1,"350":1,"351":1,"352":1,"353":1,"355":1,"356":1,"357":1,"358":1,"359":1,"360":1,"361":1,"362":1,"363":1,"364":1,"607":1,"608":1,"609":1},"2":{"4":1,"7":1,"10":5,"13":1,"16":1,"22":1,"24":4,"27":1,"30":1,"31":3,"34":9,"37":1,"40":1,"42":1,"43":2,"44":1,"47":1,"52":1,"53":2,"55":1,"59":1,"60":1,"61":1,"65":1,"69":2,"71":1,"73":1,"74":1,"76":2,"82":1,"84":2,"85":1,"88":4,"91":1,"92":1,"101":1,"105":2,"107":1,"108":1,"109":1,"110":2,"111":1,"117":1,"126":1,"128":1,"131":2,"133":1,"135":4,"136":2,"150":1,"151":2,"152":2,"155":1,"156":1,"158":1,"159":6,"161":3,"169":1,"170":2,"173":1,"176":2,"177":1,"180":1,"201":2,"206":1,"207":1,"212":1,"214":3,"232":1,"244":2,"249":1,"252":1,"255":2,"257":2,"259":1,"260":4,"263":1,"266":1,"268":2,"272":2,"275":1,"286":2,"288":1,"295":1,"309":2,"314":1,"315":1,"316":2,"317":3,"319":2,"323":1,"324":1,"325":1,"326":1,"329":1,"337":1,"352":1,"364":1,"365":1,"366":1,"369":2,"373":1,"377":1,"380":1,"384":1,"385":3,"387":1,"388":1,"392":2,"403":1,"406":3,"415":1,"419":4,"421":1,"422":6,"423":1,"424":1,"425":1,"444":1,"445":1,"451":1,"455":1,"457":1,"464":1,"471":3,"478":2,"480":1,"484":2,"485":1,"488":4,"491":1,"495":2,"500":1,"501":1,"503":1,"504":1,"509":1,"529":3,"530":1,"536":1,"539":1,"540":1,"544":1,"545":1,"546":1,"548":2,"550":1,"554":3,"557":1,"560":1,"571":1,"575":1,"577":3,"579":1,"588":1,"589":1,"598":1,"610":1,"611":1,"615":1,"620":1,"624":1,"628":2,"634":1,"649":1,"657":1,"660":1,"671":1,"679":1,"719":1,"721":2,"727":1}}],["anyway",{"2":{"297":1,"302":1,"552":1}}],["anything",{"2":{"178":1,"630":1}}],["anytrust",{"2":{"83":1,"84":1,"105":1,"107":1}}],["anyone",{"2":{"116":1,"248":1,"284":1,"289":1,"380":1,"402":1,"423":1,"627":1}}],["any",{"2":{"4":1,"8":1,"9":1,"11":1,"12":1,"15":1,"16":1,"24":4,"25":1,"27":1,"30":5,"31":19,"32":4,"33":2,"34":36,"36":3,"37":2,"45":2,"48":1,"89":1,"133":1,"172":1,"201":1,"229":1,"231":1,"233":2,"235":1,"269":1,"280":1,"282":1,"283":1,"307":1,"312":1,"351":1,"402":1,"417":1,"459":1,"472":2,"488":2,"503":4,"539":1,"554":1,"559":1,"562":1,"565":1,"571":1,"601":1,"637":1,"667":1,"680":1,"693":2,"696":1,"721":1}}],["and",{"0":{"39":1,"41":1,"45":1,"66":1,"75":1,"86":1,"87":1,"90":1,"100":1,"203":1,"205":1,"213":1,"241":1,"260":1,"268":1,"279":1,"280":1,"287":1,"288":1,"290":1,"310":1,"340":1,"367":1,"383":1,"384":1,"395":1,"410":1,"430":1,"432":1,"435":1,"436":1,"440":1,"441":1,"466":1,"491":1,"542":1,"571":1,"579":1,"593":1,"661":1,"667":1,"688":1,"692":1},"1":{"40":1,"41":1,"42":1,"43":1,"44":1,"45":1,"46":1,"47":1,"48":1,"67":1,"68":1,"69":1,"70":1,"71":1,"72":1,"76":1,"101":1,"102":1,"103":1,"242":1,"243":1,"244":1,"288":1,"289":1,"384":1,"385":2,"386":2,"387":2,"388":1,"441":1,"442":1,"594":1,"595":1,"596":1,"597":1,"598":1,"599":1,"662":1},"2":{"0":1,"1":1,"2":4,"3":3,"5":1,"7":5,"8":8,"9":6,"10":1,"11":3,"13":1,"19":3,"21":2,"22":1,"23":3,"24":10,"26":1,"27":10,"28":1,"30":7,"31":20,"32":3,"33":11,"34":37,"36":12,"37":7,"38":6,"39":1,"40":6,"41":9,"42":7,"43":6,"44":8,"45":11,"46":11,"47":4,"48":3,"50":4,"52":4,"53":2,"54":1,"56":2,"57":3,"58":1,"59":1,"60":1,"61":1,"62":1,"63":2,"64":2,"65":10,"66":3,"71":3,"72":1,"73":1,"75":4,"78":1,"79":5,"80":3,"83":2,"84":13,"85":4,"86":3,"87":1,"88":6,"89":1,"90":2,"91":5,"92":2,"94":2,"101":1,"103":1,"105":2,"107":9,"108":3,"109":1,"113":2,"114":1,"115":1,"116":1,"117":4,"119":5,"120":1,"128":1,"130":1,"131":9,"133":3,"134":3,"135":8,"136":3,"137":1,"139":10,"141":1,"146":5,"148":4,"149":2,"150":1,"151":2,"152":3,"153":1,"155":5,"156":2,"158":3,"159":6,"161":2,"162":3,"163":1,"166":3,"167":1,"170":3,"171":2,"172":2,"173":5,"176":3,"177":3,"178":2,"180":1,"181":5,"183":1,"184":2,"185":3,"189":4,"192":2,"194":1,"195":2,"201":3,"205":5,"206":5,"207":3,"208":1,"209":1,"212":2,"213":2,"214":1,"215":1,"217":1,"218":1,"219":1,"221":1,"222":2,"223":3,"225":1,"228":2,"229":4,"232":4,"233":2,"235":1,"237":2,"238":1,"246":1,"247":1,"248":1,"249":4,"251":1,"252":1,"255":3,"256":4,"260":8,"263":1,"264":3,"265":3,"266":3,"268":11,"269":1,"270":2,"271":1,"272":5,"274":1,"276":1,"278":5,"279":2,"280":1,"282":2,"283":1,"284":2,"285":4,"288":4,"289":5,"290":1,"291":2,"292":2,"293":4,"294":1,"295":1,"297":1,"299":1,"302":3,"304":6,"305":2,"307":6,"308":2,"309":1,"311":1,"312":1,"313":1,"314":2,"316":1,"317":4,"319":1,"320":2,"329":2,"331":1,"334":1,"335":1,"337":1,"340":1,"346":2,"347":1,"348":1,"351":9,"352":1,"353":2,"354":1,"355":1,"357":5,"360":1,"365":3,"367":3,"369":1,"371":1,"372":1,"373":7,"374":5,"375":1,"376":2,"377":1,"379":1,"380":4,"382":1,"384":2,"385":3,"386":6,"387":3,"388":2,"389":1,"392":14,"395":1,"396":2,"397":3,"398":2,"399":3,"400":4,"402":2,"403":3,"406":1,"407":1,"408":3,"409":2,"410":2,"411":2,"412":4,"414":2,"415":3,"416":4,"417":1,"418":3,"419":8,"421":3,"422":9,"423":2,"424":7,"425":3,"426":2,"429":1,"430":1,"431":1,"434":1,"435":2,"438":4,"439":1,"440":4,"441":1,"442":4,"444":2,"445":2,"447":4,"449":6,"450":3,"451":1,"458":1,"460":1,"462":1,"463":1,"465":2,"470":1,"471":3,"477":1,"478":5,"485":1,"488":6,"491":2,"494":5,"495":2,"501":3,"503":9,"504":1,"507":2,"509":7,"510":1,"511":1,"512":2,"514":1,"515":2,"519":1,"521":1,"522":2,"523":1,"526":1,"527":1,"528":1,"531":1,"533":1,"535":1,"538":1,"539":2,"541":2,"542":1,"544":2,"548":1,"552":3,"554":1,"555":1,"556":1,"557":2,"559":7,"561":1,"562":1,"564":1,"566":1,"568":1,"573":1,"575":2,"576":2,"578":1,"579":1,"584":1,"585":4,"586":2,"588":2,"594":1,"596":1,"598":1,"600":1,"601":2,"602":1,"606":1,"608":1,"610":2,"611":4,"613":2,"625":2,"626":1,"627":4,"628":1,"629":2,"630":1,"631":2,"632":5,"635":2,"638":1,"639":1,"650":1,"655":1,"658":1,"659":1,"660":1,"666":2,"667":4,"671":3,"679":4,"685":2,"688":3,"689":1,"691":2,"692":1,"696":3,"698":4,"701":1,"706":1,"707":2,"708":1,"709":3,"712":1,"714":1,"715":3,"716":1,"722":1,"727":3,"728":1}}],["alias",{"0":{"464":1},"2":{"464":1,"492":2}}],["alignitems",{"2":{"268":4}}],["align",{"2":{"44":1}}],["aligned",{"2":{"9":1,"24":1}}],["although",{"2":{"425":3}}],["alternatives",{"2":{"600":1}}],["alternatively",{"2":{"119":1,"139":2,"181":1,"188":1,"312":1,"352":1,"418":1,"425":1}}],["alternative",{"2":{"110":1,"169":1,"170":1}}],["algorithm",{"2":{"421":1}}],["alertmanager",{"2":{"503":2}}],["alerts",{"0":{"505":1},"2":{"503":2,"505":1}}],["alert",{"2":{"398":4}}],["alerting",{"2":{"27":1}}],["along",{"2":{"135":1,"148":1,"159":1,"161":2,"207":1,"263":1,"309":1,"365":1,"438":1}}],["alongside",{"2":{"48":1,"105":1,"627":1}}],["alphab",{"2":{"452":1,"453":1,"674":1,"675":1,"676":1,"679":1,"683":1}}],["alpha",{"2":{"94":1,"149":2,"377":1}}],["alchemy",{"2":{"88":3}}],["alcohol",{"2":{"41":1}}],["already",{"2":{"24":1,"46":1,"47":1,"50":1,"89":1,"196":1,"279":1,"302":1,"305":1,"311":1,"388":1,"419":1,"505":1,"518":1,"543":1,"568":1,"571":1,"586":1,"632":1,"633":1,"635":1,"644":1}}],["also",{"2":{"10":1,"24":1,"41":3,"44":1,"45":1,"46":2,"48":1,"69":1,"84":2,"109":1,"136":1,"139":2,"149":1,"166":1,"180":1,"184":1,"191":1,"206":1,"237":1,"243":1,"244":1,"255":1,"260":4,"263":1,"267":1,"288":1,"295":1,"309":1,"312":1,"315":1,"317":1,"360":1,"365":1,"398":2,"419":1,"424":1,"425":1,"460":1,"478":2,"501":1,"504":1,"509":1,"517":1,"529":1,"547":1,"566":1,"573":1,"574":1,"601":1,"626":1,"648":1,"649":1,"659":1,"685":1,"688":1,"703":1,"707":1,"708":1}}],["allocation",{"0":{"449":1},"1":{"450":1},"2":{"449":1,"450":1}}],["allocating",{"2":{"46":1}}],["allocated",{"2":{"449":1,"450":1,"495":1}}],["allocate",{"2":{"41":1}}],["allotting",{"2":{"43":1}}],["allowing",{"2":{"47":1,"351":1,"637":1,"685":1}}],["allow",{"2":{"44":2,"84":1,"90":1,"167":1,"195":1,"201":2,"206":1,"215":1,"315":1,"357":1,"380":1,"398":2,"402":1,"442":3,"541":1,"542":1,"548":2,"552":1,"578":1,"587":1,"629":4,"635":4,"654":1,"689":1,"690":1,"691":1,"717":1}}],["allows",{"2":{"23":1,"46":2,"47":1,"84":1,"105":1,"109":2,"110":1,"116":1,"129":1,"131":1,"150":1,"152":1,"163":1,"165":1,"168":1,"170":1,"172":1,"174":1,"176":2,"248":1,"267":1,"268":3,"280":1,"293":1,"302":1,"320":1,"350":1,"373":3,"380":1,"398":1,"417":1,"426":1,"460":1,"505":1,"541":2,"552":2,"557":1,"575":1,"606":1,"611":1,"652":1,"690":1,"707":2,"708":2}}],["allowed",{"2":{"15":1}}],["alleged",{"2":{"31":1}}],["all",{"0":{"240":1,"596":1},"2":{"3":1,"5":2,"10":1,"11":2,"27":2,"32":1,"33":1,"34":1,"47":1,"48":1,"52":3,"64":1,"76":1,"91":2,"134":2,"162":1,"166":1,"201":1,"206":2,"229":1,"231":3,"232":2,"233":1,"260":3,"268":2,"272":1,"308":1,"313":1,"319":6,"345":1,"351":1,"363":1,"372":1,"373":1,"385":1,"386":1,"387":1,"392":2,"396":1,"402":3,"404":3,"405":1,"411":1,"419":3,"422":3,"425":1,"441":1,"442":1,"448":1,"450":1,"466":1,"471":1,"488":1,"492":1,"519":2,"532":1,"536":1,"541":1,"552":1,"575":1,"578":1,"594":1,"596":2,"599":1,"602":1,"605":3,"607":2,"612":1,"613":1,"631":2,"644":2,"648":2,"650":4,"666":1,"679":1,"680":1,"696":1,"727":1}}],["always",{"2":{"2":1,"40":1,"107":1,"237":1,"357":1,"385":1,"466":1,"610":2,"703":1}}],["a",{"0":{"48":1,"60":1,"61":1,"73":1,"100":1,"102":1,"103":1,"112":1,"134":1,"151":1,"177":1,"178":1,"179":1,"180":1,"189":1,"197":1,"198":1,"199":1,"225":1,"226":1,"228":1,"229":1,"231":1,"234":1,"245":1,"271":1,"272":1,"308":1,"309":1,"334":1,"335":1,"347":1,"356":1,"357":1,"358":1,"367":1,"392":1,"393":1,"399":1,"400":1,"420":1,"424":1,"430":1,"435":1,"455":1,"477":1,"486":1,"489":1,"500":1,"511":1,"519":1,"527":1,"529":1,"530":1,"534":1,"536":1,"537":1,"540":1,"560":1,"587":1,"589":1,"594":2,"612":1,"621":1,"632":1,"643":1,"645":1,"646":1,"652":1,"654":1,"663":1,"710":1,"717":1,"719":1,"721":1},"1":{"74":1,"75":1,"76":1,"77":1,"78":1,"101":1,"102":1,"103":1,"113":1,"114":1,"115":1,"116":1,"117":1,"118":1,"119":1,"120":1,"121":1,"122":1,"178":1,"179":1,"180":1,"235":1,"236":1,"237":1,"238":1,"239":1,"240":1,"241":1,"242":1,"243":1,"244":1,"246":1,"247":1,"248":1,"249":1,"250":1,"251":1,"252":1,"253":1,"254":1,"273":1,"274":1,"275":1,"421":1,"425":1,"478":1,"479":1,"480":1,"481":1,"482":1,"483":1,"484":1,"485":1,"486":1,"487":1,"512":1,"513":1,"514":1,"515":1,"516":1,"517":1,"518":1,"519":1,"520":1,"521":1,"522":1,"523":1,"524":1,"525":1,"526":1,"527":1,"528":1,"529":1,"530":1,"531":1,"532":1,"533":1,"535":1,"536":1,"537":1,"588":1,"589":1,"590":2,"591":2,"592":2,"593":1,"594":1,"595":3,"596":3,"597":1,"598":1,"599":1,"600":1,"601":1,"602":1,"603":1,"604":1,"605":1,"606":1,"607":1,"608":1,"609":1,"610":1,"611":1,"613":1,"614":1,"615":1,"616":1,"617":1,"618":1,"619":1,"620":1,"621":1,"622":1,"623":1,"644":1,"645":1,"646":1,"647":1,"648":1,"649":1,"650":1,"651":1,"652":1,"653":1,"655":1,"656":1,"657":1,"658":1,"659":1,"660":1,"661":1,"662":1,"663":1,"664":1,"665":1,"711":1,"712":1,"713":1,"714":1,"715":1,"716":1,"718":1,"719":1,"720":2,"721":2,"722":1,"723":1,"724":1,"725":1,"726":1,"727":1,"728":1,"729":1,"730":1},"2":{"2":4,"3":3,"7":1,"8":3,"13":2,"14":5,"15":4,"16":2,"19":3,"21":1,"22":2,"23":2,"24":5,"25":1,"26":1,"27":2,"29":1,"30":5,"31":10,"32":3,"33":2,"34":15,"35":2,"36":6,"37":1,"39":3,"40":12,"41":3,"42":8,"43":7,"44":7,"45":3,"46":3,"47":8,"48":4,"50":1,"52":2,"53":2,"54":1,"56":1,"57":1,"59":1,"60":7,"61":5,"65":5,"66":2,"71":1,"72":1,"73":1,"74":1,"75":3,"76":3,"77":1,"78":1,"79":3,"80":3,"82":2,"83":2,"84":15,"85":3,"86":1,"88":7,"89":10,"90":1,"91":2,"92":1,"101":1,"102":2,"103":3,"105":1,"106":1,"107":21,"108":5,"109":3,"110":2,"113":2,"115":2,"116":3,"117":2,"119":2,"120":3,"121":3,"122":2,"126":2,"127":3,"128":8,"129":4,"131":13,"133":3,"134":15,"135":19,"136":3,"137":3,"139":6,"145":3,"146":10,"148":17,"150":5,"152":11,"153":2,"155":4,"156":3,"158":2,"159":8,"161":2,"163":1,"165":1,"166":2,"167":5,"168":1,"169":1,"170":7,"171":1,"172":6,"173":12,"174":1,"176":3,"177":3,"178":3,"181":7,"184":1,"185":1,"186":1,"189":1,"191":2,"192":1,"195":1,"196":2,"197":1,"198":1,"199":1,"201":2,"203":2,"204":1,"205":3,"206":7,"207":3,"208":3,"211":1,"212":1,"213":4,"214":2,"217":1,"218":1,"222":3,"223":1,"225":1,"226":1,"228":3,"229":5,"231":7,"232":3,"233":2,"234":1,"235":1,"237":8,"238":1,"239":1,"243":2,"244":1,"245":1,"246":1,"247":2,"248":3,"249":2,"251":3,"252":3,"253":3,"254":1,"255":2,"256":1,"258":1,"259":1,"260":7,"262":1,"263":5,"265":7,"266":1,"267":3,"268":5,"270":1,"271":1,"272":8,"273":1,"274":1,"278":1,"280":1,"283":1,"284":2,"285":4,"286":2,"288":1,"289":2,"290":5,"291":2,"292":1,"293":2,"294":1,"295":1,"297":1,"302":1,"303":1,"307":3,"308":2,"309":2,"310":2,"311":1,"312":1,"314":2,"315":1,"316":3,"317":7,"318":1,"320":2,"329":3,"331":1,"334":3,"337":1,"346":1,"347":4,"348":1,"351":6,"352":2,"354":2,"355":3,"357":4,"358":2,"360":2,"365":1,"367":3,"370":3,"371":1,"372":1,"373":24,"374":4,"375":1,"376":2,"377":1,"378":3,"379":3,"380":5,"382":1,"383":1,"384":6,"385":10,"386":13,"387":4,"388":7,"392":10,"394":2,"395":2,"396":1,"398":2,"399":1,"400":1,"402":2,"403":5,"404":3,"405":6,"406":5,"407":3,"408":2,"409":4,"410":1,"411":2,"412":4,"414":1,"415":5,"416":9,"417":3,"418":7,"419":4,"421":5,"422":10,"423":3,"424":13,"425":9,"426":2,"429":1,"430":1,"431":1,"435":1,"438":5,"439":3,"441":3,"442":2,"444":4,"447":2,"448":1,"449":2,"450":1,"451":4,"457":1,"458":6,"459":3,"460":1,"465":2,"466":2,"471":3,"473":2,"478":6,"485":5,"486":2,"487":2,"488":10,"491":4,"494":7,"495":4,"497":2,"499":1,"500":1,"501":2,"502":2,"503":16,"504":1,"505":3,"506":3,"507":2,"509":10,"510":2,"511":4,"515":1,"517":4,"518":1,"519":3,"522":3,"523":1,"525":2,"526":4,"527":3,"529":2,"530":2,"531":2,"532":1,"533":3,"534":1,"537":1,"538":1,"539":1,"540":2,"541":5,"542":2,"543":1,"547":6,"548":2,"550":1,"551":1,"552":5,"553":3,"556":4,"557":1,"559":2,"560":2,"561":1,"565":2,"568":1,"569":1,"571":5,"573":3,"574":4,"575":2,"576":2,"577":1,"578":6,"579":1,"580":2,"583":3,"587":1,"588":3,"592":3,"594":1,"598":2,"599":1,"600":1,"601":3,"602":2,"603":1,"606":1,"607":1,"608":2,"610":1,"611":2,"612":3,"620":7,"621":2,"622":2,"624":2,"625":3,"626":2,"627":1,"628":2,"630":1,"632":1,"633":3,"634":17,"638":3,"639":2,"643":1,"644":2,"646":2,"652":1,"653":1,"654":1,"656":1,"660":7,"661":1,"663":2,"664":1,"666":1,"667":1,"670":2,"671":9,"672":1,"679":2,"681":1,"685":2,"688":4,"691":2,"693":2,"696":4,"701":1,"703":1,"706":1,"707":1,"708":1,"709":3,"710":1,"711":2,"712":1,"717":1,"718":1,"719":1,"720":1,"721":3,"722":4,"724":1,"726":2,"727":3,"728":1}}],["bzsidenode",{"2":{"158":6,"159":6,"161":8}}],["blind",{"2":{"519":2}}],["blue",{"2":{"146":1}}],["blobtypes",{"2":{"387":4,"392":6}}],["blobtxs",{"2":{"389":1}}],["blobtx",{"2":{"382":1,"392":4}}],["blobpointer",{"2":{"107":1}}],["blobspace",{"0":{"437":1,"457":1},"1":{"438":1,"439":1}}],["blobsharerange",{"2":{"161":4}}],["blobs",{"0":{"301":1,"382":1},"1":{"383":1,"384":1,"385":1,"386":1,"387":1,"388":1,"389":1,"390":1,"391":1,"392":1,"393":1},"2":{"107":1,"146":1,"299":1,"319":1,"349":2,"352":2,"382":1,"385":1,"386":3,"438":2,"441":1,"471":1,"679":1,"688":1}}],["blobstreamwrapper",{"2":{"160":4}}],["blobstreamxdatacommitmentstored",{"2":{"151":4,"159":5,"160":3}}],["blobstreamxwrapper",{"2":{"151":14,"159":6,"160":4}}],["blobstreamx",{"0":{"198":1},"2":{"89":4,"107":3,"109":2,"126":2,"127":15,"129":1,"151":14,"153":6,"159":8,"161":4,"207":3,"208":1}}],["blobstream",{"0":{"95":1,"109":1,"123":1,"126":1,"132":1,"133":1,"141":1,"144":1,"151":1,"163":1,"196":1,"200":1,"201":1,"202":1,"206":1,"207":1,"208":1,"209":1,"211":1},"1":{"96":1,"97":1,"124":1,"125":1,"126":1,"127":1,"128":1,"129":1,"130":1,"131":1,"133":1,"134":1,"135":1,"136":1,"137":1,"138":1,"139":1,"140":1,"141":1,"142":1,"143":1,"145":1,"146":1,"147":1,"148":1,"149":1,"150":1,"151":1,"152":1,"153":1,"154":1,"155":1,"156":1,"157":1,"158":1,"159":1,"160":1,"161":1,"162":1,"164":1,"165":1,"166":1,"167":1,"168":1,"169":1,"170":1,"171":1,"172":1,"173":1,"174":1,"175":1,"176":1,"177":1,"178":1,"179":1,"180":1,"181":1,"182":1,"183":1,"184":1,"185":1,"186":1,"187":1,"188":1,"189":1,"190":1,"191":1,"192":1,"193":1,"194":1,"195":1,"197":1,"198":1,"199":1,"201":1,"202":1,"203":2,"204":2,"205":2,"206":1,"207":1,"208":2,"209":2,"210":2,"211":2},"2":{"27":1,"88":3,"89":6,"94":1,"95":1,"106":1,"108":1,"109":1,"110":1,"126":4,"127":8,"130":1,"131":9,"133":3,"134":2,"135":4,"136":3,"137":1,"139":2,"141":1,"146":2,"148":8,"150":2,"159":2,"161":8,"162":1,"163":2,"166":2,"169":1,"172":1,"173":2,"176":2,"184":1,"196":1,"198":1,"201":7,"203":2,"204":2,"205":3,"206":6,"207":9,"208":3,"209":4,"210":8,"211":2,"255":1,"376":3,"377":4,"378":1,"379":2,"626":3}}],["blob",{"0":{"334":1},"2":{"89":1,"91":1,"107":1,"108":2,"146":2,"148":1,"151":4,"154":1,"155":1,"159":4,"161":4,"312":7,"317":12,"318":1,"319":10,"320":3,"334":5,"351":6,"352":2,"373":2,"382":2,"385":6,"386":6,"387":6,"390":2,"391":3,"392":14,"393":1,"438":1,"439":1,"441":1,"671":4}}],["blocktimes",{"2":{"283":1}}],["blockdataroot",{"2":{"159":3,"161":8}}],["blockres",{"2":{"151":2}}],["blocknumber",{"2":{"121":2,"151":2,"159":2,"253":2,"348":2}}],["block",{"0":{"90":1,"139":1,"142":1,"151":1,"326":1,"388":1,"407":1,"596":1},"2":{"89":1,"91":12,"127":16,"128":2,"129":1,"131":5,"133":1,"134":34,"135":2,"136":4,"137":17,"139":15,"146":4,"148":3,"150":3,"151":10,"152":1,"159":20,"161":6,"166":3,"169":1,"170":3,"172":1,"181":1,"185":1,"201":1,"206":2,"207":4,"213":1,"283":1,"290":1,"302":1,"317":1,"319":6,"320":1,"326":10,"334":1,"347":3,"351":2,"373":5,"374":2,"388":2,"392":2,"399":1,"402":3,"403":4,"404":2,"405":4,"406":4,"407":2,"410":1,"413":1,"415":1,"416":8,"417":3,"418":4,"419":3,"421":2,"423":2,"424":10,"425":13,"438":3,"440":1,"442":4,"445":3,"448":1,"460":1,"471":1,"478":2,"495":1,"507":2,"594":1,"596":2,"599":3,"600":2,"601":1,"611":3,"613":2,"625":2,"626":1,"629":4,"655":1,"671":3,"679":1,"721":1,"727":1}}],["blockspaceracecelestia",{"2":{"367":1}}],["blockspacerace",{"2":{"367":3}}],["blockspace",{"2":{"367":3}}],["blocks",{"0":{"137":1},"2":{"84":2,"90":1,"134":3,"135":4,"136":1,"137":12,"139":13,"148":4,"159":2,"170":2,"176":2,"213":1,"283":2,"285":1,"290":1,"380":1,"388":2,"402":1,"406":2,"407":1,"410":1,"417":1,"418":1,"441":3,"442":3,"445":1,"478":3,"509":2,"517":1,"596":3,"599":4,"601":1,"688":2,"696":2,"715":1}}],["blockscout",{"2":{"69":1,"90":2}}],["blockchains",{"0":{"230":1,"283":1,"414":1,"422":1},"1":{"231":1,"232":1,"233":1},"2":{"59":3,"62":1,"65":1,"136":1,"218":1,"223":2,"229":3,"231":2,"232":2,"233":1,"256":4,"288":1,"374":2,"402":2,"404":1,"412":1,"414":1,"422":6,"423":1,"509":1}}],["blockchain",{"0":{"229":1,"256":1,"411":1,"420":1},"1":{"257":1,"258":1,"259":1,"260":1,"261":1,"262":1,"263":1,"264":1,"265":1,"266":1,"267":1,"268":1,"269":1,"270":1,"271":1,"421":1},"2":{"19":1,"31":1,"40":1,"57":1,"59":1,"134":3,"135":3,"139":1,"143":1,"215":3,"228":1,"229":3,"231":3,"256":1,"268":2,"283":2,"284":2,"285":1,"288":3,"289":3,"365":3,"374":1,"380":1,"392":2,"402":1,"403":1,"409":1,"411":1,"412":1,"414":1,"415":2,"421":3,"422":1,"423":2,"426":1,"444":2,"457":1,"458":2,"587":1,"588":1,"600":2,"602":1,"624":1,"627":1,"696":1}}],["blog",{"2":{"36":1,"38":1,"260":43,"262":6,"265":3,"268":12}}],["bidirectional",{"2":{"569":1}}],["bitszn",{"2":{"674":1,"675":1,"676":1,"689":1,"690":1,"691":1}}],["bitter",{"2":{"519":2}}],["bitcoin",{"2":{"256":1}}],["billion",{"2":{"450":1}}],["bip44",{"2":{"398":2}}],["bip39",{"2":{"239":1}}],["bi",{"2":{"386":2}}],["bigger",{"2":{"402":1,"406":1}}],["big",{"2":{"151":8,"155":4,"158":4,"159":8,"161":18}}],["binaries",{"2":{"304":1,"540":2,"560":2,"584":1}}],["binarymerkleproof",{"2":{"127":4,"128":1,"151":2,"153":4,"158":9,"159":5,"161":8}}],["binary",{"0":{"539":1,"540":1,"560":1,"562":1},"2":{"94":1,"102":1,"108":2,"127":2,"128":1,"131":1,"148":1,"150":1,"152":3,"153":2,"159":2,"172":2,"173":1,"236":4,"307":4,"367":1,"483":1,"488":2,"510":2,"515":1,"538":1,"539":5,"540":4,"559":4,"560":4,"562":1,"575":1,"625":4,"658":1}}],["bin",{"2":{"243":2,"244":2,"304":2,"305":9,"507":2,"539":1,"540":1,"560":1,"585":2,"586":9}}],["bind",{"2":{"151":6,"159":2,"160":2,"161":4,"607":1}}],["bindings",{"2":{"151":2,"161":2,"207":1}}],["binding",{"2":{"31":1,"151":2,"159":2,"161":2}}],["biometric",{"2":{"33":3}}],["b",{"2":{"31":2,"32":1,"34":2,"139":10,"161":24,"307":1,"449":1,"450":1,"539":6,"559":1,"634":8}}],["brevity",{"2":{"422":1}}],["break",{"2":{"466":1,"565":1}}],["breaking",{"2":{"387":1}}],["breaks",{"2":{"37":1,"43":1}}],["brew",{"2":{"304":4,"503":1,"585":4}}],["br",{"2":{"268":2}}],["brush",{"2":{"229":1}}],["broken",{"2":{"519":2,"625":1}}],["browsers",{"2":{"297":1}}],["browser",{"0":{"429":1,"434":1},"2":{"80":1,"429":1,"503":1,"681":1}}],["broadcasting",{"2":{"471":1,"522":2,"531":2,"679":1,"688":1,"689":1,"727":2}}],["broadcasttx",{"2":{"392":2}}],["broadcastforge",{"2":{"263":1,"264":1}}],["broadcast",{"2":{"75":1,"263":1,"264":1,"272":1,"423":1,"507":4,"531":2,"691":1}}],["broadcastcd",{"2":{"75":1,"272":1}}],["broader",{"2":{"30":1,"43":1,"46":1,"65":1,"667":1}}],["branding",{"0":{"58":1}}],["brand",{"2":{"47":1,"58":1}}],["branch",{"2":{"2":3}}],["brightlystake",{"2":{"674":1,"675":1,"676":1,"689":1,"690":1,"691":1}}],["brightest",{"2":{"65":1}}],["bridging",{"0":{"66":1},"1":{"67":1,"68":1,"69":1,"70":1,"71":1,"72":1},"2":{"66":1,"85":1,"206":2,"218":1}}],["bridge=",{"2":{"542":4}}],["bridgefulllight",{"2":{"237":1,"238":1,"239":1}}],["bridgesudo",{"2":{"557":1}}],["bridges",{"2":{"206":1,"696":1}}],["bridgecreator",{"2":{"98":1,"99":1}}],["bridge",{"0":{"67":1,"70":1,"218":1,"477":1,"478":1,"480":1,"482":1,"484":1,"485":1,"486":1,"487":1,"594":1,"688":1,"692":1,"715":1,"724":1,"725":1,"726":1},"1":{"68":1,"69":1,"71":1,"72":1,"478":1,"479":1,"480":1,"481":2,"482":1,"483":2,"484":2,"485":2,"486":3,"487":2,"595":1,"596":1,"726":1},"2":{"24":1,"27":3,"66":2,"68":1,"69":1,"71":1,"72":2,"82":2,"86":1,"91":8,"92":1,"98":1,"99":1,"103":1,"135":2,"161":4,"201":1,"218":2,"235":1,"237":2,"238":2,"239":2,"246":1,"255":1,"308":1,"422":1,"470":6,"471":2,"477":2,"478":4,"479":1,"483":1,"484":9,"485":12,"486":9,"487":2,"541":2,"542":4,"552":2,"555":1,"556":4,"557":10,"565":2,"566":1,"569":1,"571":3,"574":1,"594":1,"678":1,"679":2,"685":1,"688":2,"692":7,"696":1,"707":1,"715":20,"716":1,"722":3,"724":2,"725":2,"726":2}}],["briefly",{"2":{"44":1,"85":1}}],["bringing",{"2":{"503":1}}],["bring",{"2":{"41":1,"65":1}}],["bound",{"2":{"671":1}}],["bounds",{"2":{"172":2,"177":1}}],["bot",{"2":{"630":1}}],["both",{"2":{"23":1,"86":1,"91":1,"95":1,"119":1,"135":1,"184":1,"205":1,"285":1,"355":1,"386":1,"415":1,"419":1,"422":1,"438":1,"447":1,"503":1,"511":1,"531":1,"538":1,"630":1,"709":1}}],["bonded",{"2":{"509":2}}],["bonddenomalias",{"2":{"464":1}}],["bonddenom",{"2":{"463":1}}],["bonus",{"2":{"375":1}}],["bona",{"2":{"34":1}}],["border",{"2":{"268":2}}],["borrowing",{"2":{"40":1}}],["bob",{"2":{"260":4}}],["box",{"2":{"233":1}}],["boxes",{"2":{"71":1}}],["bootstrappeers",{"2":{"568":2}}],["bootstrappers",{"2":{"568":1,"679":1}}],["bootstrapper",{"2":{"542":1,"568":3,"692":1}}],["bootstrapping",{"0":{"458":1},"2":{"442":1}}],["bootstrap",{"0":{"568":1},"2":{"458":1,"571":1}}],["boolean",{"2":{"544":2,"547":1}}],["bool",{"2":{"127":2,"151":2,"161":2,"260":6}}],["boost",{"2":{"46":1}}],["book",{"2":{"40":1,"143":1,"381":1}}],["board",{"2":{"43":2}}],["body",{"2":{"7":1,"522":2}}],["bak",{"2":{"592":10,"608":2}}],["bare",{"2":{"577":1}}],["barrier",{"2":{"84":2}}],["bad",{"2":{"407":2}}],["balances",{"0":{"525":1},"2":{"411":2,"471":1,"493":5,"522":1,"525":6,"537":2,"679":1,"688":1}}],["balance",{"0":{"322":1,"323":1,"329":1},"2":{"91":4,"205":1,"255":1,"270":1,"322":3,"323":7,"399":1,"400":1,"495":1,"525":3,"598":1,"620":1,"660":1}}],["batchsubmitter",{"2":{"351":2}}],["batcheraddr",{"2":{"351":2}}],["batcher",{"2":{"213":3,"285":1,"290":3,"343":1,"346":1,"348":1,"351":3,"352":1,"353":1}}],["batched",{"2":{"204":1}}],["batches",{"0":{"353":1},"2":{"84":2,"88":1,"107":1,"109":2,"128":1,"151":2,"159":2,"161":2,"169":1,"207":2,"213":1,"285":1,"290":1,"353":2,"424":1}}],["batchposter",{"2":{"110":1}}],["batch",{"0":{"88":1},"2":{"84":4,"85":1,"86":2,"88":2,"89":5,"91":3,"93":1,"107":8,"129":2,"148":4,"159":1,"342":2,"343":1,"351":3,"353":4,"359":2,"629":8}}],["backward",{"2":{"624":1}}],["backed",{"2":{"610":2}}],["backers",{"2":{"449":2,"450":2}}],["backend=test",{"2":{"727":2}}],["backend=",{"2":{"507":2}}],["backend",{"2":{"237":6,"238":6,"239":6,"243":2,"244":2,"264":2,"270":2,"310":4,"367":2,"485":2,"489":4,"507":8,"527":3,"531":4,"536":2,"547":1,"620":2,"646":2,"647":2,"649":2,"661":4}}],["background",{"0":{"710":1},"1":{"711":1,"712":1,"713":1,"714":1,"715":1,"716":1},"2":{"150":2,"151":4,"161":2,"373":8,"374":2,"487":1,"603":1,"622":1,"664":1,"710":1,"711":1,"712":1,"716":1,"726":1}}],["back",{"2":{"72":1,"88":1,"107":1,"110":1,"139":1,"177":3,"213":1,"255":1,"290":1,"350":1,"351":1,"352":1,"373":5,"374":2,"375":1,"628":1,"638":1}}],["basically",{"2":{"233":1}}],["basic",{"0":{"313":1},"1":{"314":1,"315":1,"316":1},"2":{"114":2,"246":2,"260":2,"268":2,"312":1,"313":1,"503":1,"572":1,"576":2,"610":1}}],["basechainhandle",{"2":{"634":8}}],["baseaccount",{"2":{"524":3}}],["basefeepergas",{"2":{"347":1}}],["base64str",{"2":{"373":4}}],["base64=true",{"2":{"319":5}}],["base64",{"2":{"155":1,"156":1,"268":2,"317":4,"319":4,"335":1,"348":3,"372":2,"373":9}}],["base",{"0":{"97":1,"99":1,"524":1},"2":{"82":2,"84":13,"85":6,"86":1,"89":2,"91":12,"92":1,"95":1,"97":1,"121":1,"183":1,"207":1,"210":2,"253":1,"268":1,"376":1,"422":2,"523":4}}],["based",{"2":{"23":2,"24":1,"80":1,"131":1,"203":1,"278":1,"283":2,"307":1,"320":1,"365":1,"385":2,"386":3,"396":1,"414":1,"426":1,"439":1,"444":1,"445":1,"458":1,"539":1,"559":1}}],["bashmoniker=",{"2":{"727":1}}],["bashmake",{"2":{"307":4,"343":1,"344":1,"345":2,"360":1,"361":1,"362":1,"363":1,"539":1,"559":4}}],["bashhermes",{"2":{"630":1,"631":1,"633":3,"634":2}}],["bashvalidator",{"2":{"721":1}}],["bashvim",{"2":{"629":1}}],["bashver=",{"2":{"305":3,"586":3}}],["bashpersistent",{"2":{"592":3}}],["bashprometheus",{"2":{"503":1}}],["bashprivate",{"2":{"91":1,"264":1,"270":1}}],["bashsuccess",{"2":{"631":1,"634":2}}],["bashsudo",{"2":{"304":4,"557":2,"578":1,"585":4}}],["bashseeds=$",{"2":{"592":2}}],["bashnano",{"2":{"557":1}}],["bashnpm",{"2":{"270":1}}],["bashbash",{"2":{"540":1,"560":1}}],["bashbalances",{"2":{"525":2}}],["bashblockhash",{"2":{"121":1,"253":1}}],["bash$home",{"2":{"357":6}}],["bashgas",{"2":{"522":1}}],["bashgrafana",{"2":{"503":1}}],["bashgrpcurl",{"2":{"349":1,"352":1}}],["bashgo",{"2":{"305":1,"586":1}}],["bashgit",{"2":{"87":1,"307":3,"341":2,"539":3,"559":3}}],["bashzsh",{"2":{"305":1,"586":1}}],["bashanvil",{"2":{"263":1}}],["bashanvilanvil",{"2":{"251":1}}],["bashroot",{"2":{"244":1}}],["bashdocker",{"2":{"243":4,"244":4,"346":3,"577":3,"579":3,"580":3}}],["bashenter",{"2":{"721":1}}],["basherror",{"2":{"557":1}}],["bashecho",{"2":{"121":1,"253":1,"305":2,"586":2}}],["bashexport",{"2":{"75":2,"76":1,"120":2,"251":1,"252":2,"263":1,"264":1,"272":3,"314":2,"315":1,"329":1,"334":1,"342":1,"347":1,"348":1,"359":1,"515":1,"521":1,"527":1,"529":1,"542":2,"577":7}}],["bash0x000000000000000000000000000000000000000000000000000000000000000a0x000000000000000000000000000000000000000000000000000000000000000a",{"2":{"121":1,"253":1}}],["bash",{"2":{"116":1,"117":1,"120":1,"237":3,"238":3,"239":3,"243":2,"244":1,"248":1,"249":1,"252":1,"304":3,"305":4,"310":2,"316":2,"334":1,"502":1,"507":1,"515":1,"519":1,"520":2,"522":3,"523":2,"524":2,"525":2,"540":1,"547":1,"553":2,"560":1,"578":1,"579":1,"580":1,"585":3,"586":4,"592":1,"635":2}}],["bashfatal",{"2":{"556":1}}],["bashforge",{"2":{"115":1,"116":1,"117":1,"120":1,"247":1,"248":1,"249":1,"251":1,"252":1,"259":1,"261":1,"263":1,"264":1}}],["bashfunding",{"2":{"91":1}}],["bashcurl",{"2":{"334":1,"502":1}}],["bashcp",{"2":{"265":1}}],["bashcontainer",{"2":{"243":1,"244":1}}],["bashcelestia",{"2":{"89":2,"307":1,"308":3,"309":6,"310":3,"312":3,"315":1,"316":2,"317":4,"319":5,"320":1,"322":1,"323":2,"324":1,"325":1,"326":2,"327":1,"328":1,"329":3,"330":1,"331":2,"332":1,"390":1,"391":1,"471":2,"527":3,"529":2,"542":1,"553":2,"554":2,"555":2,"559":1,"592":6,"637":1,"679":1,"721":2,"724":1,"725":1,"727":1}}],["bashcast",{"2":{"76":3,"121":2,"253":2,"263":2,"264":2,"272":3,"347":1,"348":1}}],["bashcd",{"2":{"75":2,"77":1,"115":1,"247":1,"265":1,"272":2,"273":1,"305":4,"307":1,"341":1,"342":1,"349":1,"359":1,"514":1,"516":1,"519":1,"539":1,"559":1,"578":1,"586":4,"602":3}}],["bashyarn",{"2":{"1":1}}],["bandwidth",{"2":{"283":2,"303":1,"417":1,"479":1,"588":1,"614":1,"656":1,"718":1}}],["bank",{"2":{"278":1,"493":10,"507":4,"525":4,"537":2}}],["banner",{"2":{"44":1}}],["banners",{"2":{"44":1}}],["ban",{"0":{"15":1,"16":1},"2":{"14":1,"15":2,"16":1}}],["buckets",{"2":{"629":8}}],["bug",{"2":{"556":2,"696":1}}],["bugs",{"2":{"31":1}}],["burned",{"2":{"494":1}}],["buffer",{"2":{"268":6}}],["bubs",{"0":{"212":1,"214":1,"245":1,"252":1,"272":1},"1":{"213":1,"214":1,"215":2,"216":2,"217":2,"218":2,"219":2,"220":2,"221":2,"222":1,"246":1,"247":1,"248":1,"249":1,"250":1,"251":1,"252":1,"253":1,"254":1,"273":1,"274":1,"275":1},"2":{"212":1,"213":2,"214":2,"215":1,"216":1,"217":3,"218":2,"219":2,"220":1,"221":1,"222":3,"245":1,"246":5,"252":8,"254":1,"272":6,"286":3,"291":1}}],["budget",{"2":{"41":3,"46":1}}],["buy",{"2":{"34":1}}],["businesses",{"2":{"44":1}}],["business",{"2":{"34":1}}],["buttonstyle",{"2":{"268":6}}],["buttoncontainerstyle",{"2":{"268":6}}],["button",{"2":{"85":1,"86":1,"268":16,"398":4,"434":1,"503":1}}],["but",{"2":{"8":1,"24":1,"27":1,"30":1,"32":1,"33":1,"34":2,"35":1,"40":1,"41":3,"44":1,"47":1,"48":1,"65":1,"84":4,"107":2,"108":1,"134":1,"135":2,"168":1,"205":1,"206":2,"229":1,"264":1,"280":1,"281":1,"290":1,"370":2,"379":1,"388":1,"411":1,"424":1,"441":1,"450":2,"466":1,"504":1,"515":1,"518":1,"599":2,"611":1,"612":1,"625":1,"630":1,"645":1,"667":1,"685":1,"696":1,"722":1}}],["builds",{"2":{"515":1}}],["buildmake",{"2":{"307":1,"559":1}}],["builders",{"2":{"231":1,"289":1}}],["buildforge",{"2":{"116":1,"248":1}}],["build",{"0":{"223":1,"342":1,"359":1,"372":1},"1":{"224":1,"225":1,"226":1,"227":1,"228":1,"229":1,"230":1,"231":1,"232":1,"233":1,"343":1,"344":1,"345":1,"346":1},"2":{"43":1,"116":1,"122":1,"163":1,"176":1,"181":1,"201":1,"211":1,"214":1,"231":1,"248":1,"254":1,"256":3,"275":1,"277":1,"304":5,"305":1,"307":10,"342":1,"359":1,"370":1,"373":1,"380":1,"457":1,"514":2,"515":3,"516":2,"519":2,"539":1,"559":10,"585":4,"586":1,"639":1}}],["building",{"0":{"1":1,"214":1,"224":1,"265":1,"370":1,"420":1,"539":1},"1":{"215":1,"216":1,"217":1,"218":1,"219":1,"220":1,"221":1,"225":1,"226":1,"227":1,"266":1,"371":1,"372":1,"373":1,"374":1,"421":1},"2":{"225":1,"257":1,"265":1,"293":1,"304":1,"367":1,"381":1,"559":1,"584":1,"658":1}}],["built",{"0":{"213":1,"540":1,"560":1},"2":{"1":1,"108":1,"166":1,"176":1,"203":1,"206":1,"212":1,"213":1,"236":1,"275":1,"284":2,"289":1,"290":1,"291":1,"374":1,"380":1,"421":2,"442":1,"459":1,"478":2,"509":1,"538":1,"540":3,"560":3,"627":2,"637":1}}],["bytes=",{"2":{"671":1}}],["bytes28",{"2":{"155":2,"156":2}}],["bytes1",{"2":{"155":2,"156":2}}],["bytes",{"0":{"671":1},"2":{"127":4,"153":2,"154":2,"155":5,"156":1,"157":4,"161":6,"172":1,"317":1,"351":4,"385":1,"392":2,"417":2,"418":3,"424":1,"670":1,"671":4}}],["bytes32",{"2":{"127":8,"155":2,"158":3,"159":2,"161":2,"379":1}}],["byte",{"2":{"89":3,"134":8,"151":10,"155":17,"156":7,"158":4,"159":6,"161":36,"205":1,"373":7,"374":4,"386":1,"387":1,"392":4}}],["by",{"0":{"151":1,"326":1,"597":1},"2":{"8":1,"17":1,"19":1,"21":1,"23":2,"24":2,"30":6,"31":4,"33":3,"34":11,"38":1,"41":1,"42":1,"43":1,"46":1,"47":1,"48":1,"60":1,"65":2,"79":1,"83":1,"84":2,"85":1,"87":1,"88":1,"89":2,"91":3,"107":1,"110":1,"116":1,"117":2,"119":1,"127":2,"129":1,"131":7,"133":1,"134":1,"135":5,"137":2,"139":2,"148":3,"149":1,"150":1,"151":6,"153":4,"156":1,"159":2,"161":6,"162":1,"166":2,"169":1,"170":1,"172":2,"173":2,"176":1,"181":1,"185":1,"201":2,"207":1,"208":1,"213":3,"231":1,"232":1,"233":1,"235":1,"243":1,"244":2,"248":1,"249":2,"255":1,"260":6,"264":1,"265":1,"278":1,"282":2,"283":1,"288":2,"289":1,"290":2,"297":1,"299":1,"301":1,"307":1,"310":3,"317":3,"326":4,"327":4,"332":6,"343":1,"353":2,"357":1,"358":1,"367":1,"373":3,"376":1,"377":2,"379":1,"380":1,"383":2,"384":6,"385":5,"386":6,"387":2,"388":2,"399":1,"403":1,"404":1,"405":1,"406":1,"407":1,"416":2,"417":1,"419":2,"421":3,"423":2,"424":5,"425":3,"439":1,"441":3,"442":3,"445":1,"450":1,"451":1,"458":1,"459":1,"485":2,"488":4,"494":2,"502":1,"503":3,"504":1,"507":2,"508":1,"509":1,"523":1,"537":1,"541":3,"543":1,"552":3,"559":1,"566":2,"568":2,"571":2,"578":1,"592":1,"599":1,"600":1,"601":1,"607":2,"610":2,"620":2,"625":2,"627":2,"628":1,"634":1,"641":1,"642":1,"643":1,"644":1,"651":1,"653":1,"658":1,"661":3,"671":4,"680":1,"689":1,"696":2,"703":1,"721":1}}],["beautiful",{"2":{"265":1}}],["bearer",{"2":{"334":2}}],["bear",{"2":{"31":2,"34":2}}],["bedrock",{"2":{"213":1,"290":2,"346":1,"354":1,"357":1,"360":1}}],["belong",{"2":{"153":2,"166":2,"172":1}}],["below",{"2":{"21":1,"22":1,"34":1,"47":1,"66":1,"84":1,"89":1,"95":1,"96":1,"97":1,"120":1,"136":1,"146":1,"159":1,"169":1,"186":1,"208":1,"252":1,"348":1,"349":1,"360":1,"421":1,"424":2,"442":1,"445":1,"449":1,"450":1,"537":1,"539":1,"540":1,"560":1,"579":1,"580":1,"620":1,"660":1,"688":1,"692":1,"721":1,"727":2}}],["bech",{"2":{"721":2}}],["bech32prefixconspub",{"2":{"398":2}}],["bech32prefixconsaddr",{"2":{"398":2}}],["bech32prefixvalpub",{"2":{"398":2}}],["bech32prefixvaladdr",{"2":{"398":2}}],["bech32prefixaccpub",{"2":{"398":2}}],["bech32prefixaccaddr",{"2":{"398":2}}],["bech32config",{"2":{"398":2}}],["because",{"2":{"108":1,"137":1,"171":1,"233":1,"263":1,"264":1,"283":1,"402":1,"412":1,"628":1}}],["becomes",{"2":{"402":2,"571":1}}],["become",{"2":{"31":1,"53":1,"56":1}}],["behind",{"2":{"107":1,"283":1,"466":1,"608":1,"703":1}}],["behave",{"2":{"506":1}}],["behavior",{"2":{"8":2,"9":2,"11":1,"13":2,"14":1,"15":1,"16":1,"312":1,"509":1,"601":1,"655":1}}],["behalf",{"2":{"30":1,"34":2}}],["beer",{"2":{"41":1}}],["been",{"2":{"27":1,"31":1,"34":4,"91":2,"94":1,"107":1,"110":2,"127":3,"166":1,"263":1,"264":1,"297":1,"302":1,"379":1,"402":1,"411":1,"412":1,"442":2,"509":1,"523":1,"537":1,"552":1,"634":1,"659":1,"670":1,"712":1,"716":1}}],["beginner",{"0":{"573":1},"2":{"573":1}}],["beginning",{"2":{"44":2,"155":2,"159":1,"445":1}}],["beginkey",{"2":{"155":3,"173":2}}],["begin",{"2":{"38":1,"139":1,"173":1,"268":2,"503":1}}],["benefits",{"0":{"230":1},"1":{"231":1,"232":1,"233":1},"2":{"40":1,"44":1,"59":1,"62":1,"65":2,"223":2,"256":1}}],["benefit",{"2":{"34":2,"44":2,"46":1,"109":1}}],["beneficial",{"2":{"34":1,"48":1}}],["better",{"2":{"47":1,"48":1,"135":1,"195":1,"222":1,"442":1}}],["between",{"0":{"410":1,"632":1},"2":{"31":1,"42":1,"43":1,"85":1,"139":1,"173":1,"181":1,"205":1,"218":2,"260":2,"422":1,"445":1,"541":1,"552":1,"613":1,"627":2,"628":1,"629":1,"632":1,"634":1,"696":1,"715":1}}],["betaarabicamocha",{"2":{"663":1}}],["betamochaarabica",{"2":{"305":1,"307":1,"308":1,"309":2,"310":1,"357":2,"398":1,"484":1,"486":1,"539":1,"559":1,"577":2,"579":1,"580":1,"586":1,"592":4,"602":1,"619":1,"621":1,"659":1,"661":1}}],["beta",{"0":{"452":1,"545":1,"666":1,"698":1,"699":1},"1":{"667":1,"668":1,"669":1,"670":1,"671":1,"672":1,"673":1,"674":1,"675":1,"676":1,"677":1,"678":1,"679":1,"680":1,"681":1,"682":1,"683":1,"684":1,"699":1},"2":{"25":1,"110":1,"210":3,"237":1,"243":1,"244":1,"255":1,"285":1,"310":1,"334":1,"357":1,"360":1,"452":1,"485":1,"545":1,"550":1,"551":3,"577":1,"620":1,"660":1,"666":2,"667":2,"672":2,"679":1,"682":1,"698":1}}],["be",{"0":{"153":1},"1":{"154":1,"155":1,"156":1,"157":1,"158":1,"159":1,"160":1,"161":1},"2":{"8":2,"11":2,"13":1,"21":1,"23":3,"26":3,"27":1,"29":1,"31":2,"33":1,"34":14,"40":3,"41":3,"43":1,"44":3,"45":4,"48":4,"53":1,"65":1,"71":2,"72":1,"76":1,"77":1,"83":1,"84":5,"85":1,"88":1,"89":4,"90":2,"95":1,"103":1,"107":3,"108":2,"109":2,"110":1,"117":1,"119":2,"122":1,"126":1,"131":1,"134":4,"135":5,"136":2,"137":3,"139":3,"145":2,"146":2,"148":5,"149":1,"150":2,"152":3,"154":1,"155":2,"156":2,"157":1,"158":1,"159":6,"160":1,"161":3,"162":1,"166":3,"167":4,"168":1,"169":2,"170":4,"172":2,"173":6,"176":2,"178":3,"180":2,"181":5,"183":1,"184":1,"185":4,"186":2,"188":2,"189":2,"191":1,"192":2,"195":1,"201":1,"207":6,"208":2,"211":1,"213":1,"231":4,"233":2,"235":1,"236":1,"237":2,"243":1,"244":1,"249":1,"255":3,"256":2,"257":1,"260":2,"263":2,"265":1,"268":1,"270":2,"271":1,"272":1,"273":1,"275":2,"278":1,"282":1,"283":3,"294":1,"297":2,"302":1,"304":1,"307":1,"308":1,"309":1,"310":1,"313":2,"314":2,"317":6,"320":2,"323":1,"334":1,"351":3,"352":1,"353":3,"357":2,"360":2,"365":3,"366":1,"367":4,"368":1,"369":1,"370":1,"375":1,"376":3,"379":2,"383":1,"384":1,"385":3,"386":1,"387":2,"388":3,"392":4,"394":2,"395":2,"399":1,"400":1,"403":1,"406":1,"407":3,"409":2,"412":2,"413":1,"416":1,"417":1,"418":3,"419":1,"422":3,"424":2,"425":1,"438":2,"439":1,"440":1,"441":4,"442":3,"445":2,"450":2,"458":1,"466":2,"471":1,"483":1,"484":1,"485":1,"486":1,"491":3,"494":5,"495":1,"501":1,"502":1,"504":1,"505":1,"506":1,"507":2,"509":4,"515":1,"526":1,"532":2,"537":1,"539":2,"540":1,"541":1,"544":1,"548":2,"550":2,"552":2,"554":4,"557":1,"560":1,"566":1,"568":1,"571":2,"577":1,"578":2,"584":1,"594":1,"597":1,"601":5,"603":1,"606":1,"607":1,"610":2,"611":1,"620":4,"621":1,"623":2,"624":2,"625":3,"626":3,"627":1,"629":1,"630":2,"634":2,"637":3,"638":1,"643":1,"648":1,"649":1,"650":1,"653":1,"660":2,"661":1,"663":1,"667":1,"670":1,"671":4,"690":1,"692":1,"701":3,"703":2,"707":1,"712":1,"714":2,"715":2,"716":1,"724":1,"727":3}}],["best",{"2":{"8":1,"60":1,"65":1,"232":1,"469":1,"504":1,"672":1,"685":1,"707":1}}],["being",{"2":{"8":1,"24":1,"26":1,"84":1,"89":2,"102":1,"108":1,"137":1,"155":2,"156":1,"157":1,"158":1,"159":3,"185":1,"278":1,"320":1,"396":1,"409":1,"425":1,"478":1,"550":1,"594":1,"638":1}}],["before",{"0":{"36":1},"2":{"2":1,"24":2,"27":1,"42":2,"43":1,"45":1,"77":1,"85":1,"89":2,"110":1,"136":1,"139":3,"169":1,"207":1,"208":1,"269":1,"273":1,"312":1,"373":2,"398":3,"424":1,"517":1,"522":2,"550":1,"578":1,"601":1,"628":1,"644":1,"647":1,"712":1,"727":2}}],["r6a",{"2":{"378":1}}],["r",{"2":{"327":2,"332":2,"348":4,"449":1,"450":2}}],["rf",{"2":{"305":8,"307":2,"539":2,"556":4,"559":2,"586":8,"592":2,"602":6}}],["rfc",{"2":{"128":1}}],["rgba",{"2":{"268":2}}],["r1",{"2":{"146":1}}],["r0",{"2":{"146":1}}],["rm",{"2":{"102":2,"305":16,"307":2,"539":2,"556":4,"559":2,"586":16,"592":2,"602":6}}],["rc",{"2":{"94":1,"102":2}}],["rblocks",{"2":{"86":1}}],["rpcs",{"2":{"471":1,"679":1}}],["rpcurls",{"2":{"267":2}}],["rpc",{"0":{"88":1,"215":1,"280":1,"293":1,"295":1,"296":1,"299":1,"311":1,"393":1,"470":1,"606":1,"607":1,"674":1,"679":1,"680":1,"688":1,"689":1},"1":{"216":1,"217":1,"294":1,"295":1,"296":1,"300":1,"301":1,"302":1,"303":1,"304":1,"305":1,"306":1,"307":1,"308":1,"309":1,"310":1,"311":1,"312":2,"313":2,"314":2,"315":2,"316":2,"317":2,"318":2,"319":2,"320":2,"321":2,"322":2,"323":2,"324":2,"325":2,"326":2,"327":2,"328":2,"329":2,"330":2,"331":2,"332":2,"333":1,"334":1,"335":1,"336":1,"337":1,"607":1,"608":1,"609":1,"681":1},"2":{"75":7,"76":16,"88":3,"89":18,"91":7,"107":1,"120":7,"121":9,"135":4,"142":1,"145":1,"146":1,"150":2,"151":8,"159":4,"160":2,"161":6,"215":2,"216":1,"217":1,"243":2,"244":2,"246":1,"251":7,"252":10,"253":9,"263":2,"264":6,"269":1,"272":25,"280":2,"292":1,"293":3,"295":3,"296":1,"299":1,"302":1,"308":1,"309":3,"311":1,"312":2,"313":4,"314":1,"315":1,"324":1,"325":1,"334":1,"336":1,"337":2,"347":2,"348":2,"360":3,"367":1,"370":3,"372":2,"373":7,"377":2,"393":1,"398":10,"399":2,"400":2,"470":4,"471":5,"478":1,"484":1,"485":2,"493":4,"529":8,"530":1,"531":5,"541":2,"542":1,"552":2,"553":2,"561":1,"577":5,"601":4,"606":1,"607":3,"608":1,"611":3,"620":1,"629":16,"660":3,"661":3,"674":23,"675":1,"676":2,"679":12,"688":4,"689":12,"691":1,"708":1,"724":1}}],["rsvp",{"2":{"36":1}}],["rigorous",{"2":{"666":1}}],["rights",{"2":{"30":2}}],["right",{"2":{"9":1,"34":1,"68":1,"89":1,"223":1,"268":6,"270":1,"434":1,"458":1,"625":1}}],["riit57fvj0gm7tgacaaa",{"2":{"349":2}}],["richer",{"2":{"293":1}}],["risk",{"2":{"201":1}}],["risks",{"2":{"34":2}}],["router",{"2":{"541":1,"552":1}}],["routine",{"2":{"42":1}}],["rounds",{"2":{"405":2}}],["round",{"2":{"326":2}}],["roughly",{"2":{"283":1,"388":1}}],["robusta",{"2":{"316":2}}],["robust",{"2":{"214":1}}],["rowproof",{"2":{"157":1,"158":1,"161":4}}],["rowproofs",{"0":{"158":1},"2":{"153":2,"158":2,"159":1,"161":2,"173":4}}],["rowroots",{"0":{"157":1},"2":{"153":4,"157":3,"161":4}}],["row",{"2":{"108":3,"131":4,"146":1,"148":6,"152":7,"153":2,"155":1,"158":1,"159":1,"166":2,"172":2,"173":7,"326":2,"418":1,"424":4}}],["rows",{"2":{"108":2,"131":3,"148":1,"152":1,"153":8,"155":1,"157":1,"158":1,"166":2,"172":2,"416":1}}],["ro",{"2":{"102":2}}],["roots",{"2":{"107":1,"108":5,"109":1,"131":2,"137":1,"146":1,"148":4,"152":3,"153":2,"155":1,"157":8,"159":1,"161":6,"166":6,"204":1,"206":2,"326":4,"416":3,"417":1,"418":1,"424":2,"425":1,"478":1}}],["root",{"0":{"150":1},"2":{"68":1,"87":2,"90":1,"91":1,"94":1,"108":2,"109":2,"110":1,"121":2,"127":4,"128":4,"131":10,"134":1,"136":5,"137":1,"148":9,"150":8,"151":12,"152":8,"153":14,"157":4,"158":1,"159":26,"161":10,"162":2,"166":8,"167":2,"172":5,"173":15,"185":4,"189":3,"192":2,"201":1,"206":4,"244":2,"253":2,"265":1,"267":2,"313":1,"344":1,"361":1,"416":1,"418":1,"419":1,"424":2,"438":1}}],["rooms",{"2":{"40":1}}],["role",{"0":{"456":1},"1":{"457":1,"458":1,"459":1,"460":1,"461":1,"462":1,"463":1,"464":1},"2":{"60":1,"61":1,"84":1,"373":2,"460":1}}],["roles",{"2":{"36":1}}],["rolling",{"2":{"509":1}}],["rollback",{"2":{"488":8}}],["rollkit",{"0":{"380":1},"1":{"381":1},"2":{"48":1,"61":2,"264":1,"270":1,"349":2,"360":4,"380":4,"381":1}}],["rollupinclusionproofs",{"2":{"154":1,"155":1,"156":1,"158":1,"159":1,"180":1}}],["rollupuserlogic",{"2":{"98":1,"99":1}}],["rollupadminlogic",{"2":{"98":1,"99":1}}],["rollupeventinbox",{"2":{"98":1,"99":1}}],["rollupcreator",{"2":{"89":1,"96":1,"97":1,"98":1,"99":1}}],["rollup",{"0":{"61":1,"66":1,"67":1,"69":1,"70":1,"71":1,"73":1,"79":1,"112":1,"133":1,"135":1,"138":1,"161":1,"228":1,"264":1},"1":{"67":1,"68":2,"69":2,"70":1,"71":2,"72":2,"74":1,"75":1,"76":1,"77":1,"78":1,"80":1,"81":1,"82":1,"83":1,"84":1,"85":1,"86":1,"87":1,"88":1,"89":1,"90":1,"91":1,"92":1,"93":1,"94":1,"95":1,"96":1,"97":1,"98":1,"99":1,"113":1,"114":1,"115":1,"116":1,"117":1,"118":1,"119":1,"120":1,"121":1,"122":1,"136":1,"137":1,"139":1,"140":1,"141":1,"142":1,"143":1},"2":{"44":1,"61":4,"66":2,"69":4,"71":1,"73":2,"74":1,"79":3,"82":1,"83":3,"86":2,"88":1,"90":1,"91":8,"92":2,"101":1,"111":1,"113":1,"114":1,"122":2,"127":15,"131":1,"133":3,"134":11,"135":15,"139":4,"143":1,"146":2,"148":2,"149":1,"152":1,"156":1,"157":1,"159":2,"161":4,"163":1,"167":4,"168":2,"169":1,"170":7,"173":3,"176":3,"177":7,"178":2,"180":1,"181":7,"184":2,"185":5,"188":2,"189":4,"192":4,"195":1,"206":1,"208":1,"209":1,"213":2,"214":1,"226":1,"228":1,"229":1,"231":2,"232":1,"233":5,"258":2,"264":2,"268":2,"269":1,"270":3,"271":1,"275":1,"285":1,"290":1,"351":4,"352":1,"380":2,"381":4,"419":1,"440":2,"442":7,"449":1,"457":1,"620":2}}],["rollups",{"0":{"104":1,"163":1,"176":1,"177":1,"178":1,"179":1,"180":1,"181":1,"182":1,"183":1,"184":1,"185":1,"186":1,"187":1,"188":1,"190":1,"191":1,"211":1,"226":1,"442":1,"458":1},"1":{"105":1,"106":1,"107":1,"108":1,"109":1,"110":1,"111":1,"164":1,"165":1,"166":1,"167":1,"168":1,"169":1,"170":1,"171":1,"172":1,"173":1,"174":1,"175":1,"176":1,"177":2,"178":3,"179":3,"180":3,"181":2,"182":3,"183":3,"184":1,"185":2,"186":3,"187":3,"188":2,"189":3,"190":3,"191":3,"192":2,"193":2,"194":2,"195":1},"2":{"3":1,"43":1,"48":1,"59":1,"61":1,"82":1,"103":1,"133":1,"136":3,"162":1,"163":2,"169":1,"176":3,"177":1,"184":2,"189":1,"195":2,"206":4,"211":2,"213":1,"223":1,"227":1,"228":1,"229":1,"231":2,"232":6,"233":3,"350":1,"358":1,"380":2,"381":1,"403":1,"413":1,"422":1,"440":1,"442":3,"458":1,"465":1,"466":1,"666":1,"685":1,"698":1,"701":1,"703":1}}],["roster",{"2":{"27":1}}],["rust",{"2":{"627":1,"628":1}}],["rule",{"2":{"31":1,"34":5}}],["rules",{"2":{"23":1,"135":1,"422":1}}],["runtime",{"2":{"634":8}}],["runners",{"2":{"471":1}}],["running",{"0":{"100":1,"102":1,"103":1,"243":1,"261":1,"544":1,"696":1},"1":{"101":1,"102":1,"103":1,"545":1,"546":1,"547":1},"2":{"60":1,"74":1,"80":1,"87":1,"88":1,"89":1,"91":6,"92":1,"103":1,"107":2,"114":1,"117":2,"119":1,"126":1,"135":1,"206":1,"244":1,"249":2,"258":2,"270":1,"271":1,"277":1,"302":1,"303":1,"304":3,"307":1,"310":3,"313":1,"314":1,"334":1,"343":1,"346":1,"360":1,"367":3,"368":1,"377":1,"380":1,"384":1,"406":1,"469":1,"479":1,"485":2,"499":1,"501":1,"502":1,"503":1,"504":1,"507":2,"517":1,"530":1,"543":2,"548":1,"557":1,"559":1,"571":1,"573":1,"574":2,"577":1,"578":1,"580":1,"584":1,"585":2,"588":3,"592":1,"614":1,"620":2,"622":1,"623":1,"628":1,"630":1,"644":1,"653":1,"656":1,"661":3,"665":1,"672":1,"685":1,"688":1,"703":1,"707":1,"708":1,"709":1,"710":1,"711":1,"712":2,"718":1,"722":3}}],["rundown",{"2":{"223":1}}],["runs",{"2":{"117":2,"249":2,"386":1,"503":2,"515":1}}],["run",{"0":{"60":1,"89":1,"90":1,"199":1,"367":1,"485":1,"486":1,"514":1,"572":1,"618":1,"621":1,"663":1,"725":1,"727":1},"1":{"486":1,"573":1,"574":1,"619":1,"620":1,"726":1},"2":{"1":1,"24":2,"27":1,"57":1,"60":2,"89":1,"90":1,"91":3,"92":1,"102":4,"103":1,"116":3,"117":1,"135":1,"199":1,"233":2,"242":1,"243":3,"248":3,"249":1,"251":1,"252":1,"257":1,"261":1,"262":2,"264":1,"270":7,"293":1,"295":1,"297":1,"304":1,"305":2,"309":2,"311":1,"317":1,"338":1,"344":1,"346":2,"349":1,"355":1,"357":3,"361":1,"362":1,"363":1,"367":1,"368":2,"371":1,"374":3,"377":2,"378":2,"469":1,"470":1,"471":1,"478":3,"483":1,"484":1,"486":1,"488":2,"500":1,"503":4,"508":1,"514":3,"515":2,"517":1,"522":1,"531":1,"536":1,"537":1,"539":2,"540":1,"541":2,"548":1,"552":2,"554":1,"556":1,"560":1,"561":2,"562":1,"572":1,"574":2,"575":6,"577":9,"579":9,"580":9,"584":1,"586":2,"598":1,"602":1,"603":2,"619":1,"621":1,"622":1,"627":1,"631":1,"645":1,"646":1,"647":1,"649":2,"650":1,"652":2,"659":1,"663":1,"672":2,"679":1,"685":3,"688":1,"696":1,"706":1,"707":1,"708":1,"709":3,"721":2,"722":1,"724":1,"725":1,"727":2}}],["rate=0",{"2":{"500":6,"727":6}}],["rates",{"2":{"445":1}}],["rate",{"2":{"400":3,"445":1}}],["rather",{"2":{"205":1,"229":1,"445":1,"465":1}}],["ram",{"2":{"285":1,"303":1,"307":1,"479":1,"559":1,"588":1,"614":1,"656":1,"718":1}}],["rapid",{"2":{"265":1}}],["rapidly",{"2":{"50":1}}],["rainbow",{"2":{"266":2,"267":8,"268":2}}],["rainbowkitprovider",{"2":{"267":6}}],["rainbowkit",{"2":{"256":1,"265":1,"266":2,"267":10,"268":2}}],["raises",{"2":{"84":1}}],["raise",{"2":{"4":1,"36":1}}],["raas",{"0":{"358":1},"2":{"226":1,"358":2}}],["rawlog",{"2":{"392":4}}],["raw",{"2":{"154":1,"304":2,"400":1,"478":2,"522":2,"585":2,"592":10,"655":1,"721":2,"727":2}}],["rawmessage",{"2":{"134":2,"137":2}}],["rand",{"2":{"89":1}}],["randomly",{"2":{"416":1}}],["random",{"2":{"89":1,"181":1,"405":1,"423":1,"425":2}}],["ranges",{"0":{"376":1},"1":{"377":1,"378":1,"379":1},"2":{"207":3}}],["range",{"2":{"89":1,"107":3,"151":6,"152":2,"155":4,"157":2,"158":4,"159":8,"161":26,"172":1,"206":1,"207":2,"373":4,"377":1,"378":1,"379":2,"419":2}}],["radio",{"2":{"34":1}}],["race",{"2":{"7":1,"367":3}}],["reinitializing",{"2":{"644":1}}],["reinitialize",{"2":{"550":1}}],["reindexing",{"2":{"611":2}}],["reindex",{"2":{"611":1}}],["rejoin",{"2":{"509":1}}],["rejected",{"2":{"425":1,"638":1}}],["rejects",{"2":{"425":1}}],["reject",{"2":{"9":1,"418":1}}],["reed",{"2":{"416":2,"418":5,"421":1,"424":1}}],["redelegate",{"2":{"432":1,"436":1}}],["redeploy",{"2":{"264":1}}],["reducing",{"2":{"423":1}}],["reduce",{"2":{"231":1}}],["reduced",{"2":{"40":1,"178":1,"231":1,"425":1,"667":1,"698":1}}],["redundant",{"2":{"407":1}}],["redundancy",{"2":{"288":1}}],["revert",{"2":{"161":2}}],["reveal",{"2":{"108":1}}],["revisit",{"2":{"46":1}}],["revised",{"2":{"30":1}}],["reviewing",{"2":{"38":1}}],["reviewed",{"2":{"11":1,"21":1,"24":1}}],["review",{"0":{"85":1},"2":{"2":1,"36":1,"53":1,"73":1,"84":1,"85":2,"272":1,"435":2}}],["retain",{"0":{"596":1},"2":{"594":1,"596":5,"599":4}}],["retrial",{"2":{"425":1}}],["retries",{"2":{"360":2}}],["retried",{"2":{"159":1}}],["retrievable",{"2":{"412":1,"440":1,"680":1}}],["retrievability",{"0":{"414":1,"440":1,"441":1},"1":{"441":1,"442":1},"2":{"411":1,"440":1,"441":1}}],["retrieval",{"2":{"408":1}}],["retrieving",{"0":{"319":1},"2":{"107":1,"408":1,"412":1,"573":1,"601":1}}],["retrieves",{"2":{"213":1}}],["retrieved",{"2":{"159":1,"313":1,"351":1,"409":1,"442":1}}],["retrieve",{"2":{"107":2,"268":1,"299":1,"319":1,"335":1,"351":1,"357":2,"409":1}}],["retryable",{"2":{"91":6}}],["retryables",{"2":{"91":4}}],["return",{"2":{"120":1,"127":2,"137":10,"139":12,"151":2,"160":8,"252":1,"260":4,"268":4,"319":3,"329":1,"373":12,"374":1,"392":18,"398":2}}],["returns",{"2":{"89":1,"127":2,"135":4,"170":1,"260":4,"317":1,"351":1,"373":2}}],["returned",{"2":{"76":1,"268":2,"272":1,"319":1,"351":2,"419":1,"509":1,"638":1,"721":1}}],["rebasing",{"2":{"84":1}}],["rename",{"2":{"490":2,"626":1}}],["renaming",{"2":{"490":2}}],["render",{"2":{"267":2}}],["rentals",{"2":{"46":1}}],["renewals",{"2":{"23":1}}],["renewal",{"2":{"23":1,"26":1}}],["refresh",{"2":{"629":2}}],["refreshments",{"0":{"41":1},"2":{"36":1,"41":6,"44":1}}],["refund",{"2":{"384":1,"386":1}}],["refunded",{"2":{"384":1}}],["referred",{"2":{"422":1,"424":2}}],["referrals",{"2":{"43":2}}],["refers",{"2":{"172":2,"478":1}}],["referencing",{"2":{"136":1,"168":2,"170":1,"171":1}}],["references",{"2":{"185":1}}],["referenced",{"2":{"166":1,"170":1,"181":1,"185":1,"189":1,"192":1}}],["reference",{"2":{"0":1,"3":1,"30":1,"42":1,"148":1,"177":1,"180":1,"184":1,"188":1,"189":1,"364":1,"633":6}}],["refer",{"2":{"84":1,"237":1,"243":2,"309":1,"310":1,"318":1,"354":1,"367":1,"386":2,"419":1,"445":1,"470":1,"484":1,"485":1,"509":1,"537":1,"550":1,"562":1,"575":1,"577":2,"603":1,"620":2,"653":1,"660":2,"685":1,"724":1,"730":1}}],["reward",{"0":{"413":1},"2":{"31":1,"34":1,"43":1,"414":1}}],["rewards",{"0":{"496":1},"2":{"30":5,"31":10,"32":3,"33":1,"34":37,"288":1,"432":1,"436":1,"444":2,"448":1,"450":1,"451":1,"459":1,"460":1,"495":1,"496":3,"509":1}}],["regen",{"2":{"149":4}}],["regions",{"2":{"47":1}}],["registry",{"2":{"400":1,"601":1}}],["registration",{"2":{"34":8,"36":1,"37":1,"53":1}}],["register",{"2":{"34":2,"629":2}}],["registered",{"2":{"34":5,"36":1}}],["regular",{"2":{"42":1,"48":1,"82":1,"421":1,"506":1}}],["regulation",{"2":{"31":1,"34":6}}],["regardless",{"2":{"7":1,"167":1,"394":1,"396":1}}],["reaped",{"2":{"388":1}}],["reapplying",{"2":{"26":1}}],["reapply",{"2":{"24":1}}],["reactjs",{"2":{"398":1}}],["reactmarkdown",{"2":{"268":4}}],["reactdom",{"2":{"267":4}}],["react",{"0":{"398":1},"2":{"256":1,"265":4,"266":2,"267":2,"268":2,"270":1,"397":1,"398":4}}],["reaching",{"2":{"455":1}}],["reaches",{"2":{"405":1,"445":1}}],["reached",{"2":{"392":2}}],["reach",{"2":{"24":1,"36":1,"43":1,"44":1,"45":1,"46":3,"256":1}}],["real",{"2":{"88":1,"666":1,"712":1,"716":1}}],["readme",{"2":{"510":1}}],["readable",{"2":{"268":2}}],["reads",{"2":{"213":1,"290":2}}],["reader",{"2":{"135":1}}],["reading",{"0":{"268":1},"2":{"135":2,"213":4,"290":4,"408":1,"561":1}}],["read",{"0":{"348":1},"2":{"30":1,"71":1,"76":1,"79":2,"94":1,"107":1,"108":1,"121":1,"134":1,"135":1,"253":1,"256":1,"263":1,"264":1,"268":3,"272":1,"297":1,"301":1,"302":1,"319":1,"348":1,"351":1,"385":1,"583":1,"680":1,"685":1}}],["ready",{"2":{"27":1,"76":1,"120":1,"149":1,"225":1,"231":1,"252":2,"254":1,"270":1,"272":1,"365":1,"517":1,"601":1,"624":1,"625":1,"626":1,"727":1}}],["reasonable",{"2":{"378":1}}],["reasonably",{"2":{"8":1}}],["reason",{"2":{"25":1,"31":2,"283":1}}],["reasons",{"2":{"9":1}}],["resubmission",{"0":{"394":1,"395":1},"1":{"395":1,"396":1},"2":{"394":1}}],["resubmitted",{"2":{"395":1}}],["resubmit",{"2":{"388":2,"396":1}}],["results",{"2":{"148":1,"149":1,"326":2,"509":1,"611":4}}],["resulting",{"2":{"148":1,"418":1,"422":1}}],["result",{"2":{"24":2,"32":1,"117":2,"121":2,"154":1,"155":1,"249":2,"253":2,"317":6,"319":8,"320":2,"322":2,"323":2,"324":2,"325":2,"326":2,"327":2,"329":2,"332":2,"334":3,"349":1,"416":1,"419":2,"422":3,"583":1,"712":2}}],["resolved",{"2":{"628":1}}],["resolve",{"2":{"422":1}}],["resolving",{"2":{"285":1,"357":1}}],["resolution",{"2":{"229":1,"556":1}}],["resources",{"0":{"333":1,"508":1,"604":1,"730":1},"1":{"334":1,"335":1,"336":1,"337":1,"605":1,"606":1,"607":1,"608":1,"609":1,"610":1,"611":1},"2":{"45":1,"46":2,"49":1,"50":2,"56":1,"57":1,"64":2,"93":1,"143":1,"277":1,"730":2}}],["resource",{"2":{"43":1,"56":1,"65":1,"205":1,"417":2,"423":1,"598":1}}],["resides",{"2":{"108":1}}],["resilience",{"2":{"27":1}}],["resetting",{"0":{"554":1}}],["reset",{"0":{"605":1,"644":1},"2":{"550":1,"555":4,"605":2,"626":1,"644":2}}],["reserved",{"2":{"172":1,"424":2,"671":1}}],["reserves",{"2":{"34":1}}],["research",{"2":{"42":1,"44":1,"449":1}}],["restartsec=3",{"2":{"712":2,"714":2,"715":2,"716":2}}],["restart=on",{"2":{"712":2,"714":2,"715":2,"716":2}}],["restart",{"0":{"609":1},"2":{"502":1,"544":1,"557":3,"562":1,"609":1,"610":1,"611":1}}],["restaurant",{"2":{"40":1}}],["restaurants",{"2":{"40":2}}],["rest",{"2":{"136":1,"297":1,"367":1,"398":10,"399":1,"400":1,"552":1,"628":1,"629":2,"675":3,"690":4}}],["restrict",{"2":{"233":1}}],["restricted",{"2":{"34":5}}],["restrictions",{"2":{"31":1,"34":4,"450":1}}],["resale",{"2":{"34":2}}],["resp",{"2":{"373":4,"392":20}}],["respectively",{"2":{"351":1,"377":1,"471":1,"679":1,"688":1}}],["respective",{"2":{"28":1,"135":1,"139":1,"146":1,"471":1,"500":1,"601":2,"679":1,"685":1,"688":1}}],["respect",{"2":{"11":1,"30":1,"31":1,"32":1,"34":3}}],["respectfully",{"2":{"52":1}}],["respectful",{"2":{"8":1}}],["responses",{"0":{"611":1},"2":{"48":1,"365":1,"611":9}}],["response",{"2":{"9":1,"48":1,"155":4,"156":1,"158":4,"159":1,"243":1,"244":1,"268":4,"319":3,"322":1,"323":1,"324":1,"325":1,"373":12,"392":4,"416":1}}],["responsible",{"2":{"9":1,"11":1,"31":1,"32":1,"33":2,"84":1,"85":1,"88":1,"135":2,"213":1,"290":1,"351":1,"423":1,"442":2}}],["responsibilities",{"0":{"9":1}}],["responsibility",{"2":{"8":1,"9":1,"131":2,"396":1,"412":1}}],["requiring",{"2":{"423":1}}],["requires",{"2":{"128":1,"169":1,"191":1,"206":1,"209":1,"353":1,"378":1,"379":1,"412":1,"588":1}}],["requirement",{"2":{"41":1,"418":1,"506":1,"588":1}}],["requirements",{"0":{"303":1,"479":1,"588":1,"599":1,"614":1,"640":1,"656":1,"718":1},"2":{"24":1,"34":4,"36":1,"43":1,"60":1,"61":1,"84":1,"195":1,"283":2,"303":1,"479":1,"588":1,"599":1,"614":1,"640":1,"656":1,"718":1}}],["required",{"2":{"32":1,"34":1,"35":1,"36":1,"84":1,"186":1,"213":1,"231":1,"243":1,"264":1,"309":1,"367":1,"371":1,"372":1,"386":1,"418":1,"484":1,"537":1,"541":1,"550":1,"551":2,"552":1,"575":1,"577":1,"601":1,"603":1,"620":1,"653":1,"660":1,"722":1,"724":1}}],["require",{"2":{"31":1,"103":1,"127":4,"162":1,"167":1,"169":1,"206":1,"232":1,"260":2,"312":1,"412":1,"413":1,"418":2,"422":2,"598":1,"624":1}}],["requisites",{"0":{"258":1,"564":1},"2":{"264":1}}],["requesting",{"0":{"376":1},"1":{"377":1,"378":1,"379":1},"2":{"379":1,"431":1}}],["requests",{"2":{"313":1,"377":1,"379":1,"414":2,"419":1,"425":1,"442":1,"478":1,"606":1}}],["requestheaderrange",{"2":{"207":1,"379":1}}],["requested",{"2":{"13":1,"207":1,"379":1}}],["request",{"0":{"379":1},"2":{"2":2,"44":1,"107":1,"310":1,"314":1,"367":1,"376":1,"377":1,"379":1,"398":2,"424":1,"473":1,"485":1,"509":1,"600":1,"620":1,"662":1,"693":1,"696":1}}],["rec",{"2":{"392":4}}],["recipe",{"2":{"381":1}}],["recipient",{"2":{"329":1,"495":2}}],["recipients",{"2":{"34":1}}],["recursive",{"2":{"341":2}}],["recursivegit",{"2":{"341":2}}],["recency",{"2":{"441":2}}],["recent",{"2":{"107":1,"347":1,"680":1}}],["receiptsch",{"2":{"351":2}}],["receiptsroot",{"2":{"347":2}}],["receipt",{"2":{"32":3,"34":4,"450":1}}],["receiving",{"2":{"27":1,"31":2,"34":1,"367":1}}],["receiver",{"2":{"76":2,"91":4,"272":2}}],["receiver=",{"2":{"76":2,"272":2}}],["received",{"2":{"24":1,"33":1,"160":1,"416":1,"425":2,"655":1}}],["receives",{"2":{"23":3,"402":1,"448":1,"460":1}}],["receive",{"2":{"21":1,"23":1,"27":1,"29":1,"30":1,"31":4,"33":3,"34":3,"53":1,"54":1,"320":1,"406":1,"416":1,"451":1,"667":1,"698":1}}],["recap",{"2":{"54":1}}],["recruiting",{"2":{"48":2}}],["reconstructing",{"2":{"418":1}}],["reconstruction",{"0":{"407":1}}],["reconstruct",{"2":{"406":1,"407":1,"442":1}}],["recoverable",{"2":{"418":1}}],["recovered",{"2":{"416":1}}],["recover",{"2":{"239":6,"519":2,"536":1}}],["recompute",{"2":{"108":1}}],["recommendation",{"2":{"588":1}}],["recommendations",{"2":{"41":1}}],["recommend",{"2":{"84":1,"103":1,"256":1,"277":1,"465":1,"561":2,"685":1}}],["recommends",{"2":{"47":1}}],["recommended",{"0":{"377":1},"2":{"24":1,"35":1,"88":1,"234":1,"267":2,"296":1,"303":1,"307":1,"398":2,"425":1,"479":1,"485":1,"494":1,"559":1,"573":1,"583":1,"588":2,"614":1,"620":1,"656":1,"709":1,"718":1,"722":1}}],["recognizable",{"2":{"84":1}}],["recognize",{"2":{"53":1}}],["recognition",{"2":{"43":1}}],["recorded",{"2":{"46":4}}],["record",{"2":{"40":1}}],["recording",{"0":{"46":1},"2":{"40":2,"46":6}}],["recordings",{"2":{"5":1,"40":1,"46":3}}],["recourse",{"2":{"31":2}}],["remainder",{"2":{"264":1,"509":1,"515":1,"522":1}}],["remain",{"2":{"232":1,"440":1,"501":1,"596":1,"645":1}}],["remaining",{"2":{"155":2,"156":1,"229":1,"351":1,"450":4,"525":1,"671":1}}],["remains",{"2":{"19":1,"201":1,"667":1,"698":1}}],["removing",{"2":{"509":1}}],["removed",{"2":{"91":2,"297":1,"302":1,"509":2,"552":1}}],["removechainowner",{"2":{"91":2}}],["remove",{"2":{"9":1,"267":2,"307":1,"348":2,"363":1,"539":1,"555":1,"559":1}}],["remote",{"2":{"46":1,"215":1,"311":1,"370":1,"478":1,"488":2,"503":2,"566":3,"571":1}}],["remember",{"2":{"40":1,"44":1,"217":1,"233":1,"348":1,"610":1,"611":1,"667":1}}],["reminders",{"2":{"45":1}}],["reminder",{"2":{"36":1,"45":1,"71":1}}],["repeat",{"2":{"648":1}}],["replicated",{"2":{"422":1}}],["replacement",{"2":{"380":1}}],["replace",{"2":{"181":1,"314":1,"388":1,"515":1,"650":1}}],["replaced",{"2":{"159":2}}],["replaces",{"2":{"149":1,"351":1,"421":1,"607":1}}],["replacing",{"2":{"91":2,"243":1,"244":1,"380":1,"537":1}}],["replaying",{"2":{"442":1}}],["replay",{"2":{"94":1,"108":1}}],["repo",{"2":{"126":3,"336":1,"341":1,"478":2,"500":1,"650":1,"652":1}}],["report",{"2":{"28":1,"556":2}}],["reporting",{"2":{"24":1}}],["reporter",{"2":{"11":1}}],["reported",{"2":{"11":1}}],["repository",{"0":{"87":1,"287":1,"341":1},"1":{"288":1,"289":1},"2":{"0":1,"1":1,"2":2,"4":1,"87":3,"90":1,"91":1,"126":1,"207":1,"213":1,"235":1,"236":1,"287":1,"307":1,"349":1,"356":1,"539":1,"559":1,"592":2,"728":1}}],["represents",{"2":{"107":2,"312":1}}],["represented",{"2":{"34":1}}],["represent",{"2":{"32":1,"33":1,"34":2,"146":1}}],["representation",{"2":{"34":2,"135":1,"155":2,"156":2}}],["representations",{"0":{"34":1},"2":{"31":1,"34":2}}],["representative",{"2":{"10":1}}],["representing",{"2":{"10":2}}],["reloadsudo",{"2":{"557":1}}],["reload",{"2":{"557":2}}],["rely",{"2":{"204":1,"279":1,"442":1}}],["relying",{"2":{"32":1,"203":1,"442":1}}],["relaying",{"0":{"627":1},"1":{"628":1,"629":1,"630":1,"631":1,"632":1,"633":1,"634":1,"635":1,"636":1,"637":1,"638":1},"2":{"255":1,"628":1,"630":3}}],["relayers",{"2":{"627":1}}],["relayer",{"0":{"630":1,"636":1},"2":{"207":1,"627":3,"628":2,"636":1}}],["relayed",{"2":{"107":2,"128":1,"129":1,"133":1,"136":1,"137":1,"379":1}}],["relay",{"2":{"107":1,"627":1,"629":1}}],["relays",{"2":{"95":1,"109":1,"201":1}}],["relatively",{"2":{"283":1,"394":1}}],["relative",{"0":{"283":1},"2":{"191":1,"283":1}}],["relations",{"2":{"45":1}}],["relationships",{"2":{"38":1,"42":1}}],["relating",{"2":{"32":1}}],["related",{"2":{"31":1,"33":1,"34":1,"44":1,"48":1,"64":1,"411":1,"620":1,"660":1}}],["releasing",{"2":{"375":1}}],["releases",{"2":{"540":1,"560":1,"626":1}}],["release",{"2":{"351":1,"441":1,"625":1,"626":1,"627":1}}],["released",{"2":{"31":1,"628":1}}],["relevant",{"2":{"30":1,"34":1,"43":2,"45":2,"48":1,"392":2,"415":1,"425":1,"469":2,"655":1,"672":2,"685":2}}],["relevantly",{"2":{"2":1}}],["relies",{"2":{"601":1,"627":1}}],["relied",{"2":{"34":1}}],["reliability",{"2":{"33":1}}],["reliance",{"2":{"31":3,"34":2}}],["religion",{"2":{"7":1}}],["re",{"2":{"2":1,"40":1,"41":1,"52":1,"75":1,"76":1,"79":2,"84":2,"89":1,"120":1,"126":1,"148":1,"171":1,"176":1,"181":1,"223":1,"225":1,"232":1,"244":1,"252":2,"254":1,"256":1,"267":2,"269":1,"270":2,"272":2,"307":1,"336":1,"356":1,"365":1,"380":1,"392":4,"394":1,"399":1,"400":1,"505":1,"550":1,"554":1,"556":1,"559":1,"561":1,"578":2,"632":1}}],["snaps",{"2":{"602":12}}],["snapshots",{"2":{"442":4,"599":1,"601":3}}],["snapshot",{"2":{"411":1,"602":1}}],["snap",{"2":{"602":24}}],["snippets",{"2":{"149":1}}],["ss",{"2":{"386":1}}],["ssd",{"2":{"285":1,"303":1,"479":1,"588":3,"614":1,"656":1,"718":1}}],["sgvsbg8gd28ybgqh",{"2":{"352":2}}],["svg",{"0":{"335":1},"2":{"335":2,"400":1}}],["skip",{"2":{"196":1,"302":1,"632":1,"709":1,"722":1}}],["skipped",{"2":{"117":2,"120":2,"249":2,"252":2}}],["skillsets",{"2":{"52":1}}],["square",{"0":{"147":1},"2":{"107":1,"108":2,"131":3,"136":1,"146":2,"151":2,"161":2,"166":1,"170":2,"171":1,"172":9,"173":1,"424":3,"671":4}}],["sdk",{"2":{"94":4,"149":7,"233":1,"235":1,"278":4,"279":2,"280":1,"282":1,"283":1,"380":1,"381":1,"382":1,"385":1,"392":2,"421":1,"426":1,"444":3,"448":1,"459":1,"469":1,"478":1,"488":2,"494":2,"506":1,"509":1,"628":1,"672":1,"690":1,"691":1}}],["swap",{"2":{"290":1,"626":1}}],["swag",{"0":{"63":1},"2":{"43":1}}],["switching",{"2":{"550":1}}],["switch",{"2":{"83":1,"213":1,"270":1,"290":1,"398":2,"400":1,"503":1}}],["symbol",{"2":{"267":2,"269":1,"400":2}}],["sync|async|block",{"2":{"531":2}}],["sync",{"0":{"601":1,"602":1},"2":{"282":3,"409":1,"442":3,"530":1,"531":2,"587":1,"600":2,"601":2,"602":2,"696":1,"712":4,"716":1}}],["syncing",{"0":{"282":1,"600":1},"1":{"601":1,"602":1},"2":{"282":1,"408":1,"440":1,"487":1,"588":1,"600":1,"714":1,"715":2,"716":1,"726":1}}],["synced",{"2":{"80":1,"357":2,"360":1}}],["syndtr",{"2":{"149":4}}],["systemctl",{"2":{"557":4,"712":6,"714":4,"715":4,"716":6}}],["systemd",{"0":{"487":1,"499":1,"622":1,"664":1,"710":1,"712":1,"726":1},"1":{"711":1,"712":1,"713":1,"714":1,"715":1,"716":1},"2":{"487":1,"499":7,"557":2,"603":1,"622":1,"664":1,"710":1,"712":6,"714":5,"715":5,"716":4,"726":1}}],["systems",{"2":{"279":1,"288":1,"503":2,"540":1,"560":1}}],["system",{"2":{"23":1,"36":1,"203":1,"288":1,"295":1,"307":1,"385":1,"386":2,"422":2,"501":2,"504":1,"505":1,"557":5,"559":1,"575":1,"631":1,"712":4,"714":4,"715":4,"716":4}}],["src",{"2":{"77":2,"116":1,"117":2,"120":2,"159":2,"248":1,"249":2,"251":2,"252":2,"259":1,"260":3,"262":2,"267":1,"270":2,"273":2,"398":8}}],["smartstake",{"2":{"683":1}}],["smart",{"0":{"112":1,"116":1,"117":1,"118":1,"121":1,"227":1,"245":1,"248":1,"249":1,"250":1,"253":1},"1":{"113":1,"114":1,"115":1,"116":1,"117":1,"118":1,"119":2,"120":2,"121":1,"122":1,"246":1,"247":1,"248":1,"249":1,"250":1,"251":2,"252":2,"253":1,"254":1},"2":{"73":1,"74":1,"79":1,"85":2,"88":1,"113":2,"116":1,"122":1,"131":7,"133":1,"148":1,"150":1,"153":5,"159":2,"162":1,"167":1,"172":1,"173":2,"206":2,"207":1,"208":1,"209":1,"210":1,"217":1,"222":1,"227":1,"229":1,"231":1,"233":1,"245":1,"248":1,"252":1,"254":1,"256":1,"258":1,"259":1,"260":2,"262":1,"264":1,"268":2,"272":2,"286":1,"376":1,"411":1,"458":1}}],["smartphone",{"2":{"40":1}}],["small",{"2":{"46":1,"107":1,"134":1,"195":1,"232":1,"405":1,"417":1,"423":1}}],["smooth",{"2":{"43":1,"588":1}}],["sandbox",{"2":{"681":1}}],["sanctions",{"2":{"24":1,"31":1}}],["sausage",{"2":{"519":2}}],["sashabaranov",{"2":{"372":2}}],["save",{"0":{"515":1},"2":{"237":1,"367":1,"494":1,"503":1,"536":1,"554":1,"599":1,"610":1,"611":1,"646":1,"680":1}}],["saved",{"2":{"146":1,"148":1,"167":1,"182":1,"185":1}}],["saving",{"0":{"599":1},"2":{"43":1,"169":1,"494":1}}],["said",{"2":{"94":1}}],["sampling",{"0":{"405":1,"406":1,"416":1,"665":1},"1":{"417":1,"418":1},"2":{"59":1,"201":1,"232":1,"283":1,"328":2,"404":1,"405":5,"406":1,"407":1,"415":1,"416":3,"418":1,"423":2,"441":3,"478":1,"654":1,"667":1,"696":1,"707":1,"716":1}}],["samples",{"2":{"417":1}}],["sampled",{"2":{"406":1,"425":1}}],["sampler",{"0":{"328":1}}],["sample",{"0":{"59":1,"60":1,"61":1,"62":1},"2":{"59":2,"60":2,"61":2,"62":2,"370":1,"423":1,"441":1}}],["same",{"0":{"388":1},"2":{"34":1,"75":1,"110":1,"134":1,"148":1,"168":1,"173":1,"182":1,"185":1,"189":2,"203":1,"272":1,"280":1,"295":1,"346":1,"352":1,"368":1,"392":2,"422":2,"503":4,"509":1,"645":1,"647":1,"727":1}}],["safely",{"2":{"109":1,"345":1,"362":1,"423":1,"553":1}}],["safe",{"2":{"52":1,"367":1,"508":1,"519":2}}],["satisfied",{"2":{"34":1}}],["satisfy",{"2":{"34":1}}],["sales",{"2":{"34":1,"44":1,"472":1,"693":1}}],["sale",{"2":{"34":3}}],["shnode",{"2":{"715":1}}],["shpeer=",{"2":{"652":1}}],["ship",{"2":{"652":1}}],["shirts",{"2":{"43":1}}],["shtia",{"2":{"647":1}}],["shcurl",{"2":{"712":1}}],["shcat",{"2":{"712":1,"714":1,"715":1,"716":1}}],["shcd",{"2":{"514":1,"592":1}}],["shceles",{"2":{"500":1}}],["shcelestia",{"2":{"367":2,"484":4,"485":3,"486":3,"489":1,"491":3,"492":2,"493":5,"494":5,"495":2,"496":1,"497":2,"498":1,"536":3,"537":1,"539":1,"544":1,"545":1,"546":1,"550":1,"603":1,"605":1,"619":3,"620":1,"621":3,"644":1,"650":1,"653":1,"659":3,"660":3,"661":3,"663":3}}],["shsed",{"2":{"607":1}}],["shstaking",{"2":{"500":1,"649":1}}],["shsudo",{"2":{"499":1,"712":4,"714":2,"715":2,"716":4}}],["shkey",{"2":{"500":1,"646":1}}],["shkeyname=",{"2":{"367":1}}],["shvalidator",{"2":{"500":1,"645":1}}],["shgo",{"2":{"374":1}}],["shgit",{"2":{"126":1,"236":1}}],["shmkdir",{"2":{"371":1}}],["shexternal",{"2":{"608":1}}],["shexport",{"2":{"368":1}}],["sherror",{"2":{"554":1}}],["shevm",{"2":{"500":1}}],["shecho",{"2":{"368":1}}],["sha3uncles",{"2":{"347":2}}],["sharing",{"2":{"46":1,"65":1,"442":1}}],["shareversionzero",{"2":{"392":2}}],["shareindexinrowmajororder",{"2":{"173":2}}],["shareindexinrow",{"2":{"173":4}}],["shareproof",{"2":{"161":2,"173":6}}],["shareproofs",{"0":{"155":1},"2":{"153":2,"155":3,"161":4,"173":2}}],["sharesize",{"2":{"386":1}}],["sharesproof",{"2":{"153":3,"155":1,"156":1,"157":1,"158":1,"161":18}}],["shares",{"2":{"127":2,"131":5,"134":3,"136":1,"146":4,"148":3,"152":3,"153":14,"154":1,"155":2,"156":1,"157":1,"158":1,"161":4,"167":1,"170":1,"173":4,"178":1,"203":1,"319":2,"373":9,"385":2,"386":2,"387":1,"416":5,"418":4,"419":3,"423":1,"424":4,"425":3,"478":1,"613":2,"671":4}}],["shared",{"2":{"65":1,"256":2,"288":1,"649":1}}],["share",{"0":{"165":1,"166":1,"167":1,"168":1,"169":1,"181":1,"182":1,"183":1,"188":1,"190":1,"191":1},"1":{"166":1,"167":1,"168":1,"169":1,"182":1,"183":1,"189":1,"190":1,"191":1},"2":{"27":1,"31":1,"36":1,"38":1,"40":1,"43":1,"45":4,"46":3,"48":1,"134":6,"146":2,"148":5,"152":4,"161":2,"165":1,"166":11,"167":2,"168":1,"170":2,"171":1,"172":2,"173":9,"174":1,"178":2,"181":7,"182":1,"184":1,"186":1,"188":1,"189":5,"192":7,"268":2,"271":1,"317":1,"319":8,"327":2,"332":3,"334":2,"373":6,"374":1,"385":1,"386":2,"416":1,"424":1,"425":1,"442":2,"451":1,"500":1,"554":4,"648":1,"650":1,"652":3,"671":3}}],["shutdown",{"2":{"623":1}}],["shut",{"2":{"345":1,"352":1}}],["sh",{"2":{"236":2,"240":1,"243":2,"244":2,"264":3,"304":2,"337":1,"367":1,"485":1,"490":1,"507":4,"514":1,"515":2,"540":2,"556":1,"560":2,"585":2,"620":1,"661":2}}],["shforge",{"2":{"126":1}}],["shove",{"2":{"519":2}}],["shoutout",{"2":{"44":1}}],["shouldn",{"2":{"412":1}}],["should",{"2":{"27":1,"40":1,"44":1,"56":1,"84":1,"85":1,"116":1,"117":1,"184":1,"185":1,"201":1,"237":1,"248":1,"249":1,"270":1,"283":2,"317":1,"353":2,"367":1,"384":1,"385":1,"395":1,"442":1,"458":1,"503":1,"540":1,"553":1,"557":1,"560":1,"596":1,"598":1,"601":3,"626":1,"631":1,"634":2,"639":1,"645":1,"649":1,"650":2,"714":1,"715":1,"721":2,"727":2}}],["shown",{"2":{"208":1,"727":1}}],["shows",{"2":{"148":1,"180":1,"332":1,"392":2,"419":1,"421":1}}],["showcase",{"2":{"44":1}}],["show",{"2":{"43":1,"44":1,"172":1,"256":1,"305":1,"307":1,"334":1,"338":1,"352":1,"373":2,"374":1,"398":1,"469":1,"500":2,"525":2,"553":1,"559":1,"586":1,"652":2,"659":2,"672":1,"685":1,"706":1,"721":2,"727":3}}],["shortly",{"2":{"69":1}}],["short",{"2":{"34":1,"43":1,"579":1,"625":1}}],["scan",{"2":{"627":1}}],["scattered",{"2":{"365":1}}],["scavengermkdir",{"2":{"371":1}}],["scavenger",{"0":{"365":1,"370":1},"1":{"366":1,"367":1,"368":1,"369":1,"370":1,"371":2,"372":2,"373":2,"374":2,"375":1},"2":{"298":1,"365":1,"367":2,"371":7}}],["scale",{"2":{"232":2,"417":1}}],["scales",{"2":{"163":1,"201":1,"423":2}}],["scaling",{"0":{"232":1},"2":{"232":2,"402":1,"403":1,"415":1}}],["scalable",{"2":{"205":1,"213":1,"256":1,"290":1,"412":1,"415":1,"423":1}}],["scalability",{"0":{"205":1,"417":1},"2":{"109":1,"205":2}}],["scraped",{"2":{"503":2}}],["scrape",{"2":{"503":16}}],["scratches",{"2":{"233":1}}],["scratch",{"2":{"231":1}}],["scroll",{"2":{"400":1}}],["scrolling",{"2":{"399":1}}],["scripting",{"2":{"262":1}}],["scripts",{"2":{"119":1,"507":2,"514":2,"515":2}}],["script",{"0":{"87":1,"262":1},"2":{"68":1,"75":4,"87":5,"90":1,"91":7,"92":1,"262":7,"263":4,"264":6,"272":4,"367":1,"374":1,"377":2,"507":2,"514":1,"540":1,"560":1}}],["screening",{"2":{"31":1}}],["screen",{"2":{"24":1,"40":1,"431":1,"435":1}}],["scheme",{"0":{"148":1},"2":{"166":1,"416":1,"418":1,"421":1,"424":1}}],["schemes",{"2":{"58":1,"278":1}}],["schedules",{"2":{"450":1}}],["schedule",{"2":{"36":1,"37":1,"41":1,"43":1,"48":1,"450":2,"455":1}}],["schools",{"2":{"44":1}}],["scene",{"2":{"42":1}}],["scope",{"0":{"10":1},"2":{"62":1}}],["slightly",{"2":{"671":1}}],["slides",{"2":{"43":1}}],["slower",{"2":{"703":1}}],["slowest",{"2":{"600":1}}],["slots",{"2":{"21":1,"23":1}}],["sl",{"2":{"540":2,"560":2,"592":10}}],["slack",{"2":{"505":1}}],["slashboard",{"2":{"683":1}}],["slashing",{"0":{"509":1},"2":{"278":1,"498":2,"509":3}}],["slashed",{"2":{"24":1,"25":1,"27":1,"509":3}}],["soundness",{"2":{"425":1}}],["sourcing",{"2":{"231":1}}],["source",{"0":{"539":1,"559":1},"2":{"89":1,"148":1,"288":1,"305":4,"351":2,"495":2,"503":4,"538":1,"548":1,"586":4,"628":1,"629":4,"691":1}}],["sources",{"2":{"33":1,"503":1}}],["soon",{"2":{"258":1}}],["software",{"0":{"468":1,"669":1,"687":1,"699":1,"702":1,"704":1},"2":{"60":1,"61":1,"110":1,"201":1,"288":1,"304":1,"340":1,"466":1,"584":1,"624":2,"685":1,"708":1,"709":1}}],["sovereign",{"0":{"61":1,"264":1},"2":{"59":1,"61":5,"271":1,"380":4,"381":1,"466":1,"620":2,"685":1,"701":1,"703":1}}],["soda",{"2":{"41":1}}],["somewhere",{"2":{"646":1}}],["sometimes",{"2":{"478":1,"613":1}}],["something",{"2":{"102":1,"108":1,"326":2,"503":1,"542":1,"660":1}}],["some",{"0":{"173":1,"406":1,"414":1},"2":{"40":1,"46":1,"82":1,"90":1,"134":3,"135":1,"137":1,"153":4,"159":2,"228":1,"232":1,"233":1,"278":1,"281":1,"317":1,"365":1,"373":2,"374":1,"392":2,"413":1,"414":1,"424":1,"440":1,"442":1,"450":1,"494":1,"495":1,"503":1,"507":4,"530":1,"532":1,"548":1,"565":1,"625":1,"634":8}}],["someone",{"2":{"37":1,"45":1}}],["solomon",{"2":{"416":2,"418":5,"421":1,"424":1}}],["solution",{"2":{"139":1,"163":1,"195":1,"201":1,"205":1,"415":1,"418":1,"422":1,"423":1}}],["solutions",{"2":{"109":1,"388":1,"403":1,"415":1}}],["solc",{"2":{"116":2,"248":2}}],["solid",{"2":{"268":2}}],["solidityuint256",{"2":{"173":1}}],["soliditypragma",{"2":{"161":1}}],["solidity",{"2":{"114":1,"116":4,"117":3,"126":1,"127":4,"153":1,"155":6,"156":1,"158":4,"159":2,"161":2,"162":1,"166":1,"210":1,"246":1,"248":4,"249":3,"260":6,"262":4}}],["soliciting",{"2":{"38":1}}],["solicits",{"2":{"34":1}}],["solicit",{"2":{"34":1}}],["solicited",{"2":{"34":1}}],["solicitation",{"2":{"34":3}}],["sol",{"2":{"75":2,"77":3,"109":1,"116":1,"117":7,"120":2,"127":7,"129":1,"131":1,"154":1,"155":1,"156":1,"158":1,"159":1,"161":4,"207":1,"208":1,"248":1,"249":7,"251":2,"252":2,"259":1,"260":6,"262":5,"263":2,"264":2,"265":2,"272":2,"273":3}}],["sold",{"2":{"34":1}}],["solely",{"2":{"31":1,"442":1}}],["sole",{"2":{"25":1,"30":1,"31":1,"34":1}}],["so",{"2":{"23":1,"24":2,"29":1,"32":1,"34":1,"41":2,"45":1,"71":1,"89":1,"126":1,"134":2,"148":2,"155":3,"171":1,"172":1,"185":1,"225":1,"237":1,"262":1,"264":1,"265":1,"267":2,"270":1,"305":1,"309":1,"313":1,"358":1,"365":1,"367":1,"399":1,"404":1,"406":1,"407":2,"419":1,"425":1,"440":1,"466":1,"484":1,"486":1,"522":1,"527":1,"554":1,"583":1,"586":1,"599":1,"605":1,"621":1,"626":2,"639":1,"644":1,"652":2,"663":1,"679":1,"680":1}}],["sorted",{"2":{"396":1}}],["sort",{"2":{"15":1,"16":1,"34":1}}],["social",{"2":{"10":1,"14":1,"36":1,"38":1,"45":4,"46":1,"275":1,"494":1}}],["socio",{"2":{"7":1}}],["spidey",{"2":{"674":1,"675":1,"676":1,"689":1,"690":1,"691":1}}],["spinning",{"2":{"547":1}}],["spins",{"2":{"380":1}}],["spin",{"0":{"271":1,"643":1},"1":{"644":1,"645":1,"646":1,"647":1,"648":1,"649":1,"650":1,"651":1,"652":1,"653":1},"2":{"643":1}}],["splits",{"2":{"424":1}}],["split",{"2":{"146":1,"416":1,"418":1,"424":1,"438":1,"449":1}}],["sparse",{"2":{"671":2}}],["sparsesharesneeded",{"2":{"386":1}}],["spans",{"0":{"170":1,"171":1,"172":1,"173":1,"174":1,"175":1,"177":1,"178":1,"179":1,"180":1,"185":1,"186":1,"187":1},"1":{"171":1,"172":1,"173":1,"174":1,"175":1,"178":1,"179":1,"180":1,"186":1,"187":1},"2":{"152":1,"170":3,"171":1,"172":3,"173":7,"174":1,"177":4,"180":1,"181":2,"184":1,"185":5,"195":3}}],["span",{"0":{"353":1},"2":{"134":12,"135":4,"136":2,"137":8,"139":3,"153":4,"155":1,"172":1,"353":2}}],["space",{"2":{"52":1,"504":1,"610":1,"611":2,"674":1,"675":1,"676":1,"680":1}}],["spaces",{"2":{"10":2,"14":1,"40":5,"44":1}}],["spdx",{"2":{"116":2,"117":2,"127":2,"248":2,"249":2,"260":4,"262":2}}],["sponsoring",{"2":{"44":1}}],["sponsorship",{"2":{"44":5}}],["sponsors",{"0":{"44":1},"2":{"44":9,"45":1,"52":1}}],["spread",{"2":{"42":1,"45":1}}],["speed",{"2":{"395":1}}],["spend",{"2":{"386":1,"448":1,"494":1,"495":7}}],["spec",{"2":{"681":1}}],["specialization",{"2":{"422":1}}],["specialized",{"2":{"422":3}}],["specialize",{"2":{"229":1}}],["special",{"2":{"351":1}}],["specif",{"2":{"153":2,"159":2}}],["specifying",{"2":{"384":1,"503":1}}],["specify",{"2":{"84":1,"237":1,"309":2,"320":1,"367":2,"388":1,"484":2,"579":2}}],["specifies",{"2":{"84":1,"237":2}}],["specified",{"2":{"14":1,"15":1,"85":1,"237":1,"320":1,"334":1,"377":1,"384":2,"386":1,"441":1,"463":1,"488":2,"548":1}}],["specifications",{"2":{"134":1,"509":1,"625":1,"670":1}}],["specifically",{"2":{"65":1,"88":1,"402":1,"411":1,"601":1}}],["specific",{"2":{"34":1,"41":1,"43":1,"48":1,"84":1,"107":1,"129":2,"134":2,"135":4,"137":1,"203":1,"205":1,"207":1,"256":1,"307":1,"312":2,"331":1,"360":1,"373":1,"398":1,"504":1,"541":1,"552":1,"625":1,"648":1}}],["specs",{"2":{"146":2,"148":1,"155":2,"156":2,"159":4,"319":1,"387":1,"494":2}}],["speak",{"2":{"43":1}}],["speaking",{"2":{"42":1,"48":1}}],["speaker",{"0":{"65":1},"2":{"36":1,"40":1,"42":1,"43":3,"56":1,"65":5}}],["speakers",{"0":{"43":1},"2":{"36":3,"37":1,"38":1,"40":1,"42":5,"43":13,"44":2,"45":3,"48":2,"52":1,"62":1,"65":2}}],["sed",{"2":{"592":10,"608":2}}],["severe",{"2":{"509":1}}],["severity",{"2":{"403":1}}],["several",{"2":{"107":1,"309":1,"376":1,"450":1,"625":1,"671":1,"694":1,"696":1}}],["semantic",{"2":{"307":1,"559":1}}],["semi",{"2":{"203":1}}],["sequential",{"2":{"233":1,"629":4}}],["sequence",{"0":{"170":1,"171":1,"172":1,"173":1,"174":1,"175":1,"177":1,"178":1,"179":1,"180":1,"185":1,"186":1,"187":1},"1":{"171":1,"172":1,"173":1,"174":1,"175":1,"178":1,"179":1,"180":1,"186":1,"187":1},"2":{"170":3,"171":1,"172":3,"173":7,"174":1,"177":4,"180":1,"181":2,"184":1,"185":5,"195":3,"255":1,"352":1,"385":1,"388":1,"392":2,"422":2,"523":2,"524":2,"630":1}}],["sequencers",{"2":{"160":1,"181":1,"231":1}}],["sequencersignature",{"2":{"134":2,"137":2,"139":2}}],["sequenceraddress",{"2":{"134":2,"139":2}}],["sequencer",{"0":{"135":1},"1":{"136":1,"137":1},"2":{"108":1,"110":2,"134":3,"135":7,"136":2,"137":8,"139":8,"255":1,"342":2,"343":1,"359":2,"380":1}}],["sequencerinbox",{"2":{"89":2,"98":2,"99":2,"109":1}}],["separation",{"2":{"422":1}}],["separating",{"2":{"232":1}}],["separately",{"2":{"181":1}}],["separated",{"2":{"146":1,"592":2,"651":2}}],["separate",{"2":{"31":2,"155":1,"295":1,"416":1,"425":1,"478":2,"547":1}}],["sepoliafaucet",{"2":{"82":1}}],["sepolia",{"0":{"82":1,"96":1,"97":1,"98":1,"99":1},"2":{"69":2,"71":2,"79":1,"80":1,"82":3,"83":1,"85":1,"88":4,"91":2,"95":2,"96":1,"97":1,"210":3,"214":1,"216":1,"217":1,"218":1,"219":1,"220":1,"252":2,"272":2}}],["session",{"2":{"59":1,"60":1,"61":1}}],["sessions",{"2":{"44":1,"65":1}}],["seamless",{"2":{"667":1}}],["seamlessly",{"2":{"255":1}}],["search",{"0":{"430":1}}],["sealfields",{"2":{"347":2}}],["seasoned",{"2":{"56":1}}],["seating",{"2":{"37":1,"40":1}}],["sentry",{"2":{"592":1,"629":6}}],["sent",{"2":{"91":2,"148":1,"229":1,"351":1,"385":1,"537":1}}],["sensitive",{"2":{"86":1,"293":1}}],["sense",{"2":{"42":1,"46":1}}],["sendtransaction",{"2":{"351":2}}],["sender",{"2":{"260":4,"351":1,"424":1,"438":1,"634":16}}],["sending",{"2":{"215":1,"255":1,"263":1,"264":1,"329":1,"351":1,"424":1,"537":1}}],["sends",{"2":{"110":1,"351":1}}],["send",{"2":{"36":2,"38":1,"45":3,"47":1,"76":5,"119":2,"121":3,"253":3,"263":3,"264":3,"272":5,"346":1,"352":1,"493":6,"495":1,"507":8,"509":1,"529":1,"613":1,"638":1}}],["self",{"2":{"488":2,"500":2,"727":4}}],["selected",{"2":{"442":1}}],["selecting",{"2":{"40":1,"88":1,"105":1,"512":1}}],["selection",{"0":{"550":1},"1":{"551":1},"2":{"27":1,"65":1}}],["select",{"0":{"427":1,"430":1,"435":1},"2":{"36":1,"68":2,"83":1,"160":2,"237":1,"265":4,"269":1,"367":1,"429":1,"430":1,"431":2,"434":2,"435":1,"522":1,"539":1,"550":1,"599":1,"610":1,"685":3,"729":1}}],["sell",{"2":{"34":5}}],["selling",{"2":{"34":3}}],["serving",{"2":{"408":1,"414":1,"442":1}}],["servicecat",{"2":{"712":1,"714":1,"715":1,"716":1}}],["servicenano",{"2":{"557":1}}],["service",{"0":{"226":1,"276":1,"278":1},"1":{"277":1,"278":1,"279":2,"280":2,"281":2,"282":2,"283":2},"2":{"30":5,"276":1,"295":1,"343":1,"358":1,"499":7,"557":9,"607":3,"667":1,"710":1,"712":8,"714":7,"715":7,"716":7}}],["services",{"0":{"570":1},"1":{"571":1},"2":{"24":1,"30":4,"31":9,"33":3,"40":1,"244":2,"674":2,"675":2,"676":2,"683":1,"689":2,"690":2,"691":2}}],["served",{"2":{"502":2}}],["serves",{"2":{"217":1,"425":1,"478":1,"638":1,"666":1}}],["servergrafana",{"2":{"503":1}}],["server",{"2":{"53":1,"251":1,"268":2,"360":1,"367":1,"392":2,"473":1,"503":6,"541":2,"548":1,"552":2,"630":1,"693":1}}],["servers",{"2":{"27":1,"601":2}}],["serve",{"2":{"50":1,"412":1,"414":1,"606":1}}],["serialized",{"2":{"107":1}}],["serious",{"2":{"15":1}}],["series",{"2":{"14":1,"449":1,"450":1,"503":2}}],["secretvite",{"2":{"266":1}}],["secret=your",{"2":{"266":2}}],["secret",{"2":{"266":2,"268":2}}],["secp256k1",{"2":{"244":2,"278":1,"519":2,"520":4,"524":2}}],["sections",{"2":{"195":1,"209":1,"469":1,"565":1,"672":1,"685":1,"707":1,"709":1}}],["section",{"2":{"30":1,"33":1,"34":1,"42":1,"64":1,"81":1,"84":2,"88":1,"107":1,"117":1,"131":1,"136":1,"155":1,"159":1,"161":4,"169":1,"172":1,"181":2,"184":1,"185":1,"189":1,"192":2,"223":1,"237":1,"243":3,"249":1,"302":1,"309":1,"310":1,"311":1,"316":1,"318":1,"360":1,"367":1,"421":1,"471":1,"484":1,"485":1,"515":1,"526":1,"537":1,"539":1,"542":1,"544":1,"550":1,"553":1,"557":1,"559":1,"562":1,"566":1,"573":1,"574":1,"575":1,"577":1,"579":1,"603":1,"610":1,"611":1,"620":2,"627":1,"632":1,"653":1,"660":2,"706":1,"709":2,"722":1,"724":1}}],["securing",{"2":{"42":1,"43":1,"44":2}}],["securities",{"2":{"31":1,"34":13}}],["security",{"0":{"203":1,"406":1,"407":1},"2":{"11":1,"27":2,"31":1,"34":1,"201":1,"203":1,"205":1,"231":2,"402":1,"409":1,"583":1}}],["securely",{"2":{"86":1,"163":1,"201":1,"255":1,"268":2,"423":1,"625":1}}],["secure",{"2":{"25":1,"36":1,"205":1,"412":1,"426":1,"451":1,"459":2,"547":1,"600":1}}],["seconds",{"2":{"91":2,"243":4,"244":4,"283":1,"503":4}}],["second",{"2":{"23":4,"155":1,"181":1,"207":1,"260":6,"388":1,"394":1,"395":1,"396":1,"406":1,"633":1}}],["setgaslimitandfee",{"2":{"392":2}}],["setcontent",{"2":{"268":4}}],["setposts",{"2":{"268":4}}],["setviewstate",{"2":{"268":6}}],["sets",{"2":{"117":1,"249":1,"314":1,"398":1}}],["setnumber",{"2":{"116":3,"117":7,"121":2,"248":3,"249":7,"253":2}}],["settitle",{"2":{"268":4}}],["settingpruning",{"2":{"599":1}}],["settingmin",{"2":{"596":1}}],["settings",{"2":{"68":1,"503":1,"594":1,"610":1,"611":1}}],["setting",{"0":{"91":1,"304":1,"320":1,"477":1,"480":1,"513":1,"518":1,"587":1,"589":1,"612":1,"615":1,"654":1,"657":1,"710":1,"717":1,"719":1},"1":{"478":1,"479":1,"480":1,"481":2,"482":1,"483":1,"484":1,"485":1,"486":1,"487":1,"514":1,"515":1,"516":1,"517":1,"519":1,"520":1,"521":1,"522":1,"523":1,"524":1,"525":1,"588":1,"589":1,"590":2,"591":2,"592":2,"593":1,"594":1,"595":1,"596":1,"597":1,"598":1,"599":1,"600":1,"601":1,"602":1,"603":1,"604":1,"605":1,"606":1,"607":1,"608":1,"609":1,"610":1,"611":1,"613":1,"614":1,"615":1,"616":2,"617":1,"618":1,"619":1,"620":1,"621":1,"622":1,"623":1,"655":1,"656":1,"657":1,"658":2,"659":1,"660":1,"661":1,"662":1,"663":1,"664":1,"665":1,"711":1,"712":1,"713":1,"714":1,"715":1,"716":1,"718":1,"719":1,"720":2,"721":2,"722":1,"723":1,"724":1,"725":1,"726":1,"727":1,"728":1,"729":1,"730":1},"2":{"8":1,"47":2,"60":1,"61":1,"91":8,"214":1,"234":1,"304":1,"308":1,"315":1,"329":1,"353":1,"366":1,"392":4,"477":1,"487":1,"539":1,"557":1,"559":1,"571":1,"584":1,"592":1,"596":2,"599":1,"612":1,"616":1,"626":1,"654":1,"657":1,"664":1,"711":1,"717":1,"719":1,"726":1}}],["settling",{"2":{"206":1}}],["settlement",{"2":{"146":1,"163":1,"169":1,"170":1,"177":1,"178":1,"181":2,"183":1,"184":2,"185":3,"188":1,"189":1,"192":1,"206":1,"229":1,"285":1,"415":2,"421":1,"422":4}}],["settles",{"2":{"105":1}}],["settle",{"2":{"79":1,"85":1,"109":1,"206":1}}],["settled",{"2":{"69":1,"72":1}}],["setupprivate",{"2":{"91":1}}],["setup",{"0":{"75":1,"81":1,"87":1,"115":1,"247":1,"259":1,"339":1,"340":1,"377":1,"481":1,"502":1,"575":1,"578":1,"616":1,"641":1},"1":{"76":1,"82":1,"83":1,"84":1,"85":1,"86":1,"87":1,"88":1,"89":1,"90":1,"91":1,"92":1,"260":1,"261":1,"262":1,"263":1,"340":1,"341":1,"576":1,"577":1,"578":1,"579":2,"580":2,"581":1,"582":1,"583":1},"2":{"27":2,"36":1,"68":1,"84":1,"87":5,"90":1,"91":2,"117":3,"213":1,"249":3,"260":2,"262":2,"264":2,"311":1,"354":1,"357":2,"360":1,"367":1,"368":1,"370":1,"372":2,"373":3,"392":2,"503":2,"505":1,"518":1,"540":1,"543":1,"560":1,"601":1,"627":1,"641":1,"722":1}}],["setupsigner",{"2":{"392":2}}],["setups",{"2":{"24":1}}],["set",{"0":{"521":1,"530":1,"590":1,"592":1},"2":{"19":2,"21":1,"23":1,"24":1,"27":1,"30":1,"36":1,"37":1,"42":1,"47":2,"71":1,"75":2,"76":2,"88":1,"89":4,"91":10,"115":1,"116":1,"120":2,"131":1,"134":4,"137":1,"166":1,"176":1,"203":1,"231":2,"233":1,"247":1,"248":1,"251":1,"252":3,"260":2,"263":1,"264":1,"268":1,"272":4,"293":1,"295":1,"304":1,"305":1,"312":4,"313":2,"314":5,"315":4,"316":4,"317":1,"319":1,"320":3,"326":2,"329":1,"334":1,"342":1,"347":1,"348":1,"353":1,"359":1,"373":2,"386":4,"411":1,"416":1,"422":1,"444":2,"450":1,"487":1,"491":3,"503":1,"504":1,"509":3,"515":1,"521":1,"526":2,"527":2,"529":2,"530":1,"531":1,"532":1,"547":1,"557":1,"566":1,"569":1,"577":3,"578":1,"579":1,"586":1,"587":1,"592":5,"597":1,"601":1,"610":1,"611":1,"631":1,"646":1,"657":1,"696":1,"726":1}}],["seeing",{"2":{"634":2,"714":1,"715":1}}],["seeds",{"2":{"592":13}}],["seed",{"2":{"449":1,"450":1,"592":2,"630":5,"728":1}}],["sees",{"2":{"403":1}}],["seen",{"2":{"260":2}}],["seeking",{"2":{"307":1,"559":1}}],["seek",{"2":{"40":1,"43":1}}],["see",{"2":{"17":1,"33":1,"68":1,"69":4,"71":3,"72":1,"85":1,"89":1,"91":1,"92":1,"108":1,"109":1,"136":2,"143":1,"155":2,"156":2,"208":1,"221":1,"251":1,"270":1,"319":1,"352":1,"366":1,"373":1,"392":4,"399":1,"400":1,"402":1,"421":1,"432":1,"436":1,"438":1,"447":1,"462":1,"478":3,"492":1,"493":1,"502":1,"503":1,"510":1,"515":1,"519":1,"531":2,"537":1,"539":1,"540":1,"560":1,"601":1,"610":1,"628":2,"631":1,"679":1,"712":1,"714":1,"715":2,"716":1,"721":2,"727":1}}],["sexualized",{"2":{"8":1}}],["sexual",{"2":{"7":1,"8":1}}],["sex",{"2":{"7":1}}],["sum",{"2":{"371":3}}],["summarize",{"2":{"146":1}}],["summarizes",{"2":{"135":4}}],["summary",{"2":{"59":1,"60":1,"61":1,"205":1}}],["sudo",{"2":{"304":2,"305":16,"499":4,"585":2,"586":16,"712":2,"714":4,"715":4,"716":2}}],["sudden",{"2":{"19":1}}],["suite",{"2":{"288":1}}],["suitable",{"2":{"36":1,"610":2}}],["superior",{"2":{"293":1}}],["superchain",{"2":{"288":1}}],["supply",{"0":{"443":1},"1":{"444":1,"445":1,"446":1,"447":1,"448":1,"449":1,"450":1},"2":{"360":1,"445":1,"449":1,"450":6,"455":1,"478":1}}],["supplemented",{"2":{"30":1}}],["supplemental",{"0":{"30":1,"33":1},"1":{"31":1,"32":1,"33":1,"34":1},"2":{"30":2}}],["supporters",{"2":{"449":2}}],["supported",{"2":{"166":1,"255":1,"282":1,"681":1}}],["supports",{"2":{"279":1,"380":1,"444":1}}],["supportive",{"2":{"52":1}}],["supporting",{"2":{"44":1,"47":1,"62":1}}],["support",{"2":{"36":1,"39":1,"41":1,"44":4,"46":1,"49":1,"50":1,"63":2,"79":2,"88":1,"108":1,"212":1,"232":1,"233":2,"288":1,"380":1,"442":2,"444":1,"501":1,"506":1,"583":1,"627":1,"628":1,"690":1,"722":1}}],["surface",{"2":{"233":1}}],["sure",{"2":{"89":2,"125":1,"126":1,"148":2,"149":2,"162":1,"166":1,"173":1,"181":2,"185":2,"189":1,"192":1,"237":1,"261":1,"271":1,"314":1,"360":1,"367":2,"369":1,"501":1,"502":2,"526":1,"539":1,"541":3,"552":3,"564":1,"623":1,"712":1}}],["surveys",{"2":{"37":1}}],["suggest",{"2":{"398":2}}],["suggested",{"0":{"442":1},"2":{"388":1,"442":1}}],["suggests",{"2":{"47":1}}],["suggestions",{"0":{"4":1},"2":{"4":1,"40":1}}],["sustainable",{"2":{"44":1}}],["sustained",{"2":{"15":1,"16":1}}],["sufficiently",{"2":{"425":1}}],["sufficient",{"2":{"34":1,"40":1,"172":1,"418":2}}],["suffer",{"2":{"32":1}}],["succinctgateway",{"0":{"197":1},"2":{"196":1,"197":1,"198":1,"208":2}}],["succinct",{"2":{"196":1,"201":1,"206":2,"377":5,"379":1}}],["succinctlabs",{"2":{"126":2,"151":2,"161":2}}],["succeeds",{"2":{"107":1}}],["success",{"2":{"38":1,"50":1,"69":1,"319":1,"631":3,"634":2}}],["successful",{"2":{"30":1,"35":1,"42":1,"44":1,"45":1,"57":1,"63":1,"91":1,"116":2,"117":1,"120":1,"248":2,"249":1,"252":1,"334":1,"373":1,"392":4,"425":1,"626":1,"721":1}}],["successfully",{"2":{"24":1,"31":1,"88":1,"131":2,"166":1,"263":1,"379":1,"402":1,"405":1,"409":1,"487":1,"537":2,"539":1,"629":1,"630":1,"634":1,"712":1,"714":1,"715":1,"716":1,"726":1}}],["such",{"2":{"8":1,"30":3,"31":10,"33":1,"34":9,"36":1,"42":1,"43":2,"44":1,"84":1,"134":2,"135":1,"255":1,"276":1,"308":1,"312":2,"385":1,"392":2,"403":1,"406":1,"408":1,"409":1,"412":1,"418":1,"422":1,"424":1,"425":2,"440":1,"460":1,"504":1,"506":1,"541":1,"552":1,"620":1,"660":1,"670":1}}],["subcommands",{"0":{"492":1},"2":{"488":6}}],["submodule",{"2":{"341":4}}],["submitdata",{"2":{"392":2}}],["submitpayforblob",{"2":{"373":2,"392":2,"393":1}}],["submitgrpcurl",{"2":{"352":1}}],["submitfraudproof",{"2":{"161":8}}],["submitter",{"2":{"343":1,"494":1}}],["submitted",{"2":{"85":1,"134":8,"151":2,"154":2,"159":2,"161":2,"170":2,"319":1,"320":1,"373":3,"374":1,"376":1,"388":3,"495":1,"629":2,"650":1}}],["submitting",{"0":{"142":1,"317":1,"334":1,"382":1,"388":1},"1":{"318":1,"383":1,"384":1,"385":1,"386":1,"387":1,"388":1,"389":1,"390":1,"391":1,"392":1,"393":1},"2":{"143":1,"156":1,"178":1,"181":1,"184":1,"185":1,"317":1,"318":1,"319":1,"350":1,"373":2,"385":1,"387":1,"439":1,"494":1,"665":1}}],["submits",{"2":{"135":2,"351":1,"373":2,"627":1}}],["submitheader",{"2":{"135":4,"137":2}}],["submitblockdata",{"2":{"135":2,"137":2}}],["submitrollupblock",{"2":{"127":2}}],["submit",{"0":{"318":1,"728":1},"2":{"85":1,"142":1,"161":5,"169":1,"299":1,"312":6,"317":12,"320":2,"334":2,"337":1,"352":1,"357":1,"365":1,"373":1,"374":2,"375":2,"382":2,"386":1,"388":2,"389":1,"391":3,"392":6,"393":2,"425":1,"438":2,"447":1,"448":1,"457":1,"494":10,"495":4,"620":2,"660":2,"728":1}}],["submission",{"2":{"53":1,"184":1,"213":1,"290":1,"320":1,"334":1,"373":1,"495":1}}],["submissions",{"2":{"21":1}}],["subdomain",{"2":{"334":1}}],["subtract",{"2":{"280":1}}],["subtrees",{"2":{"424":1}}],["subtree",{"2":{"166":6}}],["subspace",{"2":{"494":4}}],["subset",{"2":{"447":1}}],["subsequently",{"2":{"133":1}}],["subsequent",{"2":{"23":1,"34":1}}],["subscription",{"2":{"160":6}}],["subscriptions",{"2":{"88":1}}],["subscribe",{"2":{"88":3,"89":1}}],["sub",{"2":{"34":1,"392":2}}],["subjectivity",{"2":{"601":1}}],["subject",{"2":{"24":1,"31":1,"43":1,"278":1,"283":1,"396":1,"450":2,"466":1}}],["sig",{"2":{"508":1}}],["signer",{"2":{"268":4,"392":12,"522":2}}],["signedtx",{"2":{"507":2}}],["signed",{"2":{"137":1,"507":4}}],["signal",{"2":{"176":1}}],["signatures",{"2":{"326":2,"506":1,"522":2}}],["signature",{"2":{"134":4,"137":4,"139":4,"326":2,"385":1,"424":1,"438":1}}],["signbytes",{"2":{"137":2}}],["signs",{"2":{"136":1}}],["sign",{"2":{"88":1,"137":2,"369":1,"392":6,"507":6}}],["significant",{"2":{"441":1,"588":1,"598":1,"611":1}}],["significantly",{"2":{"44":1,"231":1,"283":1}}],["signing",{"0":{"500":1},"2":{"86":1,"265":1,"278":1,"279":1,"295":1,"297":1,"509":2,"522":2,"727":2}}],["signup",{"0":{"53":1}}],["situation",{"2":{"407":1}}],["site",{"0":{"0":1,"1":1},"1":{"1":1,"2":1,"3":1,"4":1},"2":{"1":1,"398":8}}],["sidebar",{"2":{"503":2}}],["sidenode",{"2":{"158":4,"159":4,"161":8}}],["sidenodes",{"2":{"151":2,"155":6,"158":6,"159":2,"161":6}}],["side",{"2":{"155":2,"158":2,"166":2,"634":8}}],["simultaneous",{"2":{"352":1,"502":2}}],["simulated",{"2":{"255":1}}],["simplify",{"2":{"288":1}}],["simplicity",{"2":{"170":1}}],["simply",{"2":{"135":2,"290":1,"478":1,"503":1,"638":1}}],["simpler",{"2":{"174":1,"405":1}}],["simplerollup",{"2":{"161":8}}],["simple",{"2":{"133":1,"136":1,"161":1,"178":1,"186":1,"208":1,"231":1}}],["simplest",{"2":{"82":1,"139":1,"610":1}}],["similarly",{"2":{"185":1,"204":1,"360":1,"421":1,"434":1,"458":1,"506":1}}],["similar",{"2":{"89":1,"91":1,"116":1,"117":1,"120":1,"121":2,"159":1,"184":1,"192":1,"248":1,"249":1,"252":1,"253":2,"283":1,"319":1,"322":1,"348":1,"349":1,"357":1,"384":1,"423":1,"495":1,"557":1,"579":1,"580":1,"634":2,"721":2,"727":1}}],["since",{"2":{"88":1,"89":1,"107":1,"128":1,"151":2,"159":2,"161":2,"167":1,"184":1,"282":1,"384":1,"386":1,"403":1,"407":1,"417":1,"418":1,"425":1,"445":1,"505":1,"529":1,"666":1}}],["single",{"2":{"14":1,"86":1,"129":1,"135":1,"146":1,"170":1,"178":3,"303":1,"380":1,"409":1,"412":1,"422":1,"507":4,"514":3,"515":2,"656":1}}],["sizeofdatainbytes",{"2":{"387":2}}],["sizes",{"2":{"195":1,"386":1}}],["size",{"2":{"7":1,"42":2,"89":4,"131":3,"156":1,"167":1,"169":1,"172":5,"320":1,"347":2,"385":4,"386":3,"387":1,"406":2,"411":1,"417":1,"423":1,"424":2,"438":1,"439":1,"444":1,"629":8,"671":9}}],["s",{"0":{"84":1,"86":1,"90":1,"415":1},"1":{"416":1,"417":1,"418":1,"419":1,"420":1,"421":1},"2":{"5":1,"8":1,"17":1,"19":1,"21":2,"22":1,"23":1,"24":2,"25":2,"27":1,"28":1,"31":3,"33":2,"34":12,"41":1,"42":3,"44":1,"46":1,"48":1,"53":1,"56":1,"65":1,"75":2,"79":1,"80":1,"81":1,"84":9,"85":9,"86":4,"88":4,"89":1,"90":1,"91":3,"92":1,"101":1,"105":1,"108":1,"109":2,"116":1,"117":1,"120":1,"131":4,"134":1,"137":28,"139":3,"146":1,"148":1,"163":1,"172":1,"177":1,"185":3,"201":2,"205":2,"206":5,"207":1,"213":1,"228":1,"229":1,"233":3,"248":1,"249":1,"251":1,"255":1,"257":1,"260":3,"262":2,"263":2,"264":2,"266":1,"270":1,"272":2,"277":1,"280":1,"288":1,"292":1,"301":1,"308":1,"309":1,"311":1,"312":1,"314":1,"322":1,"323":1,"324":1,"325":1,"326":1,"329":1,"330":1,"336":1,"348":2,"365":6,"367":3,"370":1,"373":2,"374":1,"377":1,"380":1,"381":1,"385":2,"386":2,"392":2,"396":1,"398":1,"400":1,"409":1,"411":1,"412":1,"415":1,"416":1,"417":1,"424":1,"425":1,"438":1,"448":1,"450":2,"452":1,"453":1,"457":1,"459":1,"460":1,"466":1,"469":1,"471":2,"473":1,"485":1,"488":2,"489":1,"494":1,"499":4,"502":3,"503":3,"504":2,"505":1,"509":1,"529":1,"531":2,"541":1,"544":1,"547":1,"552":1,"554":1,"557":2,"561":1,"565":1,"583":1,"592":10,"602":12,"607":2,"608":2,"610":1,"620":2,"627":1,"629":2,"631":1,"654":1,"660":3,"666":2,"667":1,"672":1,"679":2,"683":1,"685":1,"688":2,"689":2,"693":1,"698":1,"701":1,"712":2}}],["style",{"2":{"583":1}}],["style=",{"2":{"267":2,"268":26}}],["styles",{"2":{"267":2,"398":6}}],["stop",{"0":{"345":1,"362":1,"623":1},"2":{"362":1,"494":1,"553":1,"623":1}}],["stopbroadcast",{"2":{"262":2}}],["stops",{"2":{"255":1}}],["storing",{"2":{"110":1,"409":1,"410":1,"412":1,"414":1,"442":2,"715":1}}],["stories",{"2":{"48":1}}],["storecd",{"2":{"578":1}}],["storecelestia",{"2":{"312":1,"315":1,"317":1,"319":4,"320":1,"322":1,"323":2,"324":1,"325":1,"326":1,"327":1,"328":1,"329":2,"330":1,"332":1}}],["storesudo",{"2":{"578":1}}],["stores",{"2":{"421":1,"612":1,"696":1}}],["store```",{"2":{"326":2}}],["store=$home",{"2":{"315":2}}],["store",{"0":{"315":1,"413":1,"553":1,"555":1,"579":1},"1":{"316":1},"2":{"86":1,"107":1,"110":1,"237":3,"308":1,"312":6,"313":7,"315":8,"316":7,"317":9,"319":14,"320":3,"322":3,"323":6,"324":3,"325":3,"326":5,"327":7,"328":3,"329":10,"330":3,"331":4,"332":7,"357":3,"385":1,"408":1,"412":2,"413":1,"414":1,"417":1,"421":1,"440":1,"442":1,"478":1,"553":9,"555":5,"556":4,"578":4,"579":9,"580":8,"588":1,"611":3,"613":1,"629":8,"659":2,"680":1}}],["stored",{"2":{"84":3,"107":1,"148":1,"206":1,"308":1,"379":1,"406":1,"407":1,"409":1,"440":1,"441":2,"489":1,"531":2,"611":2}}],["storage",{"0":{"408":1,"409":1,"410":1,"412":1,"578":1,"593":1,"599":1,"612":1,"613":1,"615":1,"618":1,"619":1,"620":1,"621":1,"622":1,"623":1,"714":1},"1":{"579":1,"580":1,"594":1,"595":1,"596":1,"597":1,"598":1,"599":1,"613":1,"614":1,"615":1,"616":2,"617":1,"618":1,"619":2,"620":2,"621":1,"622":1,"623":1},"2":{"48":1,"110":1,"260":6,"303":1,"377":1,"408":2,"409":2,"410":1,"412":2,"414":2,"441":1,"479":1,"503":2,"580":1,"588":5,"598":1,"599":1,"610":1,"611":1,"612":1,"613":2,"614":2,"620":3,"621":2,"622":2,"623":1,"656":1,"678":1,"680":1,"685":1,"696":1,"707":1,"714":2,"718":1}}],["stuck",{"2":{"255":1}}],["stub",{"2":{"127":1}}],["stick",{"2":{"170":1}}],["sticking",{"2":{"84":1}}],["still",{"2":{"166":2,"398":1,"417":1,"421":1,"466":1,"566":1,"599":1,"620":1,"667":1}}],["stdencoding",{"2":{"373":4}}],["std",{"2":{"117":2,"249":2,"260":2,"262":2}}],["steal",{"2":{"403":1}}],["steady",{"2":{"19":1,"23":1}}],["step",{"0":{"68":1,"69":1,"71":1,"72":1,"82":1,"83":1,"84":1,"85":1,"86":1,"87":1,"88":1,"89":1,"90":1,"91":1},"2":{"60":2,"71":1,"83":1,"84":1,"85":2,"88":1,"134":1,"137":1,"139":1,"161":1,"173":1,"196":1,"310":1,"434":1,"508":2,"632":1,"643":1,"648":1,"649":1,"667":1}}],["steps",{"0":{"111":1,"122":1,"222":1,"237":1,"238":1,"239":1,"254":1,"275":1,"291":1,"375":1,"517":1,"561":1},"2":{"35":1,"88":1,"102":1,"103":1,"354":1,"375":1,"477":1,"539":2,"540":1,"559":1,"560":1,"628":1,"630":1,"639":1,"643":1,"711":1,"727":3}}],["stronger",{"0":{"414":1}}],["strong",{"2":{"412":1}}],["strike",{"2":{"205":1}}],["string",{"2":{"76":2,"150":2,"260":30,"263":4,"264":4,"272":2,"313":3,"314":1,"317":2,"334":1,"373":16,"374":4,"392":2,"470":1,"471":1,"541":2,"551":2,"552":2,"679":1,"688":1}}],["straight",{"2":{"681":1}}],["straightforward",{"2":{"191":1}}],["strategy",{"2":{"556":1}}],["strange",{"2":{"30":1}}],["struct",{"2":{"107":1,"134":10,"135":6,"153":4,"155":9,"156":2,"158":5,"159":4,"260":2}}],["structures",{"0":{"128":1},"2":{"108":1,"134":2,"146":1,"159":2,"392":2}}],["structure",{"0":{"3":1},"2":{"23":1,"134":2,"155":2,"158":2}}],["streaming",{"0":{"200":1},"1":{"201":1,"202":1,"203":1,"204":1,"205":1,"206":1,"207":1,"208":1,"209":1,"210":1,"211":1},"2":{"46":4}}],["stress",{"2":{"41":1}}],["strengthen",{"2":{"38":1}}],["stands",{"2":{"463":1}}],["standard",{"2":{"47":1,"260":1,"278":2,"280":1,"307":1,"382":1,"383":1,"418":4,"439":1,"559":1,"627":1}}],["standards",{"0":{"8":1},"2":{"9":1,"15":1,"16":1}}],["stage",{"2":{"201":1,"626":1,"667":1}}],["stack",{"0":{"213":1,"256":1,"284":1,"287":1,"288":1,"290":1,"338":1,"350":1,"354":1},"1":{"257":1,"258":1,"259":1,"260":1,"261":1,"262":1,"263":1,"264":1,"265":1,"266":1,"267":1,"268":1,"269":1,"270":1,"271":1,"285":1,"286":1,"287":1,"288":2,"289":2,"290":1,"291":1,"339":1,"340":1,"341":1,"342":1,"343":1,"344":1,"345":1,"346":1,"347":1,"348":1,"349":1,"350":1,"351":2,"352":2,"353":1,"355":1,"356":1,"357":1,"358":1,"359":1,"360":1,"361":1,"362":1,"363":1,"364":1},"2":{"105":1,"212":1,"214":1,"222":1,"231":2,"233":1,"255":1,"256":1,"275":2,"284":1,"285":1,"286":2,"288":3,"289":1,"291":1,"338":1,"340":1,"346":1,"352":1,"354":1,"355":1,"356":1,"364":1,"422":2}}],["stakin",{"2":{"674":1,"675":1,"676":1}}],["staking",{"0":{"443":1,"451":1,"463":1,"464":1},"1":{"444":1,"445":1,"446":1,"447":1,"448":1,"449":1,"450":1,"452":1,"453":1},"2":{"84":1,"85":1,"278":1,"426":1,"444":3,"450":1,"451":2,"452":1,"453":1,"459":1,"460":1,"463":2,"494":6,"497":4,"500":1,"721":2,"727":2}}],["stakeflow",{"2":{"674":1,"675":1,"676":1,"682":1,"689":1,"690":1,"691":1,"694":1}}],["staked",{"2":{"429":1,"450":1}}],["stakecurrency",{"2":{"398":2}}],["stakers",{"2":{"447":1,"448":1}}],["staker",{"2":{"91":3,"674":1,"675":1,"676":1}}],["stakes",{"2":{"84":2}}],["stake",{"0":{"426":1,"428":1,"431":1,"433":1,"435":1,"444":1,"459":1,"721":1},"1":{"427":1,"428":1,"429":2,"430":2,"431":2,"432":2,"433":1,"434":2,"435":2,"436":2},"2":{"19":1,"22":1,"24":1,"84":6,"203":1,"426":3,"429":1,"431":1,"432":1,"434":1,"435":1,"436":1,"444":4,"451":1,"452":1,"459":2,"478":1,"494":1,"509":4,"721":1}}],["static",{"2":{"385":4,"386":2,"503":2}}],["stats",{"0":{"328":1},"2":{"328":2}}],["stateroot",{"2":{"347":2}}],["stateless",{"2":{"131":1}}],["state",{"0":{"411":1,"598":1,"601":1},"2":{"34":6,"84":4,"121":1,"136":2,"139":1,"161":4,"177":3,"181":1,"185":1,"189":1,"192":1,"206":1,"213":1,"229":1,"253":1,"282":1,"283":1,"290":1,"312":2,"322":2,"323":4,"325":2,"329":6,"343":1,"373":2,"384":1,"385":1,"386":1,"388":1,"393":1,"403":1,"411":4,"421":3,"422":8,"424":2,"440":2,"442":2,"471":3,"478":2,"488":6,"530":1,"598":1,"599":1,"601":3,"602":1,"611":3,"620":1,"626":1,"637":1,"660":1,"679":2,"688":2,"689":1,"691":1}}],["states",{"2":{"34":9}}],["statements",{"0":{"372":1},"2":{"372":1}}],["statement",{"2":{"27":1,"373":1}}],["status",{"0":{"221":1},"2":{"7":1,"69":1,"121":2,"221":2,"243":2,"244":2,"253":2,"395":1,"470":1,"488":4,"510":1,"679":2,"712":4,"716":2}}],["stable",{"2":{"23":1,"341":1,"667":1,"698":1}}],["stability",{"0":{"466":1,"667":1},"2":{"19":1}}],["startcelestia",{"2":{"603":1,"653":1,"725":1,"727":1}}],["starts",{"2":{"343":1,"351":1,"445":1}}],["startbroadcast",{"2":{"262":2}}],["startblock",{"2":{"151":6,"159":4}}],["startprank",{"2":{"260":2}}],["startindex",{"2":{"170":2,"172":2}}],["starting",{"2":{"21":1,"50":1,"134":1,"150":1,"152":1,"422":1,"448":1,"485":1,"553":1,"649":1,"728":1}}],["start",{"0":{"343":1,"360":1,"487":1,"577":1,"580":1,"603":1,"620":1,"622":1,"636":1,"660":1,"664":1,"706":1,"712":1,"726":1},"1":{"707":1,"708":1,"709":1},"2":{"75":1,"89":3,"90":1,"134":2,"135":1,"150":2,"151":4,"155":3,"161":4,"170":1,"222":1,"225":1,"231":1,"243":2,"244":3,"251":1,"257":1,"263":1,"264":1,"268":4,"272":1,"290":1,"291":1,"309":12,"310":8,"312":1,"342":1,"343":1,"345":1,"359":1,"360":5,"367":3,"444":1,"471":6,"484":1,"485":8,"486":7,"488":4,"507":2,"523":2,"535":1,"537":2,"542":3,"544":3,"545":2,"546":2,"547":4,"550":2,"553":3,"554":3,"572":1,"577":6,"579":1,"580":10,"601":1,"603":2,"605":1,"620":5,"621":7,"628":1,"629":6,"636":2,"653":2,"660":9,"661":8,"663":7,"679":3,"709":1,"712":5,"714":5,"715":6,"716":7,"725":1,"727":2}}],["startup",{"2":{"40":1,"715":1}}],["started",{"0":{"124":1,"257":1,"277":1,"709":1},"1":{"125":1,"126":1,"127":1,"258":1,"259":1,"260":1,"261":1,"262":1,"263":1,"264":1,"265":1,"266":1,"267":1,"268":1,"269":1,"270":1,"271":1},"2":{"1":1,"27":1,"39":1,"47":1,"90":1,"243":1,"259":1,"263":1,"268":2,"270":1,"277":1,"296":1,"356":1,"365":1,"381":2,"452":1,"453":1,"540":1,"553":1,"560":1,"573":2,"578":1,"706":1,"707":1,"708":1,"709":1,"712":1,"716":1}}],["stay",{"2":{"5":1,"505":1}}],["gprc",{"2":{"676":1}}],["gptprompt",{"2":{"374":4}}],["gpt3dot5turbo",{"2":{"373":2}}],["gpt3",{"2":{"373":4,"374":2}}],["gpt",{"2":{"365":2,"366":1,"373":3,"374":3,"375":1}}],["gg",{"2":{"630":2}}],["global",{"2":{"503":3,"629":2}}],["glance",{"0":{"455":1}}],["gcpbb",{"2":{"386":1}}],["gz",{"2":{"305":24,"586":24}}],["gbps",{"2":{"479":2,"588":2,"614":2,"718":2}}],["gb",{"2":{"285":1,"303":1,"479":1,"588":4,"614":1,"656":1,"718":2}}],["ghcr",{"2":{"243":2,"346":1,"360":4,"577":6,"579":6,"580":6}}],["gmid",{"2":{"268":2}}],["gms",{"2":{"76":3,"272":3}}],["gmportalscript",{"2":{"75":2,"272":2}}],["gmportal",{"2":{"75":2,"77":7,"272":2,"273":7}}],["gm",{"2":{"75":8,"76":5,"77":8,"78":2,"222":1,"254":1,"272":13,"273":8,"274":2,"286":1,"312":4,"317":3,"319":6,"320":2,"381":2}}],["gt",{"2":{"69":1,"75":2,"76":2,"89":8,"137":2,"149":8,"207":1,"237":13,"238":12,"239":12,"243":5,"244":8,"260":4,"263":2,"264":2,"265":2,"267":14,"268":118,"272":4,"305":8,"309":6,"310":21,"312":6,"314":6,"315":4,"317":8,"319":8,"323":2,"326":4,"329":6,"331":4,"346":4,"347":1,"360":13,"390":4,"391":4,"398":14,"471":8,"473":3,"484":8,"485":4,"486":12,"490":8,"491":6,"492":8,"493":18,"494":20,"495":4,"496":6,"497":16,"498":4,"499":12,"500":2,"503":2,"507":2,"515":2,"531":4,"542":10,"544":10,"545":4,"546":4,"547":16,"550":10,"553":10,"554":8,"555":4,"556":8,"580":2,"586":8,"602":12,"620":4,"621":12,"630":4,"660":4,"661":6,"662":3,"663":12,"679":8,"693":3,"712":2,"714":2,"715":4,"716":4,"721":4,"724":2,"727":2}}],["gone",{"2":{"535":1}}],["google",{"2":{"392":4}}],["goods",{"2":{"288":1}}],["good",{"2":{"275":1,"503":1,"578":2}}],["goimport",{"2":{"387":1,"392":1}}],["going",{"2":{"71":1,"122":1,"358":1,"517":1,"566":1,"696":1,"712":1}}],["godotenv",{"2":{"372":2,"373":2}}],["gofile",{"2":{"372":1}}],["gofunc",{"2":{"137":2,"139":1,"155":1,"156":1,"157":1,"158":1,"159":1,"351":2,"373":2,"374":1}}],["go$ver",{"2":{"305":24,"586":24}}],["govcelestia",{"2":{"494":1}}],["gov",{"2":{"278":1,"494":12,"495":2}}],["governing",{"2":{"460":1}}],["governance",{"0":{"443":1,"446":1,"460":1,"494":1},"1":{"444":1,"445":1,"446":1,"447":2,"448":2,"449":1,"450":1,"495":1},"2":{"288":1,"385":1,"386":2,"421":1,"426":1,"447":2,"448":1,"450":1,"460":3,"494":3,"495":1,"671":3}}],["governmental",{"2":{"34":1}}],["government",{"2":{"33":1}}],["governed",{"2":{"31":1}}],["govern",{"2":{"30":1}}],["gobin",{"2":{"236":2}}],["go\\t",{"2":{"159":1}}],["go\\tsharesproof",{"2":{"152":1}}],["gopackage",{"2":{"150":1,"151":1,"161":1,"372":1}}],["golangnodearabica",{"2":{"305":2,"586":2}}],["golangnodemocha",{"2":{"305":2,"586":2}}],["golangnodemainnet",{"2":{"305":2,"586":2}}],["golang",{"0":{"305":1,"336":1,"340":1,"586":1},"2":{"150":1,"152":1,"155":1,"157":1,"158":1,"159":1,"161":1,"207":1,"294":1,"305":11,"307":1,"365":2,"366":1,"371":1,"374":1,"392":4,"559":1,"586":11}}],["goleveldb",{"2":{"149":4}}],["gogo",{"2":{"149":2,"374":1}}],["gotten",{"2":{"154":1,"173":1}}],["got",{"2":{"139":2,"319":1,"370":1,"373":2}}],["gotype",{"2":{"134":2,"135":1}}],["gossiped",{"2":{"384":1,"394":1,"416":1}}],["gossiping",{"2":{"139":2,"395":1}}],["gossip",{"2":{"139":1,"613":1}}],["goals",{"2":{"231":1}}],["goal",{"2":{"134":1,"380":1}}],["goes",{"2":{"60":1,"72":1,"153":1,"177":3,"280":1,"559":1}}],["go",{"0":{"371":1},"2":{"50":1,"107":1,"134":1,"135":1,"143":1,"149":5,"151":6,"160":1,"161":6,"207":1,"219":1,"220":1,"234":2,"271":1,"290":1,"304":1,"305":18,"312":1,"336":2,"349":6,"351":4,"352":2,"369":1,"371":6,"372":2,"373":6,"374":1,"422":1,"477":1,"489":1,"503":1,"515":2,"534":1,"539":1,"544":1,"556":2,"578":2,"584":1,"586":18,"627":4,"632":1,"709":1}}],["gather",{"2":{"544":1}}],["gateway",{"0":{"297":1,"298":1},"1":{"298":1},"2":{"89":2,"91":2,"196":2,"292":1,"293":1,"297":3,"298":1,"302":1,"360":2,"367":8,"552":3,"690":1}}],["game",{"2":{"365":2,"372":1,"374":1}}],["gaming",{"2":{"43":1,"232":1,"275":1}}],["gaining",{"2":{"403":1}}],["gain",{"2":{"48":1,"56":1,"65":1,"365":1,"409":1}}],["gained",{"2":{"38":1}}],["gap",{"2":{"42":1}}],["gauge",{"2":{"42":1}}],["gas=220000",{"2":{"727":1}}],["gas=220000moniker=",{"2":{"727":1}}],["gasperblobbyte",{"2":{"385":1}}],["gasperbyte",{"2":{"385":1,"386":2}}],["gaspricestep",{"2":{"398":2}}],["gasprice",{"2":{"348":2,"384":1}}],["gastoconsume",{"2":{"385":2,"386":1}}],["gaslimit",{"2":{"317":4,"329":1,"347":2,"373":4,"374":4,"384":1,"387":2,"392":4}}],["gasused",{"2":{"121":2,"253":2,"347":2}}],["gas",{"0":{"320":1,"384":1,"385":1,"386":1,"387":1},"1":{"385":1,"386":1,"387":1},"2":{"31":1,"80":1,"82":1,"84":3,"85":2,"89":4,"117":2,"205":1,"249":2,"317":2,"320":12,"329":7,"348":2,"351":1,"374":1,"379":1,"383":2,"384":9,"385":9,"386":18,"387":5,"392":6,"400":3,"439":1,"458":1,"496":2,"498":2,"522":12,"529":2,"629":16,"721":4,"727":4}}],["grpcaddr",{"2":{"392":4}}],["grpc",{"0":{"392":1,"676":1,"691":1},"2":{"309":3,"334":1,"360":4,"367":2,"392":10,"470":3,"471":3,"484":2,"485":1,"503":3,"541":2,"620":2,"629":12,"660":3,"676":19,"679":8,"688":1,"690":3,"691":15}}],["grove",{"0":{"681":1},"2":{"681":2}}],["groundbreaking",{"2":{"667":1}}],["groupname",{"2":{"267":2}}],["group",{"2":{"42":2,"45":1,"46":1,"47":13,"48":7}}],["groups",{"2":{"40":1,"42":2,"45":5,"47":2}}],["growth",{"2":{"65":1,"288":1}}],["grow",{"2":{"50":1}}],["growing",{"2":{"50":1}}],["grep",{"2":{"353":1}}],["grepping",{"2":{"353":1}}],["green",{"2":{"146":1}}],["greet",{"2":{"37":1}}],["greater",{"2":{"195":1,"601":1}}],["greatly",{"2":{"46":1}}],["great",{"2":{"41":1,"291":1}}],["grafana",{"2":{"503":3,"505":3,"548":1,"679":1}}],["graphical",{"2":{"506":1}}],["graphics",{"2":{"36":1}}],["graph",{"2":{"260":2}}],["gravity",{"2":{"201":1}}],["gradients",{"2":{"146":1}}],["grading",{"2":{"23":1}}],["grassroots",{"2":{"49":1}}],["gratitude",{"2":{"43":1,"44":1}}],["granter",{"2":{"522":2}}],["granted",{"2":{"53":1}}],["grant",{"2":{"34":1}}],["granting",{"2":{"34":1}}],["gracefully",{"2":{"8":1,"623":1}}],["guru",{"2":{"682":1}}],["guidance",{"0":{"39":1},"1":{"40":1,"41":1,"42":1,"43":1,"44":1,"45":1,"46":1,"47":1,"48":1}}],["guide",{"0":{"35":1,"256":1,"311":1,"336":1,"563":1,"627":1,"639":1,"706":1},"1":{"36":1,"37":1,"38":1,"39":1,"40":1,"41":1,"42":1,"43":1,"44":1,"45":1,"46":1,"47":1,"48":1,"257":1,"258":1,"259":1,"260":1,"261":1,"262":1,"263":1,"264":1,"265":1,"266":1,"267":1,"268":1,"269":1,"270":1,"271":1,"312":1,"313":1,"314":1,"315":1,"316":1,"317":1,"318":1,"319":1,"320":1,"321":1,"322":1,"323":1,"324":1,"325":1,"326":1,"327":1,"328":1,"329":1,"330":1,"331":1,"332":1,"564":1,"565":1,"566":1,"567":1,"568":1,"569":1,"570":1,"571":1,"628":1,"629":1,"630":1,"631":1,"632":1,"633":1,"634":1,"635":1,"636":1,"637":1,"638":1,"640":1,"641":1,"642":1,"643":1,"644":1,"645":1,"646":1,"647":1,"648":1,"649":1,"650":1,"651":1,"652":1,"653":1,"707":1,"708":1,"709":1},"2":{"49":1,"56":1,"60":1,"66":2,"79":3,"83":1,"91":1,"113":1,"198":1,"236":1,"242":1,"256":2,"290":1,"302":1,"338":1,"364":1,"469":2,"504":1,"508":2,"511":1,"534":1,"538":1,"548":1,"553":1,"587":1,"611":1,"612":1,"627":1,"639":2,"641":1,"642":1,"654":1,"666":1,"672":2,"685":2,"696":1,"709":1,"717":1}}],["guidelines",{"0":{"2":1,"12":1,"58":1},"1":{"13":1,"14":1,"15":1,"16":1},"2":{"2":1,"12":1,"17":1,"39":2}}],["guides",{"2":{"0":1,"3":1,"24":2,"39":1,"92":1,"223":1,"610":1,"685":1,"696":1}}],["guaranteeing",{"2":{"423":1}}],["guaranteed",{"2":{"281":1,"425":1,"442":2}}],["guarantees",{"2":{"201":1,"442":1}}],["guarantee",{"2":{"27":1,"412":1,"413":1,"416":1,"418":1,"440":1,"471":1,"679":1,"688":1}}],["gitcd",{"2":{"592":1}}],["gitgit",{"2":{"87":1}}],["git",{"2":{"75":6,"87":1,"126":1,"236":3,"272":4,"304":4,"307":7,"341":6,"349":4,"371":4,"539":7,"559":7,"585":4,"592":3}}],["githubusercontent",{"2":{"304":2,"400":1,"585":2,"592":10}}],["github",{"2":{"27":2,"75":3,"87":2,"149":18,"150":2,"151":14,"155":2,"156":2,"159":2,"161":22,"236":2,"272":3,"287":1,"297":1,"302":1,"307":2,"341":2,"349":2,"371":2,"372":10,"387":2,"392":16,"508":2,"531":2,"539":2,"552":1,"559":2,"592":2,"696":1}}],["given",{"0":{"556":1},"2":{"107":1,"169":1,"195":1,"334":1,"351":1,"373":6,"406":1,"419":1,"466":1,"554":2,"556":2,"598":1,"671":1,"701":1,"703":1}}],["gives",{"2":{"45":1,"65":1,"109":1,"182":1,"444":1,"671":1}}],["give",{"0":{"271":1},"2":{"44":1,"223":1,"260":2,"263":1,"355":1,"578":1}}],["giving",{"2":{"8":1,"40":1,"109":1}}],["gifts",{"2":{"43":1}}],["g",{"2":{"23":1,"31":1,"33":1,"38":1,"48":2,"312":1,"405":1,"419":1,"425":1,"503":1,"607":2}}],["geoblock",{"2":{"31":1}}],["geolocation",{"2":{"24":1}}],["gentx",{"2":{"488":2,"500":4,"649":5,"650":7}}],["gentxscelestia",{"2":{"650":1}}],["gentxs",{"2":{"488":2,"650":1}}],["gentle",{"2":{"79":1,"80":1,"101":1}}],["genutil",{"2":{"278":1}}],["genuine",{"2":{"53":1,"65":1}}],["genuinely",{"2":{"52":1}}],["genesis",{"0":{"449":1,"500":1,"647":1,"649":1,"650":1},"1":{"450":1},"2":{"159":2,"409":1,"441":1,"448":1,"449":2,"450":1,"455":1,"488":20,"500":3,"532":1,"542":10,"571":1,"592":7,"647":2,"648":3,"649":2,"650":5}}],["generic",{"2":{"117":1,"249":1}}],["generator",{"2":{"265":1}}],["generating",{"0":{"237":1},"2":{"166":1,"172":1}}],["generation",{"2":{"149":1,"171":1,"423":1}}],["generate",{"2":{"166":1,"206":1,"234":1,"237":1,"319":1,"365":1,"367":1,"369":1,"370":1,"378":1,"407":2,"418":1,"488":2,"500":1,"507":4,"534":1,"626":1}}],["generates",{"2":{"150":1}}],["generated",{"2":{"84":4,"133":1,"152":2,"173":1,"264":1,"310":2,"317":1,"322":1,"370":1,"379":2,"418":1,"473":1,"485":1,"620":1,"646":1,"649":1,"661":1,"662":1,"693":1}}],["generally",{"2":{"385":1}}],["generalized",{"2":{"170":1}}],["general",{"0":{"31":1,"625":1},"1":{"626":1},"2":{"27":1,"34":1,"195":1,"229":1,"235":1,"385":1,"416":1,"450":1,"625":1}}],["gender",{"2":{"7":1}}],["getall",{"2":{"442":2}}],["getallgms",{"2":{"76":2,"272":2}}],["getaddress",{"2":{"392":2}}],["getdataasprompt",{"2":{"373":3,"374":2}}],["getenv",{"2":{"373":8}}],["getelementbyid",{"2":{"267":2}}],["getgrpcurl",{"2":{"349":1}}],["getbyheight",{"2":{"326":1}}],["getblockdata",{"2":{"135":2,"139":2}}],["getlatestblock",{"2":{"139":2}}],["getlatestrollupheight",{"2":{"135":2}}],["getheader",{"2":{"135":4,"139":2,"373":2}}],["geth",{"2":{"119":1,"213":1,"290":1,"343":1}}],["getsharesbynamespace",{"2":{"373":2}}],["getsigner",{"2":{"268":2}}],["gets",{"2":{"107":1,"108":1,"146":1,"148":2,"170":1,"509":1}}],["gettotalgms",{"2":{"76":2,"272":2}}],["getting",{"0":{"124":1,"257":1,"277":1,"709":1},"1":{"125":1,"126":1,"127":1,"258":1,"259":1,"260":1,"261":1,"262":1,"263":1,"264":1,"265":1,"266":1,"267":1,"268":1,"269":1,"270":1,"271":1},"2":{"25":2,"91":2,"93":1,"268":2,"277":1,"366":1,"373":2,"402":1}}],["get",{"0":{"324":1,"325":1,"326":1,"328":1},"2":{"1":1,"27":1,"39":1,"42":1,"47":1,"107":1,"108":2,"151":8,"159":2,"161":4,"173":2,"228":1,"229":1,"232":1,"255":1,"259":1,"296":1,"317":1,"319":15,"324":1,"325":1,"326":6,"327":4,"331":1,"332":7,"349":1,"356":1,"365":1,"367":1,"368":1,"374":3,"375":1,"377":1,"381":2,"386":1,"388":1,"392":2,"402":1,"452":1,"453":1,"485":1,"493":1,"494":1,"504":1,"509":1,"537":1,"540":1,"560":1,"573":2,"592":1,"620":2,"630":1,"660":1,"662":1,"706":1,"707":1,"708":1,"709":1,"721":1}}],["occupied",{"2":{"671":1}}],["occurs",{"2":{"403":1}}],["occurring",{"2":{"283":1}}],["occur",{"2":{"139":1,"494":1}}],["occasional",{"2":{"667":1,"698":1}}],["oil",{"2":{"519":2}}],["o",{"2":{"417":1,"418":2,"602":12}}],["o1w4npv7w4caztgjomx71w96m63+xzynarxlu7wwvrrwbeb6cw8r93yhxt1r4+txcbgve76obzf5jltnu22gksd2cl+83d8dgjx0fkcwzd0vofkgmboky1utddu8704s2mwgnne09s1bzw+n9fq6fkfw7pvwjl200ecs0ofj3hfpaeywnlgyygqc89dh+98gd5trdu4anql9afmw+sdzjtc9s0fzlwyroos0bvk3w7evnpmwxe5qrdzklbmv1lzi4ofrrxlhgmbyoajhhsen+b81lguh33hdet8k9nvkksf2+w3xul6upsxydpbwsv2ghskr+yfuldbvyl1rotvts1zxlpepz0m1e",{"2":{"349":2}}],["old",{"2":{"345":1,"410":1,"550":1,"554":1}}],["older",{"2":{"31":1,"441":1,"624":1}}],["ostrich",{"2":{"519":2}}],["os|file|kwallet|pass|test|memory",{"2":{"489":1,"531":2}}],["oss",{"2":{"281":1}}],["os",{"2":{"150":2,"151":2,"161":2,"304":1,"372":2,"373":8,"502":2,"585":1}}],["omitempty",{"2":{"134":2}}],["ok",{"2":{"117":2,"249":2}}],["owners",{"2":{"91":6}}],["ownership",{"2":{"34":1,"91":2,"260":2}}],["owner",{"2":{"84":1,"91":5,"260":12}}],["own",{"2":{"34":2,"84":1,"89":1,"201":1,"231":1,"233":1,"258":1,"266":1,"284":1,"289":1,"290":1,"293":1,"335":1,"338":1,"355":1,"357":2,"370":1,"419":1,"422":1,"423":1,"442":1,"444":1,"458":1,"459":1,"471":1,"484":1,"525":1,"529":1,"537":1,"539":1,"568":1,"620":1,"639":1,"645":1,"660":1,"679":1,"681":1,"721":1}}],["otel",{"2":{"24":1,"545":2,"546":2,"548":3}}],["otherwise",{"2":{"11":1,"34":4,"89":1,"107":1,"109":1,"181":1,"317":1,"334":1,"494":1,"643":1,"712":1}}],["others",{"2":{"8":1,"652":1}}],["other",{"0":{"283":1,"648":1},"2":{"3":1,"8":2,"9":1,"13":1,"24":2,"25":1,"27":1,"30":2,"31":10,"32":1,"34":12,"40":1,"42":5,"43":2,"45":2,"47":1,"48":3,"52":1,"64":1,"84":1,"105":1,"107":1,"119":1,"129":1,"134":2,"143":1,"162":1,"176":1,"195":1,"205":1,"215":1,"232":1,"237":1,"281":2,"283":1,"293":1,"302":1,"346":1,"383":1,"386":1,"402":1,"409":1,"412":1,"419":1,"422":2,"442":3,"448":1,"459":1,"472":1,"478":1,"607":1,"637":1,"648":2,"649":1,"650":2,"651":2,"652":2,"693":1,"707":1}}],["override",{"2":{"503":2}}],["overpaying",{"2":{"386":1}}],["overpay",{"2":{"384":1}}],["overhead",{"2":{"284":1,"289":1}}],["over",{"0":{"371":1,"634":1},"2":{"59":1,"60":1,"122":1,"134":4,"136":1,"137":1,"148":3,"165":1,"207":1,"229":1,"234":2,"256":1,"268":2,"297":1,"304":1,"310":1,"312":1,"345":1,"367":1,"369":1,"371":1,"442":1,"460":1,"477":1,"534":1,"544":1,"559":1,"584":1,"634":2,"709":1}}],["overview",{"0":{"64":1,"105":1,"113":1,"146":1,"439":1,"454":1,"478":1,"613":1,"655":1,"696":1},"1":{"147":1,"148":1,"455":1,"456":1,"457":1,"458":1,"459":1,"460":1,"461":1,"462":1,"463":1,"464":1},"2":{"22":1,"37":1,"47":1,"59":1,"60":1,"61":1,"69":1,"71":1,"79":1,"80":1,"94":1,"101":1,"131":2,"148":1,"257":1,"277":1,"432":1,"436":1,"439":1,"548":1,"561":1}}],["overall",{"2":{"8":1,"41":1,"94":1}}],["observe",{"2":{"556":1}}],["observer",{"2":{"545":2}}],["observance",{"2":{"34":1}}],["objects",{"2":{"268":2}}],["object",{"2":{"88":2}}],["objective",{"2":{"27":1}}],["objectives",{"0":{"19":1},"2":{"19":1,"36":1}}],["obligation",{"2":{"34":1}}],["obligations",{"2":{"33":1}}],["obligated",{"2":{"11":1}}],["obtained",{"2":{"34":2,"366":1}}],["obtaining",{"2":{"34":2}}],["obtain",{"2":{"32":1,"33":1,"44":1,"88":1}}],["onto",{"2":{"252":1,"380":1,"414":1}}],["oncall",{"2":{"505":2}}],["onchange=",{"2":{"268":4}}],["onchain",{"0":{"379":1},"2":{"109":1,"110":1,"201":2,"208":1,"209":1,"232":1,"275":1,"285":1,"403":1,"450":1,"727":1}}],["onclick=",{"2":{"268":8,"398":2}}],["once",{"2":{"25":1,"85":1,"88":1,"107":1,"120":1,"137":1,"185":1,"192":1,"228":1,"234":1,"252":1,"263":3,"264":1,"268":2,"310":1,"312":1,"405":1,"412":1,"424":1,"440":1,"441":2,"442":1,"485":1,"503":2,"522":1,"601":1,"620":1,"623":1,"648":1,"650":2,"661":1}}],["onboarding",{"0":{"48":1}}],["onecelestia",{"2":{"493":1}}],["ones",{"2":{"167":1,"185":1,"626":1,"632":1,"635":1}}],["onestepproofentry",{"2":{"98":1,"99":1}}],["onestepproverhostio",{"2":{"98":1,"99":1}}],["onestepprovermath",{"2":{"98":1,"99":1}}],["onestepprovermemory",{"2":{"98":1,"99":1}}],["onestepprover0",{"2":{"98":1,"99":1}}],["one",{"0":{"388":1},"2":{"34":5,"45":1,"46":1,"47":1,"48":1,"66":2,"76":1,"88":1,"89":1,"102":1,"105":1,"107":1,"108":1,"116":1,"117":1,"143":1,"167":1,"176":1,"182":1,"185":1,"189":3,"207":1,"210":1,"229":1,"231":1,"232":1,"233":1,"237":1,"243":2,"244":2,"248":1,"249":1,"263":1,"272":1,"302":1,"309":3,"370":1,"376":2,"382":1,"385":1,"388":1,"400":2,"406":1,"418":2,"419":1,"422":2,"424":2,"425":2,"484":1,"485":2,"488":4,"493":4,"494":3,"503":3,"515":1,"517":1,"518":2,"527":1,"529":2,"531":2,"553":2,"556":1,"571":2,"613":1,"620":1,"627":2,"629":2,"650":1,"660":3,"661":2,"671":2,"674":1,"675":1,"676":1,"679":1,"688":1,"690":1,"691":1,"722":1}}],["onlyowner",{"2":{"260":8}}],["only",{"2":{"34":2,"89":3,"94":2,"107":2,"108":3,"109":1,"139":2,"167":1,"170":1,"173":1,"178":1,"206":1,"232":1,"233":1,"237":1,"260":2,"278":1,"280":1,"289":1,"297":1,"302":1,"380":1,"386":1,"392":2,"407":1,"409":1,"412":1,"415":1,"416":1,"417":1,"418":2,"419":2,"421":1,"422":3,"423":2,"424":2,"441":2,"442":1,"495":1,"507":2,"519":2,"536":1,"541":1,"583":1,"592":1,"623":1,"639":1,"679":1,"692":1,"709":1,"727":1}}],["online",{"2":{"10":1,"37":1,"44":1,"45":1,"712":2,"714":2,"715":2,"716":2}}],["onwards",{"2":{"23":1}}],["on",{"0":{"73":1,"149":1,"214":1,"224":1,"245":1,"270":1,"272":1,"316":1,"349":1,"444":1,"451":1,"509":1,"518":1,"529":1,"599":1,"608":1,"696":1},"1":{"74":1,"75":1,"76":1,"77":1,"78":1,"150":1,"151":1,"152":1,"215":1,"216":1,"217":1,"218":1,"219":1,"220":1,"221":1,"225":1,"226":1,"227":1,"246":1,"247":1,"248":1,"249":1,"250":1,"251":1,"252":1,"253":1,"254":1,"273":1,"274":1,"275":1,"452":1,"453":1,"519":1,"520":1,"521":1,"522":1,"523":1,"524":1,"525":1},"2":{"8":2,"23":1,"24":5,"27":5,"30":3,"31":3,"32":1,"33":1,"34":5,"36":1,"39":1,"40":3,"41":2,"43":2,"44":1,"45":4,"46":5,"47":1,"48":1,"53":1,"59":1,"60":1,"61":2,"65":1,"68":1,"69":2,"71":3,"76":1,"77":3,"78":1,"79":1,"80":2,"82":1,"84":10,"85":2,"86":1,"88":1,"89":5,"90":1,"91":51,"95":1,"105":1,"107":3,"108":1,"109":4,"110":3,"113":1,"114":1,"117":1,"119":2,"129":1,"133":2,"135":6,"137":1,"146":1,"148":3,"163":1,"165":1,"166":3,"167":1,"168":1,"169":2,"172":2,"173":2,"176":1,"178":1,"180":1,"196":1,"201":2,"203":2,"204":4,"205":2,"206":5,"207":2,"208":1,"210":3,"211":1,"212":1,"213":6,"214":1,"222":2,"223":2,"225":1,"227":1,"229":2,"233":1,"234":1,"237":1,"242":2,"243":4,"244":1,"246":1,"249":1,"251":1,"255":4,"256":1,"267":2,"268":5,"269":1,"270":1,"271":1,"272":1,"273":3,"274":1,"275":3,"277":1,"279":1,"281":1,"283":1,"285":3,"286":3,"290":4,"293":2,"297":1,"302":1,"304":1,"307":1,"308":1,"309":2,"310":3,"314":3,"315":1,"319":1,"320":1,"334":1,"347":1,"348":1,"351":1,"357":1,"360":1,"365":2,"367":5,"368":1,"373":1,"376":2,"377":1,"379":1,"380":3,"385":3,"386":6,"392":2,"396":1,"403":1,"405":1,"406":1,"414":1,"415":1,"416":1,"417":1,"419":1,"421":2,"422":3,"425":1,"426":2,"431":1,"435":1,"438":1,"439":3,"441":2,"442":6,"444":3,"445":1,"447":2,"457":2,"458":2,"460":1,"465":3,"466":2,"469":1,"471":1,"473":2,"478":2,"480":1,"484":4,"485":4,"487":1,"489":1,"494":3,"495":1,"500":1,"502":2,"503":9,"504":1,"505":2,"508":1,"509":1,"510":2,"511":1,"517":3,"525":2,"526":2,"529":2,"530":1,"533":2,"537":2,"539":2,"540":2,"541":4,"543":1,"548":3,"550":2,"552":5,"553":1,"559":1,"560":1,"575":1,"577":3,"578":5,"583":1,"585":1,"587":1,"589":1,"590":1,"591":1,"597":1,"599":1,"600":1,"601":2,"603":2,"607":3,"610":4,"611":1,"615":1,"620":6,"624":2,"625":3,"626":4,"627":3,"629":2,"638":1,"653":2,"654":1,"655":1,"657":2,"660":7,"664":1,"665":1,"671":3,"672":1,"679":1,"680":1,"681":2,"685":7,"689":1,"693":2,"696":4,"701":3,"703":3,"706":1,"709":2,"717":1,"719":2,"720":1,"724":4,"726":1,"727":1}}],["opqym",{"2":{"520":2,"524":2}}],["opt",{"2":{"458":1,"557":4}}],["optimally",{"2":{"422":1}}],["optimal",{"2":{"307":1,"559":1}}],["optimized",{"2":{"422":1}}],["optimize",{"2":{"283":1}}],["optimismcd",{"2":{"341":1}}],["optimism",{"0":{"288":1},"2":{"213":2,"275":1,"284":1,"285":2,"287":1,"288":6,"290":1,"341":3,"342":2,"344":1,"351":1,"354":1,"357":1,"359":2,"360":1,"361":1,"450":1}}],["optimised",{"2":{"201":1}}],["optimistically",{"2":{"176":1,"177":1}}],["optimistic",{"0":{"176":1,"177":1,"178":1,"179":1,"180":1,"181":1,"182":1,"183":1},"1":{"177":1,"178":2,"179":2,"180":2,"181":1,"182":2,"183":2},"2":{"136":1,"139":1,"163":1,"169":1,"176":4,"177":1,"180":1,"184":1,"195":1,"206":1}}],["option",{"2":{"105":1,"307":2,"355":1,"478":1,"494":2,"503":1,"559":2,"566":1,"610":4,"611":1,"709":1,"727":1}}],["options",{"0":{"240":1,"610":1,"729":1},"2":{"40":2,"41":1,"46":1,"83":1,"223":1,"225":1,"231":1,"233":1,"268":4,"376":1,"392":6,"488":1,"489":1,"492":1,"493":1,"494":2,"502":2,"522":4,"610":1,"672":1,"685":1,"696":1,"709":1,"729":1}}],["optionally",{"2":{"69":1,"71":2,"317":1,"592":1}}],["optional",{"0":{"318":1,"486":1,"487":1,"530":1,"531":1,"605":1,"606":1,"610":1,"611":1,"621":1,"622":1,"644":1,"648":1,"663":1,"664":1,"722":1,"726":1,"729":1},"1":{"607":1,"608":1,"609":1,"723":1,"724":1,"725":1,"726":1},"2":{"24":1,"40":1,"45":1,"76":1,"307":1,"317":8,"344":1,"399":1,"400":2,"425":1,"522":1,"559":1,"603":1}}],["ops",{"2":{"346":1,"357":1,"360":1}}],["oprexqlg9er1oey1de4mkwvmjlfnqoocg==",{"2":{"326":2}}],["op",{"0":{"213":1,"284":1,"287":1,"288":1,"290":1,"338":1,"350":1,"354":1},"1":{"285":1,"286":1,"287":1,"288":2,"289":2,"290":1,"291":1,"339":1,"340":1,"341":1,"342":1,"343":1,"344":1,"345":1,"346":1,"347":1,"348":1,"349":1,"350":1,"351":2,"352":2,"353":1,"355":1,"356":1,"357":1,"358":1,"359":1,"360":1,"361":1,"362":1,"363":1,"364":1},"2":{"212":1,"213":7,"214":1,"222":1,"255":1,"275":1,"284":1,"285":2,"286":2,"288":3,"290":7,"291":1,"338":1,"340":1,"341":3,"343":4,"346":1,"351":4,"352":2,"353":1,"354":1,"355":1,"356":1,"364":1}}],["opportunities",{"2":{"48":1,"65":1}}],["opportunity",{"2":{"19":1,"30":1,"32":1}}],["operating",{"2":{"540":1,"557":1,"560":1,"575":1}}],["operations",{"2":{"263":1,"264":1,"385":1,"424":1,"463":1,"465":1,"485":1,"620":1}}],["operation",{"2":{"30":1,"86":1,"117":1,"249":1,"588":1}}],["operators",{"2":{"449":1,"548":1,"667":1,"696":1}}],["operator",{"2":{"307":1,"377":3,"378":1,"379":1,"501":1,"559":1,"610":1}}],["operates",{"2":{"205":1,"351":1}}],["operate",{"2":{"107":1,"422":1}}],["operated",{"2":{"24":1,"30":1}}],["opened",{"2":{"635":1}}],["opentelemetry",{"2":{"547":1,"548":1}}],["openaikey",{"2":{"373":4}}],["openai",{"0":{"369":1},"2":{"365":2,"366":1,"369":3,"370":3,"372":4,"373":12,"374":1}}],["openrpc",{"2":{"94":1,"295":1}}],["openssl",{"2":{"89":1}}],["opening",{"2":{"34":1,"557":2}}],["open",{"0":{"429":1,"434":1,"557":1,"634":1},"2":{"7":1,"23":1,"26":1,"117":1,"243":1,"249":1,"260":1,"268":2,"269":2,"285":1,"288":1,"309":1,"357":1,"367":1,"370":1,"429":1,"470":3,"471":1,"484":1,"495":2,"502":2,"503":2,"537":1,"541":3,"548":1,"550":1,"552":2,"557":9,"577":1,"603":1,"620":1,"628":1,"651":1,"653":1,"660":1,"691":1,"724":1}}],["opinions",{"2":{"8":1}}],["our",{"0":{"7":1,"8":1,"120":1},"2":{"7":1,"8":2,"9":1,"10":1,"30":1,"31":1,"33":3,"48":1,"52":2,"53":2,"55":1,"64":1,"65":2,"122":1,"251":1,"261":2,"262":1,"264":2,"269":2,"271":1,"305":1,"317":1,"322":1,"323":1,"367":5,"372":1,"373":4,"374":1,"476":1,"517":1,"529":2,"536":1,"561":1,"579":1,"586":1,"630":1,"646":1,"695":1,"705":1}}],["outbound",{"2":{"627":1}}],["outbox",{"2":{"98":1,"99":1}}],["outcome",{"2":{"424":1}}],["outercontainerstyle",{"2":{"268":4}}],["outsource",{"2":{"231":1}}],["outsourcing",{"2":{"231":1}}],["outside",{"2":{"34":2,"194":1,"385":1,"412":1,"441":1}}],["outline",{"2":{"133":2,"139":1,"163":1}}],["outlined",{"2":{"23":1,"102":1,"103":1,"139":1,"181":1}}],["outputs",{"2":{"499":2}}],["output",{"2":{"75":3,"116":1,"117":1,"120":1,"121":1,"248":1,"249":1,"252":1,"253":1,"264":1,"272":3,"304":2,"305":1,"307":1,"317":3,"319":4,"326":2,"332":1,"347":1,"348":1,"373":2,"374":2,"488":2,"507":6,"515":2,"519":1,"520":1,"522":1,"523":1,"524":1,"525":2,"531":4,"536":1,"539":1,"540":2,"559":1,"560":2,"585":2,"586":1,"631":1,"634":2,"646":1,"652":2,"659":1,"721":2,"727":1}}],["outputinfo",{"2":{"68":1,"92":1}}],["out",{"0":{"66":1,"70":1,"270":1},"1":{"67":1,"68":1,"69":1,"70":1,"71":2,"72":2},"2":{"0":1,"36":3,"43":1,"44":1,"45":1,"46":1,"66":2,"77":3,"92":1,"107":1,"127":2,"130":1,"135":1,"167":1,"172":2,"177":1,"189":1,"192":1,"201":1,"229":1,"233":1,"251":1,"254":1,"263":1,"265":2,"269":1,"271":1,"273":3,"277":1,"290":1,"298":1,"307":1,"309":1,"312":1,"335":1,"341":1,"345":1,"352":1,"355":1,"388":1,"405":1,"485":1,"502":2,"503":1,"539":1,"548":1,"559":1,"639":1,"665":1,"679":1,"685":2,"703":1,"707":1}}],["oracle",{"0":{"108":1},"2":{"106":1,"108":3,"129":1}}],["oral",{"2":{"34":1}}],["originate",{"2":{"394":1}}],["originally",{"2":{"384":1,"553":1}}],["original",{"2":{"79":1,"172":1,"351":2,"388":1,"416":1,"418":3,"419":1,"424":2,"425":1,"523":2}}],["origin",{"2":{"71":1,"396":1,"527":4,"528":1}}],["orientation",{"2":{"7":1}}],["orbitsetupscriptconfig",{"2":{"86":1,"87":1}}],["orbit",{"0":{"66":1,"69":1,"79":1,"84":1,"85":1},"1":{"67":1,"68":1,"69":1,"70":1,"71":1,"72":1,"80":1,"81":1,"82":1,"83":1,"84":1,"85":1,"86":1,"87":1,"88":1,"89":1,"90":1,"91":1,"92":1,"93":1,"94":1,"95":1,"96":1,"97":1,"98":1,"99":1},"2":{"66":2,"68":2,"73":1,"79":5,"80":2,"81":1,"82":2,"83":1,"84":8,"85":7,"87":5,"89":1,"90":1,"91":28,"92":1,"94":4,"95":1,"101":3,"105":4,"106":1,"108":1}}],["ordered",{"2":{"419":1,"422":2}}],["orders",{"2":{"289":1}}],["ordering",{"2":{"41":1,"229":1,"423":1,"634":2}}],["order",{"0":{"556":1},"2":{"30":1,"31":1,"41":1,"43":1,"76":1,"88":1,"107":1,"108":2,"145":1,"148":1,"172":1,"173":1,"228":1,"256":1,"272":1,"313":1,"314":1,"315":1,"324":1,"325":1,"334":1,"354":1,"357":2,"367":1,"369":1,"384":1,"417":1,"422":1,"440":1,"442":1,"469":1,"470":1,"486":1,"556":2,"573":1,"579":1,"603":1,"620":1,"621":1,"623":1,"624":1,"625":2,"626":3,"648":1,"660":1,"663":1,"672":2,"688":1,"709":1,"727":2}}],["organizing",{"2":{"39":1,"42":2,"47":2,"52":1,"56":1,"63":1}}],["organizers",{"2":{"40":1,"43":1,"44":1,"48":2,"49":1,"50":2,"52":1,"53":1,"55":1,"56":2,"65":1}}],["organizer",{"2":{"39":1,"43":1,"47":4,"56":1,"63":1,"65":1}}],["organize",{"2":{"35":2,"42":1,"43":1}}],["organization",{"2":{"27":1}}],["org",{"0":{"6":1,"52":1},"1":{"7":1,"8":1,"9":1,"10":1,"11":1,"12":1,"13":1,"14":1,"15":1,"16":1,"17":1},"2":{"7":1,"11":1,"17":3,"25":1,"46":1,"47":2,"52":1,"63":1,"84":1,"289":1,"305":8,"392":4,"540":2,"560":2,"586":8,"674":2,"675":2,"676":2,"689":1,"690":1,"691":1}}],["or",{"0":{"100":1,"316":1,"364":1,"530":1},"1":{"101":1,"102":1,"103":1},"2":{"2":1,"4":1,"7":2,"8":8,"9":2,"10":2,"11":1,"13":2,"14":2,"15":2,"16":2,"23":1,"24":6,"25":4,"27":3,"30":9,"31":33,"32":3,"33":6,"34":50,"36":3,"37":4,"38":1,"40":4,"41":3,"42":3,"43":5,"44":4,"45":3,"46":2,"47":1,"48":2,"65":1,"82":1,"83":1,"84":3,"89":2,"91":5,"92":1,"94":1,"102":1,"105":1,"110":2,"131":1,"133":1,"135":1,"136":1,"139":2,"143":2,"145":1,"146":2,"148":1,"159":1,"163":1,"167":1,"172":1,"184":1,"203":2,"204":1,"206":1,"229":2,"231":1,"232":1,"233":3,"235":1,"236":1,"237":2,"244":1,"246":1,"255":5,"256":1,"262":1,"265":1,"266":1,"267":2,"268":2,"269":1,"280":1,"297":2,"307":1,"308":1,"309":2,"310":2,"312":1,"313":1,"316":2,"317":1,"334":2,"341":1,"350":2,"355":1,"357":1,"360":1,"364":1,"366":1,"367":1,"382":1,"384":1,"385":1,"388":1,"392":4,"394":1,"395":1,"396":1,"403":2,"409":1,"413":1,"418":1,"422":1,"425":1,"432":1,"436":1,"438":1,"442":2,"444":1,"450":2,"458":1,"463":1,"465":1,"471":3,"472":1,"478":1,"484":3,"488":4,"489":1,"500":1,"503":6,"505":3,"506":1,"509":1,"529":1,"530":2,"537":1,"547":2,"554":2,"575":1,"576":1,"577":2,"579":1,"599":1,"601":1,"607":1,"620":3,"624":1,"635":1,"639":1,"649":1,"652":1,"660":4,"661":1,"667":2,"671":1,"679":3,"681":1,"688":1,"690":1,"693":1,"696":2,"698":1,"711":1,"724":1}}],["often",{"2":{"40":1,"42":1,"43":1,"45":1,"384":1,"386":1,"624":1}}],["off",{"2":{"352":1,"379":1,"503":1}}],["offloads",{"2":{"228":1}}],["offline",{"2":{"10":1,"509":1}}],["offchain",{"2":{"79":2,"91":1,"205":1,"209":1,"403":1}}],["offshore",{"2":{"34":1}}],["offense",{"2":{"509":1}}],["offensive",{"2":{"9":1}}],["offers",{"2":{"57":1,"83":1,"205":1,"214":1,"709":1}}],["offered",{"2":{"34":1,"42":1}}],["offer",{"2":{"34":5,"40":4,"41":1,"43":1,"44":2,"205":1}}],["offering",{"2":{"34":1,"41":1,"105":1,"212":1}}],["officially",{"2":{"10":1}}],["official",{"2":{"0":1,"10":2,"89":1}}],["of",{"0":{"6":1,"19":1,"52":1,"66":1,"146":1,"151":1,"170":1,"171":1,"172":1,"173":2,"174":1,"175":1,"177":1,"178":1,"179":1,"180":1,"185":1,"186":1,"187":1,"204":1,"230":1,"231":1,"286":2,"323":1,"329":1,"344":1,"351":1,"361":1,"406":1,"412":1,"414":1,"418":1,"424":1,"444":1,"454":1,"456":1,"459":1,"478":1,"516":1,"525":1,"553":1,"613":1,"655":1},"1":{"7":1,"8":1,"9":1,"10":1,"11":1,"12":1,"13":1,"14":1,"15":1,"16":1,"17":1,"67":1,"68":1,"69":1,"70":1,"71":1,"72":1,"147":1,"148":1,"171":1,"172":1,"173":1,"174":1,"175":1,"178":1,"179":1,"180":1,"186":1,"187":1,"231":1,"232":1,"233":1,"425":1,"455":1,"456":1,"457":2,"458":2,"459":2,"460":2,"461":2,"462":2,"463":2,"464":2},"2":{"0":1,"2":1,"7":2,"8":6,"9":2,"10":2,"11":3,"12":2,"13":3,"14":3,"15":4,"16":6,"17":4,"19":3,"21":1,"22":3,"23":5,"24":7,"25":1,"26":2,"27":3,"28":1,"30":9,"31":14,"32":5,"33":10,"34":47,"36":2,"37":1,"38":1,"39":3,"40":3,"41":1,"42":8,"43":5,"44":6,"45":1,"46":2,"47":5,"48":4,"49":1,"50":1,"52":5,"53":2,"55":1,"56":2,"59":2,"60":1,"61":1,"62":1,"64":1,"65":7,"66":3,"68":1,"69":3,"71":1,"72":1,"75":1,"76":1,"77":1,"79":3,"82":2,"84":6,"85":3,"87":2,"88":1,"89":3,"90":1,"91":5,"92":3,"94":1,"102":1,"105":1,"106":2,"107":8,"108":2,"109":2,"110":2,"114":2,"116":1,"117":3,"121":1,"126":1,"128":3,"131":5,"134":28,"135":10,"136":3,"137":2,"139":8,"143":2,"146":4,"148":12,"149":2,"150":2,"152":5,"153":8,"154":1,"155":11,"156":2,"157":1,"158":10,"159":12,"161":8,"162":1,"163":1,"166":2,"167":4,"168":2,"169":1,"170":9,"171":1,"172":10,"173":20,"174":2,"176":2,"177":6,"178":1,"180":2,"181":10,"183":2,"184":1,"185":9,"186":1,"189":4,"192":4,"194":1,"195":3,"196":1,"201":4,"203":2,"204":1,"205":3,"206":4,"207":1,"213":2,"218":1,"222":2,"223":1,"228":1,"229":6,"231":7,"232":6,"233":3,"234":1,"236":2,"237":1,"242":1,"243":1,"246":2,"248":1,"249":3,"252":1,"253":1,"255":3,"257":1,"260":5,"263":4,"264":2,"265":1,"267":2,"268":2,"272":2,"273":1,"275":1,"278":1,"281":1,"283":2,"290":3,"291":1,"293":3,"295":1,"302":1,"304":1,"305":1,"307":2,"309":1,"311":1,"312":1,"313":2,"314":2,"316":1,"317":8,"319":3,"320":3,"322":1,"323":4,"326":2,"329":3,"332":1,"344":2,"345":1,"346":2,"347":1,"350":1,"351":6,"354":2,"355":2,"357":1,"360":3,"361":2,"365":2,"367":4,"373":1,"374":2,"376":1,"377":1,"378":1,"379":1,"380":3,"382":1,"383":2,"384":5,"385":10,"386":8,"387":3,"388":3,"392":4,"394":2,"395":1,"396":1,"399":1,"400":1,"402":2,"403":4,"405":3,"406":3,"407":1,"408":1,"409":2,"411":2,"412":4,"413":2,"414":1,"415":2,"416":5,"417":5,"418":8,"419":7,"421":7,"422":11,"423":4,"424":15,"425":4,"426":1,"430":1,"431":1,"435":1,"438":3,"439":2,"440":3,"441":6,"442":4,"444":5,"445":3,"447":2,"448":1,"449":4,"450":5,"451":2,"455":1,"457":1,"458":2,"459":3,"460":2,"469":1,"470":2,"471":1,"472":2,"473":1,"478":3,"484":3,"485":1,"486":7,"494":3,"495":1,"501":1,"502":4,"503":2,"504":1,"505":1,"506":1,"507":2,"509":7,"510":1,"515":2,"516":1,"517":1,"522":1,"525":3,"526":2,"527":4,"530":2,"537":1,"539":2,"547":1,"548":2,"550":1,"553":1,"556":1,"557":1,"559":2,"561":1,"565":2,"568":1,"569":1,"571":1,"575":1,"576":2,"577":1,"583":1,"586":1,"588":4,"592":2,"596":1,"598":1,"599":3,"600":2,"601":1,"603":1,"611":4,"620":3,"621":7,"624":1,"625":3,"627":2,"628":2,"630":1,"646":3,"650":2,"651":2,"653":1,"655":1,"659":1,"660":3,"663":7,"666":3,"670":1,"671":2,"672":4,"679":2,"680":1,"685":5,"688":1,"693":3,"696":1,"698":2,"707":1,"709":1,"714":1,"715":1,"721":2,"722":1,"724":2,"727":5,"730":1}}],["mzonder",{"2":{"674":1,"675":1,"676":1}}],["m",{"2":{"452":1,"683":1}}],["mb",{"2":{"303":1,"656":1}}],["mbidstyle",{"2":{"268":4}}],["msgcreatevestingaccount",{"2":{"522":2}}],["msgsubmitproposal",{"2":{"494":2}}],["msg",{"2":{"260":4,"373":4,"424":1,"494":1,"629":4}}],["msgpayforblobs",{"2":{"165":1,"301":2,"382":1,"386":2}}],["mykey",{"2":{"264":3,"270":2}}],["my",{"2":{"244":4,"260":14,"263":2,"264":2,"357":1,"578":5,"579":8,"580":8,"661":6}}],["myrollup",{"2":{"127":2}}],["mnemonics",{"2":{"367":1}}],["mnemonic",{"2":{"239":2,"351":1,"519":3,"536":1,"630":4}}],["md013",{"2":{"326":2}}],["md",{"2":{"159":2}}],["mkdir",{"2":{"115":2,"247":2,"578":2,"602":6}}],["mutual",{"0":{"569":1}}],["mutually",{"2":{"42":1}}],["multiaddress",{"2":{"715":1}}],["multiaddresses",{"2":{"568":1}}],["multi",{"2":{"508":1}}],["multiplier",{"2":{"629":4}}],["multiplies",{"2":{"386":1}}],["multiple",{"0":{"388":1},"2":{"43":1,"103":1,"131":1,"148":1,"152":2,"153":8,"155":4,"170":2,"232":1,"302":1,"405":1,"408":1,"413":1,"416":1,"419":1,"422":2,"424":1,"475":1,"503":1,"682":1}}],["multiproof",{"2":{"155":2}}],["multisign",{"2":{"507":2}}],["multisig",{"0":{"506":1},"1":{"507":1,"508":1},"2":{"84":1,"506":3,"507":18,"508":3}}],["much",{"2":{"41":1,"89":1,"229":1,"231":1,"232":1,"649":1}}],["mustnewv0",{"2":{"392":2}}],["must",{"2":{"30":1,"31":2,"84":2,"89":1,"256":1,"305":1,"357":2,"360":1,"384":1,"388":3,"415":1,"417":1,"419":3,"422":2,"486":2,"494":1,"554":4,"574":1,"586":1,"597":1,"601":1,"620":1,"621":2,"643":1,"644":1,"645":1,"650":2,"660":1,"663":2,"671":1,"727":1}}],["milestone",{"2":{"685":1}}],["million",{"2":{"499":2}}],["mib",{"2":{"671":1}}],["middleware",{"2":{"638":1}}],["migrate",{"2":{"488":4}}],["might",{"2":{"40":3,"169":1,"229":1,"309":1,"312":1,"418":1,"494":1,"557":1,"592":1,"598":1,"660":1,"679":1,"701":1}}],["microtia",{"0":{"464":1},"2":{"464":1}}],["micro",{"2":{"463":1}}],["microphone",{"2":{"40":2}}],["microphones",{"2":{"36":1,"40":1}}],["mixhash",{"2":{"347":2}}],["mit",{"2":{"260":4,"262":2}}],["mirrors",{"2":{"214":1}}],["miner",{"2":{"347":2}}],["mined",{"2":{"91":4}}],["mintscan",{"2":{"400":1,"682":1,"694":1}}],["mint",{"2":{"278":1}}],["minority",{"2":{"494":1}}],["minor",{"2":{"278":1}}],["minns",{"2":{"155":2,"161":2}}],["minnamespace",{"2":{"155":4,"161":4}}],["min",{"2":{"155":4,"500":2,"596":1,"599":4,"629":4,"727":3}}],["minimized",{"2":{"415":1}}],["minimal",{"2":{"127":1,"178":1,"278":1,"284":1,"289":2,"307":1,"559":1,"667":1}}],["minimum",{"2":{"24":1,"84":1,"91":4,"126":1,"155":2,"228":1,"303":2,"384":1,"406":1,"479":2,"588":1,"614":2,"656":2,"718":1}}],["minutes",{"2":{"43":1,"107":1,"378":1,"380":1,"601":1}}],["minute",{"0":{"582":1},"2":{"41":1,"48":1,"388":1,"509":1}}],["mind",{"2":{"41":1,"237":2,"592":1,"727":1}}],["mindful",{"2":{"41":1}}],["misbehaviour",{"2":{"629":2}}],["mismatch",{"2":{"418":1}}],["misuse",{"2":{"89":1}}],["mission",{"2":{"19":1}}],["mistakes",{"2":{"8":1}}],["mounted",{"2":{"244":1}}],["mounting",{"0":{"244":1},"2":{"244":1,"578":1}}],["mount",{"2":{"243":1,"244":1,"357":1,"578":2,"579":1,"583":1}}],["moment",{"2":{"178":1,"181":1,"263":1,"548":1}}],["moved",{"2":{"422":1,"650":1}}],["move",{"2":{"69":2,"71":1,"87":1,"310":1}}],["moniker",{"2":{"727":3}}],["moniker=$moniker",{"2":{"500":2,"727":2}}],["moniker=validator",{"2":{"500":2}}],["monitors",{"2":{"510":1}}],["monitor",{"0":{"510":1},"2":{"395":1,"396":1,"503":4,"510":1,"548":1}}],["monitoring",{"0":{"395":1},"2":{"24":1,"27":1,"304":1,"501":1,"505":1,"585":1}}],["monolithic",{"0":{"422":1},"2":{"59":1,"213":1,"229":2,"277":1,"290":1,"422":3}}],["monthly",{"2":{"42":1,"54":1}}],["month",{"2":{"24":1,"26":1}}],["months",{"2":{"22":1,"23":6,"24":1,"26":1,"27":1,"166":1}}],["mocha4",{"2":{"688":1,"689":1,"691":1}}],["mocharesturl",{"2":{"398":2}}],["mocharpcurl",{"2":{"398":2}}],["mochachainid",{"2":{"315":3,"357":4,"398":2,"551":1,"592":8,"602":2,"721":2,"727":2}}],["mochacelestia",{"2":{"89":1,"308":1,"309":2,"310":1,"484":1,"485":1,"486":1,"493":1,"546":1,"553":1,"555":1,"619":1,"621":1,"659":1,"660":1,"661":1,"663":1}}],["mochadocker",{"2":{"243":1}}],["mochaversions",{"2":{"243":2,"307":2,"539":4,"559":2,"577":2,"579":2,"580":2}}],["mocha",{"0":{"89":1,"453":1,"526":1,"529":1,"546":1,"626":1,"685":1,"693":1,"703":1,"704":1},"1":{"527":1,"528":1,"529":1,"530":1,"531":1,"686":1,"687":1,"688":1,"689":1,"690":1,"691":1,"692":1,"693":1,"694":1,"695":1},"2":{"24":2,"27":1,"79":1,"80":2,"89":7,"95":1,"107":1,"210":3,"213":2,"237":4,"243":4,"244":8,"307":1,"308":1,"309":5,"310":3,"317":2,"357":1,"360":1,"398":6,"400":5,"453":3,"465":2,"466":2,"484":2,"485":6,"486":1,"493":5,"511":1,"517":1,"525":1,"526":2,"529":7,"531":4,"533":1,"537":1,"546":4,"551":2,"553":7,"555":1,"577":1,"619":1,"620":3,"621":1,"625":1,"626":5,"627":1,"629":9,"630":2,"632":1,"633":6,"634":6,"635":3,"659":1,"660":5,"661":3,"662":1,"663":1,"685":7,"688":6,"689":9,"690":9,"691":14,"692":16,"693":2,"694":2,"703":4,"709":2,"724":1,"727":1}}],["moduleencodingregisters",{"2":{"392":2}}],["module",{"2":{"312":5,"315":2,"331":2,"387":1,"398":2,"494":4,"509":1,"637":1,"638":1}}],["modules",{"2":{"265":1,"278":2,"385":1}}],["modularizes",{"2":{"289":1}}],["modularity",{"0":{"59":1},"2":{"48":1,"59":1}}],["modular",{"0":{"35":1,"48":1,"49":1,"57":1,"62":1,"200":1,"229":1,"230":1,"256":1,"422":1},"1":{"36":1,"37":1,"38":1,"39":1,"40":1,"41":1,"42":1,"43":1,"44":1,"45":1,"46":1,"47":1,"48":1,"50":1,"51":1,"52":1,"53":1,"54":1,"55":1,"56":1,"58":1,"59":1,"60":1,"61":1,"62":1,"63":1,"201":1,"202":1,"203":1,"204":1,"205":1,"206":1,"207":1,"208":1,"209":1,"210":1,"211":1,"231":1,"232":1,"233":1,"257":1,"258":1,"259":1,"260":1,"261":1,"262":1,"263":1,"264":1,"265":1,"266":1,"267":1,"268":1,"269":1,"270":1,"271":1},"2":{"19":1,"35":2,"39":1,"40":1,"41":2,"42":3,"43":4,"45":1,"46":1,"47":2,"48":2,"49":2,"50":2,"53":3,"54":1,"56":2,"57":2,"59":4,"62":3,"63":1,"65":9,"206":1,"223":2,"229":4,"231":2,"232":2,"233":1,"256":3,"268":2,"277":2,"284":1,"289":1,"290":1,"374":2,"402":1,"422":3,"423":2,"457":1,"458":1,"682":1}}],["modifiable",{"2":{"494":1}}],["modifications",{"2":{"278":1}}],["modifier",{"2":{"260":4}}],["modified",{"2":{"213":1,"283":1,"290":1,"354":2,"419":1,"421":1,"478":1}}],["modifying",{"2":{"651":1}}],["modify",{"0":{"651":1},"2":{"264":1,"312":1,"360":1,"557":1,"651":1}}],["mod",{"2":{"149":3,"371":3,"515":2,"627":1}}],["mode=",{"2":{"507":2}}],["model",{"2":{"373":2,"386":1,"460":1}}],["modern",{"2":{"265":1,"691":1}}],["moderation",{"2":{"9":1}}],["mode",{"2":{"68":2,"268":2,"380":1,"458":1,"507":4,"531":4,"629":12}}],["mozilla",{"2":{"17":1}}],["more",{"0":{"140":1},"1":{"141":1,"142":1,"143":1},"2":{"5":1,"21":1,"25":1,"33":1,"40":1,"41":1,"42":1,"44":1,"71":1,"79":3,"84":3,"85":1,"90":1,"92":1,"105":1,"108":1,"110":1,"116":1,"117":1,"134":2,"135":2,"136":2,"146":2,"148":2,"161":1,"166":1,"167":1,"172":1,"173":2,"180":1,"188":1,"192":1,"194":1,"205":2,"207":1,"208":2,"211":1,"213":1,"214":1,"229":1,"232":3,"233":1,"237":1,"243":1,"248":1,"249":1,"256":1,"268":2,"279":1,"288":1,"289":1,"290":1,"297":1,"298":1,"301":1,"302":2,"307":1,"310":1,"319":1,"320":1,"373":2,"382":1,"386":1,"387":1,"391":1,"393":1,"402":1,"405":1,"406":1,"416":1,"417":3,"419":1,"421":3,"423":1,"425":1,"447":1,"460":1,"485":1,"489":1,"494":1,"505":1,"509":3,"518":1,"519":1,"529":1,"531":2,"532":1,"539":1,"540":2,"552":1,"557":1,"559":1,"560":1,"575":1,"601":1,"620":1,"660":1,"680":1,"681":1,"696":1,"721":2}}],["mostly",{"2":{"620":1}}],["most",{"2":{"0":1,"43":1,"48":5,"50":1,"88":1,"281":1,"404":1,"412":1,"565":1,"575":1,"600":1,"626":1,"655":1}}],["mesa",{"2":{"674":1,"675":1,"676":1,"679":1,"688":1,"689":1,"691":1}}],["messages",{"2":{"36":1,"38":1,"188":1,"278":1,"365":1,"373":2,"494":2,"522":2}}],["message",{"2":{"2":1,"47":3,"107":1,"110":1,"319":1,"373":5,"375":1,"386":1,"424":2,"532":1,"537":1}}],["me",{"2":{"266":2,"267":8,"268":2,"608":2,"652":2,"689":1,"690":1,"691":1}}],["mechanisms",{"2":{"85":1,"110":1,"139":1,"385":1,"386":1}}],["mechanism",{"0":{"110":1,"204":1,"350":1},"1":{"351":1,"352":1},"2":{"85":1,"106":1,"110":4,"162":1,"206":1,"255":2,"350":1,"351":3,"352":1,"404":1,"405":1,"415":1,"442":1,"444":1,"451":1,"505":1,"509":1,"600":1}}],["measured",{"2":{"84":1}}],["meant",{"2":{"297":1}}],["mean",{"2":{"135":2}}],["meaningful",{"2":{"283":1}}],["meaning",{"2":{"30":1,"166":2,"283":1,"478":1,"602":1}}],["means",{"2":{"22":1,"148":1,"232":2,"260":2,"278":1,"307":1,"308":1,"384":1,"388":1,"417":2,"422":2,"426":1,"439":1,"441":1,"574":1,"575":1,"607":2}}],["metric",{"2":{"544":1}}],["metricscurl",{"2":{"502":1}}],["metrics",{"0":{"501":1,"543":1,"544":1,"548":1},"1":{"502":1,"503":1,"504":1,"505":1,"544":1,"545":2,"546":2,"547":2,"548":1},"2":{"33":1,"501":3,"502":8,"503":3,"504":2,"543":2,"544":12,"545":6,"546":6,"547":15,"548":2,"628":1}}],["met",{"2":{"384":1}}],["meta",{"2":{"268":4}}],["metadata",{"2":{"159":2,"232":1,"425":1,"655":1}}],["metamaskwallet",{"2":{"267":4}}],["metamask",{"0":{"269":1},"2":{"80":1,"258":1,"269":2,"270":3}}],["methods",{"2":{"312":1,"393":1,"442":1}}],["method",{"2":{"129":1,"131":2,"156":1,"172":1,"207":1,"213":1,"282":1,"290":1,"312":5,"315":2,"331":3,"332":1,"334":2,"398":2,"442":3}}],["menu",{"2":{"68":1,"331":2,"358":1,"400":1,"529":1,"539":1,"540":1,"560":1}}],["mentioned",{"2":{"102":1,"688":1}}],["mention",{"2":{"44":2}}],["memo",{"2":{"522":2,"629":4}}],["memory",{"2":{"136":1,"161":2,"260":26,"303":1,"479":1,"504":1,"588":1,"614":1,"656":1,"718":1}}],["memorable",{"2":{"65":1,"84":1}}],["mempool",{"0":{"383":1},"1":{"384":1,"385":1,"386":1,"387":1,"388":1},"2":{"255":1,"383":2,"384":2,"388":2,"392":2,"439":1}}],["member",{"2":{"47":1}}],["members",{"0":{"48":1},"2":{"7":1,"23":1,"36":1,"45":3,"47":2,"48":2,"204":1,"388":1,"449":1,"495":1}}],["mev",{"2":{"48":1}}],["merge",{"2":{"554":1}}],["merkelized",{"2":{"438":1}}],["merkelizes",{"2":{"206":1}}],["merkleized",{"2":{"128":1,"148":1}}],["merkle",{"0":{"192":1,"193":1,"194":1,"419":1},"1":{"193":1,"194":1},"2":{"108":1,"128":4,"129":1,"131":1,"136":1,"148":3,"150":2,"151":4,"152":4,"153":2,"155":4,"158":4,"159":4,"161":8,"166":6,"172":2,"173":5,"186":1,"188":1,"194":1,"204":1,"206":2,"302":1,"415":1,"416":6,"417":1,"418":2,"419":2,"421":2,"424":4,"425":3}}],["merchandise",{"2":{"63":2}}],["merits",{"2":{"34":1}}],["merely",{"2":{"27":1}}],["meets",{"2":{"588":1}}],["meetups",{"2":{"38":1,"40":4,"42":5,"43":1,"44":1,"45":1,"46":4,"47":6,"48":2,"49":1,"57":3,"63":2,"65":1}}],["meetup",{"0":{"35":1,"36":1,"37":1,"38":1,"47":1,"48":1,"49":1,"57":1,"62":1},"1":{"36":1,"37":1,"38":1,"39":1,"40":1,"41":1,"42":1,"43":1,"44":1,"45":1,"46":1,"47":1,"48":1,"50":1,"51":1,"52":1,"53":1,"54":1,"55":1,"56":1,"58":1,"59":1,"60":1,"61":1,"62":1,"63":1},"2":{"35":2,"36":2,"38":1,"39":3,"40":4,"41":2,"42":7,"43":9,"44":5,"45":8,"46":4,"47":10,"48":5,"49":1,"50":4,"52":1,"53":2,"54":1,"56":4,"62":2,"63":1,"65":6}}],["meet",{"2":{"22":1}}],["media",{"2":{"3":1,"10":1,"14":1,"36":1,"38":1,"45":4,"46":1,"275":1}}],["mamaki",{"2":{"626":3}}],["mac",{"2":{"305":2,"576":1,"578":1,"586":2}}],["machines",{"2":{"422":1,"503":1,"607":1}}],["machine",{"2":{"2":1,"80":1,"114":1,"214":1,"242":1,"243":1,"246":1,"285":1,"308":1,"309":1,"367":2,"368":1,"378":1,"384":1,"386":1,"421":3,"422":1,"424":2,"478":1,"480":1,"484":1,"503":1,"517":2,"537":1,"541":1,"550":1,"552":1,"575":1,"577":1,"578":3,"589":1,"603":1,"615":1,"620":1,"637":1,"653":1,"657":1,"660":1,"719":1,"724":1}}],["magnitude",{"2":{"283":1}}],["map",{"2":{"268":8}}],["mappings",{"2":{"260":2}}],["mapping",{"2":{"108":1,"155":2,"158":1,"260":4}}],["marginright",{"2":{"268":4}}],["margintop",{"2":{"268":4}}],["marginbottom",{"2":{"268":2}}],["margin",{"2":{"267":2,"268":4}}],["marking",{"2":{"441":1}}],["markdownlint",{"2":{"326":2}}],["markdown",{"2":{"266":1}}],["markdownnpm",{"2":{"266":1}}],["marks",{"2":{"105":1,"666":1,"698":1}}],["marketing",{"0":{"45":1},"2":{"45":1}}],["market",{"0":{"383":1,"439":1},"1":{"384":1,"385":1,"386":1,"387":1,"388":1},"2":{"34":3}}],["mark",{"2":{"27":1}}],["mass",{"2":{"256":1}}],["master",{"2":{"159":2,"400":1,"592":10}}],["majority",{"2":{"418":1}}],["major",{"2":{"173":1,"421":1}}],["malicious",{"2":{"139":1,"418":1,"509":1}}],["math",{"2":{"151":2,"161":2,"372":2}}],["matrix",{"0":{"94":1},"2":{"89":1,"416":4,"418":1,"424":5,"425":1}}],["matches",{"2":{"89":1,"244":1,"351":1}}],["match",{"2":{"77":1,"134":1,"149":1,"273":1,"351":1}}],["matters",{"2":{"34":1,"172":1}}],["materials",{"0":{"56":1},"2":{"0":1,"3":1,"36":2,"37":1,"57":1}}],["made",{"2":{"34":6,"79":1,"109":1,"206":1,"233":1,"290":1,"292":1,"370":1,"406":1,"421":1,"424":2,"425":1,"438":2,"442":1,"517":1,"525":1}}],["manual",{"2":{"442":1}}],["manually",{"2":{"265":1,"269":1,"388":1,"392":2,"395":1,"442":1,"568":1,"571":1,"592":2}}],["many",{"0":{"557":1},"2":{"45":1,"134":1,"143":1,"232":1,"275":1,"279":1,"304":1,"385":1,"387":1,"422":1,"445":1,"448":1,"466":1,"503":1,"557":2,"585":1,"696":1}}],["manager",{"2":{"503":1}}],["management",{"0":{"279":1,"490":1,"493":1},"1":{"491":1},"2":{"279":1,"351":1,"575":1}}],["managed",{"2":{"166":1,"441":1}}],["manage",{"0":{"432":1,"436":1},"2":{"31":1,"36":1,"42":1,"235":1,"392":2,"488":2,"611":1}}],["manner",{"2":{"25":1,"31":1,"34":1,"583":1}}],["maxvalidators",{"2":{"494":4}}],["maxwidth",{"2":{"268":4}}],["maximize",{"2":{"233":1}}],["maximum",{"0":{"671":1},"2":{"155":2,"386":1,"502":2,"671":8}}],["maxns",{"2":{"155":2,"161":2}}],["maxnamespace",{"2":{"155":4,"161":4}}],["max",{"2":{"27":1,"155":4,"207":1,"494":3,"500":4,"502":2,"629":20,"670":1,"727":4}}],["may",{"0":{"413":1},"2":{"11":1,"13":1,"14":1,"15":1,"23":1,"24":1,"30":3,"31":6,"32":1,"33":4,"34":11,"40":2,"44":2,"45":3,"76":1,"83":1,"85":1,"89":1,"162":1,"272":1,"385":1,"394":1,"395":1,"413":1,"425":2,"441":1,"442":1,"448":1,"450":1,"458":1,"530":1,"541":1,"552":1,"557":1,"578":1,"601":1,"610":1,"667":1,"698":1}}],["mail",{"2":{"10":1,"34":1}}],["maintenance",{"2":{"449":1}}],["maintained",{"2":{"207":1}}],["maintaining",{"2":{"52":1,"60":1,"84":1,"388":1,"626":1}}],["maintains",{"2":{"23":1}}],["maintain",{"2":{"19":1,"23":1,"24":1,"34":2,"38":1,"42":1,"43":1,"45":1,"626":1}}],["mainfunction",{"2":{"374":1}}],["mainnetresturl",{"2":{"398":2}}],["mainnetrpcurl",{"2":{"398":2}}],["mainnetchainid",{"2":{"398":2,"551":2,"592":8,"602":2}}],["mainnetversions",{"2":{"307":2,"539":4,"559":2,"577":2,"579":2,"580":2}}],["mainnet",{"0":{"452":1,"545":1,"666":1,"698":1,"699":1},"1":{"667":1,"668":1,"669":1,"670":1,"671":1,"672":1,"673":1,"674":1,"675":1,"676":1,"677":1,"678":1,"679":1,"680":1,"681":1,"682":1,"683":1,"684":1,"699":1},"2":{"24":3,"25":1,"27":4,"30":2,"110":1,"207":2,"210":4,"237":1,"243":1,"244":1,"255":1,"269":1,"305":1,"307":1,"308":1,"309":2,"310":2,"334":1,"357":2,"360":1,"398":5,"452":1,"472":2,"484":1,"485":1,"486":1,"539":1,"545":1,"550":1,"551":3,"559":1,"577":3,"579":1,"580":1,"586":1,"592":4,"602":1,"619":1,"620":1,"621":1,"625":1,"659":1,"660":1,"661":1,"663":1,"666":3,"667":2,"672":2,"674":1,"675":1,"676":1,"679":1,"682":2,"683":1,"693":2,"698":2}}],["main",{"2":{"2":1,"77":1,"92":1,"150":4,"151":4,"161":4,"267":2,"273":1,"312":1,"346":1,"372":6,"373":3,"374":4,"421":1,"508":1}}],["making",{"2":{"2":1,"47":1,"48":1,"91":1,"206":1,"214":1,"313":1,"422":1,"423":1,"607":1,"608":1,"609":1,"610":1,"611":1}}],["makeconfig",{"2":{"392":2}}],["makecd",{"2":{"342":1,"359":1}}],["makes",{"0":{"406":1},"2":{"34":1,"185":1,"213":1,"383":1}}],["make",{"2":{"0":1,"2":1,"7":1,"31":1,"50":1,"88":1,"89":1,"107":1,"121":1,"125":1,"126":1,"130":1,"149":3,"151":2,"155":5,"157":2,"158":4,"159":2,"160":2,"161":12,"166":1,"181":1,"206":1,"231":1,"232":2,"236":5,"237":1,"253":1,"261":1,"268":2,"304":4,"342":1,"359":1,"360":1,"367":1,"369":1,"416":1,"424":1,"442":1,"501":1,"502":2,"511":1,"522":1,"533":1,"541":3,"552":3,"564":1,"585":4,"607":1,"696":1,"712":1}}],["ccv",{"2":{"629":4}}],["cycle",{"2":{"438":1}}],["cmd",{"2":{"360":2}}],["cn",{"2":{"341":3}}],["cgo",{"2":{"307":1,"559":1}}],["cte",{"2":{"267":2,"269":1}}],["ctx",{"2":{"150":2,"151":10,"152":2,"159":4,"160":6,"161":4,"373":6,"374":4}}],["css",{"2":{"267":4,"398":2}}],["cfg",{"2":{"88":1,"89":2,"102":1}}],["cpu",{"2":{"285":1,"303":1,"479":1,"588":1,"614":1,"656":1,"718":1}}],["cp",{"2":{"77":2,"265":1,"273":2,"371":4}}],["cd",{"2":{"75":4,"115":2,"236":2,"247":2,"259":2,"272":2,"305":4,"307":3,"341":2,"342":2,"349":2,"359":2,"371":2,"539":2,"559":3,"586":4,"602":3}}],["cip",{"2":{"441":1,"670":2}}],["city",{"2":{"42":1,"44":1,"45":1}}],["circulation",{"2":{"450":1}}],["circulating",{"2":{"450":4}}],["circulated",{"2":{"34":1}}],["circuits",{"2":{"188":1,"207":1}}],["circuit",{"0":{"189":1},"2":{"167":1,"185":1,"186":1,"189":3,"191":1,"192":4,"194":1,"208":1}}],["circumvention",{"2":{"31":1}}],["circumvent",{"2":{"31":1}}],["cherry",{"2":{"505":1}}],["checking",{"0":{"425":1},"2":{"277":1,"419":1}}],["checks",{"2":{"127":2,"131":3}}],["checkout",{"2":{"75":2,"307":6,"341":4,"539":6,"559":6}}],["check",{"0":{"322":1,"323":1,"516":1},"2":{"37":1,"71":1,"110":1,"127":1,"130":1,"131":1,"135":1,"139":2,"148":1,"159":1,"161":5,"172":2,"192":1,"196":1,"204":1,"228":1,"243":1,"244":1,"254":1,"298":1,"307":2,"309":1,"313":1,"335":1,"341":1,"347":1,"380":1,"392":4,"405":1,"415":1,"419":1,"502":3,"516":1,"522":1,"523":1,"524":1,"525":1,"536":1,"537":1,"539":2,"547":1,"548":1,"554":1,"559":2,"627":1,"631":4,"665":1,"679":1,"712":3,"716":2,"721":1}}],["checklist",{"2":{"35":1}}],["chown",{"2":{"578":2}}],["chose",{"2":{"466":1}}],["chosen",{"2":{"32":1,"43":1,"85":1,"351":1,"610":2}}],["choices",{"2":{"373":2}}],["choice",{"2":{"360":1,"430":1,"435":1,"727":2}}],["choosing",{"2":{"252":1,"646":1}}],["chooses",{"2":{"416":1}}],["choose",{"0":{"71":1,"225":1},"2":{"69":1,"71":1,"84":1,"357":1,"435":1,"442":1,"503":1,"522":1,"530":1,"540":2,"547":1,"553":1,"560":2,"610":1,"645":1}}],["chart",{"2":{"449":1}}],["charge",{"2":{"444":1}}],["charged",{"2":{"386":1}}],["characters",{"2":{"27":1}}],["characteristics",{"2":{"7":1}}],["chatmessageroleuser",{"2":{"373":2}}],["chatcompletion",{"2":{"373":2}}],["chatcompletionmessage",{"2":{"373":2}}],["chatcompletionrequest",{"2":{"373":2}}],["chainode",{"2":{"674":1,"675":1,"676":1,"677":1}}],["chain=celestia",{"2":{"452":1}}],["chainname",{"2":{"398":12}}],["chainid",{"2":{"398":16,"634":8}}],["chainlist",{"2":{"84":1}}],["chain",{"0":{"68":1,"84":1,"85":1,"86":2,"90":1,"91":1,"134":1,"198":1,"231":1,"269":1,"399":1,"400":1,"551":1,"649":1},"2":{"68":2,"71":2,"77":2,"79":1,"82":1,"83":1,"84":26,"85":12,"86":7,"88":2,"89":3,"90":1,"91":56,"92":2,"105":1,"107":6,"108":1,"110":1,"129":1,"134":3,"135":1,"169":1,"170":1,"178":1,"196":2,"197":1,"201":1,"206":1,"208":1,"229":1,"237":1,"243":1,"264":2,"267":6,"268":1,"269":1,"270":1,"273":2,"278":1,"280":1,"282":1,"283":1,"310":1,"326":2,"350":1,"377":2,"379":2,"380":1,"398":2,"399":5,"400":4,"402":2,"403":1,"423":1,"440":1,"458":1,"471":1,"485":1,"493":6,"494":12,"496":4,"497":8,"498":4,"500":6,"505":1,"507":12,"522":6,"529":3,"530":1,"531":7,"542":2,"551":3,"592":6,"601":1,"620":1,"626":1,"627":3,"628":1,"629":4,"630":6,"633":12,"634":22,"637":1,"638":1,"645":4,"649":3,"660":1,"679":1,"680":1,"688":1,"721":2,"727":2}}],["chains=",{"2":{"267":2}}],["chains",{"0":{"632":1},"2":{"27":1,"68":1,"79":1,"80":2,"84":3,"91":1,"95":2,"101":1,"105":1,"139":1,"163":1,"207":1,"213":1,"267":8,"269":1,"283":1,"290":1,"381":1,"400":1,"444":1,"448":1,"452":1,"453":1,"627":1,"628":1,"629":30,"630":1,"631":2,"632":1,"635":4}}],["challengemanager",{"2":{"98":1,"99":1}}],["challenge",{"2":{"84":1,"85":1,"109":1}}],["challenges",{"2":{"48":1,"256":2,"375":1}}],["challenging",{"2":{"44":1}}],["changing",{"0":{"553":1},"2":{"233":1,"599":1}}],["changeable",{"2":{"447":2}}],["changed",{"2":{"77":1,"91":2,"117":2,"120":2,"249":2,"252":2,"273":1}}],["change",{"0":{"531":1},"2":{"75":1,"77":1,"264":1,"265":1,"272":1,"273":1,"278":1,"280":1,"283":1,"307":1,"351":1,"357":1,"385":1,"441":1,"447":1,"494":6,"500":2,"503":1,"514":1,"539":1,"553":1,"559":1,"569":1,"592":1,"727":2}}],["changes",{"2":{"2":4,"19":1,"43":1,"45":1,"94":2,"388":1,"395":1,"421":2,"466":2,"478":1,"494":4,"609":1,"610":1,"611":1,"620":1,"667":1,"701":1}}],["chan",{"2":{"160":2,"351":2}}],["channelid",{"2":{"634":4}}],["channel",{"2":{"40":1,"53":2,"54":1,"89":4,"367":1,"473":1,"476":1,"537":1,"629":4,"634":13,"635":4,"684":1,"693":1,"695":1,"705":1}}],["channelside",{"2":{"634":4}}],["channels",{"0":{"635":1},"2":{"14":1,"36":1,"46":2,"48":1,"310":1,"629":2,"632":1,"635":2}}],["chances",{"2":{"44":1}}],["chance",{"2":{"27":1,"601":1}}],["c",{"2":{"31":2,"32":1,"34":2,"304":2,"305":8,"540":2,"553":1,"560":2,"585":2,"586":8,"602":6,"623":1}}],["cumulo",{"2":{"674":1,"675":1,"676":1,"689":1,"690":1,"691":1}}],["cumulativegasused",{"2":{"121":2,"253":2}}],["culmination",{"2":{"666":1,"698":1}}],["custody",{"0":{"279":1}}],["custodians",{"2":{"276":1}}],["custodian",{"2":{"24":1}}],["custom=",{"2":{"542":4}}],["custom=robusta",{"2":{"316":2}}],["custom=private",{"2":{"316":2}}],["customtargetblock",{"2":{"207":3}}],["customizable",{"2":{"267":1}}],["customizability",{"0":{"233":1},"2":{"233":1}}],["customizations",{"2":{"233":1}}],["customization",{"2":{"233":1}}],["customizing",{"2":{"84":1}}],["customized",{"2":{"233":1}}],["customize",{"2":{"47":2,"84":1,"237":1}}],["custom",{"0":{"68":1,"316":1,"399":1,"400":1,"486":1,"542":1,"621":1,"663":1},"2":{"47":3,"68":1,"80":1,"82":1,"84":1,"89":1,"207":1,"233":1,"237":1,"269":1,"316":4,"397":1,"399":1,"400":3,"486":9,"542":3,"554":2,"577":1,"621":9,"663":9}}],["curl",{"0":{"318":1,"334":1},"2":{"304":6,"318":1,"334":1,"360":2,"540":2,"560":2,"585":6,"592":10,"602":6,"652":2}}],["currencies",{"2":{"398":2}}],["currency",{"2":{"269":1,"400":1,"458":1}}],["currentitem",{"2":{"260":4}}],["currentid",{"2":{"260":4}}],["current",{"2":{"43":1,"95":1,"206":1,"236":2,"256":1,"385":1,"411":1,"490":2,"671":1}}],["currently",{"2":{"23":1,"48":1,"148":1,"210":1,"225":1,"255":1,"280":1,"281":1,"285":1,"297":1,"380":2,"384":1,"389":1,"452":1,"453":1,"556":1,"627":1,"628":1,"700":1}}],["curated",{"2":{"43":1,"65":1}}],["cups",{"2":{"41":1}}],["ceid",{"2":{"373":2}}],["celopstia",{"2":{"275":1}}],["cel",{"0":{"235":1,"240":1,"241":1},"1":{"236":1,"237":1,"238":1,"239":1,"240":1,"242":1,"243":1,"244":1},"2":{"234":2,"235":1,"236":5,"237":7,"238":6,"239":6,"240":2,"243":3,"244":3,"307":3,"310":4,"367":2,"485":3,"553":3,"559":3,"575":1,"620":3,"661":5}}],["celesvaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u43cv6hd",{"2":{"721":1}}],["celesvaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u43cv6hdenter",{"2":{"721":1}}],["celesvaloper",{"2":{"721":2}}],["celes",{"2":{"244":4,"357":1,"661":6}}],["celestiavaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u4q4gx4p",{"2":{"721":2}}],["celestiavaloper",{"2":{"721":1}}],["celestiacommunity",{"2":{"630":1}}],["celestiascan",{"2":{"475":1,"694":1}}],["celestiatestnet",{"2":{"400":1}}],["celestia1xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxcelestia",{"2":{"537":1}}],["celestia1xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",{"2":{"537":2}}],["celestia1xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx$request",{"2":{"537":1}}],["celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j",{"2":{"520":2,"522":2,"524":2}}],["celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5",{"2":{"519":2,"520":2,"522":2,"523":2}}],["celestia17adsjkuecgjheugrdrwdqv9uh3qkrfmj9xzawx",{"2":{"495":2}}],["celestia10d07y265gmmuvt4z0w9aw880jnsr700jtgz4v7",{"2":{"494":2}}],["celestia10rtd9lhel2cuh6c659l25yncl6atcyt37umard",{"2":{"323":2}}],["celestia1czpgn3hdh9sodm06d5qk23xzgpq2uyc8ggdqgw",{"2":{"493":2}}],["celestia1krkle0n547u0znz3unnln8paft2dq4z3rznv86",{"2":{"337":2}}],["celestia1znk24rh52pgcd9z5x2x42jztjh6raaaphuvrt3",{"2":{"325":2}}],["celestia1",{"2":{"310":1,"367":2,"473":1,"662":1,"693":1}}],["celestia1wkhyhr7ngf0ayqlpnsnxg4d72hfs5453dvunm9",{"2":{"244":2}}],["celestialightnodeclient",{"2":{"135":6,"137":2,"139":2}}],["celestiaheight",{"2":{"134":4}}],["celestiaorg",{"2":{"87":2,"110":1,"149":6,"151":2,"155":2,"156":2,"159":2,"161":4,"213":1,"236":2,"243":2,"287":1,"307":2,"341":2,"346":1,"351":1,"371":2,"372":6,"387":2,"392":12,"395":1,"539":2,"559":2,"577":6,"579":6,"580":6,"592":12}}],["celestia",{"0":{"0":1,"6":1,"18":1,"49":1,"52":1,"58":1,"60":1,"92":1,"104":1,"142":1,"147":1,"151":1,"213":1,"224":1,"234":1,"276":1,"278":1,"287":1,"289":1,"290":1,"299":1,"306":1,"307":1,"308":1,"349":1,"354":1,"367":1,"382":1,"390":1,"391":1,"393":1,"397":1,"398":1,"404":1,"406":1,"412":1,"415":1,"424":1,"430":1,"441":1,"444":1,"451":1,"477":1,"482":1,"483":1,"509":1,"511":1,"534":1,"538":1,"543":1,"558":1,"587":1,"591":1,"612":1,"617":1,"624":1,"639":1,"642":1,"643":1,"654":1,"658":1,"696":1,"697":1,"707":1,"708":1,"712":1,"714":1,"715":1,"716":1,"717":1,"722":1,"723":1},"1":{"1":1,"2":1,"3":1,"4":1,"7":1,"8":1,"9":1,"10":1,"11":1,"12":1,"13":1,"14":1,"15":1,"16":1,"17":1,"19":1,"20":1,"21":1,"22":1,"23":1,"24":1,"25":1,"26":1,"27":1,"28":1,"29":1,"50":1,"51":1,"52":1,"53":1,"54":1,"55":1,"56":1,"105":1,"106":1,"107":1,"108":1,"109":1,"110":1,"111":1,"225":1,"226":1,"227":1,"235":1,"236":1,"237":1,"238":1,"239":1,"240":1,"241":1,"242":1,"243":1,"244":1,"277":1,"278":1,"279":2,"280":2,"281":2,"282":2,"283":2,"288":1,"289":1,"300":1,"301":1,"302":1,"303":1,"304":1,"305":1,"306":1,"307":2,"308":2,"309":2,"310":2,"311":1,"312":1,"313":1,"314":1,"315":1,"316":1,"317":1,"318":1,"319":1,"320":1,"321":1,"322":1,"323":1,"324":1,"325":1,"326":1,"327":1,"328":1,"329":1,"330":1,"331":1,"332":1,"333":1,"334":1,"335":1,"336":1,"337":1,"355":1,"356":1,"357":1,"358":1,"359":1,"360":1,"361":1,"362":1,"363":1,"364":1,"383":1,"384":1,"385":1,"386":1,"387":1,"388":1,"389":1,"390":1,"391":1,"392":1,"393":1,"398":1,"399":1,"400":1,"416":1,"417":1,"418":1,"419":1,"420":1,"421":1,"425":1,"452":1,"453":1,"478":1,"479":1,"480":1,"481":1,"482":1,"483":2,"484":2,"485":2,"486":2,"487":2,"512":1,"513":1,"514":1,"515":1,"516":1,"517":1,"518":1,"519":1,"520":1,"521":1,"522":1,"523":1,"524":1,"525":1,"526":1,"527":1,"528":1,"529":1,"530":1,"531":1,"532":1,"533":1,"535":1,"536":1,"537":1,"539":1,"540":1,"541":1,"544":1,"545":1,"546":1,"547":1,"548":1,"559":1,"560":1,"561":1,"562":1,"588":1,"589":1,"590":1,"591":1,"592":1,"593":1,"594":1,"595":1,"596":1,"597":1,"598":1,"599":1,"600":1,"601":1,"602":1,"603":1,"604":1,"605":1,"606":1,"607":1,"608":1,"609":1,"610":1,"611":1,"613":1,"614":1,"615":1,"616":1,"617":1,"618":2,"619":2,"620":2,"621":2,"622":2,"623":2,"625":1,"626":1,"640":1,"641":1,"642":1,"643":1,"644":2,"645":2,"646":2,"647":2,"648":2,"649":2,"650":2,"651":2,"652":2,"653":2,"655":1,"656":1,"657":1,"658":1,"659":1,"660":1,"661":1,"662":1,"663":1,"664":1,"665":1,"698":1,"699":1,"700":1,"701":1,"702":1,"703":1,"704":1,"705":1,"718":1,"719":1,"720":1,"721":1,"722":1,"723":2,"724":2,"725":2,"726":2,"727":1,"728":1,"729":1,"730":1},"2":{"0":2,"3":3,"5":2,"7":1,"8":1,"11":1,"19":3,"21":3,"22":1,"24":3,"25":2,"27":5,"30":8,"33":3,"40":2,"41":2,"42":1,"43":4,"44":2,"45":2,"46":6,"47":15,"48":5,"49":1,"50":1,"52":2,"54":1,"56":1,"57":2,"59":2,"60":4,"61":2,"62":1,"63":2,"64":1,"65":4,"79":3,"80":1,"83":5,"88":1,"89":12,"94":5,"101":1,"102":4,"105":2,"106":1,"107":12,"108":6,"109":4,"110":3,"111":1,"127":1,"128":1,"129":1,"130":2,"131":6,"133":4,"134":20,"135":11,"136":4,"137":3,"139":4,"142":2,"145":1,"146":4,"148":4,"149":3,"151":2,"153":2,"154":1,"155":4,"156":5,"159":8,"161":4,"162":2,"163":2,"165":1,"166":5,"169":1,"170":4,"172":3,"173":2,"176":2,"177":1,"178":1,"181":2,"182":1,"184":1,"185":2,"189":1,"192":1,"201":5,"203":2,"204":2,"205":1,"206":12,"207":1,"210":1,"212":2,"213":7,"214":1,"222":1,"223":1,"225":1,"228":1,"232":1,"233":1,"234":1,"235":1,"236":5,"237":2,"242":1,"243":11,"244":17,"255":5,"256":3,"257":1,"258":1,"259":4,"260":2,"262":2,"264":1,"275":1,"276":1,"277":4,"278":2,"279":1,"280":3,"283":2,"284":1,"285":1,"286":1,"289":4,"290":6,"291":1,"292":2,"293":1,"296":1,"299":1,"301":1,"302":1,"304":2,"305":1,"307":11,"308":2,"309":5,"310":7,"311":2,"312":3,"313":1,"314":4,"315":2,"316":13,"317":5,"319":2,"322":1,"326":1,"327":2,"332":4,"334":6,"335":1,"340":1,"341":2,"343":1,"346":1,"350":1,"351":6,"352":1,"354":1,"355":2,"356":2,"357":14,"360":18,"365":3,"366":1,"367":3,"368":3,"372":4,"373":7,"374":1,"375":3,"377":1,"380":6,"381":1,"382":1,"383":1,"384":2,"387":3,"388":2,"390":1,"391":1,"392":16,"393":1,"394":2,"395":1,"397":1,"398":22,"399":4,"400":5,"402":1,"404":1,"406":1,"407":2,"412":1,"415":3,"416":3,"417":4,"418":1,"419":2,"421":12,"423":4,"424":3,"425":7,"426":2,"430":1,"434":1,"438":3,"439":2,"440":2,"441":2,"442":5,"444":3,"445":1,"448":2,"449":6,"450":1,"451":1,"452":2,"453":3,"457":2,"458":1,"459":2,"460":2,"465":1,"466":1,"470":11,"471":5,"472":2,"473":4,"474":1,"475":1,"477":1,"478":11,"483":2,"484":3,"485":3,"486":3,"488":6,"490":8,"491":2,"492":3,"493":2,"494":4,"495":1,"497":2,"500":10,"501":2,"502":3,"503":5,"504":1,"506":1,"507":18,"509":2,"510":1,"511":1,"512":1,"514":3,"515":10,"516":4,"517":1,"519":4,"520":2,"522":6,"523":2,"524":2,"525":4,"526":1,"527":2,"534":2,"535":2,"537":4,"538":1,"539":10,"540":12,"541":1,"542":6,"543":2,"544":2,"545":3,"546":2,"547":4,"548":2,"550":3,"551":2,"552":2,"553":9,"554":2,"555":1,"556":10,"557":15,"559":11,"560":12,"561":1,"564":1,"565":3,"566":1,"569":1,"571":6,"575":6,"577":15,"579":22,"580":22,"583":1,"584":1,"586":1,"587":2,"588":1,"591":1,"592":19,"594":1,"598":1,"601":1,"602":42,"603":2,"605":2,"607":2,"608":2,"609":1,"610":2,"611":2,"612":3,"613":1,"617":1,"620":2,"621":3,"622":1,"624":1,"625":1,"626":4,"627":2,"628":1,"629":9,"630":1,"637":3,"638":2,"639":2,"642":1,"644":3,"645":2,"646":2,"647":2,"648":1,"649":3,"650":4,"651":1,"652":2,"653":2,"654":2,"655":2,"658":2,"660":4,"661":4,"662":3,"663":3,"666":3,"667":1,"670":1,"674":23,"675":22,"676":21,"677":1,"679":8,"681":2,"682":6,"683":4,"685":1,"688":4,"689":12,"690":8,"691":12,"692":8,"693":6,"694":3,"696":3,"698":1,"700":1,"701":1,"706":2,"707":3,"708":4,"709":4,"711":1,"712":16,"714":14,"715":18,"716":16,"717":2,"722":4,"723":1,"724":2,"727":4}}],["cele",{"2":{"243":2,"244":2}}],["celenium",{"2":{"91":1,"334":1,"475":1,"682":1,"694":1}}],["centric",{"2":{"205":1}}],["centralized",{"2":{"135":1,"203":2,"205":1,"268":2}}],["center",{"2":{"27":1,"268":4,"398":2}}],["certificate",{"2":{"34":1}}],["certainty",{"2":{"108":1,"109":1}}],["certain",{"0":{"34":1},"2":{"22":1,"30":2,"31":1,"33":1,"172":1,"413":1,"494":1}}],["cross",{"2":{"627":1}}],["crate",{"2":{"628":1}}],["crashed",{"2":{"505":1}}],["craft",{"2":{"275":1}}],["crisis",{"2":{"278":1}}],["critical",{"2":{"45":1,"402":1,"460":1,"501":1,"522":2,"722":1}}],["criteria",{"0":{"24":1,"25":1},"2":{"21":1,"22":1,"23":1,"24":3}}],["cryptography",{"2":{"281":1}}],["crypto",{"2":{"151":2,"161":2,"244":2,"288":1,"392":2,"519":2,"520":4,"524":2}}],["crucial",{"2":{"42":1,"44":1,"84":2,"86":2,"89":1,"667":1}}],["credibility",{"2":{"44":1}}],["credentials",{"2":{"31":1,"266":1,"392":2}}],["creation",{"2":{"288":1}}],["creating",{"0":{"137":1,"268":1,"489":1,"650":1},"2":{"40":1,"88":1,"91":2,"135":1,"136":2,"137":1,"162":1,"166":1,"267":1,"288":1,"373":3,"458":1,"517":2,"557":2,"632":1,"720":1}}],["creator",{"2":{"260":2}}],["creators",{"2":{"79":1}}],["createpayforblob",{"2":{"392":2}}],["createpost",{"2":{"260":10,"263":2,"264":2,"268":6}}],["createchatcompletion",{"2":{"373":2}}],["createclient",{"2":{"267":4,"373":4,"374":2}}],["createnamespaceid",{"2":{"373":3,"374":2}}],["createroot",{"2":{"267":2}}],["creates",{"2":{"260":2,"351":1}}],["created",{"2":{"65":1,"117":1,"150":1,"160":1,"173":1,"243":2,"244":2,"249":1,"259":1,"265":1,"266":1,"310":2,"506":1,"517":1,"523":1,"532":1,"537":1,"629":1,"634":1,"635":2,"648":1,"650":1,"661":1,"662":1,"712":1,"714":1,"715":1,"716":1}}],["create",{"0":{"116":1,"234":1,"248":1,"511":1,"519":1,"522":1,"527":1,"529":1,"534":1,"536":1,"632":1,"633":1,"646":1,"649":1},"1":{"235":1,"236":1,"237":1,"238":1,"239":1,"240":1,"241":1,"242":1,"243":1,"244":1,"512":1,"513":1,"514":1,"515":1,"516":1,"517":1,"518":1,"519":1,"520":1,"521":1,"522":1,"523":1,"524":1,"525":1,"526":1,"527":1,"528":1,"529":1,"530":1,"531":1,"532":1,"533":1,"535":1,"536":1,"537":1,"634":1,"635":1},"2":{"2":3,"34":2,"36":1,"44":2,"46":2,"47":2,"65":2,"88":1,"120":2,"135":1,"137":1,"148":1,"159":1,"161":2,"167":1,"172":1,"176":1,"181":1,"207":1,"251":2,"252":3,"259":1,"260":6,"265":3,"266":2,"267":2,"268":20,"310":1,"313":1,"335":1,"372":1,"373":3,"389":1,"392":4,"398":1,"485":1,"488":2,"495":1,"500":2,"503":2,"507":4,"511":1,"518":1,"519":1,"522":9,"525":1,"526":1,"529":3,"533":2,"536":1,"539":1,"578":2,"579":1,"620":1,"627":1,"632":1,"633":7,"634":6,"646":1,"649":1,"650":1,"661":1,"670":1,"712":1,"714":1,"715":1,"721":1,"727":3}}],["cake",{"2":{"505":1}}],["cadence",{"2":{"376":2,"377":1}}],["cautious",{"2":{"557":1}}],["caution",{"2":{"263":1}}],["cause",{"2":{"352":1,"409":1,"445":1,"671":1}}],["caused",{"2":{"31":2}}],["car",{"2":{"519":2}}],["carrying",{"2":{"488":2}}],["carry",{"2":{"229":1}}],["carries",{"2":{"146":1}}],["care",{"2":{"107":1}}],["carefully",{"2":{"30":1}}],["catching",{"2":{"712":1}}],["catch",{"2":{"54":1,"398":2}}],["categories",{"2":{"449":1}}],["category",{"0":{"286":1},"2":{"3":3,"285":1,"290":1,"358":1,"449":1,"450":2}}],["cater",{"2":{"48":1}}],["catering",{"0":{"41":1},"2":{"41":1,"44":1}}],["cafes",{"2":{"42":1}}],["capacities",{"2":{"278":1}}],["capacity",{"2":{"232":3,"417":1,"588":1}}],["capability",{"2":{"278":1,"627":1}}],["capabilities",{"2":{"222":1,"293":1,"557":1}}],["capable",{"2":{"31":1,"79":1}}],["capture",{"2":{"37":1}}],["cast",{"2":{"76":2,"121":3,"253":3,"263":3,"264":2,"272":2}}],["caste",{"2":{"7":1}}],["casual",{"2":{"42":2,"48":1}}],["cases",{"2":{"177":1,"394":1,"421":1}}],["case",{"2":{"34":1,"69":1,"82":1,"109":1,"160":2,"169":1,"173":1,"181":1,"183":1,"184":1,"186":1,"195":1,"213":1,"255":1,"290":1,"350":1,"352":1,"385":3,"414":1,"536":1,"569":1,"632":1,"721":1}}],["cancel",{"2":{"373":4,"374":4}}],["canceling",{"2":{"45":1}}],["canonical",{"2":{"108":1,"388":1}}],["cannot",{"2":{"25":1,"34":1,"46":1,"351":1,"403":2,"409":1,"422":1}}],["can",{"0":{"414":1},"2":{"8":1,"24":2,"29":1,"35":2,"36":1,"40":3,"41":5,"43":3,"44":5,"45":3,"46":5,"47":3,"48":2,"54":1,"60":1,"61":1,"65":3,"69":2,"72":1,"76":1,"77":2,"84":4,"88":1,"90":1,"95":1,"102":3,"107":6,"108":5,"109":4,"110":1,"116":1,"119":1,"121":1,"126":1,"131":1,"133":1,"134":5,"135":5,"137":3,"146":3,"148":4,"150":2,"152":3,"154":2,"155":4,"156":2,"157":1,"158":1,"159":6,"160":2,"161":2,"162":2,"166":1,"167":1,"170":4,"172":2,"173":5,"176":3,"177":4,"178":2,"180":2,"181":2,"184":2,"185":2,"186":1,"189":1,"192":1,"195":1,"196":2,"199":1,"201":1,"204":1,"207":6,"208":1,"210":1,"211":1,"213":2,"215":1,"217":1,"222":1,"231":4,"232":1,"233":4,"234":1,"235":1,"236":1,"237":2,"238":1,"243":1,"248":1,"253":1,"255":5,"260":7,"261":1,"262":1,"263":4,"264":5,"265":1,"267":1,"268":1,"269":1,"272":1,"273":2,"282":1,"290":1,"291":1,"292":1,"294":1,"297":2,"302":2,"304":1,"309":1,"310":5,"312":1,"313":2,"314":1,"315":2,"317":3,"319":1,"320":1,"329":1,"330":1,"334":1,"335":1,"336":1,"346":1,"351":2,"352":2,"353":2,"357":1,"358":1,"360":1,"367":3,"370":2,"373":4,"376":1,"381":1,"385":1,"387":3,"388":1,"389":1,"392":6,"395":1,"397":1,"398":2,"399":2,"400":1,"402":3,"403":1,"406":3,"409":2,"416":1,"417":2,"419":3,"422":2,"423":1,"424":1,"425":2,"426":1,"432":1,"436":1,"438":1,"440":1,"442":2,"445":2,"447":1,"451":1,"458":1,"459":1,"466":3,"470":2,"471":1,"473":1,"475":1,"478":1,"484":1,"485":3,"489":1,"494":10,"495":4,"496":1,"497":2,"498":1,"499":1,"500":2,"501":2,"502":2,"503":4,"504":1,"505":2,"506":1,"509":2,"516":1,"522":1,"525":2,"529":1,"530":1,"531":1,"532":2,"534":1,"536":2,"537":1,"539":1,"540":1,"542":1,"544":1,"548":1,"552":1,"554":1,"557":1,"560":1,"562":1,"566":1,"571":2,"572":1,"573":2,"575":2,"577":1,"578":3,"579":1,"583":1,"584":1,"592":4,"594":1,"599":1,"601":2,"603":1,"605":1,"606":1,"607":2,"610":1,"611":2,"616":1,"617":1,"620":5,"625":2,"626":2,"627":2,"630":1,"632":1,"633":1,"637":2,"643":1,"644":1,"648":1,"652":5,"653":1,"661":3,"662":1,"665":1,"670":1,"671":3,"672":1,"679":2,"680":1,"681":2,"682":1,"685":1,"690":1,"692":1,"693":1,"694":1,"696":3,"700":1,"703":1,"709":1,"715":1,"721":3,"722":1,"723":1,"724":1}}],["caldera",{"2":{"212":1,"213":1,"214":1,"216":1,"217":1,"219":1,"220":1,"252":2,"272":2}}],["calculation",{"0":{"386":1}}],["calculating",{"2":{"131":1,"172":1,"384":1,"387":1}}],["calculates",{"2":{"385":1,"386":1}}],["calculated",{"2":{"320":1,"384":1,"386":3,"439":1,"445":1,"671":1}}],["calculate",{"2":{"155":2,"158":2,"172":1,"386":1,"445":1}}],["callopts",{"2":{"151":2}}],["calling",{"2":{"117":2,"249":2}}],["called",{"2":{"116":1,"117":1,"134":1,"201":1,"235":1,"244":1,"248":1,"249":1,"251":1,"373":2,"382":1,"387":1,"403":1,"404":1,"425":1,"624":1}}],["calldatatxcandidate",{"2":{"351":1}}],["calldata",{"2":{"110":1,"127":8,"213":3,"255":2,"290":3,"351":9}}],["calls",{"2":{"54":1,"121":1,"208":1,"253":1,"690":1}}],["call",{"0":{"348":1},"2":{"5":2,"43":1,"53":1,"54":1,"76":4,"110":1,"121":4,"215":1,"253":4,"263":3,"264":3,"268":2,"272":4,"311":1,"324":1,"325":1,"348":1,"379":1,"381":1}}],["calendar",{"0":{"5":1},"2":{"5":3,"45":1,"54":1,"64":1,"71":1}}],["cleanmake",{"2":{"345":1,"363":1}}],["clean",{"0":{"363":1},"2":{"345":2,"363":1,"644":1}}],["clearing",{"0":{"555":1}}],["clears",{"2":{"394":1}}],["clear",{"2":{"2":1,"43":1,"629":4}}],["clock",{"2":{"629":4}}],["cloning",{"2":{"592":1}}],["cloned",{"2":{"87":1}}],["clone",{"0":{"87":1,"341":1},"2":{"1":1,"2":2,"75":3,"87":3,"236":2,"272":3,"307":3,"341":3,"349":3,"371":2,"539":3,"559":3,"592":2}}],["cloud",{"2":{"378":1,"503":1,"541":1,"547":1,"552":1,"682":1}}],["closely",{"2":{"214":1}}],["close",{"2":{"151":4,"159":4,"160":2,"161":2,"373":5,"374":4,"392":2}}],["closer",{"2":{"131":1}}],["cli",{"0":{"299":1,"311":1,"390":1,"391":1,"488":1},"1":{"300":1,"301":1,"302":1,"303":1,"304":1,"305":1,"306":1,"307":1,"308":1,"309":1,"310":1,"311":1,"312":2,"313":2,"314":2,"315":2,"316":2,"317":2,"318":2,"319":2,"320":2,"321":2,"322":2,"323":2,"324":2,"325":2,"326":2,"327":2,"328":2,"329":2,"330":2,"331":2,"332":2,"333":1,"334":1,"335":1,"336":1,"337":1,"489":1,"490":1,"491":1,"492":1,"493":1,"494":1,"495":1,"496":1,"497":1,"498":1,"499":1,"500":1},"2":{"121":1,"253":1,"264":1,"296":1,"302":1,"311":1,"312":2,"313":3,"314":1,"315":1,"331":1,"488":2,"531":2,"536":1,"539":1,"540":1,"542":1,"561":1,"628":2,"637":1,"661":1}}],["clientid",{"2":{"634":8}}],["client=",{"2":{"267":2}}],["clients",{"0":{"633":1,"634":1},"1":{"634":1,"635":1},"2":{"135":1,"283":1,"629":2,"632":2,"633":2,"634":2,"696":1,"709":1}}],["client",{"0":{"132":1,"531":1},"1":{"133":1,"134":1,"135":1,"136":1,"137":1,"138":1,"139":1,"140":1,"141":1,"142":1,"143":1},"2":{"89":1,"109":1,"135":2,"143":1,"149":1,"150":3,"151":4,"152":1,"155":20,"156":4,"157":4,"158":6,"159":8,"161":50,"201":2,"206":2,"209":2,"260":2,"266":2,"267":2,"268":8,"280":1,"336":2,"372":2,"373":41,"374":10,"387":1,"531":4,"601":1,"633":8,"634":8,"690":1,"707":1}}],["clicking",{"2":{"231":1,"399":2}}],["click",{"2":{"68":1,"69":3,"71":4,"72":1,"83":1,"85":1,"86":1,"219":1,"269":1,"399":1,"400":3,"435":1,"503":3}}],["clubs",{"2":{"40":2}}],["classname=",{"2":{"398":4}}],["classes",{"2":{"16":1}}],["clang",{"2":{"304":4,"585":4}}],["claim",{"0":{"72":1,"496":1},"2":{"30":1,"31":5,"71":1,"72":3,"432":1,"436":1,"496":1}}],["claiming",{"2":{"30":1,"31":1}}],["clarifying",{"2":{"9":1}}],["clarity",{"2":{"2":1,"13":1}}],["copied",{"2":{"602":1}}],["copy",{"0":{"371":1},"2":{"89":1,"133":1,"265":2,"307":1,"347":1,"371":1,"409":1,"539":1,"559":1}}],["coin",{"2":{"399":1,"400":1,"405":1}}],["coingeckoid",{"2":{"398":6}}],["coindecimals",{"2":{"398":6}}],["coindenom",{"2":{"398":6}}],["coinminimaldenom",{"2":{"398":6}}],["cointype",{"2":{"398":2}}],["cosmwasm",{"2":{"381":1}}],["cosmosnetwork",{"2":{"630":1}}],["cosmossdk",{"2":{"372":2,"629":4}}],["cosmosmath",{"2":{"372":2,"373":2,"374":2}}],["cosmostation",{"0":{"400":1},"2":{"367":1,"397":1,"400":2}}],["cosmos",{"2":{"149":7,"233":1,"235":1,"244":2,"278":4,"279":2,"280":1,"282":1,"283":1,"380":1,"381":1,"382":1,"385":1,"392":4,"399":1,"400":1,"421":1,"426":1,"444":3,"448":1,"452":1,"459":3,"469":1,"478":1,"488":2,"489":1,"494":5,"503":3,"506":1,"508":2,"509":1,"519":2,"520":4,"522":2,"523":2,"524":4,"532":1,"627":1,"628":1,"629":9,"630":1,"672":1,"690":1,"691":1,"712":2,"714":2,"715":2}}],["costly",{"2":{"167":1}}],["cost",{"2":{"40":1,"82":1,"181":1,"183":1,"231":1,"284":1,"385":6,"386":5,"423":1}}],["costs",{"2":{"40":2,"41":1,"44":3,"46":1,"168":1,"191":1,"205":1,"385":3,"387":1}}],["coral",{"2":{"519":2}}],["corner",{"2":{"399":1,"400":1}}],["cors",{"2":{"268":2}}],["corresponds",{"2":{"192":1,"317":1}}],["corresponding",{"2":{"146":1,"148":2,"150":1,"151":2,"159":3,"161":2,"165":1,"166":3,"170":1,"416":1,"425":2,"512":1,"627":1}}],["correctly",{"2":{"117":2,"185":1,"189":1,"192":1,"249":2,"305":1,"351":1,"422":1,"586":1,"628":1,"631":1,"712":1,"716":1}}],["correct",{"2":{"108":1,"185":1,"237":1,"305":1,"373":1,"416":1,"486":1,"539":1,"541":2,"552":2,"586":1,"621":1,"639":1,"663":1,"685":1}}],["correction",{"0":{"13":1}}],["corrective",{"2":{"9":1}}],["cores",{"2":{"479":1,"614":1,"718":1}}],["core",{"0":{"309":1,"566":1},"2":{"59":1,"89":3,"107":1,"149":3,"229":1,"243":2,"244":2,"280":1,"285":1,"301":1,"303":1,"309":15,"310":6,"334":2,"360":3,"367":7,"395":1,"421":4,"449":3,"450":1,"458":1,"470":1,"471":9,"478":8,"484":10,"485":6,"486":6,"544":2,"545":2,"546":2,"547":4,"550":2,"553":2,"554":2,"566":2,"571":1,"577":6,"580":6,"588":1,"620":3,"621":6,"639":1,"656":1,"660":7,"661":6,"663":6,"679":7,"685":1,"688":1,"716":2,"724":2}}],["cooperation",{"2":{"52":1}}],["coordinating",{"2":{"624":1}}],["coordination",{"2":{"36":1,"288":1,"466":1,"624":1,"703":1}}],["coordinated",{"2":{"667":1}}],["coordinates",{"2":{"416":2}}],["coordinate",{"2":{"36":1,"46":2}}],["coworking",{"2":{"44":1}}],["coding",{"2":{"43":1,"407":2,"421":1,"438":1}}],["codelab",{"2":{"503":2}}],["coded",{"2":{"406":1,"407":1,"478":1,"671":1}}],["codespace",{"2":{"392":4,"522":2,"721":2,"727":2}}],["codebase",{"2":{"107":1,"701":1}}],["code",{"0":{"6":1,"52":1},"1":{"7":1,"8":1,"9":1,"10":1,"11":1,"12":1,"13":1,"14":1,"15":1,"16":1,"17":1},"2":{"9":2,"10":1,"12":1,"14":1,"15":1,"17":4,"25":1,"52":4,"64":1,"107":2,"108":1,"117":1,"133":1,"207":1,"213":1,"249":1,"260":2,"262":1,"268":1,"337":2,"365":1,"392":18,"398":1,"478":1,"503":1,"522":2,"625":1,"721":1,"727":1}}],["covers",{"2":{"59":1,"60":1,"61":1,"62":1,"66":1,"79":1,"207":1,"397":1,"587":1,"708":2}}],["cover",{"2":{"44":1,"46":1,"47":1,"66":1,"82":1,"209":1,"223":1,"256":1,"299":1,"312":1,"526":1,"542":1}}],["covering",{"2":{"44":1,"47":2}}],["covered",{"2":{"42":1,"574":1,"707":1,"709":1}}],["covenant",{"2":{"17":4}}],["co",{"2":{"40":2,"44":2,"45":1,"47":1,"56":1}}],["course",{"2":{"232":1}}],["counterparty",{"2":{"629":2}}],["counterforge",{"2":{"120":1,"251":1,"252":1}}],["countertest",{"2":{"117":4,"249":4}}],["counter",{"2":{"115":6,"116":7,"117":31,"120":3,"247":6,"248":7,"249":31,"251":3,"252":3}}],["count",{"2":{"93":1}}],["country",{"2":{"24":1,"34":1}}],["couldn",{"2":{"46":2,"151":2,"159":2}}],["could",{"2":{"8":1,"133":1,"134":1,"136":1,"139":3,"152":1,"172":1,"173":1,"233":1,"275":1,"280":1,"312":1,"385":1,"386":1,"407":2,"418":1,"425":1,"442":3,"505":1}}],["columns",{"2":{"131":1,"148":1,"172":1,"416":1}}],["column",{"2":{"108":1,"131":3,"146":1,"148":2,"172":1,"268":4,"326":2,"418":1,"424":3}}],["collide",{"2":{"503":1}}],["collaboration",{"2":{"44":1,"50":1,"288":1}}],["collaborate",{"2":{"40":1}}],["collects",{"2":{"380":1}}],["collectively",{"2":{"417":1}}],["collective",{"2":{"288":2}}],["collection",{"2":{"30":2,"49":1,"56":1,"134":1}}],["collect",{"2":{"33":5,"37":1,"488":4,"650":2}}],["collector",{"2":{"24":1,"91":4,"502":2,"547":2,"548":3}}],["color",{"2":{"7":1,"58":1}}],["cohorts",{"2":{"22":1,"23":3,"26":1}}],["cohort",{"0":{"22":1,"28":1},"2":{"22":3,"23":12,"24":1,"25":1,"26":1,"28":1}}],["conjured",{"2":{"503":1}}],["conn",{"2":{"392":6}}],["connects",{"2":{"548":1}}],["connectbutton",{"2":{"268":4}}],["connectors",{"2":{"267":4}}],["connectorsforwallets",{"2":{"267":4}}],["connectivity",{"2":{"425":1}}],["connecting",{"0":{"594":1},"1":{"595":1,"596":1},"2":{"135":2,"258":1,"679":1,"709":1,"727":1}}],["connectionid",{"2":{"634":8}}],["connectionside",{"2":{"634":4}}],["connections",{"0":{"547":1},"2":{"40":1,"42":1,"43":1,"65":1,"88":1,"502":6,"541":3,"552":3,"606":1,"627":2,"629":2,"632":2,"635":1,"651":2,"715":1}}],["connection",{"0":{"632":1,"634":1},"2":{"31":1,"32":1,"34":3,"44":1,"88":5,"103":1,"107":1,"135":6,"309":1,"392":2,"478":1,"485":1,"547":5,"620":1,"632":1,"634":30,"660":2}}],["connect",{"0":{"309":1},"2":{"38":1,"42":1,"43":1,"55":1,"83":1,"107":1,"151":2,"159":2,"161":2,"215":1,"217":1,"265":1,"267":3,"268":4,"334":1,"381":1,"398":2,"425":1,"452":1,"453":1,"469":2,"477":1,"478":1,"592":2,"612":1,"672":2,"685":4,"696":1,"724":1,"727":2}}],["connected",{"2":{"24":1,"270":1,"399":1,"400":1,"406":2,"421":1,"594":1,"703":1}}],["congested",{"2":{"255":1}}],["congests",{"2":{"232":1}}],["congestion",{"2":{"232":1,"394":1,"425":1}}],["congratulations",{"0":{"92":1},"2":{"122":1,"254":1,"517":1,"525":1,"533":1,"577":1,"580":1,"634":1}}],["concerned",{"2":{"408":1}}],["concepts",{"0":{"164":1},"1":{"165":1,"166":1,"167":1,"168":1,"169":1,"170":1,"171":1,"172":1,"173":1,"174":1,"175":1}}],["concept",{"2":{"59":1,"441":1}}],["conclusion",{"0":{"162":1,"195":1,"533":1}}],["concise",{"2":{"2":1}}],["conf",{"2":{"102":2}}],["confidence",{"2":{"109":1,"405":2}}],["configs",{"2":{"102":2,"503":4}}],["configurable",{"2":{"110":1}}],["configurations",{"0":{"593":1},"1":{"594":1,"595":1,"596":1,"597":1,"598":1,"599":1},"2":{"86":1,"237":1,"609":1,"629":1,"631":1}}],["configuration",{"0":{"86":1,"87":1,"610":1,"611":1,"629":1,"631":1,"729":1},"2":{"84":4,"85":1,"86":3,"266":1,"267":4,"268":1,"488":4,"502":2,"503":3,"531":3,"536":1,"542":1,"554":2,"598":1,"608":1,"610":2,"611":1,"629":1,"631":2,"729":1}}],["configuring",{"0":{"266":1,"267":1,"606":1},"1":{"607":1,"608":1,"609":1},"2":{"84":2,"86":1,"575":1,"628":1}}],["configures",{"2":{"489":1}}],["configurechains",{"2":{"267":4}}],["configured",{"2":{"89":1,"270":1,"353":1,"384":1,"630":1}}],["configure",{"0":{"84":1,"635":1},"2":{"83":1,"84":2,"89":1,"91":3,"266":1,"268":2,"269":1,"606":1,"610":1,"611":1,"632":1,"729":1}}],["config",{"0":{"68":1,"554":1,"563":1,"565":1,"651":1},"1":{"564":1,"565":1,"566":2,"567":2,"568":2,"569":2,"570":2,"571":2},"2":{"68":1,"77":1,"86":4,"87":1,"88":1,"89":3,"102":5,"273":1,"304":4,"345":1,"351":4,"357":6,"392":2,"488":2,"489":2,"502":1,"503":8,"527":2,"531":2,"536":2,"550":2,"554":7,"556":1,"565":4,"569":1,"571":1,"585":4,"592":27,"594":1,"597":1,"601":2,"607":5,"608":4,"610":2,"611":2,"628":1,"629":3,"631":5,"635":1,"648":1,"649":1,"650":4,"651":2,"659":1,"729":1}}],["confirmed",{"2":{"108":1,"109":1,"392":2,"432":1,"436":1,"629":2}}],["confirmations",{"2":{"204":1}}],["confirmation",{"2":{"53":1,"137":1,"537":1,"629":2,"659":1}}],["confirm",{"0":{"432":1,"436":1},"2":{"33":1,"34":2,"36":1,"131":1,"204":1,"503":1,"522":2,"727":1}}],["conferences",{"2":{"45":1}}],["conference",{"2":{"45":1}}],["conversion",{"2":{"155":1,"455":1}}],["conversations",{"2":{"8":1,"34":1,"37":1}}],["converted",{"2":{"157":1,"317":1}}],["converting",{"0":{"153":1},"1":{"154":1,"155":1,"156":1,"157":1,"158":1,"159":1,"160":1,"161":1}}],["convert",{"2":{"121":1,"154":1,"155":1,"156":1,"158":1,"253":1,"373":1,"374":1}}],["convenient",{"2":{"36":1}}],["conditioned",{"2":{"34":1}}],["conditions",{"2":{"31":1,"189":1,"509":1}}],["conducting",{"2":{"406":1}}],["conducted",{"2":{"34":1}}],["conduct",{"0":{"6":1,"52":1},"1":{"7":1,"8":1,"9":1,"10":1,"11":1,"12":1,"13":1,"14":1,"15":1,"16":1,"17":1},"2":{"8":1,"9":1,"10":1,"12":1,"14":1,"15":1,"17":4,"25":1,"36":1,"52":3,"64":1,"405":1,"696":1}}],["consoleconfirm",{"2":{"727":1}}],["consolecode",{"2":{"721":1,"727":1}}],["console$request",{"2":{"662":1}}],["console$",{"2":{"488":1}}],["console",{"2":{"349":1}}],["consoleblockhash",{"2":{"348":1}}],["consolebasefeepergas",{"2":{"347":1}}],["cons",{"0":{"169":1,"175":1,"179":1,"183":1,"187":1,"191":1,"194":1}}],["consumer",{"2":{"629":4}}],["consume",{"2":{"386":1,"611":1}}],["consumes",{"2":{"386":1}}],["consumed",{"2":{"149":1,"385":2}}],["consuming",{"2":{"107":1}}],["consulted",{"2":{"32":1}}],["consist",{"2":{"135":1,"387":1}}],["consists",{"2":{"134":1,"421":1,"422":1,"424":3,"438":1}}],["consistency",{"2":{"47":1}}],["consistent",{"2":{"47":2}}],["considerable",{"2":{"611":1}}],["considerations",{"0":{"548":1},"2":{"107":2,"544":1,"548":1}}],["consider",{"2":{"40":1,"42":2,"43":1,"46":3,"317":1,"405":1,"599":1}}],["considered",{"2":{"8":1}}],["consensus",{"0":{"390":1,"392":1,"471":1,"530":1,"587":1,"589":1,"594":1,"603":1,"604":1,"673":1,"674":1,"711":1},"1":{"588":1,"589":1,"590":2,"591":2,"592":2,"593":1,"594":1,"595":2,"596":2,"597":1,"598":1,"599":1,"600":1,"601":1,"602":1,"603":1,"604":1,"605":2,"606":2,"607":2,"608":2,"609":2,"610":2,"611":2,"674":1,"675":1,"676":1,"677":1,"712":1},"2":{"89":2,"107":1,"130":1,"133":1,"135":1,"145":1,"146":1,"206":3,"229":1,"232":1,"233":1,"278":1,"283":1,"284":1,"289":2,"309":2,"326":2,"334":2,"380":3,"384":1,"392":4,"418":1,"421":1,"422":6,"423":2,"444":1,"459":1,"465":1,"470":2,"471":3,"477":1,"478":3,"494":1,"503":1,"517":1,"526":1,"530":2,"540":1,"541":1,"574":2,"587":3,"588":3,"594":1,"600":1,"603":1,"606":1,"612":1,"626":1,"661":2,"673":1,"679":5,"685":2,"688":6,"689":2,"691":4,"696":5,"708":2,"711":1,"717":1,"719":1,"722":1,"727":1,"730":2}}],["consents",{"2":{"34":1}}],["consent",{"2":{"33":1}}],["consequence",{"2":{"13":1,"14":1,"15":1,"16":1,"418":1}}],["consequences",{"2":{"12":1,"14":1,"32":1,"34":1,"403":2}}],["const",{"2":{"267":10,"268":54,"270":2,"398":8}}],["construction",{"2":{"169":1,"172":1,"181":2}}],["constructions",{"2":{"162":1,"176":1,"184":1}}],["constructing",{"2":{"159":2}}],["constructive",{"2":{"8":1}}],["constructed",{"2":{"156":1,"207":1,"406":1}}],["construct",{"2":{"153":1,"155":4,"158":1,"375":1}}],["constructs",{"2":{"146":1,"180":1}}],["constructor",{"2":{"127":2,"260":2}}],["constantly",{"2":{"627":1}}],["constants",{"2":{"305":6,"315":3,"319":1,"357":8,"398":18,"399":1,"551":4,"586":6,"592":22,"602":6,"721":2,"727":2}}],["constant",{"2":{"89":1,"386":1,"671":2}}],["constitute",{"2":{"30":1,"34":1}}],["context",{"2":{"150":4,"151":10,"161":8,"372":2,"373":14,"374":4,"392":8}}],["contents",{"0":{"286":1},"2":{"375":1}}],["content",{"2":{"2":1,"31":1,"46":2,"65":1,"260":8,"268":17,"334":2,"373":4,"494":2,"712":1,"714":1,"715":1,"716":1}}],["containerization",{"2":{"575":1}}],["container",{"0":{"244":1},"2":{"243":9,"244":11,"346":6,"578":5,"579":4,"580":2}}],["containerstyle",{"2":{"267":4}}],["containers",{"0":{"346":1},"2":{"243":1,"244":1,"345":1,"346":2}}],["contained",{"2":{"160":1,"165":1}}],["containing",{"2":{"148":1,"151":4,"155":1,"157":1,"159":5,"161":4,"170":3,"178":1,"181":1,"424":1,"503":2}}],["contain",{"2":{"86":1,"146":1,"148":1,"152":2,"155":1,"173":1,"382":1,"416":1,"424":1,"442":1,"495":1,"626":1}}],["contains",{"2":{"86":1,"89":1,"92":1,"116":1,"134":2,"151":4,"153":4,"159":5,"161":2,"248":1,"347":1,"469":1,"495":1,"556":1,"672":1,"685":1}}],["contacts",{"2":{"45":1,"46":1}}],["contact",{"2":{"41":1,"48":1}}],["contrastingly",{"2":{"418":1}}],["contrast",{"2":{"203":1,"410":1}}],["contracttest",{"2":{"260":2}}],["contractaddress",{"2":{"121":2,"253":2,"268":6,"270":3}}],["contractforge",{"2":{"115":1,"247":1}}],["contractscript",{"2":{"262":2,"263":2,"264":2}}],["contracts",{"0":{"123":1,"126":1,"210":1,"227":1},"1":{"124":1,"125":1,"126":1,"127":1,"128":1,"129":1,"130":1,"131":1},"2":{"75":3,"77":3,"79":1,"82":1,"84":4,"85":4,"86":2,"89":2,"91":4,"92":1,"94":3,"95":1,"126":2,"127":6,"129":1,"131":1,"136":2,"137":4,"141":1,"143":2,"153":1,"178":1,"196":1,"199":1,"201":1,"209":1,"210":2,"217":1,"227":1,"229":1,"233":1,"260":3,"265":1,"272":3,"273":3}}],["contract",{"0":{"75":1,"76":1,"95":1,"112":1,"116":1,"117":1,"118":1,"121":1,"151":1,"197":1,"198":1,"245":1,"248":1,"249":1,"250":1,"253":1,"260":1},"1":{"76":1,"96":1,"97":1,"113":1,"114":1,"115":1,"116":1,"117":1,"118":1,"119":2,"120":2,"121":1,"122":1,"246":1,"247":1,"248":1,"249":1,"250":1,"251":2,"252":2,"253":1,"254":1},"2":{"31":1,"34":1,"73":1,"74":1,"75":6,"76":5,"77":4,"84":1,"85":2,"88":1,"89":4,"91":4,"98":1,"99":1,"107":2,"109":1,"113":2,"115":1,"116":8,"117":11,"120":5,"121":1,"122":1,"127":6,"129":1,"130":1,"131":7,"133":1,"135":4,"146":1,"148":4,"149":1,"150":2,"151":10,"153":6,"159":9,"160":2,"161":10,"162":1,"163":1,"167":2,"169":1,"170":1,"172":1,"173":2,"177":1,"181":2,"183":1,"184":2,"185":3,"188":1,"189":1,"192":1,"196":1,"197":1,"198":1,"204":1,"206":2,"207":8,"208":5,"209":1,"210":2,"222":1,"231":1,"245":1,"247":1,"248":8,"249":11,"251":3,"252":4,"253":1,"254":1,"256":1,"258":1,"259":2,"260":15,"261":1,"262":7,"263":13,"264":11,"265":2,"268":16,"270":3,"272":13,"273":4,"286":1,"376":3,"377":4,"379":2,"411":1,"458":1}}],["controlled",{"2":{"84":1}}],["control",{"2":{"30":1,"31":1,"34":1,"553":1,"611":1,"623":1}}],["contributor",{"2":{"17":4,"449":1}}],["contributors",{"2":{"7":1,"449":1,"450":1}}],["contributing",{"2":{"8":1}}],["contributiondao",{"2":{"674":1,"675":1,"676":1}}],["contributions",{"2":{"2":1,"9":1,"21":1,"27":3,"288":1}}],["contribution",{"0":{"2":1},"2":{"2":2,"43":1,"105":1}}],["contributes",{"2":{"8":1,"385":1}}],["contribute",{"2":{"7":1,"24":1,"27":2,"44":1}}],["continuing",{"2":{"727":1}}],["continuousvestingaccount",{"2":{"523":3}}],["continuous",{"2":{"511":1,"522":1}}],["continuously",{"2":{"450":4,"466":1}}],["continues",{"2":{"501":1,"698":1}}],["continue",{"2":{"71":1,"85":1,"255":1,"351":1,"441":1,"442":1,"624":1}}],["continued",{"2":{"14":1,"34":1,"44":1}}],["continually",{"2":{"29":1}}],["comcelestia",{"2":{"471":1}}],["combination",{"2":{"351":1}}],["combines",{"2":{"365":1}}],["combined",{"0":{"327":1},"2":{"232":1}}],["combine",{"2":{"33":1}}],["comfortable",{"2":{"307":1,"559":1}}],["coming",{"2":{"258":1,"714":1,"715":1}}],["comexport",{"2":{"577":1}}],["cometbft",{"2":{"203":1,"206":1,"444":1,"671":1}}],["come",{"2":{"133":1,"206":1,"223":1,"666":1}}],["comes",{"2":{"107":1}}],["com",{"0":{"47":1},"2":{"45":2,"47":6,"75":2,"82":1,"87":2,"89":2,"149":18,"150":2,"151":14,"159":2,"161":22,"236":2,"272":2,"287":1,"304":2,"307":2,"309":2,"341":2,"349":2,"371":2,"372":10,"387":2,"392":16,"399":3,"400":1,"453":1,"470":11,"471":1,"474":1,"475":2,"485":2,"508":2,"531":2,"539":2,"546":2,"559":2,"577":1,"585":2,"592":12,"629":4,"660":2,"661":2,"674":10,"675":9,"676":8,"683":1,"688":2,"689":7,"690":7,"691":9,"692":8,"694":1}}],["computing",{"2":{"283":1}}],["computation",{"2":{"205":1}}],["computed",{"2":{"173":1,"385":1,"416":1}}],["computesquaresizefromshareproof",{"2":{"131":1}}],["computesquaresizefromrowproof",{"2":{"131":1,"172":1}}],["computes",{"2":{"131":2,"385":1,"424":1}}],["compute",{"2":{"108":1,"424":1}}],["comprised",{"2":{"439":1}}],["comprises",{"2":{"159":1}}],["comprehensive",{"2":{"0":1}}],["compilation",{"2":{"117":2,"120":2,"249":2,"252":2}}],["compiling",{"2":{"116":4,"117":2,"120":2,"248":4,"249":2,"252":2,"304":1,"585":1}}],["compiler",{"2":{"116":2,"126":1,"248":2}}],["compile",{"2":{"116":1,"248":1}}],["compatmode",{"2":{"628":2}}],["compatibility",{"0":{"94":1,"281":1},"2":{"89":1}}],["compatible",{"0":{"699":1,"702":1,"704":1},"2":{"79":1,"94":1,"212":1,"214":1,"227":1,"257":1,"281":1,"307":1,"624":1,"628":1}}],["compared",{"2":{"205":1}}],["compare",{"2":{"173":1}}],["comparing",{"2":{"167":1}}],["compact",{"0":{"167":1},"2":{"167":4,"168":1,"181":1}}],["companion",{"2":{"56":1}}],["companies",{"2":{"44":3}}],["company",{"2":{"30":5,"31":8,"32":3,"34":8,"44":1}}],["compelling",{"2":{"44":1}}],["component",{"2":{"94":1,"233":1,"268":2,"398":2}}],["components",{"0":{"106":1},"1":{"107":1,"108":1,"109":1,"110":1},"2":{"27":1,"39":1,"106":1,"209":1,"288":1,"382":1,"387":2,"398":8,"421":1,"625":1,"666":1}}],["compose",{"2":{"80":1,"89":1,"90":1,"243":2,"244":6,"353":2,"357":1,"360":1}}],["composition",{"2":{"28":1}}],["complex",{"2":{"610":1}}],["completing",{"2":{"79":1,"727":1}}],["completion",{"2":{"30":1,"34":3}}],["completes",{"2":{"405":1}}],["completely",{"2":{"316":1}}],["completed",{"2":{"33":1,"234":1,"539":1,"559":1}}],["complete",{"2":{"24":1,"31":3,"32":1,"53":1,"85":1,"91":2,"234":1,"264":2,"375":1,"419":1,"447":1}}],["compliant",{"2":{"128":1}}],["compliance",{"2":{"24":1,"34":3}}],["comply",{"2":{"25":1,"33":1}}],["complaints",{"2":{"11":1}}],["comma",{"2":{"592":2,"651":2}}],["commands",{"0":{"327":1,"488":1},"1":{"489":1,"490":1,"491":1,"492":1,"493":1,"494":1,"495":1,"496":1,"497":1,"498":1,"499":1,"500":1},"2":{"236":1,"304":2,"307":1,"313":2,"367":1,"392":2,"488":2,"494":1,"500":1,"539":1,"540":1,"554":1,"559":1,"585":2,"592":1}}],["command",{"0":{"312":1,"507":1},"2":{"76":2,"89":2,"91":1,"115":1,"116":1,"121":2,"126":1,"170":1,"236":1,"239":1,"243":2,"244":5,"247":1,"248":1,"252":1,"253":2,"270":1,"272":2,"304":2,"307":1,"309":2,"310":4,"311":1,"312":3,"314":1,"315":1,"316":1,"317":1,"319":3,"323":1,"326":1,"330":1,"331":2,"332":1,"344":1,"345":1,"352":1,"360":2,"361":1,"362":1,"363":1,"367":1,"374":1,"471":1,"473":1,"484":1,"485":1,"488":4,"492":4,"494":4,"495":2,"496":1,"497":2,"498":1,"499":3,"506":1,"522":4,"527":1,"529":2,"531":1,"537":1,"540":1,"544":2,"550":1,"553":1,"554":2,"555":1,"559":1,"560":1,"566":1,"577":1,"579":2,"580":2,"585":2,"602":1,"607":1,"611":1,"619":1,"620":1,"633":2,"634":1,"644":1,"645":1,"646":2,"647":1,"648":1,"649":2,"650":2,"652":2,"653":1,"659":1,"660":1,"661":3,"662":1,"693":1,"721":1}}],["commission",{"2":{"24":1,"34":2,"496":2,"500":6,"727":6}}],["commitment",{"0":{"148":1,"165":1,"166":1,"167":1,"168":1,"169":1,"376":1},"1":{"166":1,"167":1,"168":1,"169":1,"377":1,"378":1,"379":1},"2":{"148":5,"153":2,"159":2,"165":3,"166":9,"167":2,"168":2,"171":2,"173":3,"174":1,"181":6,"182":1,"184":1,"185":2,"186":1,"189":5,"192":7,"207":2,"213":5,"290":5,"317":5,"319":10,"320":3,"334":2,"343":1,"351":2,"379":1,"416":1,"418":1,"424":6,"438":2}}],["commitments",{"0":{"160":1,"181":1,"182":1,"183":1,"188":1,"190":1,"191":1},"1":{"182":1,"183":1,"189":1,"190":1,"191":1},"2":{"109":1,"134":6,"136":1,"137":1,"160":1,"167":1,"178":1,"181":2,"184":1,"185":1,"188":1,"201":1,"416":1,"425":1}}],["committing",{"0":{"136":1},"2":{"151":2,"159":2,"161":2}}],["committees",{"0":{"202":1},"1":{"203":1,"204":1,"205":1},"2":{"203":1}}],["committee",{"2":{"83":1}}],["committedto",{"2":{"161":4}}],["committed",{"0":{"151":1},"2":{"47":1,"128":1,"131":9,"134":1,"139":1,"146":2,"148":4,"150":2,"151":8,"153":8,"159":6,"161":4,"162":1,"166":1,"169":1,"170":1,"172":1,"173":2,"207":1,"288":1,"392":4,"424":1}}],["commitforge",{"2":{"126":1}}],["commits",{"2":{"9":1,"131":1,"136":1,"153":2,"159":2,"176":1,"192":1}}],["commit",{"2":{"2":3,"94":1,"126":1,"131":1,"136":1,"148":1,"181":1,"307":1,"326":4,"559":1}}],["commonly",{"2":{"403":1,"503":1}}],["common",{"2":{"17":1,"60":1,"151":2,"161":2,"351":2,"422":2,"547":1,"655":1}}],["communities",{"2":{"65":1}}],["community",{"0":{"5":1,"48":1,"52":1,"64":1,"448":1,"495":1},"2":{"2":1,"3":2,"4":1,"5":5,"7":2,"8":2,"9":2,"10":3,"11":2,"12":2,"13":3,"14":2,"15":3,"16":3,"17":1,"25":1,"27":1,"36":1,"38":1,"39":1,"40":4,"43":2,"45":2,"47":1,"50":1,"52":3,"53":1,"55":1,"57":1,"63":1,"64":3,"65":1,"309":1,"388":1,"444":1,"448":2,"460":2,"471":1,"494":1,"495":11,"666":1,"667":1,"679":1,"684":1,"698":1,"703":1}}],["communicating",{"2":{"503":2}}],["communications",{"0":{"45":1}}],["communication",{"2":{"15":1,"34":1,"45":1,"55":1,"260":2,"541":1,"547":1,"552":1,"569":1,"627":1,"690":1,"715":1}}],["communicates",{"2":{"380":1}}],["communicate",{"2":{"9":1,"42":1,"89":1,"369":1,"722":1}}],["comments",{"2":{"8":1,"9":1}}],["fsudo",{"2":{"712":1,"714":1,"715":1,"716":1}}],["fssl",{"2":{"304":2,"585":2}}],["f5nodes",{"2":{"674":1,"675":1,"676":1,"689":1,"690":1,"691":1}}],["fc",{"2":{"386":2}}],["fmt",{"2":{"139":4,"150":2,"151":4,"159":2,"160":2,"161":2,"372":2,"373":8,"392":8}}],["f",{"2":{"139":22,"346":4,"353":1,"360":2,"712":1,"714":1,"715":1,"716":1}}],["floor",{"2":{"455":1}}],["flow",{"2":{"23":1,"43":1}}],["flipping",{"2":{"405":1}}],["flat",{"2":{"384":3,"387":1,"439":1}}],["flag",{"2":{"107":1,"309":1,"312":1,"313":1,"314":1,"315":1,"319":1,"320":3,"326":2,"334":2,"367":1,"484":1,"511":1,"522":2,"527":1,"531":1,"539":1,"541":1,"544":1,"547":3,"551":1,"552":1,"553":1,"566":1,"579":1,"626":1}}],["flags",{"0":{"313":1,"544":1},"1":{"314":1,"315":1,"316":1,"545":1,"546":1,"547":1},"2":{"89":1,"102":2,"237":2,"312":4,"313":2,"390":2,"492":4,"529":2,"542":2,"544":2,"577":1,"626":1}}],["flexdirection",{"2":{"268":4}}],["flex",{"2":{"268":6}}],["flexibility",{"0":{"205":1},"2":{"195":1,"205":1,"233":1,"422":1}}],["flexible",{"2":{"41":1}}],["fun",{"2":{"365":1}}],["func",{"2":{"137":2,"139":3,"150":2,"151":6,"155":7,"156":1,"157":1,"158":1,"159":1,"161":22,"351":2,"372":2,"373":12,"374":1,"392":2}}],["functionalities",{"2":{"685":1}}],["functionality",{"2":{"36":1,"94":1,"135":2,"260":1,"280":1,"297":1}}],["functional",{"2":{"667":1}}],["functionid",{"2":{"377":1}}],["functioning",{"2":{"351":1}}],["function",{"2":{"110":1,"116":6,"117":13,"127":4,"131":5,"161":2,"248":6,"249":13,"260":24,"262":4,"268":10,"351":4,"372":1,"373":10,"374":1,"377":2,"385":2,"386":7,"387":3,"392":8,"398":5,"419":1,"422":1}}],["functions",{"0":{"373":1,"374":1},"2":{"108":1,"116":1,"117":1,"131":4,"229":3,"231":1,"248":1,"249":1,"260":1,"351":1,"373":3,"374":1,"422":5,"509":1}}],["fund",{"0":{"528":1,"537":1},"2":{"76":1,"91":1,"119":1,"272":1,"310":2,"448":1,"485":1,"495":2,"528":1,"537":1,"620":1,"661":1}}],["funded",{"0":{"119":1},"2":{"74":1,"78":1,"80":1,"119":1,"252":1,"272":1,"274":1,"302":1,"310":1,"337":1,"357":1,"360":1,"630":1}}],["funds",{"2":{"46":1,"69":2,"71":1,"82":1,"255":1,"310":1,"367":1,"403":1,"448":1,"494":1,"495":3,"507":4,"509":1,"662":1}}],["funding",{"2":{"44":1,"91":3,"288":1}}],["fullnodes",{"2":{"139":1}}],["fullnode",{"2":{"135":6,"139":4}}],["fully",{"2":{"80":1,"357":1,"441":2,"450":1,"681":1}}],["full",{"0":{"100":1,"102":1,"103":1,"138":1,"151":1,"256":1,"530":1,"587":1,"589":1,"612":1,"613":1,"615":1,"618":1,"619":1,"620":1,"621":1,"622":1,"623":1,"688":1,"692":1,"714":1},"1":{"101":1,"102":1,"103":1,"139":1,"140":1,"141":1,"142":1,"143":1,"257":1,"258":1,"259":1,"260":1,"261":1,"262":1,"263":1,"264":1,"265":1,"266":1,"267":1,"268":1,"269":1,"270":1,"271":1,"588":1,"589":1,"590":2,"591":2,"592":2,"593":1,"594":1,"595":1,"596":1,"597":1,"598":1,"599":1,"600":1,"601":1,"602":1,"603":1,"604":1,"605":1,"606":1,"607":1,"608":1,"609":1,"610":1,"611":1,"613":1,"614":1,"615":1,"616":2,"617":1,"618":1,"619":2,"620":2,"621":1,"622":1,"623":1},"2":{"34":1,"55":1,"88":1,"89":3,"92":2,"102":1,"103":1,"107":2,"108":1,"135":4,"139":1,"145":3,"177":1,"181":1,"235":1,"237":2,"238":2,"239":2,"256":1,"283":1,"308":1,"380":1,"406":3,"407":4,"416":3,"425":1,"470":2,"471":2,"488":2,"526":1,"527":1,"529":1,"530":2,"553":4,"555":1,"565":2,"580":1,"587":2,"588":3,"603":1,"606":1,"612":2,"613":2,"614":1,"619":6,"620":7,"621":9,"622":2,"623":1,"670":1,"673":1,"674":1,"675":1,"676":1,"678":1,"679":3,"685":2,"688":5,"689":1,"690":1,"691":3,"692":7,"696":3,"707":1,"711":1,"714":17,"719":1,"730":1}}],["further",{"2":{"34":2,"237":1}}],["furthermore",{"2":{"27":1,"53":1,"380":2,"419":1,"422":1}}],["future",{"2":{"22":1,"37":1,"38":2,"42":1,"43":1,"48":1,"94":1,"233":1,"280":1,"297":1,"302":1,"442":1,"449":1,"450":1,"548":1,"552":1,"625":1,"715":1}}],["federation",{"2":{"503":2}}],["federal",{"2":{"34":1}}],["feasible",{"2":{"417":1}}],["feature",{"2":{"84":1,"106":1,"233":1,"255":1,"626":1}}],["features",{"2":{"65":1,"232":1,"293":1,"295":1,"307":1,"415":2,"559":1,"624":1,"625":6,"639":1}}],["fetchposts",{"2":{"260":8,"263":2,"264":2,"268":8}}],["fetchpost",{"2":{"260":4}}],["fetches",{"2":{"260":4}}],["fetched",{"2":{"108":1}}],["fetcher",{"2":{"107":1}}],["fetch",{"2":{"107":1,"268":4,"365":1}}],["fetching",{"2":{"88":1}}],["few",{"2":{"77":1,"139":1,"163":1,"169":1,"225":1,"231":2,"232":1,"273":1,"290":1,"373":1,"529":1,"601":1}}],["fellow",{"2":{"40":1,"50":1,"55":1}}],["feel",{"2":{"696":1}}],["feecurrencies",{"2":{"398":2}}],["fee=gas",{"2":{"386":1}}],["feegrant",{"2":{"278":1,"385":1}}],["fee",{"0":{"383":1,"386":1,"439":1},"1":{"384":1,"385":1,"386":1,"387":1,"388":1},"2":{"84":1,"85":1,"91":20,"317":5,"329":6,"373":4,"374":5,"384":5,"386":3,"439":2,"444":1,"457":1,"522":2}}],["fees=21000utia",{"2":{"721":1,"727":2}}],["fees=21000utiacelestia",{"2":{"721":1}}],["fees",{"0":{"384":1},"1":{"385":1,"386":1,"387":1},"2":{"31":1,"320":1,"384":3,"388":1,"394":1,"396":1,"439":3,"507":4,"522":6,"529":2,"629":4}}],["feedback",{"0":{"4":1,"29":1},"2":{"4":1,"8":1,"29":1,"36":1,"37":1,"38":2,"666":1,"696":1}}],["february",{"2":{"21":1}}],["fatalf",{"2":{"373":10}}],["fatal",{"0":{"556":1},"2":{"373":2,"556":2}}],["faster",{"2":{"568":1}}],["fastest",{"2":{"540":1,"560":1}}],["fast",{"2":{"282":1,"284":1}}],["falsediscard",{"2":{"611":1}}],["false",{"2":{"151":2,"357":1,"425":3,"522":2,"541":1,"547":2,"552":1,"566":1,"568":1,"569":1,"611":3,"629":24,"712":1}}],["fallen",{"2":{"352":1}}],["falls",{"2":{"351":1,"628":1}}],["fallback",{"0":{"110":1,"255":1,"350":1,"351":1,"352":1},"1":{"351":1,"352":1},"2":{"106":1,"110":6,"255":6,"350":1,"351":5,"352":2}}],["fall",{"2":{"88":1,"110":1,"139":1,"255":1,"350":1}}],["faults",{"2":{"135":1}}],["faulty",{"2":{"109":1,"178":1}}],["faucets",{"2":{"357":1,"630":1}}],["faucet",{"0":{"219":1,"472":1,"693":1},"1":{"473":1,"474":1},"2":{"75":1,"80":1,"82":1,"119":1,"219":2,"246":1,"272":1,"310":2,"337":1,"367":1,"472":1,"473":3,"474":2,"528":1,"537":2,"693":4}}],["familiarize",{"2":{"375":1}}],["familiarity",{"2":{"80":1,"101":1}}],["family",{"2":{"45":2}}],["face",{"2":{"256":1}}],["fact",{"2":{"108":1,"146":1,"167":1,"173":1}}],["factories",{"2":{"89":1}}],["factory",{"2":{"85":1}}],["factors",{"2":{"34":1,"42":1,"671":2}}],["facilitating",{"2":{"85":1}}],["facilitated",{"2":{"65":1}}],["facilitate",{"2":{"37":1,"260":2,"421":1}}],["failures",{"2":{"288":1}}],["failure",{"2":{"139":4,"712":2,"714":2,"715":2,"716":2}}],["failed",{"2":{"117":2,"249":2,"398":2}}],["fails",{"2":{"110":1,"425":1}}],["fail",{"2":{"31":1,"314":1,"352":1}}],["failing",{"2":{"25":1}}],["fairly",{"2":{"11":1,"278":1}}],["fair",{"2":{"9":1,"19":1,"466":1}}],["faq",{"0":{"401":1},"1":{"402":1,"403":1,"404":1,"405":1,"406":1,"407":1,"408":1,"409":1,"410":1,"411":1,"412":1,"413":1,"414":1},"2":{"17":2}}],["frequently",{"2":{"701":1}}],["fresh",{"2":{"212":1,"605":1}}],["free",{"2":{"7":1,"31":1,"40":1,"442":2,"523":2,"696":1}}],["frameworks",{"2":{"233":2,"265":1,"285":1}}],["framework",{"0":{"225":1},"2":{"265":2,"444":1,"691":1}}],["frame",{"2":{"84":1}}],["fraudulent",{"2":{"139":1,"601":1}}],["fraud",{"0":{"131":1,"418":1},"2":{"33":1,"108":2,"131":1,"136":1,"139":3,"161":5,"162":1,"176":1,"177":4,"178":1,"181":2,"184":1,"232":2,"380":1,"406":1,"407":3,"418":1,"422":1,"613":2}}],["frontendcd",{"2":{"77":1,"273":1}}],["frontend",{"0":{"77":1,"78":1,"265":1,"273":1,"274":1},"1":{"266":1},"2":{"75":3,"76":1,"77":5,"258":1,"263":1,"264":1,"265":10,"266":1,"272":4,"273":5,"381":1}}],["from=$validator",{"2":{"721":2,"727":2}}],["from=",{"2":{"496":2}}],["from",{"0":{"70":1,"71":1,"388":1,"539":1,"559":1},"1":{"71":1,"72":1},"2":{"2":1,"4":1,"8":1,"13":1,"15":1,"16":1,"17":1,"27":2,"30":1,"31":5,"33":4,"34":9,"36":1,"37":1,"38":1,"43":1,"46":1,"47":1,"49":1,"50":1,"52":1,"54":1,"65":3,"66":1,"68":1,"69":1,"75":3,"76":2,"77":1,"81":1,"84":2,"85":1,"87":1,"88":2,"89":1,"90":1,"91":8,"95":1,"107":3,"108":1,"109":1,"116":1,"120":1,"121":1,"126":1,"130":1,"131":4,"135":8,"137":1,"139":8,"154":1,"155":2,"156":2,"159":2,"161":4,"170":1,"171":1,"172":1,"177":1,"184":1,"201":1,"204":1,"206":1,"207":1,"212":2,"213":6,"231":2,"232":1,"233":2,"238":1,"239":1,"243":1,"246":2,"248":1,"251":1,"252":1,"253":1,"256":1,"258":1,"262":2,"264":3,"267":14,"268":23,"270":1,"272":5,"273":1,"280":1,"288":1,"290":6,"299":1,"304":2,"310":1,"313":3,"315":1,"316":1,"317":1,"319":3,"324":1,"326":1,"337":1,"344":1,"347":1,"348":2,"349":1,"351":4,"352":1,"361":1,"363":1,"365":1,"367":2,"370":1,"373":2,"374":3,"375":1,"377":3,"384":3,"388":2,"392":6,"394":1,"398":4,"403":1,"406":1,"407":1,"409":1,"410":1,"411":2,"419":1,"422":1,"423":2,"425":2,"441":2,"442":2,"444":1,"450":5,"465":1,"466":1,"471":1,"473":1,"478":2,"484":1,"485":1,"491":1,"493":5,"494":7,"495":2,"497":4,"498":2,"503":7,"506":2,"507":14,"509":2,"515":1,"521":3,"522":8,"524":1,"527":2,"529":2,"530":1,"531":1,"538":1,"554":1,"556":1,"562":1,"569":1,"577":1,"585":2,"592":1,"601":2,"602":3,"606":1,"607":1,"620":2,"630":1,"638":2,"660":1,"662":1,"679":1,"688":3,"693":1,"701":1,"715":2,"716":1,"727":2}}],["friends",{"2":{"44":1,"45":2,"643":1}}],["friendly",{"2":{"41":2}}],["firewall",{"2":{"541":2,"552":2}}],["firstly",{"2":{"213":1}}],["first",{"2":{"23":4,"43":1,"50":1,"57":1,"59":1,"73":1,"84":1,"89":2,"91":1,"105":1,"115":1,"120":2,"134":3,"135":1,"136":1,"137":1,"139":1,"151":2,"155":2,"156":1,"159":4,"161":6,"163":1,"170":1,"181":1,"201":2,"207":1,"213":1,"234":1,"236":1,"247":1,"251":2,"252":2,"256":1,"260":8,"263":2,"264":4,"266":1,"268":2,"270":3,"272":2,"277":1,"290":2,"304":1,"329":1,"340":1,"348":1,"360":1,"367":2,"370":1,"373":1,"385":2,"388":1,"449":1,"455":1,"457":1,"469":1,"494":3,"495":1,"500":1,"503":1,"512":1,"519":1,"526":1,"536":1,"553":1,"561":1,"578":1,"585":1,"625":1,"671":1,"672":1,"685":1,"719":1}}],["figment",{"2":{"508":1}}],["figure",{"2":{"419":1,"421":1,"424":2}}],["figures",{"2":{"65":1}}],["five",{"2":{"449":1}}],["fit",{"0":{"411":1}}],["fits",{"2":{"232":1}}],["fields",{"2":{"601":1}}],["field",{"2":{"154":1,"155":1,"159":1,"569":1,"608":2}}],["fide",{"2":{"34":1}}],["fine",{"2":{"425":1}}],["finished",{"2":{"116":2,"117":2,"248":2,"249":2,"373":2,"374":2,"716":1}}],["finish",{"0":{"91":1}}],["finger",{"2":{"41":1}}],["finality",{"2":{"229":1}}],["finalize",{"2":{"36":1,"435":1}}],["finally",{"2":{"103":1,"109":1,"159":1,"169":1,"173":1,"373":1,"385":1,"503":1}}],["final",{"2":{"36":1,"374":1,"422":3,"505":1,"507":2,"650":2}}],["financial",{"2":{"34":1,"44":1}}],["finding",{"2":{"5":1,"40":1,"44":1,"46":1,"48":2}}],["find",{"0":{"347":1,"349":1},"2":{"0":1,"4":1,"43":1,"44":1,"47":1,"60":1,"61":1,"64":1,"75":1,"88":2,"91":1,"96":1,"97":1,"107":3,"131":1,"136":2,"151":2,"159":2,"168":1,"207":1,"208":1,"272":1,"287":1,"297":1,"302":1,"309":1,"310":1,"346":2,"470":1,"471":1,"485":1,"505":1,"515":1,"529":1,"552":1,"557":1,"565":1,"568":1,"583":1,"620":1,"626":1,"648":1,"649":1,"661":1,"679":2,"681":1,"692":1,"709":1,"724":1}}],["fix",{"2":{"388":1,"557":1}}],["fixed",{"2":{"385":2,"386":2}}],["fixes",{"2":{"24":1}}],["fixing",{"2":{"2":1}}],["fill",{"2":{"503":1}}],["filled",{"2":{"23":1}}],["filter",{"0":{"638":1},"2":{"629":8,"635":4,"638":1}}],["filters",{"2":{"351":1}}],["filteropts",{"2":{"151":2,"159":2}}],["filterdatacommitmentstored",{"2":{"151":2,"159":2}}],["file=",{"2":{"503":2}}],["filecoin",{"2":{"414":1}}],["file",{"0":{"650":1,"651":1},"2":{"84":3,"86":3,"92":1,"102":3,"116":1,"117":2,"248":1,"249":2,"260":1,"265":1,"266":1,"353":1,"368":1,"370":1,"373":5,"377":1,"392":2,"488":6,"491":2,"494":5,"495":4,"500":1,"502":1,"503":1,"504":1,"531":2,"536":1,"539":1,"540":1,"554":1,"556":2,"557":9,"560":1,"571":1,"592":5,"607":1,"610":1,"611":1,"630":9,"648":1,"649":2,"650":4,"651":2,"712":2,"714":2,"715":2,"716":1,"729":1}}],["filesystem",{"2":{"583":1}}],["fileserror",{"2":{"557":1}}],["files",{"0":{"86":1,"87":1,"371":1,"557":1,"631":1},"2":{"3":1,"58":1,"85":1,"86":3,"87":1,"116":2,"117":2,"120":2,"248":2,"249":2,"252":2,"258":1,"304":1,"351":1,"371":1,"488":2,"557":2,"585":1,"650":4}}],["fontsize",{"2":{"268":4}}],["four",{"2":{"166":1,"229":1,"389":1,"422":3,"494":1}}],["found",{"2":{"84":2,"95":1,"102":1,"108":1,"110":1,"131":1,"135":1,"146":1,"148":1,"155":1,"156":1,"158":1,"159":1,"166":1,"167":1,"170":1,"173":2,"180":2,"207":1,"211":1,"213":1,"314":1,"334":1,"337":2,"365":1,"387":2,"445":1,"471":1,"494":2,"601":3,"637":1,"670":1}}],["foundry",{"2":{"75":1,"114":1,"115":1,"121":1,"125":1,"246":1,"247":1,"253":1,"256":1,"258":1,"259":2,"260":1,"265":2,"272":1}}],["foundation",{"0":{"18":1,"20":1},"1":{"19":1,"20":1,"21":2,"22":2,"23":2,"24":2,"25":2,"26":1,"27":1,"28":1,"29":1},"2":{"19":2,"21":1,"22":2,"24":2,"25":1,"27":2,"28":1,"288":1,"449":1}}],["folders",{"2":{"515":4,"605":1,"629":1}}],["folder",{"2":{"92":1,"108":1,"115":1,"247":1,"539":1}}],["followed",{"2":{"136":1}}],["follows",{"2":{"24":1,"135":1,"153":1,"155":2,"156":2,"157":1,"158":1,"160":1,"161":1,"167":1,"173":1,"189":1,"192":1,"312":1,"503":1,"637":1,"671":1}}],["follow",{"2":{"12":1,"35":2,"38":1,"48":1,"88":1,"102":1,"198":1,"264":1,"304":2,"346":2,"377":1,"422":1,"481":1,"483":1,"487":1,"504":1,"540":1,"560":1,"574":1,"575":1,"585":2,"590":1,"591":1,"603":1,"616":1,"617":1,"622":1,"628":1,"630":1,"639":1,"640":1,"643":1,"664":1,"685":1,"719":1,"720":1,"723":1,"726":1,"727":1,"729":1}}],["following",{"2":{"1":1,"27":1,"34":3,"41":1,"42":1,"52":1,"53":1,"54":1,"83":1,"91":2,"95":1,"107":2,"115":1,"116":2,"117":1,"125":1,"134":1,"149":1,"153":1,"155":2,"158":1,"166":1,"170":1,"176":1,"177":1,"185":1,"189":1,"196":1,"207":2,"210":1,"215":1,"234":1,"236":1,"237":1,"239":1,"244":1,"247":1,"248":2,"249":1,"260":2,"262":1,"264":1,"265":1,"266":1,"268":1,"269":1,"270":1,"303":1,"307":1,"310":4,"313":1,"314":1,"315":1,"316":1,"317":3,"319":3,"330":1,"331":2,"344":1,"345":1,"346":2,"361":1,"362":1,"363":1,"366":1,"367":1,"368":2,"370":1,"371":1,"373":2,"374":2,"376":1,"381":2,"393":1,"398":1,"399":1,"400":1,"419":1,"421":1,"422":1,"424":1,"431":1,"432":1,"435":2,"436":1,"452":1,"453":1,"473":1,"478":1,"479":1,"480":1,"484":1,"485":2,"494":1,"495":1,"496":1,"497":1,"498":1,"499":1,"500":1,"503":1,"509":1,"522":1,"541":1,"543":1,"544":1,"548":1,"552":1,"553":1,"554":2,"559":1,"565":1,"580":1,"588":1,"589":1,"592":1,"594":1,"602":1,"603":1,"614":1,"615":1,"619":1,"620":2,"627":1,"629":1,"631":1,"633":1,"634":1,"639":1,"641":1,"642":1,"644":1,"645":1,"646":1,"647":1,"648":1,"649":1,"650":3,"651":2,"652":3,"653":1,"655":1,"656":1,"659":1,"661":3,"662":1,"671":1,"683":1,"693":1,"718":1,"719":1,"724":1,"725":1,"727":2}}],["foster",{"2":{"52":1,"65":1}}],["fosters",{"2":{"43":1,"288":1}}],["fostering",{"2":{"19":1,"41":1,"49":1}}],["focus",{"2":{"458":1,"465":1,"543":1}}],["focuses",{"2":{"293":1}}],["focused",{"2":{"42":2,"205":1,"465":1,"701":1,"703":1}}],["focusing",{"2":{"8":2,"43":1}}],["foods",{"2":{"41":1}}],["food",{"2":{"41":2}}],["forward",{"2":{"712":1}}],["fortunately",{"2":{"503":1}}],["forth",{"2":{"30":1}}],["forget",{"2":{"71":1,"519":2}}],["forge",{"2":{"42":1,"75":2,"116":1,"117":2,"248":1,"249":2,"252":1,"260":3,"262":2,"272":2}}],["forces",{"2":{"42":1}}],["forest",{"2":{"424":1}}],["forever",{"2":{"412":1}}],["foreign",{"2":{"34":1}}],["foregoing",{"2":{"33":1}}],["formula",{"2":{"387":1}}],["formcontainerstyle",{"2":{"268":4}}],["former",{"2":{"423":1}}],["formerly",{"2":{"201":1}}],["formed",{"2":{"34":1}}],["formatting",{"0":{"312":1}}],["format",{"2":{"153":1,"154":1,"238":1,"268":2,"312":1,"317":3,"319":1,"323":1,"326":1,"329":1,"491":1,"494":1,"531":2}}],["formation",{"2":{"33":1}}],["formal",{"2":{"42":1}}],["forms",{"2":{"32":1,"36":1,"37":1,"134":1}}],["forming",{"2":{"31":1}}],["form",{"0":{"53":1},"2":{"27":1,"29":1,"34":1,"53":1,"84":1,"450":1}}],["fork",{"2":{"2":3,"263":2,"388":1}}],["for",{"0":{"48":1,"88":1,"89":1,"131":1,"160":1,"237":1,"238":1,"239":1,"240":1,"276":1,"406":1,"407":1,"420":1,"430":1,"437":1,"442":1,"457":1,"500":1,"604":1,"649":1,"688":1,"699":1,"702":1,"704":1},"1":{"277":1,"278":1,"279":1,"280":1,"281":1,"282":1,"283":1,"421":1,"438":1,"439":1,"605":1,"606":1,"607":1,"608":1,"609":1,"610":1,"611":1},"2":{"0":1,"2":1,"3":4,"4":1,"5":2,"7":1,"8":3,"9":2,"11":1,"12":1,"14":2,"15":1,"17":1,"19":1,"22":2,"23":3,"24":3,"25":2,"26":3,"27":3,"30":3,"31":6,"32":2,"33":4,"34":12,"36":3,"37":1,"38":4,"39":1,"40":13,"41":6,"42":4,"43":6,"44":4,"45":2,"46":5,"47":8,"48":3,"49":2,"50":1,"52":2,"53":2,"56":1,"57":1,"60":2,"61":1,"63":3,"64":1,"65":1,"69":3,"75":1,"79":1,"80":1,"82":2,"84":15,"85":1,"86":5,"88":6,"89":13,"90":1,"91":14,"94":4,"102":1,"105":2,"107":11,"108":5,"110":3,"113":1,"115":1,"116":1,"117":2,"119":1,"121":1,"126":1,"127":1,"129":1,"131":1,"133":1,"134":2,"135":12,"136":4,"137":3,"139":3,"143":2,"145":1,"146":1,"148":2,"149":1,"151":2,"154":1,"159":6,"160":4,"161":12,"163":1,"168":1,"169":2,"170":1,"172":1,"174":1,"177":1,"178":1,"185":1,"192":1,"195":1,"201":2,"203":1,"205":1,"206":2,"207":3,"208":3,"209":1,"213":3,"214":2,"215":2,"221":1,"223":3,"225":1,"232":1,"233":2,"235":1,"236":1,"237":5,"243":5,"244":1,"247":1,"248":1,"249":2,"253":1,"255":2,"256":1,"257":1,"258":3,"260":7,"264":1,"265":4,"266":2,"267":4,"268":4,"272":1,"276":1,"278":2,"279":1,"280":2,"281":1,"285":1,"287":1,"288":3,"289":1,"290":2,"292":1,"294":1,"295":1,"297":1,"298":1,"302":1,"303":3,"304":3,"305":1,"307":2,"309":3,"310":5,"312":2,"313":1,"315":1,"316":1,"317":1,"319":2,"320":2,"322":1,"323":6,"331":1,"334":1,"340":2,"341":1,"342":1,"346":1,"347":1,"348":1,"351":4,"352":1,"353":1,"354":1,"356":1,"357":1,"359":1,"360":5,"365":1,"366":1,"367":3,"368":2,"369":1,"370":3,"372":1,"373":9,"375":2,"376":2,"377":2,"379":2,"380":2,"384":2,"385":3,"386":12,"387":4,"388":1,"392":4,"398":2,"402":2,"403":1,"404":3,"405":4,"406":3,"408":1,"409":1,"410":1,"412":2,"413":2,"414":2,"415":1,"416":4,"417":3,"418":2,"419":7,"421":2,"422":4,"423":2,"424":3,"425":5,"431":1,"435":1,"440":1,"441":1,"442":8,"445":2,"449":2,"450":2,"452":1,"453":1,"457":2,"458":1,"459":1,"463":1,"464":1,"465":2,"469":1,"471":4,"475":1,"476":1,"479":3,"481":1,"483":1,"484":1,"485":6,"488":4,"491":2,"494":1,"501":2,"502":4,"503":1,"505":1,"506":1,"509":5,"510":1,"511":3,"515":1,"517":1,"522":3,"526":2,"527":3,"529":1,"530":1,"531":5,"534":1,"535":1,"536":1,"537":2,"539":2,"540":2,"541":3,"542":1,"543":3,"545":1,"546":1,"547":2,"548":2,"550":1,"551":3,"552":3,"555":1,"556":1,"557":2,"559":1,"560":2,"565":4,"566":2,"569":1,"571":3,"573":1,"574":1,"575":3,"576":2,"577":2,"578":3,"579":1,"583":3,"584":1,"585":2,"586":1,"588":4,"592":2,"598":1,"599":1,"600":1,"601":1,"603":1,"606":1,"608":2,"610":2,"611":3,"614":3,"616":1,"617":1,"620":11,"624":1,"625":2,"626":5,"627":2,"628":1,"629":2,"630":4,"631":2,"635":2,"637":1,"639":1,"644":1,"645":3,"646":1,"648":2,"649":1,"650":1,"653":1,"655":1,"656":3,"660":8,"661":3,"666":2,"670":1,"671":1,"672":1,"679":6,"681":1,"682":1,"683":1,"684":1,"685":3,"688":4,"690":1,"692":1,"694":1,"695":1,"698":1,"701":1,"705":1,"710":1,"712":1,"715":1,"718":3,"721":1,"722":1,"723":1,"724":1,"730":2}}],["tls",{"0":{"547":1},"2":{"547":7}}],["tls=false",{"2":{"547":2}}],["tls=true",{"2":{"545":2,"546":2,"547":2}}],["tls=",{"2":{"544":2}}],["tld",{"2":{"334":1}}],["tb",{"2":{"479":1,"614":1}}],["tbd",{"2":{"127":2,"278":1}}],["tcp",{"2":{"150":2,"151":2,"243":2,"244":2,"324":6,"470":4,"531":2,"541":2,"542":4,"552":1,"607":4,"715":2}}],["tmproto",{"2":{"155":2,"161":2,"392":6}}],["tm",{"2":{"149":2,"278":1}}],["txoption",{"2":{"392":2}}],["txtsudo",{"2":{"499":1}}],["txt",{"2":{"491":1,"499":5,"592":10}}],["txtcelestia",{"2":{"491":1}}],["txtnode",{"2":{"370":1}}],["txtvite",{"2":{"266":1}}],["txreceipt",{"2":{"351":2}}],["txdata",{"2":{"351":8}}],["txmgr",{"2":{"351":5}}],["txhash",{"2":{"329":1,"522":2,"721":3,"727":2}}],["txhashcelestia",{"2":{"329":1}}],["txsizecost",{"2":{"386":1}}],["txsizecostperbyte",{"2":{"385":1}}],["txsharerange",{"2":{"170":1}}],["txs",{"2":{"134":4,"137":6,"351":2,"488":2}}],["tx",{"2":{"91":6,"151":14,"159":4,"161":2,"268":4,"347":2,"348":2,"384":1,"385":1,"390":2,"424":1,"488":4,"493":6,"494":6,"495":2,"496":2,"497":4,"498":2,"507":10,"522":8,"529":4,"610":4,"629":10,"637":2,"721":7,"727":6}}],["ts",{"2":{"89":1,"342":1,"359":1}}],["tsx",{"2":{"77":2,"273":2}}],["tuning",{"2":{"425":1}}],["tuplerootnonce",{"2":{"153":2,"159":5}}],["tuple",{"2":{"127":10,"128":1,"136":1,"137":1,"148":4,"150":3,"151":2,"153":14,"159":17,"162":1,"166":3,"167":1,"172":1,"173":4}}],["tutorial",{"0":{"296":1,"298":1,"299":1},"1":{"300":1,"301":1,"302":1,"303":1,"304":1,"305":1,"306":1,"307":1,"308":1,"309":1,"310":1,"311":1,"312":1,"313":1,"314":1,"315":1,"316":1,"317":1,"318":1,"319":1,"320":1,"321":1,"322":1,"323":1,"324":1,"325":1,"326":1,"327":1,"328":1,"329":1,"330":1,"331":1,"332":1,"333":1,"334":1,"335":1,"336":1,"337":1},"2":{"79":1,"113":1,"115":1,"122":1,"234":3,"245":1,"247":1,"254":1,"256":1,"257":1,"258":1,"264":3,"271":1,"272":1,"280":1,"296":1,"298":1,"299":1,"304":1,"311":1,"335":1,"336":1,"365":3,"367":1,"375":2,"391":1,"426":1,"477":1,"480":1,"481":1,"483":1,"487":1,"511":1,"526":2,"530":1,"535":1,"538":1,"539":1,"543":5,"559":1,"561":1,"573":1,"574":1,"577":1,"589":1,"591":1,"603":1,"612":1,"615":1,"616":1,"617":1,"622":1,"629":1,"654":1,"657":1,"664":1,"665":1,"696":1,"717":1,"719":1,"720":1,"723":1,"726":1,"727":1}}],["tutorials",{"0":{"381":1},"2":{"0":1,"3":1,"24":1,"223":1,"381":2,"696":1}}],["turned",{"2":{"597":1}}],["turn",{"2":{"48":1,"68":1}}],["twitch",{"2":{"46":1}}],["twitter",{"2":{"27":1,"45":2,"46":1}}],["two",{"2":{"45":1,"66":1,"71":1,"116":1,"117":1,"146":2,"152":2,"166":1,"167":1,"173":1,"209":1,"213":1,"229":1,"248":1,"249":1,"290":1,"292":1,"351":1,"382":1,"415":1,"419":1,"424":1,"438":1,"478":1,"494":1,"600":1,"601":1,"625":1,"626":2,"627":1,"634":1,"662":1,"700":1,"709":1}}],["tasked",{"2":{"407":1}}],["tasks",{"2":{"30":1,"91":1,"304":1,"422":1,"585":1}}],["tar",{"2":{"304":4,"305":32,"585":4,"586":32,"602":24}}],["targets",{"2":{"503":6}}],["targetblock",{"2":{"379":1}}],["target",{"2":{"149":1,"201":1,"268":4,"445":1,"488":2,"556":1,"712":4,"714":4,"715":4,"716":4}}],["targeting",{"2":{"88":1}}],["taking",{"2":{"107":1,"155":1,"156":1,"159":1,"167":1,"600":1}}],["takes",{"2":{"107":1,"117":1,"150":1,"152":1,"153":1,"249":1,"317":1,"373":1,"378":1}}],["takeaways",{"2":{"38":1}}],["take",{"2":{"9":1,"53":1,"107":1,"116":1,"131":1,"134":1,"154":1,"158":1,"173":1,"189":1,"192":1,"248":1,"263":2,"416":1,"421":1,"425":1,"530":1,"551":1,"571":1,"601":1}}],["tab=staking",{"2":{"452":1}}],["tables",{"2":{"260":2}}],["table",{"0":{"286":1},"2":{"84":1,"449":1,"450":1,"471":1}}],["tab",{"2":{"69":2,"71":2,"219":1,"237":1}}],["tag",{"2":{"45":1,"243":2,"307":6,"539":12,"559":6,"577":6,"579":6,"580":6}}],["tags",{"2":{"36":1,"307":6,"341":2,"539":6,"559":6}}],["talented",{"2":{"43":1}}],["talks",{"2":{"40":1,"43":1,"46":1,"48":5}}],["talk",{"2":{"40":1,"43":1}}],["tailored",{"2":{"53":1,"65":1}}],["tailor",{"2":{"42":1,"44":1,"48":1}}],["tax",{"2":{"32":6,"34":2}}],["taxes",{"0":{"32":1},"2":{"31":1,"32":1}}],["t",{"0":{"412":1},"2":{"27":1,"35":1,"40":1,"43":1,"45":1,"46":3,"71":1,"88":1,"89":1,"107":2,"117":3,"139":1,"145":1,"151":2,"154":1,"155":1,"156":1,"158":1,"159":3,"172":1,"181":1,"207":1,"231":1,"232":1,"233":2,"249":3,"260":1,"297":2,"313":1,"337":1,"392":2,"398":2,"406":1,"407":2,"412":2,"453":1,"515":4,"527":1,"529":1,"561":1,"607":1,"610":1,"612":1,"628":1,"632":1,"671":1,"689":1,"690":1,"701":1,"709":1}}],["tiny",{"2":{"400":1}}],["tia",{"0":{"426":1,"428":1,"431":1,"432":1,"433":1,"435":1,"436":1,"449":1,"454":1,"455":1,"456":1,"462":1},"1":{"427":1,"428":1,"429":2,"430":2,"431":2,"432":2,"433":1,"434":2,"435":2,"436":2,"450":1,"455":1,"456":1,"457":2,"458":2,"459":2,"460":2,"461":2,"462":2,"463":2,"464":2},"2":{"329":2,"398":6,"400":1,"426":3,"431":1,"444":1,"445":3,"447":1,"448":1,"449":1,"450":3,"455":2,"457":2,"458":1,"459":1,"460":1,"462":1,"463":2}}],["ticketing",{"2":{"275":1}}],["title",{"2":{"260":24,"268":12,"494":6,"495":2}}],["ti",{"2":{"243":2,"244":2}}],["tips",{"2":{"344":1,"440":1}}],["tip",{"2":{"76":1,"89":1,"237":1,"243":1,"285":1,"302":1,"308":1,"309":1,"310":1,"316":1,"348":2,"367":1,"378":1,"485":1,"522":2,"537":1,"550":1,"554":2,"603":1,"620":1,"628":1,"653":1,"660":1,"679":1,"724":1}}],["timing",{"2":{"65":1,"388":1}}],["timeout",{"2":{"360":2,"392":2,"522":2,"629":4}}],["timeseries",{"2":{"503":2}}],["timestamp",{"2":{"326":2,"347":2,"445":1,"522":3,"721":2,"727":2}}],["times",{"2":{"34":1,"416":1,"624":1,"701":1}}],["timely",{"2":{"25":1}}],["time",{"2":{"14":1,"15":1,"31":2,"34":2,"36":1,"45":1,"84":2,"88":1,"116":1,"139":1,"207":1,"231":1,"248":1,"256":1,"283":1,"326":2,"353":1,"374":1,"378":1,"385":1,"388":2,"392":2,"409":1,"424":1,"445":1,"503":2,"509":1,"515":1,"522":2,"523":4,"530":1,"561":1,"625":1,"629":4,"631":1,"712":1,"716":1}}],["tiers",{"2":{"23":1}}],["tier",{"2":{"23":1,"65":1}}],["tr",{"2":{"592":10,"602":6}}],["try",{"2":{"262":1,"312":1,"337":1,"357":1,"398":2,"638":1}}],["trying",{"2":{"107":1,"337":1,"556":1,"557":1}}],["trpc",{"2":{"150":4,"151":8,"152":2,"161":2}}],["trees",{"0":{"419":1},"2":{"302":1,"415":1,"419":1}}],["tree",{"2":{"108":2,"127":2,"128":1,"148":2,"155":6,"158":6,"186":1,"419":2,"421":2,"424":1}}],["treated",{"2":{"33":1,"385":1}}],["tries",{"2":{"425":1}}],["triggers",{"2":{"352":1}}],["triggered",{"2":{"255":3}}],["triggering",{"2":{"177":1,"352":1}}],["trigger",{"2":{"177":1,"352":1}}],["trick",{"2":{"108":1}}],["tripods",{"2":{"40":1}}],["truth",{"2":{"148":1}}],["true",{"2":{"89":6,"127":2,"151":2,"260":6,"267":4,"319":1,"502":4,"541":2,"547":1,"552":2,"568":1,"569":1,"611":1,"629":10}}],["trusts",{"2":{"571":1}}],["trustedpeers",{"2":{"571":1}}],["trustedpeer",{"0":{"571":1}}],["trustedhash",{"0":{"571":1},"2":{"571":1}}],["trusted",{"2":{"56":1,"478":2,"571":5,"601":2,"629":4,"674":1,"675":1,"676":1,"689":1,"690":1,"691":1}}],["trust",{"2":{"45":1,"415":1,"425":1,"601":4,"629":4}}],["trustworthy",{"2":{"19":1}}],["troubles",{"2":{"679":1}}],["troubleshooting",{"0":{"337":1,"549":1,"583":1},"1":{"550":1,"551":1,"552":1,"553":1,"554":1,"555":1,"556":1,"557":1},"2":{"60":1,"93":1,"237":1,"243":2,"309":1,"310":1,"367":1,"484":1,"485":1,"537":1,"562":1,"577":1,"603":1,"620":2,"653":1,"660":2,"724":1}}],["trolling",{"2":{"8":1}}],["traditional",{"2":{"186":1,"213":1,"290":1,"422":1}}],["trackers",{"2":{"504":1}}],["track",{"2":{"36":1,"108":1,"136":1,"367":1,"544":1}}],["transform",{"2":{"268":2}}],["transferred",{"2":{"637":1}}],["transferring",{"2":{"34":1,"91":2}}],["transferhermes",{"2":{"634":1}}],["transferownership",{"2":{"260":2}}],["transfers",{"2":{"260":2}}],["transfer",{"0":{"329":1,"637":1},"1":{"638":1},"2":{"34":6,"76":1,"84":1,"218":1,"272":1,"329":7,"450":1,"493":1,"627":1,"629":4,"634":7,"635":4,"637":6,"638":1}}],["transact",{"2":{"255":1,"452":1}}],["transactopts",{"2":{"161":2}}],["transactionindex",{"2":{"121":2,"253":2,"348":2}}],["transactionhash",{"2":{"121":2,"253":2}}],["transactions",{"0":{"388":1,"438":1,"597":1},"2":{"33":2,"34":2,"66":1,"69":2,"71":1,"72":1,"79":1,"85":2,"86":1,"90":1,"136":1,"139":1,"143":1,"146":1,"206":1,"215":1,"228":3,"229":2,"232":1,"255":1,"263":1,"264":1,"265":1,"278":1,"279":1,"289":1,"301":1,"310":1,"347":3,"351":4,"365":1,"380":2,"382":2,"383":2,"388":1,"392":4,"394":3,"395":1,"396":2,"399":1,"400":1,"402":1,"409":1,"415":1,"416":1,"417":1,"421":1,"422":6,"423":1,"424":6,"425":2,"438":2,"439":2,"457":1,"463":1,"471":1,"485":1,"488":2,"508":1,"597":1,"610":5,"611":1,"620":2,"660":1,"661":1,"665":1,"679":1,"688":1,"689":1,"691":1,"729":1}}],["transaction",{"0":{"152":1,"347":1,"348":1,"394":1,"424":1,"595":1,"610":1,"649":1,"729":1},"1":{"395":1,"396":1,"425":1},"2":{"33":1,"34":2,"69":4,"71":3,"72":3,"83":3,"84":3,"85":4,"91":14,"120":2,"136":1,"139":1,"146":3,"151":8,"152":3,"154":1,"155":1,"159":2,"161":4,"162":1,"165":1,"170":2,"173":1,"178":2,"189":1,"229":2,"232":2,"233":1,"252":2,"255":1,"317":3,"319":3,"320":1,"329":1,"334":1,"347":4,"348":2,"351":5,"352":3,"373":2,"374":1,"375":1,"382":1,"384":2,"385":3,"386":10,"387":1,"388":8,"392":14,"395":2,"402":2,"403":3,"404":1,"408":3,"409":3,"410":2,"411":2,"413":2,"414":2,"415":1,"422":1,"424":3,"431":1,"432":1,"435":2,"436":1,"438":4,"439":2,"506":1,"507":4,"511":1,"522":3,"529":1,"531":2,"594":1,"597":1,"649":1,"671":4,"727":3,"729":1}}],["transitions",{"2":{"139":1,"177":1,"185":1,"189":1,"192":1,"422":1}}],["transition",{"2":{"19":1,"161":4,"177":2,"181":1}}],["translations",{"2":{"17":2}}],["tee",{"2":{"712":2,"714":2,"715":2,"716":2}}],["territories",{"2":{"667":1}}],["terminal",{"2":{"75":2,"76":2,"272":4,"304":1,"334":1,"346":1,"540":2,"560":2,"623":1,"659":1}}],["termination",{"2":{"34":1}}],["term",{"2":{"34":1,"445":1}}],["terms",{"0":{"30":1,"31":1},"1":{"31":1,"32":1,"33":1,"34":1},"2":{"14":1,"15":1,"30":15,"31":1,"33":2,"34":7}}],["teach",{"2":{"311":1}}],["team",{"2":{"27":1,"36":1,"41":1,"43":1,"45":1,"52":1,"79":1,"91":1,"168":1,"189":1,"192":1,"195":1,"377":1,"478":1,"624":1,"625":1}}],["text|json",{"2":{"531":2}}],["text$request",{"2":{"310":1,"473":1,"537":1,"693":1}}],["textarea",{"2":{"268":2}}],["textalign",{"2":{"268":4}}],["text",{"2":{"268":2,"317":2,"319":1,"494":1,"495":1,"531":2,"651":1}}],["testtia",{"2":{"647":1}}],["testkey",{"2":{"646":1}}],["test3",{"2":{"507":4}}],["test2sig",{"2":{"507":4}}],["test2",{"2":{"507":8}}],["test1sig",{"2":{"507":4}}],["test1",{"2":{"507":8}}],["testcelestia",{"2":{"489":1,"527":1,"536":1}}],["testcreatepost",{"2":{"260":2}}],["testonlyowner",{"2":{"260":2}}],["testfetchposts",{"2":{"260":2}}],["testforge",{"2":{"117":1,"249":1}}],["testupdatepost",{"2":{"260":2}}],["teststaking",{"2":{"649":1}}],["tests",{"0":{"260":1},"2":{"117":4,"249":4,"261":1}}],["testsetnumber",{"2":{"117":6,"249":6}}],["testincrement",{"2":{"117":6,"249":6}}],["testing",{"0":{"270":1,"352":1},"2":{"113":1,"117":1,"119":1,"217":1,"249":1,"352":1,"465":3,"466":1,"625":1,"666":1,"681":1,"698":1}}],["test",{"0":{"117":1,"249":1,"261":1},"2":{"117":14,"120":1,"155":1,"156":1,"158":1,"159":1,"214":2,"237":6,"238":6,"239":6,"243":4,"244":4,"249":14,"251":2,"260":7,"261":2,"262":1,"263":2,"264":3,"269":1,"270":3,"271":1,"291":1,"310":4,"355":1,"360":2,"367":3,"371":8,"485":3,"486":1,"489":1,"503":1,"507":6,"517":1,"527":1,"531":2,"536":1,"620":3,"621":1,"629":4,"631":1,"639":2,"646":1,"647":1,"649":1,"661":4,"663":1,"685":2,"703":1,"707":1,"709":1}}],["tested",{"2":{"94":1,"262":1,"264":1,"281":1,"285":1,"353":1}}],["testnetwork",{"2":{"639":1}}],["testnets",{"0":{"700":1},"1":{"701":1,"702":1,"703":1,"704":1},"2":{"30":1,"80":1,"101":1,"210":1,"233":1,"398":1,"400":1,"625":2,"630":1,"700":1}}],["testnet",{"0":{"30":1,"89":1,"212":1,"245":1,"272":1,"453":1,"546":1,"643":1,"662":1,"685":1,"693":1,"703":1,"704":1},"1":{"31":1,"32":1,"33":1,"34":1,"213":1,"214":1,"215":1,"216":1,"217":1,"218":1,"219":1,"220":1,"221":1,"222":1,"246":1,"247":1,"248":1,"249":1,"250":1,"251":1,"252":1,"253":1,"254":1,"273":1,"274":1,"275":1,"644":1,"645":1,"646":1,"647":1,"648":1,"649":1,"650":1,"651":1,"652":1,"653":1,"686":1,"687":1,"688":1,"689":1,"690":1,"691":1,"692":1,"693":1,"694":1,"695":1},"2":{"24":3,"27":1,"30":5,"68":1,"79":2,"80":3,"82":5,"85":5,"88":1,"95":1,"210":4,"212":1,"213":5,"214":1,"215":1,"217":3,"218":2,"219":1,"222":3,"243":1,"245":1,"246":1,"254":1,"257":1,"263":1,"267":4,"272":2,"286":3,"291":2,"307":1,"309":1,"310":2,"337":2,"398":2,"400":5,"449":1,"453":4,"465":3,"466":1,"484":1,"485":4,"511":1,"517":1,"525":1,"526":2,"529":1,"533":1,"546":1,"592":1,"620":4,"625":1,"626":1,"627":2,"629":11,"630":3,"632":1,"633":6,"634":8,"635":3,"639":1,"643":2,"645":1,"648":2,"653":1,"660":1,"661":1,"662":2,"685":1,"689":3,"690":3,"691":3,"693":1,"694":4,"703":2,"709":1,"724":1}}],["temp",{"2":{"540":1,"560":1}}],["temporarily",{"2":{"509":1}}],["temporary",{"0":{"15":1},"2":{"14":1,"15":1,"255":1,"517":1}}],["template",{"2":{"353":1}}],["templates",{"2":{"89":1}}],["tendermintrpc",{"2":{"107":1}}],["tendermint",{"2":{"89":3,"149":4,"150":4,"151":10,"161":22,"278":2,"279":1,"280":1,"282":1,"283":2,"326":4,"377":1,"380":1,"392":6,"421":6,"478":1,"488":8,"500":2,"531":2,"605":2,"628":2,"634":8,"644":2,"652":2,"727":2}}],["tell",{"2":{"48":1}}],["telegram",{"2":{"45":2,"476":1,"505":1,"684":1,"695":1,"705":1}}],["television",{"2":{"34":1}}],["telemetry",{"2":{"33":1,"629":6}}],["tech",{"2":{"42":2,"44":1,"105":1,"674":1,"675":1,"676":1,"677":1}}],["technologies",{"2":{"48":1,"59":1,"232":1,"667":1}}],["technology",{"2":{"8":1,"42":1,"289":1,"365":1}}],["technical",{"2":{"3":1,"27":1,"31":1,"36":1,"48":1}}],["typically",{"2":{"203":1,"462":1}}],["typical",{"2":{"136":1,"229":1,"283":2}}],["type=$node",{"2":{"577":6,"579":6,"580":6}}],["type=full",{"2":{"577":1}}],["type=fullexport",{"2":{"577":1}}],["type=bridge",{"2":{"577":1}}],["type=bridgeexport",{"2":{"577":1}}],["type=1",{"2":{"353":1}}],["type=spanbatch",{"2":{"353":1}}],["type=lightexport",{"2":{"577":1}}],["type=light",{"2":{"243":2,"244":2,"360":2,"577":1}}],["type",{"0":{"83":1,"355":1},"1":{"356":1,"357":1,"358":1},"2":{"117":1,"121":2,"134":8,"135":7,"155":2,"158":2,"176":1,"228":1,"235":1,"237":7,"238":6,"239":6,"243":2,"244":6,"249":1,"253":2,"310":4,"314":4,"326":4,"334":2,"353":2,"367":2,"399":1,"400":1,"403":2,"469":1,"470":2,"471":2,"485":2,"494":4,"519":4,"520":8,"522":2,"523":3,"524":5,"542":2,"544":2,"545":2,"546":2,"547":4,"550":4,"553":12,"554":4,"555":2,"556":4,"561":1,"565":1,"577":13,"579":8,"580":8,"620":2,"629":12,"661":4,"672":1,"679":2,"685":3,"696":1}}],["typescript",{"2":{"342":1,"359":1}}],["types",{"2":{"3":1,"42":1,"161":6,"171":1,"181":1,"232":1,"308":1,"351":2,"387":2,"392":4,"413":1,"469":1,"470":1,"471":1,"541":1,"552":1,"561":1,"583":1,"672":2,"679":1,"685":1,"688":1,"707":1}}],["typography",{"2":{"58":1}}],["typos",{"2":{"2":1}}],["though",{"2":{"422":1,"424":1}}],["thought",{"2":{"134":2}}],["thorough",{"2":{"183":1}}],["those",{"2":{"8":1,"14":1,"15":1,"39":1,"40":1,"46":3,"49":1,"134":1,"135":1,"148":1,"166":1,"228":2,"229":1,"278":1,"282":1,"381":1,"411":1,"413":1,"416":1,"494":1,"624":1,"625":3,"626":2,"650":1,"685":1}}],["thus",{"2":{"88":1,"89":1,"107":1,"109":1,"418":1,"422":3,"424":2}}],["thank",{"2":{"44":1,"79":1}}],["thanks",{"2":{"39":1,"106":1}}],["thanking",{"2":{"38":1}}],["than",{"2":{"25":1,"31":1,"34":2,"76":1,"205":1,"213":1,"229":1,"237":1,"272":1,"283":1,"290":1,"383":1,"411":1,"441":1,"442":1,"445":2,"502":2,"509":1,"529":1,"553":1,"557":1,"671":1}}],["that",{"0":{"151":1,"161":1,"177":1,"178":1,"179":1,"180":1,"181":1,"182":1,"183":1,"185":1,"186":1,"187":1,"188":1,"190":1,"191":1,"406":1},"1":{"178":1,"179":1,"180":1,"182":1,"183":1,"186":1,"187":1,"189":1,"190":1,"191":1},"2":{"7":1,"8":1,"9":2,"19":1,"22":1,"24":4,"30":2,"31":9,"32":3,"33":1,"34":21,"35":1,"36":1,"37":1,"40":2,"41":1,"42":1,"44":1,"48":1,"53":2,"65":3,"76":2,"78":1,"84":3,"85":2,"87":1,"89":6,"91":2,"94":1,"102":2,"103":1,"105":1,"107":3,"108":4,"109":4,"117":3,"126":2,"127":4,"129":2,"130":1,"131":13,"133":2,"134":18,"135":18,"136":4,"137":2,"139":6,"146":3,"148":14,"150":1,"151":6,"152":2,"153":14,"154":2,"155":1,"156":1,"159":11,"161":11,"162":2,"163":3,"165":1,"166":7,"167":1,"168":1,"170":1,"172":7,"173":8,"176":6,"177":6,"178":2,"180":1,"181":4,"183":1,"184":1,"185":4,"186":2,"189":3,"192":6,"195":2,"199":1,"201":1,"204":1,"206":4,"207":4,"208":1,"213":2,"214":1,"215":1,"218":1,"222":1,"223":1,"225":1,"228":2,"229":2,"231":1,"232":3,"233":3,"234":1,"235":1,"237":3,"243":1,"249":3,"252":1,"255":1,"257":3,"262":1,"263":1,"264":2,"265":2,"266":1,"267":1,"268":2,"270":2,"271":1,"272":2,"274":1,"278":1,"280":1,"282":1,"283":4,"289":1,"290":1,"291":1,"304":1,"305":1,"307":1,"309":1,"310":1,"312":5,"313":1,"314":1,"319":1,"329":1,"337":1,"347":1,"351":3,"352":2,"353":1,"357":1,"365":2,"367":1,"370":1,"372":1,"373":8,"374":3,"375":1,"377":1,"379":1,"380":1,"381":1,"383":1,"384":3,"385":1,"386":5,"388":2,"392":6,"398":5,"399":1,"400":1,"402":4,"404":1,"405":1,"406":8,"407":2,"409":1,"410":1,"411":1,"412":1,"413":6,"414":1,"415":2,"416":3,"417":2,"418":1,"419":6,"421":4,"422":9,"423":4,"424":5,"425":4,"426":1,"435":1,"439":1,"440":5,"441":2,"442":2,"450":1,"458":1,"462":1,"465":2,"471":1,"478":1,"484":1,"485":1,"487":1,"494":5,"495":2,"503":5,"506":1,"509":3,"510":2,"512":1,"515":1,"517":2,"523":2,"526":1,"529":1,"531":1,"537":1,"540":1,"541":4,"542":1,"544":1,"548":1,"552":4,"554":2,"556":1,"557":2,"559":1,"560":1,"561":1,"564":1,"571":1,"572":1,"574":1,"575":2,"578":1,"585":1,"586":1,"588":1,"592":2,"598":1,"600":1,"611":1,"612":1,"613":1,"620":2,"622":1,"624":1,"626":1,"627":2,"628":2,"630":3,"632":2,"634":1,"635":1,"650":1,"652":2,"655":1,"659":1,"661":1,"666":1,"667":1,"671":3,"679":1,"685":1,"688":3,"700":1,"708":1,"709":1,"712":1,"721":1,"726":1,"727":3}}],["think",{"2":{"134":1}}],["things",{"0":{"414":1},"2":{"34":1,"77":1,"91":2,"267":1,"273":1,"317":1,"355":1,"373":1}}],["third",{"2":{"23":4,"31":10,"33":5,"34":1,"276":1,"602":1}}],["this",{"0":{"411":1},"2":{"1":1,"2":2,"4":1,"9":1,"10":1,"12":1,"14":1,"15":1,"17":2,"23":4,"24":1,"27":1,"33":2,"42":2,"43":3,"44":2,"45":1,"46":2,"47":1,"48":7,"49":1,"50":1,"52":1,"53":1,"55":1,"56":2,"57":2,"59":1,"60":2,"61":2,"63":1,"64":1,"65":1,"66":2,"69":1,"76":2,"77":1,"79":4,"81":1,"82":2,"83":4,"84":7,"85":4,"86":2,"88":3,"89":1,"90":2,"91":2,"92":1,"94":3,"102":2,"103":1,"107":3,"108":3,"109":1,"113":1,"115":1,"116":1,"117":3,"126":1,"131":5,"133":4,"134":8,"135":6,"136":3,"137":3,"139":1,"148":4,"149":1,"150":2,"152":2,"154":3,"155":3,"159":3,"161":4,"163":1,"165":1,"167":1,"169":2,"172":5,"173":3,"177":3,"181":2,"184":1,"185":2,"191":1,"196":2,"201":2,"206":2,"213":5,"217":1,"223":2,"232":1,"233":1,"234":3,"236":1,"237":1,"243":1,"244":3,"245":1,"247":1,"248":1,"249":3,"255":2,"256":3,"257":1,"260":3,"262":1,"263":1,"268":4,"271":1,"272":2,"273":1,"275":2,"276":1,"278":1,"280":2,"282":1,"283":1,"285":2,"287":1,"288":2,"290":5,"291":2,"299":1,"302":1,"304":4,"307":2,"308":1,"309":2,"311":1,"312":1,"315":1,"316":2,"317":1,"319":3,"324":1,"325":1,"326":2,"329":1,"332":1,"334":1,"338":1,"343":1,"347":1,"351":3,"352":2,"353":2,"356":1,"357":4,"360":3,"365":3,"367":1,"368":1,"371":1,"375":2,"379":1,"384":1,"385":3,"386":3,"387":3,"388":4,"392":10,"394":1,"395":1,"397":1,"398":4,"402":1,"403":1,"404":1,"405":1,"406":2,"414":1,"417":2,"418":2,"419":2,"421":1,"422":4,"423":1,"424":2,"425":2,"426":2,"429":1,"439":1,"440":1,"441":2,"442":4,"450":1,"458":1,"463":1,"469":1,"471":1,"472":1,"477":1,"478":2,"485":1,"494":3,"499":2,"503":7,"504":1,"507":4,"509":2,"511":2,"515":2,"517":1,"519":2,"522":1,"526":2,"530":1,"531":5,"534":1,"535":1,"536":1,"537":1,"538":1,"539":1,"540":1,"542":2,"543":3,"547":2,"548":2,"550":2,"553":2,"554":1,"555":1,"557":5,"559":4,"560":1,"566":2,"569":2,"573":1,"574":1,"575":2,"578":4,"579":1,"583":1,"584":2,"585":2,"587":1,"588":1,"592":1,"594":1,"596":2,"598":1,"599":3,"600":1,"601":2,"605":1,"606":1,"607":2,"608":1,"610":7,"611":3,"612":1,"623":1,"624":1,"626":3,"627":2,"629":1,"632":2,"634":2,"639":2,"643":1,"645":1,"646":2,"649":1,"650":5,"654":1,"655":1,"657":1,"660":1,"665":1,"666":1,"667":1,"671":1,"672":1,"679":1,"685":2,"690":1,"693":1,"696":3,"698":1,"706":1,"709":1,"715":1,"717":1,"722":1,"727":1}}],["thread",{"2":{"405":1}}],["threatening",{"2":{"9":1}}],["threshold",{"2":{"384":1,"506":1,"507":2,"629":4}}],["three",{"2":{"209":1,"320":1,"419":2,"422":1}}],["throughput",{"2":{"201":1,"205":1,"206":1,"417":1,"422":2}}],["throughout",{"2":{"34":1,"52":1,"365":1}}],["through",{"2":{"14":1,"33":1,"34":1,"38":1,"45":1,"72":1,"89":1,"91":2,"208":1,"290":1,"365":1,"385":1,"386":1,"395":1,"451":1,"460":1,"502":1,"503":1,"535":1,"538":1,"553":1,"612":1,"637":1,"654":1,"696":1,"714":1,"715":1,"717":1,"721":1}}],["theta",{"2":{"629":9,"630":2,"632":1,"633":6,"634":8,"635":3}}],["then",{"2":{"68":1,"69":1,"84":1,"85":1,"90":1,"107":1,"108":1,"117":1,"133":2,"134":1,"136":2,"139":1,"148":5,"150":1,"152":3,"155":1,"156":1,"160":1,"161":1,"162":1,"166":1,"172":1,"173":1,"177":2,"181":1,"189":3,"192":1,"195":1,"208":1,"213":2,"228":1,"239":1,"249":1,"263":2,"264":1,"268":4,"269":1,"290":2,"304":2,"315":1,"319":1,"329":1,"337":1,"345":1,"367":1,"373":3,"374":1,"375":1,"376":1,"385":2,"386":1,"387":1,"388":1,"392":2,"399":1,"400":1,"402":1,"416":2,"425":3,"435":1,"438":1,"439":1,"491":2,"494":2,"500":3,"503":3,"542":1,"548":2,"553":1,"554":1,"556":1,"568":1,"585":2,"601":1,"650":1}}],["therefore",{"2":{"135":1,"380":1,"392":2,"402":1,"417":1,"466":1}}],["therefor",{"2":{"34":1}}],["therefrom",{"2":{"34":1}}],["thereof",{"2":{"34":2}}],["thereto",{"2":{"31":1,"34":2}}],["there",{"0":{"413":1},"2":{"23":1,"42":1,"45":1,"50":1,"84":1,"116":1,"117":1,"139":1,"167":1,"181":1,"184":1,"217":1,"229":1,"231":1,"232":1,"248":1,"249":1,"275":1,"280":1,"290":1,"292":1,"309":1,"317":1,"384":1,"406":1,"407":2,"412":1,"413":1,"416":1,"417":1,"422":1,"472":1,"475":1,"494":3,"551":1,"600":1,"627":1,"671":2,"681":1,"682":1,"693":1,"694":1,"696":1}}],["these",{"2":{"12":1,"14":1,"15":1,"30":7,"33":2,"34":9,"35":1,"39":1,"40":1,"41":1,"45":1,"50":1,"85":1,"86":1,"88":1,"89":1,"131":2,"134":1,"135":2,"146":1,"148":1,"152":1,"155":1,"158":1,"166":1,"167":2,"169":1,"173":2,"180":1,"185":2,"192":1,"207":1,"209":1,"229":1,"232":1,"256":2,"275":1,"277":1,"283":1,"304":2,"313":2,"320":1,"351":2,"377":1,"385":2,"387":1,"416":1,"422":2,"424":2,"442":1,"470":1,"495":1,"503":2,"529":1,"540":1,"542":1,"560":1,"583":1,"585":2,"601":1,"609":1,"611":1,"627":1,"643":1,"670":1,"671":1,"679":2,"688":1,"727":1}}],["they",{"2":{"8":1,"9":1,"12":1,"23":1,"30":1,"31":2,"34":2,"40":1,"44":4,"45":2,"46":1,"47":1,"65":1,"84":2,"86":1,"139":1,"158":1,"167":1,"176":2,"177":2,"181":2,"215":1,"228":2,"229":1,"231":1,"233":1,"281":1,"312":1,"384":1,"386":1,"388":3,"394":1,"396":1,"398":2,"404":2,"406":2,"407":1,"413":1,"414":2,"417":1,"422":1,"442":1,"466":1,"495":1,"509":3,"624":1,"632":1,"649":1,"650":1,"652":1,"655":2,"679":1,"688":1,"692":1,"703":1}}],["their",{"2":{"8":1,"28":1,"31":1,"34":1,"36":1,"38":1,"40":2,"42":2,"43":2,"44":6,"45":3,"46":2,"47":2,"48":1,"53":1,"65":1,"84":2,"133":1,"148":1,"162":1,"163":3,"171":1,"176":1,"177":1,"201":1,"212":1,"213":1,"214":1,"223":1,"231":2,"232":2,"233":3,"256":1,"260":2,"267":1,"284":1,"288":1,"289":1,"290":1,"293":1,"295":1,"299":1,"302":1,"357":1,"384":1,"388":1,"396":1,"409":1,"413":1,"422":1,"423":2,"440":1,"442":4,"447":1,"458":3,"459":2,"465":1,"466":1,"471":1,"503":2,"509":3,"548":1,"597":1,"601":1,"624":1,"625":2,"648":1,"650":3,"679":1,"685":2,"688":1,"703":1}}],["theme",{"2":{"43":2}}],["them",{"2":{"2":1,"34":1,"38":1,"40":1,"44":4,"45":1,"46":3,"47":3,"102":1,"107":1,"130":1,"148":1,"155":3,"158":1,"162":1,"166":1,"170":1,"180":2,"213":1,"229":1,"260":2,"289":1,"290":1,"297":1,"302":1,"375":1,"380":1,"415":1,"444":1,"469":1,"509":1,"552":1,"554":1,"601":1,"649":1,"650":1,"672":1,"685":1,"688":1}}],["the",{"0":{"1":1,"18":1,"19":1,"36":1,"37":1,"38":1,"76":1,"77":1,"78":1,"87":1,"88":1,"130":1,"139":1,"146":1,"147":1,"148":1,"153":2,"161":1,"198":1,"213":1,"235":1,"260":1,"261":1,"262":1,"264":1,"265":1,"267":1,"273":1,"274":1,"285":1,"286":1,"288":1,"320":1,"323":1,"344":2,"348":1,"349":1,"352":1,"359":1,"360":1,"361":2,"362":1,"363":1,"370":1,"374":1,"388":1,"390":1,"391":1,"392":1,"403":1,"406":1,"409":1,"410":1,"424":1,"481":1,"482":1,"484":1,"485":1,"486":1,"487":1,"498":1,"513":1,"514":1,"515":1,"516":2,"523":1,"524":1,"525":2,"553":1,"555":1,"556":2,"579":1,"580":1,"590":1,"592":1,"603":1,"609":1,"616":1,"618":1,"619":1,"620":1,"621":1,"622":1,"623":1,"636":1,"649":1,"650":1,"653":1,"659":1,"660":1,"663":1,"697":1,"712":1,"722":1,"724":1,"725":1,"726":1,"727":1},"1":{"19":1,"20":1,"21":1,"22":1,"23":1,"24":1,"25":1,"26":1,"27":1,"28":1,"29":1,"147":1,"148":1,"154":2,"155":2,"156":2,"157":2,"158":2,"159":2,"160":2,"161":2,"236":1,"237":1,"238":1,"239":1,"240":1,"266":1,"286":1,"371":1,"372":1,"373":1,"374":1,"425":1,"483":1,"484":1,"485":1,"486":2,"487":1,"514":1,"515":1,"516":1,"517":1,"619":1,"620":1,"660":1,"661":1,"662":1,"663":1,"664":1,"698":1,"699":1,"700":1,"701":1,"702":1,"703":1,"704":1,"705":1,"723":1,"724":1,"725":1,"726":2},"2":{"0":2,"1":2,"2":3,"3":2,"4":2,"5":4,"8":4,"9":1,"10":1,"11":3,"12":1,"13":4,"14":2,"15":3,"16":1,"17":2,"19":5,"21":7,"22":3,"23":9,"24":16,"25":4,"26":3,"27":12,"28":2,"29":2,"30":24,"31":25,"32":7,"33":15,"34":105,"35":1,"36":8,"37":4,"38":6,"39":2,"40":12,"41":4,"42":13,"43":15,"44":14,"45":9,"46":8,"47":7,"48":15,"49":2,"50":5,"52":5,"53":5,"54":2,"55":1,"56":2,"57":3,"59":4,"60":3,"61":3,"62":4,"63":1,"64":2,"65":15,"68":5,"69":7,"71":9,"72":6,"73":1,"75":12,"76":9,"77":7,"79":8,"82":6,"83":6,"84":40,"85":13,"86":7,"87":6,"88":12,"89":31,"90":2,"91":46,"92":3,"94":8,"95":5,"102":10,"103":2,"105":5,"106":3,"107":31,"108":18,"109":5,"110":14,"111":1,"113":3,"115":2,"116":10,"117":27,"119":3,"120":4,"121":7,"125":1,"126":6,"127":7,"128":2,"129":4,"130":4,"131":34,"133":2,"134":54,"135":37,"136":25,"137":17,"139":34,"141":1,"142":2,"143":4,"145":2,"146":15,"148":29,"149":9,"150":10,"151":38,"152":16,"153":58,"154":8,"155":46,"156":12,"157":5,"158":24,"159":78,"160":5,"161":56,"162":6,"163":3,"165":4,"166":23,"167":15,"168":6,"169":8,"170":23,"171":3,"172":30,"173":47,"174":1,"176":4,"177":12,"178":6,"180":3,"181":23,"182":2,"183":5,"184":10,"185":29,"186":3,"188":2,"189":28,"191":1,"192":24,"194":1,"195":8,"196":2,"197":1,"198":1,"199":1,"201":7,"203":1,"204":4,"205":3,"206":15,"207":17,"208":9,"209":2,"210":3,"211":2,"213":21,"215":4,"217":4,"218":2,"219":2,"220":1,"221":2,"222":1,"223":3,"229":4,"231":6,"232":5,"233":6,"234":3,"235":4,"236":4,"237":13,"238":1,"239":1,"242":1,"243":13,"244":11,"245":1,"246":2,"247":2,"248":10,"249":27,"251":3,"252":4,"253":7,"254":1,"255":6,"256":2,"257":1,"260":29,"262":3,"263":10,"264":21,"265":9,"266":6,"267":4,"268":29,"269":3,"270":14,"272":23,"273":7,"276":1,"278":5,"279":2,"280":7,"281":2,"282":2,"283":5,"284":1,"285":1,"287":1,"288":9,"289":2,"290":15,"291":3,"292":5,"293":6,"294":1,"295":4,"296":3,"297":5,"298":1,"299":2,"301":1,"302":5,"303":1,"304":6,"305":4,"307":15,"308":2,"309":10,"310":16,"311":2,"312":16,"313":10,"314":8,"315":6,"316":4,"317":21,"318":1,"319":19,"320":13,"322":4,"323":6,"324":3,"326":4,"329":4,"330":1,"331":3,"332":4,"334":8,"335":1,"336":2,"337":2,"341":3,"343":3,"344":5,"345":4,"346":11,"347":2,"348":8,"349":2,"350":2,"351":35,"352":9,"353":1,"354":2,"355":1,"356":4,"357":6,"358":2,"360":7,"361":5,"362":2,"363":2,"364":1,"365":7,"366":1,"367":19,"368":5,"369":1,"370":5,"371":3,"372":3,"373":31,"374":15,"375":3,"376":5,"377":12,"378":1,"379":9,"380":6,"381":2,"383":1,"384":10,"385":26,"386":43,"387":6,"388":13,"391":1,"392":48,"393":3,"394":2,"395":1,"396":1,"398":19,"399":3,"400":3,"402":14,"403":9,"404":2,"405":2,"406":4,"407":7,"408":2,"409":3,"411":5,"412":6,"414":2,"415":4,"416":18,"417":9,"418":17,"419":32,"421":26,"422":24,"423":10,"424":47,"425":32,"426":3,"429":1,"430":2,"431":2,"432":2,"434":1,"435":4,"436":2,"438":16,"439":3,"440":4,"441":6,"442":10,"444":3,"445":9,"447":1,"449":3,"450":10,"451":2,"452":2,"453":2,"455":1,"457":2,"458":2,"459":1,"460":2,"462":1,"463":2,"464":1,"465":2,"466":3,"469":5,"470":1,"471":7,"473":2,"474":1,"477":3,"478":13,"479":2,"480":1,"481":2,"483":3,"484":12,"485":9,"486":5,"487":3,"488":10,"489":3,"491":6,"494":22,"495":9,"496":1,"497":2,"498":1,"499":3,"500":3,"501":1,"502":7,"503":26,"504":4,"505":5,"506":3,"507":14,"509":20,"510":2,"511":2,"512":1,"514":2,"515":10,"516":1,"517":5,"519":4,"521":5,"522":15,"523":4,"524":3,"525":7,"526":3,"527":3,"528":1,"529":7,"530":4,"531":8,"532":1,"533":1,"536":3,"537":9,"539":16,"540":10,"541":6,"542":2,"543":2,"544":6,"547":5,"548":11,"550":3,"551":2,"552":8,"553":4,"554":4,"555":1,"556":8,"557":12,"559":15,"560":9,"561":2,"562":3,"565":2,"566":3,"568":5,"569":3,"571":5,"572":1,"573":2,"574":1,"575":8,"577":9,"578":7,"579":9,"580":2,"583":3,"585":5,"586":4,"587":1,"588":4,"589":1,"590":1,"591":1,"592":10,"594":1,"596":1,"597":1,"598":3,"599":5,"600":6,"601":10,"602":5,"603":4,"606":1,"607":5,"608":3,"609":1,"610":6,"611":10,"612":1,"613":2,"614":2,"615":1,"616":1,"617":1,"619":1,"620":16,"621":5,"622":2,"623":4,"624":7,"625":8,"626":11,"627":5,"628":3,"629":5,"630":4,"631":5,"632":2,"633":2,"634":3,"635":2,"636":1,"637":5,"638":3,"639":1,"641":1,"642":1,"644":2,"645":5,"646":4,"647":2,"648":6,"649":6,"650":10,"651":4,"652":6,"653":3,"655":3,"656":1,"657":1,"658":1,"659":4,"660":12,"661":10,"662":3,"663":5,"664":2,"666":6,"667":2,"670":2,"671":16,"672":6,"679":6,"680":2,"681":1,"683":1,"684":1,"685":10,"688":3,"689":4,"690":5,"691":3,"692":2,"693":2,"696":7,"698":3,"701":1,"703":2,"706":1,"707":1,"708":3,"709":4,"711":1,"712":1,"714":2,"715":4,"716":5,"717":1,"718":1,"719":2,"720":1,"721":10,"722":3,"723":1,"724":6,"725":1,"726":3,"727":12,"728":1,"729":1,"730":2}}],["todo",{"2":{"372":2,"392":6}}],["today",{"2":{"210":1,"257":1,"499":2}}],["tomlvim",{"2":{"629":1}}],["tomldiscard",{"2":{"611":1}}],["tomlexternal",{"2":{"608":1}}],["tomlsed",{"2":{"607":1}}],["tomlseeds=$",{"2":{"592":2}}],["tomlrpc",{"2":{"601":1}}],["tomlpruning",{"2":{"598":1,"599":1}}],["tomlpersistent",{"2":{"592":3}}],["tomlmin",{"2":{"596":1}}],["tomlindexer",{"2":{"595":1,"597":1,"610":1}}],["toml`",{"2":{"592":2}}],["toml",{"0":{"531":1,"563":1,"565":1},"1":{"564":1,"565":1,"566":2,"567":2,"568":2,"569":2,"570":2,"571":2},"2":{"357":3,"502":1,"503":1,"531":8,"565":4,"592":7,"594":1,"596":1,"597":1,"598":1,"599":2,"601":1,"607":2,"608":1,"610":2,"611":2,"629":3,"631":1,"635":1,"651":1,"729":1}}],["toml$home",{"2":{"357":3}}],["tombstoned",{"2":{"25":1}}],["toggleview",{"2":{"268":6}}],["together",{"2":{"52":1,"351":1,"416":1,"422":1,"424":1,"425":1,"503":1}}],["tostring",{"2":{"268":4}}],["toattestationproof",{"2":{"159":2,"161":4}}],["torowproofs",{"2":{"158":2,"161":4}}],["torowroots",{"2":{"157":2,"161":4}}],["tonamespacenode",{"2":{"155":4,"157":2,"161":6}}],["tonamespacemerklemultiproofs",{"2":{"155":2,"161":4}}],["token=",{"2":{"370":2}}],["token=$",{"2":{"314":4,"316":2,"334":2,"368":2}}],["tokenecho",{"2":{"368":1}}],["tokencelestia",{"2":{"312":1}}],["tokenbridgeretryablesender",{"2":{"96":1,"97":1}}],["tokenbridgecreator",{"2":{"96":1,"97":1}}],["tokenbridgedeployment",{"2":{"91":2}}],["token",{"0":{"314":1,"316":1,"462":1,"493":1,"638":1},"2":{"80":1,"82":2,"84":8,"86":1,"89":8,"91":13,"93":1,"312":7,"313":9,"314":6,"315":3,"316":2,"317":4,"322":1,"334":3,"368":2,"370":2,"373":3,"426":1,"458":3,"493":1,"495":1,"637":1,"638":3}}],["tokens",{"0":{"431":1,"497":1,"662":1},"2":{"8":1,"30":2,"33":1,"43":1,"80":1,"84":1,"85":1,"310":2,"329":1,"337":2,"367":1,"426":1,"431":1,"432":1,"436":1,"449":1,"450":6,"472":2,"473":1,"485":3,"493":1,"497":2,"522":1,"537":2,"620":3,"627":1,"630":2,"638":1,"661":1,"662":1,"693":3,"721":1,"727":1}}],["touch",{"2":{"65":1}}],["too",{"0":{"557":1},"2":{"557":3,"649":2}}],["tools",{"2":{"213":1,"290":1}}],["toolkit",{"0":{"57":1},"1":{"58":1,"59":1,"60":1,"61":1,"62":1,"63":1},"2":{"56":1,"57":1}}],["tool",{"2":{"31":1,"55":1,"265":1,"352":1,"488":2,"501":1,"611":1}}],["toolings",{"2":{"548":1}}],["tooling",{"2":{"24":1,"166":1,"182":1}}],["took",{"2":{"31":1}}],["topology",{"2":{"601":1}}],["topic",{"2":{"36":1,"42":1,"43":1,"411":1,"494":1}}],["topics",{"2":{"2":1,"36":1,"43":2,"48":2,"209":1}}],["top",{"2":{"24":2,"65":1,"68":1,"293":1,"380":1,"399":1,"400":1,"421":1,"422":1,"434":1,"515":1,"674":1,"675":1,"676":1,"682":1,"694":1}}],["totaldifficulty",{"2":{"347":2}}],["total",{"2":{"21":1,"22":1,"76":1,"151":2,"158":3,"159":2,"161":4,"172":1,"272":1,"326":4,"384":1,"385":2,"386":6,"445":1,"449":1,"455":1,"525":4,"671":5}}],["towards",{"2":{"19":1,"25":1,"44":1,"509":1}}],["toward",{"2":{"8":1,"16":1}}],["to",{"0":{"59":1,"67":1,"69":1,"71":1,"104":1,"112":1,"136":1,"142":1,"143":1,"151":1,"153":1,"163":1,"198":1,"200":1,"209":1,"244":1,"264":1,"269":1,"283":1,"284":1,"309":1,"329":1,"354":1,"364":1,"382":1,"392":1,"393":1,"398":1,"399":1,"400":1,"414":1,"426":1,"499":1,"511":1,"556":1,"572":1,"594":1,"696":1,"721":1},"1":{"68":1,"69":1,"105":1,"106":1,"107":1,"108":1,"109":1,"110":1,"111":1,"113":1,"114":1,"115":1,"116":1,"117":1,"118":1,"119":1,"120":1,"121":1,"122":1,"154":1,"155":1,"156":1,"157":1,"158":1,"159":1,"160":1,"161":1,"164":1,"165":1,"166":1,"167":1,"168":1,"169":1,"170":1,"171":1,"172":1,"173":1,"174":1,"175":1,"176":1,"177":1,"178":1,"179":1,"180":1,"181":1,"182":1,"183":1,"184":1,"185":1,"186":1,"187":1,"188":1,"189":1,"190":1,"191":1,"192":1,"193":1,"194":1,"195":1,"201":1,"202":1,"203":1,"204":1,"205":1,"206":1,"207":1,"208":1,"209":1,"210":1,"211":1,"285":1,"286":1,"287":1,"288":1,"289":1,"290":1,"291":1,"355":1,"356":1,"357":1,"358":1,"359":1,"360":1,"361":1,"362":1,"363":1,"364":1,"383":1,"384":1,"385":1,"386":1,"387":1,"388":1,"389":1,"390":1,"391":1,"392":1,"393":1,"427":1,"428":1,"429":1,"430":1,"431":1,"432":1,"433":1,"434":1,"435":1,"436":1,"512":1,"513":1,"514":1,"515":1,"516":1,"517":1,"518":1,"519":1,"520":1,"521":1,"522":1,"523":1,"524":1,"525":1,"526":1,"527":1,"528":1,"529":1,"530":1,"531":1,"532":1,"533":1,"573":1,"574":1,"595":1,"596":1},"2":{"0":2,"1":1,"2":4,"5":2,"7":3,"8":3,"9":3,"11":2,"14":1,"15":1,"17":1,"19":5,"21":5,"22":1,"23":1,"24":7,"25":3,"26":4,"27":5,"30":9,"31":20,"32":5,"33":12,"34":51,"35":3,"36":7,"37":1,"38":3,"39":2,"40":7,"41":4,"42":8,"43":14,"44":14,"45":10,"46":7,"47":14,"48":13,"49":1,"50":3,"52":4,"53":4,"54":2,"55":2,"56":2,"57":4,"59":2,"60":1,"61":1,"62":2,"63":1,"64":1,"65":6,"66":1,"69":3,"71":7,"72":3,"73":1,"74":1,"76":7,"77":2,"79":3,"82":5,"83":7,"84":19,"85":10,"86":5,"88":15,"89":21,"90":2,"91":21,"92":1,"94":1,"95":1,"102":2,"103":3,"105":2,"106":1,"107":15,"108":6,"109":4,"110":8,"111":1,"113":3,"116":3,"117":4,"119":1,"120":5,"121":4,"122":4,"125":1,"127":6,"128":2,"129":1,"130":4,"131":21,"133":8,"134":23,"135":21,"136":8,"137":8,"139":13,"142":2,"143":1,"145":3,"146":5,"148":17,"149":5,"150":8,"151":14,"152":9,"153":27,"154":2,"155":12,"156":4,"158":7,"159":27,"161":18,"162":3,"163":4,"166":24,"167":6,"168":3,"169":4,"170":5,"172":10,"173":18,"176":6,"177":6,"178":5,"180":2,"181":14,"182":1,"183":2,"184":3,"185":8,"186":1,"188":5,"189":5,"191":3,"192":8,"196":3,"197":1,"198":2,"201":5,"203":1,"204":1,"205":4,"206":15,"207":3,"208":3,"211":1,"212":2,"213":7,"214":2,"215":2,"217":3,"218":1,"219":2,"220":2,"221":1,"223":2,"225":1,"228":3,"229":5,"231":6,"232":6,"233":4,"234":1,"235":1,"236":1,"237":8,"239":1,"242":1,"243":7,"244":4,"245":1,"248":3,"249":4,"251":2,"252":5,"253":4,"254":3,"255":7,"256":7,"258":3,"259":1,"260":5,"261":1,"262":3,"263":3,"264":12,"265":1,"266":1,"267":6,"268":15,"269":2,"270":11,"271":1,"272":9,"273":2,"275":1,"277":2,"278":3,"280":2,"281":1,"283":3,"284":2,"285":2,"288":5,"289":2,"290":9,"291":1,"293":2,"295":3,"296":1,"297":4,"299":2,"301":2,"302":3,"304":3,"305":3,"307":2,"309":7,"310":8,"311":2,"312":9,"313":3,"314":5,"315":6,"316":1,"317":8,"318":1,"319":9,"320":4,"322":1,"324":1,"325":1,"329":1,"330":1,"331":2,"332":2,"334":4,"335":2,"337":1,"338":1,"341":1,"342":1,"344":1,"345":2,"346":3,"348":5,"349":1,"350":3,"351":11,"352":4,"353":1,"354":5,"355":2,"356":3,"357":13,"358":2,"359":1,"360":4,"361":1,"362":1,"363":1,"364":2,"365":8,"366":1,"367":10,"368":1,"369":4,"370":4,"371":2,"372":1,"373":15,"374":6,"375":5,"376":5,"377":1,"378":2,"379":2,"380":7,"381":1,"382":2,"384":3,"385":4,"386":9,"388":8,"392":26,"396":2,"397":4,"398":14,"399":4,"400":3,"402":7,"403":3,"404":4,"405":2,"406":3,"407":7,"408":1,"409":5,"412":7,"413":4,"415":7,"416":4,"417":3,"418":6,"419":6,"421":9,"422":8,"423":8,"424":11,"425":8,"426":2,"429":1,"434":2,"435":1,"438":6,"440":2,"441":3,"442":18,"444":3,"445":4,"447":3,"448":3,"449":2,"450":8,"451":2,"452":1,"453":1,"457":1,"458":5,"459":2,"460":1,"466":3,"469":9,"470":2,"471":7,"472":1,"477":1,"478":5,"483":1,"484":5,"485":7,"486":1,"488":6,"491":4,"492":1,"493":6,"494":12,"495":10,"497":2,"501":2,"502":6,"503":23,"504":3,"505":4,"506":2,"507":8,"508":2,"509":10,"510":1,"511":3,"512":1,"515":1,"516":1,"517":3,"518":1,"519":2,"521":3,"522":8,"523":1,"525":1,"526":2,"527":3,"528":1,"529":6,"530":2,"531":3,"532":1,"533":2,"535":1,"536":2,"537":6,"539":5,"540":4,"541":6,"542":1,"544":4,"547":10,"548":6,"550":4,"551":2,"552":6,"553":3,"554":3,"556":4,"557":6,"559":1,"560":4,"561":2,"562":2,"566":1,"568":4,"569":3,"571":1,"572":1,"573":3,"574":1,"575":10,"577":5,"578":9,"579":6,"580":2,"583":3,"584":1,"585":1,"586":3,"587":2,"588":4,"592":5,"594":2,"597":2,"598":2,"599":3,"600":2,"601":3,"602":1,"603":4,"606":2,"607":3,"608":2,"609":1,"610":12,"611":9,"612":1,"613":1,"620":12,"621":1,"622":1,"623":2,"624":5,"625":6,"626":9,"627":3,"628":2,"629":2,"630":7,"631":1,"632":2,"633":2,"634":4,"635":1,"637":2,"638":4,"639":4,"640":1,"641":1,"642":1,"643":2,"646":1,"648":3,"649":1,"650":4,"651":6,"652":2,"653":3,"654":1,"655":1,"660":10,"661":2,"662":2,"663":1,"666":2,"667":2,"670":1,"671":2,"672":10,"679":8,"680":1,"685":21,"688":5,"689":2,"690":2,"691":3,"692":1,"693":1,"696":4,"698":1,"701":3,"703":3,"706":2,"707":4,"708":2,"709":9,"711":1,"712":2,"714":1,"715":1,"716":1,"717":1,"721":7,"722":4,"724":4,"727":11,"728":1,"729":3,"730":1}}]],"serializationVersion":2}';export{e as default}; diff --git a/assets/chunks/ArabicaVersionTags.e649ff5a.js b/assets/chunks/ArabicaVersionTags.e649ff5a.js new file mode 100644 index 00000000000..492fb27ce8f --- /dev/null +++ b/assets/chunks/ArabicaVersionTags.e649ff5a.js @@ -0,0 +1 @@ +import{a as r}from"./arabica_versions.d7daf483.js";import{c as s}from"./constants.dc70dfe7.js";import{_ as o,o as n,c as l,k as e,t as a}from"./framework.51d6c45b.js";const c={name:"ArabicaVersionTags",data(){return{arabicaVersions:r,constants:s}}},i=e("tr",null,[e("th",null,"Software"),e("th",null,"Version")],-1),p=e("td",null,"celestia-node",-1),_=["href"],h=e("td",null,"celestia-app",-1),u=["href"];function g(d,f,b,m,t,V){return n(),l("table",null,[i,e("tr",null,[p,e("td",null,[e("a",{href:`https://github.com/celestiaorg/celestia-node/releases/tag/${t.arabicaVersions["node-latest-tag"]}`,target:"_blank",rel:"noopener noreferrer"},a(t.arabicaVersions["node-latest-tag"]),9,_)])]),e("tr",null,[h,e("td",null,[e("a",{href:`https://github.com/celestiaorg/celestia-app/releases/tag/${t.arabicaVersions["app-latest-tag"]}`,target:"_blank",rel:"noopener noreferrer"},a(t.arabicaVersions["app-latest-tag"]),9,u)])])])}const B=o(c,[["render",g]]);export{B as A}; diff --git a/assets/chunks/MainnetVersionTags.4372f6f0.js b/assets/chunks/MainnetVersionTags.4372f6f0.js new file mode 100644 index 00000000000..085cd435af9 --- /dev/null +++ b/assets/chunks/MainnetVersionTags.4372f6f0.js @@ -0,0 +1 @@ +import{m as s}from"./mainnet_versions.84d9fd1c.js";import{c as o}from"./constants.dc70dfe7.js";import{_ as r,o as a,c as l,k as e,t as n}from"./framework.51d6c45b.js";const i={name:"MainnetVersionTags",data(){return{mainnetVersions:s,constants:o}}},c=e("tr",null,[e("th",null,"Software"),e("th",null,"Version")],-1),p=e("td",null,"celestia-node",-1),_=["href"],m=e("td",null,"celestia-app",-1),h=["href"];function u(g,d,f,V,t,b){return a(),l("table",null,[c,e("tr",null,[p,e("td",null,[e("a",{href:`https://github.com/celestiaorg/celestia-node/releases/tag/${t.mainnetVersions["node-latest-tag"]}`,target:"_blank",rel:"noopener noreferrer"},n(t.mainnetVersions["node-latest-tag"]),9,_)])]),e("tr",null,[m,e("td",null,[e("a",{href:`https://github.com/celestiaorg/celestia-app/releases/tag/${t.mainnetVersions["app-latest-tag"]}`,target:"_blank",rel:"noopener noreferrer"},n(t.mainnetVersions["app-latest-tag"]),9,h)])])])}const M=r(i,[["render",u]]);export{M}; diff --git a/assets/chunks/MochaVersionTags.139813c5.js b/assets/chunks/MochaVersionTags.139813c5.js new file mode 100644 index 00000000000..bdb2471cfc2 --- /dev/null +++ b/assets/chunks/MochaVersionTags.139813c5.js @@ -0,0 +1 @@ +import{m as s}from"./mocha_versions.3ba4a823.js";import{c as r}from"./constants.dc70dfe7.js";import{_ as a,o as n,c as l,k as e,t as o}from"./framework.51d6c45b.js";const c={name:"MochaVersionTags",data(){return{mochaVersions:s,constants:r}}},i=e("tr",null,[e("th",null,"Software"),e("th",null,"Version")],-1),h=e("td",null,"celestia-node",-1),p=["href"],_=e("td",null,"celestia-app",-1),m=["href"];function u(g,d,f,V,t,b){return n(),l("table",null,[i,e("tr",null,[h,e("td",null,[e("a",{href:`https://github.com/celestiaorg/celestia-node/releases/tag/${t.mochaVersions["node-latest-tag"]}`,target:"_blank",rel:"noopener noreferrer"},o(t.mochaVersions["node-latest-tag"]),9,p)])]),e("tr",null,[_,e("td",null,[e("a",{href:`https://github.com/celestiaorg/celestia-app/releases/tag/${t.mochaVersions["app-latest-tag"]}`,target:"_blank",rel:"noopener noreferrer"},o(t.mochaVersions["app-latest-tag"]),9,m)])])])}const M=a(c,[["render",u]]);export{M}; diff --git a/assets/chunks/VPLocalSearchBox.bcaf1d49.js b/assets/chunks/VPLocalSearchBox.bcaf1d49.js new file mode 100644 index 00000000000..7e61c127631 --- /dev/null +++ b/assets/chunks/VPLocalSearchBox.bcaf1d49.js @@ -0,0 +1,7 @@ +import{X as pt,h as ie,x as Be,ah as kt,ai as Nt,d as It,E as be,aj as et,g as we,ak as Dt,al as _t,y as Ot,am as Rt,j as De,O as he,V as xe,an as Mt,S as Lt,U as Pt,ao as zt,Y as Bt,s as Vt,ap as $t,o as X,b as Wt,k as F,a1 as jt,l as U,aq as Kt,ar as Jt,as as Ut,c as te,n as tt,e as Fe,D as rt,F as at,a as de,t as ve,at as Ht,p as Gt,m as qt,au as nt,av as Qt,a7 as Yt,ad as Zt,_ as Xt}from"./framework.51d6c45b.js";import{u as er,c as tr}from"./theme.6bee4cb8.js";const rr={root:()=>pt(()=>import("./@localSearchIndexroot.00d83d1a.js"),[])};/*! +* tabbable 6.2.0 +* @license MIT, https://github.com/focus-trap/tabbable/blob/master/LICENSE +*/var yt=["input:not([inert])","select:not([inert])","textarea:not([inert])","a[href]:not([inert])","button:not([inert])","[tabindex]:not(slot):not([inert])","audio[controls]:not([inert])","video[controls]:not([inert])",'[contenteditable]:not([contenteditable="false"]):not([inert])',"details>summary:first-of-type:not([inert])","details:not([inert])"],Ae=yt.join(","),mt=typeof Element>"u",se=mt?function(){}:Element.prototype.matches||Element.prototype.msMatchesSelector||Element.prototype.webkitMatchesSelector,Ce=!mt&&Element.prototype.getRootNode?function(o){var e;return o==null||(e=o.getRootNode)===null||e===void 0?void 0:e.call(o)}:function(o){return o==null?void 0:o.ownerDocument},Te=function o(e,t){var r;t===void 0&&(t=!0);var n=e==null||(r=e.getAttribute)===null||r===void 0?void 0:r.call(e,"inert"),a=n===""||n==="true",i=a||t&&e&&o(e.parentNode);return i},ar=function(e){var t,r=e==null||(t=e.getAttribute)===null||t===void 0?void 0:t.call(e,"contenteditable");return r===""||r==="true"},gt=function(e,t,r){if(Te(e))return[];var n=Array.prototype.slice.apply(e.querySelectorAll(Ae));return t&&se.call(e,Ae)&&n.unshift(e),n=n.filter(r),n},bt=function o(e,t,r){for(var n=[],a=Array.from(e);a.length;){var i=a.shift();if(!Te(i,!1))if(i.tagName==="SLOT"){var s=i.assignedElements(),u=s.length?s:i.children,l=o(u,!0,r);r.flatten?n.push.apply(n,l):n.push({scopeParent:i,candidates:l})}else{var h=se.call(i,Ae);h&&r.filter(i)&&(t||!e.includes(i))&&n.push(i);var d=i.shadowRoot||typeof r.getShadowRoot=="function"&&r.getShadowRoot(i),v=!Te(d,!1)&&(!r.shadowRootFilter||r.shadowRootFilter(i));if(d&&v){var m=o(d===!0?i.children:d.children,!0,r);r.flatten?n.push.apply(n,m):n.push({scopeParent:i,candidates:m})}else a.unshift.apply(a,i.children)}}return n},wt=function(e){return!isNaN(parseInt(e.getAttribute("tabindex"),10))},oe=function(e){if(!e)throw new Error("No node provided");return e.tabIndex<0&&(/^(AUDIO|VIDEO|DETAILS)$/.test(e.tagName)||ar(e))&&!wt(e)?0:e.tabIndex},nr=function(e,t){var r=oe(e);return r<0&&t&&!wt(e)?0:r},ir=function(e,t){return e.tabIndex===t.tabIndex?e.documentOrder-t.documentOrder:e.tabIndex-t.tabIndex},xt=function(e){return e.tagName==="INPUT"},or=function(e){return xt(e)&&e.type==="hidden"},sr=function(e){var t=e.tagName==="DETAILS"&&Array.prototype.slice.apply(e.children).some(function(r){return r.tagName==="SUMMARY"});return t},ur=function(e,t){for(var r=0;rsummary:first-of-type"),i=a?e.parentElement:e;if(se.call(i,"details:not([open]) *"))return!0;if(!r||r==="full"||r==="legacy-full"){if(typeof n=="function"){for(var s=e;e;){var u=e.parentElement,l=Ce(e);if(u&&!u.shadowRoot&&n(u)===!0)return it(e);e.assignedSlot?e=e.assignedSlot:!u&&l!==e.ownerDocument?e=l.host:e=u}e=s}if(hr(e))return!e.getClientRects().length;if(r!=="legacy-full")return!0}else if(r==="non-zero-area")return it(e);return!1},vr=function(e){if(/^(INPUT|BUTTON|SELECT|TEXTAREA)$/.test(e.tagName))for(var t=e.parentElement;t;){if(t.tagName==="FIELDSET"&&t.disabled){for(var r=0;r=0)},yr=function o(e){var t=[],r=[];return e.forEach(function(n,a){var i=!!n.scopeParent,s=i?n.scopeParent:n,u=nr(s,i),l=i?o(n.candidates):s;u===0?i?t.push.apply(t,l):t.push(s):r.push({documentOrder:a,tabIndex:u,item:n,isScope:i,content:l})}),r.sort(ir).reduce(function(n,a){return a.isScope?n.push.apply(n,a.content):n.push(a.content),n},[]).concat(t)},mr=function(e,t){t=t||{};var r;return t.getShadowRoot?r=bt([e],t.includeContainer,{filter:Ve.bind(null,t),flatten:!1,getShadowRoot:t.getShadowRoot,shadowRootFilter:pr}):r=gt(e,t.includeContainer,Ve.bind(null,t)),yr(r)},gr=function(e,t){t=t||{};var r;return t.getShadowRoot?r=bt([e],t.includeContainer,{filter:ke.bind(null,t),flatten:!0,getShadowRoot:t.getShadowRoot}):r=gt(e,t.includeContainer,ke.bind(null,t)),r},ue=function(e,t){if(t=t||{},!e)throw new Error("No node provided");return se.call(e,Ae)===!1?!1:Ve(t,e)},br=yt.concat("iframe").join(","),_e=function(e,t){if(t=t||{},!e)throw new Error("No node provided");return se.call(e,br)===!1?!1:ke(t,e)};/*! +* focus-trap 7.5.4 +* @license MIT, https://github.com/focus-trap/focus-trap/blob/master/LICENSE +*/function ot(o,e){var t=Object.keys(o);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(o);e&&(r=r.filter(function(n){return Object.getOwnPropertyDescriptor(o,n).enumerable})),t.push.apply(t,r)}return t}function st(o){for(var e=1;e0){var r=e[e.length-1];r!==t&&r.pause()}var n=e.indexOf(t);n===-1||e.splice(n,1),e.push(t)},deactivateTrap:function(e,t){var r=e.indexOf(t);r!==-1&&e.splice(r,1),e.length>0&&e[e.length-1].unpause()}},Er=function(e){return e.tagName&&e.tagName.toLowerCase()==="input"&&typeof e.select=="function"},Sr=function(e){return(e==null?void 0:e.key)==="Escape"||(e==null?void 0:e.key)==="Esc"||(e==null?void 0:e.keyCode)===27},me=function(e){return(e==null?void 0:e.key)==="Tab"||(e==null?void 0:e.keyCode)===9},Ar=function(e){return me(e)&&!e.shiftKey},Cr=function(e){return me(e)&&e.shiftKey},lt=function(e){return setTimeout(e,0)},ct=function(e,t){var r=-1;return e.every(function(n,a){return t(n)?(r=a,!1):!0}),r},pe=function(e){for(var t=arguments.length,r=new Array(t>1?t-1:0),n=1;n1?p-1:0),k=1;k=0)c=r.activeElement;else{var f=i.tabbableGroups[0],p=f&&f.firstTabbableNode;c=p||h("fallbackFocus")}if(!c)throw new Error("Your focus-trap needs to have at least one focusable element");return c},v=function(){if(i.containerGroups=i.containers.map(function(c){var f=mr(c,a.tabbableOptions),p=gr(c,a.tabbableOptions),N=f.length>0?f[0]:void 0,k=f.length>0?f[f.length-1]:void 0,M=p.find(function(w){return ue(w)}),z=p.slice().reverse().find(function(w){return ue(w)}),y=!!f.find(function(w){return oe(w)>0});return{container:c,tabbableNodes:f,focusableNodes:p,posTabIndexesFound:y,firstTabbableNode:N,lastTabbableNode:k,firstDomTabbableNode:M,lastDomTabbableNode:z,nextTabbableNode:function(B){var G=arguments.length>1&&arguments[1]!==void 0?arguments[1]:!0,j=f.indexOf(B);return j<0?G?p.slice(p.indexOf(B)+1).find(function(K){return ue(K)}):p.slice(0,p.indexOf(B)).reverse().find(function(K){return ue(K)}):f[j+(G?1:-1)]}}}),i.tabbableGroups=i.containerGroups.filter(function(c){return c.tabbableNodes.length>0}),i.tabbableGroups.length<=0&&!h("fallbackFocus"))throw new Error("Your focus-trap must have at least one container with at least one tabbable node in it at all times");if(i.containerGroups.find(function(c){return c.posTabIndexesFound})&&i.containerGroups.length>1)throw new Error("At least one node with a positive tabindex was found in one of your focus-trap's multiple containers. Positive tabindexes are only supported in single-container focus-traps.")},m=function x(c){var f=c.activeElement;if(f)return f.shadowRoot&&f.shadowRoot.activeElement!==null?x(f.shadowRoot):f},b=function x(c){if(c!==!1&&c!==m(document)){if(!c||!c.focus){x(d());return}c.focus({preventScroll:!!a.preventScroll}),i.mostRecentlyFocusedNode=c,Er(c)&&c.select()}},E=function(c){var f=h("setReturnFocus",c);return f||(f===!1?!1:c)},g=function(c){var f=c.target,p=c.event,N=c.isBackward,k=N===void 0?!1:N;f=f||Ee(p),v();var M=null;if(i.tabbableGroups.length>0){var z=l(f,p),y=z>=0?i.containerGroups[z]:void 0;if(z<0)k?M=i.tabbableGroups[i.tabbableGroups.length-1].lastTabbableNode:M=i.tabbableGroups[0].firstTabbableNode;else if(k){var w=ct(i.tabbableGroups,function(J){var q=J.firstTabbableNode;return f===q});if(w<0&&(y.container===f||_e(f,a.tabbableOptions)&&!ue(f,a.tabbableOptions)&&!y.nextTabbableNode(f,!1))&&(w=z),w>=0){var B=w===0?i.tabbableGroups.length-1:w-1,G=i.tabbableGroups[B];M=oe(f)>=0?G.lastTabbableNode:G.lastDomTabbableNode}else me(p)||(M=y.nextTabbableNode(f,!1))}else{var j=ct(i.tabbableGroups,function(J){var q=J.lastTabbableNode;return f===q});if(j<0&&(y.container===f||_e(f,a.tabbableOptions)&&!ue(f,a.tabbableOptions)&&!y.nextTabbableNode(f))&&(j=z),j>=0){var K=j===i.tabbableGroups.length-1?0:j+1,V=i.tabbableGroups[K];M=oe(f)>=0?V.firstTabbableNode:V.firstDomTabbableNode}else me(p)||(M=y.nextTabbableNode(f))}}else M=h("fallbackFocus");return M},S=function(c){var f=Ee(c);if(!(l(f,c)>=0)){if(pe(a.clickOutsideDeactivates,c)){s.deactivate({returnFocus:a.returnFocusOnDeactivate});return}pe(a.allowOutsideClick,c)||c.preventDefault()}},T=function(c){var f=Ee(c),p=l(f,c)>=0;if(p||f instanceof Document)p&&(i.mostRecentlyFocusedNode=f);else{c.stopImmediatePropagation();var N,k=!0;if(i.mostRecentlyFocusedNode)if(oe(i.mostRecentlyFocusedNode)>0){var M=l(i.mostRecentlyFocusedNode),z=i.containerGroups[M].tabbableNodes;if(z.length>0){var y=z.findIndex(function(w){return w===i.mostRecentlyFocusedNode});y>=0&&(a.isKeyForward(i.recentNavEvent)?y+1=0&&(N=z[y-1],k=!1))}}else i.containerGroups.some(function(w){return w.tabbableNodes.some(function(B){return oe(B)>0})})||(k=!1);else k=!1;k&&(N=g({target:i.mostRecentlyFocusedNode,isBackward:a.isKeyBackward(i.recentNavEvent)})),b(N||i.mostRecentlyFocusedNode||d())}i.recentNavEvent=void 0},C=function(c){var f=arguments.length>1&&arguments[1]!==void 0?arguments[1]:!1;i.recentNavEvent=c;var p=g({event:c,isBackward:f});p&&(me(c)&&c.preventDefault(),b(p))},R=function(c){if(Sr(c)&&pe(a.escapeDeactivates,c)!==!1){c.preventDefault(),s.deactivate();return}(a.isKeyForward(c)||a.isKeyBackward(c))&&C(c,a.isKeyBackward(c))},L=function(c){var f=Ee(c);l(f,c)>=0||pe(a.clickOutsideDeactivates,c)||pe(a.allowOutsideClick,c)||(c.preventDefault(),c.stopImmediatePropagation())},P=function(){if(i.active)return ut.activateTrap(n,s),i.delayInitialFocusTimer=a.delayInitialFocus?lt(function(){b(d())}):b(d()),r.addEventListener("focusin",T,!0),r.addEventListener("mousedown",S,{capture:!0,passive:!1}),r.addEventListener("touchstart",S,{capture:!0,passive:!1}),r.addEventListener("click",L,{capture:!0,passive:!1}),r.addEventListener("keydown",R,{capture:!0,passive:!1}),s},I=function(){if(i.active)return r.removeEventListener("focusin",T,!0),r.removeEventListener("mousedown",S,!0),r.removeEventListener("touchstart",S,!0),r.removeEventListener("click",L,!0),r.removeEventListener("keydown",R,!0),s},A=function(c){var f=c.some(function(p){var N=Array.from(p.removedNodes);return N.some(function(k){return k===i.mostRecentlyFocusedNode})});f&&b(d())},_=typeof window<"u"&&"MutationObserver"in window?new MutationObserver(A):void 0,O=function(){_&&(_.disconnect(),i.active&&!i.paused&&i.containers.map(function(c){_.observe(c,{subtree:!0,childList:!0})}))};return s={get active(){return i.active},get paused(){return i.paused},activate:function(c){if(i.active)return this;var f=u(c,"onActivate"),p=u(c,"onPostActivate"),N=u(c,"checkCanFocusTrap");N||v(),i.active=!0,i.paused=!1,i.nodeFocusedBeforeActivation=r.activeElement,f==null||f();var k=function(){N&&v(),P(),O(),p==null||p()};return N?(N(i.containers.concat()).then(k,k),this):(k(),this)},deactivate:function(c){if(!i.active)return this;var f=st({onDeactivate:a.onDeactivate,onPostDeactivate:a.onPostDeactivate,checkCanReturnFocus:a.checkCanReturnFocus},c);clearTimeout(i.delayInitialFocusTimer),i.delayInitialFocusTimer=void 0,I(),i.active=!1,i.paused=!1,O(),ut.deactivateTrap(n,s);var p=u(f,"onDeactivate"),N=u(f,"onPostDeactivate"),k=u(f,"checkCanReturnFocus"),M=u(f,"returnFocus","returnFocusOnDeactivate");p==null||p();var z=function(){lt(function(){M&&b(E(i.nodeFocusedBeforeActivation)),N==null||N()})};return M&&k?(k(E(i.nodeFocusedBeforeActivation)).then(z,z),this):(z(),this)},pause:function(c){if(i.paused||!i.active)return this;var f=u(c,"onPause"),p=u(c,"onPostPause");return i.paused=!0,f==null||f(),I(),O(),p==null||p(),this},unpause:function(c){if(!i.paused||!i.active)return this;var f=u(c,"onUnpause"),p=u(c,"onPostUnpause");return i.paused=!1,f==null||f(),v(),P(),O(),p==null||p(),this},updateContainerElements:function(c){var f=[].concat(c).filter(Boolean);return i.containers=f.map(function(p){return typeof p=="string"?r.querySelector(p):p}),i.active&&v(),O(),this}},s.updateContainerElements(e),s};function Nr(o,e={}){let t;const{immediate:r,...n}=e,a=ie(!1),i=ie(!1),s=d=>t&&t.activate(d),u=d=>t&&t.deactivate(d),l=()=>{t&&(t.pause(),i.value=!0)},h=()=>{t&&(t.unpause(),i.value=!1)};return Be(()=>kt(o),d=>{d&&(t=kr(d,{...n,onActivate(){a.value=!0,e.onActivate&&e.onActivate()},onDeactivate(){a.value=!1,e.onDeactivate&&e.onDeactivate()}}),r&&s())},{flush:"post"}),Nt(()=>u()),{hasFocus:a,isPaused:i,activate:s,deactivate:u,pause:l,unpause:h}}class ce{constructor(e,t=!0,r=[],n=5e3){this.ctx=e,this.iframes=t,this.exclude=r,this.iframesTimeout=n}static matches(e,t){const r=typeof t=="string"?[t]:t,n=e.matches||e.matchesSelector||e.msMatchesSelector||e.mozMatchesSelector||e.oMatchesSelector||e.webkitMatchesSelector;if(n){let a=!1;return r.every(i=>n.call(e,i)?(a=!0,!1):!0),a}else return!1}getContexts(){let e,t=[];return typeof this.ctx>"u"||!this.ctx?e=[]:NodeList.prototype.isPrototypeOf(this.ctx)?e=Array.prototype.slice.call(this.ctx):Array.isArray(this.ctx)?e=this.ctx:typeof this.ctx=="string"?e=Array.prototype.slice.call(document.querySelectorAll(this.ctx)):e=[this.ctx],e.forEach(r=>{const n=t.filter(a=>a.contains(r)).length>0;t.indexOf(r)===-1&&!n&&t.push(r)}),t}getIframeContents(e,t,r=()=>{}){let n;try{const a=e.contentWindow;if(n=a.document,!a||!n)throw new Error("iframe inaccessible")}catch{r()}n&&t(n)}isIframeBlank(e){const t="about:blank",r=e.getAttribute("src").trim();return e.contentWindow.location.href===t&&r!==t&&r}observeIframeLoad(e,t,r){let n=!1,a=null;const i=()=>{if(!n){n=!0,clearTimeout(a);try{this.isIframeBlank(e)||(e.removeEventListener("load",i),this.getIframeContents(e,t,r))}catch{r()}}};e.addEventListener("load",i),a=setTimeout(i,this.iframesTimeout)}onIframeReady(e,t,r){try{e.contentWindow.document.readyState==="complete"?this.isIframeBlank(e)?this.observeIframeLoad(e,t,r):this.getIframeContents(e,t,r):this.observeIframeLoad(e,t,r)}catch{r()}}waitForIframes(e,t){let r=0;this.forEachIframe(e,()=>!0,n=>{r++,this.waitForIframes(n.querySelector("html"),()=>{--r||t()})},n=>{n||t()})}forEachIframe(e,t,r,n=()=>{}){let a=e.querySelectorAll("iframe"),i=a.length,s=0;a=Array.prototype.slice.call(a);const u=()=>{--i<=0&&n(s)};i||u(),a.forEach(l=>{ce.matches(l,this.exclude)?u():this.onIframeReady(l,h=>{t(l)&&(s++,r(h)),u()},u)})}createIterator(e,t,r){return document.createNodeIterator(e,t,r,!1)}createInstanceOnIframe(e){return new ce(e.querySelector("html"),this.iframes)}compareNodeIframe(e,t,r){const n=e.compareDocumentPosition(r),a=Node.DOCUMENT_POSITION_PRECEDING;if(n&a)if(t!==null){const i=t.compareDocumentPosition(r),s=Node.DOCUMENT_POSITION_FOLLOWING;if(i&s)return!0}else return!0;return!1}getIteratorNode(e){const t=e.previousNode();let r;return t===null?r=e.nextNode():r=e.nextNode()&&e.nextNode(),{prevNode:t,node:r}}checkIframeFilter(e,t,r,n){let a=!1,i=!1;return n.forEach((s,u)=>{s.val===r&&(a=u,i=s.handled)}),this.compareNodeIframe(e,t,r)?(a===!1&&!i?n.push({val:r,handled:!0}):a!==!1&&!i&&(n[a].handled=!0),!0):(a===!1&&n.push({val:r,handled:!1}),!1)}handleOpenIframes(e,t,r,n){e.forEach(a=>{a.handled||this.getIframeContents(a.val,i=>{this.createInstanceOnIframe(i).forEachNode(t,r,n)})})}iterateThroughNodes(e,t,r,n,a){const i=this.createIterator(t,e,n);let s=[],u=[],l,h,d=()=>({prevNode:h,node:l}=this.getIteratorNode(i),l);for(;d();)this.iframes&&this.forEachIframe(t,v=>this.checkIframeFilter(l,h,v,s),v=>{this.createInstanceOnIframe(v).forEachNode(e,m=>u.push(m),n)}),u.push(l);u.forEach(v=>{r(v)}),this.iframes&&this.handleOpenIframes(s,e,r,n),a()}forEachNode(e,t,r,n=()=>{}){const a=this.getContexts();let i=a.length;i||n(),a.forEach(s=>{const u=()=>{this.iterateThroughNodes(e,s,t,r,()=>{--i<=0&&n()})};this.iframes?this.waitForIframes(s,u):u()})}}let Ir=class{constructor(e){this.ctx=e,this.ie=!1;const t=window.navigator.userAgent;(t.indexOf("MSIE")>-1||t.indexOf("Trident")>-1)&&(this.ie=!0)}set opt(e){this._opt=Object.assign({},{element:"",className:"",exclude:[],iframes:!1,iframesTimeout:5e3,separateWordSearch:!0,diacritics:!0,synonyms:{},accuracy:"partially",acrossElements:!1,caseSensitive:!1,ignoreJoiners:!1,ignoreGroups:0,ignorePunctuation:[],wildcards:"disabled",each:()=>{},noMatch:()=>{},filter:()=>!0,done:()=>{},debug:!1,log:window.console},e)}get opt(){return this._opt}get iterator(){return new ce(this.ctx,this.opt.iframes,this.opt.exclude,this.opt.iframesTimeout)}log(e,t="debug"){const r=this.opt.log;this.opt.debug&&typeof r=="object"&&typeof r[t]=="function"&&r[t](`mark.js: ${e}`)}escapeStr(e){return e.replace(/[\-\[\]\/\{\}\(\)\*\+\?\.\\\^\$\|]/g,"\\$&")}createRegExp(e){return this.opt.wildcards!=="disabled"&&(e=this.setupWildcardsRegExp(e)),e=this.escapeStr(e),Object.keys(this.opt.synonyms).length&&(e=this.createSynonymsRegExp(e)),(this.opt.ignoreJoiners||this.opt.ignorePunctuation.length)&&(e=this.setupIgnoreJoinersRegExp(e)),this.opt.diacritics&&(e=this.createDiacriticsRegExp(e)),e=this.createMergedBlanksRegExp(e),(this.opt.ignoreJoiners||this.opt.ignorePunctuation.length)&&(e=this.createJoinersRegExp(e)),this.opt.wildcards!=="disabled"&&(e=this.createWildcardsRegExp(e)),e=this.createAccuracyRegExp(e),e}createSynonymsRegExp(e){const t=this.opt.synonyms,r=this.opt.caseSensitive?"":"i",n=this.opt.ignoreJoiners||this.opt.ignorePunctuation.length?"\0":"";for(let a in t)if(t.hasOwnProperty(a)){const i=t[a],s=this.opt.wildcards!=="disabled"?this.setupWildcardsRegExp(a):this.escapeStr(a),u=this.opt.wildcards!=="disabled"?this.setupWildcardsRegExp(i):this.escapeStr(i);s!==""&&u!==""&&(e=e.replace(new RegExp(`(${this.escapeStr(s)}|${this.escapeStr(u)})`,`gm${r}`),n+`(${this.processSynomyms(s)}|${this.processSynomyms(u)})`+n))}return e}processSynomyms(e){return(this.opt.ignoreJoiners||this.opt.ignorePunctuation.length)&&(e=this.setupIgnoreJoinersRegExp(e)),e}setupWildcardsRegExp(e){return e=e.replace(/(?:\\)*\?/g,t=>t.charAt(0)==="\\"?"?":""),e.replace(/(?:\\)*\*/g,t=>t.charAt(0)==="\\"?"*":"")}createWildcardsRegExp(e){let t=this.opt.wildcards==="withSpaces";return e.replace(/\u0001/g,t?"[\\S\\s]?":"\\S?").replace(/\u0002/g,t?"[\\S\\s]*?":"\\S*")}setupIgnoreJoinersRegExp(e){return e.replace(/[^(|)\\]/g,(t,r,n)=>{let a=n.charAt(r+1);return/[(|)\\]/.test(a)||a===""?t:t+"\0"})}createJoinersRegExp(e){let t=[];const r=this.opt.ignorePunctuation;return Array.isArray(r)&&r.length&&t.push(this.escapeStr(r.join(""))),this.opt.ignoreJoiners&&t.push("\\u00ad\\u200b\\u200c\\u200d"),t.length?e.split(/\u0000+/).join(`[${t.join("")}]*`):e}createDiacriticsRegExp(e){const t=this.opt.caseSensitive?"":"i",r=this.opt.caseSensitive?["aàáảãạăằắẳẵặâầấẩẫậäåāą","AÀÁẢÃẠĂẰẮẲẴẶÂẦẤẨẪẬÄÅĀĄ","cçćč","CÇĆČ","dđď","DĐĎ","eèéẻẽẹêềếểễệëěēę","EÈÉẺẼẸÊỀẾỂỄỆËĚĒĘ","iìíỉĩịîïī","IÌÍỈĨỊÎÏĪ","lł","LŁ","nñňń","NÑŇŃ","oòóỏõọôồốổỗộơởỡớờợöøō","OÒÓỎÕỌÔỒỐỔỖỘƠỞỠỚỜỢÖØŌ","rř","RŘ","sšśșş","SŠŚȘŞ","tťțţ","TŤȚŢ","uùúủũụưừứửữựûüůū","UÙÚỦŨỤƯỪỨỬỮỰÛÜŮŪ","yýỳỷỹỵÿ","YÝỲỶỸỴŸ","zžżź","ZŽŻŹ"]:["aàáảãạăằắẳẵặâầấẩẫậäåāąAÀÁẢÃẠĂẰẮẲẴẶÂẦẤẨẪẬÄÅĀĄ","cçćčCÇĆČ","dđďDĐĎ","eèéẻẽẹêềếểễệëěēęEÈÉẺẼẸÊỀẾỂỄỆËĚĒĘ","iìíỉĩịîïīIÌÍỈĨỊÎÏĪ","lłLŁ","nñňńNÑŇŃ","oòóỏõọôồốổỗộơởỡớờợöøōOÒÓỎÕỌÔỒỐỔỖỘƠỞỠỚỜỢÖØŌ","rřRŘ","sšśșşSŠŚȘŞ","tťțţTŤȚŢ","uùúủũụưừứửữựûüůūUÙÚỦŨỤƯỪỨỬỮỰÛÜŮŪ","yýỳỷỹỵÿYÝỲỶỸỴŸ","zžżźZŽŻŹ"];let n=[];return e.split("").forEach(a=>{r.every(i=>{if(i.indexOf(a)!==-1){if(n.indexOf(i)>-1)return!1;e=e.replace(new RegExp(`[${i}]`,`gm${t}`),`[${i}]`),n.push(i)}return!0})}),e}createMergedBlanksRegExp(e){return e.replace(/[\s]+/gmi,"[\\s]+")}createAccuracyRegExp(e){const t="!\"#$%&'()*+,-./:;<=>?@[\\]^_`{|}~¡¿";let r=this.opt.accuracy,n=typeof r=="string"?r:r.value,a=typeof r=="string"?[]:r.limiters,i="";switch(a.forEach(s=>{i+=`|${this.escapeStr(s)}`}),n){case"partially":default:return`()(${e})`;case"complementary":return i="\\s"+(i||this.escapeStr(t)),`()([^${i}]*${e}[^${i}]*)`;case"exactly":return`(^|\\s${i})(${e})(?=$|\\s${i})`}}getSeparatedKeywords(e){let t=[];return e.forEach(r=>{this.opt.separateWordSearch?r.split(" ").forEach(n=>{n.trim()&&t.indexOf(n)===-1&&t.push(n)}):r.trim()&&t.indexOf(r)===-1&&t.push(r)}),{keywords:t.sort((r,n)=>n.length-r.length),length:t.length}}isNumeric(e){return Number(parseFloat(e))==e}checkRanges(e){if(!Array.isArray(e)||Object.prototype.toString.call(e[0])!=="[object Object]")return this.log("markRanges() will only accept an array of objects"),this.opt.noMatch(e),[];const t=[];let r=0;return e.sort((n,a)=>n.start-a.start).forEach(n=>{let{start:a,end:i,valid:s}=this.callNoMatchOnInvalidRanges(n,r);s&&(n.start=a,n.length=i-a,t.push(n),r=i)}),t}callNoMatchOnInvalidRanges(e,t){let r,n,a=!1;return e&&typeof e.start<"u"?(r=parseInt(e.start,10),n=r+parseInt(e.length,10),this.isNumeric(e.start)&&this.isNumeric(e.length)&&n-t>0&&n-r>0?a=!0:(this.log(`Ignoring invalid or overlapping range: ${JSON.stringify(e)}`),this.opt.noMatch(e))):(this.log(`Ignoring invalid range: ${JSON.stringify(e)}`),this.opt.noMatch(e)),{start:r,end:n,valid:a}}checkWhitespaceRanges(e,t,r){let n,a=!0,i=r.length,s=t-i,u=parseInt(e.start,10)-s;return u=u>i?i:u,n=u+parseInt(e.length,10),n>i&&(n=i,this.log(`End range automatically set to the max value of ${i}`)),u<0||n-u<0||u>i||n>i?(a=!1,this.log(`Invalid range: ${JSON.stringify(e)}`),this.opt.noMatch(e)):r.substring(u,n).replace(/\s+/g,"")===""&&(a=!1,this.log("Skipping whitespace only range: "+JSON.stringify(e)),this.opt.noMatch(e)),{start:u,end:n,valid:a}}getTextNodes(e){let t="",r=[];this.iterator.forEachNode(NodeFilter.SHOW_TEXT,n=>{r.push({start:t.length,end:(t+=n.textContent).length,node:n})},n=>this.matchesExclude(n.parentNode)?NodeFilter.FILTER_REJECT:NodeFilter.FILTER_ACCEPT,()=>{e({value:t,nodes:r})})}matchesExclude(e){return ce.matches(e,this.opt.exclude.concat(["script","style","title","head","html"]))}wrapRangeInTextNode(e,t,r){const n=this.opt.element?this.opt.element:"mark",a=e.splitText(t),i=a.splitText(r-t);let s=document.createElement(n);return s.setAttribute("data-markjs","true"),this.opt.className&&s.setAttribute("class",this.opt.className),s.textContent=a.textContent,a.parentNode.replaceChild(s,a),i}wrapRangeInMappedTextNode(e,t,r,n,a){e.nodes.every((i,s)=>{const u=e.nodes[s+1];if(typeof u>"u"||u.start>t){if(!n(i.node))return!1;const l=t-i.start,h=(r>i.end?i.end:r)-i.start,d=e.value.substr(0,i.start),v=e.value.substr(h+i.start);if(i.node=this.wrapRangeInTextNode(i.node,l,h),e.value=d+v,e.nodes.forEach((m,b)=>{b>=s&&(e.nodes[b].start>0&&b!==s&&(e.nodes[b].start-=h),e.nodes[b].end-=h)}),r-=h,a(i.node.previousSibling,i.start),r>i.end)t=i.end;else return!1}return!0})}wrapMatches(e,t,r,n,a){const i=t===0?0:t+1;this.getTextNodes(s=>{s.nodes.forEach(u=>{u=u.node;let l;for(;(l=e.exec(u.textContent))!==null&&l[i]!=="";){if(!r(l[i],u))continue;let h=l.index;if(i!==0)for(let d=1;d{let u;for(;(u=e.exec(s.value))!==null&&u[i]!=="";){let l=u.index;if(i!==0)for(let d=1;dr(u[i],d),(d,v)=>{e.lastIndex=v,n(d)})}a()})}wrapRangeFromIndex(e,t,r,n){this.getTextNodes(a=>{const i=a.value.length;e.forEach((s,u)=>{let{start:l,end:h,valid:d}=this.checkWhitespaceRanges(s,i,a.value);d&&this.wrapRangeInMappedTextNode(a,l,h,v=>t(v,s,a.value.substring(l,h),u),v=>{r(v,s)})}),n()})}unwrapMatches(e){const t=e.parentNode;let r=document.createDocumentFragment();for(;e.firstChild;)r.appendChild(e.removeChild(e.firstChild));t.replaceChild(r,e),this.ie?this.normalizeTextNode(t):t.normalize()}normalizeTextNode(e){if(e){if(e.nodeType===3)for(;e.nextSibling&&e.nextSibling.nodeType===3;)e.nodeValue+=e.nextSibling.nodeValue,e.parentNode.removeChild(e.nextSibling);else this.normalizeTextNode(e.firstChild);this.normalizeTextNode(e.nextSibling)}}markRegExp(e,t){this.opt=t,this.log(`Searching with expression "${e}"`);let r=0,n="wrapMatches";const a=i=>{r++,this.opt.each(i)};this.opt.acrossElements&&(n="wrapMatchesAcrossElements"),this[n](e,this.opt.ignoreGroups,(i,s)=>this.opt.filter(s,i,r),a,()=>{r===0&&this.opt.noMatch(e),this.opt.done(r)})}mark(e,t){this.opt=t;let r=0,n="wrapMatches";const{keywords:a,length:i}=this.getSeparatedKeywords(typeof e=="string"?[e]:e),s=this.opt.caseSensitive?"":"i",u=l=>{let h=new RegExp(this.createRegExp(l),`gm${s}`),d=0;this.log(`Searching with expression "${h}"`),this[n](h,1,(v,m)=>this.opt.filter(m,l,r,d),v=>{d++,r++,this.opt.each(v)},()=>{d===0&&this.opt.noMatch(l),a[i-1]===l?this.opt.done(r):u(a[a.indexOf(l)+1])})};this.opt.acrossElements&&(n="wrapMatchesAcrossElements"),i===0?this.opt.done(r):u(a[0])}markRanges(e,t){this.opt=t;let r=0,n=this.checkRanges(e);n&&n.length?(this.log("Starting to mark with the following ranges: "+JSON.stringify(n)),this.wrapRangeFromIndex(n,(a,i,s,u)=>this.opt.filter(a,i,s,u),(a,i)=>{r++,this.opt.each(a,i)},()=>{this.opt.done(r)})):this.opt.done(r)}unmark(e){this.opt=e;let t=this.opt.element?this.opt.element:"*";t+="[data-markjs]",this.opt.className&&(t+=`.${this.opt.className}`),this.log(`Removal selector "${t}"`),this.iterator.forEachNode(NodeFilter.SHOW_ELEMENT,r=>{this.unwrapMatches(r)},r=>{const n=ce.matches(r,t),a=this.matchesExclude(r);return!n||a?NodeFilter.FILTER_REJECT:NodeFilter.FILTER_ACCEPT},this.opt.done)}};function Dr(o){const e=new Ir(o);return this.mark=(t,r)=>(e.mark(t,r),this),this.markRegExp=(t,r)=>(e.markRegExp(t,r),this),this.markRanges=(t,r)=>(e.markRanges(t,r),this),this.unmark=t=>(e.unmark(t),this),this}var $=function(){return $=Object.assign||function(e){for(var t,r=1,n=arguments.length;r0&&a[a.length-1])&&(l[0]===6||l[0]===2)){t=0;continue}if(l[0]===3&&(!a||l[1]>a[0]&&l[1]=o.length&&(o=void 0),{value:o&&o[r++],done:!o}}};throw new TypeError(e?"Object is not iterable.":"Symbol.iterator is not defined.")}function W(o,e){var t=typeof Symbol=="function"&&o[Symbol.iterator];if(!t)return o;var r=t.call(o),n,a=[],i;try{for(;(e===void 0||e-- >0)&&!(n=r.next()).done;)a.push(n.value)}catch(s){i={error:s}}finally{try{n&&!n.done&&(t=r.return)&&t.call(r)}finally{if(i)throw i.error}}return a}var Rr="ENTRIES",Ft="KEYS",Et="VALUES",H="",Oe=function(){function o(e,t){var r=e._tree,n=Array.from(r.keys());this.set=e,this._type=t,this._path=n.length>0?[{node:r,keys:n}]:[]}return o.prototype.next=function(){var e=this.dive();return this.backtrack(),e},o.prototype.dive=function(){if(this._path.length===0)return{done:!0,value:void 0};var e=le(this._path),t=e.node,r=e.keys;if(le(r)===H)return{done:!1,value:this.result()};var n=t.get(le(r));return this._path.push({node:n,keys:Array.from(n.keys())}),this.dive()},o.prototype.backtrack=function(){if(this._path.length!==0){var e=le(this._path).keys;e.pop(),!(e.length>0)&&(this._path.pop(),this.backtrack())}},o.prototype.key=function(){return this.set._prefix+this._path.map(function(e){var t=e.keys;return le(t)}).filter(function(e){return e!==H}).join("")},o.prototype.value=function(){return le(this._path).node.get(H)},o.prototype.result=function(){switch(this._type){case Et:return this.value();case Ft:return this.key();default:return[this.key(),this.value()]}},o.prototype[Symbol.iterator]=function(){return this},o}(),le=function(o){return o[o.length-1]},Mr=function(o,e,t){var r=new Map;if(e===void 0)return r;for(var n=e.length+1,a=n+t,i=new Uint8Array(a*n).fill(t+1),s=0;st)continue e}St(o.get(m),e,t,r,n,E,i,s+m)}}}catch(f){u={error:f}}finally{try{v&&!v.done&&(l=d.return)&&l.call(d)}finally{if(u)throw u.error}}},Re=function(){function o(e,t){e===void 0&&(e=new Map),t===void 0&&(t=""),this._size=void 0,this._tree=e,this._prefix=t}return o.prototype.atPrefix=function(e){var t,r;if(!e.startsWith(this._prefix))throw new Error("Mismatched prefix");var n=W(Ne(this._tree,e.slice(this._prefix.length)),2),a=n[0],i=n[1];if(a===void 0){var s=W(Ke(i),2),u=s[0],l=s[1];try{for(var h=D(u.keys()),d=h.next();!d.done;d=h.next()){var v=d.value;if(v!==H&&v.startsWith(l)){var m=new Map;return m.set(v.slice(l.length),u.get(v)),new o(m,e)}}}catch(b){t={error:b}}finally{try{d&&!d.done&&(r=h.return)&&r.call(h)}finally{if(t)throw t.error}}}return new o(a,e)},o.prototype.clear=function(){this._size=void 0,this._tree.clear()},o.prototype.delete=function(e){return this._size=void 0,Lr(this._tree,e)},o.prototype.entries=function(){return new Oe(this,Rr)},o.prototype.forEach=function(e){var t,r;try{for(var n=D(this),a=n.next();!a.done;a=n.next()){var i=W(a.value,2),s=i[0],u=i[1];e(s,u,this)}}catch(l){t={error:l}}finally{try{a&&!a.done&&(r=n.return)&&r.call(n)}finally{if(t)throw t.error}}},o.prototype.fuzzyGet=function(e,t){return Mr(this._tree,e,t)},o.prototype.get=function(e){var t=$e(this._tree,e);return t!==void 0?t.get(H):void 0},o.prototype.has=function(e){var t=$e(this._tree,e);return t!==void 0&&t.has(H)},o.prototype.keys=function(){return new Oe(this,Ft)},o.prototype.set=function(e,t){if(typeof e!="string")throw new Error("key must be a string");this._size=void 0;var r=Me(this._tree,e);return r.set(H,t),this},Object.defineProperty(o.prototype,"size",{get:function(){if(this._size)return this._size;this._size=0;for(var e=this.entries();!e.next().done;)this._size+=1;return this._size},enumerable:!1,configurable:!0}),o.prototype.update=function(e,t){if(typeof e!="string")throw new Error("key must be a string");this._size=void 0;var r=Me(this._tree,e);return r.set(H,t(r.get(H))),this},o.prototype.fetch=function(e,t){if(typeof e!="string")throw new Error("key must be a string");this._size=void 0;var r=Me(this._tree,e),n=r.get(H);return n===void 0&&r.set(H,n=t()),n},o.prototype.values=function(){return new Oe(this,Et)},o.prototype[Symbol.iterator]=function(){return this.entries()},o.from=function(e){var t,r,n=new o;try{for(var a=D(e),i=a.next();!i.done;i=a.next()){var s=W(i.value,2),u=s[0],l=s[1];n.set(u,l)}}catch(h){t={error:h}}finally{try{i&&!i.done&&(r=a.return)&&r.call(a)}finally{if(t)throw t.error}}return n},o.fromObject=function(e){return o.from(Object.entries(e))},o}(),Ne=function(o,e,t){var r,n;if(t===void 0&&(t=[]),e.length===0||o==null)return[o,t];try{for(var a=D(o.keys()),i=a.next();!i.done;i=a.next()){var s=i.value;if(s!==H&&e.startsWith(s))return t.push([o,s]),Ne(o.get(s),e.slice(s.length),t)}}catch(u){r={error:u}}finally{try{i&&!i.done&&(n=a.return)&&n.call(a)}finally{if(r)throw r.error}}return t.push([o,e]),Ne(void 0,"",t)},$e=function(o,e){var t,r;if(e.length===0||o==null)return o;try{for(var n=D(o.keys()),a=n.next();!a.done;a=n.next()){var i=a.value;if(i!==H&&e.startsWith(i))return $e(o.get(i),e.slice(i.length))}}catch(s){t={error:s}}finally{try{a&&!a.done&&(r=n.return)&&r.call(n)}finally{if(t)throw t.error}}},Me=function(o,e){var t,r,n=e.length;e:for(var a=0;o&&a0)throw new Error("Expected documents to be present. Omit the argument to remove all documents.");this._index=new Re,this._documentCount=0,this._documentIds=new Map,this._idToShortId=new Map,this._fieldLength=new Map,this._avgFieldLength=[],this._storedFields=new Map,this._nextId=0}},o.prototype.discard=function(e){var t=this,r=this._idToShortId.get(e);if(r==null)throw new Error("MiniSearch: cannot discard document with ID ".concat(e,": it is not in the index"));this._idToShortId.delete(e),this._documentIds.delete(r),this._storedFields.delete(r),(this._fieldLength.get(r)||[]).forEach(function(n,a){t.removeFieldLength(r,a,t._documentCount,n)}),this._fieldLength.delete(r),this._documentCount-=1,this._dirtCount+=1,this.maybeAutoVacuum()},o.prototype.maybeAutoVacuum=function(){if(this._options.autoVacuum!==!1){var e=this._options.autoVacuum,t=e.minDirtFactor,r=e.minDirtCount,n=e.batchSize,a=e.batchWait;this.conditionalVacuum({batchSize:n,batchWait:a},{minDirtCount:r,minDirtFactor:t})}},o.prototype.discardAll=function(e){var t,r,n=this._options.autoVacuum;try{this._options.autoVacuum=!1;try{for(var a=D(e),i=a.next();!i.done;i=a.next()){var s=i.value;this.discard(s)}}catch(u){t={error:u}}finally{try{i&&!i.done&&(r=a.return)&&r.call(a)}finally{if(t)throw t.error}}}finally{this._options.autoVacuum=n}this.maybeAutoVacuum()},o.prototype.replace=function(e){var t=this._options,r=t.idField,n=t.extractField,a=n(e,r);this.discard(a),this.add(e)},o.prototype.vacuum=function(e){return e===void 0&&(e={}),this.conditionalVacuum(e)},o.prototype.conditionalVacuum=function(e,t){var r=this;return this._currentVacuum?(this._enqueuedVacuumConditions=this._enqueuedVacuumConditions&&t,this._enqueuedVacuum!=null?this._enqueuedVacuum:(this._enqueuedVacuum=this._currentVacuum.then(function(){var n=r._enqueuedVacuumConditions;return r._enqueuedVacuumConditions=je,r.performVacuuming(e,n)}),this._enqueuedVacuum)):this.vacuumConditionsMet(t)===!1?Promise.resolve():(this._currentVacuum=this.performVacuuming(e),this._currentVacuum)},o.prototype.performVacuuming=function(e,t){return _r(this,void 0,void 0,function(){var r,n,a,i,s,u,l,h,d,v,m,b,E,g,S,T,C,R,L,P,I,A,_,O,x;return Or(this,function(c){switch(c.label){case 0:if(r=this._dirtCount,!this.vacuumConditionsMet(t))return[3,10];n=e.batchSize||We.batchSize,a=e.batchWait||We.batchWait,i=1,c.label=1;case 1:c.trys.push([1,7,8,9]),s=D(this._index),u=s.next(),c.label=2;case 2:if(u.done)return[3,6];l=W(u.value,2),h=l[0],d=l[1];try{for(v=(A=void 0,D(d)),m=v.next();!m.done;m=v.next()){b=W(m.value,2),E=b[0],g=b[1];try{for(S=(O=void 0,D(g)),T=S.next();!T.done;T=S.next())C=W(T.value,1),R=C[0],!this._documentIds.has(R)&&(g.size<=1?d.delete(E):g.delete(R))}catch(f){O={error:f}}finally{try{T&&!T.done&&(x=S.return)&&x.call(S)}finally{if(O)throw O.error}}}}catch(f){A={error:f}}finally{try{m&&!m.done&&(_=v.return)&&_.call(v)}finally{if(A)throw A.error}}return this._index.get(h).size===0&&this._index.delete(h),i%n!==0?[3,4]:[4,new Promise(function(f){return setTimeout(f,a)})];case 3:c.sent(),c.label=4;case 4:i+=1,c.label=5;case 5:return u=s.next(),[3,2];case 6:return[3,9];case 7:return L=c.sent(),P={error:L},[3,9];case 8:try{u&&!u.done&&(I=s.return)&&I.call(s)}finally{if(P)throw P.error}return[7];case 9:this._dirtCount-=r,c.label=10;case 10:return[4,null];case 11:return c.sent(),this._currentVacuum=this._enqueuedVacuum,this._enqueuedVacuum=null,[2]}})})},o.prototype.vacuumConditionsMet=function(e){if(e==null)return!0;var t=e.minDirtCount,r=e.minDirtFactor;return t=t||ze.minDirtCount,r=r||ze.minDirtFactor,this.dirtCount>=t&&this.dirtFactor>=r},Object.defineProperty(o.prototype,"isVacuuming",{get:function(){return this._currentVacuum!=null},enumerable:!1,configurable:!0}),Object.defineProperty(o.prototype,"dirtCount",{get:function(){return this._dirtCount},enumerable:!1,configurable:!0}),Object.defineProperty(o.prototype,"dirtFactor",{get:function(){return this._dirtCount/(1+this._documentCount+this._dirtCount)},enumerable:!1,configurable:!0}),o.prototype.has=function(e){return this._idToShortId.has(e)},o.prototype.getStoredFields=function(e){var t=this._idToShortId.get(e);if(t!=null)return this._storedFields.get(t)},o.prototype.search=function(e,t){var r,n;t===void 0&&(t={});var a=this.executeQuery(e,t),i=[];try{for(var s=D(a),u=s.next();!u.done;u=s.next()){var l=W(u.value,2),h=l[0],d=l[1],v=d.score,m=d.terms,b=d.match,E=m.length,g={id:this._documentIds.get(h),score:v*E,terms:Object.keys(b),match:b};Object.assign(g,this._storedFields.get(h)),(t.filter==null||t.filter(g))&&i.push(g)}}catch(S){r={error:S}}finally{try{u&&!u.done&&(n=s.return)&&n.call(s)}finally{if(r)throw r.error}}return i.sort(dt),i},o.prototype.autoSuggest=function(e,t){var r,n,a,i;t===void 0&&(t={}),t=$($({},this._options.autoSuggestOptions),t);var s=new Map;try{for(var u=D(this.search(e,t)),l=u.next();!l.done;l=u.next()){var h=l.value,d=h.score,v=h.terms,m=v.join(" "),b=s.get(m);b!=null?(b.score+=d,b.count+=1):s.set(m,{score:d,terms:v,count:1})}}catch(L){r={error:L}}finally{try{l&&!l.done&&(n=u.return)&&n.call(u)}finally{if(r)throw r.error}}var E=[];try{for(var g=D(s),S=g.next();!S.done;S=g.next()){var T=W(S.value,2),b=T[0],C=T[1],d=C.score,v=C.terms,R=C.count;E.push({suggestion:b,terms:v,score:d/R})}}catch(L){a={error:L}}finally{try{S&&!S.done&&(i=g.return)&&i.call(g)}finally{if(a)throw a.error}}return E.sort(dt),E},Object.defineProperty(o.prototype,"documentCount",{get:function(){return this._documentCount},enumerable:!1,configurable:!0}),Object.defineProperty(o.prototype,"termCount",{get:function(){return this._index.size},enumerable:!1,configurable:!0}),o.loadJSON=function(e,t){if(t==null)throw new Error("MiniSearch: loadJSON should be given the same options used when serializing the index");return this.loadJS(JSON.parse(e),t)},o.getDefault=function(e){if(Pe.hasOwnProperty(e))return Le(Pe,e);throw new Error('MiniSearch: unknown option "'.concat(e,'"'))},o.loadJS=function(e,t){var r,n,a,i,s,u,l=e.index,h=e.documentCount,d=e.nextId,v=e.documentIds,m=e.fieldIds,b=e.fieldLength,E=e.averageFieldLength,g=e.storedFields,S=e.dirtCount,T=e.serializationVersion;if(T!==1&&T!==2)throw new Error("MiniSearch: cannot deserialize an index created with an incompatible version");var C=new o(t);C._documentCount=h,C._nextId=d,C._documentIds=Se(v),C._idToShortId=new Map,C._fieldIds=m,C._fieldLength=Se(b),C._avgFieldLength=E,C._storedFields=Se(g),C._dirtCount=S||0,C._index=new Re;try{for(var R=D(C._documentIds),L=R.next();!L.done;L=R.next()){var P=W(L.value,2),I=P[0],A=P[1];C._idToShortId.set(A,I)}}catch(y){r={error:y}}finally{try{L&&!L.done&&(n=R.return)&&n.call(R)}finally{if(r)throw r.error}}try{for(var _=D(l),O=_.next();!O.done;O=_.next()){var x=W(O.value,2),c=x[0],f=x[1],p=new Map;try{for(var N=(s=void 0,D(Object.keys(f))),k=N.next();!k.done;k=N.next()){var M=k.value,z=f[M];T===1&&(z=z.ds),p.set(parseInt(M,10),Se(z))}}catch(y){s={error:y}}finally{try{k&&!k.done&&(u=N.return)&&u.call(N)}finally{if(s)throw s.error}}C._index.set(c,p)}}catch(y){a={error:y}}finally{try{O&&!O.done&&(i=_.return)&&i.call(_)}finally{if(a)throw a.error}}return C},o.prototype.executeQuery=function(e,t){var r=this;if(t===void 0&&(t={}),typeof e!="string"){var n=$($($({},t),e),{queries:void 0}),a=e.queries.map(function(g){return r.executeQuery(g,n)});return this.combineResults(a,n.combineWith)}var i=this._options,s=i.tokenize,u=i.processTerm,l=i.searchOptions,h=$($({tokenize:s,processTerm:u},l),t),d=h.tokenize,v=h.processTerm,m=d(e).flatMap(function(g){return v(g)}).filter(function(g){return!!g}),b=m.map(Wr(h)),E=b.map(function(g){return r.executeQuerySpec(g,h)});return this.combineResults(E,h.combineWith)},o.prototype.executeQuerySpec=function(e,t){var r,n,a,i,s=$($({},this._options.searchOptions),t),u=(s.fields||this._options.fields).reduce(function(M,z){var y;return $($({},M),(y={},y[z]=Le(s.boost,z)||1,y))},{}),l=s.boostDocument,h=s.weights,d=s.maxFuzzy,v=s.bm25,m=$($({},ft.weights),h),b=m.fuzzy,E=m.prefix,g=this._index.get(e.term),S=this.termResults(e.term,e.term,1,g,u,l,v),T,C;if(e.prefix&&(T=this._index.atPrefix(e.term)),e.fuzzy){var R=e.fuzzy===!0?.2:e.fuzzy,L=R<1?Math.min(d,Math.round(e.term.length*R)):R;L&&(C=this._index.fuzzyGet(e.term,L))}if(T)try{for(var P=D(T),I=P.next();!I.done;I=P.next()){var A=W(I.value,2),_=A[0],O=A[1],x=_.length-e.term.length;if(x){C==null||C.delete(_);var c=E*_.length/(_.length+.3*x);this.termResults(e.term,_,c,O,u,l,v,S)}}}catch(M){r={error:M}}finally{try{I&&!I.done&&(n=P.return)&&n.call(P)}finally{if(r)throw r.error}}if(C)try{for(var f=D(C.keys()),p=f.next();!p.done;p=f.next()){var _=p.value,N=W(C.get(_),2),k=N[0],x=N[1];if(x){var c=b*_.length/(_.length+x);this.termResults(e.term,_,c,k,u,l,v,S)}}}catch(M){a={error:M}}finally{try{p&&!p.done&&(i=f.return)&&i.call(f)}finally{if(a)throw a.error}}return S},o.prototype.combineResults=function(e,t){if(t===void 0&&(t=Je),e.length===0)return new Map;var r=t.toLowerCase();return e.reduce(Br[r])||new Map},o.prototype.toJSON=function(){var e,t,r,n,a=[];try{for(var i=D(this._index),s=i.next();!s.done;s=i.next()){var u=W(s.value,2),l=u[0],h=u[1],d={};try{for(var v=(r=void 0,D(h)),m=v.next();!m.done;m=v.next()){var b=W(m.value,2),E=b[0],g=b[1];d[E]=Object.fromEntries(g)}}catch(S){r={error:S}}finally{try{m&&!m.done&&(n=v.return)&&n.call(v)}finally{if(r)throw r.error}}a.push([l,d])}}catch(S){e={error:S}}finally{try{s&&!s.done&&(t=i.return)&&t.call(i)}finally{if(e)throw e.error}}return{documentCount:this._documentCount,nextId:this._nextId,documentIds:Object.fromEntries(this._documentIds),fieldIds:this._fieldIds,fieldLength:Object.fromEntries(this._fieldLength),averageFieldLength:this._avgFieldLength,storedFields:Object.fromEntries(this._storedFields),dirtCount:this._dirtCount,index:a,serializationVersion:2}},o.prototype.termResults=function(e,t,r,n,a,i,s,u){var l,h,d,v,m;if(u===void 0&&(u=new Map),n==null)return u;try{for(var b=D(Object.keys(a)),E=b.next();!E.done;E=b.next()){var g=E.value,S=a[g],T=this._fieldIds[g],C=n.get(T);if(C!=null){var R=C.size,L=this._avgFieldLength[T];try{for(var P=(d=void 0,D(C.keys())),I=P.next();!I.done;I=P.next()){var A=I.value;if(!this._documentIds.has(A)){this.removeTerm(T,A,t),R-=1;continue}var _=i?i(this._documentIds.get(A),t,this._storedFields.get(A)):1;if(_){var O=C.get(A),x=this._fieldLength.get(A)[T],c=$r(O,R,this._documentCount,x,L,s),f=r*S*_*c,p=u.get(A);if(p){p.score+=f,Kr(p.terms,e);var N=Le(p.match,t);N?N.push(g):p.match[t]=[g]}else u.set(A,{score:f,terms:[e],match:(m={},m[t]=[g],m)})}}}catch(k){d={error:k}}finally{try{I&&!I.done&&(v=P.return)&&v.call(P)}finally{if(d)throw d.error}}}}}catch(k){l={error:k}}finally{try{E&&!E.done&&(h=b.return)&&h.call(b)}finally{if(l)throw l.error}}return u},o.prototype.addTerm=function(e,t,r){var n=this._index.fetch(r,vt),a=n.get(e);if(a==null)a=new Map,a.set(t,1),n.set(e,a);else{var i=a.get(t);a.set(t,(i||0)+1)}},o.prototype.removeTerm=function(e,t,r){if(!this._index.has(r)){this.warnDocumentChanged(t,e,r);return}var n=this._index.fetch(r,vt),a=n.get(e);a==null||a.get(t)==null?this.warnDocumentChanged(t,e,r):a.get(t)<=1?a.size<=1?n.delete(e):a.delete(t):a.set(t,a.get(t)-1),this._index.get(r).size===0&&this._index.delete(r)},o.prototype.warnDocumentChanged=function(e,t,r){var n,a;try{for(var i=D(Object.keys(this._fieldIds)),s=i.next();!s.done;s=i.next()){var u=s.value;if(this._fieldIds[u]===t){this._options.logger("warn","MiniSearch: document with ID ".concat(this._documentIds.get(e),' has changed before removal: term "').concat(r,'" was not present in field "').concat(u,'". Removing a document after it has changed can corrupt the index!'),"version_conflict");return}}}catch(l){n={error:l}}finally{try{s&&!s.done&&(a=i.return)&&a.call(i)}finally{if(n)throw n.error}}},o.prototype.addDocumentId=function(e){var t=this._nextId;return this._idToShortId.set(e,t),this._documentIds.set(t,e),this._documentCount+=1,this._nextId+=1,t},o.prototype.addFields=function(e){for(var t=0;t(Gt("data-v-2813d7e3"),o=o(),qt(),o),Ur=["aria-owns"],Hr={class:"shell"},Gr=["title"],qr=Y(()=>F("svg",{class:"search-icon",width:"18",height:"18",viewBox:"0 0 24 24","aria-hidden":"true"},[F("g",{fill:"none",stroke:"currentColor","stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2"},[F("circle",{cx:"11",cy:"11",r:"8"}),F("path",{d:"m21 21l-4.35-4.35"})])],-1)),Qr=[qr],Yr={class:"search-actions before"},Zr=["title"],Xr=Y(()=>F("svg",{width:"18",height:"18",viewBox:"0 0 24 24","aria-hidden":"true"},[F("path",{fill:"none",stroke:"currentColor","stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"M19 12H5m7 7l-7-7l7-7"})],-1)),ea=[Xr],ta=["placeholder"],ra={class:"search-actions"},aa=["title"],na=Y(()=>F("svg",{width:"18",height:"18",viewBox:"0 0 24 24","aria-hidden":"true"},[F("path",{fill:"none",stroke:"currentColor","stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"M3 14h7v7H3zM3 3h7v7H3zm11 1h7m-7 5h7m-7 6h7m-7 5h7"})],-1)),ia=[na],oa=["disabled","title"],sa=Y(()=>F("svg",{width:"18",height:"18",viewBox:"0 0 24 24","aria-hidden":"true"},[F("path",{fill:"none",stroke:"currentColor","stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"M20 5H9l-7 7l7 7h11a2 2 0 0 0 2-2V7a2 2 0 0 0-2-2Zm-2 4l-6 6m0-6l6 6"})],-1)),ua=[sa],la=["id","role","aria-labelledby"],ca=["aria-selected"],fa=["href","aria-label","onMouseenter","onFocusin"],ha={class:"titles"},da=Y(()=>F("span",{class:"title-icon"},"#",-1)),va=["innerHTML"],pa=Y(()=>F("svg",{width:"18",height:"18",viewBox:"0 0 24 24"},[F("path",{fill:"none",stroke:"currentColor","stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"m9 18l6-6l-6-6"})],-1)),ya={class:"title main"},ma=["innerHTML"],ga={key:0,class:"excerpt-wrapper"},ba={key:0,class:"excerpt",inert:""},wa=["innerHTML"],xa=Y(()=>F("div",{class:"excerpt-gradient-bottom"},null,-1)),Fa=Y(()=>F("div",{class:"excerpt-gradient-top"},null,-1)),Ea={key:0,class:"no-results"},Sa={class:"search-keyboard-shortcuts"},Aa=["aria-label"],Ca=Y(()=>F("svg",{width:"14",height:"14",viewBox:"0 0 24 24"},[F("path",{fill:"none",stroke:"currentColor","stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"M12 19V5m-7 7l7-7l7 7"})],-1)),Ta=[Ca],ka=["aria-label"],Na=Y(()=>F("svg",{width:"14",height:"14",viewBox:"0 0 24 24"},[F("path",{fill:"none",stroke:"currentColor","stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"M12 5v14m7-7l-7 7l-7-7"})],-1)),Ia=[Na],Da=["aria-label"],_a=Y(()=>F("svg",{width:"14",height:"14",viewBox:"0 0 24 24"},[F("g",{fill:"none",stroke:"currentcolor","stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2"},[F("path",{d:"m9 10l-5 5l5 5"}),F("path",{d:"M20 4v7a4 4 0 0 1-4 4H4"})])],-1)),Oa=[_a],Ra=["aria-label"],Ma=It({__name:"VPLocalSearchBox",emits:["close"],setup(o,{emit:e}){var M,z;const t=e,r=be(),n=be(),a=be(rr),i=er(),{activate:s}=Nr(r,{immediate:!0,allowOutsideClick:!0,clickOutsideDeactivates:!0,escapeDeactivates:!0}),{localeIndex:u,theme:l}=i,h=et(async()=>{var y,w,B,G,j,K,V,J,q;return nt(zr.loadJSON((B=await((w=(y=a.value)[u.value])==null?void 0:w.call(y)))==null?void 0:B.default,{fields:["title","titles","text"],storeFields:["title","titles"],searchOptions:{fuzzy:.2,prefix:!0,boost:{title:4,text:2,titles:1},...((G=l.value.search)==null?void 0:G.provider)==="local"&&((K=(j=l.value.search.options)==null?void 0:j.miniSearch)==null?void 0:K.searchOptions)},...((V=l.value.search)==null?void 0:V.provider)==="local"&&((q=(J=l.value.search.options)==null?void 0:J.miniSearch)==null?void 0:q.options)}))}),v=we(()=>{var y,w;return((y=l.value.search)==null?void 0:y.provider)==="local"&&((w=l.value.search.options)==null?void 0:w.disableQueryPersistence)===!0}).value?ie(""):Dt("vitepress:local-search-filter",""),m=_t("vitepress:local-search-detailed-list",((M=l.value.search)==null?void 0:M.provider)==="local"&&((z=l.value.search.options)==null?void 0:z.detailedView)===!0),b=we(()=>{var y,w,B;return((y=l.value.search)==null?void 0:y.provider)==="local"&&(((w=l.value.search.options)==null?void 0:w.disableDetailedView)===!0||((B=l.value.search.options)==null?void 0:B.detailedView)===!1)}),E=we(()=>{var w,B,G,j,K,V,J;const y=((w=l.value.search)==null?void 0:w.options)??l.value.algolia;return((K=(j=(G=(B=y==null?void 0:y.locales)==null?void 0:B[u.value])==null?void 0:G.translations)==null?void 0:j.button)==null?void 0:K.buttonText)||((J=(V=y==null?void 0:y.translations)==null?void 0:V.button)==null?void 0:J.buttonText)||"Search"});Ot(()=>{b.value&&(m.value=!1)});const g=be([]),S=ie(!1);Be(v,()=>{S.value=!1});const T=et(async()=>{if(n.value)return nt(new Dr(n.value))},null);Rt(()=>[h.value,v.value,m.value],async([y,w,B],G,j)=>{var Ue,He,Ge,qe;let K=!1;if(j(()=>{K=!0}),!y)return;g.value=y.search(w).slice(0,16),S.value=!0;const V=B?await Promise.all(g.value.map(Q=>C(Q.id))):[];if(K)return;const J=new Map;for(const{id:Q,mod:re}of V){const ae=Q.slice(0,Q.indexOf("#"));let ee=J.get(ae);if(ee)continue;ee=new Map,J.set(ae,ee);const Z=re.default??re;if(Z!=null&&Z.render||Z!=null&&Z.setup){const ne=Qt(Z);ne.config.warnHandler=()=>{},ne.provide(Yt,i),Object.defineProperties(ne.config.globalProperties,{$frontmatter:{get(){return i.frontmatter.value}},$params:{get(){return i.page.value.params}}});const Qe=document.createElement("div");ne.mount(Qe),Qe.querySelectorAll("h1, h2, h3, h4, h5, h6").forEach(fe=>{var Xe;const ge=(Xe=fe.querySelector("a"))==null?void 0:Xe.getAttribute("href"),Ye=(ge==null?void 0:ge.startsWith("#"))&&ge.slice(1);if(!Ye)return;let Ze="";for(;(fe=fe.nextElementSibling)&&!/^h[1-6]$/i.test(fe.tagName);)Ze+=fe.outerHTML;ee.set(Ye,Ze)}),ne.unmount()}if(K)return}const q=new Set;if(g.value=g.value.map(Q=>{const[re,ae]=Q.id.split("#"),ee=J.get(re),Z=(ee==null?void 0:ee.get(ae))??"";for(const ne in Q.match)q.add(ne);return{...Q,text:Z}}),await he(),K)return;await new Promise(Q=>{var re;(re=T.value)==null||re.unmark({done:()=>{var ae;(ae=T.value)==null||ae.markRegExp(k(q),{done:Q})}})});const Ie=((Ue=r.value)==null?void 0:Ue.querySelectorAll(".result .excerpt"))??[];for(const Q of Ie)(He=Q.querySelector('mark[data-markjs="true"]'))==null||He.scrollIntoView({block:"center"});(qe=(Ge=n.value)==null?void 0:Ge.firstElementChild)==null||qe.scrollIntoView({block:"start"})},{debounce:200,immediate:!0});async function C(y){const w=Zt(y.slice(0,y.indexOf("#")));try{if(!w)throw new Error(`Cannot find file for id: ${y}`);return{id:y,mod:await pt(()=>import(w),[])}}catch(B){return console.error(B),{id:y,mod:{}}}}const R=ie(),L=we(()=>{var y;return((y=v.value)==null?void 0:y.length)<=0});function P(y=!0){var w,B;(w=R.value)==null||w.focus(),y&&((B=R.value)==null||B.select())}De(()=>{P()});function I(y){y.pointerType==="mouse"&&P()}const A=ie(-1),_=ie(!1);Be(g,y=>{A.value=y.length?0:-1,O()});function O(){he(()=>{const y=document.querySelector(".result.selected");y&&y.scrollIntoView({block:"nearest"})})}xe("ArrowUp",y=>{y.preventDefault(),A.value--,A.value<0&&(A.value=g.value.length-1),_.value=!0,O()}),xe("ArrowDown",y=>{y.preventDefault(),A.value++,A.value>=g.value.length&&(A.value=0),_.value=!0,O()});const x=Mt();xe("Enter",y=>{if(y.target instanceof HTMLButtonElement&&y.target.type!=="submit")return;const w=g.value[A.value];if(y.target instanceof HTMLInputElement&&!w){y.preventDefault();return}w&&(x.go(w.id),t("close"))}),xe("Escape",()=>{t("close")});const c={modal:{displayDetails:"Display detailed list",resetButtonTitle:"Reset search",backButtonTitle:"Close search",noResultsText:"No results for",footer:{selectText:"to select",selectKeyAriaLabel:"enter",navigateText:"to navigate",navigateUpKeyAriaLabel:"up arrow",navigateDownKeyAriaLabel:"down arrow",closeText:"to close",closeKeyAriaLabel:"escape"}}},f=Lt(tr)(Pt(()=>{var y;return(y=l.value.search)==null?void 0:y.options}),c);De(()=>{window.history.pushState(null,"",null)}),zt("popstate",y=>{y.preventDefault(),t("close")});const p=Bt(Vt?document.body:null);De(()=>{he(()=>{p.value=!0,he().then(()=>s())})}),$t(()=>{p.value=!1});function N(){v.value="",he().then(()=>P(!1))}function k(y){return new RegExp([...y].sort((w,B)=>B.length-w.length).map(w=>`(${w.replace(/[|\\{}()[\]^$+*?.]/g,"\\$&").replace(/-/g,"\\x2d")})`).join("|"),"gi")}return(y,w)=>{var B,G,j,K;return X(),Wt(Ht,{to:"body"},[F("div",{ref_key:"el",ref:r,role:"button","aria-owns":(B=g.value)!=null&&B.length?"localsearch-list":void 0,"aria-expanded":"true","aria-haspopup":"listbox","aria-labelledby":"localsearch-label",class:"VPLocalSearchBox"},[F("div",{class:"backdrop",onClick:w[0]||(w[0]=V=>y.$emit("close"))}),F("div",Hr,[F("form",{class:"search-bar",onPointerup:w[4]||(w[4]=V=>I(V)),onSubmit:w[5]||(w[5]=jt(()=>{},["prevent"]))},[F("label",{title:E.value,id:"localsearch-label",for:"localsearch-input"},Qr,8,Gr),F("div",Yr,[F("button",{class:"back-button",title:U(f)("modal.backButtonTitle"),onClick:w[1]||(w[1]=V=>y.$emit("close"))},ea,8,Zr)]),Kt(F("input",{ref_key:"searchInput",ref:R,"onUpdate:modelValue":w[2]||(w[2]=V=>Ut(v)?v.value=V:null),placeholder:E.value,id:"localsearch-input","aria-labelledby":"localsearch-label",class:"search-input"},null,8,ta),[[Jt,U(v)]]),F("div",ra,[b.value?Fe("",!0):(X(),te("button",{key:0,class:tt(["toggle-layout-button",{"detailed-list":U(m)}]),type:"button",title:U(f)("modal.displayDetails"),onClick:w[3]||(w[3]=V=>A.value>-1&&(m.value=!U(m)))},ia,10,aa)),F("button",{class:"clear-button",type:"reset",disabled:L.value,title:U(f)("modal.resetButtonTitle"),onClick:N},ua,8,oa)])],32),F("ul",{ref_key:"resultsEl",ref:n,id:(G=g.value)!=null&&G.length?"localsearch-list":void 0,role:(j=g.value)!=null&&j.length?"listbox":void 0,"aria-labelledby":(K=g.value)!=null&&K.length?"localsearch-label":void 0,class:"results",onMousemove:w[7]||(w[7]=V=>_.value=!1)},[(X(!0),te(at,null,rt(g.value,(V,J)=>(X(),te("li",{key:V.id,role:"option","aria-selected":A.value===J?"true":"false"},[F("a",{href:V.id,class:tt(["result",{selected:A.value===J}]),"aria-label":[...V.titles,V.title].join(" > "),onMouseenter:q=>!_.value&&(A.value=J),onFocusin:q=>A.value=J,onClick:w[6]||(w[6]=q=>y.$emit("close"))},[F("div",null,[F("div",ha,[da,(X(!0),te(at,null,rt(V.titles,(q,Ie)=>(X(),te("span",{key:Ie,class:"title"},[F("span",{class:"text",innerHTML:q},null,8,va),pa]))),128)),F("span",ya,[F("span",{class:"text",innerHTML:V.title},null,8,ma)])]),U(m)?(X(),te("div",ga,[V.text?(X(),te("div",ba,[F("div",{class:"vp-doc",innerHTML:V.text},null,8,wa)])):Fe("",!0),xa,Fa])):Fe("",!0)])],42,fa)],8,ca))),128)),U(v)&&!g.value.length&&S.value?(X(),te("li",Ea,[de(ve(U(f)("modal.noResultsText"))+' "',1),F("strong",null,ve(U(v)),1),de('" ')])):Fe("",!0)],40,la),F("div",Sa,[F("span",null,[F("kbd",{"aria-label":U(f)("modal.footer.navigateUpKeyAriaLabel")},Ta,8,Aa),F("kbd",{"aria-label":U(f)("modal.footer.navigateDownKeyAriaLabel")},Ia,8,ka),de(" "+ve(U(f)("modal.footer.navigateText")),1)]),F("span",null,[F("kbd",{"aria-label":U(f)("modal.footer.selectKeyAriaLabel")},Oa,8,Da),de(" "+ve(U(f)("modal.footer.selectText")),1)]),F("span",null,[F("kbd",{"aria-label":U(f)("modal.footer.closeKeyAriaLabel")},"esc",8,Ra),de(" "+ve(U(f)("modal.footer.closeText")),1)])])])],8,Ur)])}}});const Va=Xt(Ma,[["__scopeId","data-v-2813d7e3"]]);export{Va as default}; diff --git a/assets/chunks/arabica_versions.d7daf483.js b/assets/chunks/arabica_versions.d7daf483.js new file mode 100644 index 00000000000..4c2c86719ef --- /dev/null +++ b/assets/chunks/arabica_versions.d7daf483.js @@ -0,0 +1 @@ +const a=Object.freeze({"app-latest-tag":"v1.7.0","app-latest-sha":"915847191e80d836f862eea2664949d9a240abea","core-latest-tag":"v1.35.0-tm-v0.34.29","core-latest-sha":"f95a434e674610b3a2c5e7d26bd87da69ccaa213","node-latest-tag":"v0.13.3","node-latest-sha":"05238b3e087eb9ecd3b9684cd0125f2400f6f0c7"});export{a}; diff --git a/assets/chunks/constants.dc70dfe7.js b/assets/chunks/constants.dc70dfe7.js new file mode 100644 index 00000000000..b6e04813abb --- /dev/null +++ b/assets/chunks/constants.dc70dfe7.js @@ -0,0 +1 @@ +const a=Object.freeze({golangNodeMainnet:"1.21.1",golangNodeMocha:"1.22.0",golangNodeArabica:"1.22.0",golangApp:"1.22.0",golangCore:"1.22.0",golang:"1.21.1",arabicaChainId:"arabica-11",mainnetChainId:"celestia",mochaChainId:"mocha-4",arabicaRollkitVersion:"v0.10.5",mochaRollkitVersion:"v0.10.5",mainnetRollkitVersion:"v0.10.5",localCelestiaDevnetVersion:"v0.8.2",golangBlobstream:"1.21.4",orchrelayVersion:"v1.0.1",mochaRpcUrl:"https://rpc-mocha.pops.one/",mochaRestUrl:"https://api-mocha.pops.one/",arabicaRpcUrl:"https://rpc.celestia-arabica-11.com/",arabicaRestUrl:"https://api.celestia-arabica-11.com",mainnetRpcUrl:"https://rpc.lunaroasis.net/",mainnetRestUrl:"https://api.lunaroasis.net/"});export{a as c}; diff --git a/assets/chunks/framework.51d6c45b.js b/assets/chunks/framework.51d6c45b.js new file mode 100644 index 00000000000..4c451b75b45 --- /dev/null +++ b/assets/chunks/framework.51d6c45b.js @@ -0,0 +1,2 @@ +function gr(e,t){const n=Object.create(null),r=e.split(",");for(let s=0;s!!n[s.toLowerCase()]:s=>!!n[s]}const te={},mt=[],Oe=()=>{},fi=()=>!1,di=/^on[^a-z]/,zt=e=>di.test(e),mr=e=>e.startsWith("onUpdate:"),ue=Object.assign,yr=(e,t)=>{const n=e.indexOf(t);n>-1&&e.splice(n,1)},hi=Object.prototype.hasOwnProperty,Y=(e,t)=>hi.call(e,t),k=Array.isArray,yt=e=>Tn(e)==="[object Map]",Vs=e=>Tn(e)==="[object Set]",K=e=>typeof e=="function",re=e=>typeof e=="string",_r=e=>typeof e=="symbol",ee=e=>e!==null&&typeof e=="object",qs=e=>(ee(e)||K(e))&&K(e.then)&&K(e.catch),zs=Object.prototype.toString,Tn=e=>zs.call(e),pi=e=>Tn(e).slice(8,-1),Ys=e=>Tn(e)==="[object Object]",br=e=>re(e)&&e!=="NaN"&&e[0]!=="-"&&""+parseInt(e,10)===e,It=gr(",key,ref,ref_for,ref_key,onVnodeBeforeMount,onVnodeMounted,onVnodeBeforeUpdate,onVnodeUpdated,onVnodeBeforeUnmount,onVnodeUnmounted"),An=e=>{const t=Object.create(null);return n=>t[n]||(t[n]=e(n))},gi=/-(\w)/g,Ne=An(e=>e.replace(gi,(t,n)=>n?n.toUpperCase():"")),mi=/\B([A-Z])/g,ft=An(e=>e.replace(mi,"-$1").toLowerCase()),Sn=An(e=>e.charAt(0).toUpperCase()+e.slice(1)),dn=An(e=>e?`on${Sn(e)}`:""),ut=(e,t)=>!Object.is(e,t),hn=(e,t)=>{for(let n=0;n{Object.defineProperty(e,t,{configurable:!0,enumerable:!1,value:n})},Gn=e=>{const t=parseFloat(e);return isNaN(t)?e:t},yi=e=>{const t=re(e)?Number(e):NaN;return isNaN(t)?e:t};let Yr;const er=()=>Yr||(Yr=typeof globalThis<"u"?globalThis:typeof self<"u"?self:typeof window<"u"?window:typeof global<"u"?global:{});function vr(e){if(k(e)){const t={};for(let n=0;n{if(n){const r=n.split(bi);r.length>1&&(t[r[0].trim()]=r[1].trim())}}),t}function wr(e){let t="";if(re(e))t=e;else if(k(e))for(let n=0;nre(e)?e:e==null?"":k(e)||ee(e)&&(e.toString===zs||!K(e.toString))?JSON.stringify(e,Xs,2):String(e),Xs=(e,t)=>t&&t.__v_isRef?Xs(e,t.value):yt(t)?{[`Map(${t.size})`]:[...t.entries()].reduce((n,[r,s])=>(n[`${r} =>`]=s,n),{})}:Vs(t)?{[`Set(${t.size})`]:[...t.values()]}:ee(t)&&!k(t)&&!Ys(t)?String(t):t;let be;class xi{constructor(t=!1){this.detached=t,this._active=!0,this.effects=[],this.cleanups=[],this.parent=be,!t&&be&&(this.index=(be.scopes||(be.scopes=[])).push(this)-1)}get active(){return this._active}run(t){if(this._active){const n=be;try{return be=this,t()}finally{be=n}}}on(){be=this}off(){be=this.parent}stop(t){if(this._active){let n,r;for(n=0,r=this.effects.length;n{const t=new Set(e);return t.w=0,t.n=0,t},Zs=e=>(e.w&Xe)>0,Gs=e=>(e.n&Xe)>0,Si=({deps:e})=>{if(e.length)for(let t=0;t{const{deps:t}=e;if(t.length){let n=0;for(let r=0;r{(f==="length"||f>=c)&&l.push(a)})}else switch(n!==void 0&&l.push(i.get(n)),t){case"add":k(e)?br(n)&&l.push(i.get("length")):(l.push(i.get(lt)),yt(e)&&l.push(i.get(nr)));break;case"delete":k(e)||(l.push(i.get(lt)),yt(e)&&l.push(i.get(nr)));break;case"set":yt(e)&&l.push(i.get(lt));break}if(l.length===1)l[0]&&rr(l[0]);else{const c=[];for(const a of l)a&&c.push(...a);rr(Cr(c))}}function rr(e,t){const n=k(e)?e:[...e];for(const r of n)r.computed&&Xr(r);for(const r of n)r.computed||Xr(r)}function Xr(e,t){(e!==Se||e.allowRecurse)&&(e.scheduler?e.scheduler():e.run())}function Oi(e,t){var n;return(n=yn.get(e))==null?void 0:n.get(t)}const Pi=gr("__proto__,__v_isRef,__isVue"),no=new Set(Object.getOwnPropertyNames(Symbol).filter(e=>e!=="arguments"&&e!=="caller").map(e=>Symbol[e]).filter(_r)),Qr=Mi();function Mi(){const e={};return["includes","indexOf","lastIndexOf"].forEach(t=>{e[t]=function(...n){const r=J(this);for(let o=0,i=this.length;o{e[t]=function(...n){At();const r=J(this)[t].apply(this,n);return St(),r}}),e}function Fi(e){const t=J(this);return me(t,"has",e),t.hasOwnProperty(e)}class ro{constructor(t=!1,n=!1){this._isReadonly=t,this._shallow=n}get(t,n,r){const s=this._isReadonly,o=this._shallow;if(n==="__v_isReactive")return!s;if(n==="__v_isReadonly")return s;if(n==="__v_isShallow")return o;if(n==="__v_raw"&&r===(s?o?Vi:lo:o?io:oo).get(t))return t;const i=k(t);if(!s){if(i&&Y(Qr,n))return Reflect.get(Qr,n,r);if(n==="hasOwnProperty")return Fi}const l=Reflect.get(t,n,r);return(_r(n)?no.has(n):Pi(n))||(s||me(t,"get",n),o)?l:ce(l)?i&&br(n)?l:l.value:ee(l)?s?Pn(l):On(l):l}}class so extends ro{constructor(t=!1){super(!1,t)}set(t,n,r,s){let o=t[n];if(Ct(o)&&ce(o)&&!ce(r))return!1;if(!this._shallow&&(!_n(r)&&!Ct(r)&&(o=J(o),r=J(r)),!k(t)&&ce(o)&&!ce(r)))return o.value=r,!0;const i=k(t)&&br(n)?Number(n)e,Rn=e=>Reflect.getPrototypeOf(e);function Zt(e,t,n=!1,r=!1){e=e.__v_raw;const s=J(e),o=J(t);n||(ut(t,o)&&me(s,"get",t),me(s,"get",o));const{has:i}=Rn(s),l=r?xr:n?Sr:Bt;if(i.call(s,t))return l(e.get(t));if(i.call(s,o))return l(e.get(o));e!==s&&e.get(t)}function Gt(e,t=!1){const n=this.__v_raw,r=J(n),s=J(e);return t||(ut(e,s)&&me(r,"has",e),me(r,"has",s)),e===s?n.has(e):n.has(e)||n.has(s)}function en(e,t=!1){return e=e.__v_raw,!t&&me(J(e),"iterate",lt),Reflect.get(e,"size",e)}function Zr(e){e=J(e);const t=J(this);return Rn(t).has.call(t,e)||(t.add(e),je(t,"add",e,e)),this}function Gr(e,t){t=J(t);const n=J(this),{has:r,get:s}=Rn(n);let o=r.call(n,e);o||(e=J(e),o=r.call(n,e));const i=s.call(n,e);return n.set(e,t),o?ut(t,i)&&je(n,"set",e,t):je(n,"add",e,t),this}function es(e){const t=J(this),{has:n,get:r}=Rn(t);let s=n.call(t,e);s||(e=J(e),s=n.call(t,e)),r&&r.call(t,e);const o=t.delete(e);return s&&je(t,"delete",e,void 0),o}function ts(){const e=J(this),t=e.size!==0,n=e.clear();return t&&je(e,"clear",void 0,void 0),n}function tn(e,t){return function(r,s){const o=this,i=o.__v_raw,l=J(i),c=t?xr:e?Sr:Bt;return!e&&me(l,"iterate",lt),i.forEach((a,f)=>r.call(s,c(a),c(f),o))}}function nn(e,t,n){return function(...r){const s=this.__v_raw,o=J(s),i=yt(o),l=e==="entries"||e===Symbol.iterator&&i,c=e==="keys"&&i,a=s[e](...r),f=n?xr:t?Sr:Bt;return!t&&me(o,"iterate",c?nr:lt),{next(){const{value:h,done:g}=a.next();return g?{value:h,done:g}:{value:l?[f(h[0]),f(h[1])]:f(h),done:g}},[Symbol.iterator](){return this}}}}function ke(e){return function(...t){return e==="delete"?!1:this}}function $i(){const e={get(o){return Zt(this,o)},get size(){return en(this)},has:Gt,add:Zr,set:Gr,delete:es,clear:ts,forEach:tn(!1,!1)},t={get(o){return Zt(this,o,!1,!0)},get size(){return en(this)},has:Gt,add:Zr,set:Gr,delete:es,clear:ts,forEach:tn(!1,!0)},n={get(o){return Zt(this,o,!0)},get size(){return en(this,!0)},has(o){return Gt.call(this,o,!0)},add:ke("add"),set:ke("set"),delete:ke("delete"),clear:ke("clear"),forEach:tn(!0,!1)},r={get(o){return Zt(this,o,!0,!0)},get size(){return en(this,!0)},has(o){return Gt.call(this,o,!0)},add:ke("add"),set:ke("set"),delete:ke("delete"),clear:ke("clear"),forEach:tn(!0,!0)};return["keys","values","entries",Symbol.iterator].forEach(o=>{e[o]=nn(o,!1,!1),n[o]=nn(o,!0,!1),t[o]=nn(o,!1,!0),r[o]=nn(o,!0,!0)}),[e,n,t,r]}const[ji,Di,Bi,ki]=$i();function Tr(e,t){const n=t?e?ki:Bi:e?Di:ji;return(r,s,o)=>s==="__v_isReactive"?!e:s==="__v_isReadonly"?e:s==="__v_raw"?r:Reflect.get(Y(n,s)&&s in r?n:r,s,o)}const Ui={get:Tr(!1,!1)},Ki={get:Tr(!1,!0)},Wi={get:Tr(!0,!1)},oo=new WeakMap,io=new WeakMap,lo=new WeakMap,Vi=new WeakMap;function qi(e){switch(e){case"Object":case"Array":return 1;case"Map":case"Set":case"WeakMap":case"WeakSet":return 2;default:return 0}}function zi(e){return e.__v_skip||!Object.isExtensible(e)?0:qi(pi(e))}function On(e){return Ct(e)?e:Ar(e,!1,Li,Ui,oo)}function Yi(e){return Ar(e,!1,Hi,Ki,io)}function Pn(e){return Ar(e,!0,Ni,Wi,lo)}function Ar(e,t,n,r,s){if(!ee(e)||e.__v_raw&&!(t&&e.__v_isReactive))return e;const o=s.get(e);if(o)return o;const i=zi(e);if(i===0)return e;const l=new Proxy(e,i===2?r:n);return s.set(e,l),l}function _t(e){return Ct(e)?_t(e.__v_raw):!!(e&&e.__v_isReactive)}function Ct(e){return!!(e&&e.__v_isReadonly)}function _n(e){return!!(e&&e.__v_isShallow)}function co(e){return _t(e)||Ct(e)}function J(e){const t=e&&e.__v_raw;return t?J(t):e}function Lt(e){return mn(e,"__v_skip",!0),e}const Bt=e=>ee(e)?On(e):e,Sr=e=>ee(e)?Pn(e):e;function Rr(e){ze&&Se&&(e=J(e),to(e.dep||(e.dep=Cr())))}function Or(e,t){e=J(e);const n=e.dep;n&&rr(n)}function ce(e){return!!(e&&e.__v_isRef===!0)}function ae(e){return ao(e,!1)}function Pr(e){return ao(e,!0)}function ao(e,t){return ce(e)?e:new Ji(e,t)}class Ji{constructor(t,n){this.__v_isShallow=n,this.dep=void 0,this.__v_isRef=!0,this._rawValue=n?t:J(t),this._value=n?t:Bt(t)}get value(){return Rr(this),this._value}set value(t){const n=this.__v_isShallow||_n(t)||Ct(t);t=n?t:J(t),ut(t,this._rawValue)&&(this._rawValue=t,this._value=n?t:Bt(t),Or(this))}}function Mr(e){return ce(e)?e.value:e}const Xi={get:(e,t,n)=>Mr(Reflect.get(e,t,n)),set:(e,t,n,r)=>{const s=e[t];return ce(s)&&!ce(n)?(s.value=n,!0):Reflect.set(e,t,n,r)}};function uo(e){return _t(e)?e:new Proxy(e,Xi)}class Qi{constructor(t){this.dep=void 0,this.__v_isRef=!0;const{get:n,set:r}=t(()=>Rr(this),()=>Or(this));this._get=n,this._set=r}get value(){return this._get()}set value(t){this._set(t)}}function Zi(e){return new Qi(e)}class Gi{constructor(t,n,r){this._object=t,this._key=n,this._defaultValue=r,this.__v_isRef=!0}get value(){const t=this._object[this._key];return t===void 0?this._defaultValue:t}set value(t){this._object[this._key]=t}get dep(){return Oi(J(this._object),this._key)}}class el{constructor(t){this._getter=t,this.__v_isRef=!0,this.__v_isReadonly=!0}get value(){return this._getter()}}function tl(e,t,n){return ce(e)?e:K(e)?new el(e):ee(e)&&arguments.length>1?nl(e,t,n):ae(e)}function nl(e,t,n){const r=e[t];return ce(r)?r:new Gi(e,t,n)}class rl{constructor(t,n,r,s){this._setter=n,this.dep=void 0,this.__v_isRef=!0,this.__v_isReadonly=!1,this._dirty=!0,this.effect=new Er(t,()=>{this._dirty||(this._dirty=!0,Or(this))}),this.effect.computed=this,this.effect.active=this._cacheable=!s,this.__v_isReadonly=r}get value(){const t=J(this);return Rr(t),(t._dirty||!t._cacheable)&&(t._dirty=!1,t._value=t.effect.run()),t._value}set value(t){this._setter(t)}}function sl(e,t,n=!1){let r,s;const o=K(e);return o?(r=e,s=Oe):(r=e.get,s=e.set),new rl(r,s,o||!s,n)}function Ye(e,t,n,r){let s;try{s=r?e(...r):e()}catch(o){Yt(o,t,n)}return s}function xe(e,t,n,r){if(K(e)){const o=Ye(e,t,n,r);return o&&qs(o)&&o.catch(i=>{Yt(i,t,n)}),o}const s=[];for(let o=0;o>>1;Ut(de[r])Le&&de.splice(t,1)}function cl(e){k(e)?bt.push(...e):(!$e||!$e.includes(e,e.allowRecurse?rt+1:rt))&&bt.push(e),ho()}function ns(e,t=kt?Le+1:0){for(;tUt(n)-Ut(r)),rt=0;rt<$e.length;rt++)$e[rt]();$e=null,rt=0}}const Ut=e=>e.id==null?1/0:e.id,al=(e,t)=>{const n=Ut(e)-Ut(t);if(n===0){if(e.pre&&!t.pre)return-1;if(t.pre&&!e.pre)return 1}return n};function po(e){sr=!1,kt=!0,de.sort(al);const t=Oe;try{for(Le=0;Lere(w)?w.trim():w)),h&&(s=n.map(Gn))}let l,c=r[l=dn(t)]||r[l=dn(Ne(t))];!c&&o&&(c=r[l=dn(ft(t))]),c&&xe(c,e,6,s);const a=r[l+"Once"];if(a){if(!e.emitted)e.emitted={};else if(e.emitted[l])return;e.emitted[l]=!0,xe(a,e,6,s)}}function go(e,t,n=!1){const r=t.emitsCache,s=r.get(e);if(s!==void 0)return s;const o=e.emits;let i={},l=!1;if(!K(e)){const c=a=>{const f=go(a,t,!0);f&&(l=!0,ue(i,f))};!n&&t.mixins.length&&t.mixins.forEach(c),e.extends&&c(e.extends),e.mixins&&e.mixins.forEach(c)}return!o&&!l?(ee(e)&&r.set(e,null),null):(k(o)?o.forEach(c=>i[c]=null):ue(i,o),ee(e)&&r.set(e,i),i)}function In(e,t){return!e||!zt(t)?!1:(t=t.slice(2).replace(/Once$/,""),Y(e,t[0].toLowerCase()+t.slice(1))||Y(e,ft(t))||Y(e,t))}let fe=null,Ln=null;function vn(e){const t=fe;return fe=e,Ln=e&&e.type.__scopeId||null,t}function La(e){Ln=e}function Na(){Ln=null}function fl(e,t=fe,n){if(!t||e._n)return e;const r=(...s)=>{r._d&&gs(-1);const o=vn(t);let i;try{i=e(...s)}finally{vn(o),r._d&&gs(1)}return i};return r._n=!0,r._c=!0,r._d=!0,r}function Kn(e){const{type:t,vnode:n,proxy:r,withProxy:s,props:o,propsOptions:[i],slots:l,attrs:c,emit:a,render:f,renderCache:h,data:g,setupState:w,ctx:E,inheritAttrs:T}=e;let H,m;const _=vn(e);try{if(n.shapeFlag&4){const S=s||r;H=Ae(f.call(S,S,h,o,w,g,E)),m=c}else{const S=t;H=Ae(S.length>1?S(o,{attrs:c,slots:l,emit:a}):S(o,null)),m=t.props?c:dl(c)}}catch(S){jt.length=0,Yt(S,e,1),H=se(ve)}let N=H;if(m&&T!==!1){const S=Object.keys(m),{shapeFlag:B}=N;S.length&&B&7&&(i&&S.some(mr)&&(m=hl(m,i)),N=Qe(N,m))}return n.dirs&&(N=Qe(N),N.dirs=N.dirs?N.dirs.concat(n.dirs):n.dirs),n.transition&&(N.transition=n.transition),H=N,vn(_),H}const dl=e=>{let t;for(const n in e)(n==="class"||n==="style"||zt(n))&&((t||(t={}))[n]=e[n]);return t},hl=(e,t)=>{const n={};for(const r in e)(!mr(r)||!(r.slice(9)in t))&&(n[r]=e[r]);return n};function pl(e,t,n){const{props:r,children:s,component:o}=e,{props:i,children:l,patchFlag:c}=t,a=o.emitsOptions;if(t.dirs||t.transition)return!0;if(n&&c>=0){if(c&1024)return!0;if(c&16)return r?rs(r,i,a):!!i;if(c&8){const f=t.dynamicProps;for(let h=0;he.__isSuspense;function mo(e,t){t&&t.pendingBranch?k(e)?t.effects.push(...e):t.effects.push(e):cl(e)}function Ir(e,t){return Nn(e,null,t)}function Ha(e,t){return Nn(e,null,{flush:"post"})}const rn={};function Je(e,t,n){return Nn(e,t,n)}function Nn(e,t,{immediate:n,deep:r,flush:s,onTrack:o,onTrigger:i}=te){var l;const c=Qs()===((l=le)==null?void 0:l.scope)?le:null;let a,f=!1,h=!1;if(ce(e)?(a=()=>e.value,f=_n(e)):_t(e)?(a=()=>e,r=!0):k(e)?(h=!0,f=e.some(S=>_t(S)||_n(S)),a=()=>e.map(S=>{if(ce(S))return S.value;if(_t(S))return it(S);if(K(S))return Ye(S,c,2)})):K(e)?t?a=()=>Ye(e,c,2):a=()=>{if(!(c&&c.isUnmounted))return g&&g(),xe(e,c,3,[w])}:a=Oe,t&&r){const S=a;a=()=>it(S())}let g,w=S=>{g=_.onStop=()=>{Ye(S,c,4)}},E;if(Tt)if(w=Oe,t?n&&xe(t,c,3,[a(),h?[]:void 0,w]):a(),s==="sync"){const S=dc();E=S.__watcherHandles||(S.__watcherHandles=[])}else return Oe;let T=h?new Array(e.length).fill(rn):rn;const H=()=>{if(_.active)if(t){const S=_.run();(r||f||(h?S.some((B,W)=>ut(B,T[W])):ut(S,T)))&&(g&&g(),xe(t,c,3,[S,T===rn?void 0:h&&T[0]===rn?[]:T,w]),T=S)}else _.run()};H.allowRecurse=!!t;let m;s==="sync"?m=H:s==="post"?m=()=>pe(H,c&&c.suspense):(H.pre=!0,c&&(H.id=c.uid),m=()=>Fn(H));const _=new Er(a,m);t?n?H():T=_.run():s==="post"?pe(_.run.bind(_),c&&c.suspense):_.run();const N=()=>{_.stop(),c&&c.scope&&yr(c.scope.effects,_)};return E&&E.push(N),N}function yl(e,t,n){const r=this.proxy,s=re(e)?e.includes(".")?yo(r,e):()=>r[e]:e.bind(r,r);let o;K(t)?o=t:(o=t.handler,n=t);const i=le;xt(this);const l=Nn(s,o.bind(r),n);return i?xt(i):ct(),l}function yo(e,t){const n=t.split(".");return()=>{let r=e;for(let s=0;s{it(n,t)});else if(Ys(e))for(const n in e)it(e[n],t);return e}function $a(e,t){const n=fe;if(n===null)return e;const r=Bn(n)||n.proxy,s=e.dirs||(e.dirs=[]);for(let o=0;o{e.isMounted=!0}),Co(()=>{e.isUnmounting=!0}),e}const we=[Function,Array],_o={mode:String,appear:Boolean,persisted:Boolean,onBeforeEnter:we,onEnter:we,onAfterEnter:we,onEnterCancelled:we,onBeforeLeave:we,onLeave:we,onAfterLeave:we,onLeaveCancelled:we,onBeforeAppear:we,onAppear:we,onAfterAppear:we,onAppearCancelled:we},bl={name:"BaseTransition",props:_o,setup(e,{slots:t}){const n=Dn(),r=_l();let s;return()=>{const o=t.default&&vo(t.default(),!0);if(!o||!o.length)return;let i=o[0];if(o.length>1){for(const T of o)if(T.type!==ve){i=T;break}}const l=J(e),{mode:c}=l;if(r.isLeaving)return Wn(i);const a=ss(i);if(!a)return Wn(i);const f=or(a,l,r,n);ir(a,f);const h=n.subTree,g=h&&ss(h);let w=!1;const{getTransitionKey:E}=a.type;if(E){const T=E();s===void 0?s=T:T!==s&&(s=T,w=!0)}if(g&&g.type!==ve&&(!st(a,g)||w)){const T=or(g,l,r,n);if(ir(g,T),c==="out-in")return r.isLeaving=!0,T.afterLeave=()=>{r.isLeaving=!1,n.update.active!==!1&&n.update()},Wn(i);c==="in-out"&&a.type!==ve&&(T.delayLeave=(H,m,_)=>{const N=bo(r,g);N[String(g.key)]=g,H[Ve]=()=>{m(),H[Ve]=void 0,delete f.delayedLeave},f.delayedLeave=_})}return i}}},vl=bl;function bo(e,t){const{leavingVNodes:n}=e;let r=n.get(t.type);return r||(r=Object.create(null),n.set(t.type,r)),r}function or(e,t,n,r){const{appear:s,mode:o,persisted:i=!1,onBeforeEnter:l,onEnter:c,onAfterEnter:a,onEnterCancelled:f,onBeforeLeave:h,onLeave:g,onAfterLeave:w,onLeaveCancelled:E,onBeforeAppear:T,onAppear:H,onAfterAppear:m,onAppearCancelled:_}=t,N=String(e.key),S=bo(n,e),B=(y,P)=>{y&&xe(y,r,9,P)},W=(y,P)=>{const I=P[1];B(y,P),k(y)?y.every(V=>V.length<=1)&&I():y.length<=1&&I()},D={mode:o,persisted:i,beforeEnter(y){let P=l;if(!n.isMounted)if(s)P=T||l;else return;y[Ve]&&y[Ve](!0);const I=S[N];I&&st(e,I)&&I.el[Ve]&&I.el[Ve](),B(P,[y])},enter(y){let P=c,I=a,V=f;if(!n.isMounted)if(s)P=H||c,I=m||a,V=_||f;else return;let M=!1;const q=y[sn]=L=>{M||(M=!0,L?B(V,[y]):B(I,[y]),D.delayedLeave&&D.delayedLeave(),y[sn]=void 0)};P?W(P,[y,q]):q()},leave(y,P){const I=String(e.key);if(y[sn]&&y[sn](!0),n.isUnmounting)return P();B(h,[y]);let V=!1;const M=y[Ve]=q=>{V||(V=!0,P(),q?B(E,[y]):B(w,[y]),y[Ve]=void 0,S[I]===e&&delete S[I])};S[I]=e,g?W(g,[y,M]):M()},clone(y){return or(y,t,n,r)}};return D}function Wn(e){if(Jt(e))return e=Qe(e),e.children=null,e}function ss(e){return Jt(e)?e.children?e.children[0]:void 0:e}function ir(e,t){e.shapeFlag&6&&e.component?ir(e.component.subTree,t):e.shapeFlag&128?(e.ssContent.transition=t.clone(e.ssContent),e.ssFallback.transition=t.clone(e.ssFallback)):e.transition=t}function vo(e,t=!1,n){let r=[],s=0;for(let o=0;o1)for(let o=0;oue({name:e.name},t,{setup:e}))():e}const vt=e=>!!e.type.__asyncLoader;/*! #__NO_SIDE_EFFECTS__ */function ja(e){K(e)&&(e={loader:e});const{loader:t,loadingComponent:n,errorComponent:r,delay:s=200,timeout:o,suspensible:i=!0,onError:l}=e;let c=null,a,f=0;const h=()=>(f++,c=null,g()),g=()=>{let w;return c||(w=c=t().catch(E=>{if(E=E instanceof Error?E:new Error(String(E)),l)return new Promise((T,H)=>{l(E,()=>T(h()),()=>H(E),f+1)});throw E}).then(E=>w!==c&&c?c:(E&&(E.__esModule||E[Symbol.toStringTag]==="Module")&&(E=E.default),a=E,E)))};return Lr({name:"AsyncComponentWrapper",__asyncLoader:g,get __asyncResolved(){return a},setup(){const w=le;if(a)return()=>Vn(a,w);const E=_=>{c=null,Yt(_,w,13,!r)};if(i&&w.suspense||Tt)return g().then(_=>()=>Vn(_,w)).catch(_=>(E(_),()=>r?se(r,{error:_}):null));const T=ae(!1),H=ae(),m=ae(!!s);return s&&setTimeout(()=>{m.value=!1},s),o!=null&&setTimeout(()=>{if(!T.value&&!H.value){const _=new Error(`Async component timed out after ${o}ms.`);E(_),H.value=_}},o),g().then(()=>{T.value=!0,w.parent&&Jt(w.parent.vnode)&&Fn(w.parent.update)}).catch(_=>{E(_),H.value=_}),()=>{if(T.value&&a)return Vn(a,w);if(H.value&&r)return se(r,{error:H.value});if(n&&!m.value)return se(n)}}})}function Vn(e,t){const{ref:n,props:r,children:s,ce:o}=t.vnode,i=se(e,r,s);return i.ref=n,i.ce=o,delete t.vnode.ce,i}const Jt=e=>e.type.__isKeepAlive;function wl(e,t){wo(e,"a",t)}function Cl(e,t){wo(e,"da",t)}function wo(e,t,n=le){const r=e.__wdc||(e.__wdc=()=>{let s=n;for(;s;){if(s.isDeactivated)return;s=s.parent}return e()});if(Hn(t,r,n),n){let s=n.parent;for(;s&&s.parent;)Jt(s.parent.vnode)&&El(r,t,n,s),s=s.parent}}function El(e,t,n,r){const s=Hn(t,e,r,!0);$n(()=>{yr(r[t],s)},n)}function Hn(e,t,n=le,r=!1){if(n){const s=n[e]||(n[e]=[]),o=t.__weh||(t.__weh=(...i)=>{if(n.isUnmounted)return;At(),xt(n);const l=xe(t,n,e,i);return ct(),St(),l});return r?s.unshift(o):s.push(o),o}}const Be=e=>(t,n=le)=>(!Tt||e==="sp")&&Hn(e,(...r)=>t(...r),n),xl=Be("bm"),Rt=Be("m"),Tl=Be("bu"),Al=Be("u"),Co=Be("bum"),$n=Be("um"),Sl=Be("sp"),Rl=Be("rtg"),Ol=Be("rtc");function Pl(e,t=le){Hn("ec",e,t)}const Nr="components";function Da(e,t){return xo(Nr,e,!0,t)||e}const Eo=Symbol.for("v-ndc");function Ba(e){return re(e)?xo(Nr,e,!1)||e:e||Eo}function xo(e,t,n=!0,r=!1){const s=fe||le;if(s){const o=s.type;if(e===Nr){const l=ac(o,!1);if(l&&(l===t||l===Ne(t)||l===Sn(Ne(t))))return o}const i=os(s[e]||o[e],t)||os(s.appContext[e],t);return!i&&r?o:i}}function os(e,t){return e&&(e[t]||e[Ne(t)]||e[Sn(Ne(t))])}function ka(e,t,n,r){let s;const o=n&&n[r];if(k(e)||re(e)){s=new Array(e.length);for(let i=0,l=e.length;it(i,l,void 0,o&&o[l]));else{const i=Object.keys(e);s=new Array(i.length);for(let l=0,c=i.length;lxn(t)?!(t.type===ve||t.type===ge&&!To(t.children)):!0)?e:null}function Ka(e,t){const n={};for(const r in e)n[t&&/[A-Z]/.test(r)?`on:${r}`:dn(r)]=e[r];return n}const lr=e=>e?ko(e)?Bn(e)||e.proxy:lr(e.parent):null,Nt=ue(Object.create(null),{$:e=>e,$el:e=>e.vnode.el,$data:e=>e.data,$props:e=>e.props,$attrs:e=>e.attrs,$slots:e=>e.slots,$refs:e=>e.refs,$parent:e=>lr(e.parent),$root:e=>lr(e.root),$emit:e=>e.emit,$options:e=>Hr(e),$forceUpdate:e=>e.f||(e.f=()=>Fn(e.update)),$nextTick:e=>e.n||(e.n=Mn.bind(e.proxy)),$watch:e=>yl.bind(e)}),qn=(e,t)=>e!==te&&!e.__isScriptSetup&&Y(e,t),Ml={get({_:e},t){const{ctx:n,setupState:r,data:s,props:o,accessCache:i,type:l,appContext:c}=e;let a;if(t[0]!=="$"){const w=i[t];if(w!==void 0)switch(w){case 1:return r[t];case 2:return s[t];case 4:return n[t];case 3:return o[t]}else{if(qn(r,t))return i[t]=1,r[t];if(s!==te&&Y(s,t))return i[t]=2,s[t];if((a=e.propsOptions[0])&&Y(a,t))return i[t]=3,o[t];if(n!==te&&Y(n,t))return i[t]=4,n[t];cr&&(i[t]=0)}}const f=Nt[t];let h,g;if(f)return t==="$attrs"&&me(e,"get",t),f(e);if((h=l.__cssModules)&&(h=h[t]))return h;if(n!==te&&Y(n,t))return i[t]=4,n[t];if(g=c.config.globalProperties,Y(g,t))return g[t]},set({_:e},t,n){const{data:r,setupState:s,ctx:o}=e;return qn(s,t)?(s[t]=n,!0):r!==te&&Y(r,t)?(r[t]=n,!0):Y(e.props,t)||t[0]==="$"&&t.slice(1)in e?!1:(o[t]=n,!0)},has({_:{data:e,setupState:t,accessCache:n,ctx:r,appContext:s,propsOptions:o}},i){let l;return!!n[i]||e!==te&&Y(e,i)||qn(t,i)||(l=o[0])&&Y(l,i)||Y(r,i)||Y(Nt,i)||Y(s.config.globalProperties,i)},defineProperty(e,t,n){return n.get!=null?e._.accessCache[t]=0:Y(n,"value")&&this.set(e,t,n.value,null),Reflect.defineProperty(e,t,n)}};function Wa(){return Fl().slots}function Fl(){const e=Dn();return e.setupContext||(e.setupContext=Ko(e))}function is(e){return k(e)?e.reduce((t,n)=>(t[n]=null,t),{}):e}let cr=!0;function Il(e){const t=Hr(e),n=e.proxy,r=e.ctx;cr=!1,t.beforeCreate&&ls(t.beforeCreate,e,"bc");const{data:s,computed:o,methods:i,watch:l,provide:c,inject:a,created:f,beforeMount:h,mounted:g,beforeUpdate:w,updated:E,activated:T,deactivated:H,beforeDestroy:m,beforeUnmount:_,destroyed:N,unmounted:S,render:B,renderTracked:W,renderTriggered:D,errorCaptured:y,serverPrefetch:P,expose:I,inheritAttrs:V,components:M,directives:q,filters:L}=t;if(a&&Ll(a,r,null),i)for(const ne in i){const Z=i[ne];K(Z)&&(r[ne]=Z.bind(n))}if(s){const ne=s.call(n,n);ee(ne)&&(e.data=On(ne))}if(cr=!0,o)for(const ne in o){const Z=o[ne],Ze=K(Z)?Z.bind(n,n):K(Z.get)?Z.get.bind(n,n):Oe,Xt=!K(Z)&&K(Z.set)?Z.set.bind(n):Oe,Ge=ie({get:Ze,set:Xt});Object.defineProperty(r,ne,{enumerable:!0,configurable:!0,get:()=>Ge.value,set:Me=>Ge.value=Me})}if(l)for(const ne in l)Ao(l[ne],r,n,ne);if(c){const ne=K(c)?c.call(n):c;Reflect.ownKeys(ne).forEach(Z=>{Bl(Z,ne[Z])})}f&&ls(f,e,"c");function X(ne,Z){k(Z)?Z.forEach(Ze=>ne(Ze.bind(n))):Z&&ne(Z.bind(n))}if(X(xl,h),X(Rt,g),X(Tl,w),X(Al,E),X(wl,T),X(Cl,H),X(Pl,y),X(Ol,W),X(Rl,D),X(Co,_),X($n,S),X(Sl,P),k(I))if(I.length){const ne=e.exposed||(e.exposed={});I.forEach(Z=>{Object.defineProperty(ne,Z,{get:()=>n[Z],set:Ze=>n[Z]=Ze})})}else e.exposed||(e.exposed={});B&&e.render===Oe&&(e.render=B),V!=null&&(e.inheritAttrs=V),M&&(e.components=M),q&&(e.directives=q)}function Ll(e,t,n=Oe){k(e)&&(e=ar(e));for(const r in e){const s=e[r];let o;ee(s)?"default"in s?o=wt(s.from||r,s.default,!0):o=wt(s.from||r):o=wt(s),ce(o)?Object.defineProperty(t,r,{enumerable:!0,configurable:!0,get:()=>o.value,set:i=>o.value=i}):t[r]=o}}function ls(e,t,n){xe(k(e)?e.map(r=>r.bind(t.proxy)):e.bind(t.proxy),t,n)}function Ao(e,t,n,r){const s=r.includes(".")?yo(n,r):()=>n[r];if(re(e)){const o=t[e];K(o)&&Je(s,o)}else if(K(e))Je(s,e.bind(n));else if(ee(e))if(k(e))e.forEach(o=>Ao(o,t,n,r));else{const o=K(e.handler)?e.handler.bind(n):t[e.handler];K(o)&&Je(s,o,e)}}function Hr(e){const t=e.type,{mixins:n,extends:r}=t,{mixins:s,optionsCache:o,config:{optionMergeStrategies:i}}=e.appContext,l=o.get(t);let c;return l?c=l:!s.length&&!n&&!r?c=t:(c={},s.length&&s.forEach(a=>wn(c,a,i,!0)),wn(c,t,i)),ee(t)&&o.set(t,c),c}function wn(e,t,n,r=!1){const{mixins:s,extends:o}=t;o&&wn(e,o,n,!0),s&&s.forEach(i=>wn(e,i,n,!0));for(const i in t)if(!(r&&i==="expose")){const l=Nl[i]||n&&n[i];e[i]=l?l(e[i],t[i]):t[i]}return e}const Nl={data:cs,props:as,emits:as,methods:Ft,computed:Ft,beforeCreate:he,created:he,beforeMount:he,mounted:he,beforeUpdate:he,updated:he,beforeDestroy:he,beforeUnmount:he,destroyed:he,unmounted:he,activated:he,deactivated:he,errorCaptured:he,serverPrefetch:he,components:Ft,directives:Ft,watch:$l,provide:cs,inject:Hl};function cs(e,t){return t?e?function(){return ue(K(e)?e.call(this,this):e,K(t)?t.call(this,this):t)}:t:e}function Hl(e,t){return Ft(ar(e),ar(t))}function ar(e){if(k(e)){const t={};for(let n=0;n1)return n&&K(t)?t.call(r&&r.proxy):t}}function kl(e,t,n,r=!1){const s={},o={};mn(o,jn,1),e.propsDefaults=Object.create(null),Ro(e,t,s,o);for(const i in e.propsOptions[0])i in s||(s[i]=void 0);n?e.props=r?s:Yi(s):e.type.props?e.props=s:e.props=o,e.attrs=o}function Ul(e,t,n,r){const{props:s,attrs:o,vnode:{patchFlag:i}}=e,l=J(s),[c]=e.propsOptions;let a=!1;if((r||i>0)&&!(i&16)){if(i&8){const f=e.vnode.dynamicProps;for(let h=0;h{c=!0;const[g,w]=Oo(h,t,!0);ue(i,g),w&&l.push(...w)};!n&&t.mixins.length&&t.mixins.forEach(f),e.extends&&f(e.extends),e.mixins&&e.mixins.forEach(f)}if(!o&&!c)return ee(e)&&r.set(e,mt),mt;if(k(o))for(let f=0;f-1,w[1]=T<0||E-1||Y(w,"default"))&&l.push(h)}}}const a=[i,l];return ee(e)&&r.set(e,a),a}function us(e){return e[0]!=="$"}function fs(e){const t=e&&e.toString().match(/^\s*(function|class) (\w+)/);return t?t[2]:e===null?"null":""}function ds(e,t){return fs(e)===fs(t)}function hs(e,t){return k(t)?t.findIndex(n=>ds(n,e)):K(t)&&ds(t,e)?0:-1}const Po=e=>e[0]==="_"||e==="$stable",$r=e=>k(e)?e.map(Ae):[Ae(e)],Kl=(e,t,n)=>{if(t._n)return t;const r=fl((...s)=>$r(t(...s)),n);return r._c=!1,r},Mo=(e,t,n)=>{const r=e._ctx;for(const s in e){if(Po(s))continue;const o=e[s];if(K(o))t[s]=Kl(s,o,r);else if(o!=null){const i=$r(o);t[s]=()=>i}}},Fo=(e,t)=>{const n=$r(t);e.slots.default=()=>n},Wl=(e,t)=>{if(e.vnode.shapeFlag&32){const n=t._;n?(e.slots=J(t),mn(t,"_",n)):Mo(t,e.slots={})}else e.slots={},t&&Fo(e,t);mn(e.slots,jn,1)},Vl=(e,t,n)=>{const{vnode:r,slots:s}=e;let o=!0,i=te;if(r.shapeFlag&32){const l=t._;l?n&&l===1?o=!1:(ue(s,t),!n&&l===1&&delete s._):(o=!t.$stable,Mo(t,s)),i=t}else t&&(Fo(e,t),i={default:1});if(o)for(const l in s)!Po(l)&&i[l]==null&&delete s[l]};function En(e,t,n,r,s=!1){if(k(e)){e.forEach((g,w)=>En(g,t&&(k(t)?t[w]:t),n,r,s));return}if(vt(r)&&!s)return;const o=r.shapeFlag&4?Bn(r.component)||r.component.proxy:r.el,i=s?null:o,{i:l,r:c}=e,a=t&&t.r,f=l.refs===te?l.refs={}:l.refs,h=l.setupState;if(a!=null&&a!==c&&(re(a)?(f[a]=null,Y(h,a)&&(h[a]=null)):ce(a)&&(a.value=null)),K(c))Ye(c,l,12,[i,f]);else{const g=re(c),w=ce(c);if(g||w){const E=()=>{if(e.f){const T=g?Y(h,c)?h[c]:f[c]:c.value;s?k(T)&&yr(T,o):k(T)?T.includes(o)||T.push(o):g?(f[c]=[o],Y(h,c)&&(h[c]=f[c])):(c.value=[o],e.k&&(f[e.k]=c.value))}else g?(f[c]=i,Y(h,c)&&(h[c]=i)):w&&(c.value=i,e.k&&(f[e.k]=i))};i?(E.id=-1,pe(E,n)):E()}}}let Ue=!1;const on=e=>/svg/.test(e.namespaceURI)&&e.tagName!=="foreignObject",ln=e=>e.nodeType===8;function ql(e){const{mt:t,p:n,o:{patchProp:r,createText:s,nextSibling:o,parentNode:i,remove:l,insert:c,createComment:a}}=e,f=(m,_)=>{if(!_.hasChildNodes()){n(null,m,_),bn(),_._vnode=m;return}Ue=!1,h(_.firstChild,m,null,null,null),bn(),_._vnode=m,Ue&&console.error("Hydration completed but contains mismatches.")},h=(m,_,N,S,B,W=!1)=>{const D=ln(m)&&m.data==="[",y=()=>T(m,_,N,S,B,D),{type:P,ref:I,shapeFlag:V,patchFlag:M}=_;let q=m.nodeType;_.el=m,M===-2&&(W=!1,_.dynamicChildren=null);let L=null;switch(P){case Et:q!==3?_.children===""?(c(_.el=s(""),i(m),m),L=m):L=y():(m.data!==_.children&&(Ue=!0,m.data=_.children),L=o(m));break;case ve:q!==8||D?L=y():L=o(m);break;case $t:if(D&&(m=o(m),q=m.nodeType),q===1||q===3){L=m;const ye=!_.children.length;for(let X=0;X<_.staticCount;X++)ye&&(_.children+=L.nodeType===1?L.outerHTML:L.data),X===_.staticCount-1&&(_.anchor=L),L=o(L);return D?o(L):L}else y();break;case ge:D?L=E(m,_,N,S,B,W):L=y();break;default:if(V&1)q!==1||_.type.toLowerCase()!==m.tagName.toLowerCase()?L=y():L=g(m,_,N,S,B,W);else if(V&6){_.slotScopeIds=B;const ye=i(m);if(t(_,ye,null,N,S,on(ye),W),L=D?H(m):o(m),L&&ln(L)&&L.data==="teleport end"&&(L=o(L)),vt(_)){let X;D?(X=se(ge),X.anchor=L?L.previousSibling:ye.lastChild):X=m.nodeType===3?Bo(""):se("div"),X.el=m,_.component.subTree=X}}else V&64?q!==8?L=y():L=_.type.hydrate(m,_,N,S,B,W,e,w):V&128&&(L=_.type.hydrate(m,_,N,S,on(i(m)),B,W,e,h))}return I!=null&&En(I,null,S,_),L},g=(m,_,N,S,B,W)=>{W=W||!!_.dynamicChildren;const{type:D,props:y,patchFlag:P,shapeFlag:I,dirs:V}=_,M=D==="input"&&V||D==="option";if(M||P!==-1){if(V&&Ie(_,null,N,"created"),y)if(M||!W||P&48)for(const L in y)(M&&L.endsWith("value")||zt(L)&&!It(L))&&r(m,L,null,y[L],!1,void 0,N);else y.onClick&&r(m,"onClick",null,y.onClick,!1,void 0,N);let q;if((q=y&&y.onVnodeBeforeMount)&&Ce(q,N,_),V&&Ie(_,null,N,"beforeMount"),((q=y&&y.onVnodeMounted)||V)&&mo(()=>{q&&Ce(q,N,_),V&&Ie(_,null,N,"mounted")},S),I&16&&!(y&&(y.innerHTML||y.textContent))){let L=w(m.firstChild,_,m,N,S,B,W);for(;L;){Ue=!0;const ye=L;L=L.nextSibling,l(ye)}}else I&8&&m.textContent!==_.children&&(Ue=!0,m.textContent=_.children)}return m.nextSibling},w=(m,_,N,S,B,W,D)=>{D=D||!!_.dynamicChildren;const y=_.children,P=y.length;for(let I=0;I{const{slotScopeIds:D}=_;D&&(B=B?B.concat(D):D);const y=i(m),P=w(o(m),_,y,N,S,B,W);return P&&ln(P)&&P.data==="]"?o(_.anchor=P):(Ue=!0,c(_.anchor=a("]"),y,P),P)},T=(m,_,N,S,B,W)=>{if(Ue=!0,_.el=null,W){const P=H(m);for(;;){const I=o(m);if(I&&I!==P)l(I);else break}}const D=o(m),y=i(m);return l(m),n(null,_,y,D,N,S,on(y),B),D},H=m=>{let _=0;for(;m;)if(m=o(m),m&&ln(m)&&(m.data==="["&&_++,m.data==="]")){if(_===0)return o(m);_--}return m};return[f,h]}const pe=mo;function zl(e){return Io(e)}function Yl(e){return Io(e,ql)}function Io(e,t){const n=er();n.__VUE__=!0;const{insert:r,remove:s,patchProp:o,createElement:i,createText:l,createComment:c,setText:a,setElementText:f,parentNode:h,nextSibling:g,setScopeId:w=Oe,insertStaticContent:E}=e,T=(u,d,p,v=null,b=null,A=null,O=!1,x=null,R=!!d.dynamicChildren)=>{if(u===d)return;u&&!st(u,d)&&(v=Qt(u),Me(u,b,A,!0),u=null),d.patchFlag===-2&&(R=!1,d.dynamicChildren=null);const{type:C,ref:$,shapeFlag:F}=d;switch(C){case Et:H(u,d,p,v);break;case ve:m(u,d,p,v);break;case $t:u==null&&_(d,p,v,O);break;case ge:M(u,d,p,v,b,A,O,x,R);break;default:F&1?B(u,d,p,v,b,A,O,x,R):F&6?q(u,d,p,v,b,A,O,x,R):(F&64||F&128)&&C.process(u,d,p,v,b,A,O,x,R,dt)}$!=null&&b&&En($,u&&u.ref,A,d||u,!d)},H=(u,d,p,v)=>{if(u==null)r(d.el=l(d.children),p,v);else{const b=d.el=u.el;d.children!==u.children&&a(b,d.children)}},m=(u,d,p,v)=>{u==null?r(d.el=c(d.children||""),p,v):d.el=u.el},_=(u,d,p,v)=>{[u.el,u.anchor]=E(u.children,d,p,v,u.el,u.anchor)},N=({el:u,anchor:d},p,v)=>{let b;for(;u&&u!==d;)b=g(u),r(u,p,v),u=b;r(d,p,v)},S=({el:u,anchor:d})=>{let p;for(;u&&u!==d;)p=g(u),s(u),u=p;s(d)},B=(u,d,p,v,b,A,O,x,R)=>{O=O||d.type==="svg",u==null?W(d,p,v,b,A,O,x,R):P(u,d,b,A,O,x,R)},W=(u,d,p,v,b,A,O,x)=>{let R,C;const{type:$,props:F,shapeFlag:j,transition:U,dirs:z}=u;if(R=u.el=i(u.type,A,F&&F.is,F),j&8?f(R,u.children):j&16&&y(u.children,R,null,v,b,A&&$!=="foreignObject",O,x),z&&Ie(u,null,v,"created"),D(R,u,u.scopeId,O,v),F){for(const Q in F)Q!=="value"&&!It(Q)&&o(R,Q,null,F[Q],A,u.children,v,b,He);"value"in F&&o(R,"value",null,F.value),(C=F.onVnodeBeforeMount)&&Ce(C,v,u)}z&&Ie(u,null,v,"beforeMount");const G=(!b||b&&!b.pendingBranch)&&U&&!U.persisted;G&&U.beforeEnter(R),r(R,d,p),((C=F&&F.onVnodeMounted)||G||z)&&pe(()=>{C&&Ce(C,v,u),G&&U.enter(R),z&&Ie(u,null,v,"mounted")},b)},D=(u,d,p,v,b)=>{if(p&&w(u,p),v)for(let A=0;A{for(let C=R;C{const x=d.el=u.el;let{patchFlag:R,dynamicChildren:C,dirs:$}=d;R|=u.patchFlag&16;const F=u.props||te,j=d.props||te;let U;p&&et(p,!1),(U=j.onVnodeBeforeUpdate)&&Ce(U,p,d,u),$&&Ie(d,u,p,"beforeUpdate"),p&&et(p,!0);const z=b&&d.type!=="foreignObject";if(C?I(u.dynamicChildren,C,x,p,v,z,A):O||Z(u,d,x,null,p,v,z,A,!1),R>0){if(R&16)V(x,d,F,j,p,v,b);else if(R&2&&F.class!==j.class&&o(x,"class",null,j.class,b),R&4&&o(x,"style",F.style,j.style,b),R&8){const G=d.dynamicProps;for(let Q=0;Q{U&&Ce(U,p,d,u),$&&Ie(d,u,p,"updated")},v)},I=(u,d,p,v,b,A,O)=>{for(let x=0;x{if(p!==v){if(p!==te)for(const x in p)!It(x)&&!(x in v)&&o(u,x,p[x],null,O,d.children,b,A,He);for(const x in v){if(It(x))continue;const R=v[x],C=p[x];R!==C&&x!=="value"&&o(u,x,C,R,O,d.children,b,A,He)}"value"in v&&o(u,"value",p.value,v.value)}},M=(u,d,p,v,b,A,O,x,R)=>{const C=d.el=u?u.el:l(""),$=d.anchor=u?u.anchor:l("");let{patchFlag:F,dynamicChildren:j,slotScopeIds:U}=d;U&&(x=x?x.concat(U):U),u==null?(r(C,p,v),r($,p,v),y(d.children,p,$,b,A,O,x,R)):F>0&&F&64&&j&&u.dynamicChildren?(I(u.dynamicChildren,j,p,b,A,O,x),(d.key!=null||b&&d===b.subTree)&&jr(u,d,!0)):Z(u,d,p,$,b,A,O,x,R)},q=(u,d,p,v,b,A,O,x,R)=>{d.slotScopeIds=x,u==null?d.shapeFlag&512?b.ctx.activate(d,p,v,O,R):L(d,p,v,b,A,O,R):ye(u,d,R)},L=(u,d,p,v,b,A,O)=>{const x=u.component=oc(u,v,b);if(Jt(u)&&(x.ctx.renderer=dt),ic(x),x.asyncDep){if(b&&b.registerDep(x,X),!u.el){const R=x.subTree=se(ve);m(null,R,d,p)}return}X(x,u,d,p,b,A,O)},ye=(u,d,p)=>{const v=d.component=u.component;if(pl(u,d,p))if(v.asyncDep&&!v.asyncResolved){ne(v,d,p);return}else v.next=d,ll(v.update),v.update();else d.el=u.el,v.vnode=d},X=(u,d,p,v,b,A,O)=>{const x=()=>{if(u.isMounted){let{next:$,bu:F,u:j,parent:U,vnode:z}=u,G=$,Q;et(u,!1),$?($.el=z.el,ne(u,$,O)):$=z,F&&hn(F),(Q=$.props&&$.props.onVnodeBeforeUpdate)&&Ce(Q,U,$,z),et(u,!0);const oe=Kn(u),Te=u.subTree;u.subTree=oe,T(Te,oe,h(Te.el),Qt(Te),u,b,A),$.el=oe.el,G===null&&gl(u,oe.el),j&&pe(j,b),(Q=$.props&&$.props.onVnodeUpdated)&&pe(()=>Ce(Q,U,$,z),b)}else{let $;const{el:F,props:j}=d,{bm:U,m:z,parent:G}=u,Q=vt(d);if(et(u,!1),U&&hn(U),!Q&&($=j&&j.onVnodeBeforeMount)&&Ce($,G,d),et(u,!0),F&&Un){const oe=()=>{u.subTree=Kn(u),Un(F,u.subTree,u,b,null)};Q?d.type.__asyncLoader().then(()=>!u.isUnmounted&&oe()):oe()}else{const oe=u.subTree=Kn(u);T(null,oe,p,v,u,b,A),d.el=oe.el}if(z&&pe(z,b),!Q&&($=j&&j.onVnodeMounted)){const oe=d;pe(()=>Ce($,G,oe),b)}(d.shapeFlag&256||G&&vt(G.vnode)&&G.vnode.shapeFlag&256)&&u.a&&pe(u.a,b),u.isMounted=!0,d=p=v=null}},R=u.effect=new Er(x,()=>Fn(C),u.scope),C=u.update=()=>R.run();C.id=u.uid,et(u,!0),C()},ne=(u,d,p)=>{d.component=u;const v=u.vnode.props;u.vnode=d,u.next=null,Ul(u,d.props,v,p),Vl(u,d.children,p),At(),ns(),St()},Z=(u,d,p,v,b,A,O,x,R=!1)=>{const C=u&&u.children,$=u?u.shapeFlag:0,F=d.children,{patchFlag:j,shapeFlag:U}=d;if(j>0){if(j&128){Xt(C,F,p,v,b,A,O,x,R);return}else if(j&256){Ze(C,F,p,v,b,A,O,x,R);return}}U&8?($&16&&He(C,b,A),F!==C&&f(p,F)):$&16?U&16?Xt(C,F,p,v,b,A,O,x,R):He(C,b,A,!0):($&8&&f(p,""),U&16&&y(F,p,v,b,A,O,x,R))},Ze=(u,d,p,v,b,A,O,x,R)=>{u=u||mt,d=d||mt;const C=u.length,$=d.length,F=Math.min(C,$);let j;for(j=0;j$?He(u,b,A,!0,!1,F):y(d,p,v,b,A,O,x,R,F)},Xt=(u,d,p,v,b,A,O,x,R)=>{let C=0;const $=d.length;let F=u.length-1,j=$-1;for(;C<=F&&C<=j;){const U=u[C],z=d[C]=R?qe(d[C]):Ae(d[C]);if(st(U,z))T(U,z,p,null,b,A,O,x,R);else break;C++}for(;C<=F&&C<=j;){const U=u[F],z=d[j]=R?qe(d[j]):Ae(d[j]);if(st(U,z))T(U,z,p,null,b,A,O,x,R);else break;F--,j--}if(C>F){if(C<=j){const U=j+1,z=U<$?d[U].el:v;for(;C<=j;)T(null,d[C]=R?qe(d[C]):Ae(d[C]),p,z,b,A,O,x,R),C++}}else if(C>j)for(;C<=F;)Me(u[C],b,A,!0),C++;else{const U=C,z=C,G=new Map;for(C=z;C<=j;C++){const _e=d[C]=R?qe(d[C]):Ae(d[C]);_e.key!=null&&G.set(_e.key,C)}let Q,oe=0;const Te=j-z+1;let ht=!1,Vr=0;const Ot=new Array(Te);for(C=0;C=Te){Me(_e,b,A,!0);continue}let Fe;if(_e.key!=null)Fe=G.get(_e.key);else for(Q=z;Q<=j;Q++)if(Ot[Q-z]===0&&st(_e,d[Q])){Fe=Q;break}Fe===void 0?Me(_e,b,A,!0):(Ot[Fe-z]=C+1,Fe>=Vr?Vr=Fe:ht=!0,T(_e,d[Fe],p,null,b,A,O,x,R),oe++)}const qr=ht?Jl(Ot):mt;for(Q=qr.length-1,C=Te-1;C>=0;C--){const _e=z+C,Fe=d[_e],zr=_e+1<$?d[_e+1].el:v;Ot[C]===0?T(null,Fe,p,zr,b,A,O,x,R):ht&&(Q<0||C!==qr[Q]?Ge(Fe,p,zr,2):Q--)}}},Ge=(u,d,p,v,b=null)=>{const{el:A,type:O,transition:x,children:R,shapeFlag:C}=u;if(C&6){Ge(u.component.subTree,d,p,v);return}if(C&128){u.suspense.move(d,p,v);return}if(C&64){O.move(u,d,p,dt);return}if(O===ge){r(A,d,p);for(let F=0;Fx.enter(A),b);else{const{leave:F,delayLeave:j,afterLeave:U}=x,z=()=>r(A,d,p),G=()=>{F(A,()=>{z(),U&&U()})};j?j(A,z,G):G()}else r(A,d,p)},Me=(u,d,p,v=!1,b=!1)=>{const{type:A,props:O,ref:x,children:R,dynamicChildren:C,shapeFlag:$,patchFlag:F,dirs:j}=u;if(x!=null&&En(x,null,p,u,!0),$&256){d.ctx.deactivate(u);return}const U=$&1&&j,z=!vt(u);let G;if(z&&(G=O&&O.onVnodeBeforeUnmount)&&Ce(G,d,u),$&6)ui(u.component,p,v);else{if($&128){u.suspense.unmount(p,v);return}U&&Ie(u,null,d,"beforeUnmount"),$&64?u.type.remove(u,d,p,b,dt,v):C&&(A!==ge||F>0&&F&64)?He(C,d,p,!1,!0):(A===ge&&F&384||!b&&$&16)&&He(R,d,p),v&&Kr(u)}(z&&(G=O&&O.onVnodeUnmounted)||U)&&pe(()=>{G&&Ce(G,d,u),U&&Ie(u,null,d,"unmounted")},p)},Kr=u=>{const{type:d,el:p,anchor:v,transition:b}=u;if(d===ge){ai(p,v);return}if(d===$t){S(u);return}const A=()=>{s(p),b&&!b.persisted&&b.afterLeave&&b.afterLeave()};if(u.shapeFlag&1&&b&&!b.persisted){const{leave:O,delayLeave:x}=b,R=()=>O(p,A);x?x(u.el,A,R):R()}else A()},ai=(u,d)=>{let p;for(;u!==d;)p=g(u),s(u),u=p;s(d)},ui=(u,d,p)=>{const{bum:v,scope:b,update:A,subTree:O,um:x}=u;v&&hn(v),b.stop(),A&&(A.active=!1,Me(O,u,d,p)),x&&pe(x,d),pe(()=>{u.isUnmounted=!0},d),d&&d.pendingBranch&&!d.isUnmounted&&u.asyncDep&&!u.asyncResolved&&u.suspenseId===d.pendingId&&(d.deps--,d.deps===0&&d.resolve())},He=(u,d,p,v=!1,b=!1,A=0)=>{for(let O=A;Ou.shapeFlag&6?Qt(u.component.subTree):u.shapeFlag&128?u.suspense.next():g(u.anchor||u.el),Wr=(u,d,p)=>{u==null?d._vnode&&Me(d._vnode,null,null,!0):T(d._vnode||null,u,d,null,null,null,p),ns(),bn(),d._vnode=u},dt={p:T,um:Me,m:Ge,r:Kr,mt:L,mc:y,pc:Z,pbc:I,n:Qt,o:e};let kn,Un;return t&&([kn,Un]=t(dt)),{render:Wr,hydrate:kn,createApp:Dl(Wr,kn)}}function et({effect:e,update:t},n){e.allowRecurse=t.allowRecurse=n}function jr(e,t,n=!1){const r=e.children,s=t.children;if(k(r)&&k(s))for(let o=0;o>1,e[n[l]]0&&(t[r]=n[o-1]),n[o]=r)}}for(o=n.length,i=n[o-1];o-- >0;)n[o]=i,i=t[i];return n}const Xl=e=>e.__isTeleport,Ht=e=>e&&(e.disabled||e.disabled===""),ps=e=>typeof SVGElement<"u"&&e instanceof SVGElement,fr=(e,t)=>{const n=e&&e.to;return re(n)?t?t(n):null:n},Ql={__isTeleport:!0,process(e,t,n,r,s,o,i,l,c,a){const{mc:f,pc:h,pbc:g,o:{insert:w,querySelector:E,createText:T,createComment:H}}=a,m=Ht(t.props);let{shapeFlag:_,children:N,dynamicChildren:S}=t;if(e==null){const B=t.el=T(""),W=t.anchor=T("");w(B,n,r),w(W,n,r);const D=t.target=fr(t.props,E),y=t.targetAnchor=T("");D&&(w(y,D),i=i||ps(D));const P=(I,V)=>{_&16&&f(N,I,V,s,o,i,l,c)};m?P(n,W):D&&P(D,y)}else{t.el=e.el;const B=t.anchor=e.anchor,W=t.target=e.target,D=t.targetAnchor=e.targetAnchor,y=Ht(e.props),P=y?n:W,I=y?B:D;if(i=i||ps(W),S?(g(e.dynamicChildren,S,P,s,o,i,l),jr(e,t,!0)):c||h(e,t,P,I,s,o,i,l,!1),m)y?t.props&&e.props&&t.props.to!==e.props.to&&(t.props.to=e.props.to):cn(t,n,B,a,1);else if((t.props&&t.props.to)!==(e.props&&e.props.to)){const V=t.target=fr(t.props,E);V&&cn(t,V,null,a,0)}else y&&cn(t,W,D,a,1)}Lo(t)},remove(e,t,n,r,{um:s,o:{remove:o}},i){const{shapeFlag:l,children:c,anchor:a,targetAnchor:f,target:h,props:g}=e;if(h&&o(f),i&&o(a),l&16){const w=i||!Ht(g);for(let E=0;E0?Re||mt:null,Gl(),Kt>0&&Re&&Re.push(e),e}function qa(e,t,n,r,s,o){return Ho(Do(e,t,n,r,s,o,!0))}function $o(e,t,n,r,s){return Ho(se(e,t,n,r,s,!0))}function xn(e){return e?e.__v_isVNode===!0:!1}function st(e,t){return e.type===t.type&&e.key===t.key}const jn="__vInternal",jo=({key:e})=>e??null,pn=({ref:e,ref_key:t,ref_for:n})=>(typeof e=="number"&&(e=""+e),e!=null?re(e)||ce(e)||K(e)?{i:fe,r:e,k:t,f:!!n}:e:null);function Do(e,t=null,n=null,r=0,s=null,o=e===ge?0:1,i=!1,l=!1){const c={__v_isVNode:!0,__v_skip:!0,type:e,props:t,key:t&&jo(t),ref:t&&pn(t),scopeId:Ln,slotScopeIds:null,children:n,component:null,suspense:null,ssContent:null,ssFallback:null,dirs:null,transition:null,el:null,anchor:null,target:null,targetAnchor:null,staticCount:0,shapeFlag:o,patchFlag:r,dynamicProps:s,dynamicChildren:null,appContext:null,ctx:fe};return l?(Dr(c,n),o&128&&e.normalize(c)):n&&(c.shapeFlag|=re(n)?8:16),Kt>0&&!i&&Re&&(c.patchFlag>0||o&6)&&c.patchFlag!==32&&Re.push(c),c}const se=ec;function ec(e,t=null,n=null,r=0,s=null,o=!1){if((!e||e===Eo)&&(e=ve),xn(e)){const l=Qe(e,t,!0);return n&&Dr(l,n),Kt>0&&!o&&Re&&(l.shapeFlag&6?Re[Re.indexOf(e)]=l:Re.push(l)),l.patchFlag|=-2,l}if(uc(e)&&(e=e.__vccOpts),t){t=tc(t);let{class:l,style:c}=t;l&&!re(l)&&(t.class=wr(l)),ee(c)&&(co(c)&&!k(c)&&(c=ue({},c)),t.style=vr(c))}const i=re(e)?1:ml(e)?128:Xl(e)?64:ee(e)?4:K(e)?2:0;return Do(e,t,n,r,s,i,o,!0)}function tc(e){return e?co(e)||jn in e?ue({},e):e:null}function Qe(e,t,n=!1){const{props:r,ref:s,patchFlag:o,children:i}=e,l=t?nc(r||{},t):r;return{__v_isVNode:!0,__v_skip:!0,type:e.type,props:l,key:l&&jo(l),ref:t&&t.ref?n&&s?k(s)?s.concat(pn(t)):[s,pn(t)]:pn(t):s,scopeId:e.scopeId,slotScopeIds:e.slotScopeIds,children:i,target:e.target,targetAnchor:e.targetAnchor,staticCount:e.staticCount,shapeFlag:e.shapeFlag,patchFlag:t&&e.type!==ge?o===-1?16:o|16:o,dynamicProps:e.dynamicProps,dynamicChildren:e.dynamicChildren,appContext:e.appContext,dirs:e.dirs,transition:e.transition,component:e.component,suspense:e.suspense,ssContent:e.ssContent&&Qe(e.ssContent),ssFallback:e.ssFallback&&Qe(e.ssFallback),el:e.el,anchor:e.anchor,ctx:e.ctx,ce:e.ce}}function Bo(e=" ",t=0){return se(Et,null,e,t)}function za(e,t){const n=se($t,null,e);return n.staticCount=t,n}function Ya(e="",t=!1){return t?(No(),$o(ve,null,e)):se(ve,null,e)}function Ae(e){return e==null||typeof e=="boolean"?se(ve):k(e)?se(ge,null,e.slice()):typeof e=="object"?qe(e):se(Et,null,String(e))}function qe(e){return e.el===null&&e.patchFlag!==-1||e.memo?e:Qe(e)}function Dr(e,t){let n=0;const{shapeFlag:r}=e;if(t==null)t=null;else if(k(t))n=16;else if(typeof t=="object")if(r&65){const s=t.default;s&&(s._c&&(s._d=!1),Dr(e,s()),s._c&&(s._d=!0));return}else{n=32;const s=t._;!s&&!(jn in t)?t._ctx=fe:s===3&&fe&&(fe.slots._===1?t._=1:(t._=2,e.patchFlag|=1024))}else K(t)?(t={default:t,_ctx:fe},n=32):(t=String(t),r&64?(n=16,t=[Bo(t)]):n=8);e.children=t,e.shapeFlag|=n}function nc(...e){const t={};for(let n=0;nle||fe;let Br,pt,ms="__VUE_INSTANCE_SETTERS__";(pt=er()[ms])||(pt=er()[ms]=[]),pt.push(e=>le=e),Br=e=>{pt.length>1?pt.forEach(t=>t(e)):pt[0](e)};const xt=e=>{Br(e),e.scope.on()},ct=()=>{le&&le.scope.off(),Br(null)};function ko(e){return e.vnode.shapeFlag&4}let Tt=!1;function ic(e,t=!1){Tt=t;const{props:n,children:r}=e.vnode,s=ko(e);kl(e,n,s,t),Wl(e,r);const o=s?lc(e,t):void 0;return Tt=!1,o}function lc(e,t){const n=e.type;e.accessCache=Object.create(null),e.proxy=Lt(new Proxy(e.ctx,Ml));const{setup:r}=n;if(r){const s=e.setupContext=r.length>1?Ko(e):null;xt(e),At();const o=Ye(r,e,0,[e.props,s]);if(St(),ct(),qs(o)){if(o.then(ct,ct),t)return o.then(i=>{ys(e,i,t)}).catch(i=>{Yt(i,e,0)});e.asyncDep=o}else ys(e,o,t)}else Uo(e,t)}function ys(e,t,n){K(t)?e.type.__ssrInlineRender?e.ssrRender=t:e.render=t:ee(t)&&(e.setupState=uo(t)),Uo(e,n)}let _s;function Uo(e,t,n){const r=e.type;if(!e.render){if(!t&&_s&&!r.render){const s=r.template||Hr(e).template;if(s){const{isCustomElement:o,compilerOptions:i}=e.appContext.config,{delimiters:l,compilerOptions:c}=r,a=ue(ue({isCustomElement:o,delimiters:l},i),c);r.render=_s(s,a)}}e.render=r.render||Oe}{xt(e),At();try{Il(e)}finally{St(),ct()}}}function cc(e){return e.attrsProxy||(e.attrsProxy=new Proxy(e.attrs,{get(t,n){return me(e,"get","$attrs"),t[n]}}))}function Ko(e){const t=n=>{e.exposed=n||{}};return{get attrs(){return cc(e)},slots:e.slots,emit:e.emit,expose:t}}function Bn(e){if(e.exposed)return e.exposeProxy||(e.exposeProxy=new Proxy(uo(Lt(e.exposed)),{get(t,n){if(n in t)return t[n];if(n in Nt)return Nt[n](e)},has(t,n){return n in t||n in Nt}}))}function ac(e,t=!0){return K(e)?e.displayName||e.name:e.name||t&&e.__name}function uc(e){return K(e)&&"__vccOpts"in e}const ie=(e,t)=>sl(e,t,Tt);function dr(e,t,n){const r=arguments.length;return r===2?ee(t)&&!k(t)?xn(t)?se(e,null,[t]):se(e,t):se(e,null,t):(r>3?n=Array.prototype.slice.call(arguments,2):r===3&&xn(n)&&(n=[n]),se(e,t,n))}const fc=Symbol.for("v-scx"),dc=()=>wt(fc),hc="3.3.6",pc="http://www.w3.org/2000/svg",ot=typeof document<"u"?document:null,bs=ot&&ot.createElement("template"),gc={insert:(e,t,n)=>{t.insertBefore(e,n||null)},remove:e=>{const t=e.parentNode;t&&t.removeChild(e)},createElement:(e,t,n,r)=>{const s=t?ot.createElementNS(pc,e):ot.createElement(e,n?{is:n}:void 0);return e==="select"&&r&&r.multiple!=null&&s.setAttribute("multiple",r.multiple),s},createText:e=>ot.createTextNode(e),createComment:e=>ot.createComment(e),setText:(e,t)=>{e.nodeValue=t},setElementText:(e,t)=>{e.textContent=t},parentNode:e=>e.parentNode,nextSibling:e=>e.nextSibling,querySelector:e=>ot.querySelector(e),setScopeId(e,t){e.setAttribute(t,"")},insertStaticContent(e,t,n,r,s,o){const i=n?n.previousSibling:t.lastChild;if(s&&(s===o||s.nextSibling))for(;t.insertBefore(s.cloneNode(!0),n),!(s===o||!(s=s.nextSibling)););else{bs.innerHTML=r?`${e}`:e;const l=bs.content;if(r){const c=l.firstChild;for(;c.firstChild;)l.appendChild(c.firstChild);l.removeChild(c)}t.insertBefore(l,n)}return[i?i.nextSibling:t.firstChild,n?n.previousSibling:t.lastChild]}},Ke="transition",Pt="animation",Wt=Symbol("_vtc"),Wo=(e,{slots:t})=>dr(vl,mc(e),t);Wo.displayName="Transition";const Vo={name:String,type:String,css:{type:Boolean,default:!0},duration:[String,Number,Object],enterFromClass:String,enterActiveClass:String,enterToClass:String,appearFromClass:String,appearActiveClass:String,appearToClass:String,leaveFromClass:String,leaveActiveClass:String,leaveToClass:String};Wo.props=ue({},_o,Vo);const tt=(e,t=[])=>{k(e)?e.forEach(n=>n(...t)):e&&e(...t)},vs=e=>e?k(e)?e.some(t=>t.length>1):e.length>1:!1;function mc(e){const t={};for(const M in e)M in Vo||(t[M]=e[M]);if(e.css===!1)return t;const{name:n="v",type:r,duration:s,enterFromClass:o=`${n}-enter-from`,enterActiveClass:i=`${n}-enter-active`,enterToClass:l=`${n}-enter-to`,appearFromClass:c=o,appearActiveClass:a=i,appearToClass:f=l,leaveFromClass:h=`${n}-leave-from`,leaveActiveClass:g=`${n}-leave-active`,leaveToClass:w=`${n}-leave-to`}=e,E=yc(s),T=E&&E[0],H=E&&E[1],{onBeforeEnter:m,onEnter:_,onEnterCancelled:N,onLeave:S,onLeaveCancelled:B,onBeforeAppear:W=m,onAppear:D=_,onAppearCancelled:y=N}=t,P=(M,q,L)=>{nt(M,q?f:l),nt(M,q?a:i),L&&L()},I=(M,q)=>{M._isLeaving=!1,nt(M,h),nt(M,w),nt(M,g),q&&q()},V=M=>(q,L)=>{const ye=M?D:_,X=()=>P(q,M,L);tt(ye,[q,X]),ws(()=>{nt(q,M?c:o),We(q,M?f:l),vs(ye)||Cs(q,r,T,X)})};return ue(t,{onBeforeEnter(M){tt(m,[M]),We(M,o),We(M,i)},onBeforeAppear(M){tt(W,[M]),We(M,c),We(M,a)},onEnter:V(!1),onAppear:V(!0),onLeave(M,q){M._isLeaving=!0;const L=()=>I(M,q);We(M,h),vc(),We(M,g),ws(()=>{M._isLeaving&&(nt(M,h),We(M,w),vs(S)||Cs(M,r,H,L))}),tt(S,[M,L])},onEnterCancelled(M){P(M,!1),tt(N,[M])},onAppearCancelled(M){P(M,!0),tt(y,[M])},onLeaveCancelled(M){I(M),tt(B,[M])}})}function yc(e){if(e==null)return null;if(ee(e))return[zn(e.enter),zn(e.leave)];{const t=zn(e);return[t,t]}}function zn(e){return yi(e)}function We(e,t){t.split(/\s+/).forEach(n=>n&&e.classList.add(n)),(e[Wt]||(e[Wt]=new Set)).add(t)}function nt(e,t){t.split(/\s+/).forEach(r=>r&&e.classList.remove(r));const n=e[Wt];n&&(n.delete(t),n.size||(e[Wt]=void 0))}function ws(e){requestAnimationFrame(()=>{requestAnimationFrame(e)})}let _c=0;function Cs(e,t,n,r){const s=e._endId=++_c,o=()=>{s===e._endId&&r()};if(n)return setTimeout(o,n);const{type:i,timeout:l,propCount:c}=bc(e,t);if(!i)return r();const a=i+"end";let f=0;const h=()=>{e.removeEventListener(a,g),o()},g=w=>{w.target===e&&++f>=c&&h()};setTimeout(()=>{f(n[E]||"").split(", "),s=r(`${Ke}Delay`),o=r(`${Ke}Duration`),i=Es(s,o),l=r(`${Pt}Delay`),c=r(`${Pt}Duration`),a=Es(l,c);let f=null,h=0,g=0;t===Ke?i>0&&(f=Ke,h=i,g=o.length):t===Pt?a>0&&(f=Pt,h=a,g=c.length):(h=Math.max(i,a),f=h>0?i>a?Ke:Pt:null,g=f?f===Ke?o.length:c.length:0);const w=f===Ke&&/\b(transform|all)(,|$)/.test(r(`${Ke}Property`).toString());return{type:f,timeout:h,propCount:g,hasTransform:w}}function Es(e,t){for(;e.lengthxs(n)+xs(e[r])))}function xs(e){return e==="auto"?0:Number(e.slice(0,-1).replace(",","."))*1e3}function vc(){return document.body.offsetHeight}function wc(e,t,n){const r=e[Wt];r&&(t=(t?[t,...r]:[...r]).join(" ")),t==null?e.removeAttribute("class"):n?e.setAttribute("class",t):e.className=t}const Cc=Symbol("_vod");function Ec(e,t,n){const r=e.style,s=re(n);if(n&&!s){if(t&&!re(t))for(const o in t)n[o]==null&&hr(r,o,"");for(const o in n)hr(r,o,n[o])}else{const o=r.display;s?t!==n&&(r.cssText=n):t&&e.removeAttribute("style"),Cc in e&&(r.display=o)}}const Ts=/\s*!important$/;function hr(e,t,n){if(k(n))n.forEach(r=>hr(e,t,r));else if(n==null&&(n=""),t.startsWith("--"))e.setProperty(t,n);else{const r=xc(e,t);Ts.test(n)?e.setProperty(ft(r),n.replace(Ts,""),"important"):e[r]=n}}const As=["Webkit","Moz","ms"],Yn={};function xc(e,t){const n=Yn[t];if(n)return n;let r=Ne(t);if(r!=="filter"&&r in e)return Yn[t]=r;r=Sn(r);for(let s=0;sJn||(Pc.then(()=>Jn=0),Jn=Date.now());function Fc(e,t){const n=r=>{if(!r._vts)r._vts=Date.now();else if(r._vts<=n.attached)return;xe(Ic(r,n.value),t,5,[r])};return n.value=e,n.attached=Mc(),n}function Ic(e,t){if(k(t)){const n=e.stopImmediatePropagation;return e.stopImmediatePropagation=()=>{n.call(e),e._stopped=!0},t.map(r=>s=>!s._stopped&&r&&r(s))}else return t}const Ps=/^on[a-z]/,Lc=(e,t,n,r,s=!1,o,i,l,c)=>{t==="class"?wc(e,r,s):t==="style"?Ec(e,n,r):zt(t)?mr(t)||Rc(e,t,n,r,i):(t[0]==="."?(t=t.slice(1),!0):t[0]==="^"?(t=t.slice(1),!1):Nc(e,t,r,s))?Ac(e,t,r,o,i,l,c):(t==="true-value"?e._trueValue=r:t==="false-value"&&(e._falseValue=r),Tc(e,t,r,s))};function Nc(e,t,n,r){return r?!!(t==="innerHTML"||t==="textContent"||t in e&&Ps.test(t)&&K(n)):t==="spellcheck"||t==="draggable"||t==="translate"||t==="form"||t==="list"&&e.tagName==="INPUT"||t==="type"&&e.tagName==="TEXTAREA"||Ps.test(t)&&re(n)?!1:t in e}const Ms=e=>{const t=e.props["onUpdate:modelValue"]||!1;return k(t)?n=>hn(t,n):t};function Hc(e){e.target.composing=!0}function Fs(e){const t=e.target;t.composing&&(t.composing=!1,t.dispatchEvent(new Event("input")))}const Xn=Symbol("_assign"),Ja={created(e,{modifiers:{lazy:t,trim:n,number:r}},s){e[Xn]=Ms(s);const o=r||s.props&&s.props.type==="number";gt(e,t?"change":"input",i=>{if(i.target.composing)return;let l=e.value;n&&(l=l.trim()),o&&(l=Gn(l)),e[Xn](l)}),n&>(e,"change",()=>{e.value=e.value.trim()}),t||(gt(e,"compositionstart",Hc),gt(e,"compositionend",Fs),gt(e,"change",Fs))},mounted(e,{value:t}){e.value=t??""},beforeUpdate(e,{value:t,modifiers:{lazy:n,trim:r,number:s}},o){if(e[Xn]=Ms(o),e.composing||document.activeElement===e&&e.type!=="range"&&(n||r&&e.value.trim()===t||(s||e.type==="number")&&Gn(e.value)===t))return;const i=t??"";e.value!==i&&(e.value=i)}},$c=["ctrl","shift","alt","meta"],jc={stop:e=>e.stopPropagation(),prevent:e=>e.preventDefault(),self:e=>e.target!==e.currentTarget,ctrl:e=>!e.ctrlKey,shift:e=>!e.shiftKey,alt:e=>!e.altKey,meta:e=>!e.metaKey,left:e=>"button"in e&&e.button!==0,middle:e=>"button"in e&&e.button!==1,right:e=>"button"in e&&e.button!==2,exact:(e,t)=>$c.some(n=>e[`${n}Key`]&&!t.includes(n))},Xa=(e,t)=>(n,...r)=>{for(let s=0;sn=>{if(!("key"in n))return;const r=ft(n.key);if(t.some(s=>s===r||Dc[s]===r))return e(n)},qo=ue({patchProp:Lc},gc);let Dt,Is=!1;function Bc(){return Dt||(Dt=zl(qo))}function kc(){return Dt=Is?Dt:Yl(qo),Is=!0,Dt}const Za=(...e)=>{const t=Bc().createApp(...e),{mount:n}=t;return t.mount=r=>{const s=zo(r);if(!s)return;const o=t._component;!K(o)&&!o.render&&!o.template&&(o.template=s.innerHTML),s.innerHTML="";const i=n(s,!1,s instanceof SVGElement);return s instanceof Element&&(s.removeAttribute("v-cloak"),s.setAttribute("data-v-app","")),i},t},Ga=(...e)=>{const t=kc().createApp(...e),{mount:n}=t;return t.mount=r=>{const s=zo(r);if(s)return n(s,!0,s instanceof SVGElement)},t};function zo(e){return re(e)?document.querySelector(e):e}const eu=(e,t)=>{const n=e.__vccOpts||e;for(const[r,s]of t)n[r]=s;return n},Uc="modulepreload",Kc=function(e){return"/"+e},Ls={},tu=function(t,n,r){if(!n||n.length===0)return t();const s=document.getElementsByTagName("link");return Promise.all(n.map(o=>{if(o=Kc(o),o in Ls)return;Ls[o]=!0;const i=o.endsWith(".css"),l=i?'[rel="stylesheet"]':"";if(!!r)for(let f=s.length-1;f>=0;f--){const h=s[f];if(h.href===o&&(!i||h.rel==="stylesheet"))return}else if(document.querySelector(`link[href="${o}"]${l}`))return;const a=document.createElement("link");if(a.rel=i?"stylesheet":Uc,i||(a.as="script",a.crossOrigin=""),a.href=o,document.head.appendChild(a),i)return new Promise((f,h)=>{a.addEventListener("load",f),a.addEventListener("error",()=>h(new Error(`Unable to preload CSS for ${o}`)))})})).then(()=>t()).catch(o=>{const i=new Event("vite:preloadError",{cancelable:!0});if(i.payload=o,window.dispatchEvent(i),!i.defaultPrevented)throw o})},Wc=window.__VP_SITE_DATA__;function kr(e){return Qs()?(Ai(e),!0):!1}function Pe(e){return typeof e=="function"?e():Mr(e)}function nu(e,t){const n=(t==null?void 0:t.computedGetter)===!1?Mr:Pe;return function(...r){return ie(()=>e.apply(this,r.map(s=>n(s))))}}const Yo=typeof window<"u"&&typeof document<"u",Vc=Object.prototype.toString,qc=e=>Vc.call(e)==="[object Object]",Vt=()=>{},Ns=zc();function zc(){var e;return Yo&&((e=window==null?void 0:window.navigator)==null?void 0:e.userAgent)&&/iP(ad|hone|od)/.test(window.navigator.userAgent)}function Yc(e,t){function n(...r){return new Promise((s,o)=>{Promise.resolve(e(()=>t.apply(this,r),{fn:t,thisArg:this,args:r})).then(s).catch(o)})}return n}const Jo=e=>e();function Jc(e,t={}){let n,r,s=Vt;const o=l=>{clearTimeout(l),s(),s=Vt};return l=>{const c=Pe(e),a=Pe(t.maxWait);return n&&o(n),c<=0||a!==void 0&&a<=0?(r&&(o(r),r=null),Promise.resolve(l())):new Promise((f,h)=>{s=t.rejectOnCancel?h:f,a&&!r&&(r=setTimeout(()=>{n&&o(n),r=null,f(l())},a)),n=setTimeout(()=>{r&&o(r),r=null,f(l())},c)})}}function Xc(e=Jo){const t=ae(!0);function n(){t.value=!1}function r(){t.value=!0}const s=(...o)=>{t.value&&e(...o)};return{isActive:Pn(t),pause:n,resume:r,eventFilter:s}}function Xo(...e){if(e.length!==1)return tl(...e);const t=e[0];return typeof t=="function"?Pn(Zi(()=>({get:t,set:Vt}))):ae(t)}function Qo(e,t,n={}){const{eventFilter:r=Jo,...s}=n;return Je(e,Yc(r,t),s)}function Qc(e,t,n={}){const{eventFilter:r,...s}=n,{eventFilter:o,pause:i,resume:l,isActive:c}=Xc(r);return{stop:Qo(e,t,{...s,eventFilter:o}),pause:i,resume:l,isActive:c}}function Zc(e,t=!0){Dn()?Rt(e):t?e():Mn(e)}function ru(e,t,n={}){const{debounce:r=0,maxWait:s=void 0,...o}=n;return Qo(e,t,{...o,eventFilter:Jc(r,{maxWait:s})})}function su(e,t,n){let r;ce(n)?r={evaluating:n}:r=n||{};const{lazy:s=!1,evaluating:o=void 0,shallow:i=!0,onError:l=Vt}=r,c=ae(!s),a=i?Pr(t):ae(t);let f=0;return Ir(async h=>{if(!c.value)return;f++;const g=f;let w=!1;o&&Promise.resolve().then(()=>{o.value=!0});try{const E=await e(T=>{h(()=>{o&&(o.value=!1),w||T()})});g===f&&(a.value=E)}catch(E){l(E)}finally{o&&g===f&&(o.value=!1),w=!0}}),s?ie(()=>(c.value=!0,a.value)):a}function Zo(e){var t;const n=Pe(e);return(t=n==null?void 0:n.$el)!=null?t:n}const De=Yo?window:void 0;function qt(...e){let t,n,r,s;if(typeof e[0]=="string"||Array.isArray(e[0])?([n,r,s]=e,t=De):[t,n,r,s]=e,!t)return Vt;Array.isArray(n)||(n=[n]),Array.isArray(r)||(r=[r]);const o=[],i=()=>{o.forEach(f=>f()),o.length=0},l=(f,h,g,w)=>(f.addEventListener(h,g,w),()=>f.removeEventListener(h,g,w)),c=Je(()=>[Zo(t),Pe(s)],([f,h])=>{if(i(),!f)return;const g=qc(h)?{...h}:h;o.push(...n.flatMap(w=>r.map(E=>l(f,w,E,g))))},{immediate:!0,flush:"post"}),a=()=>{c(),i()};return kr(a),a}function Gc(e){return typeof e=="function"?e:typeof e=="string"?t=>t.key===e:Array.isArray(e)?t=>e.includes(t.key):()=>!0}function ou(...e){let t,n,r={};e.length===3?(t=e[0],n=e[1],r=e[2]):e.length===2?typeof e[1]=="object"?(t=!0,n=e[0],r=e[1]):(t=e[0],n=e[1]):(t=!0,n=e[0]);const{target:s=De,eventName:o="keydown",passive:i=!1,dedupe:l=!1}=r,c=Gc(t);return qt(s,o,f=>{f.repeat&&Pe(l)||c(f)&&n(f)},i)}function ea(){const e=ae(!1);return Dn()&&Rt(()=>{e.value=!0}),e}function ta(e){const t=ea();return ie(()=>(t.value,!!e()))}function na(e,t={}){const{window:n=De}=t,r=ta(()=>n&&"matchMedia"in n&&typeof n.matchMedia=="function");let s;const o=ae(!1),i=a=>{o.value=a.matches},l=()=>{s&&("removeEventListener"in s?s.removeEventListener("change",i):s.removeListener(i))},c=Ir(()=>{r.value&&(l(),s=n.matchMedia(Pe(e)),"addEventListener"in s?s.addEventListener("change",i):s.addListener(i),o.value=s.matches)});return kr(()=>{c(),l(),s=void 0}),o}const an=typeof globalThis<"u"?globalThis:typeof window<"u"?window:typeof global<"u"?global:typeof self<"u"?self:{},un="__vueuse_ssr_handlers__",ra=sa();function sa(){return un in an||(an[un]=an[un]||{}),an[un]}function Go(e,t){return ra[e]||t}function oa(e){return e==null?"any":e instanceof Set?"set":e instanceof Map?"map":e instanceof Date?"date":typeof e=="boolean"?"boolean":typeof e=="string"?"string":typeof e=="object"?"object":Number.isNaN(e)?"any":"number"}const ia={boolean:{read:e=>e==="true",write:e=>String(e)},object:{read:e=>JSON.parse(e),write:e=>JSON.stringify(e)},number:{read:e=>Number.parseFloat(e),write:e=>String(e)},any:{read:e=>e,write:e=>String(e)},string:{read:e=>e,write:e=>String(e)},map:{read:e=>new Map(JSON.parse(e)),write:e=>JSON.stringify(Array.from(e.entries()))},set:{read:e=>new Set(JSON.parse(e)),write:e=>JSON.stringify(Array.from(e))},date:{read:e=>new Date(e),write:e=>e.toISOString()}},Hs="vueuse-storage";function Ur(e,t,n,r={}){var s;const{flush:o="pre",deep:i=!0,listenToStorageChanges:l=!0,writeDefaults:c=!0,mergeDefaults:a=!1,shallow:f,window:h=De,eventFilter:g,onError:w=y=>{console.error(y)}}=r,E=(f?Pr:ae)(t);if(!n)try{n=Go("getDefaultStorage",()=>{var y;return(y=De)==null?void 0:y.localStorage})()}catch(y){w(y)}if(!n)return E;const T=Pe(t),H=oa(T),m=(s=r.serializer)!=null?s:ia[H],{pause:_,resume:N}=Qc(E,()=>S(E.value),{flush:o,deep:i,eventFilter:g});return h&&l&&(qt(h,"storage",D),qt(h,Hs,W)),D(),E;function S(y){try{if(y==null)n.removeItem(e);else{const P=m.write(y),I=n.getItem(e);I!==P&&(n.setItem(e,P),h&&h.dispatchEvent(new CustomEvent(Hs,{detail:{key:e,oldValue:I,newValue:P,storageArea:n}})))}}catch(P){w(P)}}function B(y){const P=y?y.newValue:n.getItem(e);if(P==null)return c&&T!==null&&n.setItem(e,m.write(T)),T;if(!y&&a){const I=m.read(P);return typeof a=="function"?a(I,T):H==="object"&&!Array.isArray(I)?{...T,...I}:I}else return typeof P!="string"?P:m.read(P)}function W(y){D(y.detail)}function D(y){if(!(y&&y.storageArea!==n)){if(y&&y.key==null){E.value=T;return}if(!(y&&y.key!==e)){_();try{(y==null?void 0:y.newValue)!==m.write(E.value)&&(E.value=B(y))}catch(P){w(P)}finally{y?Mn(N):N()}}}}}function la(e){return na("(prefers-color-scheme: dark)",e)}function ca(e={}){const{selector:t="html",attribute:n="class",initialValue:r="auto",window:s=De,storage:o,storageKey:i="vueuse-color-scheme",listenToStorageChanges:l=!0,storageRef:c,emitAuto:a,disableTransition:f=!0}=e,h={auto:"",light:"light",dark:"dark",...e.modes||{}},g=la({window:s}),w=ie(()=>g.value?"dark":"light"),E=c||(i==null?Xo(r):Ur(i,r,o,{window:s,listenToStorageChanges:l})),T=ie(()=>E.value==="auto"?w.value:E.value),H=Go("updateHTMLAttrs",(S,B,W)=>{const D=typeof S=="string"?s==null?void 0:s.document.querySelector(S):Zo(S);if(!D)return;let y;if(f){y=s.document.createElement("style");const P="*,*::before,*::after{-webkit-transition:none!important;-moz-transition:none!important;-o-transition:none!important;-ms-transition:none!important;transition:none!important}";y.appendChild(document.createTextNode(P)),s.document.head.appendChild(y)}if(B==="class"){const P=W.split(/\s/g);Object.values(h).flatMap(I=>(I||"").split(/\s/g)).filter(Boolean).forEach(I=>{P.includes(I)?D.classList.add(I):D.classList.remove(I)})}else D.setAttribute(B,W);f&&(s.getComputedStyle(y).opacity,document.head.removeChild(y))});function m(S){var B;H(t,n,(B=h[S])!=null?B:S)}function _(S){e.onChanged?e.onChanged(S,m):m(S)}Je(T,_,{flush:"post",immediate:!0}),Zc(()=>_(T.value));const N=ie({get(){return a?E.value:T.value},set(S){E.value=S}});try{return Object.assign(N,{store:E,system:w,state:T})}catch{return N}}function aa(e={}){const{valueDark:t="dark",valueLight:n=""}=e,r=ca({...e,onChanged:(o,i)=>{var l;e.onChanged?(l=e.onChanged)==null||l.call(e,o==="dark",i,o):i(o)},modes:{dark:t,light:n}});return ie({get(){return r.value==="dark"},set(o){const i=o?"dark":"light";r.system.value===i?r.value="auto":r.value=i}})}function Qn(e){return typeof Window<"u"&&e instanceof Window?e.document.documentElement:typeof Document<"u"&&e instanceof Document?e.documentElement:e}function iu(e,t,n={}){const{window:r=De}=n;return Ur(e,t,r==null?void 0:r.localStorage,n)}function ei(e){const t=window.getComputedStyle(e);if(t.overflowX==="scroll"||t.overflowY==="scroll"||t.overflowX==="auto"&&e.clientWidth1?!0:(t.preventDefault&&t.preventDefault(),!1)}function lu(e,t=!1){const n=ae(t);let r=null,s;Je(Xo(e),l=>{const c=Qn(Pe(l));if(c){const a=c;s=a.style.overflow,n.value&&(a.style.overflow="hidden")}},{immediate:!0});const o=()=>{const l=Qn(Pe(e));!l||n.value||(Ns&&(r=qt(l,"touchmove",c=>{ua(c)},{passive:!1})),l.style.overflow="hidden",n.value=!0)},i=()=>{const l=Qn(Pe(e));!l||!n.value||(Ns&&(r==null||r()),l.style.overflow=s,n.value=!1)};return kr(i),ie({get(){return n.value},set(l){l?o():i()}})}function cu(e,t,n={}){const{window:r=De}=n;return Ur(e,t,r==null?void 0:r.sessionStorage,n)}function au(e={}){const{window:t=De}=e;if(!t)return{x:ae(0),y:ae(0)};const n=ae(t.scrollX),r=ae(t.scrollY);return qt(t,"scroll",()=>{n.value=t.scrollX,r.value=t.scrollY},{capture:!1,passive:!0}),{x:n,y:r}}const ti=/^[a-z]+:/i,fa="vitepress-theme-appearance",ni=/#.*$/,da=/(index)?\.(md|html)$/,Ee=typeof document<"u",ri={relativePath:"",filePath:"",title:"404",description:"Not Found",headers:[],frontmatter:{sidebar:!1,layout:"page"},lastUpdated:0,isNotFound:!0};function ha(e,t,n=!1){if(t===void 0)return!1;if(e=$s(`/${e}`),n)return new RegExp(t).test(e);if($s(t)!==e)return!1;const r=t.match(ni);return r?(Ee?location.hash:"")===r[0]:!0}function $s(e){return decodeURI(e).replace(ni,"").replace(da,"")}function pa(e){return ti.test(e)}function ga(e,t){var r,s,o,i,l,c,a;const n=Object.keys(e.locales).find(f=>f!=="root"&&!pa(f)&&ha(t,`/${f}/`,!0))||"root";return Object.assign({},e,{localeIndex:n,lang:((r=e.locales[n])==null?void 0:r.lang)??e.lang,dir:((s=e.locales[n])==null?void 0:s.dir)??e.dir,title:((o=e.locales[n])==null?void 0:o.title)??e.title,titleTemplate:((i=e.locales[n])==null?void 0:i.titleTemplate)??e.titleTemplate,description:((l=e.locales[n])==null?void 0:l.description)??e.description,head:oi(e.head,((c=e.locales[n])==null?void 0:c.head)??[]),themeConfig:{...e.themeConfig,...(a=e.locales[n])==null?void 0:a.themeConfig}})}function si(e,t){const n=t.title||e.title,r=t.titleTemplate??e.titleTemplate;if(typeof r=="string"&&r.includes(":title"))return r.replace(/:title/g,n);const s=ma(e.title,r);return`${n}${s}`}function ma(e,t){return t===!1?"":t===!0||t===void 0?` | ${e}`:e===t?"":` | ${t}`}function ya(e,t){const[n,r]=t;if(n!=="meta")return!1;const s=Object.entries(r)[0];return s==null?!1:e.some(([o,i])=>o===n&&i[s[0]]===s[1])}function oi(e,t){return[...e.filter(n=>!ya(t,n)),...t]}const _a=/[\u0000-\u001F"#$&*+,:;<=>?[\]^`{|}\u007F]/g,ba=/^[a-z]:/i;function js(e){const t=ba.exec(e),n=t?t[0]:"";return n+e.slice(n.length).replace(_a,"_").replace(/(^|\/)_+(?=[^/]*$)/,"$1")}const va=Symbol(),at=Pr(Wc);function uu(e){const t=ie(()=>ga(at.value,e.data.relativePath)),n=t.value.appearance,r=n==="force-dark"?ae(!0):n?aa({storageKey:fa,initialValue:()=>typeof n=="string"?n:"auto",...typeof n=="object"?n:{}}):ae(!1);return{site:t,theme:ie(()=>t.value.themeConfig),page:ie(()=>e.data),frontmatter:ie(()=>e.data.frontmatter),params:ie(()=>e.data.params),lang:ie(()=>t.value.lang),dir:ie(()=>t.value.dir),localeIndex:ie(()=>t.value.localeIndex||"root"),title:ie(()=>si(t.value,e.data)),description:ie(()=>e.data.description||t.value.description),isDark:r}}function wa(){const e=wt(va);if(!e)throw new Error("vitepress data not properly injected in app");return e}function Ca(e,t){return`${e}${t}`.replace(/\/+/g,"/")}function Ds(e){return ti.test(e)||!e.startsWith("/")?e:Ca(at.value.base,e)}function Ea(e){let t=e.replace(/\.html$/,"");if(t=decodeURIComponent(t),t=t.replace(/\/$/,"/index"),Ee){const n="/";t=js(t.slice(n.length).replace(/\//g,"_")||"index")+".md";let r=__VP_HASH_MAP__[t.toLowerCase()];if(r||(t=t.endsWith("_index.md")?t.slice(0,-9)+".md":t.slice(0,-3)+"_index.md",r=__VP_HASH_MAP__[t.toLowerCase()]),!r)return null;t=`${n}assets/${t}.${r}.js`}else t=`./${js(t.slice(1).replace(/\//g,"_"))}.md.js`;return t}let gn=[];function fu(e){gn.push(e),$n(()=>{gn=gn.filter(t=>t!==e)})}const xa=Symbol(),ii="http://a.com",Ta=()=>({path:"/",component:null,data:ri});function du(e,t){const n=On(Ta()),r={route:n,go:s};async function s(l=Ee?location.href:"/"){var c,a;l=pr(l),await((c=r.onBeforeRouteChange)==null?void 0:c.call(r,l))!==!1&&(Us(l),await i(l),await((a=r.onAfterRouteChanged)==null?void 0:a.call(r,l)))}let o=null;async function i(l,c=0,a=!1){var g;if(await((g=r.onBeforePageLoad)==null?void 0:g.call(r,l))===!1)return;const f=new URL(l,ii),h=o=f.pathname;try{let w=await e(h);if(!w)throw new Error(`Page not found: ${h}`);if(o===h){o=null;const{default:E,__pageData:T}=w;if(!E)throw new Error(`Invalid route component: ${E}`);n.path=Ee?h:Ds(h),n.component=Lt(E),n.data=Lt(T),Ee&&Mn(()=>{let H=at.value.base+T.relativePath.replace(/(?:(^|\/)index)?\.md$/,"$1");if(!at.value.cleanUrls&&!H.endsWith("/")&&(H+=".html"),H!==f.pathname&&(f.pathname=H,l=H+f.search+f.hash,history.replaceState(null,"",l)),f.hash&&!c){let m=null;try{m=document.getElementById(decodeURIComponent(f.hash).slice(1))}catch(_){console.warn(_)}if(m){Bs(m,f.hash);return}}window.scrollTo(0,c)})}}catch(w){if(!/fetch|Page not found/.test(w.message)&&!/^\/404(\.html|\/)?$/.test(l)&&console.error(w),!a)try{const E=await fetch(at.value.base+"hashmap.json");window.__VP_HASH_MAP__=await E.json(),await i(l,c,!0);return}catch{}o===h&&(o=null,n.path=Ee?h:Ds(h),n.component=t?Lt(t):null,n.data=ri)}}return Ee&&(window.addEventListener("click",l=>{if(l.target.closest("button"))return;const a=l.target.closest("a");if(a&&!a.closest(".vp-raw")&&(a instanceof SVGElement||!a.download)){const{target:f}=a,{href:h,origin:g,pathname:w,hash:E,search:T}=new URL(a.href instanceof SVGAnimatedString?a.href.animVal:a.href,a.baseURI),H=window.location,m=w.match(/\.\w+$/);!l.ctrlKey&&!l.shiftKey&&!l.altKey&&!l.metaKey&&!f&&g===H.origin&&!(m&&m[0]!==".html")&&(l.preventDefault(),w===H.pathname&&T===H.search?(E!==H.hash&&(history.pushState(null,"",E),window.dispatchEvent(new Event("hashchange"))),E?Bs(a,E,a.classList.contains("header-anchor")):(Us(h),window.scrollTo(0,0))):s(h))}},{capture:!0}),window.addEventListener("popstate",l=>{i(pr(location.href),l.state&&l.state.scrollPosition||0)}),window.addEventListener("hashchange",l=>{l.preventDefault()})),r}function Aa(){const e=wt(xa);if(!e)throw new Error("useRouter() is called without provider.");return e}function li(){return Aa().route}function Bs(e,t,n=!1){let r=null;try{r=e.classList.contains("header-anchor")?e:document.getElementById(decodeURIComponent(t).slice(1))}catch(s){console.warn(s)}if(r){let a=function(){!n||Math.abs(c-window.scrollY)>window.innerHeight?window.scrollTo(0,c):window.scrollTo({left:0,top:c,behavior:"smooth"})},s=at.value.scrollOffset,o=0,i=24;if(typeof s=="object"&&"padding"in s&&(i=s.padding,s=s.selector),typeof s=="number")o=s;else if(typeof s=="string")o=ks(s,i);else if(Array.isArray(s))for(const f of s){const h=ks(f,i);if(h){o=h;break}}const l=parseInt(window.getComputedStyle(r).paddingTop,10),c=window.scrollY+r.getBoundingClientRect().top-o+l;requestAnimationFrame(a)}}function ks(e,t){const n=document.querySelector(e);if(!n)return 0;const r=n.getBoundingClientRect().bottom;return r<0?0:r+t}function Us(e){Ee&&e!==pr(location.href)&&(history.replaceState({scrollPosition:window.scrollY},document.title),history.pushState(null,"",e))}function pr(e){const t=new URL(e,ii);return t.pathname=t.pathname.replace(/(^|\/)index(\.html)?$/,"$1"),at.value.cleanUrls?t.pathname=t.pathname.replace(/\.html$/,""):!t.pathname.endsWith("/")&&!t.pathname.endsWith(".html")&&(t.pathname+=".html"),t.pathname+t.search+t.hash}const Ks=()=>gn.forEach(e=>e()),hu=Lr({name:"VitePressContent",props:{as:{type:[Object,String],default:"div"}},setup(e){const t=li(),{site:n}=wa();return()=>dr(e.as,n.value.contentProps??{style:{position:"relative"}},[t.component?dr(t.component,{onVnodeMounted:Ks,onVnodeUpdated:Ks}):"404 Page Not Found"])}}),pu=Lr({setup(e,{slots:t}){const n=ae(!1);return Rt(()=>{n.value=!0}),()=>n.value&&t.default?t.default():null}});function gu(){Ee&&window.addEventListener("click",e=>{var n;const t=e.target;if(t.matches(".vp-code-group input")){const r=(n=t.parentElement)==null?void 0:n.parentElement;if(!r)return;const s=Array.from(r.querySelectorAll("input")).indexOf(t);if(s<0)return;const o=r.querySelector(".blocks");if(!o)return;const i=Array.from(o.children).find(a=>a.classList.contains("active"));if(!i)return;const l=o.children[s];if(!l||i===l)return;i.classList.remove("active"),l.classList.add("active");const c=r==null?void 0:r.querySelector(`label[for="${t.id}"]`);c==null||c.scrollIntoView({block:"nearest"})}})}function mu(){if(Ee){const e=new WeakMap;window.addEventListener("click",t=>{var r;const n=t.target;if(n.matches('div[class*="language-"] > button.copy')){const s=n.parentElement,o=(r=n.nextElementSibling)==null?void 0:r.nextElementSibling;if(!s||!o)return;const i=/language-(shellscript|shell|bash|sh|zsh)/.test(s.className);let l="";o.querySelectorAll("span.line:not(.diff.remove)").forEach(c=>l+=(c.textContent||"")+` +`),l=l.slice(0,-1),i&&(l=l.replace(/^ *(\$|>) /gm,"").trim()),Sa(l).then(()=>{n.classList.add("copied"),clearTimeout(e.get(n));const c=setTimeout(()=>{n.classList.remove("copied"),n.blur(),e.delete(n)},2e3);e.set(n,c)})}})}}async function Sa(e){try{return navigator.clipboard.writeText(e)}catch{const t=document.createElement("textarea"),n=document.activeElement;t.value=e,t.setAttribute("readonly",""),t.style.contain="strict",t.style.position="absolute",t.style.left="-9999px",t.style.fontSize="12pt";const r=document.getSelection(),s=r?r.rangeCount>0&&r.getRangeAt(0):null;document.body.appendChild(t),t.select(),t.selectionStart=0,t.selectionEnd=e.length,document.execCommand("copy"),document.body.removeChild(t),s&&(r.removeAllRanges(),r.addRange(s)),n&&n.focus()}}function yu(e,t){let n=[],r=!0;const s=o=>{if(r){r=!1;return}const i=o.map(Ws);n.forEach((l,c)=>{const a=i.findIndex(f=>f==null?void 0:f.isEqualNode(l??null));a!==-1?delete i[a]:(l==null||l.remove(),delete n[c])}),i.forEach(l=>l&&document.head.appendChild(l)),n=[...n,...i].filter(Boolean)};Ir(()=>{const o=e.data,i=t.value,l=o&&o.description,c=o&&o.frontmatter.head||[],a=si(i,o);a!==document.title&&(document.title=a);const f=l||i.description;let h=document.querySelector("meta[name=description]");h?h.getAttribute("content")!==f&&h.setAttribute("content",f):Ws(["meta",{name:"description",content:f}]),s(oi(i.head,Oa(c)))})}function Ws([e,t,n]){const r=document.createElement(e);for(const s in t)r.setAttribute(s,t[s]);return n&&(r.innerHTML=n),e==="script"&&!t.async&&(r.async=!1),r}function Ra(e){return e[0]==="meta"&&e[1]&&e[1].name==="description"}function Oa(e){return e.filter(t=>!Ra(t))}const Zn=new Set,ci=()=>document.createElement("link"),Pa=e=>{const t=ci();t.rel="prefetch",t.href=e,document.head.appendChild(t)},Ma=e=>{const t=new XMLHttpRequest;t.open("GET",e,t.withCredentials=!0),t.send()};let fn;const Fa=Ee&&(fn=ci())&&fn.relList&&fn.relList.supports&&fn.relList.supports("prefetch")?Pa:Ma;function _u(){if(!Ee||!window.IntersectionObserver)return;let e;if((e=navigator.connection)&&(e.saveData||/2g/.test(e.effectiveType)))return;const t=window.requestIdleCallback||setTimeout;let n=null;const r=()=>{n&&n.disconnect(),n=new IntersectionObserver(o=>{o.forEach(i=>{if(i.isIntersecting){const l=i.target;n.unobserve(l);const{pathname:c}=l;if(!Zn.has(c)){Zn.add(c);const a=Ea(c);a&&Fa(a)}}})}),t(()=>{document.querySelectorAll("#app a").forEach(o=>{const{hostname:i,pathname:l}=new URL(o.href instanceof SVGAnimatedString?o.href.animVal:o.href,o.baseURI),c=l.match(/\.\w+$/);c&&c[0]!==".html"||o.target!=="_blank"&&i===location.hostname&&(l!==location.pathname?n.observe(o):Zn.add(l))})})};Rt(r);const s=li();Je(()=>s.path,r),$n(()=>{n&&n.disconnect()})}export{Qa as $,Ha as A,Al as B,Da as C,ka as D,Pr as E,ge as F,fu as G,se as H,Ba as I,ti as J,li as K,nc as L,wt as M,vr as N,Mn as O,au as P,za as Q,Pn as R,nu as S,Wo as T,tl as U,ou as V,ja as W,tu as X,lu as Y,Bl as Z,eu as _,Bo as a,Ka as a0,Xa as a1,Wa as a2,dr as a3,yu as a4,xa as a5,uu as a6,va as a7,hu as a8,pu as a9,at as aa,Ga as ab,du as ac,Ea as ad,_u as ae,mu as af,gu as ag,Zo as ah,kr as ai,su as aj,cu as ak,iu as al,ru as am,Aa as an,qt as ao,Co as ap,$a as aq,Ja as ar,ce as as,Va as at,Lt as au,Za as av,$o as b,qa as c,Lr as d,Ya as e,Ds as f,ie as g,ae as h,pa as i,Rt as j,Do as k,Mr as l,Na as m,wr as n,No as o,La as p,ha as q,Ua as r,Ee as s,Ia as t,wa as u,na as v,fl as w,Je as x,Ir as y,$n as z}; diff --git a/assets/chunks/mainnet_versions.84d9fd1c.js b/assets/chunks/mainnet_versions.84d9fd1c.js new file mode 100644 index 00000000000..9233bd2b58f --- /dev/null +++ b/assets/chunks/mainnet_versions.84d9fd1c.js @@ -0,0 +1 @@ +const a=Object.freeze({"app-latest-tag":"v1.7.0","app-latest-sha":"915847191e80d836f862eea2664949d9a240abea","core-latest-tag":"v1.35.0-tm-v0.34.29","core-latest-sha":"f95a434e674610b3a2c5e7d26bd87da69ccaa213","node-latest-tag":"v0.13.2","node-latest-sha":"c1b41b0973e9d140b7651295e879d27ad47f42c4"});export{a as m}; diff --git a/assets/chunks/mocha_versions.3ba4a823.js b/assets/chunks/mocha_versions.3ba4a823.js new file mode 100644 index 00000000000..ce9cd4122fd --- /dev/null +++ b/assets/chunks/mocha_versions.3ba4a823.js @@ -0,0 +1 @@ +const a=Object.freeze({"app-latest-tag":"v1.7.0","app-latest-sha":"915847191e80d836f862eea2664949d9a240abea","core-latest-tag":"v1.35.0-tm-v0.34.29","core-latest-sha":"f95a434e674610b3a2c5e7d26bd87da69ccaa213","node-latest-tag":"v0.13.3","node-latest-sha":"05238b3e087eb9ecd3b9684cd0125f2400f6f0c7"});export{a as m}; diff --git a/assets/chunks/theme.6bee4cb8.js b/assets/chunks/theme.6bee4cb8.js new file mode 100644 index 00000000000..811529c4095 --- /dev/null +++ b/assets/chunks/theme.6bee4cb8.js @@ -0,0 +1 @@ +import{d as b,o as a,c as i,r as u,n as T,a as x,t as L,_ as m,b as $,w as v,T as ce,e as f,u as ze,i as De,f as ue,g as k,h as M,j as G,k as c,l,p as H,m as z,q as O,s as q,v as re,x as U,y as te,z as de,A as Ve,B as Ee,C as j,F as N,D as A,E as _e,G as Y,H as h,I as F,J as we,K as se,L as Z,M as ne,N as Fe,O as Oe,P as Le,Q as Ge,R as Ue,S as je,U as Ke,V as ke,W as Re,X as qe,Y as Se,Z as Me,$ as We,a0 as Ye,a1 as Je,a2 as Xe}from"./framework.51d6c45b.js";const Ze=b({__name:"VPBadge",props:{text:{},type:{default:"tip"}},setup(s){return(e,t)=>(a(),i("span",{class:T(["VPBadge",e.type])},[u(e.$slots,"default",{},()=>[x(L(e.text),1)],!0)],2))}});const Qe=m(Ze,[["__scopeId","data-v-ea5b2908"]]),et={key:0,class:"VPBackdrop"},tt=b({__name:"VPBackdrop",props:{show:{type:Boolean}},setup(s){return(e,t)=>(a(),$(ce,{name:"fade"},{default:v(()=>[e.show?(a(),i("div",et)):f("",!0)]),_:1}))}});const st=m(tt,[["__scopeId","data-v-54a304ca"]]),P=ze;function nt(s,e){let t,n=!1;return()=>{t&&clearTimeout(t),n?t=setTimeout(s,e):(s(),(n=!0)&&setTimeout(()=>n=!1,e))}}function le(s){return/^\//.test(s)?s:`/${s}`}function J(s){const{pathname:e,search:t,hash:n,protocol:o}=new URL(s,"http://a.com");if(De(s)||s.startsWith("#")||!o.startsWith("http")||/\.(?!html|md)\w+($|\?)/i.test(s))return s;const{site:r}=P(),d=e.endsWith("/")||e.endsWith(".html")?s:s.replace(/(?:(^\.+)\/)?.*$/,`$1${e.replace(/(\.md)?$/,r.value.cleanUrls?"":".html")}${t}${n}`);return ue(d)}function X({removeCurrent:s=!0,correspondingLink:e=!1}={}){const{site:t,localeIndex:n,page:o,theme:r}=P(),d=k(()=>{var _,g;return{label:(_=t.value.locales[n.value])==null?void 0:_.label,link:((g=t.value.locales[n.value])==null?void 0:g.link)||(n.value==="root"?"/":`/${n.value}/`)}});return{localeLinks:k(()=>Object.entries(t.value.locales).flatMap(([_,g])=>s&&d.value.label===g.label?[]:{text:g.label,link:ot(g.link||(_==="root"?"/":`/${_}/`),r.value.i18nRouting!==!1&&e,o.value.relativePath.slice(d.value.link.length-1),!t.value.cleanUrls)})),currentLang:d}}function ot(s,e,t,n){return e?s.replace(/\/$/,"")+le(t.replace(/(^|\/)index\.md$/,"$1").replace(/\.md$/,n?".html":"")):s}const at=s=>(H("data-v-b9c0c15a"),s=s(),z(),s),rt={class:"NotFound"},lt={class:"code"},it={class:"title"},ct=at(()=>c("div",{class:"divider"},null,-1)),ut={class:"quote"},dt={class:"action"},_t=["href","aria-label"],vt=b({__name:"NotFound",setup(s){const{site:e,theme:t}=P(),{localeLinks:n}=X({removeCurrent:!1}),o=M("/");return G(()=>{var d;const r=window.location.pathname.replace(e.value.base,"").replace(/(^.*?\/).*$/,"/$1");n.value.length&&(o.value=((d=n.value.find(({link:p})=>p.startsWith(r)))==null?void 0:d.link)||n.value[0].link)}),(r,d)=>{var p,_,g,V,y;return a(),i("div",rt,[c("p",lt,L(((p=l(t).notFound)==null?void 0:p.code)??"404"),1),c("h1",it,L(((_=l(t).notFound)==null?void 0:_.title)??"PAGE NOT FOUND"),1),ct,c("blockquote",ut,L(((g=l(t).notFound)==null?void 0:g.quote)??"But if you don't change your direction, and if you keep looking, you may end up where you are heading."),1),c("div",dt,[c("a",{class:"link",href:l(ue)(o.value),"aria-label":((V=l(t).notFound)==null?void 0:V.linkLabel)??"go to home"},L(((y=l(t).notFound)==null?void 0:y.linkText)??"Take me home"),9,_t)])])}}});const pt=m(vt,[["__scopeId","data-v-b9c0c15a"]]);function Ne(s,e){if(Array.isArray(s))return Q(s);if(s==null)return[];e=le(e);const t=Object.keys(s).sort((o,r)=>r.split("/").length-o.split("/").length).find(o=>e.startsWith(le(o))),n=t?s[t]:[];return Array.isArray(n)?Q(n):Q(n.items,n.base)}function ht(s){const e=[];let t=0;for(const n in s){const o=s[n];if(o.items){t=e.push(o);continue}e[t]||e.push({items:[]}),e[t].items.push(o)}return e}function ft(s){const e=[];function t(n){for(const o of n)o.text&&o.link&&e.push({text:o.text,link:o.link,docFooterText:o.docFooterText}),o.items&&t(o.items)}return t(s),e}function ie(s,e){return Array.isArray(e)?e.some(t=>ie(s,t)):O(s,e.link)?!0:e.items?ie(s,e.items):!1}function Q(s,e){return[...s].map(t=>{const n={...t},o=n.base||e;return o&&n.link&&(n.link=o+n.link),n.items&&(n.items=Q(n.items,o)),n})}function D(){const{frontmatter:s,page:e,theme:t}=P(),n=re("(min-width: 960px)"),o=M(!1),r=k(()=>{const B=t.value.sidebar,w=e.value.relativePath;return B?Ne(B,w):[]}),d=M(r.value);U(r,(B,w)=>{JSON.stringify(B)!==JSON.stringify(w)&&(d.value=r.value)});const p=k(()=>s.value.sidebar!==!1&&d.value.length>0&&s.value.layout!=="home"),_=k(()=>g?s.value.aside==null?t.value.aside==="left":s.value.aside==="left":!1),g=k(()=>s.value.layout==="home"?!1:s.value.aside!=null?!!s.value.aside:t.value.aside!==!1),V=k(()=>p.value&&n.value),y=k(()=>p.value?ht(d.value):[]);function I(){o.value=!0}function S(){o.value=!1}function C(){o.value?S():I()}return{isOpen:o,sidebar:d,sidebarGroups:y,hasSidebar:p,hasAside:g,leftAside:_,isSidebarEnabled:V,open:I,close:S,toggle:C}}function mt(s,e){let t;te(()=>{t=s.value?document.activeElement:void 0}),G(()=>{window.addEventListener("keyup",n)}),de(()=>{window.removeEventListener("keyup",n)});function n(o){o.key==="Escape"&&s.value&&(e(),t==null||t.focus())}}const Ie=M(q?location.hash:"");q&&window.addEventListener("hashchange",()=>{Ie.value=location.hash});function gt(s){const{page:e}=P(),t=M(!1),n=k(()=>s.value.collapsed!=null),o=k(()=>!!s.value.link),r=M(!1),d=()=>{r.value=O(e.value.relativePath,s.value.link)};U([e,s,Ie],d),G(d);const p=k(()=>r.value?!0:s.value.items?ie(e.value.relativePath,s.value.items):!1),_=k(()=>!!(s.value.items&&s.value.items.length));te(()=>{t.value=!!(n.value&&s.value.collapsed)}),Ve(()=>{(r.value||p.value)&&(t.value=!1)});function g(){n.value&&(t.value=!t.value)}return{collapsed:t,collapsible:n,isLink:o,isActiveLink:r,hasActiveLink:p,hasChildren:_,toggle:g}}function bt(){const{hasSidebar:s}=D(),e=re("(min-width: 960px)"),t=re("(min-width: 1280px)");return{isAsideEnabled:k(()=>!t.value&&!e.value?!1:s.value?t.value:e.value)}}const $t=71;function ve(s){return typeof s.outline=="object"&&!Array.isArray(s.outline)&&s.outline.label||s.outlineTitle||"On this page"}function pe(s){const e=[...document.querySelectorAll(".VPDoc :where(h1,h2,h3,h4,h5,h6)")].filter(t=>t.id&&t.hasChildNodes()).map(t=>{const n=Number(t.tagName[1]);return{title:kt(t),link:"#"+t.id,level:n}});return yt(e,s)}function kt(s){let e="";for(const t of s.childNodes)if(t.nodeType===1){if(t.classList.contains("VPBadge")||t.classList.contains("header-anchor"))continue;e+=t.textContent}else t.nodeType===3&&(e+=t.textContent);return e.trim()}function yt(s,e){if(e===!1)return[];const t=(typeof e=="object"&&!Array.isArray(e)?e.level:e)||2,[n,o]=typeof t=="number"?[t,t]:t==="deep"?[2,6]:t;s=s.filter(d=>d.level>=n&&d.level<=o);const r=[];e:for(let d=0;d=0;_--){const g=s[_];if(g.level{requestAnimationFrame(r),window.addEventListener("scroll",n)}),Ee(()=>{d(location.hash)}),de(()=>{window.removeEventListener("scroll",n)});function r(){if(!t.value)return;const p=[].slice.call(s.value.querySelectorAll(".outline-link")),_=[].slice.call(document.querySelectorAll(".content .header-anchor")).filter(S=>p.some(C=>C.hash===S.hash&&S.offsetParent!==null)),g=window.scrollY,V=window.innerHeight,y=document.body.offsetHeight,I=Math.abs(g+V-y)<1;if(_.length&&I){d(_[_.length-1].hash);return}for(let S=0;S<_.length;S++){const C=_[S],B=_[S+1],[w,K]=Vt(S,C,B);if(w){d(K);return}}}function d(p){o&&o.classList.remove("active"),p==null?o=null:o=s.value.querySelector(`a[href="${decodeURIComponent(p)}"]`);const _=o;_?(_.classList.add("active"),e.value.style.top=_.offsetTop+33+"px",e.value.style.opacity="1"):(e.value.style.top="33px",e.value.style.opacity="0")}}function ye(s){return s.parentElement.offsetTop-$t}function Vt(s,e,t){const n=window.scrollY;return s===0&&n===0?[!0,null]:n{const o=j("VPDocOutlineItem",!0);return a(),i("ul",{class:T(t.root?"root":"nested")},[(a(!0),i(N,null,A(t.headers,({children:r,link:d,title:p})=>(a(),i("li",null,[c("a",{class:"outline-link",href:d,onClick:e,title:p},L(p),9,wt),r!=null&&r.length?(a(),$(o,{key:0,headers:r},null,8,["headers"])):f("",!0)]))),256))],2)}}});const he=m(Lt,[["__scopeId","data-v-463da30f"]]),St=s=>(H("data-v-3a6c4994"),s=s(),z(),s),Mt={class:"content"},Nt={class:"outline-title",role:"heading","aria-level":"2"},It={"aria-labelledby":"doc-outline-aria-label"},Tt=St(()=>c("span",{class:"visually-hidden",id:"doc-outline-aria-label"}," Table of Contents for current page ",-1)),Ct=b({__name:"VPDocAsideOutline",setup(s){const{frontmatter:e,theme:t}=P(),n=_e([]);Y(()=>{n.value=pe(e.value.outline??t.value.outline)});const o=M(),r=M();return Pt(o,r),(d,p)=>(a(),i("div",{class:T(["VPDocAsideOutline",{"has-outline":n.value.length>0}]),ref_key:"container",ref:o,role:"navigation"},[c("div",Mt,[c("div",{class:"outline-marker",ref_key:"marker",ref:r},null,512),c("div",Nt,L(l(ve)(l(t))),1),c("nav",It,[Tt,h(he,{headers:n.value,root:!0},null,8,["headers"])])])],2))}});const Bt=m(Ct,[["__scopeId","data-v-3a6c4994"]]),At={class:"VPDocAsideCarbonAds"},xt=b({__name:"VPDocAsideCarbonAds",props:{carbonAds:{}},setup(s){const e=()=>null;return(t,n)=>(a(),i("div",At,[h(l(e),{"carbon-ads":t.carbonAds},null,8,["carbon-ads"])]))}}),Ht=s=>(H("data-v-cb998dce"),s=s(),z(),s),zt={class:"VPDocAside"},Dt=Ht(()=>c("div",{class:"spacer"},null,-1)),Et=b({__name:"VPDocAside",setup(s){const{theme:e}=P();return(t,n)=>(a(),i("div",zt,[u(t.$slots,"aside-top",{},void 0,!0),u(t.$slots,"aside-outline-before",{},void 0,!0),h(Bt),u(t.$slots,"aside-outline-after",{},void 0,!0),Dt,u(t.$slots,"aside-ads-before",{},void 0,!0),l(e).carbonAds?(a(),$(xt,{key:0,"carbon-ads":l(e).carbonAds},null,8,["carbon-ads"])):f("",!0),u(t.$slots,"aside-ads-after",{},void 0,!0),u(t.$slots,"aside-bottom",{},void 0,!0)]))}});const Ft=m(Et,[["__scopeId","data-v-cb998dce"]]);function Ot(){const{theme:s,page:e}=P();return k(()=>{const{text:t="Edit this page",pattern:n=""}=s.value.editLink||{};let o;return typeof n=="function"?o=n(e.value):o=n.replace(/:path/g,e.value.filePath),{url:o,text:t}})}function Gt(){const{page:s,theme:e,frontmatter:t}=P();return k(()=>{var _,g,V,y,I,S,C,B;const n=Ne(e.value.sidebar,s.value.relativePath),o=ft(n),r=o.findIndex(w=>O(s.value.relativePath,w.link)),d=((_=e.value.docFooter)==null?void 0:_.prev)===!1&&!t.value.prev||t.value.prev===!1,p=((g=e.value.docFooter)==null?void 0:g.next)===!1&&!t.value.next||t.value.next===!1;return{prev:d?void 0:{text:(typeof t.value.prev=="string"?t.value.prev:typeof t.value.prev=="object"?t.value.prev.text:void 0)??((V=o[r-1])==null?void 0:V.docFooterText)??((y=o[r-1])==null?void 0:y.text),link:(typeof t.value.prev=="object"?t.value.prev.link:void 0)??((I=o[r-1])==null?void 0:I.link)},next:p?void 0:{text:(typeof t.value.next=="string"?t.value.next:typeof t.value.next=="object"?t.value.next.text:void 0)??((S=o[r+1])==null?void 0:S.docFooterText)??((C=o[r+1])==null?void 0:C.text),link:(typeof t.value.next=="object"?t.value.next.link:void 0)??((B=o[r+1])==null?void 0:B.link)}}})}const Ut={},jt={xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 24 24"},Kt=c("path",{d:"M18,23H4c-1.7,0-3-1.3-3-3V6c0-1.7,1.3-3,3-3h7c0.6,0,1,0.4,1,1s-0.4,1-1,1H4C3.4,5,3,5.4,3,6v14c0,0.6,0.4,1,1,1h14c0.6,0,1-0.4,1-1v-7c0-0.6,0.4-1,1-1s1,0.4,1,1v7C21,21.7,19.7,23,18,23z"},null,-1),Rt=c("path",{d:"M8,17c-0.3,0-0.5-0.1-0.7-0.3C7,16.5,6.9,16.1,7,15.8l1-4c0-0.2,0.1-0.3,0.3-0.5l9.5-9.5c1.2-1.2,3.2-1.2,4.4,0c1.2,1.2,1.2,3.2,0,4.4l-9.5,9.5c-0.1,0.1-0.3,0.2-0.5,0.3l-4,1C8.2,17,8.1,17,8,17zM9.9,12.5l-0.5,2.1l2.1-0.5l9.3-9.3c0.4-0.4,0.4-1.1,0-1.6c-0.4-0.4-1.2-0.4-1.6,0l0,0L9.9,12.5z M18.5,2.5L18.5,2.5L18.5,2.5z"},null,-1),qt=[Kt,Rt];function Wt(s,e){return a(),i("svg",jt,qt)}const Yt=m(Ut,[["render",Wt]]),E=b({__name:"VPLink",props:{tag:{},href:{},noIcon:{type:Boolean},target:{},rel:{}},setup(s){const e=s,t=k(()=>e.tag??(e.href?"a":"span")),n=k(()=>e.href&&we.test(e.href));return(o,r)=>(a(),$(F(t.value),{class:T(["VPLink",{link:o.href,"vp-external-link-icon":n.value,"no-icon":o.noIcon}]),href:o.href?l(J)(o.href):void 0,target:o.target??(n.value?"_blank":void 0),rel:o.rel??(n.value?"noreferrer":void 0)},{default:v(()=>[u(o.$slots,"default")]),_:3},8,["class","href","target","rel"]))}}),Jt={class:"VPLastUpdated"},Xt=["datetime"],Zt=b({__name:"VPDocFooterLastUpdated",setup(s){const{theme:e,page:t,frontmatter:n,lang:o}=P(),r=k(()=>new Date(n.value.lastUpdated??t.value.lastUpdated)),d=k(()=>r.value.toISOString()),p=M("");return G(()=>{te(()=>{var _,g,V;p.value=new Intl.DateTimeFormat((g=(_=e.value.lastUpdated)==null?void 0:_.formatOptions)!=null&&g.forceLocale?o.value:void 0,((V=e.value.lastUpdated)==null?void 0:V.formatOptions)??{dateStyle:"short",timeStyle:"short"}).format(r.value)})}),(_,g)=>{var V;return a(),i("p",Jt,[x(L(((V=l(e).lastUpdated)==null?void 0:V.text)||l(e).lastUpdatedText||"Last updated")+": ",1),c("time",{datetime:d.value},L(p.value),9,Xt)])}}});const Qt=m(Zt,[["__scopeId","data-v-19a7ae4e"]]),es={key:0,class:"VPDocFooter"},ts={key:0,class:"edit-info"},ss={key:0,class:"edit-link"},ns={key:1,class:"last-updated"},os={key:1,class:"prev-next"},as={class:"pager"},rs=["href"],ls=["innerHTML"],is=["innerHTML"],cs={class:"pager"},us=["href"],ds=["innerHTML"],_s=["innerHTML"],vs=b({__name:"VPDocFooter",setup(s){const{theme:e,page:t,frontmatter:n}=P(),o=Ot(),r=Gt(),d=k(()=>e.value.editLink&&n.value.editLink!==!1),p=k(()=>t.value.lastUpdated&&n.value.lastUpdated!==!1),_=k(()=>d.value||p.value||r.value.prev||r.value.next);return(g,V)=>{var y,I,S,C,B,w;return _.value?(a(),i("footer",es,[u(g.$slots,"doc-footer-before",{},void 0,!0),d.value||p.value?(a(),i("div",ts,[d.value?(a(),i("div",ss,[h(E,{class:"edit-link-button",href:l(o).url,"no-icon":!0},{default:v(()=>[h(Yt,{class:"edit-link-icon","aria-label":"edit icon"}),x(" "+L(l(o).text),1)]),_:1},8,["href"])])):f("",!0),p.value?(a(),i("div",ns,[h(Qt)])):f("",!0)])):f("",!0),(y=l(r).prev)!=null&&y.link||(I=l(r).next)!=null&&I.link?(a(),i("nav",os,[c("div",as,[(S=l(r).prev)!=null&&S.link?(a(),i("a",{key:0,class:"pager-link prev",href:l(J)(l(r).prev.link)},[c("span",{class:"desc",innerHTML:((C=l(e).docFooter)==null?void 0:C.prev)||"Previous page"},null,8,ls),c("span",{class:"title",innerHTML:l(r).prev.text},null,8,is)],8,rs)):f("",!0)]),c("div",cs,[(B=l(r).next)!=null&&B.link?(a(),i("a",{key:0,class:"pager-link next",href:l(J)(l(r).next.link)},[c("span",{class:"desc",innerHTML:((w=l(e).docFooter)==null?void 0:w.next)||"Next page"},null,8,ds),c("span",{class:"title",innerHTML:l(r).next.text},null,8,_s)],8,us)):f("",!0)])])):f("",!0)])):f("",!0)}}});const ps=m(vs,[["__scopeId","data-v-a2d931e4"]]),hs={},fs={xmlns:"http://www.w3.org/2000/svg","aria-hidden":"true",focusable:"false",viewBox:"0 0 24 24"},ms=c("path",{d:"M9,19c-0.3,0-0.5-0.1-0.7-0.3c-0.4-0.4-0.4-1,0-1.4l5.3-5.3L8.3,6.7c-0.4-0.4-0.4-1,0-1.4s1-0.4,1.4,0l6,6c0.4,0.4,0.4,1,0,1.4l-6,6C9.5,18.9,9.3,19,9,19z"},null,-1),gs=[ms];function bs(s,e){return a(),i("svg",fs,gs)}const fe=m(hs,[["render",bs]]),$s={key:0,class:"VPDocOutlineDropdown"},ks={key:0,class:"items"},ys=b({__name:"VPDocOutlineDropdown",setup(s){const{frontmatter:e,theme:t}=P(),n=M(!1);Y(()=>{n.value=!1});const o=_e([]);return Y(()=>{o.value=pe(e.value.outline??t.value.outline)}),(r,d)=>o.value.length>0?(a(),i("div",$s,[c("button",{onClick:d[0]||(d[0]=p=>n.value=!n.value),class:T({open:n.value})},[x(L(l(ve)(l(t)))+" ",1),h(fe,{class:"icon"})],2),n.value?(a(),i("div",ks,[h(he,{headers:o.value},null,8,["headers"])])):f("",!0)])):f("",!0)}});const Ps=m(ys,[["__scopeId","data-v-95bb0785"]]),Vs=s=>(H("data-v-a3c25e27"),s=s(),z(),s),ws={class:"container"},Ls=Vs(()=>c("div",{class:"aside-curtain"},null,-1)),Ss={class:"aside-container"},Ms={class:"aside-content"},Ns={class:"content"},Is={class:"content-container"},Ts={class:"main"},Cs=b({__name:"VPDoc",setup(s){const{theme:e}=P(),t=se(),{hasSidebar:n,hasAside:o,leftAside:r}=D(),d=k(()=>t.path.replace(/[./]+/g,"_").replace(/_html$/,""));return(p,_)=>{const g=j("Content");return a(),i("div",{class:T(["VPDoc",{"has-sidebar":l(n),"has-aside":l(o)}])},[u(p.$slots,"doc-top",{},void 0,!0),c("div",ws,[l(o)?(a(),i("div",{key:0,class:T(["aside",{"left-aside":l(r)}])},[Ls,c("div",Ss,[c("div",Ms,[h(Ft,null,{"aside-top":v(()=>[u(p.$slots,"aside-top",{},void 0,!0)]),"aside-bottom":v(()=>[u(p.$slots,"aside-bottom",{},void 0,!0)]),"aside-outline-before":v(()=>[u(p.$slots,"aside-outline-before",{},void 0,!0)]),"aside-outline-after":v(()=>[u(p.$slots,"aside-outline-after",{},void 0,!0)]),"aside-ads-before":v(()=>[u(p.$slots,"aside-ads-before",{},void 0,!0)]),"aside-ads-after":v(()=>[u(p.$slots,"aside-ads-after",{},void 0,!0)]),_:3})])])],2)):f("",!0),c("div",Ns,[c("div",Is,[u(p.$slots,"doc-before",{},void 0,!0),h(Ps),c("main",Ts,[h(g,{class:T(["vp-doc",[d.value,l(e).externalLinkIcon&&"external-link-icon-enabled"]])},null,8,["class"])]),h(ps,null,{"doc-footer-before":v(()=>[u(p.$slots,"doc-footer-before",{},void 0,!0)]),_:3}),u(p.$slots,"doc-after",{},void 0,!0)])])]),u(p.$slots,"doc-bottom",{},void 0,!0)],2)}}});const Bs=m(Cs,[["__scopeId","data-v-a3c25e27"]]),As=b({__name:"VPButton",props:{tag:{},size:{default:"medium"},theme:{default:"brand"},text:{},href:{}},setup(s){const e=s,t=k(()=>e.href&&we.test(e.href)),n=k(()=>e.tag||e.href?"a":"button");return(o,r)=>(a(),$(F(n.value),{class:T(["VPButton",[o.size,o.theme]]),href:o.href?l(J)(o.href):void 0,target:t.value?"_blank":void 0,rel:t.value?"noreferrer":void 0},{default:v(()=>[x(L(o.text),1)]),_:1},8,["class","href","target","rel"]))}});const xs=m(As,[["__scopeId","data-v-1e76fe75"]]),Hs=["src","alt"],zs=b({inheritAttrs:!1,__name:"VPImage",props:{image:{},alt:{}},setup(s){return(e,t)=>{const n=j("VPImage",!0);return e.image?(a(),i(N,{key:0},[typeof e.image=="string"||"src"in e.image?(a(),i("img",Z({key:0,class:"VPImage"},typeof e.image=="string"?e.$attrs:{...e.image,...e.$attrs},{src:l(ue)(typeof e.image=="string"?e.image:e.image.src),alt:e.alt??(typeof e.image=="string"?"":e.image.alt||"")}),null,16,Hs)):(a(),i(N,{key:1},[h(n,Z({class:"dark",image:e.image.dark,alt:e.image.alt},e.$attrs),null,16,["image","alt"]),h(n,Z({class:"light",image:e.image.light,alt:e.image.alt},e.$attrs),null,16,["image","alt"])],64))],64)):f("",!0)}}});const ee=m(zs,[["__scopeId","data-v-ab19afbb"]]),Ds=s=>(H("data-v-5a3e9999"),s=s(),z(),s),Es={class:"container"},Fs={class:"main"},Os={key:0,class:"name"},Gs=["innerHTML"],Us=["innerHTML"],js=["innerHTML"],Ks={key:0,class:"actions"},Rs={key:0,class:"image"},qs={class:"image-container"},Ws=Ds(()=>c("div",{class:"image-bg"},null,-1)),Ys=b({__name:"VPHero",props:{name:{},text:{},tagline:{},image:{},actions:{}},setup(s){const e=ne("hero-image-slot-exists");return(t,n)=>(a(),i("div",{class:T(["VPHero",{"has-image":t.image||l(e)}])},[c("div",Es,[c("div",Fs,[u(t.$slots,"home-hero-info",{},()=>[t.name?(a(),i("h1",Os,[c("span",{innerHTML:t.name,class:"clip"},null,8,Gs)])):f("",!0),t.text?(a(),i("p",{key:1,innerHTML:t.text,class:"text"},null,8,Us)):f("",!0),t.tagline?(a(),i("p",{key:2,innerHTML:t.tagline,class:"tagline"},null,8,js)):f("",!0)],!0),t.actions?(a(),i("div",Ks,[(a(!0),i(N,null,A(t.actions,o=>(a(),i("div",{key:o.link,class:"action"},[h(xs,{tag:"a",size:"medium",theme:o.theme,text:o.text,href:o.link},null,8,["theme","text","href"])]))),128))])):f("",!0)]),t.image||l(e)?(a(),i("div",Rs,[c("div",qs,[Ws,u(t.$slots,"home-hero-image",{},()=>[t.image?(a(),$(ee,{key:0,class:"image-src",image:t.image},null,8,["image"])):f("",!0)],!0)])])):f("",!0)])],2))}});const Js=m(Ys,[["__scopeId","data-v-5a3e9999"]]),Xs=b({__name:"VPHomeHero",setup(s){const{frontmatter:e}=P();return(t,n)=>l(e).hero?(a(),$(Js,{key:0,class:"VPHomeHero",name:l(e).hero.name,text:l(e).hero.text,tagline:l(e).hero.tagline,image:l(e).hero.image,actions:l(e).hero.actions},{"home-hero-info":v(()=>[u(t.$slots,"home-hero-info")]),"home-hero-image":v(()=>[u(t.$slots,"home-hero-image")]),_:3},8,["name","text","tagline","image","actions"])):f("",!0)}}),Zs={},Qs={xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 24 24"},en=c("path",{d:"M19.9,12.4c0.1-0.2,0.1-0.5,0-0.8c-0.1-0.1-0.1-0.2-0.2-0.3l-7-7c-0.4-0.4-1-0.4-1.4,0s-0.4,1,0,1.4l5.3,5.3H5c-0.6,0-1,0.4-1,1s0.4,1,1,1h11.6l-5.3,5.3c-0.4,0.4-0.4,1,0,1.4c0.2,0.2,0.5,0.3,0.7,0.3s0.5-0.1,0.7-0.3l7-7C19.8,12.6,19.9,12.5,19.9,12.4z"},null,-1),tn=[en];function sn(s,e){return a(),i("svg",Qs,tn)}const nn=m(Zs,[["render",sn]]),on={class:"box"},an={key:0,class:"icon"},rn=["innerHTML"],ln=["innerHTML"],cn=["innerHTML"],un={key:4,class:"link-text"},dn={class:"link-text-value"},_n=b({__name:"VPFeature",props:{icon:{},title:{},details:{},link:{},linkText:{},rel:{},target:{}},setup(s){return(e,t)=>(a(),$(E,{class:"VPFeature",href:e.link,rel:e.rel,target:e.target,"no-icon":!0,tag:e.link?"a":"div"},{default:v(()=>[c("article",on,[typeof e.icon=="object"&&e.icon.wrap?(a(),i("div",an,[h(ee,{image:e.icon,alt:e.icon.alt,height:e.icon.height||48,width:e.icon.width||48},null,8,["image","alt","height","width"])])):typeof e.icon=="object"?(a(),$(ee,{key:1,image:e.icon,alt:e.icon.alt,height:e.icon.height||48,width:e.icon.width||48},null,8,["image","alt","height","width"])):e.icon?(a(),i("div",{key:2,class:"icon",innerHTML:e.icon},null,8,rn)):f("",!0),c("h2",{class:"title",innerHTML:e.title},null,8,ln),e.details?(a(),i("p",{key:3,class:"details",innerHTML:e.details},null,8,cn)):f("",!0),e.linkText?(a(),i("div",un,[c("p",dn,[x(L(e.linkText)+" ",1),h(nn,{class:"link-text-icon"})])])):f("",!0)])]),_:1},8,["href","rel","target","tag"]))}});const vn=m(_n,[["__scopeId","data-v-ee984185"]]),pn={key:0,class:"VPFeatures"},hn={class:"container"},fn={class:"items"},mn=b({__name:"VPFeatures",props:{features:{}},setup(s){const e=s,t=k(()=>{const n=e.features.length;if(n){if(n===2)return"grid-2";if(n===3)return"grid-3";if(n%3===0)return"grid-6";if(n>3)return"grid-4"}else return});return(n,o)=>n.features?(a(),i("div",pn,[c("div",hn,[c("div",fn,[(a(!0),i(N,null,A(n.features,r=>(a(),i("div",{key:r.title,class:T(["item",[t.value]])},[h(vn,{icon:r.icon,title:r.title,details:r.details,link:r.link,"link-text":r.linkText,rel:r.rel,target:r.target},null,8,["icon","title","details","link","link-text","rel","target"])],2))),128))])])])):f("",!0)}});const gn=m(mn,[["__scopeId","data-v-b1eea84a"]]),bn=b({__name:"VPHomeFeatures",setup(s){const{frontmatter:e}=P();return(t,n)=>l(e).features?(a(),$(gn,{key:0,class:"VPHomeFeatures",features:l(e).features},null,8,["features"])):f("",!0)}}),$n={class:"VPHome"},kn=b({__name:"VPHome",setup(s){return(e,t)=>{const n=j("Content");return a(),i("div",$n,[u(e.$slots,"home-hero-before",{},void 0,!0),h(Xs,null,{"home-hero-info":v(()=>[u(e.$slots,"home-hero-info",{},void 0,!0)]),"home-hero-image":v(()=>[u(e.$slots,"home-hero-image",{},void 0,!0)]),_:3}),u(e.$slots,"home-hero-after",{},void 0,!0),u(e.$slots,"home-features-before",{},void 0,!0),h(bn),u(e.$slots,"home-features-after",{},void 0,!0),h(n)])}}});const yn=m(kn,[["__scopeId","data-v-20eabd3a"]]),Pn={},Vn={class:"VPPage"};function wn(s,e){const t=j("Content");return a(),i("div",Vn,[u(s.$slots,"page-top"),h(t),u(s.$slots,"page-bottom")])}const Ln=m(Pn,[["render",wn]]),Sn=b({__name:"VPContent",setup(s){const{page:e,frontmatter:t}=P(),{hasSidebar:n}=D();return(o,r)=>(a(),i("div",{class:T(["VPContent",{"has-sidebar":l(n),"is-home":l(t).layout==="home"}]),id:"VPContent"},[l(e).isNotFound?u(o.$slots,"not-found",{key:0},()=>[h(pt)],!0):l(t).layout==="page"?(a(),$(Ln,{key:1},{"page-top":v(()=>[u(o.$slots,"page-top",{},void 0,!0)]),"page-bottom":v(()=>[u(o.$slots,"page-bottom",{},void 0,!0)]),_:3})):l(t).layout==="home"?(a(),$(yn,{key:2},{"home-hero-before":v(()=>[u(o.$slots,"home-hero-before",{},void 0,!0)]),"home-hero-info":v(()=>[u(o.$slots,"home-hero-info",{},void 0,!0)]),"home-hero-image":v(()=>[u(o.$slots,"home-hero-image",{},void 0,!0)]),"home-hero-after":v(()=>[u(o.$slots,"home-hero-after",{},void 0,!0)]),"home-features-before":v(()=>[u(o.$slots,"home-features-before",{},void 0,!0)]),"home-features-after":v(()=>[u(o.$slots,"home-features-after",{},void 0,!0)]),_:3})):l(t).layout&&l(t).layout!=="doc"?(a(),$(F(l(t).layout),{key:3})):(a(),$(Bs,{key:4},{"doc-top":v(()=>[u(o.$slots,"doc-top",{},void 0,!0)]),"doc-bottom":v(()=>[u(o.$slots,"doc-bottom",{},void 0,!0)]),"doc-footer-before":v(()=>[u(o.$slots,"doc-footer-before",{},void 0,!0)]),"doc-before":v(()=>[u(o.$slots,"doc-before",{},void 0,!0)]),"doc-after":v(()=>[u(o.$slots,"doc-after",{},void 0,!0)]),"aside-top":v(()=>[u(o.$slots,"aside-top",{},void 0,!0)]),"aside-outline-before":v(()=>[u(o.$slots,"aside-outline-before",{},void 0,!0)]),"aside-outline-after":v(()=>[u(o.$slots,"aside-outline-after",{},void 0,!0)]),"aside-ads-before":v(()=>[u(o.$slots,"aside-ads-before",{},void 0,!0)]),"aside-ads-after":v(()=>[u(o.$slots,"aside-ads-after",{},void 0,!0)]),"aside-bottom":v(()=>[u(o.$slots,"aside-bottom",{},void 0,!0)]),_:3}))],2))}});const Mn=m(Sn,[["__scopeId","data-v-3cf691b6"]]),Nn={class:"container"},In=["innerHTML"],Tn=["innerHTML"],Cn=b({__name:"VPFooter",setup(s){const{theme:e,frontmatter:t}=P(),{hasSidebar:n}=D();return(o,r)=>l(e).footer&&l(t).footer!==!1?(a(),i("footer",{key:0,class:T(["VPFooter",{"has-sidebar":l(n)}])},[c("div",Nn,[l(e).footer.message?(a(),i("p",{key:0,class:"message",innerHTML:l(e).footer.message},null,8,In)):f("",!0),l(e).footer.copyright?(a(),i("p",{key:1,class:"copyright",innerHTML:l(e).footer.copyright},null,8,Tn)):f("",!0)])],2)):f("",!0)}});const Bn=m(Cn,[["__scopeId","data-v-e4279f1c"]]),An={class:"header"},xn={class:"outline"},Hn=b({__name:"VPLocalNavOutlineDropdown",props:{headers:{},navHeight:{}},setup(s){const e=s,{theme:t}=P(),n=M(!1),o=M(0),r=M();Y(()=>{n.value=!1});function d(){n.value=!n.value,o.value=window.innerHeight+Math.min(window.scrollY-e.navHeight,0)}function p(g){g.target.classList.contains("outline-link")&&(r.value&&(r.value.style.transition="none"),Oe(()=>{n.value=!1}))}function _(){n.value=!1,window.scrollTo({top:0,left:0,behavior:"smooth"})}return(g,V)=>(a(),i("div",{class:"VPLocalNavOutlineDropdown",style:Fe({"--vp-vh":o.value+"px"})},[g.headers.length>0?(a(),i("button",{key:0,onClick:d,class:T({open:n.value})},[x(L(l(ve)(l(t)))+" ",1),h(fe,{class:"icon"})],2)):(a(),i("button",{key:1,onClick:_},L(l(t).returnToTopLabel||"Return to top"),1)),h(ce,{name:"flyout"},{default:v(()=>[n.value?(a(),i("div",{key:0,ref_key:"items",ref:r,class:"items",onClick:p},[c("div",An,[c("a",{class:"top-link",href:"#",onClick:_},L(l(t).returnToTopLabel||"Return to top"),1)]),c("div",xn,[h(he,{headers:g.headers},null,8,["headers"])])],512)):f("",!0)]),_:1})],4))}});const zn=m(Hn,[["__scopeId","data-v-24251f6f"]]),Dn={},En={xmlns:"http://www.w3.org/2000/svg","aria-hidden":"true",focusable:"false",viewBox:"0 0 24 24"},Fn=c("path",{d:"M17,11H3c-0.6,0-1-0.4-1-1s0.4-1,1-1h14c0.6,0,1,0.4,1,1S17.6,11,17,11z"},null,-1),On=c("path",{d:"M21,7H3C2.4,7,2,6.6,2,6s0.4-1,1-1h18c0.6,0,1,0.4,1,1S21.6,7,21,7z"},null,-1),Gn=c("path",{d:"M21,15H3c-0.6,0-1-0.4-1-1s0.4-1,1-1h18c0.6,0,1,0.4,1,1S21.6,15,21,15z"},null,-1),Un=c("path",{d:"M17,19H3c-0.6,0-1-0.4-1-1s0.4-1,1-1h14c0.6,0,1,0.4,1,1S17.6,19,17,19z"},null,-1),jn=[Fn,On,Gn,Un];function Kn(s,e){return a(),i("svg",En,jn)}const Rn=m(Dn,[["render",Kn]]),qn=["aria-expanded"],Wn={class:"menu-text"},Yn=b({__name:"VPLocalNav",props:{open:{type:Boolean}},emits:["open-menu"],setup(s){const{theme:e,frontmatter:t}=P(),{hasSidebar:n}=D(),{y:o}=Le(),r=_e([]),d=M(0);G(()=>{d.value=parseInt(getComputedStyle(document.documentElement).getPropertyValue("--vp-nav-height"))}),Y(()=>{r.value=pe(t.value.outline??e.value.outline)});const p=k(()=>r.value.length===0&&!n.value),_=k(()=>({VPLocalNav:!0,fixed:p.value,"reached-top":o.value>=d.value}));return(g,V)=>l(t).layout!=="home"&&(!p.value||l(o)>=d.value)?(a(),i("div",{key:0,class:T(_.value)},[l(n)?(a(),i("button",{key:0,class:"menu","aria-expanded":g.open,"aria-controls":"VPSidebarNav",onClick:V[0]||(V[0]=y=>g.$emit("open-menu"))},[h(Rn,{class:"menu-icon"}),c("span",Wn,L(l(e).sidebarMenuLabel||"Menu"),1)],8,qn)):f("",!0),h(zn,{headers:r.value,navHeight:d.value},null,8,["headers","navHeight"])],2)):f("",!0)}});const Jn=m(Yn,[["__scopeId","data-v-9e669cc1"]]);function Xn(){const s=M(!1);function e(){s.value=!0,window.addEventListener("resize",o)}function t(){s.value=!1,window.removeEventListener("resize",o)}function n(){s.value?t():e()}function o(){window.outerWidth>=768&&t()}const r=se();return U(()=>r.path,t),{isScreenOpen:s,openScreen:e,closeScreen:t,toggleScreen:n}}const Zn={},Qn={class:"VPSwitch",type:"button",role:"switch"},eo={class:"check"},to={key:0,class:"icon"};function so(s,e){return a(),i("button",Qn,[c("span",eo,[s.$slots.default?(a(),i("span",to,[u(s.$slots,"default",{},void 0,!0)])):f("",!0)])])}const no=m(Zn,[["render",so],["__scopeId","data-v-1c29e291"]]),oo={},ao={xmlns:"http://www.w3.org/2000/svg","aria-hidden":"true",focusable:"false",viewBox:"0 0 24 24"},ro=c("path",{d:"M12.1,22c-0.3,0-0.6,0-0.9,0c-5.5-0.5-9.5-5.4-9-10.9c0.4-4.8,4.2-8.6,9-9c0.4,0,0.8,0.2,1,0.5c0.2,0.3,0.2,0.8-0.1,1.1c-2,2.7-1.4,6.4,1.3,8.4c2.1,1.6,5,1.6,7.1,0c0.3-0.2,0.7-0.3,1.1-0.1c0.3,0.2,0.5,0.6,0.5,1c-0.2,2.7-1.5,5.1-3.6,6.8C16.6,21.2,14.4,22,12.1,22zM9.3,4.4c-2.9,1-5,3.6-5.2,6.8c-0.4,4.4,2.8,8.3,7.2,8.7c2.1,0.2,4.2-0.4,5.8-1.8c1.1-0.9,1.9-2.1,2.4-3.4c-2.5,0.9-5.3,0.5-7.5-1.1C9.2,11.4,8.1,7.7,9.3,4.4z"},null,-1),lo=[ro];function io(s,e){return a(),i("svg",ao,lo)}const co=m(oo,[["render",io]]),uo={},_o={xmlns:"http://www.w3.org/2000/svg","aria-hidden":"true",focusable:"false",viewBox:"0 0 24 24"},vo=Ge('',9),po=[vo];function ho(s,e){return a(),i("svg",_o,po)}const fo=m(uo,[["render",ho]]),mo=b({__name:"VPSwitchAppearance",setup(s){const{isDark:e}=P(),t=ne("toggle-appearance",()=>{e.value=!e.value});return(n,o)=>(a(),$(no,{title:"toggle dark mode",class:"VPSwitchAppearance","aria-checked":l(e),onClick:l(t)},{default:v(()=>[h(fo,{class:"sun"}),h(co,{class:"moon"})]),_:1},8,["aria-checked","onClick"]))}});const me=m(mo,[["__scopeId","data-v-3329432d"]]),go={key:0,class:"VPNavBarAppearance"},bo=b({__name:"VPNavBarAppearance",setup(s){const{site:e}=P();return(t,n)=>l(e).appearance&&l(e).appearance!=="force-dark"?(a(),i("div",go,[h(me)])):f("",!0)}});const $o=m(bo,[["__scopeId","data-v-283b26e9"]]),ge=M();let Te=!1,ae=0;function ko(s){const e=M(!1);if(q){!Te&&yo(),ae++;const t=U(ge,n=>{var o,r,d;n===s.el.value||(o=s.el.value)!=null&&o.contains(n)?(e.value=!0,(r=s.onFocus)==null||r.call(s)):(e.value=!1,(d=s.onBlur)==null||d.call(s))});de(()=>{t(),ae--,ae||Po()})}return Ue(e)}function yo(){document.addEventListener("focusin",Ce),Te=!0,ge.value=document.activeElement}function Po(){document.removeEventListener("focusin",Ce)}function Ce(){ge.value=document.activeElement}const Vo={},wo={xmlns:"http://www.w3.org/2000/svg","aria-hidden":"true",focusable:"false",viewBox:"0 0 24 24"},Lo=c("path",{d:"M12,16c-0.3,0-0.5-0.1-0.7-0.3l-6-6c-0.4-0.4-0.4-1,0-1.4s1-0.4,1.4,0l5.3,5.3l5.3-5.3c0.4-0.4,1-0.4,1.4,0s0.4,1,0,1.4l-6,6C12.5,15.9,12.3,16,12,16z"},null,-1),So=[Lo];function Mo(s,e){return a(),i("svg",wo,So)}const Be=m(Vo,[["render",Mo]]),No={},Io={xmlns:"http://www.w3.org/2000/svg","aria-hidden":"true",focusable:"false",viewBox:"0 0 24 24"},To=c("circle",{cx:"12",cy:"12",r:"2"},null,-1),Co=c("circle",{cx:"19",cy:"12",r:"2"},null,-1),Bo=c("circle",{cx:"5",cy:"12",r:"2"},null,-1),Ao=[To,Co,Bo];function xo(s,e){return a(),i("svg",Io,Ao)}const Ho=m(No,[["render",xo]]),zo={class:"VPMenuLink"},Do=b({__name:"VPMenuLink",props:{item:{}},setup(s){const{page:e}=P();return(t,n)=>(a(),i("div",zo,[h(E,{class:T({active:l(O)(l(e).relativePath,t.item.activeMatch||t.item.link,!!t.item.activeMatch)}),href:t.item.link,target:t.item.target,rel:t.item.rel},{default:v(()=>[x(L(t.item.text),1)]),_:1},8,["class","href","target","rel"])]))}});const oe=m(Do,[["__scopeId","data-v-f51f088d"]]),Eo={class:"VPMenuGroup"},Fo={key:0,class:"title"},Oo=b({__name:"VPMenuGroup",props:{text:{},items:{}},setup(s){return(e,t)=>(a(),i("div",Eo,[e.text?(a(),i("p",Fo,L(e.text),1)):f("",!0),(a(!0),i(N,null,A(e.items,n=>(a(),i(N,null,["link"in n?(a(),$(oe,{key:0,item:n},null,8,["item"])):f("",!0)],64))),256))]))}});const Go=m(Oo,[["__scopeId","data-v-a6b0397c"]]),Uo={class:"VPMenu"},jo={key:0,class:"items"},Ko=b({__name:"VPMenu",props:{items:{}},setup(s){return(e,t)=>(a(),i("div",Uo,[e.items?(a(),i("div",jo,[(a(!0),i(N,null,A(e.items,n=>(a(),i(N,{key:n.text},["link"in n?(a(),$(oe,{key:0,item:n},null,8,["item"])):(a(),$(Go,{key:1,text:n.text,items:n.items},null,8,["text","items"]))],64))),128))])):f("",!0),u(e.$slots,"default",{},void 0,!0)]))}});const Ro=m(Ko,[["__scopeId","data-v-e42ed9b3"]]),qo=["aria-expanded","aria-label"],Wo={key:0,class:"text"},Yo=["innerHTML"],Jo={class:"menu"},Xo=b({__name:"VPFlyout",props:{icon:{},button:{},label:{},items:{}},setup(s){const e=M(!1),t=M();ko({el:t,onBlur:n});function n(){e.value=!1}return(o,r)=>(a(),i("div",{class:"VPFlyout",ref_key:"el",ref:t,onMouseenter:r[1]||(r[1]=d=>e.value=!0),onMouseleave:r[2]||(r[2]=d=>e.value=!1)},[c("button",{type:"button",class:"button","aria-haspopup":"true","aria-expanded":e.value,"aria-label":o.label,onClick:r[0]||(r[0]=d=>e.value=!e.value)},[o.button||o.icon?(a(),i("span",Wo,[o.icon?(a(),$(F(o.icon),{key:0,class:"option-icon"})):f("",!0),o.button?(a(),i("span",{key:1,innerHTML:o.button},null,8,Yo)):f("",!0),h(Be,{class:"text-icon"})])):(a(),$(Ho,{key:1,class:"icon"}))],8,qo),c("div",Jo,[h(Ro,{items:o.items},{default:v(()=>[u(o.$slots,"default",{},void 0,!0)]),_:3},8,["items"])])],544))}});const be=m(Xo,[["__scopeId","data-v-aa8de344"]]),Zo={discord:'Discord',facebook:'Facebook',github:'GitHub',instagram:'Instagram',linkedin:'LinkedIn',mastodon:'Mastodon',slack:'Slack',twitter:'Twitter',x:'X',youtube:'YouTube'},Qo=["href","aria-label","innerHTML"],ea=b({__name:"VPSocialLink",props:{icon:{},link:{},ariaLabel:{}},setup(s){const e=s,t=k(()=>typeof e.icon=="object"?e.icon.svg:Zo[e.icon]);return(n,o)=>(a(),i("a",{class:"VPSocialLink no-icon",href:n.link,"aria-label":n.ariaLabel??(typeof n.icon=="string"?n.icon:""),target:"_blank",rel:"noopener",innerHTML:t.value},null,8,Qo))}});const ta=m(ea,[["__scopeId","data-v-16cf740a"]]),sa={class:"VPSocialLinks"},na=b({__name:"VPSocialLinks",props:{links:{}},setup(s){return(e,t)=>(a(),i("div",sa,[(a(!0),i(N,null,A(e.links,({link:n,icon:o,ariaLabel:r})=>(a(),$(ta,{key:n,icon:o,link:n,ariaLabel:r},null,8,["icon","link","ariaLabel"]))),128))]))}});const $e=m(na,[["__scopeId","data-v-e71e869c"]]),oa={key:0,class:"group translations"},aa={class:"trans-title"},ra={key:1,class:"group"},la={class:"item appearance"},ia={class:"label"},ca={class:"appearance-action"},ua={key:2,class:"group"},da={class:"item social-links"},_a=b({__name:"VPNavBarExtra",setup(s){const{site:e,theme:t}=P(),{localeLinks:n,currentLang:o}=X({correspondingLink:!0}),r=k(()=>n.value.length&&o.value.label||e.value.appearance||t.value.socialLinks);return(d,p)=>r.value?(a(),$(be,{key:0,class:"VPNavBarExtra",label:"extra navigation"},{default:v(()=>[l(n).length&&l(o).label?(a(),i("div",oa,[c("p",aa,L(l(o).label),1),(a(!0),i(N,null,A(l(n),_=>(a(),$(oe,{key:_.link,item:_},null,8,["item"]))),128))])):f("",!0),l(e).appearance?(a(),i("div",ra,[c("div",la,[c("p",ia,L(l(t).darkModeSwitchLabel||"Appearance"),1),c("div",ca,[h(me)])])])):f("",!0),l(t).socialLinks?(a(),i("div",ua,[c("div",da,[h($e,{class:"social-links-list",links:l(t).socialLinks},null,8,["links"])])])):f("",!0)]),_:1})):f("",!0)}});const va=m(_a,[["__scopeId","data-v-c8c2ae4b"]]),pa=s=>(H("data-v-6bee1efd"),s=s(),z(),s),ha=["aria-expanded"],fa=pa(()=>c("span",{class:"container"},[c("span",{class:"top"}),c("span",{class:"middle"}),c("span",{class:"bottom"})],-1)),ma=[fa],ga=b({__name:"VPNavBarHamburger",props:{active:{type:Boolean}},emits:["click"],setup(s){return(e,t)=>(a(),i("button",{type:"button",class:T(["VPNavBarHamburger",{active:e.active}]),"aria-label":"mobile navigation","aria-expanded":e.active,"aria-controls":"VPNavScreen",onClick:t[0]||(t[0]=n=>e.$emit("click"))},ma,10,ha))}});const ba=m(ga,[["__scopeId","data-v-6bee1efd"]]),$a=["innerHTML"],ka=b({__name:"VPNavBarMenuLink",props:{item:{}},setup(s){const{page:e}=P();return(t,n)=>(a(),$(E,{class:T({VPNavBarMenuLink:!0,active:l(O)(l(e).relativePath,t.item.activeMatch||t.item.link,!!t.item.activeMatch)}),href:t.item.link,target:t.item.target,rel:t.item.rel,tabindex:"0"},{default:v(()=>[c("span",{innerHTML:t.item.text},null,8,$a)]),_:1},8,["class","href","target","rel"]))}});const ya=m(ka,[["__scopeId","data-v-cb318fec"]]),Pa=b({__name:"VPNavBarMenuGroup",props:{item:{}},setup(s){const e=s,{page:t}=P(),n=r=>"link"in r?O(t.value.relativePath,r.link,!!e.item.activeMatch):r.items.some(n),o=k(()=>n(e.item));return(r,d)=>(a(),$(be,{class:T({VPNavBarMenuGroup:!0,active:l(O)(l(t).relativePath,r.item.activeMatch,!!r.item.activeMatch)||o.value}),button:r.item.text,items:r.item.items},null,8,["class","button","items"]))}}),Va=s=>(H("data-v-f732b5d0"),s=s(),z(),s),wa={key:0,"aria-labelledby":"main-nav-aria-label",class:"VPNavBarMenu"},La=Va(()=>c("span",{id:"main-nav-aria-label",class:"visually-hidden"},"Main Navigation",-1)),Sa=b({__name:"VPNavBarMenu",setup(s){const{theme:e}=P();return(t,n)=>l(e).nav?(a(),i("nav",wa,[La,(a(!0),i(N,null,A(l(e).nav,o=>(a(),i(N,{key:o.text},["link"in o?(a(),$(ya,{key:0,item:o},null,8,["item"])):(a(),$(Pa,{key:1,item:o},null,8,["item"]))],64))),128))])):f("",!0)}});const Ma=m(Sa,[["__scopeId","data-v-f732b5d0"]]);function Na(s,e){const{localeIndex:t}=P();function n(o){var S,C;const r=o.split("."),d=s&&typeof s=="object",p=d&&((C=(S=s.locales)==null?void 0:S[t.value])==null?void 0:C.translations)||null,_=d&&s.translations||null;let g=p,V=_,y=e;const I=r.pop();for(const B of r){let w=null;const K=y==null?void 0:y[B];K&&(w=y=K);const W=V==null?void 0:V[B];W&&(w=V=W);const R=g==null?void 0:g[B];R&&(w=g=R),K||(y=w),W||(V=w),R||(g=w)}return(g==null?void 0:g[I])??(V==null?void 0:V[I])??(y==null?void 0:y[I])??""}return n}const Ia=["aria-label"],Ta={class:"DocSearch-Button-Container"},Ca=c("svg",{class:"DocSearch-Search-Icon",width:"20",height:"20",viewBox:"0 0 20 20","aria-label":"search icon"},[c("path",{d:"M14.386 14.386l4.0877 4.0877-4.0877-4.0877c-2.9418 2.9419-7.7115 2.9419-10.6533 0-2.9419-2.9418-2.9419-7.7115 0-10.6533 2.9418-2.9419 7.7115-2.9419 10.6533 0 2.9419 2.9418 2.9419 7.7115 0 10.6533z",stroke:"currentColor",fill:"none","fill-rule":"evenodd","stroke-linecap":"round","stroke-linejoin":"round"})],-1),Ba={class:"DocSearch-Button-Placeholder"},Aa=c("span",{class:"DocSearch-Button-Keys"},[c("kbd",{class:"DocSearch-Button-Key"}),c("kbd",{class:"DocSearch-Button-Key"},"K")],-1),Pe=b({__name:"VPNavBarSearchButton",setup(s){const{theme:e}=P(),t={button:{buttonText:"Search",buttonAriaLabel:"Search"}},n=je(Na)(Ke(()=>{var o;return(o=e.value.search)==null?void 0:o.options}),t);return(o,r)=>(a(),i("button",{type:"button",class:"DocSearch DocSearch-Button","aria-label":l(n)("button.buttonAriaLabel")},[c("span",Ta,[Ca,c("span",Ba,L(l(n)("button.buttonText")),1)]),Aa],8,Ia))}});const xa={class:"VPNavBarSearch"},Ha={id:"local-search"},za={key:1,id:"docsearch"},Da=b({__name:"VPNavBarSearch",setup(s){const e=Re(()=>qe(()=>import("./VPLocalSearchBox.bcaf1d49.js"),["assets/chunks/VPLocalSearchBox.bcaf1d49.js","assets/chunks/framework.51d6c45b.js"])),t=()=>null,{theme:n}=P(),o=M(!1),r=M(!1);G(()=>{});function d(){o.value||(o.value=!0,setTimeout(p,16))}function p(){const y=new Event("keydown");y.key="k",y.metaKey=!0,window.dispatchEvent(y),setTimeout(()=>{document.querySelector(".DocSearch-Modal")||p()},16)}function _(y){const I=y.target,S=I.tagName;return I.isContentEditable||S==="INPUT"||S==="SELECT"||S==="TEXTAREA"}const g=M(!1);ke("k",y=>{(y.ctrlKey||y.metaKey)&&(y.preventDefault(),g.value=!0)}),ke("/",y=>{_(y)||(y.preventDefault(),g.value=!0)});const V="local";return(y,I)=>{var S;return a(),i("div",xa,[l(V)==="local"?(a(),i(N,{key:0},[g.value?(a(),$(l(e),{key:0,onClose:I[0]||(I[0]=C=>g.value=!1)})):f("",!0),c("div",Ha,[h(Pe,{onClick:I[1]||(I[1]=C=>g.value=!0)})])],64)):l(V)==="algolia"?(a(),i(N,{key:1},[o.value?(a(),$(l(t),{key:0,algolia:((S=l(n).search)==null?void 0:S.options)??l(n).algolia,onVnodeBeforeMount:I[2]||(I[2]=C=>r.value=!0)},null,8,["algolia"])):f("",!0),r.value?f("",!0):(a(),i("div",za,[h(Pe,{onClick:d})]))],64)):f("",!0)])}}});const Ea=b({__name:"VPNavBarSocialLinks",setup(s){const{theme:e}=P();return(t,n)=>l(e).socialLinks?(a(),$($e,{key:0,class:"VPNavBarSocialLinks",links:l(e).socialLinks},null,8,["links"])):f("",!0)}});const Fa=m(Ea,[["__scopeId","data-v-ef6192dc"]]),Oa=["href"],Ga=b({__name:"VPNavBarTitle",setup(s){const{site:e,theme:t}=P(),{hasSidebar:n}=D(),{currentLang:o}=X();return(r,d)=>(a(),i("div",{class:T(["VPNavBarTitle",{"has-sidebar":l(n)}])},[c("a",{class:"title",href:l(t).logoLink??l(J)(l(o).link)},[u(r.$slots,"nav-bar-title-before",{},void 0,!0),l(t).logo?(a(),$(ee,{key:0,class:"logo",image:l(t).logo},null,8,["image"])):f("",!0),l(t).siteTitle?(a(),i(N,{key:1},[x(L(l(t).siteTitle),1)],64)):l(t).siteTitle===void 0?(a(),i(N,{key:2},[x(L(l(e).title),1)],64)):f("",!0),u(r.$slots,"nav-bar-title-after",{},void 0,!0)],8,Oa)],2))}});const Ua=m(Ga,[["__scopeId","data-v-2973dbb4"]]),ja={},Ka={xmlns:"http://www.w3.org/2000/svg","aria-hidden":"true",focusable:"false",viewBox:"0 0 24 24"},Ra=c("path",{d:"M0 0h24v24H0z",fill:"none"},null,-1),qa=c("path",{d:" M12.87 15.07l-2.54-2.51.03-.03c1.74-1.94 2.98-4.17 3.71-6.53H17V4h-7V2H8v2H1v1.99h11.17C11.5 7.92 10.44 9.75 9 11.35 8.07 10.32 7.3 9.19 6.69 8h-2c.73 1.63 1.73 3.17 2.98 4.56l-5.09 5.02L4 19l5-5 3.11 3.11.76-2.04zM18.5 10h-2L12 22h2l1.12-3h4.75L21 22h2l-4.5-12zm-2.62 7l1.62-4.33L19.12 17h-3.24z ",class:"css-c4d79v"},null,-1),Wa=[Ra,qa];function Ya(s,e){return a(),i("svg",Ka,Wa)}const Ae=m(ja,[["render",Ya]]),Ja={class:"items"},Xa={class:"title"},Za=b({__name:"VPNavBarTranslations",setup(s){const{theme:e}=P(),{localeLinks:t,currentLang:n}=X({correspondingLink:!0});return(o,r)=>l(t).length&&l(n).label?(a(),$(be,{key:0,class:"VPNavBarTranslations",icon:Ae,label:l(e).langMenuLabel||"Change language"},{default:v(()=>[c("div",Ja,[c("p",Xa,L(l(n).label),1),(a(!0),i(N,null,A(l(t),d=>(a(),$(oe,{key:d.link,item:d},null,8,["item"]))),128))])]),_:1},8,["label"])):f("",!0)}});const Qa=m(Za,[["__scopeId","data-v-ff4524ae"]]),er=s=>(H("data-v-f1abbc6e"),s=s(),z(),s),tr={class:"container"},sr={class:"title"},nr={class:"content"},or=er(()=>c("div",{class:"curtain"},null,-1)),ar={class:"content-body"},rr=b({__name:"VPNavBar",props:{isScreenOpen:{type:Boolean}},emits:["toggle-screen"],setup(s){const{y:e}=Le(),{hasSidebar:t}=D(),{frontmatter:n}=P(),o=M({});return Ve(()=>{o.value={"has-sidebar":t.value,top:n.value.layout==="home"&&e.value===0}}),(r,d)=>(a(),i("div",{class:T(["VPNavBar",o.value])},[c("div",tr,[c("div",sr,[h(Ua,null,{"nav-bar-title-before":v(()=>[u(r.$slots,"nav-bar-title-before",{},void 0,!0)]),"nav-bar-title-after":v(()=>[u(r.$slots,"nav-bar-title-after",{},void 0,!0)]),_:3})]),c("div",nr,[or,c("div",ar,[u(r.$slots,"nav-bar-content-before",{},void 0,!0),h(Da,{class:"search"}),h(Ma,{class:"menu"}),h(Qa,{class:"translations"}),h($o,{class:"appearance"}),h(Fa,{class:"social-links"}),h(va,{class:"extra"}),u(r.$slots,"nav-bar-content-after",{},void 0,!0),h(ba,{class:"hamburger",active:r.isScreenOpen,onClick:d[0]||(d[0]=p=>r.$emit("toggle-screen"))},null,8,["active"])])])])],2))}});const lr=m(rr,[["__scopeId","data-v-f1abbc6e"]]),ir={key:0,class:"VPNavScreenAppearance"},cr={class:"text"},ur=b({__name:"VPNavScreenAppearance",setup(s){const{site:e,theme:t}=P();return(n,o)=>l(e).appearance?(a(),i("div",ir,[c("p",cr,L(l(t).darkModeSwitchLabel||"Appearance"),1),h(me)])):f("",!0)}});const dr=m(ur,[["__scopeId","data-v-0dc5cf49"]]),_r=b({__name:"VPNavScreenMenuLink",props:{item:{}},setup(s){const e=ne("close-screen");return(t,n)=>(a(),$(E,{class:"VPNavScreenMenuLink",href:t.item.link,target:t.item.target,rel:t.item.rel,onClick:l(e)},{default:v(()=>[x(L(t.item.text),1)]),_:1},8,["href","target","rel","onClick"]))}});const vr=m(_r,[["__scopeId","data-v-fe523e3d"]]),pr={},hr={xmlns:"http://www.w3.org/2000/svg","aria-hidden":"true",focusable:"false",viewBox:"0 0 24 24"},fr=c("path",{d:"M18.9,10.9h-6v-6c0-0.6-0.4-1-1-1s-1,0.4-1,1v6h-6c-0.6,0-1,0.4-1,1s0.4,1,1,1h6v6c0,0.6,0.4,1,1,1s1-0.4,1-1v-6h6c0.6,0,1-0.4,1-1S19.5,10.9,18.9,10.9z"},null,-1),mr=[fr];function gr(s,e){return a(),i("svg",hr,mr)}const br=m(pr,[["render",gr]]),$r=b({__name:"VPNavScreenMenuGroupLink",props:{item:{}},setup(s){const e=ne("close-screen");return(t,n)=>(a(),$(E,{class:"VPNavScreenMenuGroupLink",href:t.item.link,target:t.item.target,rel:t.item.rel,onClick:l(e)},{default:v(()=>[x(L(t.item.text),1)]),_:1},8,["href","target","rel","onClick"]))}});const xe=m($r,[["__scopeId","data-v-aea78dd1"]]),kr={class:"VPNavScreenMenuGroupSection"},yr={key:0,class:"title"},Pr=b({__name:"VPNavScreenMenuGroupSection",props:{text:{},items:{}},setup(s){return(e,t)=>(a(),i("div",kr,[e.text?(a(),i("p",yr,L(e.text),1)):f("",!0),(a(!0),i(N,null,A(e.items,n=>(a(),$(xe,{key:n.text,item:n},null,8,["item"]))),128))]))}});const Vr=m(Pr,[["__scopeId","data-v-f60dbfa7"]]),wr=["aria-controls","aria-expanded"],Lr={class:"button-text"},Sr=["id"],Mr={key:1,class:"group"},Nr=b({__name:"VPNavScreenMenuGroup",props:{text:{},items:{}},setup(s){const e=s,t=M(!1),n=k(()=>`NavScreenGroup-${e.text.replace(" ","-").toLowerCase()}`);function o(){t.value=!t.value}return(r,d)=>(a(),i("div",{class:T(["VPNavScreenMenuGroup",{open:t.value}])},[c("button",{class:"button","aria-controls":n.value,"aria-expanded":t.value,onClick:o},[c("span",Lr,L(r.text),1),h(br,{class:"button-icon"})],8,wr),c("div",{id:n.value,class:"items"},[(a(!0),i(N,null,A(r.items,p=>(a(),i(N,{key:p.text},["link"in p?(a(),i("div",{key:p.text,class:"item"},[h(xe,{item:p},null,8,["item"])])):(a(),i("div",Mr,[h(Vr,{text:p.text,items:p.items},null,8,["text","items"])]))],64))),128))],8,Sr)],2))}});const Ir=m(Nr,[["__scopeId","data-v-c2c554ed"]]),Tr={key:0,class:"VPNavScreenMenu"},Cr=b({__name:"VPNavScreenMenu",setup(s){const{theme:e}=P();return(t,n)=>l(e).nav?(a(),i("nav",Tr,[(a(!0),i(N,null,A(l(e).nav,o=>(a(),i(N,{key:o.text},["link"in o?(a(),$(vr,{key:0,item:o},null,8,["item"])):(a(),$(Ir,{key:1,text:o.text||"",items:o.items},null,8,["text","items"]))],64))),128))])):f("",!0)}}),Br=b({__name:"VPNavScreenSocialLinks",setup(s){const{theme:e}=P();return(t,n)=>l(e).socialLinks?(a(),$($e,{key:0,class:"VPNavScreenSocialLinks",links:l(e).socialLinks},null,8,["links"])):f("",!0)}}),Ar={class:"list"},xr=b({__name:"VPNavScreenTranslations",setup(s){const{localeLinks:e,currentLang:t}=X({correspondingLink:!0}),n=M(!1);function o(){n.value=!n.value}return(r,d)=>l(e).length&&l(t).label?(a(),i("div",{key:0,class:T(["VPNavScreenTranslations",{open:n.value}])},[c("button",{class:"title",onClick:o},[h(Ae,{class:"icon lang"}),x(" "+L(l(t).label)+" ",1),h(Be,{class:"icon chevron"})]),c("ul",Ar,[(a(!0),i(N,null,A(l(e),p=>(a(),i("li",{key:p.link,class:"item"},[h(E,{class:"link",href:p.link},{default:v(()=>[x(L(p.text),1)]),_:2},1032,["href"])]))),128))])],2)):f("",!0)}});const Hr=m(xr,[["__scopeId","data-v-41505286"]]),zr={class:"container"},Dr=b({__name:"VPNavScreen",props:{open:{type:Boolean}},setup(s){const e=M(null),t=Se(q?document.body:null);return(n,o)=>(a(),$(ce,{name:"fade",onEnter:o[0]||(o[0]=r=>t.value=!0),onAfterLeave:o[1]||(o[1]=r=>t.value=!1)},{default:v(()=>[n.open?(a(),i("div",{key:0,class:"VPNavScreen",ref_key:"screen",ref:e,id:"VPNavScreen"},[c("div",zr,[u(n.$slots,"nav-screen-content-before",{},void 0,!0),h(Cr,{class:"menu"}),h(Hr,{class:"translations"}),h(dr,{class:"appearance"}),h(Br,{class:"social-links"}),u(n.$slots,"nav-screen-content-after",{},void 0,!0)])],512)):f("",!0)]),_:3}))}});const Er=m(Dr,[["__scopeId","data-v-57cce842"]]),Fr={key:0,class:"VPNav"},Or=b({__name:"VPNav",setup(s){const{isScreenOpen:e,closeScreen:t,toggleScreen:n}=Xn(),{frontmatter:o}=P(),r=k(()=>o.value.navbar!==!1);return Me("close-screen",t),te(()=>{q&&document.documentElement.classList.toggle("hide-nav",!r.value)}),(d,p)=>r.value?(a(),i("header",Fr,[h(lr,{"is-screen-open":l(e),onToggleScreen:l(n)},{"nav-bar-title-before":v(()=>[u(d.$slots,"nav-bar-title-before",{},void 0,!0)]),"nav-bar-title-after":v(()=>[u(d.$slots,"nav-bar-title-after",{},void 0,!0)]),"nav-bar-content-before":v(()=>[u(d.$slots,"nav-bar-content-before",{},void 0,!0)]),"nav-bar-content-after":v(()=>[u(d.$slots,"nav-bar-content-after",{},void 0,!0)]),_:3},8,["is-screen-open","onToggleScreen"]),h(Er,{open:l(e)},{"nav-screen-content-before":v(()=>[u(d.$slots,"nav-screen-content-before",{},void 0,!0)]),"nav-screen-content-after":v(()=>[u(d.$slots,"nav-screen-content-after",{},void 0,!0)]),_:3},8,["open"])])):f("",!0)}});const Gr=m(Or,[["__scopeId","data-v-7ad780c2"]]),Ur=s=>(H("data-v-bd01e0d5"),s=s(),z(),s),jr=["role","tabindex"],Kr=Ur(()=>c("div",{class:"indicator"},null,-1)),Rr=["onKeydown"],qr={key:1,class:"items"},Wr=b({__name:"VPSidebarItem",props:{item:{},depth:{}},setup(s){const e=s,{collapsed:t,collapsible:n,isLink:o,isActiveLink:r,hasActiveLink:d,hasChildren:p,toggle:_}=gt(k(()=>e.item)),g=k(()=>p.value?"section":"div"),V=k(()=>o.value?"a":"div"),y=k(()=>p.value?e.depth+2===7?"p":`h${e.depth+2}`:"p"),I=k(()=>o.value?void 0:"button"),S=k(()=>[[`level-${e.depth}`],{collapsible:n.value},{collapsed:t.value},{"is-link":o.value},{"is-active":r.value},{"has-active":d.value}]);function C(w){"key"in w&&w.key!=="Enter"||!e.item.link&&_()}function B(){e.item.link&&_()}return(w,K)=>{const W=j("VPSidebarItem",!0);return a(),$(F(g.value),{class:T(["VPSidebarItem",S.value])},{default:v(()=>[w.item.text?(a(),i("div",Z({key:0,class:"item",role:I.value},Ye(w.item.items?{click:C,keydown:C}:{},!0),{tabindex:w.item.items&&0}),[Kr,w.item.link?(a(),$(E,{key:0,tag:V.value,class:"link",href:w.item.link,rel:w.item.rel,target:w.item.target},{default:v(()=>[(a(),$(F(y.value),{class:"text",innerHTML:w.item.text},null,8,["innerHTML"]))]),_:1},8,["tag","href","rel","target"])):(a(),$(F(y.value),{key:1,class:"text",innerHTML:w.item.text},null,8,["innerHTML"])),w.item.collapsed!=null?(a(),i("div",{key:2,class:"caret",role:"button","aria-label":"toggle section",onClick:B,onKeydown:We(B,["enter"]),tabindex:"0"},[h(fe,{class:"caret-icon"})],40,Rr)):f("",!0)],16,jr)):f("",!0),w.item.items&&w.item.items.length?(a(),i("div",qr,[w.depth<5?(a(!0),i(N,{key:0},A(w.item.items,R=>(a(),$(W,{key:R.text,item:R,depth:w.depth+1},null,8,["item","depth"]))),128)):f("",!0)])):f("",!0)]),_:1},8,["class"])}}});const Yr=m(Wr,[["__scopeId","data-v-bd01e0d5"]]),He=s=>(H("data-v-ee2efba5"),s=s(),z(),s),Jr=He(()=>c("div",{class:"curtain"},null,-1)),Xr={class:"nav",id:"VPSidebarNav","aria-labelledby":"sidebar-aria-label",tabindex:"-1"},Zr=He(()=>c("span",{class:"visually-hidden",id:"sidebar-aria-label"}," Sidebar Navigation ",-1)),Qr=b({__name:"VPSidebar",props:{open:{type:Boolean}},setup(s){const{sidebarGroups:e,hasSidebar:t}=D(),n=s,o=M(null),r=Se(q?document.body:null);return U([n,o],()=>{var d;n.open?(r.value=!0,(d=o.value)==null||d.focus()):r.value=!1},{immediate:!0,flush:"post"}),(d,p)=>l(t)?(a(),i("aside",{key:0,class:T(["VPSidebar",{open:d.open}]),ref_key:"navEl",ref:o,onClick:p[0]||(p[0]=Je(()=>{},["stop"]))},[Jr,c("nav",Xr,[Zr,u(d.$slots,"sidebar-nav-before",{},void 0,!0),(a(!0),i(N,null,A(l(e),_=>(a(),i("div",{key:_.text,class:"group"},[h(Yr,{item:_,depth:0},null,8,["item"])]))),128)),u(d.$slots,"sidebar-nav-after",{},void 0,!0)])],2)):f("",!0)}});const el=m(Qr,[["__scopeId","data-v-ee2efba5"]]),tl=b({__name:"VPSkipLink",setup(s){const e=se(),t=M();U(()=>e.path,()=>t.value.focus());function n({target:o}){const r=document.getElementById(decodeURIComponent(o.hash).slice(1));if(r){const d=()=>{r.removeAttribute("tabindex"),r.removeEventListener("blur",d)};r.setAttribute("tabindex","-1"),r.addEventListener("blur",d),r.focus(),window.scrollTo(0,0)}}return(o,r)=>(a(),i(N,null,[c("span",{ref_key:"backToTop",ref:t,tabindex:"-1"},null,512),c("a",{href:"#VPContent",class:"VPSkipLink visually-hidden",onClick:n}," Skip to content ")],64))}});const sl=m(tl,[["__scopeId","data-v-c8291ffa"]]),nl=b({__name:"Layout",setup(s){const{isOpen:e,open:t,close:n}=D(),o=se();U(()=>o.path,n),mt(e,n);const{frontmatter:r}=P(),d=Xe(),p=k(()=>!!d["home-hero-image"]);return Me("hero-image-slot-exists",p),(_,g)=>{const V=j("Content");return l(r).layout!==!1?(a(),i("div",{key:0,class:T(["Layout",l(r).pageClass])},[u(_.$slots,"layout-top",{},void 0,!0),h(sl),h(st,{class:"backdrop",show:l(e),onClick:l(n)},null,8,["show","onClick"]),h(Gr,null,{"nav-bar-title-before":v(()=>[u(_.$slots,"nav-bar-title-before",{},void 0,!0)]),"nav-bar-title-after":v(()=>[u(_.$slots,"nav-bar-title-after",{},void 0,!0)]),"nav-bar-content-before":v(()=>[u(_.$slots,"nav-bar-content-before",{},void 0,!0)]),"nav-bar-content-after":v(()=>[u(_.$slots,"nav-bar-content-after",{},void 0,!0)]),"nav-screen-content-before":v(()=>[u(_.$slots,"nav-screen-content-before",{},void 0,!0)]),"nav-screen-content-after":v(()=>[u(_.$slots,"nav-screen-content-after",{},void 0,!0)]),_:3}),h(Jn,{open:l(e),onOpenMenu:l(t)},null,8,["open","onOpenMenu"]),h(el,{open:l(e)},{"sidebar-nav-before":v(()=>[u(_.$slots,"sidebar-nav-before",{},void 0,!0)]),"sidebar-nav-after":v(()=>[u(_.$slots,"sidebar-nav-after",{},void 0,!0)]),_:3},8,["open"]),h(Mn,null,{"page-top":v(()=>[u(_.$slots,"page-top",{},void 0,!0)]),"page-bottom":v(()=>[u(_.$slots,"page-bottom",{},void 0,!0)]),"not-found":v(()=>[u(_.$slots,"not-found",{},void 0,!0)]),"home-hero-before":v(()=>[u(_.$slots,"home-hero-before",{},void 0,!0)]),"home-hero-info":v(()=>[u(_.$slots,"home-hero-info",{},void 0,!0)]),"home-hero-image":v(()=>[u(_.$slots,"home-hero-image",{},void 0,!0)]),"home-hero-after":v(()=>[u(_.$slots,"home-hero-after",{},void 0,!0)]),"home-features-before":v(()=>[u(_.$slots,"home-features-before",{},void 0,!0)]),"home-features-after":v(()=>[u(_.$slots,"home-features-after",{},void 0,!0)]),"doc-footer-before":v(()=>[u(_.$slots,"doc-footer-before",{},void 0,!0)]),"doc-before":v(()=>[u(_.$slots,"doc-before",{},void 0,!0)]),"doc-after":v(()=>[u(_.$slots,"doc-after",{},void 0,!0)]),"doc-top":v(()=>[u(_.$slots,"doc-top",{},void 0,!0)]),"doc-bottom":v(()=>[u(_.$slots,"doc-bottom",{},void 0,!0)]),"aside-top":v(()=>[u(_.$slots,"aside-top",{},void 0,!0)]),"aside-bottom":v(()=>[u(_.$slots,"aside-bottom",{},void 0,!0)]),"aside-outline-before":v(()=>[u(_.$slots,"aside-outline-before",{},void 0,!0)]),"aside-outline-after":v(()=>[u(_.$slots,"aside-outline-after",{},void 0,!0)]),"aside-ads-before":v(()=>[u(_.$slots,"aside-ads-before",{},void 0,!0)]),"aside-ads-after":v(()=>[u(_.$slots,"aside-ads-after",{},void 0,!0)]),_:3}),h(Bn),u(_.$slots,"layout-bottom",{},void 0,!0)],2)):(a(),$(V,{key:1}))}}});const ol=m(nl,[["__scopeId","data-v-9d8abc1e"]]);const rl={Layout:ol,enhanceApp:({app:s})=>{s.component("Badge",Qe)}};export{Na as c,rl as t,P as u}; diff --git a/assets/chunks/tx-lifecycle.92aed1e6.js b/assets/chunks/tx-lifecycle.92aed1e6.js new file mode 100644 index 00000000000..b693a3caab7 --- /dev/null +++ b/assets/chunks/tx-lifecycle.92aed1e6.js @@ -0,0 +1 @@ +const e="/img/learn/tx-lifecycle.png";export{e as _}; diff --git a/assets/community_calendar.md.46d438d4.js b/assets/community_calendar.md.46d438d4.js new file mode 100644 index 00000000000..33775e948bf --- /dev/null +++ b/assets/community_calendar.md.46d438d4.js @@ -0,0 +1 @@ +import{_ as t,o as n,c as o,k as e,a}from"./chunks/framework.51d6c45b.js";const b=JSON.parse(`{"title":"Community calendar","description":"Find all the different community call events happening in Celestia's community.","frontmatter":{"description":"Find all the different community call events happening in Celestia's community.","head":[["meta",{"name":"og:title","content":"Community calendar | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/calendar.md","filePath":"community/calendar.md","lastUpdated":1698091500000}`),l={name:"community/calendar.md"},c=e("h1",{id:"community-calendar",tabindex:"-1"},[a("Community calendar "),e("a",{class:"header-anchor",href:"#community-calendar","aria-label":'Permalink to "Community calendar"'},"​")],-1),i=e("p",null,"The Celestia community calendar is available for finding all the different community call events happening in Celestia's community.",-1),r=e("p",null,[e("a",{href:"https://calendar.google.com/calendar/u/0?cid=Y19za2JzbjIzNWszYmlzdHNoZ3RvNmw5ODYyNEBncm91cC5jYWxlbmRhci5nb29nbGUuY29t",target:"_blank",rel:"noreferrer"},"Add the community calendar to your personal calendar"),a(" to stay updated with all the events.")],-1),s=e("p",null,[a("Explore "),e("a",{href:"https://github.com/celestiaorg/community-calls/blob/main/README.md",target:"_blank",rel:"noreferrer"},"past community call agendas, notes, and recordings"),a(" for more insights.")],-1),m=[c,i,r,s];function d(u,h,p,f,_,y){return n(),o("div",null,m)}const C=t(l,[["render",d]]);export{b as __pageData,C as default}; diff --git a/assets/community_calendar.md.46d438d4.lean.js b/assets/community_calendar.md.46d438d4.lean.js new file mode 100644 index 00000000000..33775e948bf --- /dev/null +++ b/assets/community_calendar.md.46d438d4.lean.js @@ -0,0 +1 @@ +import{_ as t,o as n,c as o,k as e,a}from"./chunks/framework.51d6c45b.js";const b=JSON.parse(`{"title":"Community calendar","description":"Find all the different community call events happening in Celestia's community.","frontmatter":{"description":"Find all the different community call events happening in Celestia's community.","head":[["meta",{"name":"og:title","content":"Community calendar | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/calendar.md","filePath":"community/calendar.md","lastUpdated":1698091500000}`),l={name:"community/calendar.md"},c=e("h1",{id:"community-calendar",tabindex:"-1"},[a("Community calendar "),e("a",{class:"header-anchor",href:"#community-calendar","aria-label":'Permalink to "Community calendar"'},"​")],-1),i=e("p",null,"The Celestia community calendar is available for finding all the different community call events happening in Celestia's community.",-1),r=e("p",null,[e("a",{href:"https://calendar.google.com/calendar/u/0?cid=Y19za2JzbjIzNWszYmlzdHNoZ3RvNmw5ODYyNEBncm91cC5jYWxlbmRhci5nb29nbGUuY29t",target:"_blank",rel:"noreferrer"},"Add the community calendar to your personal calendar"),a(" to stay updated with all the events.")],-1),s=e("p",null,[a("Explore "),e("a",{href:"https://github.com/celestiaorg/community-calls/blob/main/README.md",target:"_blank",rel:"noreferrer"},"past community call agendas, notes, and recordings"),a(" for more insights.")],-1),m=[c,i,r,s];function d(u,h,p,f,_,y){return n(),o("div",null,m)}const C=t(l,[["render",d]]);export{b as __pageData,C as default}; diff --git a/assets/community_coc.md.e9cffa22.js b/assets/community_coc.md.e9cffa22.js new file mode 100644 index 00000000000..b8d3eb7b648 --- /dev/null +++ b/assets/community_coc.md.e9cffa22.js @@ -0,0 +1 @@ +import{_ as e,o,c as a,Q as t}from"./chunks/framework.51d6c45b.js";const f=JSON.parse(`{"title":"Celestia.org Code of Conduct","description":"Fundamental shared values and special norms that distinguish Celestia's community.","frontmatter":{"description":"Fundamental shared values and special norms that distinguish Celestia's community.","head":[["meta",{"name":"og:title","content":"Celestia.org Code of Conduct | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/coc.md","filePath":"community/coc.md","lastUpdated":1698340253000}`),n={name:"community/coc.md"},i=t('

Celestia.org Code of Conduct

Our Pledge

We as Celestia.org members, contributors, and leaders pledge to make participation in our community a harassment-free experience for everyone, regardless of age, body size, visible or invisible disability, ethnicity, sex characteristics, gender identity and expression, level of experience, education, socio-economic status, nationality, personal appearance, race, caste, color, religion, or sexual identity and orientation.

We pledge to act and interact in ways that contribute to an open, welcoming, diverse, inclusive, and healthy community.

Our Standards

Examples of behavior that contributes to a positive environment for our community include:

  • Demonstrating empathy and kindness toward other people
  • Being respectful of differing opinions, viewpoints, and experiences
  • Giving and gracefully accepting constructive feedback
  • Accepting responsibility and apologizing to those affected by our mistakes, and learning from the experience
  • Focusing on what is best not just for us as individuals, but for the overall community
  • Contributing to conversations about Celestia’s technology and ecosystem

Examples of unacceptable behavior include:

  • The use of sexualized language or imagery, and sexual attention or advances of any kind
  • Trolling, insulting or derogatory comments, and personal or political attacks
  • Public or private harassment
  • Publishing others' private information, such as a physical or email address, without their explicit permission
  • Focusing on the prices of digital assets or tokens, or where they can be purchased
  • Other conduct which could reasonably be considered inappropriate in a professional setting

Enforcement Responsibilities

Community leaders are responsible for clarifying and enforcing our standards of acceptable behavior and will take appropriate and fair corrective action in response to any behavior that they deem inappropriate, threatening, offensive, or harmful.

Community leaders have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct, and will communicate reasons for moderation decisions when appropriate.

Scope

This Code of Conduct applies within all community spaces, and also applies when an individual is officially representing the community in public spaces. Examples of representing our community include using an official e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event.

Enforcement

Instances of abusive, harassing, or otherwise unacceptable behavior may be reported to the community leaders responsible for enforcement at Celestia.org Discord. All complaints will be reviewed and investigated promptly and fairly.

All community leaders are obligated to respect the privacy and security of the reporter of any incident.

Enforcement Guidelines

Community leaders will follow these Community Impact Guidelines in determining the consequences for any action they deem in violation of this Code of Conduct:

1. Correction

Community Impact: Use of inappropriate language or other behavior deemed unprofessional or unwelcome in the community.

Consequence: A private, written warning from community leaders, providing clarity around the nature of the violation and an explanation of why the behavior was inappropriate. A public apology may be requested.

2. Warning

Community Impact: A violation through a single incident or series of actions.

Consequence: A warning with consequences for continued behavior. No interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, for a specified period of time. This includes avoiding interactions in community spaces as well as external channels like social media. Violating these terms may lead to a temporary or permanent ban.

3. Temporary Ban

Community Impact: A serious violation of community standards, including sustained inappropriate behavior.

Consequence: A temporary ban from any sort of interaction or public communication with the community for a specified period of time. No public or private interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, is allowed during this period. Violating these terms may lead to a permanent ban.

4. Permanent Ban

Community Impact: Demonstrating a pattern of violation of community standards, including sustained inappropriate behavior, harassment of an individual, or aggression toward or disparagement of classes of individuals.

Consequence: A permanent ban from any sort of public interaction within the community.

Attribution

This Code of Conduct is adapted from the Contributor Covenant, version 2.1, available at https://www.contributor-covenant.org/version/2/1/code_of_conduct.html.

Community Impact Guidelines were inspired by Mozilla's code of conduct enforcement ladder.

For answers to common questions about this code of conduct, see the FAQ at https://www.contributor-covenant.org/faq. Translations are available at https://www.contributor-covenant.org/translations.

',35),r=[i];function s(c,l,d,p,u,h){return o(),a("div",null,r)}const g=e(n,[["render",s]]);export{f as __pageData,g as default}; diff --git a/assets/community_coc.md.e9cffa22.lean.js b/assets/community_coc.md.e9cffa22.lean.js new file mode 100644 index 00000000000..5beb4ebae73 --- /dev/null +++ b/assets/community_coc.md.e9cffa22.lean.js @@ -0,0 +1 @@ +import{_ as e,o,c as a,Q as t}from"./chunks/framework.51d6c45b.js";const f=JSON.parse(`{"title":"Celestia.org Code of Conduct","description":"Fundamental shared values and special norms that distinguish Celestia's community.","frontmatter":{"description":"Fundamental shared values and special norms that distinguish Celestia's community.","head":[["meta",{"name":"og:title","content":"Celestia.org Code of Conduct | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/coc.md","filePath":"community/coc.md","lastUpdated":1698340253000}`),n={name:"community/coc.md"},i=t("",35),r=[i];function s(c,l,d,p,u,h){return o(),a("div",null,r)}const g=e(n,[["render",s]]);export{f as __pageData,g as default}; diff --git a/assets/community_foundation-delegation-program.md.0eb350d7.js b/assets/community_foundation-delegation-program.md.0eb350d7.js new file mode 100644 index 00000000000..3be54f7f2cc --- /dev/null +++ b/assets/community_foundation-delegation-program.md.0eb350d7.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.51d6c45b.js";const o="/img/foundation-delegation-program.jpg",r="/img/cohort-timeline.jpg",b=JSON.parse('{"title":"The Celestia Foundation Delegation Program","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"The Celestia Foundation Delegation Program | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/foundation-delegation-program.md","filePath":"community/foundation-delegation-program.md","lastUpdated":1707755594000}'),n={name:"community/foundation-delegation-program.md"},l=i('

The Celestia Foundation Delegation Program

Delegation program banner

Objectives of the program

The primary objectives of the Celestia Foundation Delegation Program are:

  • To provide a fair opportunity for Celestia’s users to join the validator set, while ensuring the validator set remains proficient, trustworthy, and dependable.
  • To maintain network stability by promoting a steady transition of validators and avoiding sudden and disruptive changes in participation.
  • To enable the Celestia Foundation to use its stake towards its mission of fostering a modular blockchain network that delivers exceptional performance.

Foundation delegation process

Program launch

Prospective validators are welcome to apply to the program starting February 6, 2024. The application is designed to assess a validator’s uptime performance and contributions to the Celestia ecosystem. Of the 100 total slots in Celestia’s active validator set, up to 50 will receive delegations within the program.

Application submissions will be reviewed by the Celestia Foundation. More details about the application and eligibility criteria are described below.

Cohort process

cohort timeline

Every 4 months, the Celestia Foundation will distribute a portion of the Foundation’s total available stake to a cohort of validators who meet certain criteria, detailed below. Here is an overview of how the cohort process will work for Cohort 1 and what that means for future cohorts.

Key Points

  • Initial Cohort (Cohort 1): 50 applicants will be accepted
    • Grading System: Applicants in Cohort 1 are divided into first, second, and third place based eligibility criteria outlined in this document.
    • Delegation Duration: This varies based on the applicant’s placement in Cohort 1. First place receives 12 months of delegation, second place receives 8 months, third place receives 4 months.
TierPlacementDelegation DurationRenewal By Cohort
First PlaceApplicants 1-2012 monthsCohort 4
Second PlaceApplicants 21-358 monthsCohort 3
Third PlaceApplicants 36-504 monthsCohort 2
  • Subsequent Cohorts (Cohorts 2-onwards):
    • After Cohort 1, open slots may be filled by Cohort 1 members up for renewals or new applicants. There will be no Tiers (e.g. First Place, Second Place, Third Place) in cohorts after Cohort 1. This structure allows for a steady flow of both existing applicants and new applicants to maintain a stable set of participants in the program.

During this period, so long as the validator maintains high uptime and does not violate the rules of the program, the validator will receive the delegation for the duration of the cohort they are currently in.

Eligibility criteria

The minimum requirements for participation in the program are as follows:

  • Run an active mainnet validator or an active Mocha testnet validator for at least 1 month before application deadline
  • Run a bridge node (on mainnet if you are already an active mainnet validator or on Mocha testnet if not) that is connected and reporting to the Celestia Labs OTEL collector (for new applicants - on testnet, so that we can evaluate performance)
  • Not jailed or slashed in the 6 months before application deadline
  • Not associated with an exchange or custodian
  • Not in the top 10 validators by delegation power, unless it enters the top 10 as a result of the Foundation’s delegation under this program
  • Have 10% or less commission
  • Not based within the US, within any country subject to economic sanctions, or within any other prohibited jurisdiction, and successfully complete a compliance screen
  • Dedicated email address so that the Foundation can reach you in the event of emergency upgrades and fixes

Not adhering to any of the criteria above will automatically disqualify your application, and violating any of the criteria after you have received delegation will result in withdrawal of the delegation. A participant who loses stake due to being jailed by the protocol may reapply to the program after 2 cohort periods.

Applicants are also expected to have reviewed Celestia docs and recommended guides on devops and monitoring setups.

Other optional but important criteria:

  • Develop and maintain developer tooling, services, applications, and dashboards
  • Work on projects aligned with Celestia's values
  • Contribute to documentation and new guides and tutorials
  • Quality of infrastructure
  • Operated within a location that improves geolocation of the validator set

Undelegation criteria

  • Getting slashed/tombstoned (cannot apply for 1 year afterwards)
  • Getting jailed more than once during the cohort’s applicable delegation period
  • Violating the Celestia.org Community Code of Conduct or engaging in harmful activities towards the network
  • Failing to upgrade your node in a timely manner (24 hours or less)
  • If necessary to protect or secure Mainnet Beta or to comply with applicable law
  • For any other reason, in the Celestia Foundation’s sole discretion

Application

The program will be divided into cohorts with applications open for new applicants and renewal of existing applicants every 4 months. Validators will be delegated for up to a year. For each cohort, the deadline to apply/be evaluated (if you are reapplying) is exactly 1 month prior to the date of being delegated to.

Application details

Before applying, be ready to share the following:

  • General info
    • Security Email
    • Validator Entity Name
    • Discord ID
    • Mark if entity or individual
    • Website if any
    • Github page of your organization
    • Team experience and roster (including Twitter + Github links)
    • Which networks you validate on mainnet + links to your validators
    • A personal statement why you should receive delegation from the Foundation (max 1500 characters)
  • Infrastructure
    • Validator address and bridge node ID on MAINNET
    • If you don't run an active mainnet validator, please provide us with validator address, bridge node ID and blobstream address on Mocha-4
    • Have you been slashed or jailed in the last 6 months on Celestia or other chains you validated on.
    • Hosting provider and Data Center location (mainnet and testnet if applicable)
    • Setup of the 2 components (validator and bridge)
      • Hardware
      • Security setup (servers, private keys)
      • Monitoring and alerting
  • Contributions
    • Please list all technical contributions for Celestia and its ecosystem
    • Please list all community contributions for Celestia and its ecosystem

Please note, the objective of the program is to contribute to Celestia’s resilience and uptime. If you contribute a lot to the Celestia ecosystem, but your validator uptime is low, this will negatively impact your chance at selection for the program. Furthermore, merely receiving delegation from the Foundation under the program does not guarantee your placement in the active validator set.

Get Started with the Application Form

Cohort information

The Foundation will report each cohort’s composition and the duration of their respective delegations.

Feedback process

Validators in the program will receive a feedback form every quarter so the program can be continually improved.

',37),s=[l];function d(c,h,p,u,m,g){return t(),a("div",null,s)}const y=e(n,[["render",d]]);export{b as __pageData,y as default}; diff --git a/assets/community_foundation-delegation-program.md.0eb350d7.lean.js b/assets/community_foundation-delegation-program.md.0eb350d7.lean.js new file mode 100644 index 00000000000..e0ea789d8ca --- /dev/null +++ b/assets/community_foundation-delegation-program.md.0eb350d7.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.51d6c45b.js";const o="/img/foundation-delegation-program.jpg",r="/img/cohort-timeline.jpg",b=JSON.parse('{"title":"The Celestia Foundation Delegation Program","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"The Celestia Foundation Delegation Program | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/foundation-delegation-program.md","filePath":"community/foundation-delegation-program.md","lastUpdated":1707755594000}'),n={name:"community/foundation-delegation-program.md"},l=i("",37),s=[l];function d(c,h,p,u,m,g){return t(),a("div",null,s)}const y=e(n,[["render",d]]);export{b as __pageData,y as default}; diff --git a/assets/community_itn-tos.md.fda6f837.js b/assets/community_itn-tos.md.fda6f837.js new file mode 100644 index 00000000000..377db555a66 --- /dev/null +++ b/assets/community_itn-tos.md.fda6f837.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.51d6c45b.js";const m=JSON.parse('{"title":"SUPPLEMENTAL INCENTIVIZED TESTNET TERMS","description":"These Terms govern your ability to participate in the Incentivized Testnet Award Program.","frontmatter":{"description":"These Terms govern your ability to participate in the Incentivized Testnet Award Program.","lastUpdated":false,"editLink":false,"head":[["meta",{"name":"og:title","content":"SUPPLEMENTAL INCENTIVIZED TESTNET TERMS | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/itn-tos.md","filePath":"community/itn-tos.md","lastUpdated":1699020127000}'),r={name:"community/itn-tos.md"},n=a('

SUPPLEMENTAL INCENTIVIZED TESTNET TERMS

Last Revised on 1/16/2023

Welcome to the Supplemental Terms (these "ITN Award Program Terms" or "Terms") for the Incentivized Testnet Award Program (the "ITN Award Program" or the "Program") as operated on behalf of Strange Loop Labs AG ("Company", "we" or "us"). The ITN Award Program provides eligible users of a Testnet designated by the Company the opportunity to earn rewards, which may include Celestia tokens. These Terms are supplemental to, and incorporate by reference, the broader Celestia Terms of Service ("Services Terms") available at Celestia Terms of Service.

Defined terms used but not defined herein have the meaning set forth in the Services Terms. The Program and your participation in it is a Service as defined under the Services Terms.

These Terms govern your ability to participate in the Program and any awards you receive from that participation, which may include Celestia tokens ("ITN Rewards").

Please read these Terms carefully, as they include important information about your legal rights. By participating in the Program or claiming ITN Rewards, you are agreeing to these Terms. If you do not understand or agree to these Terms, please do not participate in the Program or claim ITN Rewards.

In order to participate in the Program you must provide certain information about you. Our collection of such information, your rights with respect to such collection, and other relevant information is described in the Celestia Privacy Policy available at Celestia Privacy Policy, and is supplemented by Section 3 of these Terms.

The Program is a discretionary Service provided by the Company, pursuant to which the Company may, in its sole discretion, provide you ITN Rewards for your successful completion of certain tasks on a Testnet designated by the Company. Please note that any such Testnet itself (as well as any other Testnets or any mainnet deployment of the Celestia Protocol) is not a Service and does not constitute an element of the Services. We do not control the Celestia Protocol and accept no liability for its operation or its deployment in any testnet or mainnet environment.

1. General Terms

1.1 You must be eighteen (18) years of age or older and capable of forming a binding contract with the Company in order to participate in the Program or receive ITN Rewards.

1.2 You agree and acknowledge that you (a) may receive ITN Rewards for free (other than applicable taxes, if any) from your participation in the Program, (b) were not previously promised ITN Rewards, unless pursuant to a separate written agreement, and (c) took no action in anticipation of or in reliance on receiving any ITN Rewards, unless pursuant to a separate written agreement.

1.3 Your eligibility to participate in or receive ITN Rewards from the Program is subject to our sole discretion. The complete list of actions you must complete to earn ITN Rewards may not have been described in the documentation released by us from time to time, you may not receive ITN Rewards even if you successfully complete such actions, and no documentation related to the Program entitles you to any ITN Rewards or to participate in the Program.

1.4 You agree and acknowledge that (a) you are not a Prohibited Person, (b) you are not a U.S. Person as defined in Rule 902(k) of Regulation S under the U.S. Securities Act of 1933, as amended (the "1933 Act" or "Act"), (c) you will not use a VPN or other tool to circumvent any geoblock or other restrictions that we may have implemented for participants in the Program, and (d) you are not participating in, and have not become eligible to participate in, the Program by receiving credentials from any other person or entity. Any circumvention or violation of the above will permanently disqualify you from participation in the Program.

1.5 You agree and acknowledge that if you are unable to claim ITN Rewards due to technical bugs, gas fees, loss of access to a Wallet or the keys thereto, or for any other reason, you will have no recourse or claim against us or any other Company Entity and that neither we nor any other Company Entity will bear any liability.

1.6 You agree and acknowledge that claiming an ITN Reward may require reliance on or an integration with third party products (e.g., a Wallet or an unaffiliated network or blockchain) that we do not control. In the event that you are unable to access such products or integrations, or if they fail for any reason, and you are unable to participate in the Program or claim ITN Rewards, you will have no recourse or claim against us or any other Company Entity and neither we nor any other Company Entity will bear any liability.

1.7 The Company may share identifying information and documentation with certain vendors or third-party providers who provide such identity verification and sanctions and watchlist screening services (the "Third-Party Services"). You agree that your access and use of such Third-Party Services is governed solely by the terms and conditions of such Third-Party Services, and the Company is not responsible or liable for, and make no representations as to any aspect of such Third-Party Services, including, without limitation, their content or the manner in which they handle, protect, manage or process data or any interaction between you and the provider of such Third-Party Services. You irrevocably waive any claim against the Company with respect to such Third-Party Services. We are not liable for any damage or loss caused or alleged to be caused by or in connection with your enablement, access or use of any such Third-Party Services, or your reliance on the privacy practices, data security processes or other policies of such Third-Party Services.

2. Taxes

2.1 You are responsible for the payment of all taxes associated with your participation in the Program and your receipt of ITN Rewards. You agree to provide the Company with any additional information and complete any required tax or other forms relating to your receipt of ITN Rewards. You may suffer adverse tax consequences as a result of your participation in the Program or your receipt of ITN Rewards. You hereby represent that (a) you have consulted with a tax adviser that you deem advisable in connection with your participation, or that you have had the opportunity to obtain tax advice but have chosen not to do so, (b) the Company has not provided you with any tax advice with respect to your participation, and (c) you are not relying on the Company for any tax advice.

3. Supplemental Privacy Information

We may collect information to help us determine the reliability or uptime of your activities within the Program, including through the use of telemetry or metrics endpoints to collect and analyse such information, and link this information to a unique identifier to represent your activities within the Program. We may display all of the foregoing information on a public dashboard.

Additionally, we may collect certain information about you from Third-Party Services and may combine information we receive from you with information we obtain from Third-Party Services, including but not limited to:

  • Transaction information. Information related to transactions in your Wallet, your Wallet address, activities performed using your Wallet, tokens received by your Wallet, or transactions initiated or completed.
  • Identification information. We collect your government identification (e.g., driver’s license, passport, etc.), proof of address, biometric information, and entity formation information if applicable. By agreeing to these Terms, you consent to our use of your biometric information, and understand and agree that our use of the biometric information is necessary for the performance of these Terms and the implementation of the Services.

We collect this information to confirm your eligibility to participate in the Program and receive ITN Rewards, comply with our legal obligations, detect and prevent fraud, and to provide you with the Program.

Any information we receive from third-party sources will be treated in accordance with the Celestia Privacy Policy, available at Celestia Privacy Policy. We are not responsible or liable for the accuracy of the information provided to us by third parties and are not responsible for any third party’s policies or practices. See Section 9 of the Celestia Privacy Policy for more information.

4. Certain Additional Representations

4.1 Receipt of Rewards Entirely for Own Account. Your eligibility to receive ITN Rewards is made in reliance upon your representation to the Company, which by your agreement to these Terms you hereby confirm, that any ITN Rewards you receive will be for your own account, not as a nominee or agent, and not with a view to the resale or distribution of any part thereof, and that you have no present intention of selling, granting any participation in, or otherwise distributing the same. By agreeing to these Terms, you further represent that you do not presently have any contract, undertaking, agreement or arrangement with any person to sell, transfer or grant participations to such person or to any third person, with respect to any ITN Rewards. If you are agreeing to these terms on behalf of an entity, that entity has not been formed for the specific purpose of obtaining the ITN Rewards.

4.2 Disclosure of Information. Your eligibility to receive ITN Rewards is made in reliance upon your representation to the Company, which by your agreement to these Terms you hereby confirm, that you have sufficient knowledge of and experience in business and financial matters to be able to evaluate the risks and merits of your participation in the Program and of any ITN Rewards and are able to bear the risks thereof. You hereby affirm that you have not relied on any representations or warranties made by the Company related to the Program, including, but not limited to, conversations of any kind, whether through oral or electronic communication, or any white paper.

4.3 Compliance with United States Securities Laws. You understand that the ITN Rewards have not been, and will not be, registered under the 1933 Act or any applicable state securities laws. You acknowledge that the availability of an exemption from the registration provisions of the Securities Act and other applicable state securities laws depends upon, among other things, the bona fide nature of your intent as described in Section 4.1 above and with respect to the accuracy of your representations as expressed throughout these Terms. You understand that the ITN Rewards may be deemed "restricted securities" under applicable United States federal and state securities laws and that, pursuant to these laws, you may be restricted from transferring any ITN Rewards unless they are registered with the Securities and Exchange Commission and qualified by state authorities, or an exemption from such registration and qualification requirements is available. You acknowledge that the Company does not undertake any obligation to register or qualify the ITN Rewards for resale, and exemptions from registration and qualification may not be available or may not permit you to transfer all or any of the ITN Rewards in the amounts or at the times proposed by you. You further acknowledge that if an exemption from registration or qualification is available, such exemption may be conditioned on various requirements including, but not limited to, the time and manner of sale, the holding period for the ITN Rewards, and on other factors outside of your control, for which the Company makes no assurances and may not be able to satisfy.

4.4 Compliance with Liechtenstein Security Law. You understand that nothing in these Terms will be deemed to constitute a prospectus of any sort in Liechtenstein or in any jurisdiction in the EU; nor does it in any way pertain to a public offering or a solicitation of an offer to buy any securities in Liechtenstein or in any jurisdiction in the EU.

4.5 No Public Market. You understand that no public market now exists for the ITN Rewards, and that the Company has not made any assurances that a public market will ever exist for the ITN Rewards.

4.6 No Solicitation. At no time were you presented with or solicited by any publicly issued or circulated newspaper, mail, radio, television or other form of general advertising or solicitation in connection with any invitation to participate in the Program or offer of the ITN Rewards.

4.7 Other Applicable Laws. You hereby represent that you have satisfied yourself as to the full observance of the laws of your jurisdiction in connection with any invitation to participate in the Program, receipt of ITN Awards, and other use of these Terms, including (a) the legal requirements within your jurisdiction for participating in the Program and receiving ITN Rewards, (b) any foreign exchange restrictions applicable to such participation or receipt, (c) any governmental or other consents that may need to be obtained, and (d) the income tax and other tax consequences, if any, that may be relevant to the receipt, holding, sale, or transfer of the ITN Rewards. Your participation in the Program and continued beneficial ownership of ITN Rewards will not violate any applicable securities or other laws of your jurisdiction.

4.8 Non-US Transaction. You are not a U.S. Person as defined in Rule 902(k) of Regulation S under the 1933 Act. The offer of the ITN Rewards to you was made in an offshore transaction (as defined in Rule 902(h) of Regulation S), no directed selling efforts (as defined in Rule 902(c) of Regulation S) were made in the United States, and you are not obtaining the ITN Rewards for the account or benefit of any U.S. Person.

4.9 Transfer Restrictions. You will not, during the Restricted Period (as defined below) offer or sell any of the ITN Rewards (or create or maintain any derivative position equivalent thereto) in the United States, to or for the account or benefit of a U.S. Person or other than in accordance with Regulation S. The Company reserves the right to impose additional transfer restrictions with respect to the ITN Rewards in its sole discretion.

4.10 Subsequent Sales. You will, after the expiration of the applicable Restricted Period, only offer, sell, pledge or otherwise transfer the ITN Rewards (or create or maintain any derivative position equivalent thereto) pursuant to registration under the 1933 Act or any available exemption therefrom and, in any case, in accordance with applicable state securities laws.

4.11 Legends. You acknowledge and agree that the ITN Rewards will be deemed to bear the following legends: (a) any legend required by the securities laws of any state or country to the extent such laws are applicable to the ITN Rewards represented by the certificate so legended, and (b): the following legend (and even without such legend the following restrictions apply):

THE ITN REWARDS HAVE NOT BEEN REGISTERED UNDER THE ACT WITH THE UNITED STATES SECURITIES AND EXCHANGE COMMISSION, AND THE COMPANY DOES NOT INTEND TO REGISTER THEM. THE ITN REWARDS HAVE BEEN OBTAINED TO HOLD FOR THE LONG TERM AND NOT WITH A VIEW TO, OR IN CONNECTION WITH, THE SALE OR DISTRIBUTION THEREFOR. PRIOR TO THE ONE YEAR ANNIVERSARY FROM THE TERMINATION OF THE ITN REWARD PROGRAM (THE "PROGRAM COMPLETION DATE" AND SUCH ONE YEAR PERIOD, THE "RESTRICTED PERIOD"), THE ITN REWARDS MAY NOT BE OFFERED OR SOLD (INCLUDING OPENING A SHORT POSITION IN SUCH ITN REWARDS) IN THE UNITED STATES OR TO U.S. PERSONS AS DEFINED BY RULE 902(k) ADOPTED UNDER THE ACT, OTHER THAN TO DISTRIBUTORS, UNLESS THE ITN REWARDS ARE REGISTERED UNDER THE ACT, OR AN EXEMPTION FROM THE REGISTRATION REQUIREMENTS OF THE ACT IS AVAILABLE. RECIPIENTS OF ITN REWARDS PRIOR TO THE ONE YEAR ANNIVERSARY OF THE PROGRAM COMPLETION DATE MAY SELL SUCH ITN REWARDS ONLY PURSUANT TO AN EXEMPTION FROM REGISTRATION UNDER THE ACT OR OTHERWISE IN ACCORDANCE WITH THE PROVISIONS OF REGULATION S OF THE ACT, OR IN TRANSACTIONS EFFECTED OUTSIDE OF THE UNITED STATES PROVIDED THEY DO NOT SOLICIT (AND NO ONE ACTING ON THEIR BEHALF SOLICITS) PURCHASERS IN THE UNITED STATES OR OTHERWISE ENGAGE(S) IN SELLING EFFORTS IN THE UNITED STATES AND PROVIDED THAT HEDGING TRANSACTIONS INVOLVING THESE ITN REWARDS MAY NOT BE CONDUCTED UNLESS IN COMPLIANCE WITH THE ACT. A HOLDER OF THE ITN REWARDS WHO IS A DISTRIBUTOR, DEALER, SUB-UNDERWRITER OR OTHER SECURITIES PROFESSIONAL, IN ADDITION, CANNOT PRIOR TO THE ONE YEAR ANNIVERSARY OF THE PROGRAM COMPLETION DATE SELL THE ITN REWARDS TO A U.S. PERSON AS DEFINED BY RULE 902(k) OF REGULATION S UNLESS THE ITN REWARDS ARE REGISTERED UNDER THE ACT OR AN EXEMPTION FROM REGISTRATION UNDER THE ACT IS AVAILABLE.

',37),i=[n];function s(c,l,d,h,u,p){return t(),o("div",null,i)}const T=e(r,[["render",s]]);export{m as __pageData,T as default}; diff --git a/assets/community_itn-tos.md.fda6f837.lean.js b/assets/community_itn-tos.md.fda6f837.lean.js new file mode 100644 index 00000000000..ca78a071b1a --- /dev/null +++ b/assets/community_itn-tos.md.fda6f837.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.51d6c45b.js";const m=JSON.parse('{"title":"SUPPLEMENTAL INCENTIVIZED TESTNET TERMS","description":"These Terms govern your ability to participate in the Incentivized Testnet Award Program.","frontmatter":{"description":"These Terms govern your ability to participate in the Incentivized Testnet Award Program.","lastUpdated":false,"editLink":false,"head":[["meta",{"name":"og:title","content":"SUPPLEMENTAL INCENTIVIZED TESTNET TERMS | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/itn-tos.md","filePath":"community/itn-tos.md","lastUpdated":1699020127000}'),r={name:"community/itn-tos.md"},n=a("",37),i=[n];function s(c,l,d,h,u,p){return t(),o("div",null,i)}const T=e(r,[["render",s]]);export{m as __pageData,T as default}; diff --git a/assets/community_modular-meetup-guide.md.ad87bef9.js b/assets/community_modular-meetup-guide.md.ad87bef9.js new file mode 100644 index 00000000000..17a90d4c0f1 --- /dev/null +++ b/assets/community_modular-meetup-guide.md.ad87bef9.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as i,Q as o}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"Modular Meetup guide","description":"A guide that helps in organizing a successful Modular Meetup.","frontmatter":{"description":"A guide that helps in organizing a successful Modular Meetup.","head":[["meta",{"name":"og:title","content":"Modular Meetup guide | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/modular-meetup-guide.md","filePath":"community/modular-meetup-guide.md","lastUpdated":1698091500000}'),a={name:"community/modular-meetup-guide.md"},n=o('

Modular Meetup guide

These are recommended steps you can follow that can help you organize a successful Modular Meetup. You aren’t required to follow the entire checklist but it is available to help aid you in your journey to organize a Modular Meetup.

Before the Meetup

  1. Determine the meetup topic and objectives.
  2. Select a suitable date and time for the event.
  3. Secure a convenient and accessible venue that can accommodate the expected number of attendees.
  4. Finalize a schedule or agenda for the event, including speakers, presentations, and activities.
  5. Prepare and send out invitations to potential attendees using various channels (email, social media, community platforms, etc.).
  6. Create and share promotional materials (graphics, blog posts, etc.) to raise awareness about the meetup.
  7. Confirm speakers, including their availability, presentation topics, and technical requirements. If you are working from a Speaker List, please reach out to Nat for any support needed on coordination with speakers.
  8. Arrange any necessary equipment, such as microphones, projectors, and whiteboards.
  9. Plan and arrange refreshments, if applicable.
  10. Prepare and print any required materials, like agendas, name tags, and feedback forms.
  11. Coordinate with volunteers or team members to help manage the event.
  12. Set up a registration system or RSVP process to track attendance.
  13. Send out reminder messages to registered attendees prior to the event.
  14. Conduct a final review of the event logistics, including venue setup, equipment functionality, and volunteer roles.

During the Meetup

  1. Set up the venue, including arranging seating, preparing equipment, and displaying any promotional materials.
  2. Designate a registration area and ensure that someone is available to greet and check-in attendees.
  3. Welcome the attendees and provide an overview of the event schedule.
  4. Introduce speakers and facilitate any presentations or discussions.
  5. Encourage networking and interaction among attendees during breaks or dedicated activities.
  6. Capture the event with photos or videos for future promotions and documentation.
  7. Collect feedback from attendees using forms, online surveys, or informal conversations if needed.

After the Meetup

  1. Send follow-up messages to attendees, thanking them for their participation and soliciting additional feedback.
  2. Evaluate the success of the event by reviewing feedback and assessing key performance indicators (e.g., attendance, engagement, etc.).
  3. Analyze and document lessons learned, identifying areas for improvement in future meetups.
  4. Share event highlights, photos, and key takeaways with the community through social media, blog posts, or newsletters.
  5. Connect with speakers, attendees, and volunteers to maintain and strengthen relationships.
  6. Begin planning for the next meetup, applying insights gained from the previous event to enhance the experience for future attendees.

Logistics and guidance

These are provided guidelines for different components of kickstarting a Modular Meetup to help you get started on the logistics of organizing a meetup. A lot of those guidelines are thanks to the wonderful write ups in Ethereum Meetup Support Program and Elastic Community Organizer Guides.

Venue

Selecting the perfect venue for your Modular Meetup is essential in creating a welcoming atmosphere for the Celestia community.

  1. Startup Incubators
    • Often they would have spaces for meetups.
  2. Libraries
    • Libraries can normally have spaces for meetups at little to no costs
  3. Co-working spaces:
    • Co-working spaces might be able to offer necessary equipment like microphones, projectors, and whiteboards.
  4. Restaurants:
    • Restaurants might have private rooms for larger groups of people that you can book, depending on the restaurant.
  5. Universities and Blockchain Clubs:
    • Partnering with universities and their local university clubs can offer you a lot of spaces for meetups, as well as necessary equipments if needed for giving a talk.
  6. Other options:
    • If you're having difficulty finding a venue, don't hesitate to ask for help within the Celestia community. Your fellow organizers and attendees may have valuable suggestions or connections.

An ideal venue should offer:

  • Sufficient seating for the audience
  • A projector and screen
  • A microphone (especially for meetups with 10+ attendees)
  • Optional: A whiteboard for speakers or workshops (not always necessary, but a nice addition)

Consider recording the event, even with a smartphone, to share with the community later. If the speaker uses a microphone, their voice will be more audible in the video. We might be able to post those recordings of the meetup talks on a meetup youtube channel after.

Remember to negotiate on costs and seek discounts, emphasizing that your meetup benefits the community.

Collaborate with your venue provider to explore options like borrowing recording equipment, tripods, or even having them record the event for you. Some providers may offer these services for free or at a reduced cost.

Catering and refreshments

Providing refreshments or catering for your Modular Meetup enhances the overall experience and encourages networking among attendees.

  1. Determine your budget
    • Identify the amount you can allocate for food and drinks at your event which will help you determine how much you can bring in refreshments. You can also contact the Celestia Devrel team for ideas and support. Costs for refreshments and drinks can vary depending on your location, so be mindful to be flexible on your plans according to your specific location and budget.
  2. Offer a variety of refreshments
    • If you're ordering in, pizza and finger foods work well, but you can also have more budget-friendly options for food.
    • Offering drinks like beer, soda, or lemonade are great, but water also works. Keep in mind that not everyone drinks alcohol so it’s not a requirement. But having at least water and plastic cups works well.
  3. Plan ahead
    • Order refreshments 1 or 2 days in advance and schedule delivery to avoid last-minute stress during the event.

By following these recommendations, you'll be able to provide enjoyable refreshments for your Celestia Modular Meetup attendees while fostering a friendly and engaging atmosphere.

Audience

Understanding your audience and estimating attendance are crucial for organizing successful Modular Meetups for Celestia.

  1. Research the local tech scene:
    • Investigate the types of meetups and events popular in your area. Attend other technology-focused events to get a sense of the audience size and interests. This information will help you tailor your meetup to attract a larger audience.
  2. Assess the availability of speakers:
    • Before organizing a meetup, ensure that you have access to a pool of knowledgeable speakers. If you anticipate difficulty in securing speakers, consider joining forces with an existing meetup group or speaking at other events before launching a new group. This approach will help spread the word about your planned Modular Meetup and gauge interest. Speakers are covered in the following section with a reference to Speaker List offered by the Modular Meetup program.
  3. Establish connections with other user groups:
    • Forge relationships with other tech-focused meetup groups to mutually promote each other's events, potentially increasing attendance.
  4. Set a regular routine for your meetup:
    • Communicate how often you plan to hold meetups, whether it's monthly or quarterly, to help attendees manage their expectations and maintain their interest.
  5. Organize casual meetups:
    • If there's a gap between more formal events, arrange casual meetups at pubs or cafes to keep people engaged and connect with potential speakers for future events.
  6. Estimate the number of attendees:
    • Consider factors such as the size of your city, the popularity of the topic, and the appeal of the event description when estimating attendance.

Speakers

Securing engaging and knowledgeable speakers is key to hosting an exceptional Celestia Modular Meetup.

  1. Define your event topic
    • Determine the theme of your event, focusing on areas such as DeFi, Gaming, NFTs, coding workshops, protocol changes, rollups, data availability, or other topics relevant to the Modular ecosystem. This will help you find speakers with expertise in the chosen subject.
  2. Utilize the Speaker List provided by the Modular Meetup program
    • As a meetup organizer, you have access to a curated list of talented speakers from Celestia Labs and the broader Modular ecosystem. This valuable resource can connect you with experts who can share their knowledge with your meetup attendees.
  3. Aim for multiple speakers
    • Ideally, invite 2-3 speakers to your event, allotting 20-40 minutes per talk. Schedule short breaks between presentations to maintain audience engagement.
  4. Organize the speaker lineup
    • Discuss the topics and slides with your speakers before the event to ensure a smooth flow. Arrange the talks in a logical order, saving the most impactful presentation for last.
  5. Invite local speakers
    • For your first meetup, consider presenting an introduction to the Celestia ecosystem or a specific area of interest. Encourage local experts or enthusiasts to speak at future meetups. This approach fosters community involvement and helps build a network of potential speakers.
  6. Seek speaker referrals
    • Ask your current speakers, attendees, or other meetup organizers for referrals. Personal connections often lead to discovering new speakers with valuable insights.
  7. Leverage your meetup discussion board
    • Post a call for speakers on your meetup discussion board to reach out to potential presenters within your community. Be clear about the event theme and requirements to attract relevant speakers.
  8. Offer incentives and appreciation
    • Reward speakers with tokens of gratitude, such as T-shirts, gifts, or public recognition, to show your appreciation for their contribution to the meetup. The Celestia Labs Devrel team can help with swag logistics if needed.

Sponsors

Finding sponsors for your meetup can be challenging, but securing financial support is crucial for covering costs related to venue and refreshments.

  1. Leverage Celestia Labs' support
    • Celestia Labs may offer co-sponsorship for your meetup. However, they also encourage organizers to find local co-sponsors to help cover costs and create a more sustainable event.
  2. Offer value to your sponsors:
    • Show potential sponsors how partnering with your event will benefit them. Include their logo on event banners, mention them in the event description, and give them a shoutout at the beginning of the event. If they desire, allow them to place a rollup banner at the venue.
  3. Reach out to your network
    • Ask friends and acquaintances if they know of coworking spaces, schools, universities, or companies interested in sponsorship. A personal connection can significantly increase the chances of securing support.
  4. Approach speakers for sponsorship
    • Request speakers or their affiliated projects to contribute towards the event's expenses, such as catering costs. This can be an effective way to obtain additional funding.
  5. Create a sponsorship deck
    • Develop a compelling sponsorship deck to pitch your event to tech companies in your city or potential online sponsors. This presentation should highlight the benefits of supporting your event and showcase past successful meetups.
  6. Research local companies
    • Investigate businesses in your area that may be interested in sponsoring your event. Tailor your pitch to align with their industry and demonstrate how the meetup can benefit their company.
  7. Engage sponsors during the event
    • Allow sponsors to briefly address the audience, participate in Q&A sessions, or mention their hiring needs. Remember to avoid sales pitches, as they can negatively impact the meetup experience.
  8. Thank your sponsors
    • Express gratitude to your sponsors at the beginning and end of the meetup. Acknowledging their support encourages continued collaboration and enhances the credibility of your event.

Communications and marketing

Effectively marketing and announcing your Modular Meetup is essential for attracting attendees and ensuring a successful event.

  1. Plan your announcement
    • Announce your meetup at least two weeks prior. This gives your audience ample time to prepare and increases the likelihood of their attendance.
  2. Utilize Celestia Labs' resources
    • Celestia Labs can help co-promote your meetup on social media and Discord. They can also add your event to their online calendar, email local contacts, and share the event via their Developer Relations Team on Twitter. Don't hesitate to reach out to them for assistance.
  3. Share on social media
    • Promote your event on Twitter and any other popular social media platforms in your area. Tag speakers, sponsors, and use relevant hashtags to increase visibility.
  4. Leverage local community groups
    • Post your event in local Telegram, Discord, or other community groups relevant to your city. These groups often have many members who may be interested in attending your event.
  5. Engage speakers and the venue
    • Maintain communication with speakers and the venue to ensure any changes can be announced in advance. Trust is critical, so avoid canceling planned meetups whenever possible.
  6. Send reminders via meetup.com (optional)
    • After announcing your event, send a warm invite to your meetup group members through meetup.com. Additionally, send a reminder email one day before the event to encourage attendance.
  7. Share with friends and family
    • Invite your friends, family, and acquaintances to the meetup, as they may help spread the word or know someone interested in the event.
  8. Utilize conference groups
    • If you've attended conferences, share your meetup in the associated Telegram or social media groups, as there may be members nearby who would be interested in attending.

Recording

Recording and live-streaming your Modular Meetup can greatly benefit those who cannot attend in person and expand the reach of your event.

  1. Plan for recording
    • Consider recording your meetup to create additional learning resources and share the knowledge with a broader audience.
  2. Consider live-streaming
    • Live-streaming your event on platforms like Twitch, YouTube, or Twitter allows remote participants to watch and engage in the meetup. This can also boost your event's reach and create a sense of inclusion for those who couldn't attend in person.
  3. Coordinate with Celestia Labs
    • Celestia Labs can potentially help cover recording costs and promote recorded meetups. Reach out to them at meetups@celestia.org to discuss available options and support. They can also assist with finding local contacts for recording if you don't have one already.
  4. Utilize available resources
    • If you have a small budget, consider allocating some funds for recording and live-streaming your event. Look for local professionals or affordable equipment rentals to ensure high-quality recordings.
  5. Share recordings on Celestia Labs' channels
    • Celestia Labs can help promote recorded meetups by sharing videos in their meetups playlist on YouTube and hosting them on their website. Ensure you coordinate with Celestia Labs to provide them with the recorded video.
  6. Promote recorded content
    • Share the recordings on your social media channels and meetup group after the event. This helps attendees revisit the content and allows those who couldn't attend to learn from the talks.

Utilizing Meetup.com platform

Celestia Labs is committed to supporting your meetup efforts by helping you with meetup.com, from setting up the group to covering organizer dues. Below is a detailed overview of how Celestia Labs can assist you.

  1. Meetup.com organizer dues
    • Celestia Labs is happy to cover organizer dues for meetup.com. To get started, send an email to meetups@celestia.org and let them help you with the process.
  2. Co-organizing existing groups
    • If you already have a meetup.com group but need Celestia Labs' assistance with organizing or covering dues, email meetups@celestia.org to add one of their employees as the Organizer.
  3. Setting up groups on Meetup.com
    • Celestia Labs will help you set up your meetup group, including logos, naming, custom URLs, group description, and other essential details. They provide a unique logo for Celestia Modular Meetups, which you can use for your group.
  4. User group naming
    • Celestia Labs recommends naming your group "Celestia Modular Meetup" to encompass various aspects of the Celestia community and create a consistent brand.
  5. Custom URL for the group
    • Meetup.com allows you to create a custom URL for your user group. Celestia Labs encourages consistent URLs across regions, making it easy for people to find your meetup group.
  6. Group description
    • Celestia Labs has a standard group description to ensure consistency across all meetups. However, if you'd like to customize it, let them know.
  7. New member intake questions
    • To improve the quality of your meetups, Celestia Labs suggests a set of intake questions for new members. The answers can help you better understand their needs and interests, allowing you to plan engaging meetups.
  8. Welcome message
    • Celestia Labs provides a welcome message for new members joining your group. If you'd like to customize this message, let them know.

Onboarding questions for community members joining a Modular Meetup

  1. Q1: How did you hear about this Modular Meetup?
    • Why: This information helps us understand the most effective channels for recruiting new participants to the meetup.
  2. Q2: What do you hope to gain by participating in this meetup? (e.g., networking, learning about Celestia, learning about Modularity, finding a job, etc.)
    • Why: This information helps us cater to the needs of the user group members. If most attendees are looking for networking opportunities, we can schedule casual meetups alongside informative talks.
  3. Q3: Are you currently using or planning to use Celestia, Rollkit, Celestia’s Node API or any of the rollups deployed on Celestia? Tell us all about it. We will use your response to help us better understand what talks would be most beneficial to the group.
    • Why: Responses to this question help us determine the most relevant talks for the group members. If we discover that most participants are interested in a specific area, we can tailor the talks accordingly.
  4. Q4: We love Celestia and the modular ecosystem, but we also appreciate other Web3 technologies. What other topics would you like to hear about? (e.g., Infrastructure, Data Storage, DID, MEV)
    • Why: Knowing our group participants' interests in other technical areas helps us understand what related topics would be useful and valuable to the group, which in turn helps when recruiting speakers.
  5. Q5: Would you be interested in speaking at a future meetup? We welcome 2-minute lightning talks to 1-hour deep dives. Would you be interested in hosting a meetup? If you answer yes to this question, the group organizers will contact you to follow up.
    • Why: One of the challenges in hosting regular meetups is finding speakers. We hope this question will identify people eager to share their stories and expertise with the group, making the lives of organizers easier.
',42),r=[n];function s(l,u,c,d,h,p){return t(),i("div",null,r)}const f=e(a,[["render",s]]);export{g as __pageData,f as default}; diff --git a/assets/community_modular-meetup-guide.md.ad87bef9.lean.js b/assets/community_modular-meetup-guide.md.ad87bef9.lean.js new file mode 100644 index 00000000000..bc1c20ca228 --- /dev/null +++ b/assets/community_modular-meetup-guide.md.ad87bef9.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as i,Q as o}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"Modular Meetup guide","description":"A guide that helps in organizing a successful Modular Meetup.","frontmatter":{"description":"A guide that helps in organizing a successful Modular Meetup.","head":[["meta",{"name":"og:title","content":"Modular Meetup guide | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/modular-meetup-guide.md","filePath":"community/modular-meetup-guide.md","lastUpdated":1698091500000}'),a={name:"community/modular-meetup-guide.md"},n=o("",42),r=[n];function s(l,u,c,d,h,p){return t(),i("div",null,r)}const f=e(a,[["render",s]]);export{g as __pageData,f as default}; diff --git a/assets/community_modular-meetup-intro.md.00a93576.js b/assets/community_modular-meetup-intro.md.00a93576.js new file mode 100644 index 00000000000..1423a321945 --- /dev/null +++ b/assets/community_modular-meetup-intro.md.00a93576.js @@ -0,0 +1 @@ +import{_ as e,o,c as a,Q as t}from"./chunks/framework.51d6c45b.js";const r="/img/Celestia_Modular_meetup2.jpg",f=JSON.parse('{"title":"Celestia Modular Meetup program","description":"The ultimate guide for Modular Meetup organizers!","frontmatter":{"description":"The ultimate guide for Modular Meetup organizers!","head":[["meta",{"name":"og:title","content":"Celestia Modular Meetup program | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/modular-meetup-intro.md","filePath":"community/modular-meetup-intro.md","lastUpdated":1709744743000}'),i={name:"community/modular-meetup-intro.md"},n=t('

Celestia Modular Meetup program

Modular Meetup Banner

Welcome to the ultimate guide for Modular Meetup organizers! This collection of resources is designed for those enthusiastic about fostering grassroots Modular Meetups with support from Celestia around the world.

Program description

The Celestia Modular Meetup Program aims to empower meetup organizers, providing education and support, and encouraging collaboration within the Web3 ecosystem. This rapidly growing community has already achieved incredible success with the first Modular Meetup in Lisbon, and will grow from there.

Join fellow enthusiasts, engage in enlightening discussions, and make the most of the insightful resources provided. These resources are designed to serve as a go-to playbook for meetup organizers, especially when starting your journey.

Important info

Celestia.org Community Code of Conduct

The purpose of our Community Code of Conduct is to foster an inclusive, welcoming, and supportive environment for everyone participating in Celestia community events. We're all here to learn from each other, expand our skillsets, and enjoy a positive experience together.

All meetup attendees, speakers, sponsors, and volunteers, including the event organizing team, are kindly asked to adhere to the following Code of Conduct. Organizers will respectfully enforce this code throughout the event. We genuinely appreciate the cooperation of all participants in maintaining a safe and empowering space for everyone.

Signup form

To become part of the program, please complete the registration form.

Following the review and approval of your submission, you will receive an email confirmation and an invitation to participate in the upcoming Modular Meetup call. Furthermore, you will be granted access to the exclusive Discord channel labeled "#modular-meetup" on our Discord server. Please take note that joining our Discord is a prerequisite for channel access. It's essential to recognize that this program is tailored for dedicated organizers with a genuine interest in nurturing their local modular ecosystem community.

Emails

As a participant in the Celestia Modular Meetup Program, you can expect to receive the following emails:

  1. Welcome email with links to calendar events and Discord channel
  2. Monthly Catch-up call invites
  3. Recap emails with notes from calls

Discord

Your active participation is key to unlocking the full potential of this vibrant community. Our primary communication tool is Discord, providing an engaging platform to connect with fellow organizers:

Materials

As a meetup organizer, you'll gain access to the Celestia Modular Meetup Program's list of resources. This collection should become your trusted companion in organizing events. Drawing upon the wisdom of seasoned event organizers, this resource is available for you and your co-organizers to explore and learn.

',23),s=[n];function l(u,c,d,p,m,h){return o(),a("div",null,s)}const y=e(i,[["render",l]]);export{f as __pageData,y as default}; diff --git a/assets/community_modular-meetup-intro.md.00a93576.lean.js b/assets/community_modular-meetup-intro.md.00a93576.lean.js new file mode 100644 index 00000000000..0f131f69011 --- /dev/null +++ b/assets/community_modular-meetup-intro.md.00a93576.lean.js @@ -0,0 +1 @@ +import{_ as e,o,c as a,Q as t}from"./chunks/framework.51d6c45b.js";const r="/img/Celestia_Modular_meetup2.jpg",f=JSON.parse('{"title":"Celestia Modular Meetup program","description":"The ultimate guide for Modular Meetup organizers!","frontmatter":{"description":"The ultimate guide for Modular Meetup organizers!","head":[["meta",{"name":"og:title","content":"Celestia Modular Meetup program | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/modular-meetup-intro.md","filePath":"community/modular-meetup-intro.md","lastUpdated":1709744743000}'),i={name:"community/modular-meetup-intro.md"},n=t("",23),s=[n];function l(u,c,d,p,m,h){return o(),a("div",null,s)}const y=e(i,[["render",l]]);export{f as __pageData,y as default}; diff --git a/assets/community_modular-meetup-toolkit.md.056ae043.js b/assets/community_modular-meetup-toolkit.md.056ae043.js new file mode 100644 index 00000000000..cfad4b7874d --- /dev/null +++ b/assets/community_modular-meetup-toolkit.md.056ae043.js @@ -0,0 +1 @@ +import{_ as e,o,c as i,Q as t}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"Modular Meetup Toolkit","description":"A toolkit for Modular Meetups.","frontmatter":{"description":"A toolkit for Modular Meetups.","head":[["meta",{"name":"og:title","content":"Modular Meetup Toolkit | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/modular-meetup-toolkit.md","filePath":"community/modular-meetup-toolkit.md","lastUpdated":1698091500000}'),l={name:"community/modular-meetup-toolkit.md"},a=t('

Modular Meetup Toolkit

Welcome to the Modular Meetups Toolkit! This kit is designed to help you run successful meetups for the Celestia community. As the first modular blockchain, Celestia offers a lot to discuss and explore. This kit includes useful resources and materials to help you plan and execute your meetups effectively.

Celestia branding guidelines

  • Brand kit
    1. Includes logo files, color schemes, typography, icons and illustrations

Sample “Introduction to Modularity” workshop presentation

  • Sample presentation - introduction to modularity
  • Summary: This is an overview presentation on Modular blockchains and dives deep into Celestia core technologies.
  • The sample presentation covers:
    1. What are modular blockchains?
    2. The benefits of modular over monolithic blockchains
    3. Introduction to Celestia: The first modular blockchain
    4. The concept of Data Availability Sampling
    5. Sovereign Rollups
    6. Q&A session

Sample “Run a Celestia light node” workshop presentation

  • Sample presentation - run a light node
  • Summary: This is an overview presentation goes over running a Celestia light node. You can find existing video presentations for this here:
  • The sample presentation covers:
    1. What is a Celestia light node?
    2. The role of light nodes in the Celestia ecosystem
    3. Setting up a light node: hardware and software requirements
    4. Step-by-step guide on how to run a Celestia light node
    5. Troubleshooting common issues
    6. Best practices for maintaining a light node
    7. Q&A session

Sample “Deploy a Sovereign Rollup” workshop presentation

  • Sample presentation - deploy a sovereign rollup
  • Summary: This is an overview presentation on deploying a sovereign rollup with Rollkit on Celestia. You can find existing video presentations for this here:
  • The sample presentation covers:
    1. What is a sovereign rollup?
    2. The role of sovereign rollups in the Celestia ecosystem
    3. Introduction to Rollkit
    4. Setting up a sovereign rollup: hardware and software requirements
    5. Q&A session

Sample “Modular Meetup Introduction” workshop presentation

Swag logistics

With this Modular Meetups Organizer Kit, you’ll have everything you need to plan and execute engaging, informative, and successful meetups for the Celestia community. Happy organizing!

',15),r=[a];function n(s,p,u,d,h,c){return o(),i("div",null,r)}const f=e(l,[["render",n]]);export{g as __pageData,f as default}; diff --git a/assets/community_modular-meetup-toolkit.md.056ae043.lean.js b/assets/community_modular-meetup-toolkit.md.056ae043.lean.js new file mode 100644 index 00000000000..4bb1aec439b --- /dev/null +++ b/assets/community_modular-meetup-toolkit.md.056ae043.lean.js @@ -0,0 +1 @@ +import{_ as e,o,c as i,Q as t}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"Modular Meetup Toolkit","description":"A toolkit for Modular Meetups.","frontmatter":{"description":"A toolkit for Modular Meetups.","head":[["meta",{"name":"og:title","content":"Modular Meetup Toolkit | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/modular-meetup-toolkit.md","filePath":"community/modular-meetup-toolkit.md","lastUpdated":1698091500000}'),l={name:"community/modular-meetup-toolkit.md"},a=t("",15),r=[a];function n(s,p,u,d,h,c){return o(),i("div",null,r)}const f=e(l,[["render",n]]);export{g as __pageData,f as default}; diff --git a/assets/community_overview.md.c9f170b3.js b/assets/community_overview.md.c9f170b3.js new file mode 100644 index 00000000000..d16bef4491f --- /dev/null +++ b/assets/community_overview.md.c9f170b3.js @@ -0,0 +1 @@ +import{_ as o,o as i,c as a,k as e,a as t}from"./chunks/framework.51d6c45b.js";const y=JSON.parse('{"title":"Community overview","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Community overview | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/overview.md","filePath":"community/overview.md","lastUpdated":1698091500000}'),n={name:"community/overview.md"},r=e("h1",{id:"community-overview",tabindex:"-1"},[t("Community overview "),e("a",{class:"header-anchor",href:"#community-overview","aria-label":'Permalink to "Community overview"'},"​")],-1),c=e("p",null,"This section will highlight all the different resources and activities for the Celestia community.",-1),s=e("p",null,[t("Here you will find links to our "),e("a",{href:"./calendar"},"community calendar"),t(", "),e("a",{href:"./coc"},"Code of Conduct"),t(" and other community-related resources.")],-1),m=[r,c,s];function l(d,u,v,h,_,f){return i(),a("div",null,m)}const w=o(n,[["render",l]]);export{y as __pageData,w as default}; diff --git a/assets/community_overview.md.c9f170b3.lean.js b/assets/community_overview.md.c9f170b3.lean.js new file mode 100644 index 00000000000..d16bef4491f --- /dev/null +++ b/assets/community_overview.md.c9f170b3.lean.js @@ -0,0 +1 @@ +import{_ as o,o as i,c as a,k as e,a as t}from"./chunks/framework.51d6c45b.js";const y=JSON.parse('{"title":"Community overview","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Community overview | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/overview.md","filePath":"community/overview.md","lastUpdated":1698091500000}'),n={name:"community/overview.md"},r=e("h1",{id:"community-overview",tabindex:"-1"},[t("Community overview "),e("a",{class:"header-anchor",href:"#community-overview","aria-label":'Permalink to "Community overview"'},"​")],-1),c=e("p",null,"This section will highlight all the different resources and activities for the Celestia community.",-1),s=e("p",null,[t("Here you will find links to our "),e("a",{href:"./calendar"},"community calendar"),t(", "),e("a",{href:"./coc"},"Code of Conduct"),t(" and other community-related resources.")],-1),m=[r,c,s];function l(d,u,v,h,_,f){return i(),a("div",null,m)}const w=o(n,[["render",l]]);export{y as __pageData,w as default}; diff --git a/assets/community_speaker-list.md.fd89d412.js b/assets/community_speaker-list.md.fd89d412.js new file mode 100644 index 00000000000..909566bcb9f --- /dev/null +++ b/assets/community_speaker-list.md.fd89d412.js @@ -0,0 +1 @@ +import{_ as t,o as a,c as s,k as e,a as i}from"./chunks/framework.51d6c45b.js";const b=JSON.parse('{"title":"Speaker list","description":"Invite a speaker from the ecosystem to your Modular Meetup.","frontmatter":{"description":"Invite a speaker from the ecosystem to your Modular Meetup.","head":[["meta",{"name":"og:title","content":"Speaker list | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/speaker-list.md","filePath":"community/speaker-list.md","lastUpdated":1698091500000}'),o={name:"community/speaker-list.md"},r=e("h1",{id:"speaker-list",tabindex:"-1"},[i("Speaker list "),e("a",{class:"header-anchor",href:"#speaker-list","aria-label":'Permalink to "Speaker list"'},"​")],-1),n=e("p",null,"As a Modular Meetup organizer, we understand the importance of delivering engaging and informative content to your attendees. That's why we've created an exclusive Speaker List specifically tailored for organizers participating in the Modular Meetup Program. This resource gives you access to a curated selection of top-tier speakers who are passionate about Celestia and the modular ecosystem. Due to privacy, the list is not shared publicly but is accessible to participants of the Modular Meetup program when they create a meetup.",-1),l=e("p",null,"The Speaker List features experts from Celestia Labs, as well as prominent figures from the broader Celestia and modular communities. Each individual is well-versed in various aspects of the modular ecosystem, ensuring that your meetup attendees gain valuable insights and deepen their understanding of modular blockchains.",-1),c=e("p",null,"By joining the Modular Meetup Program, you can enjoy the benefits of our Speaker List and bring a touch of expertise to your events. The speakers can participate either in person or virtually, depending on location and timing.",-1),p=e("p",null,"You can expect benefits from the Speaker List including high-quality presentations, interactive Q&A sessions, and knowledge-sharing opportunities facilitated by the best and brightest in the Celestia ecosystem. With our Speaker List, you'll be able to create memorable and impactful Modular Meetups that foster genuine connections and promote growth within the community.",-1),d=[r,n,l,c,p];function u(h,m,f,g,y,k){return a(),s("div",null,d)}const v=t(o,[["render",u]]);export{b as __pageData,v as default}; diff --git a/assets/community_speaker-list.md.fd89d412.lean.js b/assets/community_speaker-list.md.fd89d412.lean.js new file mode 100644 index 00000000000..909566bcb9f --- /dev/null +++ b/assets/community_speaker-list.md.fd89d412.lean.js @@ -0,0 +1 @@ +import{_ as t,o as a,c as s,k as e,a as i}from"./chunks/framework.51d6c45b.js";const b=JSON.parse('{"title":"Speaker list","description":"Invite a speaker from the ecosystem to your Modular Meetup.","frontmatter":{"description":"Invite a speaker from the ecosystem to your Modular Meetup.","head":[["meta",{"name":"og:title","content":"Speaker list | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"community/speaker-list.md","filePath":"community/speaker-list.md","lastUpdated":1698091500000}'),o={name:"community/speaker-list.md"},r=e("h1",{id:"speaker-list",tabindex:"-1"},[i("Speaker list "),e("a",{class:"header-anchor",href:"#speaker-list","aria-label":'Permalink to "Speaker list"'},"​")],-1),n=e("p",null,"As a Modular Meetup organizer, we understand the importance of delivering engaging and informative content to your attendees. That's why we've created an exclusive Speaker List specifically tailored for organizers participating in the Modular Meetup Program. This resource gives you access to a curated selection of top-tier speakers who are passionate about Celestia and the modular ecosystem. Due to privacy, the list is not shared publicly but is accessible to participants of the Modular Meetup program when they create a meetup.",-1),l=e("p",null,"The Speaker List features experts from Celestia Labs, as well as prominent figures from the broader Celestia and modular communities. Each individual is well-versed in various aspects of the modular ecosystem, ensuring that your meetup attendees gain valuable insights and deepen their understanding of modular blockchains.",-1),c=e("p",null,"By joining the Modular Meetup Program, you can enjoy the benefits of our Speaker List and bring a touch of expertise to your events. The speakers can participate either in person or virtually, depending on location and timing.",-1),p=e("p",null,"You can expect benefits from the Speaker List including high-quality presentations, interactive Q&A sessions, and knowledge-sharing opportunities facilitated by the best and brightest in the Celestia ecosystem. With our Speaker List, you'll be able to create memorable and impactful Modular Meetups that foster genuine connections and promote growth within the community.",-1),d=[r,n,l,c,p];function u(h,m,f,g,y,k){return a(),s("div",null,d)}const v=t(o,[["render",u]]);export{b as __pageData,v as default}; diff --git a/assets/developers_arbitrum-bridge.md.7321b996.js b/assets/developers_arbitrum-bridge.md.7321b996.js new file mode 100644 index 00000000000..956b0118c9d --- /dev/null +++ b/assets/developers_arbitrum-bridge.md.7321b996.js @@ -0,0 +1 @@ +import{_ as r,o as t,c as o,Q as i}from"./chunks/framework.51d6c45b.js";const e="/arbitrum/bridge-overview-deposit-and-withdrawal-l3.png",a="/arbitrum/bridge-settings.png",n="/arbitrum/add-custom-chain-to-bridge.png",s="/arbitrum/live-orbit-chains.png",l="/arbitrum/bridge-in-start.png",p="/arbitrum/bridge-in-pending-txs.png",g="/arbitrum/bridge-in-settled-txs.png",d="/arbitrum/bridge-in-explorer-rollup-tx.png",u="/arbitrum/bridge-in-success.png",c="/arbitrum/bridge-in-sepolia-tx-explorer.png",b="/arbitrum/bridge-out-small-screenshot.png",h="/arbitrum/bridge-out-begin.png",m="/arbitrum/bridge-out-pending.png",f="/arbitrum/bridge-out-begin-overview.png",w="/arbitrum/bridge-out-rollup-tx-details.png",y="/arbitrum/bridge-out-logs-details-1.png",_="/arbitrum/bridge-out-logs-explorer-2.png",x="/arbitrum/bridge-out-claim-withdrawal.png",v="/arbitrum/bridge-out-claim-success-withdrawal.png",D=JSON.parse('{"title":"Bridging in and out of your Orbit rollup","description":"A guide on how to bridge in and out of your Arbitrum Orbit rollup.","frontmatter":{"description":"A guide on how to bridge in and out of your Arbitrum Orbit rollup.","head":[["meta",{"name":"og:title","content":"Bridging in and out of your Orbit rollup | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/arbitrum-bridge.md","filePath":"developers/arbitrum-bridge.md","lastUpdated":1713191709000}'),A={name:"developers/arbitrum-bridge.md"},S=i('

Bridging in and out of your Orbit rollup

This guide covers how to bridge in (deposit) and bridge out (withdrawal) from your Arbitrum Orbit L3 rollup.

This guide will cover bridging in and out of your Orbit rollup.

Below are two example transactions, one of a deposit and one of a withdrawal:

bridge-overview-deposit-and-withdrawal-l3

Bridge in (deposit) to your rollup

Step 1: Add your custom chain config

(1a) In the Arbitrum Bridge UI, click the menu dropdown in the top right. Select Settings.

bridge-settings

(1b) Under Developer Mode, select Turn on testnet mode. Add your custom chain config from outputInfo.json in the root of your orbit-setup-script directory.

add-testnet-orbit-chain

(1c) You'll then see the chain under Live Orbit Chains:

live-orbit-chains

Step 2: Deposit to your Orbit rollup

(2a) Choose an amount of Arbitrum Sepolia ETH to bridge into your rollup. Click Move funds to <YOUR_ROLLUP_NAME>, in this case Move funds to Arbitrum L3 Rollup. Approve the transaction in your wallet.

bridge-in-start

(2b) You'll then see it load in the Pending transactions tab:

bridge-in-pending

(2c) Shortly after, in the Settled transactions tab you can see the transaction status. Click Success.

bridge-in-settled-txs

(2d) View the transaction on your local instance of Blockscout for your rollup:

bridge-in-explorer-rollup-tx

(2e) Optionally, click See Details for an overview of your deposit:

bridge-in-success

(2f) From the details page, you can also see the transaction for your deposit on Arbitrum Sepolia:

bridge-in-sepolia-tx-explorer

Bridge out (withdrawal) from your rollup

Step 1: Choose an amount to withdraw from your rollup

(1a) In the Arbitrum Bridge UI, choose your origin chain to your Arbitrum L3 Rollup and the destination chain as Arbitrum Sepolia.

bridge-out-small-screenshot

(1b) Click Move funds to Arbitrum Sepolia and read the disclaimer, check the boxes, and click Continue.

bridge-out-begin

(1c) Optionally, set a reminder on your calendar so you don't forget.

(1d) After approving the transaction in your wallet, you'll be able to see the transaction in the Pending transactions tab:

bridge-out-pending

After approximately two hours, you will be able to proceed to Step 2: Claim your withdrawal.

(1e) Click See details to see an overview of your withdrawal:

bridge-out-begin-overview

(1f) Optionally, view the transaction on your local explorer.

bridge-out-tx-details

(1g) To learn more about what is going on, click the Logs tab:

bridge-out-logs-details-1

bridge-out-logs-explorer-2

Step 2: Claim your withdrawal

After approximately 2 hours, you will be able to claim your withdrawal.

(2a) Head back to the bridge UI and you will have a notification to claim your withdrawal. Click Claim in the details of the transaction:

bridge-out-claim-withdrawal

(2b) Approve the transaction in your wallet.

(2c) After your transaction goes through, you can see the details in the bridge UI under Settled transactions:

bridge-out-claim-success-withdrawal

',50),k=[S];function C(O,B,q,P,T,I){return t(),o("div",null,k)}const L=r(A,[["render",C]]);export{D as __pageData,L as default}; diff --git a/assets/developers_arbitrum-bridge.md.7321b996.lean.js b/assets/developers_arbitrum-bridge.md.7321b996.lean.js new file mode 100644 index 00000000000..576a7ac74a5 --- /dev/null +++ b/assets/developers_arbitrum-bridge.md.7321b996.lean.js @@ -0,0 +1 @@ +import{_ as r,o as t,c as o,Q as i}from"./chunks/framework.51d6c45b.js";const e="/arbitrum/bridge-overview-deposit-and-withdrawal-l3.png",a="/arbitrum/bridge-settings.png",n="/arbitrum/add-custom-chain-to-bridge.png",s="/arbitrum/live-orbit-chains.png",l="/arbitrum/bridge-in-start.png",p="/arbitrum/bridge-in-pending-txs.png",g="/arbitrum/bridge-in-settled-txs.png",d="/arbitrum/bridge-in-explorer-rollup-tx.png",u="/arbitrum/bridge-in-success.png",c="/arbitrum/bridge-in-sepolia-tx-explorer.png",b="/arbitrum/bridge-out-small-screenshot.png",h="/arbitrum/bridge-out-begin.png",m="/arbitrum/bridge-out-pending.png",f="/arbitrum/bridge-out-begin-overview.png",w="/arbitrum/bridge-out-rollup-tx-details.png",y="/arbitrum/bridge-out-logs-details-1.png",_="/arbitrum/bridge-out-logs-explorer-2.png",x="/arbitrum/bridge-out-claim-withdrawal.png",v="/arbitrum/bridge-out-claim-success-withdrawal.png",D=JSON.parse('{"title":"Bridging in and out of your Orbit rollup","description":"A guide on how to bridge in and out of your Arbitrum Orbit rollup.","frontmatter":{"description":"A guide on how to bridge in and out of your Arbitrum Orbit rollup.","head":[["meta",{"name":"og:title","content":"Bridging in and out of your Orbit rollup | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/arbitrum-bridge.md","filePath":"developers/arbitrum-bridge.md","lastUpdated":1713191709000}'),A={name:"developers/arbitrum-bridge.md"},S=i("",50),k=[S];function C(O,B,q,P,T,I){return t(),o("div",null,k)}const L=r(A,[["render",C]]);export{D as __pageData,L as default}; diff --git a/assets/developers_arbitrum-dapp-deploy.md.3f1872f8.js b/assets/developers_arbitrum-dapp-deploy.md.3f1872f8.js new file mode 100644 index 00000000000..1ae34cf29e3 --- /dev/null +++ b/assets/developers_arbitrum-dapp-deploy.md.3f1872f8.js @@ -0,0 +1,21 @@ +import{_ as s,o as a,c as o,Q as n}from"./chunks/framework.51d6c45b.js";const l="/img/gm-arb.png",F=JSON.parse('{"title":"Deploy a dapp on your Arbitrum rollup devnet","description":"Make your own GM Portal dapp on your Arbitrum rollup.","frontmatter":{"description":"Make your own GM Portal dapp on your Arbitrum rollup.","head":[["meta",{"name":"og:title","content":"Deploy a dapp on your Arbitrum rollup devnet | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/arbitrum-dapp-deploy.md","filePath":"developers/arbitrum-dapp-deploy.md","lastUpdated":1712087050000}'),p={name:"developers/arbitrum-dapp-deploy.md"},t=n(`

Deploy a dapp on your Arbitrum rollup devnet

First, review the Arbitrum integration, Quickstart: Deploy an Arbitrum Orbit rollup, and Deploy a smart contract to your Arbitrum rollup pages.

Dependencies

Setup and contract deployment

  1. Clone the gm-portal from GitHub and start the frontend:

    bash
    cd $HOME
    +git clone https://github.com/jcstein/gm-portal.git
    +cd gm-portal && git checkout arbitrum
    +cd frontend && yarn && yarn dev
    cd $HOME
    +git clone https://github.com/jcstein/gm-portal.git
    +cd gm-portal && git checkout arbitrum
    +cd frontend && yarn && yarn dev
  2. In a new terminal instance, set your private key for the faucet as a variable and the RPC URL you're using:

    bash
    export PRIVATE_KEY=0xb6b15c8cb491557369f3c7d2c287b053eb229daa9c22138887752191c9520659
    +export ARB_RPC_URL=http://localhost:8547
    export PRIVATE_KEY=0xb6b15c8cb491557369f3c7d2c287b053eb229daa9c22138887752191c9520659
    +export ARB_RPC_URL=http://localhost:8547
  3. Change into the gm-portal/contracts directory in the same terminal and deploy the contract using Foundry:

    bash
    cd $HOME/gm-portal/contracts
    +forge script script/GmPortal.s.sol:GmPortalScript --rpc-url $ARB_RPC_URL --private-key $PRIVATE_KEY --broadcast
    cd $HOME/gm-portal/contracts
    +forge script script/GmPortal.s.sol:GmPortalScript --rpc-url $ARB_RPC_URL --private-key $PRIVATE_KEY --broadcast
  4. In the output of the deployment, find the contract address and set it as a variable:

    bash
    export CONTRACT_ADDRESS=<your-contract-address-from-the-output-above>
    export CONTRACT_ADDRESS=<your-contract-address-from-the-output-above>

Interact with the contract

Next, you're ready to interact with the contract from your terminal!

  1. Send a "gm" to the contract:

    bash
    cast send $CONTRACT_ADDRESS \\
    +"gm(string)" "gm" \\
    +--private-key $PRIVATE_KEY \\
    +--rpc-url $ARB_RPC_URL
    cast send $CONTRACT_ADDRESS \\
    +"gm(string)" "gm" \\
    +--private-key $PRIVATE_KEY \\
    +--rpc-url $ARB_RPC_URL
  2. Now that you've posted to the contract, you can read all "gms" (GMs) from the contract with this command:

    bash
    cast call $CONTRACT_ADDRESS "getAllGms()" --rpc-url $ARB_RPC_URL
    cast call $CONTRACT_ADDRESS "getAllGms()" --rpc-url $ARB_RPC_URL
  3. Next, query the total number of gms, which will be returned as a hex value:

    bash
    cast call $CONTRACT_ADDRESS "getTotalGms()" --rpc-url $ARB_RPC_URL
    cast call $CONTRACT_ADDRESS "getTotalGms()" --rpc-url $ARB_RPC_URL
  4. (Optional) In order to interact with the contract on the frontend, you'll need to fund an account that you have in your Ethereum wallet. Transfer to an external account with this command:

    bash
    export RECEIVER=<receiver ETH address>
    +cast send --private-key $PRIVATE_KEY $RECEIVER --value 1ether --rpc-url $ARB_RPC_URL
    export RECEIVER=<receiver ETH address>
    +cast send --private-key $PRIVATE_KEY $RECEIVER --value 1ether --rpc-url $ARB_RPC_URL

    TIP

    If you are in a different terminal than the one you set the private key in, you may need to set it again.

Update the frontend

Next, you will need to update a few things before you can interact with the contract on the frontend:

  1. Change the contract address on gm-portal/frontend/src/App.tsx to your contract address
  2. Match the chain info on gm-portal/frontend/src/main.tsx with the chain config of your L2
  3. If you changed the contract, update the ABI in gm-portal/frontend/GmPortal.json from gm-portal/contracts/out/GmPortal.sol/GmPortal.json. This can be done with:
bash
cd $HOME
+cp dev/gm-portal/contracts/out/GmPortal.sol/GmPortal.json dev/gm-portal/frontend
cd $HOME
+cp dev/gm-portal/contracts/out/GmPortal.sol/GmPortal.json dev/gm-portal/frontend

Interact with the frontend

Now, login with your wallet that you funded, and post a GM on your GM portal!

gm-arb

',16),e=[t];function r(c,i,y,d,E,h){return a(),o("div",null,e)}const m=s(p,[["render",r]]);export{F as __pageData,m as default}; diff --git a/assets/developers_arbitrum-dapp-deploy.md.3f1872f8.lean.js b/assets/developers_arbitrum-dapp-deploy.md.3f1872f8.lean.js new file mode 100644 index 00000000000..8391f72295e --- /dev/null +++ b/assets/developers_arbitrum-dapp-deploy.md.3f1872f8.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as o,Q as n}from"./chunks/framework.51d6c45b.js";const l="/img/gm-arb.png",F=JSON.parse('{"title":"Deploy a dapp on your Arbitrum rollup devnet","description":"Make your own GM Portal dapp on your Arbitrum rollup.","frontmatter":{"description":"Make your own GM Portal dapp on your Arbitrum rollup.","head":[["meta",{"name":"og:title","content":"Deploy a dapp on your Arbitrum rollup devnet | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/arbitrum-dapp-deploy.md","filePath":"developers/arbitrum-dapp-deploy.md","lastUpdated":1712087050000}'),p={name:"developers/arbitrum-dapp-deploy.md"},t=n("",16),e=[t];function r(c,i,y,d,E,h){return a(),o("div",null,e)}const m=s(p,[["render",r]]);export{F as __pageData,m as default}; diff --git a/assets/developers_arbitrum-deploy.md.be261105.js b/assets/developers_arbitrum-deploy.md.be261105.js new file mode 100644 index 00000000000..f1ebc836e50 --- /dev/null +++ b/assets/developers_arbitrum-deploy.md.be261105.js @@ -0,0 +1,111 @@ +import{_ as s,o as a,c as o,Q as n}from"./chunks/framework.51d6c45b.js";const e="/arbitrum/choose_da.png",t="/arbitrum/configuration.png",l="/arbitrum/download-config.png",p="/arbitrum/blockscout.png",r="/arbitrum/explorer-view.png",B=JSON.parse('{"title":"Quickstart: Deploy an Arbitrum Orbit rollup","description":"A guide on how to deploy an Arbitrum Orbit rollup using the Arbitrum Orbit deployment UI and deploying the rollup to Mocha testnet.","frontmatter":{"description":"A guide on how to deploy an Arbitrum Orbit rollup using the Arbitrum Orbit deployment UI and deploying the rollup to Mocha testnet.","head":[["meta",{"name":"og:title","content":"Quickstart: Deploy an Arbitrum Orbit rollup | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/arbitrum-deploy.md","filePath":"developers/arbitrum-deploy.md","lastUpdated":1713191709000}'),c={name:"developers/arbitrum-deploy.md"},i=n('

Quickstart: Deploy an Arbitrum Orbit rollup

This guide covers deploying a rollup using the Celestia Orbit chain deployment portal.

After completing this tutorial, you will have a local development network rollup capable of hosting EVM-compatible smart contracts. This rollup will process transactions locally, settle on the public Arbitrum Sepolia testnet, and post data to Celestia's Mocha testnet.

If you're looking to learn more about the integration of Celestia and Orbit, read the Arbitrum Orbit integration overview. If you're looking to learn more about Orbit, read A gentle introduction: Orbit chains.

Thank you, Offchain Labs!

This guide was made possible with the support and information provided by the Offchain Labs team, the creators of Arbitrum. For more detailed information and support, visit Arbitrum documentation and the original deployment guide.

Prerequisites

Setup

This section was adapted from Arbitrum's Orbit quickstart.

Step 1: Acquire Arbitrum Sepolia ETH

You'll need at least 1 testnet ETH for a regular Orbit rollup or 0.6 ETH plus 0.4 of your desired native token for Orbit rollups with a custom gas token. The funds will cover the cost of deploying the base contracts to the base chain, in this case, Arbitrum Sepolia.

The simplest way to do this is to:

  1. Use an L1 testnet ETH faucet like sepoliafaucet.com to acquire some testnet ETH on Ethereum Sepolia testnet.
  2. Bridge your L1 testnet ETH to L2 Arbitrum Sepolia using the Arbitrum bridge.

Step 2: Pick your deployment type

Visit the Celestia Orbit chain deployment portal. This portal offers the following options:

  1. Celestia Rollup: Transaction data is posted to Celestia
  2. Rollup: Transaction data is posted to Ethereum
  3. AnyTrust: Transaction data is posted by a Data Availability Committee

Connect your wallet to the deployment portal. You may be prompted to add the Arbitrum Sepolia network to your wallet and/or switch your wallet to this network; approve this.

In this guide, we will select Celestia ✨ and deploy a rollup which posts data to Celestia (1 above).

Choose Celestia for DA

Click Next. In the next step, we will configure the deployment.

Step 3: Configure your Orbit chain's deployment

The deployment portal will then display a form that looks like this:

configuration

Parameter descriptions can be found in the table below (more in-depth descriptions can be found in the deployment UI). We recommend sticking to the defaults; to learn more about customizing your Orbit chain's deployment configuration, visit How to customize your Orbit chain's deployment configuration:

ParameterDescription
Chain IDThis is a unique integer identifier for your chain's network, primarily used on chain indexes like Chainlist.org. It's not crucial for development networks, but in production, you'll need to choose a unique ID.
Chain NameThe name you assign to your Orbit chain, which helps users and developers distinguish it from other chains. It should be memorable and recognizable.
Challenge Period BlocksDetermines the time frame within which validators can dispute the state of the chain posted to the base chain. It's measured in blocks on the underlying L1 chain. A longer period allows more time for disputes but also delays withdrawals.
Stake TokenSpecifies the token that validators must stake to participate in the validation process, using the token's contract address on the base chain. This can be ETH or another token, defined by its address.
Base StakeThe minimum amount of stake token required for validators to post state assertions. A lower base stake lowers the barrier to entry but increases vulnerability to attacks, whereas a higher stake encourages honest participation but raises the entry barrier.
OwnerThe account address that has the authority to deploy, own, and update the base contracts of your Orbit chain on its base chain. In production, this is usually a high-stakes address controlled by a DAO or a multisig setup. For development chains, it's a lower-stakes administrative account.
Gas TokenThe token used for gas payments on the network, which must be natively deployed on the parent chain. There are specific requirements for custom gas tokens, such as having 18 decimals and not being a rebasing or fee-on-transfer token. This feature is primarily for Orbit AnyTrust chains.
ValidatorsThis is the number of validators for your chain, including their addresses. The first validator is auto-generated and immutable. Validators are crucial for maintaining the integrity of the chain and posting state assertions to the base chain.
Batch PosterResponsible for posting transaction batches from your Orbit chain to the base chain. An address for this role is automatically generated, with the private key stored in a configuration file.

In the Configure Validators section, specify the number of validators and their addresses for your chain. The initial validator's address is pre-generated and immutable, with its key stored in a JSON file. Validators ensure transaction integrity and state assertions on the base chain. They're added to an allow-list for validation and staking. Base contracts refer to your Orbit chain's L2 contracts, and base chain to the L2 chain they're deployed on.

In the Configure Batch Poster section, a batch poster address is auto-generated for posting transaction batches to the base contracts on the base chain. The address and its private key are also stored in a JSON configuration file. After configuring, proceed to review and deploy your Orbit chain.

After configuring your batch poster, proceed to the next step.

Step 3: Review & Deploy your Orbit chain

Now, deploy your chain's base contracts to Arbitrum Sepolia!

Click the Deploy button on the Review & Deploy page. Your wallet should prompt you to submit a transaction to the Arbitrum testnet. You'll have to pay a little gas; your wallet may denominate this in ETH; as long as you see your chosen Arbitrum testnet in the transaction details, this gas fee will be paid in testnet ETH.

Before proceeding, let's briefly review what just happened:

  1. You submitted a deployment transaction to an Orbit "factory" smart contract on the Arbitrum testnet, the public L2 chain that your local Orbit chain will settle transactions to.
  2. This Orbit smart contract then initialized your Orbit chain's base contracts with the values that you specified in the previous step, and deployed these base contracts to the Arbitrum testnet.

Your Orbit chain's base contracts are responsible for facilitating the exchange of information between your chain's node(s) and its base chain's nodes. This includes the batch posting of transactions from your Orbit chain to its base chain, the staking of tokens by your Orbit chain's validators the challenge mechanism, bridging mechanisms, and more.

Once your transaction is complete, continue to Step 4 to download your chain's configuration files and launch your chain.

Step 4: Download your chain's configuration files and launch your chain

After configuring your chain, you will need to download the necessary configuration files to launch your chain. Click the Download zip files button to download both the Rollup Config and L3 Config in a single ZIP file.

  • Rollup Config: This is the nodeConfig.json file, encapsulating your chain's node configuration. It is crucial as it contains the private keys for your validator and batch poster, essential for signing transactions for RBlocks and batch postings to your chain's base contracts on the L2 chain.

  • L3 Config: This is the orbitSetupScriptConfig.json file, which holds your chain's configuration, including configurations needed for your Token Bridge contracts.

Ensure to securely store these downloaded files as they contain sensitive information crucial for your chain's operation.

download config

Step 5: Clone the setup script repository and add your configuration files

  1. Clone the orbit-setup-script repository:

    bash
    git clone https://github.com/celestiaorg/orbit-setup-script.git
    git clone https://github.com/celestiaorg/orbit-setup-script.git
  2. Move the nodeConfig.json and orbitSetupScriptConfig.json files that you downloaded into the config directory in the root of your cloned orbit-setup-script repository.

  3. Install dependencies by running yarn install from the root of the orbit-setup-script repository.

Step 6: Pick an L2 RPC URL for the Batch Poster

In order for the Batch Poster, which is responsible for posting batches of data, to subscribe to Blobstream's smart contract events, the node most use a WebSocket connection, since an HTTP one will not support subscriptions. This RPC URL is different from the parent-chain.connection.url object used in the node config, and is not necessary when running a full node. WebSocket (WSS) URLs which are essential for real-time data fetching and interaction with the Arbitrum Sepolia network.

To establish a WebSocket connection for your rollup to Arbitrum Sepolia, it's recommended to find an RPC provider with WSS connections from Arbitrum's docs.

For this example, we will make an account on Alchemy. Follow these steps to set up your account and obtain a WSS URL using Alchemy:

  1. Visit Alchemy's website and sign up for an account.
  2. Once logged in, create a new app by selecting the Arbitrum network, specifically targeting the Arbitrum Sepolia testnet.
  3. After creating your app, navigate to the "API key" section to find your WebSocket (WSS) URL.
  4. In the next step, use this WSS URL in your nodeConfig.json under the celestia-cfg.eth-rpc object to ensure your node can establish a WebSocket connection to the Arbitrum Sepolia network and successfully subscribe to Blobstream events.

Without a WSS connection, the Batch Poster won't be able to subscribe to Blobstream events, and thus will fall back to posting data to parent chain.

Step 7: Run your light node for Mocha testnet

First, be sure that your light node is running, using a command similar to:

TIP

If you are on Linux (or are not using Docker desktop), you may need to add the extra flags: --rpc.addr 0.0.0.0 and --rpc.port 26658 to your start command for your light node.

Additionally, you will need to add host.docker.internal as a host in your docker-compose.yml:

yaml
extra_hosts:
+      - "host.docker.internal:host-gateway"
extra_hosts:
+      - "host.docker.internal:host-gateway"
bash
celestia light start --p2p.network mocha --core.ip <RPC_URL>
celestia light start --p2p.network mocha --core.ip <RPC_URL>

To set your light node's auth token, you will use the auth token that returns when you run:

bash
celestia light auth admin --p2p.network mocha
celestia light auth admin --p2p.network mocha

Since the contracts deployed through the factories above are already configured to communicate with Blobstream, you now only have to configure your node accordingly. First understand the different variables that will be set in the config:

  • enable: set it to true if you are using Celestia DA 😁
  • rpc: RPC endpoint for celestia-node
  • tendermint-rpc: a celestia-core endpoint from a full node (NOTE: only needed for a batch poster node)
  • eth-rpc: Ethereum Client WSS RPC endpoint, only used when the node is a batch poster. The eth-rpc must be WSS. Otherwise, it won't be able to subscribe to events for Blobstream.
  • namespace-id: namespace being used to post data to Celestia
  • auth-token: auth token for your Celestia Node
  • is-poster: is the node with Celestia DA the batch poster, set to true if so.
  • gas-price: how much to pay for gas (in uTIA)
  • event-channel-size: size of the events channel used by the batch poster to wait for a range of headers that contains the header for the block in which it posted a blob, before posting the batch to the base layer for verification on Blobstream X.
  • blobstreamx-address: address of the Blobstream X contract on the base chain.
    • Note that the SequencerInbox contract for each chain has a constant address for the BlobstreamX contract, thus make sure that the Blobstream X address in the SequencerInbox being used for the templates in RollupCreator matches the one in your config.

Now enable Celestia DA in your Arbitrum chain params in config/nodeConfig.json. If you'd like to use your own namespace, use a custom 10 byte value or random value using openssl rand -hex 10 for namespace-id:

WARNING

The Orbit contracts depend on the existing Blobstream X deployments. Before using these addresses, please verify the contract addresses on the official source below to avoid any issues due to incorrect addresses. This is crucial to protect against potential misuse by copy-paste errors.

ts
"celestia-cfg": {
+  "enable": true,
+  "rpc": "http://host.docker.internal:26658",
+  "tendermint-rpc": "http://consensus-full-mocha-4.celestia-mocha.com:26657",
+  "eth-rpc": "wss://<YOUR_ETH_RPC_WSS_URL>",
+  "namespace-id": "<YOUR_10_BYTE_NAMESPACE>",
+  "auth-token": "<YOUR_AUTH_TOKEN>",
+  "is-poster": true,
+  "gas-price": 0.3,
+  "event-channel-size": 100,
+  "blobstreamx-address": "0xc3e209eb245Fd59c8586777b499d6A665DF3ABD2",
+}
"celestia-cfg": {
+  "enable": true,
+  "rpc": "http://host.docker.internal:26658",
+  "tendermint-rpc": "http://consensus-full-mocha-4.celestia-mocha.com:26657",
+  "eth-rpc": "wss://<YOUR_ETH_RPC_WSS_URL>",
+  "namespace-id": "<YOUR_10_BYTE_NAMESPACE>",
+  "auth-token": "<YOUR_AUTH_TOKEN>",
+  "is-poster": true,
+  "gas-price": 0.3,
+  "event-channel-size": 100,
+  "blobstreamx-address": "0xc3e209eb245Fd59c8586777b499d6A665DF3ABD2",
+}

See the compatibility matrix in the appendix to verify you're using the right versions.

Step 8: Run your chain's node and block explorer

Start Docker, then run docker-compose up -d from the root of the orbit-setup-script repository.

A Nitro node and BlockScout explorer instance will be started. Visit http://localhost/ to access your BlockScout explorer instance - this will allow you to view your chain's transactions and blocks, which can be useful for debugging.

blockscout

After you have some activity on your rollup, it will look more like this:

explorer-view

Step 9: Finish setting up your chain

The Offchain Labs team has provided a Hardhat script that handles the following tasks:

  1. Fund the batch-poster and validator (staker) accounts on your underlying L2 chain.
  2. Deposit ETH into your account on the chain using your chain's newly deployed bridge.
  3. Deploy your Token Bridge contracts on both L2 and local Orbit chains.
  4. Configure parameters on the chain.

To run this script, issue the following command from the root of the orbit-setup-script repository, replacing YourPrivateKey with the private key of the Owner account you used to deploy your chain's contracts, and replacing http://localhost:8449 with the RPC URL of your chain's node.

First, export your private key as a variable:

bash
PRIVATE_KEY="YourPrivateKey" \\
+  L2_RPC_URL="https://sepolia-rollup.arbitrum.io/rpc" \\
+  L3_RPC_URL="http://localhost:8449" yarn run setup
PRIVATE_KEY="YourPrivateKey" \\
+  L2_RPC_URL="https://sepolia-rollup.arbitrum.io/rpc" \\
+  L3_RPC_URL="http://localhost:8449" yarn run setup

Successful logs will appear similar to:

bash
Funding batch-poster accounts on parent chain with 0.3 ETH
+Transaction hash on parent chain: 0x6c7360a96165c570dcb7ce609d748d612c5fa5b76e229cd81ba5f5c93c00f805
+Transaction was mined in block 28217647 on parent chain
+Funding staker accounts on parent chain with 0.3 ETH
+Transaction hash on parent chain: 0x59d2db6c5095b9e329c80211b7a761d20064379e3382d156b69e5cf3b5fe2fc7
+Transaction was mined in block 28217653 on parent chain
+Running Orbit Chain Native token deposit to Deposit ETH or native ERC20 token from parent chain to your account on Orbit chain ... 💰💰💰💰💰💰
+Transaction hash on parent chain:  0x8dee6e88d3b62b258c1574cbb7005e1c3cf193b60a99b5c2fcfae00819b7ed82
+0.4 ETHs are deposited to your account
+Balance not changed yet. Waiting for another 30 seconds ⏰⏰⏰⏰⏰⏰
+Balance of your account on Orbit chain increased by the native token you have just sent.
+Running tokenBridgeDeployment or erc20TokenBridge script to deploy token bridge contracts on parent chain and your Orbit chain 🌉🌉🌉🌉🌉
+Creating token bridge for rollup 0x7fbEB5BC73a11b438891022786feb2C624f275F0
+Token bridge deployed in transaction 0x4888fdf44251d456bbfca92bfc6e180cfe0b096ffbea2f6da2a203a16902214f
+Waiting for retryables...
+Retryable #1: 0xc61382d5609ab0ece36b2776349c8bdceeafdd13dde9624cdf3d746fb4cf7d79
+Retryable #2: 0xf31fd34f8a9d9057198d8b13e755e583766bd528459733d948d9ffbc980c9506
+Done!
+Weth gateway set in tx 0xf2ddc2dad90e7e2b20a772bf89f989224165659d50824b98d7340e12265abf01
+Waiting for retryables...
+Retryable #1: 0xf47dc66514fd78e4666e35abd12df7d1ae2c79f69f7dfedb8d98e4106142ab7c
+Done!
+network.json updated
+Done!
+Running l3Configuration script to configure your Orbit chain 📝📝📝📝📝
+Setting the Minimum Base Fee for the Orbit chain
+Minimum Base Fee is set on the block number 13 on the Orbit chain
+Setting the  network fee receiver for the Orbit chain
+network fee receiver is set on the block number 14 on the Orbit chain
+Setting the infrastructure fee collector address for the Orbit chain
+infrastructure fee collector address is set on the block number 15 on the Orbit chain
+Getting L1 base fee estimate
+L1 Base Fee estimate on L2 is 4989526079
+Setting L1 base fee estimate on L3 to 5158076079
+L1 base fee estimate is set on the block number 16 on the Orbit chain
+All things done! Enjoy your Orbit chain. LFG 🚀🚀🚀🚀
+Transferring ownership on L3, from rollup owner to upgrade executor 🔃🔃🔃
+Adding Upgrade Executor contract to the chain owners
+Executor has been added to chain owners on TX: 0x97b50f60b60d0e658fdbf185969db0a0327bd0ae9e57cd65af2a7f9be0eeb5b0
+Executing removeChainOwner through the UpgradeExecutor contract
+Transaction complete, rollup owner removed from chain owners on TX: 0x019850732270d8c436585c7921219252422228b5d0f559da0da219f0fa2b7216
+  Done in 58.49s.
Funding batch-poster accounts on parent chain with 0.3 ETH
+Transaction hash on parent chain: 0x6c7360a96165c570dcb7ce609d748d612c5fa5b76e229cd81ba5f5c93c00f805
+Transaction was mined in block 28217647 on parent chain
+Funding staker accounts on parent chain with 0.3 ETH
+Transaction hash on parent chain: 0x59d2db6c5095b9e329c80211b7a761d20064379e3382d156b69e5cf3b5fe2fc7
+Transaction was mined in block 28217653 on parent chain
+Running Orbit Chain Native token deposit to Deposit ETH or native ERC20 token from parent chain to your account on Orbit chain ... 💰💰💰💰💰💰
+Transaction hash on parent chain:  0x8dee6e88d3b62b258c1574cbb7005e1c3cf193b60a99b5c2fcfae00819b7ed82
+0.4 ETHs are deposited to your account
+Balance not changed yet. Waiting for another 30 seconds ⏰⏰⏰⏰⏰⏰
+Balance of your account on Orbit chain increased by the native token you have just sent.
+Running tokenBridgeDeployment or erc20TokenBridge script to deploy token bridge contracts on parent chain and your Orbit chain 🌉🌉🌉🌉🌉
+Creating token bridge for rollup 0x7fbEB5BC73a11b438891022786feb2C624f275F0
+Token bridge deployed in transaction 0x4888fdf44251d456bbfca92bfc6e180cfe0b096ffbea2f6da2a203a16902214f
+Waiting for retryables...
+Retryable #1: 0xc61382d5609ab0ece36b2776349c8bdceeafdd13dde9624cdf3d746fb4cf7d79
+Retryable #2: 0xf31fd34f8a9d9057198d8b13e755e583766bd528459733d948d9ffbc980c9506
+Done!
+Weth gateway set in tx 0xf2ddc2dad90e7e2b20a772bf89f989224165659d50824b98d7340e12265abf01
+Waiting for retryables...
+Retryable #1: 0xf47dc66514fd78e4666e35abd12df7d1ae2c79f69f7dfedb8d98e4106142ab7c
+Done!
+network.json updated
+Done!
+Running l3Configuration script to configure your Orbit chain 📝📝📝📝📝
+Setting the Minimum Base Fee for the Orbit chain
+Minimum Base Fee is set on the block number 13 on the Orbit chain
+Setting the  network fee receiver for the Orbit chain
+network fee receiver is set on the block number 14 on the Orbit chain
+Setting the infrastructure fee collector address for the Orbit chain
+infrastructure fee collector address is set on the block number 15 on the Orbit chain
+Getting L1 base fee estimate
+L1 Base Fee estimate on L2 is 4989526079
+Setting L1 base fee estimate on L3 to 5158076079
+L1 base fee estimate is set on the block number 16 on the Orbit chain
+All things done! Enjoy your Orbit chain. LFG 🚀🚀🚀🚀
+Transferring ownership on L3, from rollup owner to upgrade executor 🔃🔃🔃
+Adding Upgrade Executor contract to the chain owners
+Executor has been added to chain owners on TX: 0x97b50f60b60d0e658fdbf185969db0a0327bd0ae9e57cd65af2a7f9be0eeb5b0
+Executing removeChainOwner through the UpgradeExecutor contract
+Transaction complete, rollup owner removed from chain owners on TX: 0x019850732270d8c436585c7921219252422228b5d0f559da0da219f0fa2b7216
+  Done in 58.49s.

Find your PFB on Celenium by looking at the namespace or account you posted from.

See an example blob that was posted while making this guide.

Congratulations with Celestia underneath

Your local Orbit rollup is now running. You'll see an outputInfo.json file in the main directory of your script folder - this contains more information about your chain, including the addresses of your chain's base contracts.

In the next guides, learn how to run a full and validating full node or bridge in and out of your rollup.

Appendix

Extra resources in Arbitrum documentation:

Compatibility matrix

ComponentVersionDetails
Nitrov2.3.1-rc.1Includes the replay binary for the WASM root 0x10c65b27d5031ce2351c719072e58f3153228887f027f9f6d65300d2b5b30152. Read the overview for overall changes.
Contractsv1.2.1-celestiaIntegrates Blobstream X functionality into nitro-contracts v1.2.1
Orbit SDKv0.8.2 Orbit SDK for Celestia DAThis is not compatible with Orbit SDK v0.8.2 or with the latest changes to nitro-contracts for the Atlas upgrade. The Orbit SDK itself is in Alpha.
celestia-nodev0.13.1This integration has only been tested with celestia-node 0.13.1 and only works with said version, and with future versions after that. Under the hood, the Nitro node uses this commit of celestia-openrpc.

Blobstream X contract deployments

The Orbit contracts depend on the following Blobstream X deployments. The current deployments, which can be found at 0xc3e209eb245Fd59c8586777b499d6A665DF3ABD2 in both chains, relays headers from the Mocha-4 testnet to the chains below:

  • Arbitrum Sepolia
  • Base Sepolia

Arbitrum Sepolia

Base Sepolia

  • RollupCreator: 0x1Bb8ADd5e878b12Fa37756392642eB94C53A1Cf4
  • TokenBridgeCreator: 0xAa3b8B63cCCa3c98b948FD1d6eD875d378dE2C6c
  • TokenBridgeRetryableSender: 0x4270889AdcB82338C5FF5e64B45c0A3d31CFd08C
  • Find additional Base Sepolia deployments below

Arbitrum Sepolia additional deployments

ContractAddress
Bridge0x95FEA00e689e8D1CBa909836E1Ef1b941D5f21b1
SequencerInbox0x95CBDa89325db5529eAF1813E181f66B83A7d65a
Inbox0x3681Cbb0E95AB50b63F2FC524FbBcC78adEfBd33
RollupEventInbox0x61e154128b6a1400ea8090B4431B4aA1DBb80Cc4
Outbox0x5187a92539bB4A2befe1fc078745c84AB6d37171
ERC20Bridge0xD0a6699Fc7519966685181c80BF98D35aFa1fC95
SequencerInbox0x2588867F19E2DE51f90F0aB852C7Ad11228e3d83
ERC20Inbox0x6cB49605f10831749c6090AD09918bC61439bacE
ERC20RollupEventInbox0x7fC4D9A24949680faD666FeEe7cD6a100E39C4F0
ERC20Outbox0xA773e19DC9e822933A7e72Df9c87eD1578701D29
BridgeCreator0x3Bc040EAca40b91FA06cf55Ea91842FaC88b1AF4
OneStepProver00x5810F0916BAE1067Ca1efcc00AaaF30301af001c
OneStepProverMemory0xaC3427E621C6F10dC2ABdAB00188D92690503914
OneStepProverMath0xFB612fb83959b8ACD3E49540B29C93c5A67e05f1
OneStepProverHostIo0x630093954CbF19Fe4532A2edD0bD3B10dEcA7A4D
OneStepProofEntry0x53DEA3A90Fd6C82840a1f7224F799D622f142Df4
ChallengeManager0x01B5905B154F21a393F5B5a0C6d15B53a493C05e
RollupAdminLogic0xe371AFcb8437bF61bd831EF57Be7A2496D88488B
RollupUserLogic0xE24a60b758b51b0a3dA5E8F4F6ddf1cd0aFF646C
ValidatorUtils0x7973D0b475E898082dF25c1617CBce1917cFED17
ValidatorWalletCreator0xe2662ff9b41f39e63A850E50E013Ea66e60A4F37
RollupCreator0x79751B011BCc20F413a2c4E3AF019b6E2a9738B9
DeployHelper0xd2D353916B34a877793628049c99858f04123eE1

Base Sepolia additional deployments

ContractAddress
Bridge0xb6052122545AACD2BDda0Ca9FA56416bD968cDbc
SequencerInbox0xcd9FCa5015b5ce2B06a2266e4a5dd54D9ca39F1a
Inbox0x44B412b291fEf00398501B2cA353EA912AD0fe13
RollupEventInbox0x51D196e07a27DBA0F4461Dd6CC26108424F196f7
Outbox0x5A48aDf22f526eBD06e3e8856cFEa2490923CC55
ERC20Bridge0x9abC41fEfAe7E7543a01FA837AeC909F96147280
SequencerInbox0x8f97Cb7c643Acd7f79f3B13841b24a243dA51242
ERC20Inbox0x40f8c63e0a20B399bCd9631A22E57BB988a9400e
ERC20RollupEventInbox0x3B6e845fb9f0c8Ee4E9F6D44781f6547d9c6359a
ERC20Outbox0xc99eEA0B8e67D5b2226AB6D37882DAAf6dd7593b
BridgeCreator0xC7535F078CB3880a0FD5E54FA7A3B4EAf09b3924
OneStepProver00xf889a3174Fddd9f78E6cd250Ebf4c16F1bDd1b6a
OneStepProverMemory0x61254e43e5c1e9E801F9C56B47a9ac3EADF6d1E9
OneStepProverMath0x55527d53fdA37Dbf1924482b40AcF8625E1cAA5B
OneStepProverHostIo0x03B43F7B61Fa100611191F481Ef48aa1fc98F434
OneStepProofEntry0x89b7c7970c13BB587893a70697AD6d2A335b6A15
ChallengeManager0x04CAe899Fc0B7Ef45c529f8Bf075D54F6fB70eD9
RollupAdminLogic0x99E9D2F04352B42C18F1DA5Dd93a970F82C08aFe
RollupUserLogic0x1ae3A8DC1e7eFD37F418B2987D3DF74c5a917a8B
ValidatorUtils0x1cc4551922C069A9aDE06756BF14bF0410eA44fF
ValidatorWalletCreator0x78f8B2941ddE5a8A312814Ebd29c2E2A36f25E91
RollupCreator0x1Bb8ADd5e878b12Fa37756392642eB94C53A1Cf4
DeployHelper0x20d8153AaCC4E6D29558fa3916BfF422BEDE9B5E
`,95),y=[i];function E(d,F,h,u,b,C){return a(),o("div",null,y)}const g=s(c,[["render",E]]);export{B as __pageData,g as default}; diff --git a/assets/developers_arbitrum-deploy.md.be261105.lean.js b/assets/developers_arbitrum-deploy.md.be261105.lean.js new file mode 100644 index 00000000000..6e32c8ad3fa --- /dev/null +++ b/assets/developers_arbitrum-deploy.md.be261105.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as o,Q as n}from"./chunks/framework.51d6c45b.js";const e="/arbitrum/choose_da.png",t="/arbitrum/configuration.png",l="/arbitrum/download-config.png",p="/arbitrum/blockscout.png",r="/arbitrum/explorer-view.png",B=JSON.parse('{"title":"Quickstart: Deploy an Arbitrum Orbit rollup","description":"A guide on how to deploy an Arbitrum Orbit rollup using the Arbitrum Orbit deployment UI and deploying the rollup to Mocha testnet.","frontmatter":{"description":"A guide on how to deploy an Arbitrum Orbit rollup using the Arbitrum Orbit deployment UI and deploying the rollup to Mocha testnet.","head":[["meta",{"name":"og:title","content":"Quickstart: Deploy an Arbitrum Orbit rollup | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/arbitrum-deploy.md","filePath":"developers/arbitrum-deploy.md","lastUpdated":1713191709000}'),c={name:"developers/arbitrum-deploy.md"},i=n("",95),y=[i];function E(d,F,h,u,b,C){return a(),o("div",null,y)}const g=s(c,[["render",E]]);export{B as __pageData,g as default}; diff --git a/assets/developers_arbitrum-full-node.md.1f75a4dc.js b/assets/developers_arbitrum-full-node.md.1f75a4dc.js new file mode 100644 index 00000000000..7fcf5154588 --- /dev/null +++ b/assets/developers_arbitrum-full-node.md.1f75a4dc.js @@ -0,0 +1,3 @@ +import{_ as n,o as e,c as o,Q as a}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"Running a full node and/or validator","description":"A guide on how to run a full node or validating full node on your Orbit rollup.","frontmatter":{"description":"A guide on how to run a full node or validating full node on your Orbit rollup.","head":[["meta",{"name":"og:title","content":"Running a full node and/or validator | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/arbitrum-full-node.md","filePath":"developers/arbitrum-full-node.md","lastUpdated":1712087050000}'),t={name:"developers/arbitrum-full-node.md"},r=a(`

Running a full node and/or validator

Prerequisites

Running a full node

To run a full node, you can follow the steps outlined in the Arbitrum docs, with the difference being that you will use this image: dfcelestia/nitro-node-dev:latest instead of the one mentioned in the Arbitrum docs.

Note that you can either use the flags in the nitro binary + the flags found in the celestia package, or you can just provide a node config.json file with the celestia-cfg for them to run it, which would look something like this:

json
docker run --rm -v "$HOME/Documents/configs/nodeConfig.json:/config.json:ro" \\
+  --network host celestia-nitro:v2.3.1-rc.1 --conf.file /config.json
docker run --rm -v "$HOME/Documents/configs/nodeConfig.json:/config.json:ro" \\
+  --network host celestia-nitro:v2.3.1-rc.1 --conf.file /config.json

Running a full node with validation

The information above applies to the steps outlined to run a validating full node (validator).

Finally, note that this will require connection to a DA node, and we recommend running a Bridge node if you will be instantiating multiple rollups.

`,10),i=[r];function l(s,d,u,c,p,h){return e(),o("div",null,i)}const m=n(t,[["render",l]]);export{g as __pageData,m as default}; diff --git a/assets/developers_arbitrum-full-node.md.1f75a4dc.lean.js b/assets/developers_arbitrum-full-node.md.1f75a4dc.lean.js new file mode 100644 index 00000000000..fdab04a5142 --- /dev/null +++ b/assets/developers_arbitrum-full-node.md.1f75a4dc.lean.js @@ -0,0 +1 @@ +import{_ as n,o as e,c as o,Q as a}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"Running a full node and/or validator","description":"A guide on how to run a full node or validating full node on your Orbit rollup.","frontmatter":{"description":"A guide on how to run a full node or validating full node on your Orbit rollup.","head":[["meta",{"name":"og:title","content":"Running a full node and/or validator | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/arbitrum-full-node.md","filePath":"developers/arbitrum-full-node.md","lastUpdated":1712087050000}'),t={name:"developers/arbitrum-full-node.md"},r=a("",10),i=[r];function l(s,d,u,c,p,h){return e(),o("div",null,i)}const m=n(t,[["render",l]]);export{g as __pageData,m as default}; diff --git a/assets/developers_arbitrum-integration.md.fa70b233.js b/assets/developers_arbitrum-integration.md.fa70b233.js new file mode 100644 index 00000000000..bf38796d281 --- /dev/null +++ b/assets/developers_arbitrum-integration.md.fa70b233.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.51d6c45b.js";const r="/img/Celestia-Arbitrum.png",f=JSON.parse('{"title":"Introduction to Arbitrum rollups with Celestia as DA","description":"An overview of the integration of Arbitrum Nitro with Celestia, detailing the key features and benefits, including the Ethereum fallback mechanism.","frontmatter":{"description":"An overview of the integration of Arbitrum Nitro with Celestia, detailing the key features and benefits, including the Ethereum fallback mechanism.","head":[["meta",{"name":"og:title","content":"Introduction to Arbitrum rollups with Celestia as DA | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/arbitrum-integration.md","filePath":"developers/arbitrum-integration.md","lastUpdated":1713191709000}'),o={name:"developers/arbitrum-integration.md"},n=i('

Introduction to Arbitrum rollups with Celestia as DA

Celestia_Arbitrum

Overview

The integration of Celestia with Arbitrum Orbit and the Nitro tech stack marks the first external contribution to the Arbitrum Orbit protocol layer, offering developers an additional option for selecting a data availability layer alongside Arbitrum AnyTrust. The integration allows developers to deploy an Orbit Chain that uses Celestia for data availability and settles on Arbitrum One, Ethereum, or other EVM chains.

Learn more about Orbit in Arbitrum's introduction.

Key components

The integration of Celestia with Arbitrum orbit is possible thanks to 3 key components:

Additionally, the Ethereum fallback mechanism is a feature of the integration, which is native in Nitro.

DA provider implementation

The Arbitrum Nitro code has a DataAvailabilityProvider interface that is used across the codebase to store and retrieve data from a specific provider (eip4844 blobs, Anytrust, and now Celestia).

This integration implements the DataAvailabilityProvider interface for Celestia DA

Additionally, this integration comes with the necessary code for a Nitro chain node to post and retrieve data from Celestia.

The core logic behind posting and retrieving data happens in celestia.go where data is stored on Celestia and serialized into a small batch of data that gets published once the necessary range of headers (data roots) has been relayed to the BlobstreamX contract. Then the Read logic takes care of taking the deserialized Blob Pointer struct and consuming it in order to fetch the data from Celestia and additionally inform the fetcher about the position of the data on Celestia (we'll get back to this in the next section).

The following represents a non-exhaustive list of considerations when running a Batch Poster node for a chain with Celestia underneath:

  • You will need to use a consensus full node RPC endpoint, you can find a list of them for Mocha here
  • The Batch Poster will only post a Celestia batch to the underlying chain if the height for which it posted is in a recent range in BlobstreamX and if the verification succeeds, otherwise it will discard the batch. Since it will wait until a range is relayed, it can take several minutes for a batch to be posted, but one can always make an on-chain request for the BlobstreamX contract to relay a header promptly.

The following represents a non-exhaustive list of considerations when running a Nitro node for a chain with Celestia underneath:

  • The TendermintRpc endpoint is only needed by the batch poster, every other node can operate without a connection to a full node.
  • The message header flag for Celestia batches is 0x0c.
  • You will need to know the namespace for the chain that you are trying to connect to, but don't worry if you don't find it, as the information in the BlobPointer can be used to identify where a batch of data is in the Celestia Data Square for a given height, and thus can be used to find out the namespace as well!

Preimage Oracle Implementation

In order to support fraud proofs, this integration has the necessary code for a Nitro validator to populate its preimage mapping with Celestia hashes that then get "unpealed" in order to reveal the full data for a Blob. You can read more about the "Hash Oracle Trick".

The data structures and hashing functions for this can be found in the nitro/das/celestia/tree folder

You can see where the preimage oracle gets used in the fraud proof replay binary here

Something important to note is that the preimage oracle only keeps track of hashes for the rows in the Celestia data square in which a blob resides in, this way each Orbit chain with Celestia underneath does not need validators to recompute an entire Celestia Data Square, but instead, only have to compute the row roots for the rows in which it's data lives in, and the header data root, which is the binary merkle tree hash built using the row roots and column roots fetched from a Celestia node. Because only data roots that can be confirmed on Blobstream get accepted into the sequencer inbox, one can have a high degree of certainty that the canonical data root being unpealed as well as the row roots are in fact correct.

Blobstream X implementation

Finally, the integration only accepts batches with information that can be confirmed on BlobstreamX, which gives us a high certainty that data was made available on Celestia.

You can see how BlobstreamX is integrated into the SequencerInbox.sol contract here, which allows us to discard batches with otherwise faulty data roots, thus giving us a high degree of confidence that the data root can be safely unpacked in case of a challenge.

The Celestia and Arbitrum integration also includes Blobstream, which relays commitments to Celestia’s data root to an onchain light client on Ethereum. This allows L2 solutions that settle on Ethereum to benefit from the scalability Celestia’s data availability layer can provide.

Ethereum fallback mechanism in Nitro

By default in Arbitrum Nitro, the Ethereum fallback mechanism in the BatchPoster function is handling the process of storing data, with a fallback mechanism to store data onchain if the primary data availability storage fails.

The @celestiaorg/nitro integration uses the same fallback mechanism.

More information can be found on the Ethereum fallback mechanisms for Celestia, which enables Ethereum L2s (or L3s) to “fall back” to using Ethereum calldata for data availability in the event of downtime on Celestia Mainnet Beta.

The fallback logic for Celestia DA is configurable, providing an alternative to the previous default fallback mechanism. Additionally, an ability has been added to the Arbitrum node software which allows the sequencer to call VerifyAttestation to check if a data root has been posted on Blobstream or not, before it sends the sequencer message (data pointer) to the underlying chain.

Next steps

In the next page, learn how to deploy an Arbitrum rollup devnet using Celestia as DA.

',34),l=[n];function s(h,c,d,m,b,u){return t(),a("div",null,l)}const g=e(o,[["render",s]]);export{f as __pageData,g as default}; diff --git a/assets/developers_arbitrum-integration.md.fa70b233.lean.js b/assets/developers_arbitrum-integration.md.fa70b233.lean.js new file mode 100644 index 00000000000..2a52894c5d1 --- /dev/null +++ b/assets/developers_arbitrum-integration.md.fa70b233.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.51d6c45b.js";const r="/img/Celestia-Arbitrum.png",f=JSON.parse('{"title":"Introduction to Arbitrum rollups with Celestia as DA","description":"An overview of the integration of Arbitrum Nitro with Celestia, detailing the key features and benefits, including the Ethereum fallback mechanism.","frontmatter":{"description":"An overview of the integration of Arbitrum Nitro with Celestia, detailing the key features and benefits, including the Ethereum fallback mechanism.","head":[["meta",{"name":"og:title","content":"Introduction to Arbitrum rollups with Celestia as DA | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/arbitrum-integration.md","filePath":"developers/arbitrum-integration.md","lastUpdated":1713191709000}'),o={name:"developers/arbitrum-integration.md"},n=i("",34),l=[n];function s(h,c,d,m,b,u){return t(),a("div",null,l)}const g=e(o,[["render",s]]);export{f as __pageData,g as default}; diff --git a/assets/developers_arbitrum-smart-contract.md.3069abf8.js b/assets/developers_arbitrum-smart-contract.md.3069abf8.js new file mode 100644 index 00000000000..bbede70c15e --- /dev/null +++ b/assets/developers_arbitrum-smart-contract.md.3069abf8.js @@ -0,0 +1,133 @@ +import{_ as s,o as a,c as n,Q as o}from"./chunks/framework.51d6c45b.js";const F=JSON.parse('{"title":"Deploy a smart contract to your Arbitrum rollup","description":"A tutorial that guides you through the process of deploying a smart contract to your Arbitrum rollup using a L2 Nitro devnet, including setting up the environment, creating and testing the smart contract, and interacting with the deployed contract.","frontmatter":{"description":"A tutorial that guides you through the process of deploying a smart contract to your Arbitrum rollup using a L2 Nitro devnet, including setting up the environment, creating and testing the smart contract, and interacting with the deployed contract.","head":[["meta",{"name":"og:title","content":"Deploy a smart contract to your Arbitrum rollup | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/arbitrum-smart-contract.md","filePath":"developers/arbitrum-smart-contract.md","lastUpdated":1703278090000}'),l={name:"developers/arbitrum-smart-contract.md"},p=o(`

Deploy a smart contract to your Arbitrum rollup

Overview

Welcome to the guide on deploying a smart contract to your Arbitrum rollup. In this tutorial, you will learn how to deploy a smart contract using the L2 Nitro devnet and the provided public and private keys for testing purposes.

Prerequisites

Setup

First, in your $HOME directory, set up a new project folder for this tutorial and init the project with npm:

bash
cd $HOME
+mkdir counter-project && cd counter-project && npm init -y
cd $HOME
+mkdir counter-project && cd counter-project && npm init -y

Next, initialize a Foundry project with the following command:

bash
forge init counter_contract
forge init counter_contract

Create your smart contract

Take a look at the Counter.sol file in your counter-project/counter_contract/src directory:

solidity
// SPDX-License-Identifier: UNLICENSED
+pragma solidity ^0.8.13;
+
+contract Counter {
+    uint256 public number;
+
+    function setNumber(uint256 newNumber) public {
+        number = newNumber;
+    }
+
+    function increment() public {
+        number++;
+    }
+}
// SPDX-License-Identifier: UNLICENSED
+pragma solidity ^0.8.13;
+
+contract Counter {
+    uint256 public number;
+
+    function setNumber(uint256 newNumber) public {
+        number = newNumber;
+    }
+
+    function increment() public {
+        number++;
+    }
+}

The contract contains a public unsigned integer variable named "number". There are two public functions in this contract. The setNumber function allows anyone to set a new value for the "number" variable, while the increment function increases the value of "number" by one each time it's called.

You can learn more about Solidity and smart contract programming.

To compile the contract, run the following forge command from the $HOME/counter-project/counter_contract/ directory:

bash
forge build
forge build

Your output should look similar to the following:

bash
[⠢] Compiling...
+[⠔] Compiling 21 files with 0.8.19
+[⠑] Solc 0.8.19 finished in 1.24s
+Compiler run successful
[⠢] Compiling...
+[⠔] Compiling 21 files with 0.8.19
+[⠑] Solc 0.8.19 finished in 1.24s
+Compiler run successful

Test your smart contract

Now, open the test/Counter.t.sol file:

solidity
// SPDX-License-Identifier: UNLICENSED
+pragma solidity ^0.8.13;
+
+import "forge-std/Test.sol";
+import "../src/Counter.sol";
+
+contract CounterTest is Test {
+    Counter public counter;
+
+    function setUp() public {
+        counter = new Counter();
+        counter.setNumber(0);
+    }
+
+    function testIncrement() public {
+        counter.increment();
+        assertEq(counter.number(), 1);
+    }
+
+    function testSetNumber(uint256 x) public {
+        counter.setNumber(x);
+        assertEq(counter.number(), x);
+    }
+}
// SPDX-License-Identifier: UNLICENSED
+pragma solidity ^0.8.13;
+
+import "forge-std/Test.sol";
+import "../src/Counter.sol";
+
+contract CounterTest is Test {
+    Counter public counter;
+
+    function setUp() public {
+        counter = new Counter();
+        counter.setNumber(0);
+    }
+
+    function testIncrement() public {
+        counter.increment();
+        assertEq(counter.number(), 1);
+    }
+
+    function testSetNumber(uint256 x) public {
+        counter.setNumber(x);
+        assertEq(counter.number(), x);
+    }
+}

This file performs unit testing on the contract we created in the previous section. Here's what the test is doing:

  • The contract includes a public "Counter" type variable called "counter". In the setUp function, it initializes a new instance of the "Counter" contract and sets the "number" variable to 0.

  • There are two test functions in the contract: testIncrement and testSetNumber.

  • The testIncrement function tests the "increment" function of the "Counter" contract by calling it and then asserting that the "number" in the "Counter" contract is 1. It verifies if the increment operation correctly increases the number by one.

  • The testSetNumber function is more generic. It takes an unsigned integer argument 'x' and tests the "setNumber" function of the "Counter" contract. After calling the "setNumber" function with 'x', it asserts that the "number" in the "Counter" contract is equal to 'x'. This verifies that the "setNumber" function correctly updates the "number" in the "Counter" contract.

Now, to test your code, run the following:

bash
forge test
forge test

If the test is successful, your output should be similar to this:

bash
[⠆] Compiling...
+No files changed, compilation skipped
+
+Running 2 tests for test/Counter.t.sol:CounterTest
+[PASS] testIncrement() (gas: 28334)
+[PASS] testSetNumber(uint256) (runs: 256, μ: 27709, ~: 28409)
+Test result: ok. 2 passed; 0 failed; finished in 8.96ms
[⠆] Compiling...
+No files changed, compilation skipped
+
+Running 2 tests for test/Counter.t.sol:CounterTest
+[PASS] testIncrement() (gas: 28334)
+[PASS] testSetNumber(uint256) (runs: 256, μ: 27709, ~: 28409)
+Test result: ok. 2 passed; 0 failed; finished in 8.96ms

Deploying your smart contract

Funded accounts

Your L2 Nitro devnet will have a public and private key funded as a faucet to use for testing:

  • On both L1 and L2
    • Public key: 0x3f1Eae7D46d88F08fc2F8ed27FCb2AB183EB2d0E
    • Private key: 0xb6b15c8cb491557369f3c7d2c287b053eb229daa9c22138887752191c9520659

Alternatively, you can fund other addresses by using the scripts send-l1 and send-l2.

The L1 Geth devnet will be running at http://localhost:8545 and the L2 Nitro devnet will be on http://localhost:8547 and ws://localhost:8548.

Using our Arbitrum devnet

We will use the local RPC endpoint (http://localhost:8547) and accounts above to test with.

Let's deploy the contract now. First, set a private key from anvil:

bash
export L2_PRIVATE_KEY=0xe887f7d17d07cc7b8004053fb8826f6657084e88904bb61590e498ca04704cf2
+export ARB_RPC_URL=http://localhost:8547
export L2_PRIVATE_KEY=0xe887f7d17d07cc7b8004053fb8826f6657084e88904bb61590e498ca04704cf2
+export ARB_RPC_URL=http://localhost:8547

Now, deploy the contract:

bash
forge create --rpc-url $ARB_RPC_URL \\
+  --private-key $L2_PRIVATE_KEY \\
+  src/Counter.sol:Counter
forge create --rpc-url $ARB_RPC_URL \\
+  --private-key $L2_PRIVATE_KEY \\
+  src/Counter.sol:Counter

A successful deployment will return output similar to below:

bash
[⠆] Compiling...
+No files changed, compilation skipped
+Deployer: 0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266
+Deployed to: 0x5FbDB2315678afecb367f032d93F642f64180aa3
+Transaction hash: 0xf1a793a793cd9fc588f5132d99008565ea361eb3535d66499575e9e1908200b2
[⠆] Compiling...
+No files changed, compilation skipped
+Deployer: 0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266
+Deployed to: 0x5FbDB2315678afecb367f032d93F642f64180aa3
+Transaction hash: 0xf1a793a793cd9fc588f5132d99008565ea361eb3535d66499575e9e1908200b2

Once you've deployed the contract, you're ready to interact with it!

First, we'll set it as a variable:

bash
export CONTRACT_ADDRESS=0x5FbDB2315678afecb367f032d93F642f64180aa3
export CONTRACT_ADDRESS=0x5FbDB2315678afecb367f032d93F642f64180aa3

Interacting with your smart contract

Foundry uses cast, a CLI for performing Ethereum RPC calls.

To write to the contract, we'll use the cast send command:

bash
cast send $CONTRACT_ADDRESS "setNumber(uint256)" 10 \\
+  --rpc-url $ARB_RPC_URL --private-key $L2_PRIVATE_KEY
cast send $CONTRACT_ADDRESS "setNumber(uint256)" 10 \\
+  --rpc-url $ARB_RPC_URL --private-key $L2_PRIVATE_KEY

Your output will look similar:

bash
blockHash               0x131822bef6eb59656d7e1387c19b75be667e587006710365ec5cf58030786c42
+blockNumber             3
+contractAddress
+cumulativeGasUsed       43494
+effectiveGasPrice       3767182372
+gasUsed                 43494
+logs                    []
+logsBloom               0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+root
+status                  1
+transactionHash         0x8f15d6004598f0662dd673a9898dceef77be8cc28408cecc284b28d7be32307d
+transactionIndex        0
+type                    2
blockHash               0x131822bef6eb59656d7e1387c19b75be667e587006710365ec5cf58030786c42
+blockNumber             3
+contractAddress
+cumulativeGasUsed       43494
+effectiveGasPrice       3767182372
+gasUsed                 43494
+logs                    []
+logsBloom               0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+root
+status                  1
+transactionHash         0x8f15d6004598f0662dd673a9898dceef77be8cc28408cecc284b28d7be32307d
+transactionIndex        0
+type                    2

Now, we can make a read call to view the state of the number variable, using the cast call command:

bash
cast call $CONTRACT_ADDRESS "number()" --rpc-url $ARB_RPC_URL
cast call $CONTRACT_ADDRESS "number()" --rpc-url $ARB_RPC_URL

The result will look similar:

bash
0x000000000000000000000000000000000000000000000000000000000000000a
0x000000000000000000000000000000000000000000000000000000000000000a

Convert the result from hexadecimal to a base 10 value with:

bash
echo $((0x000000000000000000000000000000000000000000000000000000000000000a))
echo $((0x000000000000000000000000000000000000000000000000000000000000000a))

Next steps

Congratulations! You've learned how to deploy a smart contract to your Arbitrum rollup devnet.

What will you build next? In our next tutorial, we will be going over how to deploy a dapp to your Arbitrum rollup.

`,60),e=[p];function t(c,r,i,y,E,u){return a(),n("div",null,e)}const h=s(l,[["render",t]]);export{F as __pageData,h as default}; diff --git a/assets/developers_arbitrum-smart-contract.md.3069abf8.lean.js b/assets/developers_arbitrum-smart-contract.md.3069abf8.lean.js new file mode 100644 index 00000000000..82334ed0092 --- /dev/null +++ b/assets/developers_arbitrum-smart-contract.md.3069abf8.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as o}from"./chunks/framework.51d6c45b.js";const F=JSON.parse('{"title":"Deploy a smart contract to your Arbitrum rollup","description":"A tutorial that guides you through the process of deploying a smart contract to your Arbitrum rollup using a L2 Nitro devnet, including setting up the environment, creating and testing the smart contract, and interacting with the deployed contract.","frontmatter":{"description":"A tutorial that guides you through the process of deploying a smart contract to your Arbitrum rollup using a L2 Nitro devnet, including setting up the environment, creating and testing the smart contract, and interacting with the deployed contract.","head":[["meta",{"name":"og:title","content":"Deploy a smart contract to your Arbitrum rollup | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/arbitrum-smart-contract.md","filePath":"developers/arbitrum-smart-contract.md","lastUpdated":1703278090000}'),l={name:"developers/arbitrum-smart-contract.md"},p=o("",60),e=[p];function t(c,r,i,y,E,u){return a(),n("div",null,e)}const h=s(l,[["render",t]]);export{F as __pageData,h as default}; diff --git a/assets/developers_blobstream-contracts.md.8c20e3c9.js b/assets/developers_blobstream-contracts.md.8c20e3c9.js new file mode 100644 index 00000000000..d575585b2ef --- /dev/null +++ b/assets/developers_blobstream-contracts.md.8c20e3c9.js @@ -0,0 +1,89 @@ +import{_ as s,o as a,c as o,Q as e}from"./chunks/framework.51d6c45b.js";const d=JSON.parse(`{"title":"Integrate with Blobstream contracts","description":"Learn how to integrate your L2's onchain logic with Blobstream","frontmatter":{"sidebar_label":"Integrate with Blobstream contracts","description":"Learn how to integrate your L2's onchain logic with Blobstream","head":[["meta",{"name":"og:title","content":"Integrate with Blobstream contracts | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstream-contracts.md","filePath":"developers/blobstream-contracts.md","lastUpdated":1709911265000}`),n={name:"developers/blobstream-contracts.md"},l=e(`

Integrate with Blobstream contracts

Getting started

Prerequisites

Make sure to have the following installed:

Installing Blobstream X contracts

We will be using the Blobstream X implementation of Blobstream, so we can install its repo as a dependency:

Install the Blobstream X contracts repo as a dependency:

sh
forge install succinctlabs/blobstreamx --no-commit
forge install succinctlabs/blobstreamx --no-commit

Make sure that the directory you're running this command from is an initialized git repository. If not, just initialize the repo using:

sh
git init
git init

Note that the minimum Solidity compiler version for using the Blobstream contracts is 0.8.19.

Example usage

Example minimal Solidity contract for a stub ZK rollup that leverages the BlobstreamX.sol contract to check that data has been posted to Celestia:

solidity
// SPDX-License-Identifier: Apache-2.0
+pragma solidity ^0.8.19;
+
+TBD
+import "blobstream-contracts/IDAOracle.sol";
+import "blobstream-contracts/DataRootTuple.sol";
+import "blobstream-contracts/lib/tree/binary/BinaryMerkleProof.sol";
+
+contract MyRollup {
+    IDAOracle immutable blobstreamX;
+    bytes32[] public rollup_block_hashes;
+
+    constructor(IDAOracle _blobstreamX) {
+        blobstreamX = _blobstreamX;
+    }
+
+    function submitRollupBlock(
+        bytes32 _rollup_block_hash,
+        bytes calldata _zk_proof,
+        uint256 _blobstreamX_nonce,
+        DataRootTuple calldata _tuple,
+        BinaryMerkleProof calldata _proof
+    ) public {
+        // Verify that the data root tuple (analog. block header) has been
+        // attested to by the Blobstream contract.
+        require(
+            blobstreamX.verifyAttestation(_blobstreamX_nonce, _tuple, _proof)
+        );
+
+        // Verify the ZKP (zero-knowledge proof).
+        // _tuple.dataRoot is a public input, leaves (shares) are private inputs.
+        require(verifyZKP(_rollup_block_hash, _zk_proof, _tuple.dataRoot));
+
+        // Everything checks out, append rollup block hash to list.
+        rollup_block_hashes.push(_rollup_block_hash);
+    }
+
+    function verifyZKP(
+        bytes32 _rollup_block_hash,
+        bytes calldata _zk_proof,
+        bytes32 _data_root
+    ) private pure returns (bool) {
+        return true;
+    }
+}
// SPDX-License-Identifier: Apache-2.0
+pragma solidity ^0.8.19;
+
+TBD
+import "blobstream-contracts/IDAOracle.sol";
+import "blobstream-contracts/DataRootTuple.sol";
+import "blobstream-contracts/lib/tree/binary/BinaryMerkleProof.sol";
+
+contract MyRollup {
+    IDAOracle immutable blobstreamX;
+    bytes32[] public rollup_block_hashes;
+
+    constructor(IDAOracle _blobstreamX) {
+        blobstreamX = _blobstreamX;
+    }
+
+    function submitRollupBlock(
+        bytes32 _rollup_block_hash,
+        bytes calldata _zk_proof,
+        uint256 _blobstreamX_nonce,
+        DataRootTuple calldata _tuple,
+        BinaryMerkleProof calldata _proof
+    ) public {
+        // Verify that the data root tuple (analog. block header) has been
+        // attested to by the Blobstream contract.
+        require(
+            blobstreamX.verifyAttestation(_blobstreamX_nonce, _tuple, _proof)
+        );
+
+        // Verify the ZKP (zero-knowledge proof).
+        // _tuple.dataRoot is a public input, leaves (shares) are private inputs.
+        require(verifyZKP(_rollup_block_hash, _zk_proof, _tuple.dataRoot));
+
+        // Everything checks out, append rollup block hash to list.
+        rollup_block_hashes.push(_rollup_block_hash);
+    }
+
+    function verifyZKP(
+        bytes32 _rollup_block_hash,
+        bytes calldata _zk_proof,
+        bytes32 _data_root
+    ) private pure returns (bool) {
+        return true;
+    }
+}

Data structures

Each DataRootTuple is a tuple of block height and data root. It is analogous to a Celestia block header. DataRootTuples are relayed in batches, committed to as a DataRootTuples root (i.e. a Merkle root of DataRootTuples).

The BinaryMerkleProof is an RFC-6962-compliant Merkle proof. Since DataRootTuples are Merkleized in a binary Merkle tree, verifying the inclusion of a DataRootTuple against a DataRootTuples root requires verifying a Merkle inclusion proof.

Interface

The IDAOracle (Data Availability Oracle Interface) interface allows L2 contracts on Ethereum to query the BlobstreamX.sol contract for relayed DataRootTuples. The single interface method verifyAttestation verifies a Merkle inclusion proof that a DataRootTuple is included under a specific batch (indexed by batch nonce). In other words, analogously it verifies that a specific block header is included in the Celestia chain.

Querying the proof

To prove that the data was published to Celestia, check out the proof queries documentation to understand how to query the proofs from Celestia consensus nodes and make them usable in the Blobstream X verifier contract.

Verifying data inclusion for fraud proofs

A high-level overview of how a fraud-proof based L2 would interact with Blobstream can be found in the inclusion proofs documentation.

The DAVerifier library is available at blobstream-contracts/lib/verifier/DAVerifier.sol, and provides functions to verify the inclusion of individual (or multiple) shares against a DataRootTuple. The library is stateless, and allows to pass an IDAOracle interface as a parameter to verify inclusion against it.

In the DAVerifier library, we find functions that help with data inclusion verification and calculating the square size of a Celestia block. These functions work with the Blobstream X smart contract, using different proofs to check and confirm the data's availability. Let's take a closer look at these functions:

  • verifySharesToDataRootTupleRoot: This function verifies that the shares, which were posted to Celestia, were committed to by the Blobstream X smart contract. It checks that the data root was committed to by the Blobstream X smart contract and that the shares were committed to by the rows roots.
  • verifyRowRootToDataRootTupleRoot: This function verifies that a row/column root, from a Celestia block, was committed to by the Blobstream X smart contract. It checks that the data root was committed to by the Blobstream X smart contract and that the row root commits to the data root.
  • verifyMultiRowRootsToDataRootTupleRoot: This function verifies that a set of rows/columns, from a Celestia block, were committed to by the Blobstream X smart contract. It checks that the data root was committed to by the Blobstream X smart contract and that the rows roots commit to the data root.
  • computeSquareSizeFromRowProof: This function computes the Celestia block square size from a row/column root to data root binary Merkle proof. It is the user's responsibility to verify that the proof is valid and was successfully committed to using the verifyRowRootToDataRootTupleRoot() method.
  • computeSquareSizeFromShareProof: This function computes the Celestia block square size from a shares to row/column root proof. It is the user's responsibility to verify that the proof is valid and that the shares were successfully committed to using the verifySharesToDataRootTupleRoot() method.

For an overview of a demo rollup implementation, head to the next section.

`,28),t=[l];function r(p,c,i,y,E,h){return a(),o("div",null,t)}const u=s(n,[["render",r]]);export{d as __pageData,u as default}; diff --git a/assets/developers_blobstream-contracts.md.8c20e3c9.lean.js b/assets/developers_blobstream-contracts.md.8c20e3c9.lean.js new file mode 100644 index 00000000000..8ab4d058eb0 --- /dev/null +++ b/assets/developers_blobstream-contracts.md.8c20e3c9.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as o,Q as e}from"./chunks/framework.51d6c45b.js";const d=JSON.parse(`{"title":"Integrate with Blobstream contracts","description":"Learn how to integrate your L2's onchain logic with Blobstream","frontmatter":{"sidebar_label":"Integrate with Blobstream contracts","description":"Learn how to integrate your L2's onchain logic with Blobstream","head":[["meta",{"name":"og:title","content":"Integrate with Blobstream contracts | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstream-contracts.md","filePath":"developers/blobstream-contracts.md","lastUpdated":1709911265000}`),n={name:"developers/blobstream-contracts.md"},l=e("",28),t=[l];function r(p,c,i,y,E,h){return a(),o("div",null,t)}const u=s(n,[["render",r]]);export{d as __pageData,u as default}; diff --git a/assets/developers_blobstream-offchain.md.162a7fc9.js b/assets/developers_blobstream-offchain.md.162a7fc9.js new file mode 100644 index 00000000000..251e0050f44 --- /dev/null +++ b/assets/developers_blobstream-offchain.md.162a7fc9.js @@ -0,0 +1,297 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.51d6c45b.js";const u=JSON.parse(`{"title":"Integrate with Blobstream client","description":"Learn how to integrate your L2's offchain logic with Blobstream","frontmatter":{"description":"Learn how to integrate your L2's offchain logic with Blobstream","head":[["meta",{"name":"og:title","content":"Integrate with Blobstream client | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstream-offchain.md","filePath":"developers/blobstream-offchain.md","lastUpdated":1713191709000}`),e={name:"developers/blobstream-offchain.md"},o=l(`

Integrate with Blobstream client

Blobstream demo rollup

Rollups can use Blobstream for DA by posting their data to Celestia and then proving that it was posted on Ethereum. This is done identically to how any rollup or user would post data to Celestia. Then, a zero-knowledge proof that Celestia validators have come to consensus on Celestia block headers is generated, and subsequently relayed to Ethereum to the Blobstream X smart contract.

This demo rollup will outline (the outline is not an implementation! Please do not expect to copy and paste this code 🙂) a very simple Blobstream rollup to illustrate at a high level what this could look like.

Defining a chain

The first step to starting a new chain is to define the structure of the commitments that each block consists of.

go
type Block struct {
+    // Data is the data of a block that is submitted to Celestia.
+    Data \`json:"Data"\`
+    // Header is the set of commitments over a block that is submitted to
+    // Ethereum.
+    Header \`json:"Header"\`
+}
+
+// Data is the data of a block that is submitted to Celestia.
+type Data struct {
+    Txs []json.RawMessage \`json:"txs"\`
+}
+
+// Header is the set of commitments over a block that is submitted to Ethereum.
+type Header struct {
+    Height             uint64    \`json:"height"\`
+    Namespace          []byte    \`json:"namespace"\`
+    PreviousHash       []byte    \`json:"previous_hash"\`
+    Span               Span      \`json:"span"\`
+    SequencerSignature Signature \`json:"sequencer_signature,omitempty"\`
+}
type Block struct {
+    // Data is the data of a block that is submitted to Celestia.
+    Data \`json:"Data"\`
+    // Header is the set of commitments over a block that is submitted to
+    // Ethereum.
+    Header \`json:"Header"\`
+}
+
+// Data is the data of a block that is submitted to Celestia.
+type Data struct {
+    Txs []json.RawMessage \`json:"txs"\`
+}
+
+// Header is the set of commitments over a block that is submitted to Ethereum.
+type Header struct {
+    Height             uint64    \`json:"height"\`
+    Namespace          []byte    \`json:"namespace"\`
+    PreviousHash       []byte    \`json:"previous_hash"\`
+    Span               Span      \`json:"span"\`
+    SequencerSignature Signature \`json:"sequencer_signature,omitempty"\`
+}

Note the Celestia-specific structures in the header such as the Namespace and the Blobstream-specific structure called the Span. The goal of these structures is to locate the data in the Celestia block so that we can prove that data's inclusion via Blobstream if needed. Read more in the namespace specifications, and you can think of this like a chain ID. Learn more information about shares, which are small pieces of the encoded Celestia block. We use the same encoding here so that the commitments to the rollup block match those committed to by validators in the Celestia data root.

The Span could take many forms, but in this demo, we will use the following:

go
// Span describes the location of the rollup block data that is posted to
+// Celestia. This is important for other nodes to be able to prove that data in
+// the Celestia block. This can be thought of as a pointer to some data in the
+// Celestia block.
+type Span struct {
+    // CelestiaHeight is the height of the Celestia block that contains the
+    // rollup block data.
+    CelestiaHeight uint64 \`json:"celestia_height"\`
+    // DataShareStart is the index of the first share of the rollup block data.
+    DataShareStart uint64 \`json:"share_start"\`
+    // DataShareLen is length in shares of the rollup block data. This is used
+    // to identify all of the rollup block data in a Celestia block.
+    DataShareLen uint64 \`json:"share_end"\`
+}
// Span describes the location of the rollup block data that is posted to
+// Celestia. This is important for other nodes to be able to prove that data in
+// the Celestia block. This can be thought of as a pointer to some data in the
+// Celestia block.
+type Span struct {
+    // CelestiaHeight is the height of the Celestia block that contains the
+    // rollup block data.
+    CelestiaHeight uint64 \`json:"celestia_height"\`
+    // DataShareStart is the index of the first share of the rollup block data.
+    DataShareStart uint64 \`json:"share_start"\`
+    // DataShareLen is length in shares of the rollup block data. This is used
+    // to identify all of the rollup block data in a Celestia block.
+    DataShareLen uint64 \`json:"share_end"\`
+}

We can then define the blockchain as a collection of blocks and some additional information about the chain such as the sequencer address.

go
type Blockchain struct {
+    Blocks           []Block
+    SequencerAddress []byte
+    Namespace        []byte
+}
type Blockchain struct {
+    Blocks           []Block
+    SequencerAddress []byte
+    Namespace        []byte
+}

Rollup sequencer

The rollup sequencer is responsible for creating blocks and, in this demo, writing that data to Celestia and Ethereum. The rollup full node is responsible for reading that data from Celestia and Ethereum and verifying that it follows the protocol rules of that rollup.

Therefore, we can start by first defining the reading and writing interactions rollup nodes will have with both the Celestia and Ethereum networks. The actual implementations of these interfaces are left as exercises to the reader (🤪). Assume that those implementations of these interfaces are verifying the respective chain. For the connection to Celestia, this would likely mean connecting to a Celestia light node, which can detect faults in consensus such as hidden data. For the connection to Ethereum, this would likely mean running and connecting to a full node. More information on the RPC that is exposed by a Celestia light node can be found in the RPC documentation. Additionally, if you need more information on how to run a light node, you can check out the documentation.

go
// CelestiaLightNodeClient summarizes the actions that a rollup that uses
+// Blobstream for DA would need from a Celestia light node. Note that the actual
+// connection to this light node is arbitrary, but would likely involve an RPC
+// connection to a Celestia light node.
+type CelestiaLightNodeClient interface {
+    GetBlockData(Span) (Data, error)
+    SubmitBlockData(Data) (Span, error)
+}
+
+// EthereumClient summarizes the actions that a rollup that uses Blobstream for
+// DA would need from an Ethereum client.
+type EthereumClient interface {
+    // GetLatestRollupHeight returns the height of the latest rollup block by
+    // querying the appropriate contract on Ethereum.
+    LatestRollupHeight() (uint64, error)
+    // GetHeader returns the rollup header of a specific height.
+    GetHeader(uint64) (Header, error)
+    // SubmitHeader submits a header to the rollup bridge contract on Ethereum.
+    SubmitHeader(Header) error
+}
// CelestiaLightNodeClient summarizes the actions that a rollup that uses
+// Blobstream for DA would need from a Celestia light node. Note that the actual
+// connection to this light node is arbitrary, but would likely involve an RPC
+// connection to a Celestia light node.
+type CelestiaLightNodeClient interface {
+    GetBlockData(Span) (Data, error)
+    SubmitBlockData(Data) (Span, error)
+}
+
+// EthereumClient summarizes the actions that a rollup that uses Blobstream for
+// DA would need from an Ethereum client.
+type EthereumClient interface {
+    // GetLatestRollupHeight returns the height of the latest rollup block by
+    // querying the appropriate contract on Ethereum.
+    LatestRollupHeight() (uint64, error)
+    // GetHeader returns the rollup header of a specific height.
+    GetHeader(uint64) (Header, error)
+    // SubmitHeader submits a header to the rollup bridge contract on Ethereum.
+    SubmitHeader(Header) error
+}

Note that here we are waiting for the head to be posted to Ethereum, however, it would likely be better to simply download that header from the p2p network or directly from the sequencer instead.

For the purposes of this demo, we will be using a single centralized sequencer, which can be defined by simply wrapping the full node to isolate the logic to create blocks.

A rollup full node will just consist of some representation of a blockchain along with clients to read from with Celestia and Ethereum.

go
type Fullnode struct {
+    Blockchain
+    CelestiaLightNodeClient
+    EthereumClient
+}
+
+// Sequencer wraps the demo Fullnode struct to add specific functionality for
+// producing blocks.
+type Sequencer struct {
+    Fullnode
+}
type Fullnode struct {
+    Blockchain
+    CelestiaLightNodeClient
+    EthereumClient
+}
+
+// Sequencer wraps the demo Fullnode struct to add specific functionality for
+// producing blocks.
+type Sequencer struct {
+    Fullnode
+}

Committing to data

Typical blockchains commit to the transactions included in each block using a Merkle root. Rollups that use Blobstream for DA need to use the commitments that are relayed to the Blobstream contracts.

For optimistic rollups, this could be as simple as referencing the data in the Celestia block, not unlike using a pointer in memory. This is what is done below via a Span in the creating blocks section. We keep track of where the data is located in the Celestia block and the sequencer signs over that location in the header. If the sequencer commits to non-existent data or an invalid state root, then the invalid transaction is first proved to be included in the Span before the rest of the fraud proof process is followed. Find more information in the inclusion proofs documentation.

For zk rollups, this would involve creating an inclusion proof to the data root tuple root in the Blobstream contracts and then verifying that proof in the zk proof used to verify state. Find more information in the data root inclusion proof documentation.

Also, see the documentation for the data square layout and the shares of the Celestia block to see how the data is encoded in Celestia.

Creating blocks

The first step in creating a block is to post the block data to Celestia. Upon confirmation of the data being included in a block, the actual location of the data in Celestia can be determined. This data is used to create a Span which is included in the header and signed over by the sequencer. This Span can be used by contracts on Ethereum that use the Blobstream contracts to prove some specific data was included.

go
func (s *Sequencer) ProduceBlock(txs []json.RawMessage) (Block, error) {
+    data := Data{Txs: txs}
+
+    span, err := s.CelestiaLightNodeClient.SubmitBlockData(data)
+    if err != nil {
+        return Block{}, err
+    }
+
+    var lastBlock Block
+    if len(s.Blocks) > 0 {
+        lastBlock = s.Blocks[len(s.Blocks)-1]
+    }
+
+    header := Header{
+        Height:       uint64(len(s.Blocks) + 1),
+        PreviousHash: lastBlock.Header.Hash(),
+        Namespce:     s.Namespace,
+        Span: span,
+    }
+
+    signature := s.key.Sign(header.SignBytes())
+
+    header.SequencerSignature = signature
+
+    block := Block{
+        Data:   data,
+        Header: header,
+    }
+
+    s.AddBlock(block)
+
+    return block, nil
+}
func (s *Sequencer) ProduceBlock(txs []json.RawMessage) (Block, error) {
+    data := Data{Txs: txs}
+
+    span, err := s.CelestiaLightNodeClient.SubmitBlockData(data)
+    if err != nil {
+        return Block{}, err
+    }
+
+    var lastBlock Block
+    if len(s.Blocks) > 0 {
+        lastBlock = s.Blocks[len(s.Blocks)-1]
+    }
+
+    header := Header{
+        Height:       uint64(len(s.Blocks) + 1),
+        PreviousHash: lastBlock.Header.Hash(),
+        Namespce:     s.Namespace,
+        Span: span,
+    }
+
+    signature := s.key.Sign(header.SignBytes())
+
+    header.SequencerSignature = signature
+
+    block := Block{
+        Data:   data,
+        Header: header,
+    }
+
+    s.AddBlock(block)
+
+    return block, nil
+}

Note that the sequencer here is not yet posting headers to Ethereum. This is because the sequencer is waiting for the commitments from the Celestia validator set (the data root tuple roots) to be relayed to the contracts. Once the contracts are updated, the sequencer can post the header to Ethereum.

go
func (s *Sequencer) UpdateHeaders() error {
+    latestRollupHeight, err := s.EthereumClient.LatestRollupHeight()
+    if err != nil {
+        return err
+    }
+
+    for i := latestRollupHeight; i <= uint64(len(s.Blocks)+1); i++ {
+        err := s.EthereumClient.SubmitHeader(s.Blocks[i].Header)
+        if err != nil {
+            return err
+        }
+    }
+
+    return nil
+}
func (s *Sequencer) UpdateHeaders() error {
+    latestRollupHeight, err := s.EthereumClient.LatestRollupHeight()
+    if err != nil {
+        return err
+    }
+
+    for i := latestRollupHeight; i <= uint64(len(s.Blocks)+1); i++ {
+        err := s.EthereumClient.SubmitHeader(s.Blocks[i].Header)
+        if err != nil {
+            return err
+        }
+    }
+
+    return nil
+}

Rollup full node

Downloading the block

There are a few different mechanisms that could be used to download blocks. The simplest solution and what is outlined above is for Fullnodes to wait until the blocks and the headers are posted to the respective chains, and then download each as they are posted. It would also be possible to gossip the headers ahead of time and download the rollup blocks from Celestia instead of waiting for the headers to be posted to Ethereum. It's also possible to download the headers and the block data like a normal blockchain via a gossiping network and only fall back to downloading the data and headers from Celestia and Ethereum if the gossiping network is unavailable or the sequencer is malicious.

go
func (f *Fullnode) AddBlock(b Block) error {
+    // Perform validation of the block
+    if b.Header.Height != uint64(len(f.Blocks)+1) {
+        return fmt.Errorf("failure to add block: expected block height %d, got %d", len(f.Blocks)+1, b.Header.Height)
+    }
+    // Check the sequencer's signature
+    if !b.Header.SequencerSignature.IsValid(f.SequencerAddress) {
+        return fmt.Errorf("failure to add block: invalid sequencer signature")
+    }
+
+    f.Blocks = append(f.Blocks, b)
+    return nil
+}
+
+func (f *Fullnode) GetLatestBlock() error {
+    nextHeight := uint64(len(f.Blocks) + 1)
+
+    // Download the next header from ethereum before we download the block data
+    // from Celestia. Note that we could alternatively download the header
+    // directly from the sequencer instead of waiting.
+    header, err := f.EthereumClient.GetHeader(nextHeight)
+    if err != nil {
+        return err
+    }
+
+    data, err := f.CelestiaLightNodeClient.GetBlockData(header.Span)
+    if err != nil {
+        return err
+    }
+
+    return f.AddBlock(
+        Block{
+            Data:   data,
+            Header: header,
+        },
+    )
+}
func (f *Fullnode) AddBlock(b Block) error {
+    // Perform validation of the block
+    if b.Header.Height != uint64(len(f.Blocks)+1) {
+        return fmt.Errorf("failure to add block: expected block height %d, got %d", len(f.Blocks)+1, b.Header.Height)
+    }
+    // Check the sequencer's signature
+    if !b.Header.SequencerSignature.IsValid(f.SequencerAddress) {
+        return fmt.Errorf("failure to add block: invalid sequencer signature")
+    }
+
+    f.Blocks = append(f.Blocks, b)
+    return nil
+}
+
+func (f *Fullnode) GetLatestBlock() error {
+    nextHeight := uint64(len(f.Blocks) + 1)
+
+    // Download the next header from ethereum before we download the block data
+    // from Celestia. Note that we could alternatively download the header
+    // directly from the sequencer instead of waiting.
+    header, err := f.EthereumClient.GetHeader(nextHeight)
+    if err != nil {
+        return err
+    }
+
+    data, err := f.CelestiaLightNodeClient.GetBlockData(header.Span)
+    if err != nil {
+        return err
+    }
+
+    return f.AddBlock(
+        Block{
+            Data:   data,
+            Header: header,
+        },
+    )
+}

This outline of a Blobstream rollup isn't doing execution or state transitions induced by the transactions, however, that step would occur here. If fraud is detected, the fraud proof process will begin. The only difference between the fraud proof process of a normal optimistic rollup and a rollup that uses Blobstream for DA is that the full node would first prove the fraudulent transaction was committed to by the Sequencer using the Span in the header and before proceeding with the normal process.

More documentation

Proving inclusion via Blobstream

Blobstream inclusion proof docs and the verifier helper contracts.

Submitting block data to Celestia via light node

As linked above, use the Celestia light node RPC to submit the data to Celestia.

Posting headers to Ethereum

How headers are posted to Ethereum is entirely dependent upon how the rollup light client contracts work. For examples of interacting with the Ethereum blockchain programmatically, please see the go-ethereum book or one of the many other resources for submitting transactions or writing contracts.

`,42),p=[o];function t(r,c,i,E,y,h){return a(),n("div",null,p)}const b=s(e,[["render",t]]);export{u as __pageData,b as default}; diff --git a/assets/developers_blobstream-offchain.md.162a7fc9.lean.js b/assets/developers_blobstream-offchain.md.162a7fc9.lean.js new file mode 100644 index 00000000000..3cc2e653fc7 --- /dev/null +++ b/assets/developers_blobstream-offchain.md.162a7fc9.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.51d6c45b.js";const u=JSON.parse(`{"title":"Integrate with Blobstream client","description":"Learn how to integrate your L2's offchain logic with Blobstream","frontmatter":{"description":"Learn how to integrate your L2's offchain logic with Blobstream","head":[["meta",{"name":"og:title","content":"Integrate with Blobstream client | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstream-offchain.md","filePath":"developers/blobstream-offchain.md","lastUpdated":1713191709000}`),e={name:"developers/blobstream-offchain.md"},o=l("",42),p=[o];function t(r,c,i,E,y,h){return a(),n("div",null,p)}const b=s(e,[["render",t]]);export{u as __pageData,b as default}; diff --git a/assets/developers_blobstream-proof-queries.md.4443c323.js b/assets/developers_blobstream-proof-queries.md.4443c323.js new file mode 100644 index 00000000000..5a7c43a8c89 --- /dev/null +++ b/assets/developers_blobstream-proof-queries.md.4443c323.js @@ -0,0 +1,1449 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.51d6c45b.js";const o="/img/blobstream/blobstream-square.png",p="/img/blobstream/blobstream-commitment-diagram.png",m=JSON.parse('{"title":"Blobstream proofs queries","description":"Learn how to query the inclusion proofs used in Blobstream","frontmatter":{"description":"Learn how to query the inclusion proofs used in Blobstream","head":[["meta",{"name":"og:title","content":"Blobstream proofs queries | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstream-proof-queries.md","filePath":"developers/blobstream-proof-queries.md","lastUpdated":1713191709000}'),e={name:"developers/blobstream-proof-queries.md"},t=l('

Blobstream proofs queries

Prerequisites

  • Access to a Celestia full consensus node RPC endpoint (or full node). The node doesn't need to be a validating node in order for the proofs to be queried. A full node is enough.

Overview of the proof queries

To prove the inclusion of PayForBlobs (PFB) transactions, blobs or shares, committed to in a Celestia block, we use the Celestia consensus node's RPC to query for proofs that can be verified in a rollup settlement contract via Blobstream. In fact, when a PFB transaction is included in a block, it gets separated into a PFB transaction (without the blob), and the actual data blob that it carries. These two are split into shares, which are the low level constructs of a Celestia block, and saved to the corresponding Celestia block. Learn more about shares in the shares specs.

The two diagrams below summarize how a single share, which can contain a PFB transaction, or a part of the rollup data that was posted using a PFB, is committed to in Blobstream.

The share is highlighted in green. R0, R1 etc, represent the respective row and column roots, the blue and pink gradients are erasure encoded data. More details on the square layout can be found in the data square layout and data structures portion of the specs.

The Celestia square

Square

The commitment scheme

Blobstream Commitment Diagram

So to prove inclusion of a share to a Celestia block, we use Blobstream as a source of truth. Currently, we will be using the Blobstream X implementation of Blobstream, more information on Blobstream X can be found in the overview. In a nutshell, Blobstream X attests to the data posted to Celestia in the Blobstream X contract via verifying a zk-proof of the headers of a batch of Celestia blocks. Then, it keeps reference of that batch of blocks using the merkleized commitment of their (dataRoot, height) resulting in a data root tuple root. Check the above diagram which shows:

  • 0: those are the shares, that when unified, contain the PFB or the rollup data blob.
  • 1: the row and column roots are the namespace merkle tree roots over the shares. More information on the NMT in the NMT specs. These commit to the rows and columns containing the above shares.
  • 2: the data roots: which are the binary merkle tree commitment over the row and column roots. This means that if you can prove that a share is part of a row, using a namespace merkle proof. Then prove that this row is committed to by the data root. Then you can be sure that that share was published to the corresponding block.
  • 3: in order to batch multiple blocks into the same commitment, we create a commitment over the (dataRoot, height) tuple for a batch of blocks, which results in a data root tuple root. It's this commitment that gets stored in the Blobstream X smart contract.

So, if we're able to prove that a share is part of a row, then that row is committed to by a data root. Then, prove that that data root along with its height is committed to by the data root tuple root, which gets saved to the Blobstream X contract, we can be sure that that share was committed to in the corresponding Celestia block.

In this document, we will provide details on how to query the above proofs, and how to adapt them to be sent to a rollup contract for verification.

Hands-on demonstration

This part will provide the details of proof generation, and the way to make the results of the proofs queries ready to be consumed by the target rollup contract.

NOTE

For the go client snippets, make sure to have the following replaces in your go.mod:

go
// go.mod
+    github.com/cosmos/cosmos-sdk => github.com/celestiaorg/cosmos-sdk v1.18.3-sdk-v0.46.14
+    github.com/gogo/protobuf => github.com/regen-network/protobuf v1.3.3-alpha.regen.1
+    github.com/syndtr/goleveldb => github.com/syndtr/goleveldb v1.0.1-0.20210819022825-2ae1ddf74ef7
+    github.com/tendermint/tendermint => github.com/celestiaorg/celestia-core v1.32.0-tm-v0.34.29
+
+)
// go.mod
+    github.com/cosmos/cosmos-sdk => github.com/celestiaorg/cosmos-sdk v1.18.3-sdk-v0.46.14
+    github.com/gogo/protobuf => github.com/regen-network/protobuf v1.3.3-alpha.regen.1
+    github.com/syndtr/goleveldb => github.com/syndtr/goleveldb v1.0.1-0.20210819022825-2ae1ddf74ef7
+    github.com/tendermint/tendermint => github.com/celestiaorg/celestia-core v1.32.0-tm-v0.34.29
+
+)

Also, make sure to update the versions to match the latest github.com/celestiaorg/cosmos-sdk and github.com/celestiaorg/celestia-core versions.

1. Data root inclusion proof

To prove the data root is committed to by the Blobstream X smart contract, we will need to provide a Merkle proof of the data root tuple to a data root tuple root. This can be created using the data_root_inclusion_proof query.

This endpoint allows querying a data root to data root tuple root proof. It takes a block height, a starting block, and an end block, then it generates the binary Merkle proof of the DataRootTuple, corresponding to that height, to the DataRootTupleRoot which is committed to in the Blobstream X contract.

The endpoint can be queried using the golang client:

go
package main
+
+import (
+	"context"
+	"fmt"
+	"github.com/tendermint/tendermint/rpc/client/http"
+	"os"
+)
+
+func main() {
+	ctx := context.Background()
+	trpc, err := http.New("tcp://localhost:26657", "/websocket")
+	if err != nil {
+		fmt.Println(err)
+		os.Exit(1)
+	}
+	err = trpc.Start()
+	if err != nil {
+		fmt.Println(err)
+		os.Exit(1)
+	}
+	dcProof, err := trpc.DataRootInclusionProof(ctx, 15, 10, 20)
+	if err != nil {
+		fmt.Println(err)
+		os.Exit(1)
+	}
+	fmt.Println(dcProof.Proof.String())
+}
package main
+
+import (
+	"context"
+	"fmt"
+	"github.com/tendermint/tendermint/rpc/client/http"
+	"os"
+)
+
+func main() {
+	ctx := context.Background()
+	trpc, err := http.New("tcp://localhost:26657", "/websocket")
+	if err != nil {
+		fmt.Println(err)
+		os.Exit(1)
+	}
+	err = trpc.Start()
+	if err != nil {
+		fmt.Println(err)
+		os.Exit(1)
+	}
+	dcProof, err := trpc.DataRootInclusionProof(ctx, 15, 10, 20)
+	if err != nil {
+		fmt.Println(err)
+		os.Exit(1)
+	}
+	fmt.Println(dcProof.Proof.String())
+}

Full example of proving that a Celestia block was committed to by Blobstream X contract

go
package main
+
+import (
+	"context"
+	"fmt"
+	"github.com/celestiaorg/celestia-app/pkg/square"
+	"github.com/ethereum/go-ethereum/accounts/abi/bind"
+	ethcmn "github.com/ethereum/go-ethereum/common"
+	"github.com/ethereum/go-ethereum/ethclient"
+	blobstreamxwrapper "github.com/succinctlabs/blobstreamx/bindings"
+	"github.com/tendermint/tendermint/crypto/merkle"
+	"github.com/tendermint/tendermint/rpc/client/http"
+	"math/big"
+	"os"
+)
+
+func main() {
+	err := verify()
+	if err != nil {
+		fmt.Println(err)
+		os.Exit(1)
+	}
+}
+
+func verify() error {
+	ctx := context.Background()
+
+	// start the tendermint RPC client
+	trpc, err := http.New("tcp://localhost:26657", "/websocket")
+	if err != nil {
+		return err
+	}
+	err = trpc.Start()
+	if err != nil {
+		return err
+	}
+
+	// get the PayForBlob transaction that contains the published blob
+	tx, err := trpc.Tx(ctx, []byte("tx_hash"), true)
+	if err != nil {
+		return err
+	}
+
+	// get the block containing the PayForBlob transaction
+	blockRes, err := trpc.Block(ctx, &tx.Height)
+	if err != nil {
+		return err
+	}
+
+	// get the nonce corresponding to the block height that contains
+	// the PayForBlob transaction
+	// since BlobstreamX emits events when new batches are submitted,
+	// we will query the events
+	// and look for the range committing to the blob
+	// first, connect to an EVM RPC endpoint
+	ethClient, err := ethclient.Dial("evm_rpc_endpoint")
+	if err != nil {
+		return err
+	}
+	defer ethClient.Close()
+
+	// use the BlobstreamX contract binding
+	wrapper, err := blobstreamxwrapper.NewBlobstreamX(ethcmn.HexToAddress("contract_Address"), ethClient)
+	if err != nil {
+		return err
+	}
+
+	LatestBlockNumber, err := ethClient.BlockNumber(context.Background())
+	if err != nil {
+		return err
+	}
+
+	eventsIterator, err := wrapper.FilterDataCommitmentStored(
+		&bind.FilterOpts{
+			Context: ctx,
+			Start: LatestBlockNumber - 90000,
+			End: &LatestBlockNumber,
+		},
+		nil,
+		nil,
+		nil,
+	)
+	if err != nil {
+		return err
+	}
+
+	var event *blobstreamxwrapper.BlobstreamXDataCommitmentStored
+	for eventsIterator.Next() {
+		e := eventsIterator.Event
+		if int64(e.StartBlock) <= tx.Height && tx.Height < int64(e.EndBlock) {
+			event = &blobstreamxwrapper.BlobstreamXDataCommitmentStored{
+				ProofNonce:     e.ProofNonce,
+				StartBlock:     e.StartBlock,
+				EndBlock:       e.EndBlock,
+				DataCommitment: e.DataCommitment,
+			}
+			break
+		}
+	}
+	if err := eventsIterator.Error(); err != nil {
+		return err
+	}
+	err = eventsIterator.Close()
+	if err != nil {
+		return err
+	}
+	if event == nil {
+		return fmt.Errorf("couldn't find range containing the transaction height")
+	}
+
+	// get the block data root inclusion proof to the data root tuple root
+	dcProof, err := trpc.DataRootInclusionProof(ctx, uint64(tx.Height), event.StartBlock, event.EndBlock)
+	if err != nil {
+		return err
+	}
+
+	// verify that the data root was committed to by the BlobstreamX contract
+	committed, err := VerifyDataRootInclusion(ctx, wrapper, event.ProofNonce.Uint64(), uint64(tx.Height), blockRes.Block.DataHash, dcProof.Proof)
+	if err != nil {
+		return err
+	}
+	if committed {
+		fmt.Println("data root was committed to by the BlobstreamX contract")
+	} else {
+		fmt.Println("data root was not committed to by the BlobstreamX contract")
+		return nil
+	}
+    return nil
+}
+
+func VerifyDataRootInclusion(
+	_ context.Context,
+	blobstreamXwrapper *blobstreamxwrapper.BlobstreamX,
+	nonce uint64,
+	height uint64,
+	dataRoot []byte,
+	proof merkle.Proof,
+) (bool, error) {
+	tuple := blobstreamxwrapper.DataRootTuple{
+		Height:   big.NewInt(int64(height)),
+		DataRoot: *(*[32]byte)(dataRoot),
+	}
+
+	sideNodes := make([][32]byte, len(proof.Aunts))
+	for i, aunt := range proof.Aunts {
+		sideNodes[i] = *(*[32]byte)(aunt)
+	}
+	wrappedProof := blobstreamxwrapper.BinaryMerkleProof{
+		SideNodes: sideNodes,
+		Key:       big.NewInt(proof.Index),
+		NumLeaves: big.NewInt(proof.Total),
+	}
+
+	valid, err := blobstreamXwrapper.VerifyAttestation(
+		&bind.CallOpts{},
+		big.NewInt(int64(nonce)),
+		tuple,
+		wrappedProof,
+	)
+	if err != nil {
+		return false, err
+	}
+	return valid, nil
+}
package main
+
+import (
+	"context"
+	"fmt"
+	"github.com/celestiaorg/celestia-app/pkg/square"
+	"github.com/ethereum/go-ethereum/accounts/abi/bind"
+	ethcmn "github.com/ethereum/go-ethereum/common"
+	"github.com/ethereum/go-ethereum/ethclient"
+	blobstreamxwrapper "github.com/succinctlabs/blobstreamx/bindings"
+	"github.com/tendermint/tendermint/crypto/merkle"
+	"github.com/tendermint/tendermint/rpc/client/http"
+	"math/big"
+	"os"
+)
+
+func main() {
+	err := verify()
+	if err != nil {
+		fmt.Println(err)
+		os.Exit(1)
+	}
+}
+
+func verify() error {
+	ctx := context.Background()
+
+	// start the tendermint RPC client
+	trpc, err := http.New("tcp://localhost:26657", "/websocket")
+	if err != nil {
+		return err
+	}
+	err = trpc.Start()
+	if err != nil {
+		return err
+	}
+
+	// get the PayForBlob transaction that contains the published blob
+	tx, err := trpc.Tx(ctx, []byte("tx_hash"), true)
+	if err != nil {
+		return err
+	}
+
+	// get the block containing the PayForBlob transaction
+	blockRes, err := trpc.Block(ctx, &tx.Height)
+	if err != nil {
+		return err
+	}
+
+	// get the nonce corresponding to the block height that contains
+	// the PayForBlob transaction
+	// since BlobstreamX emits events when new batches are submitted,
+	// we will query the events
+	// and look for the range committing to the blob
+	// first, connect to an EVM RPC endpoint
+	ethClient, err := ethclient.Dial("evm_rpc_endpoint")
+	if err != nil {
+		return err
+	}
+	defer ethClient.Close()
+
+	// use the BlobstreamX contract binding
+	wrapper, err := blobstreamxwrapper.NewBlobstreamX(ethcmn.HexToAddress("contract_Address"), ethClient)
+	if err != nil {
+		return err
+	}
+
+	LatestBlockNumber, err := ethClient.BlockNumber(context.Background())
+	if err != nil {
+		return err
+	}
+
+	eventsIterator, err := wrapper.FilterDataCommitmentStored(
+		&bind.FilterOpts{
+			Context: ctx,
+			Start: LatestBlockNumber - 90000,
+			End: &LatestBlockNumber,
+		},
+		nil,
+		nil,
+		nil,
+	)
+	if err != nil {
+		return err
+	}
+
+	var event *blobstreamxwrapper.BlobstreamXDataCommitmentStored
+	for eventsIterator.Next() {
+		e := eventsIterator.Event
+		if int64(e.StartBlock) <= tx.Height && tx.Height < int64(e.EndBlock) {
+			event = &blobstreamxwrapper.BlobstreamXDataCommitmentStored{
+				ProofNonce:     e.ProofNonce,
+				StartBlock:     e.StartBlock,
+				EndBlock:       e.EndBlock,
+				DataCommitment: e.DataCommitment,
+			}
+			break
+		}
+	}
+	if err := eventsIterator.Error(); err != nil {
+		return err
+	}
+	err = eventsIterator.Close()
+	if err != nil {
+		return err
+	}
+	if event == nil {
+		return fmt.Errorf("couldn't find range containing the transaction height")
+	}
+
+	// get the block data root inclusion proof to the data root tuple root
+	dcProof, err := trpc.DataRootInclusionProof(ctx, uint64(tx.Height), event.StartBlock, event.EndBlock)
+	if err != nil {
+		return err
+	}
+
+	// verify that the data root was committed to by the BlobstreamX contract
+	committed, err := VerifyDataRootInclusion(ctx, wrapper, event.ProofNonce.Uint64(), uint64(tx.Height), blockRes.Block.DataHash, dcProof.Proof)
+	if err != nil {
+		return err
+	}
+	if committed {
+		fmt.Println("data root was committed to by the BlobstreamX contract")
+	} else {
+		fmt.Println("data root was not committed to by the BlobstreamX contract")
+		return nil
+	}
+    return nil
+}
+
+func VerifyDataRootInclusion(
+	_ context.Context,
+	blobstreamXwrapper *blobstreamxwrapper.BlobstreamX,
+	nonce uint64,
+	height uint64,
+	dataRoot []byte,
+	proof merkle.Proof,
+) (bool, error) {
+	tuple := blobstreamxwrapper.DataRootTuple{
+		Height:   big.NewInt(int64(height)),
+		DataRoot: *(*[32]byte)(dataRoot),
+	}
+
+	sideNodes := make([][32]byte, len(proof.Aunts))
+	for i, aunt := range proof.Aunts {
+		sideNodes[i] = *(*[32]byte)(aunt)
+	}
+	wrappedProof := blobstreamxwrapper.BinaryMerkleProof{
+		SideNodes: sideNodes,
+		Key:       big.NewInt(proof.Index),
+		NumLeaves: big.NewInt(proof.Total),
+	}
+
+	valid, err := blobstreamXwrapper.VerifyAttestation(
+		&bind.CallOpts{},
+		big.NewInt(int64(nonce)),
+		tuple,
+		wrappedProof,
+	)
+	if err != nil {
+		return false, err
+	}
+	return valid, nil
+}

2. Transaction inclusion proof

To prove that a rollup transaction is part of the data root, we will need to provide two proofs: (1) a namespace Merkle proof of the transaction to a row root. This could be done via proving the shares that contain the transaction to the row root using a namespace Merkle proof. (2) And, a binary Merkle proof of the row root to the data root.

These proofs can be generated using the ProveShares query.

This endpoint allows querying a shares proof to row roots, then a row roots to data root proofs. It takes a block height, a starting share index and an end share index which define a share range. Then, two proofs are generated:

  • An NMT proof of the shares to the row roots
  • A binary Merkle proof of the row root to the data root

NOTE

If the share range spans multiple rows, then the proof can contain multiple NMT and binary proofs.

The endpoint can be queried using the golang client:

go
	sharesProof, err := trpc.ProveShares(ctx, 15, 0, 1)
+	if err != nil {
+		...
+	}
	sharesProof, err := trpc.ProveShares(ctx, 15, 0, 1)
+	if err != nil {
+		...
+	}

Converting the proofs to be usable in the DAVerifier library

Smart contracts that use the DAVerifier library takes the following proof format:

solidity
/// @notice Contains the necessary parameters to prove that some shares, which were posted to
+/// the Celestia network, were committed to by the BlobstreamX smart contract.
+struct SharesProof {
+    // The shares that were committed to.
+    bytes[] data;
+    // The shares proof to the row roots. If the shares span multiple rows, we will have multiple nmt proofs.
+    NamespaceMerkleMultiproof[] shareProofs;
+    // The namespace of the shares.
+    Namespace namespace;
+    // The rows where the shares belong. If the shares span multiple rows, we will have multiple rows.
+    NamespaceNode[] rowRoots;
+    // The proofs of the rowRoots to the data root.
+    BinaryMerkleProof[] rowProofs;
+    // The proof of the data root tuple to the data root tuple root that was posted to the BlobstreamX contract.
+    AttestationProof attestationProof;
+}
+
+/// @notice Contains the necessary parameters needed to verify that a data root tuple
+/// was committed to, by the BlobstreamX smart contract, at some specif nonce.
+struct AttestationProof {
+    // the attestation nonce that commits to the data root tuple.
+    uint256 tupleRootNonce;
+    // the data root tuple that was committed to.
+    DataRootTuple tuple;
+    // the binary Merkle proof of the tuple to the commitment.
+    BinaryMerkleProof proof;
+}
/// @notice Contains the necessary parameters to prove that some shares, which were posted to
+/// the Celestia network, were committed to by the BlobstreamX smart contract.
+struct SharesProof {
+    // The shares that were committed to.
+    bytes[] data;
+    // The shares proof to the row roots. If the shares span multiple rows, we will have multiple nmt proofs.
+    NamespaceMerkleMultiproof[] shareProofs;
+    // The namespace of the shares.
+    Namespace namespace;
+    // The rows where the shares belong. If the shares span multiple rows, we will have multiple rows.
+    NamespaceNode[] rowRoots;
+    // The proofs of the rowRoots to the data root.
+    BinaryMerkleProof[] rowProofs;
+    // The proof of the data root tuple to the data root tuple root that was posted to the BlobstreamX contract.
+    AttestationProof attestationProof;
+}
+
+/// @notice Contains the necessary parameters needed to verify that a data root tuple
+/// was committed to, by the BlobstreamX smart contract, at some specif nonce.
+struct AttestationProof {
+    // the attestation nonce that commits to the data root tuple.
+    uint256 tupleRootNonce;
+    // the data root tuple that was committed to.
+    DataRootTuple tuple;
+    // the binary Merkle proof of the tuple to the commitment.
+    BinaryMerkleProof proof;
+}

To construct the SharesProof, we will need the proof that we queried above, and it goes as follows:

data

This is the raw shares that were submitted to Celestia in the bytes format. If we take the example blob that was submitted in the RollupInclusionProofs.t.sol, we can convert it to bytes using the abi.encode(...) as done for this variable. This can be gotten from the above result of the transaction inclusion proof query in the field data.

shareProofs

This is the shares proof to the row roots. These can contain multiple proofs if the shares containing the blob span across multiple rows. To construct them, we will use the result of the transaction inclusion proof section.

While the NamespaceMerkleMultiproof being:

solidity
/// @notice Namespace Merkle Tree Multiproof structure. Proves multiple leaves.
+struct NamespaceMerkleMultiproof {
+    // The beginning key of the leaves to verify.
+    uint256 beginKey;
+    // The ending key of the leaves to verify.
+    uint256 endKey;
+    // List of side nodes to verify and calculate tree.
+    NamespaceNode[] sideNodes;
+}
/// @notice Namespace Merkle Tree Multiproof structure. Proves multiple leaves.
+struct NamespaceMerkleMultiproof {
+    // The beginning key of the leaves to verify.
+    uint256 beginKey;
+    // The ending key of the leaves to verify.
+    uint256 endKey;
+    // List of side nodes to verify and calculate tree.
+    NamespaceNode[] sideNodes;
+}

So, we can construct the NamespaceMerkleMultiproof with the following mapping:

  • beginKey in the Solidity struct == start in the query response

  • endKey in the Solidity struct == end in the query response

  • sideNodes in the Solidity struct == nodes in the query response

  • The NamespaceNode, which is the type of the sideNodes, is defined as follows:

solidity
/// @notice Namespace Merkle Tree node.
+struct NamespaceNode {
+    // Minimum namespace.
+    Namespace min;
+    // Maximum namespace.
+    Namespace max;
+    // Node value.
+    bytes32 digest;
+}
/// @notice Namespace Merkle Tree node.
+struct NamespaceNode {
+    // Minimum namespace.
+    Namespace min;
+    // Maximum namespace.
+    Namespace max;
+    // Node value.
+    bytes32 digest;
+}

So, we construct a NamespaceNode via taking the values from the nodes field in the query response, we convert them from base64 to hex, then we use the following mapping:

  • min == the first 29 bytes in the decoded value
  • max == the second 29 bytes in the decoded value
  • digest == the remaining 32 bytes in the decoded value

The min and max are Namespace type which is:

solidity
/// @notice A representation of the Celestia-app namespace ID and its version.
+/// See: https://celestiaorg.github.io/celestia-app/specs/namespace.html
+struct Namespace {
+    // The namespace version.
+    bytes1 version;
+    // The namespace ID.
+    bytes28 id;
+}
/// @notice A representation of the Celestia-app namespace ID and its version.
+/// See: https://celestiaorg.github.io/celestia-app/specs/namespace.html
+struct Namespace {
+    // The namespace version.
+    bytes1 version;
+    // The namespace ID.
+    bytes28 id;
+}

So, to construct them, we separate the 29 bytes in the decoded value to:

  • first byte: version
  • remaining 28 bytes: id

An example of doing this can be found in the RollupInclusionProofs.t.sol test.

A golang helper that can be used to make this conversion is as follows:

go
func toNamespaceMerkleMultiProofs(proofs []*tmproto.NMTProof) []client.NamespaceMerkleMultiproof {
+	shareProofs := make([]client.NamespaceMerkleMultiproof, len(proofs))
+	for i, proof := range proofs {
+		sideNodes := make([]client.NamespaceNode, len(proof.Nodes))
+		for j, node := range proof.Nodes {
+			sideNodes[j] = *toNamespaceNode(node)
+		}
+		shareProofs[i] = client.NamespaceMerkleMultiproof{
+			BeginKey:  big.NewInt(int64(proof.Start)),
+			EndKey:    big.NewInt(int64(proof.End)),
+			SideNodes: sideNodes,
+		}
+	}
+	return shareProofs
+}
+
+func minNamespace(innerNode []byte) *client.Namespace {
+	version := innerNode[0]
+	var id [28]byte
+	copy(id[:], innerNode[1:29])
+	return &client.Namespace{
+		Version: [1]byte{version},
+		Id:      id,
+	}
+}
+
+func maxNamespace(innerNode []byte) *client.Namespace {
+	version := innerNode[29]
+	var id [28]byte
+	copy(id[:], innerNode[30:58])
+	return &client.Namespace{
+		Version: [1]byte{version},
+		Id:      id,
+	}
+}
+
+func toNamespaceNode(node []byte) *client.NamespaceNode {
+	minNs := minNamespace(node)
+	maxNs := maxNamespace(node)
+	var digest [32]byte
+	copy(digest[:], node[58:])
+	return &client.NamespaceNode{
+		Min:    *minNs,
+		Max:    *maxNs,
+		Digest: digest,
+	}
+}
func toNamespaceMerkleMultiProofs(proofs []*tmproto.NMTProof) []client.NamespaceMerkleMultiproof {
+	shareProofs := make([]client.NamespaceMerkleMultiproof, len(proofs))
+	for i, proof := range proofs {
+		sideNodes := make([]client.NamespaceNode, len(proof.Nodes))
+		for j, node := range proof.Nodes {
+			sideNodes[j] = *toNamespaceNode(node)
+		}
+		shareProofs[i] = client.NamespaceMerkleMultiproof{
+			BeginKey:  big.NewInt(int64(proof.Start)),
+			EndKey:    big.NewInt(int64(proof.End)),
+			SideNodes: sideNodes,
+		}
+	}
+	return shareProofs
+}
+
+func minNamespace(innerNode []byte) *client.Namespace {
+	version := innerNode[0]
+	var id [28]byte
+	copy(id[:], innerNode[1:29])
+	return &client.Namespace{
+		Version: [1]byte{version},
+		Id:      id,
+	}
+}
+
+func maxNamespace(innerNode []byte) *client.Namespace {
+	version := innerNode[29]
+	var id [28]byte
+	copy(id[:], innerNode[30:58])
+	return &client.Namespace{
+		Version: [1]byte{version},
+		Id:      id,
+	}
+}
+
+func toNamespaceNode(node []byte) *client.NamespaceNode {
+	minNs := minNamespace(node)
+	maxNs := maxNamespace(node)
+	var digest [32]byte
+	copy(digest[:], node[58:])
+	return &client.NamespaceNode{
+		Min:    *minNs,
+		Max:    *maxNs,
+		Digest: digest,
+	}
+}

with proofs being sharesProof.ShareProofs.

namespace

Which is the namespace used by the rollup when submitting data to Celestia. As described above, it can be constructed as follows:

solidity
/// @notice A representation of the Celestia-app namespace ID and its version.
+/// See: https://celestiaorg.github.io/celestia-app/specs/namespace.html
+struct Namespace {
+    // The namespace version.
+    bytes1 version;
+    // The namespace ID.
+    bytes28 id;
+}
/// @notice A representation of the Celestia-app namespace ID and its version.
+/// See: https://celestiaorg.github.io/celestia-app/specs/namespace.html
+struct Namespace {
+    // The namespace version.
+    bytes1 version;
+    // The namespace ID.
+    bytes28 id;
+}

Via taking the namespace value from the prove_shares query response, decoding it from base64 to hex, then:

  • first byte: version
  • remaining 28 bytes: id

An example can be found in the RollupInclusionProofs.t.sol test.

A method to convert to namespace, provided that the namespace size is 29, is as follows:

go
func namespace(namespaceID []byte, version uint8) *client.Namespace {
+	var id [28]byte
+	copy(id[:], namespaceID)
+	return &client.Namespace{
+		Version: [1]byte{version},
+		Id:      id,
+	}
+}
func namespace(namespaceID []byte, version uint8) *client.Namespace {
+	var id [28]byte
+	copy(id[:], namespaceID)
+	return &client.Namespace{
+		Version: [1]byte{version},
+		Id:      id,
+	}
+}

with namespace being sharesProof.NamespaceID.

rowRoots

Which are the roots of the rows where the shares containing the Rollup data are localised.

In golang, the proof can be converted as follows:

go
func toRowRoots(roots []bytes.HexBytes) []client.NamespaceNode {
+	rowRoots := make([]client.NamespaceNode, len(roots))
+	for i, root := range roots {
+		rowRoots[i] = *toNamespaceNode(root.Bytes())
+	}
+	return rowRoots
+}
func toRowRoots(roots []bytes.HexBytes) []client.NamespaceNode {
+	rowRoots := make([]client.NamespaceNode, len(roots))
+	for i, root := range roots {
+		rowRoots[i] = *toNamespaceNode(root.Bytes())
+	}
+	return rowRoots
+}

with roots being sharesProof.RowProof.RowRoots.

rowProofs

These are the proofs of the rows to the data root. They are of type BinaryMerkleProof:

solidity
/// @notice Merkle Tree Proof structure.
+struct BinaryMerkleProof {
+    // List of side nodes to verify and calculate tree.
+    bytes32[] sideNodes;
+    // The key of the leaf to verify.
+    uint256 key;
+    // The number of leaves in the tree
+    uint256 numLeaves;
+}
/// @notice Merkle Tree Proof structure.
+struct BinaryMerkleProof {
+    // List of side nodes to verify and calculate tree.
+    bytes32[] sideNodes;
+    // The key of the leaf to verify.
+    uint256 key;
+    // The number of leaves in the tree
+    uint256 numLeaves;
+}

To construct them, we take the response of the prove_shares query, and do the following mapping:

  • key in the Solidity struct == index in the query response
  • numLeaves in the Solidity struct == total in the query response
  • sideNodes in the Solidity struct == aunts in the query response

The type of the sideNodes is a bytes32.

An example can be found in the RollupInclusionProofs.t.sol test.

A golang helper to convert the row proofs is as follows:

go
func toRowProofs(proofs []*merkle.Proof) []client.BinaryMerkleProof {
+	rowProofs := make([]client.BinaryMerkleProof, len(proofs))
+	for i, proof := range proofs {
+		sideNodes := make( [][32]byte, len(proof.Aunts))
+		for j, sideNode :=  range proof.Aunts {
+			var bzSideNode [32]byte
+			copy(bzSideNode[:], sideNode)
+			sideNodes[j] = bzSideNode
+		}
+ 		rowProofs[i] = client.BinaryMerkleProof{
+			SideNodes: sideNodes,
+			Key:       big.NewInt(proof.Index),
+			NumLeaves: big.NewInt(proof.Total),
+		}
+	}
+	return rowProofs
+}
func toRowProofs(proofs []*merkle.Proof) []client.BinaryMerkleProof {
+	rowProofs := make([]client.BinaryMerkleProof, len(proofs))
+	for i, proof := range proofs {
+		sideNodes := make( [][32]byte, len(proof.Aunts))
+		for j, sideNode :=  range proof.Aunts {
+			var bzSideNode [32]byte
+			copy(bzSideNode[:], sideNode)
+			sideNodes[j] = bzSideNode
+		}
+ 		rowProofs[i] = client.BinaryMerkleProof{
+			SideNodes: sideNodes,
+			Key:       big.NewInt(proof.Index),
+			NumLeaves: big.NewInt(proof.Total),
+		}
+	}
+	return rowProofs
+}

with proofs being sharesProof.RowProof.Proofs.

attestationProof

This is the proof of the data root to the data root tuple root, which is committed to in the Blobstream X contract:

solidity
/// @notice Contains the necessary parameters needed to verify that a data root tuple
+/// was committed to, by the BlobstreamX smart contract, at some specif nonce.
+struct AttestationProof {
+    // the attestation nonce that commits to the data root tuple.
+    uint256 tupleRootNonce;
+    // the data root tuple that was committed to.
+    DataRootTuple tuple;
+    // the binary Merkle proof of the tuple to the commitment.
+    BinaryMerkleProof proof;
+}
/// @notice Contains the necessary parameters needed to verify that a data root tuple
+/// was committed to, by the BlobstreamX smart contract, at some specif nonce.
+struct AttestationProof {
+    // the attestation nonce that commits to the data root tuple.
+    uint256 tupleRootNonce;
+    // the data root tuple that was committed to.
+    DataRootTuple tuple;
+    // the binary Merkle proof of the tuple to the commitment.
+    BinaryMerkleProof proof;
+}
  • tupleRootNonce: the nonce at which Blobstream X committed to the batch containing the block containing the data.
  • tuple: the DataRootTuple of the block:
solidity
/// @notice A tuple of data root with metadata. Each data root is associated
+///  with a Celestia block height.
+/// @dev \`availableDataRoot\` in
+///  https://github.com/celestiaorg/celestia-specs/blob/master/src/specs/data_structures.md#header
+struct DataRootTuple {
+    // Celestia block height the data root was included in.
+    // Genesis block is height = 0.
+    // First queryable block is height = 1.
+    uint256 height;
+    // Data root.
+    bytes32 dataRoot;
+}
/// @notice A tuple of data root with metadata. Each data root is associated
+///  with a Celestia block height.
+/// @dev \`availableDataRoot\` in
+///  https://github.com/celestiaorg/celestia-specs/blob/master/src/specs/data_structures.md#header
+struct DataRootTuple {
+    // Celestia block height the data root was included in.
+    // Genesis block is height = 0.
+    // First queryable block is height = 1.
+    uint256 height;
+    // Data root.
+    bytes32 dataRoot;
+}

which comprises a dataRoot, i.e. the block containing the Rollup data data root, and the height which is the height of that block.

  • proof: the BinaryMerkleProof of the data root tuple to the data root tuple root. Constructing it is similar to constructing the row roots to data root proof in the rowProofs section.

An example can be found in the RollupInclusionProofs.t.sol test.

A golang helper to create an attestation proof:

go
func toAttestationProof(
+	nonce uint64,
+	height uint64,
+	blockDataRoot [32]byte,
+	dataRootInclusionProof merkle.Proof,
+) client.AttestationProof {
+	sideNodes := make( [][32]byte, len(dataRootInclusionProof.Aunts))
+	for i, sideNode :=  range dataRootInclusionProof.Aunts {
+		var bzSideNode [32]byte
+		copy(bzSideNode[:], sideNode)
+		sideNodes[i] = bzSideNode
+	}
+
+	return client.AttestationProof{
+		TupleRootNonce: big.NewInt(int64(nonce)),
+		Tuple:          client.DataRootTuple{
+			Height:   big.NewInt(int64(height)),
+			DataRoot: blockDataRoot,
+		},
+		Proof:          client.BinaryMerkleProof{
+			SideNodes: sideNodes,
+			Key:       big.NewInt(dataRootInclusionProof.Index),
+			NumLeaves: big.NewInt(dataRootInclusionProof.Total),
+		},
+	}
+}
func toAttestationProof(
+	nonce uint64,
+	height uint64,
+	blockDataRoot [32]byte,
+	dataRootInclusionProof merkle.Proof,
+) client.AttestationProof {
+	sideNodes := make( [][32]byte, len(dataRootInclusionProof.Aunts))
+	for i, sideNode :=  range dataRootInclusionProof.Aunts {
+		var bzSideNode [32]byte
+		copy(bzSideNode[:], sideNode)
+		sideNodes[i] = bzSideNode
+	}
+
+	return client.AttestationProof{
+		TupleRootNonce: big.NewInt(int64(nonce)),
+		Tuple:          client.DataRootTuple{
+			Height:   big.NewInt(int64(height)),
+			DataRoot: blockDataRoot,
+		},
+		Proof:          client.BinaryMerkleProof{
+			SideNodes: sideNodes,
+			Key:       big.NewInt(dataRootInclusionProof.Index),
+			NumLeaves: big.NewInt(dataRootInclusionProof.Total),
+		},
+	}
+}

with the nonce being the attestation nonce, which can be retrieved using BlobstreamX contract events. Check below for an example. And height being the Celestia Block height that contains the rollup data, along with the blockDataRoot being the data root of the block height. Finally, dataRootInclusionProof is the Celestia block data root inclusion proof to the data root tuple root that was queried at the beginning of this page.

If the dataRoot or the tupleRootNonce is unknown during the verification:

  • dataRoot: can be queried using the /block?height=15 query (15 in this example endpoint), and taking the data_hash field from the response.
  • tupleRootNonce: can be retried via querying the BlobstreamXDataCommitmentStored events from the BlobstreamX contract and looking for the nonce attesting to the corresponding data. An example:
go
	// get the nonce corresponding to the block height that contains the PayForBlob transaction
+	// since BlobstreamX emits events when new batches are submitted, we will query the events
+	// and look for the range committing to the blob
+	// first, connect to an EVM RPC endpoint
+	ethClient, err := ethclient.Dial("evm_rpc_endpoint")
+	if err != nil {
+		return err
+	}
+	defer ethClient.Close()
+
+	// use the BlobstreamX contract binding
+	wrapper, err := blobstreamxwrapper.NewBlobstreamX(ethcmn.HexToAddress("contract_Address"), ethClient)
+	if err != nil {
+		return err
+	}
+
+	LatestBlockNumber, err := ethClient.BlockNumber(ctx)
+	if err != nil {
+		return err
+	}
+
+	eventsIterator, err := wrapper.FilterDataCommitmentStored(
+		&bind.FilterOpts{
+			Context: ctx,
+			Start: LatestBlockNumber - 90000, // 90000 can be replaced with the range of EVM blocks to look for the events in
+			End: &LatestBlockNumber,
+		},
+		nil,
+		nil,
+		nil,
+	)
+	if err != nil {
+		return err
+	}
+
+	var event *blobstreamxwrapper.BlobstreamXDataCommitmentStored
+	for eventsIterator.Next() {
+		e := eventsIterator.Event
+		if int64(e.StartBlock) <= tx.Height && tx.Height < int64(e.EndBlock) {
+			event = &blobstreamxwrapper.BlobstreamXDataCommitmentStored{
+				ProofNonce:     e.ProofNonce,
+				StartBlock:     e.StartBlock,
+				EndBlock:       e.EndBlock,
+				DataCommitment: e.DataCommitment,
+			}
+			break
+		}
+	}
+	if err := eventsIterator.Error(); err != nil {
+		return err
+	}
+	err = eventsIterator.Close()
+	if err != nil {
+		return err
+	}
+	if event == nil {
+		return fmt.Errorf("couldn't find range containing the block height")
+	}
	// get the nonce corresponding to the block height that contains the PayForBlob transaction
+	// since BlobstreamX emits events when new batches are submitted, we will query the events
+	// and look for the range committing to the blob
+	// first, connect to an EVM RPC endpoint
+	ethClient, err := ethclient.Dial("evm_rpc_endpoint")
+	if err != nil {
+		return err
+	}
+	defer ethClient.Close()
+
+	// use the BlobstreamX contract binding
+	wrapper, err := blobstreamxwrapper.NewBlobstreamX(ethcmn.HexToAddress("contract_Address"), ethClient)
+	if err != nil {
+		return err
+	}
+
+	LatestBlockNumber, err := ethClient.BlockNumber(ctx)
+	if err != nil {
+		return err
+	}
+
+	eventsIterator, err := wrapper.FilterDataCommitmentStored(
+		&bind.FilterOpts{
+			Context: ctx,
+			Start: LatestBlockNumber - 90000, // 90000 can be replaced with the range of EVM blocks to look for the events in
+			End: &LatestBlockNumber,
+		},
+		nil,
+		nil,
+		nil,
+	)
+	if err != nil {
+		return err
+	}
+
+	var event *blobstreamxwrapper.BlobstreamXDataCommitmentStored
+	for eventsIterator.Next() {
+		e := eventsIterator.Event
+		if int64(e.StartBlock) <= tx.Height && tx.Height < int64(e.EndBlock) {
+			event = &blobstreamxwrapper.BlobstreamXDataCommitmentStored{
+				ProofNonce:     e.ProofNonce,
+				StartBlock:     e.StartBlock,
+				EndBlock:       e.EndBlock,
+				DataCommitment: e.DataCommitment,
+			}
+			break
+		}
+	}
+	if err := eventsIterator.Error(); err != nil {
+		return err
+	}
+	err = eventsIterator.Close()
+	if err != nil {
+		return err
+	}
+	if event == nil {
+		return fmt.Errorf("couldn't find range containing the block height")
+	}

Listening for new data commitments

For listening for new BlobstreamXDataCommitmentStored events, sequencers can use the WatchDataCommitmentStored as follows:

go
    ethClient, err := ethclient.Dial("evm_rpc")
+    if err != nil {
+	    return err
+    }
+    defer ethClient.Close()
+    blobstreamWrapper, err := blobstreamxwrapper.NewBlobstreamXFilterer(ethcmn.HexToAddress("contract_address"), ethClient)
+    if err != nil {
+	    return err
+    }
+
+    eventsChan := make(chan *blobstreamxwrapper.BlobstreamXDataCommitmentStored, 100)
+    subscription, err := blobstreamWrapper.WatchDataCommitmentStored(
+	    &bind.WatchOpts{
+			Context: ctx,
+        },
+	    eventsChan,
+	    nil,
+	    nil,
+	    nil,
+	)
+    if err != nil {
+	    return err
+    }
+    defer subscription.Unsubscribe()
+
+    for {
+	    select {
+	    case <-ctx.Done():
+		    return ctx.Err()
+		case err := <-subscription.Err():
+			return err
+		case event := <-eventsChan:
+			// process the event
+		    fmt.Println(event)
+	    }
+    }
    ethClient, err := ethclient.Dial("evm_rpc")
+    if err != nil {
+	    return err
+    }
+    defer ethClient.Close()
+    blobstreamWrapper, err := blobstreamxwrapper.NewBlobstreamXFilterer(ethcmn.HexToAddress("contract_address"), ethClient)
+    if err != nil {
+	    return err
+    }
+
+    eventsChan := make(chan *blobstreamxwrapper.BlobstreamXDataCommitmentStored, 100)
+    subscription, err := blobstreamWrapper.WatchDataCommitmentStored(
+	    &bind.WatchOpts{
+			Context: ctx,
+        },
+	    eventsChan,
+	    nil,
+	    nil,
+	    nil,
+	)
+    if err != nil {
+	    return err
+    }
+    defer subscription.Unsubscribe()
+
+    for {
+	    select {
+	    case <-ctx.Done():
+		    return ctx.Err()
+		case err := <-subscription.Err():
+			return err
+		case event := <-eventsChan:
+			// process the event
+		    fmt.Println(event)
+	    }
+    }

Then, new proofs can be created as documented above using the new data commitments contained in the received events.

Example rollup that uses the DAVerifier

An example rollup that uses the DAVerifier can be as simple as:

solidity
pragma solidity ^0.8.22;
+
+import {DAVerifier} from "@blobstream/lib/verifier/DAVerifier.sol";
+import {IDAOracle} from "@blobstream/IDAOracle.sol";
+
+contract SimpleRollup {
+    IDAOracle bridge;
+    ...
+    function submitFraudProof(SharesProof memory _sharesProof, bytes32 _root) public {
+        // (1) verify that the data is committed to by BlobstreamX contract
+        (bool committedTo, DAVerifier.ErrorCodes err) = DAVerifier.verifySharesToDataRootTupleRoot(bridge, _sharesProof, _root);
+        if (!committedTo) {
+            revert("the data was not committed to by Blobstream");
+        }
+        // (2) verify that the data is part of the rollup block
+        // (3) parse the data
+        // (4) verify invalid state transition
+        // (5) effects
+    }
+}
pragma solidity ^0.8.22;
+
+import {DAVerifier} from "@blobstream/lib/verifier/DAVerifier.sol";
+import {IDAOracle} from "@blobstream/IDAOracle.sol";
+
+contract SimpleRollup {
+    IDAOracle bridge;
+    ...
+    function submitFraudProof(SharesProof memory _sharesProof, bytes32 _root) public {
+        // (1) verify that the data is committed to by BlobstreamX contract
+        (bool committedTo, DAVerifier.ErrorCodes err) = DAVerifier.verifySharesToDataRootTupleRoot(bridge, _sharesProof, _root);
+        if (!committedTo) {
+            revert("the data was not committed to by Blobstream");
+        }
+        // (2) verify that the data is part of the rollup block
+        // (3) parse the data
+        // (4) verify invalid state transition
+        // (5) effects
+    }
+}

Then, you can submit the fraud proof using golang as follows:

go
package main
+
+import (
+	"context"
+	"fmt"
+	"github.com/celestiaorg/celestia-app/pkg/square"
+	"github.com/celestiaorg/celestia-app/x/qgb/client"
+	"github.com/ethereum/go-ethereum/accounts/abi/bind"
+	ethcmn "github.com/ethereum/go-ethereum/common"
+	"github.com/ethereum/go-ethereum/ethclient"
+	blobstreamxwrapper "github.com/succinctlabs/blobstreamx/bindings"
+	"github.com/tendermint/tendermint/crypto/merkle"
+	"github.com/tendermint/tendermint/libs/bytes"
+	tmproto "github.com/tendermint/tendermint/proto/tendermint/types"
+	"github.com/tendermint/tendermint/rpc/client/http"
+	"github.com/tendermint/tendermint/types"
+	"math/big"
+	"os"
+)
+
+func main() {
+	err := verify()
+	if err != nil {
+		fmt.Println(err)
+		os.Exit(1)
+	}
+}
+
+func verify() error {
+	ctx := context.Background()
+
+
+	// ...
+	// check the first section for this part of the implementation
+
+	// get the nonce corresponding to the block height that contains the PayForBlob transaction
+	// since Blobstream X emits events when new batches are submitted, we will query the events
+	// and look for the range committing to the blob
+	// first, connect to an EVM RPC endpoint
+	ethClient, err := ethclient.Dial("evm_rpc_endpoint")
+	if err != nil {
+		return err
+	}
+	defer ethClient.Close()
+
+	// ...
+	// check the first section for this part of the implementation
+
+	// now we will create the shares proof to be verified by the SimpleRollup
+	// contract that uses the DAVerifier library
+
+	// get the proof of the shares containing the blob to the data root
+	sharesProof, err := trpc.ProveShares(ctx, 16, uint64(blobShareRange.Start), uint64(blobShareRange.End))
+	if err != nil {
+		return err
+	}
+
+	// use the SimpleRollup contract binding to submit to it a fraud proof
+	simpleRollupWrapper, err := client.NewWrappers(ethcmn.HexToAddress("contract_Address"), ethClient)
+	if err != nil {
+		return err
+	}
+
+	// submit the fraud proof containing the share data that had the invalid state transition for example
+	// along with its proof
+	err = submitFraudProof(
+		ctx,
+		simpleRollupWrapper,
+		sharesProof,
+		event.ProofNonce.Uint64(),
+		uint64(tx.Height),
+		dcProof.Proof,
+		blockRes.Block.DataHash,
+	)
+
+	return nil
+}
+
+func submitFraudProof(
+	ctx context.Context,
+	simpleRollup *client.Wrappers,
+	sharesProof types.ShareProof,
+	nonce uint64,
+	height uint64,
+	dataRootInclusionProof merkle.Proof,
+	dataRoot []byte,
+) error {
+	var blockDataRoot [32]byte
+	copy(blockDataRoot[:], dataRoot)
+	tx, err := simpleRollup.SubmitFraudProof(
+		&bind.TransactOpts{
+			Context: ctx,
+		},
+		client.SharesProof{
+			Data:             sharesProof.Data,
+			ShareProofs:      toNamespaceMerkleMultiProofs(sharesProof.ShareProofs),
+			Namespace:        *namespace(sharesProof.NamespaceID),
+			RowRoots:         toRowRoots(sharesProof.RowProof.RowRoots),
+			RowProofs:        toRowProofs(sharesProof.RowProof.Proofs),
+			AttestationProof: toAttestationProof(nonce, height, blockDataRoot, dataRootInclusionProof),
+		},
+		blockDataRoot,
+	)
+	if err != nil {
+		return err
+	}
+	// wait for transaction
+}
+
+func toAttestationProof(
+	nonce uint64,
+	height uint64,
+	blockDataRoot [32]byte,
+	dataRootInclusionProof merkle.Proof,
+) client.AttestationProof {
+	sideNodes := make( [][32]byte, len(dataRootInclusionProof.Aunts))
+	for i, sideNode :=  range dataRootInclusionProof.Aunts {
+		var bzSideNode [32]byte
+		for k, b := range sideNode {
+			bzSideNode[k] = b
+		}
+		sideNodes[i] = bzSideNode
+	}
+
+	return client.AttestationProof{
+		TupleRootNonce: big.NewInt(int64(nonce)),
+		Tuple:          client.DataRootTuple{
+			Height:   big.NewInt(int64(height)),
+			DataRoot: blockDataRoot,
+		},
+		Proof:          client.BinaryMerkleProof{
+			SideNodes: sideNodes,
+			Key:       big.NewInt(dataRootInclusionProof.Index),
+			NumLeaves: big.NewInt(dataRootInclusionProof.Total),
+		},
+	}
+}
+
+func toRowRoots(roots []bytes.HexBytes) []client.NamespaceNode {
+	rowRoots := make([]client.NamespaceNode, len(roots))
+	for i, root := range roots {
+		rowRoots[i] = *toNamespaceNode(root.Bytes())
+	}
+	return rowRoots
+}
+
+func toRowProofs(proofs []*merkle.Proof) []client.BinaryMerkleProof {
+	rowProofs := make([]client.BinaryMerkleProof, len(proofs))
+	for i, proof := range proofs {
+		sideNodes := make( [][32]byte, len(proof.Aunts))
+		for j, sideNode :=  range proof.Aunts {
+			var bzSideNode [32]byte
+			for k, b := range sideNode {
+				bzSideNode[k] = b
+			}
+			sideNodes[j] = bzSideNode
+		}
+ 		rowProofs[i] = client.BinaryMerkleProof{
+			SideNodes: sideNodes,
+			Key:       big.NewInt(proof.Index),
+			NumLeaves: big.NewInt(proof.Total),
+		}
+	}
+	return rowProofs
+}
+
+func toNamespaceMerkleMultiProofs(proofs []*tmproto.NMTProof) []client.NamespaceMerkleMultiproof {
+	shareProofs := make([]client.NamespaceMerkleMultiproof, len(proofs))
+	for i, proof := range proofs {
+		sideNodes := make([]client.NamespaceNode, len(proof.Nodes))
+		for j, node := range proof.Nodes {
+			sideNodes[j] = *toNamespaceNode(node)
+		}
+		shareProofs[i] = client.NamespaceMerkleMultiproof{
+			BeginKey:  big.NewInt(int64(proof.Start)),
+			EndKey:    big.NewInt(int64(proof.End)),
+			SideNodes: sideNodes,
+		}
+	}
+	return shareProofs
+}
+
+func minNamespace(innerNode []byte) *client.Namespace {
+	version := innerNode[0]
+	var id [28]byte
+	for i, b := range innerNode[1:28] {
+		id[i] = b
+	}
+	return &client.Namespace{
+		Version: [1]byte{version},
+		Id:      id,
+	}
+}
+
+func maxNamespace(innerNode []byte) *client.Namespace {
+	version := innerNode[29]
+	var id [28]byte
+	for i, b := range innerNode[30:57] {
+		id[i] = b
+	}
+	return &client.Namespace{
+		Version: [1]byte{version},
+		Id:      id,
+	}
+}
+
+func toNamespaceNode(node []byte) *client.NamespaceNode {
+	minNs := minNamespace(node)
+	maxNs := maxNamespace(node)
+	var digest [32]byte
+	for i, b := range node[58:] {
+		digest[i] = b
+	}
+	return &client.NamespaceNode{
+		Min:    *minNs,
+		Max:    *maxNs,
+		Digest: digest,
+	}
+}
+
+func namespace(namespaceID []byte) *client.Namespace {
+	version := namespaceID[0]
+	var id [28]byte
+	for i, b := range namespaceID[1:] {
+		id[i] = b
+	}
+	return &client.Namespace{
+		Version: [1]byte{version},
+		Id:      id,
+	}
+}
package main
+
+import (
+	"context"
+	"fmt"
+	"github.com/celestiaorg/celestia-app/pkg/square"
+	"github.com/celestiaorg/celestia-app/x/qgb/client"
+	"github.com/ethereum/go-ethereum/accounts/abi/bind"
+	ethcmn "github.com/ethereum/go-ethereum/common"
+	"github.com/ethereum/go-ethereum/ethclient"
+	blobstreamxwrapper "github.com/succinctlabs/blobstreamx/bindings"
+	"github.com/tendermint/tendermint/crypto/merkle"
+	"github.com/tendermint/tendermint/libs/bytes"
+	tmproto "github.com/tendermint/tendermint/proto/tendermint/types"
+	"github.com/tendermint/tendermint/rpc/client/http"
+	"github.com/tendermint/tendermint/types"
+	"math/big"
+	"os"
+)
+
+func main() {
+	err := verify()
+	if err != nil {
+		fmt.Println(err)
+		os.Exit(1)
+	}
+}
+
+func verify() error {
+	ctx := context.Background()
+
+
+	// ...
+	// check the first section for this part of the implementation
+
+	// get the nonce corresponding to the block height that contains the PayForBlob transaction
+	// since Blobstream X emits events when new batches are submitted, we will query the events
+	// and look for the range committing to the blob
+	// first, connect to an EVM RPC endpoint
+	ethClient, err := ethclient.Dial("evm_rpc_endpoint")
+	if err != nil {
+		return err
+	}
+	defer ethClient.Close()
+
+	// ...
+	// check the first section for this part of the implementation
+
+	// now we will create the shares proof to be verified by the SimpleRollup
+	// contract that uses the DAVerifier library
+
+	// get the proof of the shares containing the blob to the data root
+	sharesProof, err := trpc.ProveShares(ctx, 16, uint64(blobShareRange.Start), uint64(blobShareRange.End))
+	if err != nil {
+		return err
+	}
+
+	// use the SimpleRollup contract binding to submit to it a fraud proof
+	simpleRollupWrapper, err := client.NewWrappers(ethcmn.HexToAddress("contract_Address"), ethClient)
+	if err != nil {
+		return err
+	}
+
+	// submit the fraud proof containing the share data that had the invalid state transition for example
+	// along with its proof
+	err = submitFraudProof(
+		ctx,
+		simpleRollupWrapper,
+		sharesProof,
+		event.ProofNonce.Uint64(),
+		uint64(tx.Height),
+		dcProof.Proof,
+		blockRes.Block.DataHash,
+	)
+
+	return nil
+}
+
+func submitFraudProof(
+	ctx context.Context,
+	simpleRollup *client.Wrappers,
+	sharesProof types.ShareProof,
+	nonce uint64,
+	height uint64,
+	dataRootInclusionProof merkle.Proof,
+	dataRoot []byte,
+) error {
+	var blockDataRoot [32]byte
+	copy(blockDataRoot[:], dataRoot)
+	tx, err := simpleRollup.SubmitFraudProof(
+		&bind.TransactOpts{
+			Context: ctx,
+		},
+		client.SharesProof{
+			Data:             sharesProof.Data,
+			ShareProofs:      toNamespaceMerkleMultiProofs(sharesProof.ShareProofs),
+			Namespace:        *namespace(sharesProof.NamespaceID),
+			RowRoots:         toRowRoots(sharesProof.RowProof.RowRoots),
+			RowProofs:        toRowProofs(sharesProof.RowProof.Proofs),
+			AttestationProof: toAttestationProof(nonce, height, blockDataRoot, dataRootInclusionProof),
+		},
+		blockDataRoot,
+	)
+	if err != nil {
+		return err
+	}
+	// wait for transaction
+}
+
+func toAttestationProof(
+	nonce uint64,
+	height uint64,
+	blockDataRoot [32]byte,
+	dataRootInclusionProof merkle.Proof,
+) client.AttestationProof {
+	sideNodes := make( [][32]byte, len(dataRootInclusionProof.Aunts))
+	for i, sideNode :=  range dataRootInclusionProof.Aunts {
+		var bzSideNode [32]byte
+		for k, b := range sideNode {
+			bzSideNode[k] = b
+		}
+		sideNodes[i] = bzSideNode
+	}
+
+	return client.AttestationProof{
+		TupleRootNonce: big.NewInt(int64(nonce)),
+		Tuple:          client.DataRootTuple{
+			Height:   big.NewInt(int64(height)),
+			DataRoot: blockDataRoot,
+		},
+		Proof:          client.BinaryMerkleProof{
+			SideNodes: sideNodes,
+			Key:       big.NewInt(dataRootInclusionProof.Index),
+			NumLeaves: big.NewInt(dataRootInclusionProof.Total),
+		},
+	}
+}
+
+func toRowRoots(roots []bytes.HexBytes) []client.NamespaceNode {
+	rowRoots := make([]client.NamespaceNode, len(roots))
+	for i, root := range roots {
+		rowRoots[i] = *toNamespaceNode(root.Bytes())
+	}
+	return rowRoots
+}
+
+func toRowProofs(proofs []*merkle.Proof) []client.BinaryMerkleProof {
+	rowProofs := make([]client.BinaryMerkleProof, len(proofs))
+	for i, proof := range proofs {
+		sideNodes := make( [][32]byte, len(proof.Aunts))
+		for j, sideNode :=  range proof.Aunts {
+			var bzSideNode [32]byte
+			for k, b := range sideNode {
+				bzSideNode[k] = b
+			}
+			sideNodes[j] = bzSideNode
+		}
+ 		rowProofs[i] = client.BinaryMerkleProof{
+			SideNodes: sideNodes,
+			Key:       big.NewInt(proof.Index),
+			NumLeaves: big.NewInt(proof.Total),
+		}
+	}
+	return rowProofs
+}
+
+func toNamespaceMerkleMultiProofs(proofs []*tmproto.NMTProof) []client.NamespaceMerkleMultiproof {
+	shareProofs := make([]client.NamespaceMerkleMultiproof, len(proofs))
+	for i, proof := range proofs {
+		sideNodes := make([]client.NamespaceNode, len(proof.Nodes))
+		for j, node := range proof.Nodes {
+			sideNodes[j] = *toNamespaceNode(node)
+		}
+		shareProofs[i] = client.NamespaceMerkleMultiproof{
+			BeginKey:  big.NewInt(int64(proof.Start)),
+			EndKey:    big.NewInt(int64(proof.End)),
+			SideNodes: sideNodes,
+		}
+	}
+	return shareProofs
+}
+
+func minNamespace(innerNode []byte) *client.Namespace {
+	version := innerNode[0]
+	var id [28]byte
+	for i, b := range innerNode[1:28] {
+		id[i] = b
+	}
+	return &client.Namespace{
+		Version: [1]byte{version},
+		Id:      id,
+	}
+}
+
+func maxNamespace(innerNode []byte) *client.Namespace {
+	version := innerNode[29]
+	var id [28]byte
+	for i, b := range innerNode[30:57] {
+		id[i] = b
+	}
+	return &client.Namespace{
+		Version: [1]byte{version},
+		Id:      id,
+	}
+}
+
+func toNamespaceNode(node []byte) *client.NamespaceNode {
+	minNs := minNamespace(node)
+	maxNs := maxNamespace(node)
+	var digest [32]byte
+	for i, b := range node[58:] {
+		digest[i] = b
+	}
+	return &client.NamespaceNode{
+		Min:    *minNs,
+		Max:    *maxNs,
+		Digest: digest,
+	}
+}
+
+func namespace(namespaceID []byte) *client.Namespace {
+	version := namespaceID[0]
+	var id [28]byte
+	for i, b := range namespaceID[1:] {
+		id[i] = b
+	}
+	return &client.Namespace{
+		Version: [1]byte{version},
+		Id:      id,
+	}
+}

For the step (2), check the rollup inclusion proofs documentation for more information.

Conclusion

After creating all the proofs, and verifying them:

  1. Verify inclusion proof of the transaction to Celestia data root
  2. Prove that the data root tuple is committed to by the Blobstream X smart contract

We can be sure that the data was published to Celestia, and then rollups can proceed with their normal fraud proving mechanism.

NOTE

The above proof constructions are implemented in Solidity, and may require different approaches in other programming languages.

`,109),c=[t];function r(E,y,i,d,F,h){return n(),a("div",null,c)}const b=s(e,[["render",r]]);export{m as __pageData,b as default}; diff --git a/assets/developers_blobstream-proof-queries.md.4443c323.lean.js b/assets/developers_blobstream-proof-queries.md.4443c323.lean.js new file mode 100644 index 00000000000..13b34a93e1f --- /dev/null +++ b/assets/developers_blobstream-proof-queries.md.4443c323.lean.js @@ -0,0 +1 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.51d6c45b.js";const o="/img/blobstream/blobstream-square.png",p="/img/blobstream/blobstream-commitment-diagram.png",m=JSON.parse('{"title":"Blobstream proofs queries","description":"Learn how to query the inclusion proofs used in Blobstream","frontmatter":{"description":"Learn how to query the inclusion proofs used in Blobstream","head":[["meta",{"name":"og:title","content":"Blobstream proofs queries | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstream-proof-queries.md","filePath":"developers/blobstream-proof-queries.md","lastUpdated":1713191709000}'),e={name:"developers/blobstream-proof-queries.md"},t=l("",109),c=[t];function r(E,y,i,d,F,h){return n(),a("div",null,c)}const b=s(e,[["render",r]]);export{m as __pageData,b as default}; diff --git a/assets/developers_blobstream-rollups.md.b8b40693.js b/assets/developers_blobstream-rollups.md.b8b40693.js new file mode 100644 index 00000000000..3d19ca74e44 --- /dev/null +++ b/assets/developers_blobstream-rollups.md.b8b40693.js @@ -0,0 +1,3 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.51d6c45b.js";const f=JSON.parse('{"title":"Introduction to Blobstream rollups","description":"Learn how to build rollups that use Blobstream.","frontmatter":{"description":"Learn how to build rollups that use Blobstream.","head":[["meta",{"name":"og:title","content":"Introduction to Blobstream rollups | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstream-rollups.md","filePath":"developers/blobstream-rollups.md","lastUpdated":1713450778000}'),s={name:"developers/blobstream-rollups.md"},r=a(`

Introduction to Blobstream rollups

Blobstream is the first data availability solution for EVM chains that securely scales with the number of users. It allows rollups to post their data on Celestia while proving their availability in the rollup settlement contract.

This document will outline a few ways to build optimistic or zk-rollups that post their data to Celestia and use Blobstream to prove that data's availability.

Concepts

Share commitment

The share commitment is a commitment over the data contained in the MsgPayForBlobs transaction. This commitment allows proving that the corresponding data exists on Celestia efficiently.

Share commitment: Proof details

To prove that the data corresponding to a share commitment was posted to Celestia using Blobstream, the following proofs need to be verified:

  1. share inclusion proof to the share commitment: meaning creating two merkle proofs:
    1. share merkle proof up to the subtree root corresponding to that share
    2. subtree root merkle proof to the share commitment
  2. share commitment inclusion proof to the data root tuple root: meaning four merkle proofs:
    1. subtree roots merkle proofs to the share commitment: to make sure the subtree roots are valid
    2. subtree roots merkle proofs up to the row roots: to prove that the subtree roots belong to a set of rows in the Celestia block
    3. row roots proofs to the data root: to prove that those rows belong to the Celestia Block
    4. data root tuple proof to the data root tuple: to prove that the Celestia block referenced by its height and data root, was committed to by Blobstream.

More details on the share commitment inclusion proof can be found in the commitment scheme docs and also the data square layout.

If all of these proofs are valid, then you successfully managed to prove that the data corresponding to that share commitment has been posted to Celestia.

NOTE

Generating/verifying share commitment proofs is still not supported. It still needs tooling to generate the proofs on the node side, and verifying them on the Solidity side which will be built in the upcoming months.

Share commitment: Compact proofs

There is a way to have compact proofs, when using share commitments, unlike the ones defined above; that allow less costly inclusion proofs. These require the ability to parse the protobuf encoded PFBs.

In fact, if the rollup project has a way to parse the protobuf encoded PFB, either in a smart contract or a zk-circuit, they will be able to create compact proofs of the rollup data.

These proofs will work as follows:

  • Parsing the PFB and taking out the share commitment
  • Comparing the PFB commitment to the one saved in the rollup contract
  • Proving inclusion of the PFB to the data root tuple root. This will be a compact proof since we will only be proving two shares regardless of the size of the rollup data.

More details on compact proofs can be found in ADR-011.

Share commitment: Pros

The pros of referencing rollup data using a share commitment:

  • Using the same commitment that exists on the PFB, without having to find another way of referencing the rollup data.
  • If the team has access to protobuf parsing, it allows for compact proof, but the parsing costs need to be investigated.

Share commitment: Cons

  • Large/expensive proofs in the case of having no way to parse the protobuf PFB encoding.
  • In the optimistic rollups construction, defined below, this requires waiting for the Celestia block to be committed to by Blobstream before saving updating the settlement contract. This might require waiting for a few hours, depending on the batches size on each chain, to finally submit the rollup update.

Given these limitations, an alternative design will be discussed in the next section.

Sequence of spans

An alternative way of referencing rollup data in the rollup settlement contract is using a sequence of spans.

A sequence of spans is a data pointer that allows pointing to the rollup data inside a Celestia square using its location inside the square. It can be defined using the following information:

  • height: The height of the Celestia block containing the rollup data.
  • startIndex: The index of the first share containing the rollup data.
  • dataLen: The number of shares containing the rollup data.

The startIndex and the dataLen can be queried from Celestia after the corresponding transaction gets included in a block and committed to the chain. An example of how to query them can be found in the verify command. The TxShareRange returns the start and end share of the data referenced by a transaction hash.

NOTE

If the rollup data is submitted in multiple blocks, the above sequence of spans can be generalized to include multiple blocks. For simplicity, we will stick with the data only submitted to a single Celestia block.

Sequence of spans: Proof details

Using sequence of spans is different from using the share commitment because we're referencing a location in the square, and not actual data commitment. So, the proof types and their generation are different.

Sequence of spans: Proving unavailable data

By construction, if the sequence of spans refers to a certain location in the square, that location is the data. This location can be in the reserved namespaces, the parity bytes, etc. What matters is that it's part of the square. So to prove that the sequence of spans is invalid, i.e., refers to data that is not available on Celestia, it is necessary and sufficient to show that the sequence of spans doesn't belong to the Celestia block, i.e., the span is out of bounds.

We could create this proof via generating a binary Merkle proof of any row/column to the Celestia data root. This proof will provide the total which is the number of rows/columns in the extended data square. This can be used to calculate the square size. The computeSquareSizeFromRowProof method in the DAVerifier library allows calculating the square size from a row proof or a share proof.

Then, we will use that information to check if the provided share index, in the header, is out of the square size bounds. In order words, we will check if the startIndex and the startIndex + dataLen are included in the range [0, 4*square_size].

NOTE

The square size is the number of rows of the original square.

For the data root, we will use a binary Merkle proof to prove its inclusion in a data root tuple root that was committed to by the Blobstream smart contract. More on this in the data root inclusion proofs section.

Sequence of spans: Proving inclusion of some data

The difference between using a share commitment and a sequence of spans is that when using a share commitment, an extra merkle proof is needed to prove inclusion of the share to the share commitment. However, in the case of a sequence of spans, only the usual inclusion proof of a share to the data root tuple root is needed. The inclusion of the share to the sequence of spans is gotten using the same proof.

In fact, proving that a share is part of the sequence of spans, i.e., part of the rollup data is done as follows:

  1. Prove that the data root tuple is committed to by the Blobstream smart contract:

    To prove the data root is committed to by the Blobstream smart contract, we will need to provide a Merkle proof of the data root tuple to a data root tuple root. This can be created using the data_root_inclusion_proof query. More on this can be found in the data root inclusion proofs documentation.

  2. Verify inclusion proof of the data to Celestia data root:

    To prove that the data is part of the data root, we will need to provide two proofs: a namespace Merkle proof of the data to a row root. This could be done via proving the shares that contain the data to the row root using a namespace Merkle proof. And, a binary Merkle proof of the row root to the data root.

    These proofs can be generated using the ProveShares query.

    More details on these proofs can be found in the transaction inclusion proof documentation.

  3. Prove that the data is in the sequence spans:

    To prove that the data is part of the rollup sequence of spans, we take the authenticated share proofs in step (2) and use the shares begin/end key to define the shares' positions in the row.

    Then, we use the row proof to get the row index in the extended Celestia square and get the index of the share in row major order:

    solidity
    uint256 shareIndexInRow = shareProof.shareProofs[0].beginKey;
    +uint256 shareIndexInRowMajorOrder = shareIndexInRow + shareProof.rowProofs[0].numLeaves * shareProof.rowProofs[0].key;
    uint256 shareIndexInRow = shareProof.shareProofs[0].beginKey;
    +uint256 shareIndexInRowMajorOrder = shareIndexInRow + shareProof.rowProofs[0].numLeaves * shareProof.rowProofs[0].key;

Finally, we can compare the computed index with the sequence of spans, and be sure that the data/shares is part of the rollup data.

Sequence of spans: Pros

  • Using a sequence of spans instead of the share commitment allows for simpler proofs

Sequence of spans: Cons

None

Optimistic rollups

One type of rollups that can be built with Blobstream is optimistic rollups. An optimistic rollup is a rollup that commits optimistically to a set of blocks, and allows the other parties to verify that the blocks are valid, and if they're not, they can create fraud proofs to signal that.

Celestia allows optimistic rollups to post their data on its DA layer, and to prove that the data is available using Blobstream.

To build an optimistic rollup that uses Celestia as a DA layer, the following constructions can be inspired by.

Optimistic rollups that use a sequence of spans

Optimistic rollups can post their data in Celestia, then in the rollup settlement contract, they can reference optimistically that data using a sequence of spans. Then, rollup full nodes can verify if that data is valid. If not, they can trigger a fraud proof.

When using a sequence of spans, triggering the data availability fraud proofs, which are different from the state transitions fraud proofs (left for the rollup to define), goes back to the following cases:

Optimistic rollups that use a sequence of spans: Pros

  • Not needing to verify anything at the moment of submitting the commitments to the rollup settlement contracts
  • The fraud proofs are simple and can be reduced to a single share: if, for example, a single transaction in the rollup data that was posted to Celestia is faulty, only the shares containing that transaction, which can be as minimal as a single share, need to be proven on chain and verified.

Optimistic rollups that use a sequence of spans: Cons

None

Optimistic rollups that use a sequence of spans: Example

An example optimistic rollup that uses sequence of spans to reference its data can be found in the RollupInclusionProofs. It portrays the different possible data availability proofs, constructs them and shows how to verify them.

Also, more details on querying these kinds of proofs can be found in the proof queries documentation.

Optimistic rollups that use share commitments

Another way to build a rollup is to replace the sequence of spans with a height and a share commitment. Then, users/rollup full nodes will be able to query that data and validate it. If the rollup data is not valid, they can create a fraud proof.

The first difference between the sequence of spans construction and the share commitment construction is having to verify that the provided share commitment is part of the Celestia block, referenced by its height in the moment of submitting the rollup commitments to the settlement contract. This is necessary to make sure that the commitment is part of Celestia. Otherwise, rollup sequencers can commit to random share commitments and there won't be a way to prove they're invalid.

The second difference is the proof types. In the case of a fraud proof, the proofs outlined in the proofs details of share commitment section would need to be verified to be sure that the share containing the invalid state transition is part of the rollup data. Alternatively, the rollup settlement contract would need to have a library to parse protobuf encoded PFBs, as explained in the compact proofs of share commitment section, to have less expensive proofs. The cost of parsing the protobuf is not included in this analysis and needs to be investigated separately.

Optimistic rollups that use share commitments: Pros

  • Using the same share commitment as the one saved in Celestia which gives access to existing tooling

Optimistic rollups that use share commitments: Cons

  • The proofs are expensive in the base case. And if the settlement contract is able to parse the PFBs, thorough investigations of the cost of that would need to be done.

Zk-rollups

Zk-rollups, aka validity rollups, can also use Celestia as a DA and Blobstream to verify that the data was posted. However, the submission process is different from the above constructions, since there are no fraud proofs, and everything should be verified when submitting the commitment to the settlement contract.

Similar to the optimistic case, the rollup settlement contract can reference the rollup data using either the sequence of spans approach or the share commitments. We will discuss both in this section.

Zk-rollups that use sequence of spans

When submitting the commitments to the rollup settlement contract, this latter will need to verify the following:

  1. Zk-proof of the state transitions, which is left for the rollup to define.
  2. Verify that the sequence of spans is valid, i.e., is part of the Celestia block referenced by its height, as described in the proof details section.
  3. Zk-proof of the rollup data to the data root. The verification process of this should accept a commitment as input so that the settlement contract makes sure it's the correct value that's being saved. The commitment can be the data root and the sequence of spans. And, when the rollup data is proven inside the circuit to the data root, the used data root is asserted to be the input one. Similarly, the data's location is asserted to be the same as the input sequence of spans. These arguments are the ones used in the sequence of spans verification in (2).

Once these are valid, the settlement contract can be sure that the rollup data was posted to Celestia, and the sequence of spans references it correctly.

Zk-rollups that use sequence of spans: Pros

  • The inclusion proof inside the zk-circuit is a simple proof that uses traditional merkle tree. In the case of using share commitment, as will be explained below, additional libraries that can be expensive to prove are required.

Zk-rollups that use sequence of spans: Cons

None

Zk-rollups that use share commitments

To use share commitments to reference rollup data in the zk-rollup settlement contract, the zk-circuits need to be able to deserialize protobuf encoded messages. Alternatively, more involved merkle proofs will need to be verified.

Protobuf deserialization inside a zk-circuit

One way of using the share commitment to reference the rollup data in zk-rollups is via using a protobuf deserialization library inside the zk-circuit. And the verification would proceed as follows:

  1. Zk-proof of the state transitions, which is left to the rollup team to define.
  2. Verify that the share commitment is valid using the proofs laid out in the proof details of share commitment section.
  3. The zk-proof verifier would take as argument the data root and the share commitment. Then, inside the circuit, the protobuf encoded PFB transaction will be deserialized and then verify the following:
  • The deserialized share commitment is the same as the one provided as input
  • The circuit will prove the inclusion of the PFB to the data root, then assert that the data root is the same as the one provided as input.

If the above conditions are valid, the rollup settlement contract can be sure that the rollup data was posted to Celestia and is correctly referenced.

Zk-rollups that use share commitments: Pros

None

Zk-rollups that use share commitments: Cons

  • This approach requires having access to a protobuf decoder inside a zk-circuit which is not straightforward to have. Also, the relative costs will need to be investigated.

Heavy merkle proofs usage

Similar to Protobuf deserialization inside a zk-circuit, the zk-circuit will proceed to the verification of the availability of the data. The difference is that instead of parsing the encoded protobuf, the proofs defined under the share commitment proof details section will need to be verified inside the zk-circuit as follows:

  1. Zk-proof of the state transitions, which is left to the rollup team to define.
  2. Verify that the share commitment is valid using the proofs laid out in the share commitment proof details section.
  3. The zk-proof verifier would take as argument the data root and the share commitment. Then, inside the circuit:
  • It will verify that the input share commitment corresponds to the rollup data.
  • Verify that the input data root commits to that share commitment. Check the share commitment proof details for more details

Once these proofs are valid, the rollup settlement contract can be sure that the rollup data was posted to Celestia and is correctly referenced.

heavy merkle proofs usage: Pros

None

heavy merkle proofs usage: Cons

  • More heavy usage of merkle proofs inside and outside the zk-circuit.

Conclusion

Given the above details, using the sequence of spans is the better solution in the general case as explained in the optimistic rollups that uses a sequence of spans and zk-rollups that use sequence of spans sections. The proof sizes are small and allow for greater flexibility. However, if the rollup team has different requirements, then the other designs can be explored.

`,103),i=[r];function n(l,h,c,p,u,d){return t(),o("div",null,i)}const b=e(s,[["render",n]]);export{f as __pageData,b as default}; diff --git a/assets/developers_blobstream-rollups.md.b8b40693.lean.js b/assets/developers_blobstream-rollups.md.b8b40693.lean.js new file mode 100644 index 00000000000..23d4d87f42d --- /dev/null +++ b/assets/developers_blobstream-rollups.md.b8b40693.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.51d6c45b.js";const f=JSON.parse('{"title":"Introduction to Blobstream rollups","description":"Learn how to build rollups that use Blobstream.","frontmatter":{"description":"Learn how to build rollups that use Blobstream.","head":[["meta",{"name":"og:title","content":"Introduction to Blobstream rollups | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstream-rollups.md","filePath":"developers/blobstream-rollups.md","lastUpdated":1713450778000}'),s={name:"developers/blobstream-rollups.md"},r=a("",103),i=[r];function n(l,h,c,p,u,d){return t(),o("div",null,i)}const b=e(s,[["render",n]]);export{f as __pageData,b as default}; diff --git a/assets/developers_blobstream-x-deploy.md.33965d1e.js b/assets/developers_blobstream-x-deploy.md.33965d1e.js new file mode 100644 index 00000000000..f5c5cc4f095 --- /dev/null +++ b/assets/developers_blobstream-x-deploy.md.33965d1e.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as o}from"./chunks/framework.51d6c45b.js";const b=JSON.parse('{"title":"New Blobstream X deployments","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"New Blobstream X deployments | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstream-x-deploy.md","filePath":"developers/blobstream-x-deploy.md","lastUpdated":1709911265000}'),c={name:"developers/blobstream-x-deploy.md"},n=o('

New Blobstream X deployments

If you want to deploy Blobstream X to a new chain, where a Gateway contract does not exist, you need to do the following.

If Succinct Gateway already exists on this chain, you can skip this step. You can check the list of deployed SuccinctGateway contracts.

Deploy a new SuccinctGateway contract

Deploy a new SuccinctGateway contract to the new chain.

Deploy a BlobstreamX contract to the new chain

Follow the guide to use your deployed SuccinctGateway to deploy a new Blobstream X contract.

Run a local prover

Now that you have the contracts deployed, you can run a local prover.

',9),r=[n];function l(s,d,i,p,h,m){return a(),t("div",null,r)}const u=e(c,[["render",l]]);export{b as __pageData,u as default}; diff --git a/assets/developers_blobstream-x-deploy.md.33965d1e.lean.js b/assets/developers_blobstream-x-deploy.md.33965d1e.lean.js new file mode 100644 index 00000000000..b1f2c0b0cea --- /dev/null +++ b/assets/developers_blobstream-x-deploy.md.33965d1e.lean.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as o}from"./chunks/framework.51d6c45b.js";const b=JSON.parse('{"title":"New Blobstream X deployments","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"New Blobstream X deployments | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstream-x-deploy.md","filePath":"developers/blobstream-x-deploy.md","lastUpdated":1709911265000}'),c={name:"developers/blobstream-x-deploy.md"},n=o("",9),r=[n];function l(s,d,i,p,h,m){return a(),t("div",null,r)}const u=e(c,[["render",l]]);export{b as __pageData,u as default}; diff --git a/assets/developers_blobstream.md.bbaa04fb.js b/assets/developers_blobstream.md.bbaa04fb.js new file mode 100644 index 00000000000..72acd27ced5 --- /dev/null +++ b/assets/developers_blobstream.md.bbaa04fb.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as o}from"./chunks/framework.51d6c45b.js";const r="/img/blobstream/blobstream_logo.png",s="/img/blobstream/Celestia_Blobstream_X1b.png",i="/img/blobstream/Celestia_Blobstream_X2b.png",B=JSON.parse('{"title":"Blobstream: Streaming modular DA to Ethereum","description":"Learn how to integrate your L2 with Blobstream","frontmatter":{"description":"Learn how to integrate your L2 with Blobstream","head":[["meta",{"name":"og:title","content":"Blobstream: Streaming modular DA to Ethereum | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstream.md","filePath":"developers/blobstream.md","lastUpdated":1713450778000}'),l={name:"developers/blobstream.md"},n=o('

Blobstream: Streaming modular DA to Ethereum

Blobstream logo

What is Blobstream?

Blobstream is the first data availability solution for Ethereum that securely scales with the number of users. Formerly known as the Quantum Gravity Bridge (QGB), Blobstream relays commitments to Celestia's data root to an onchain light client on Ethereum, for integration by developers into L2 contracts. This enables Ethereum developers to build high-throughput L2s using Celestia's optimised DA layer, the first with Data Availability Sampling (DAS). Any ecosystem can deploy a Blobstream light client onchain to allow L2s and L3s to access DA from Celestia.

An implementation of Blobstream, by Succinct, called Blobstream X, is out and will be used in the upcoming deployments. This implementation proves the validity of Celestia block headers on a target EVM chain using zero-knowledge (ZK) proofs, which allow inheriting all the security guarantees of Celestia.

Please note: Blobstream remains early-stage, experimental software and users should use Blobstream at their own risk.

Blobstream vs. data availability committees (DACs)

Decentralization and security

Blobstream is built on Celestia, which uses a CometBFT-based proof-of-stake system. Blobstream shares the same security assumptions as Celestia. In contrast, data availability committees (DACs), are typically centralized or semi-centralized, relying on a specific set of entities or individuals to vouch for data availability.

Mechanism of verification

Blobstream uses data availability attestations, which are Merkle roots of the batched L2 data, to confirm that the necessary data is present on Celestia. The L2 contract on Ethereum can check directly with Blobstream if the data is published on Celestia. Similarly, a DAC would rely on attestations or confirmations from its permissioned members.

Flexibility and scalability

Blobstream is designed to offer high-throughput data availability for Ethereum L2s, aiming to strike a balance between scalability and security. It operates independently of Ethereum's gas costs, as Celestia's resource pricing is more byte-focused rather than computation-centric. On the other hand, the scalability and flexibility of a DAC would depend on its specific design and implementation.

In summary, both Blobstream and DACs aim to ensure offchain data availability, but Blobstream offers a more decentralized, secure, and scalable solution compared to the potential centralized nature of DACs.

What is Blobstream X?

Blobstream X is an implementation of Blobstream with a ZK light client that bridges Celestia’s modular DA layer to Ethereum to allow high-throughput rollups to use Celestia’s DA while settling on Ethereum.

Optimistic or ZK rollups that settle on Ethereum, but wish to use Celestia for DA, require a mechanism for bridging Celestia’s data root to Ethereum as part of the settlement process. This data root is used during inclusion proofs to prove that particular rollup transactions were included and made available in the Celestia network.

Bridging Celestia’s data root to Ethereum requires running a Celestia light client as a smart contract on Ethereum, to make the latest state of the Celestia chain known on Ethereum and available to rollups. Blobstream X utilizes the latest advances in ZK proofs to generate a succinct proof that enough Celestia validators have come to consensus (according to the CometBFT consensus protocol) on a block header, and verifies this proof in the Blobstream X Ethereum smart contract to update it with the latest Celestia header.

The Blobstream X ZK proof not only verifies the consensus of Celestia validators, but it also merkelizes and hashes all the data roots in the block range from the previous update to the current update, making accessible all Celestia data roots (verifiable with a Merkle inclusion proof against the stored Merkle root) to rollups.

Blobstream X is built and deployed with Succinct's protocol.

blobstream x draft diagram

Integrate with Blobstream X

The following docs go over how developers can integrate Blobstream X.

You can find the repository for Blobstream X along with code for:

The first deployments of Blobstream X will be maintained on the following chains: Arbitrum One, Base and Ethereum Mainnet. Every 1 hour, the prover/relayer will post an update to the Blobstream X contract that will include a new data commitment range that covers a 1-hour block range from the latestBlock in the Blobstream X contract. On Ethereum Mainnet, the Blobstream X contract will be updated every 4 hours.

NOTE

Custom ranges can be requested using the BlobstreamX contract to create proofs for specific Celestia block batches. These ranges can be constructed as [latestBlock, customTargetBlock), with latestBlock as the latest block height that was committed to by the BlobstreamX contract, and latestBlock > customTargetBlock, and customTargetBlock - latestBlock <= DATA_COMMITMENT_MAX.

Block ranges that are before the contract's latestBlock can't be proven a second time in different batches.

More information can be found in the requestHeaderRange(...) method.

How Blobstream X works

As shown in the diagram below, the entrypoint for updates to the Blobstream X contract is through the SuccinctGateway smart contract, which is a simple entrypoint contract that verifies proofs (against a deployed onchain verifier for the Blobstream X circuit) and then calls the BlobstreamX.sol contract to update it. Find more information about the SuccinctGateway.

blobstream x overview diagram draft

NOTE

If the Blobstream X contract is not deployed on a desired chain, it needs to be deployed before it can be used by your rollup. See the deployment documentation for more details.

How to integrate with Blobstream X

Integrating your L2 with Blobstream X requires two components: your onchain smart contract logic, and your offchain client logic for your rollup. The next three sections cover these topics:

Deployed contracts

You can interact with the Blobstream X contracts today on testnet. The Blobstream X Solidity smart contracts are currently deployed on the following Ethereum testnets:

ContractEVM networkContract addressAttested data on Celestia
Blobstream XEthereum MainnetNot yet deployedMainnet Beta
Blobstream XArbitrum One0xA83ca7775Bc2889825BcDeDfFa5b758cf69e8794Mainnet Beta
Blobstream XBase0xA83ca7775Bc2889825BcDeDfFa5b758cf69e8794Mainnet Beta
Blobstream XSepolia0xf0c6429ebab2e7dc6e05dafb61128be21f13cb1eMocha testnet
Blobstream XArbitrum Sepolia0xc3e209eb245Fd59c8586777b499d6A665DF3ABD2Mocha testnet
Blobstream XBase Sepolia0xc3e209eb245Fd59c8586777b499d6A665DF3ABD2Mocha testnet

Blobstream rollups

More on the different ways to build a blobstream rollups can be found in the blobstream rollups documentation.

',39),c=[n];function d(h,b,m,u,p,f){return t(),a("div",null,c)}const y=e(l,[["render",d]]);export{B as __pageData,y as default}; diff --git a/assets/developers_blobstream.md.bbaa04fb.lean.js b/assets/developers_blobstream.md.bbaa04fb.lean.js new file mode 100644 index 00000000000..bd42ec05f64 --- /dev/null +++ b/assets/developers_blobstream.md.bbaa04fb.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as o}from"./chunks/framework.51d6c45b.js";const r="/img/blobstream/blobstream_logo.png",s="/img/blobstream/Celestia_Blobstream_X1b.png",i="/img/blobstream/Celestia_Blobstream_X2b.png",B=JSON.parse('{"title":"Blobstream: Streaming modular DA to Ethereum","description":"Learn how to integrate your L2 with Blobstream","frontmatter":{"description":"Learn how to integrate your L2 with Blobstream","head":[["meta",{"name":"og:title","content":"Blobstream: Streaming modular DA to Ethereum | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstream.md","filePath":"developers/blobstream.md","lastUpdated":1713450778000}'),l={name:"developers/blobstream.md"},n=o("",39),c=[n];function d(h,b,m,u,p,f){return t(),a("div",null,c)}const y=e(l,[["render",d]]);export{B as __pageData,y as default}; diff --git a/assets/developers_bubs-testnet.md.50110685.js b/assets/developers_bubs-testnet.md.50110685.js new file mode 100644 index 00000000000..ae3fe424628 --- /dev/null +++ b/assets/developers_bubs-testnet.md.50110685.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as s}from"./chunks/framework.51d6c45b.js";const o="/img/Celestia_Bubs_Testnet.jpg",f=JSON.parse('{"title":"Bubs testnet","description":"The first testnet built with OP Stack and Celestia.","frontmatter":{"description":"The first testnet built with OP Stack and Celestia.","head":[["meta",{"name":"og:title","content":"Bubs testnet | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/bubs-testnet.md","filePath":"developers/bubs-testnet.md","lastUpdated":1711392621000}'),r={name:"developers/bubs-testnet.md"},i=s('

Bubs testnet

Bubs testnet

Bubs Testnet is a fresh offering from Caldera with support from Celestia Labs, built with OP Stack and Celestia, and is dedicated to providing developers with an EVM-compatible execution layer to deploy their EVM applications on.

Built with the OP Stack and Celestia

The Bubs Testnet is a testnet rollup, a modified version of optimism-bedrock that uses Celestia as a data availability (DA) layer. This integration can be found in the @celestiaorg/optimism repository. The testnet is hosted by Caldera, who makes it easy to launch rollups with no code required.

In this setup, data handling is accomplished in two ways. Firstly, data is written to the DA layer, in this case, Celestia (on the Mocha testnet). Then, the data commitment is written to the op-batcher. When reading, the op-node retrieves the data back from the DA layer by first reading the data commitment from the op-batcher, then reading the data from the DA layer using the data commitment. Hence, while previously op-node was reading from calldata on Ethereum, it now reads data from Celestia. View the namespace for Bubs on Celestia's Mocha testnet.

The tools involved in the data handling process include op-batcher, which batches up rollup blocks and posts them to Ethereum, op-geth that handles execution, and op-proposer responsible for state commitment submission.

By using Celestia as a DA layer, existing L2s can switch from posting their data as calldata on Ethereum to posting to Celestia. The commitment to the block is posted on Celestia, which is purpose-built for data availability. This is more scalable than the traditional method of posting this data as calldata on monolithic chains.

Building on Bubs

Bubs Testnet provides a robust environment for developers to test their Ethereum Virtual Machine (EVM) applications. It offers an EVM-compatible execution layer, making it an ideal platform for developers looking to build and test applications in a setting that closely mirrors an OP Stack rollup on Celestia.

Learn more at https://bubs-sepolia.hub.caldera.xyz/.

RPC URLs

Remote Procedure Call (RPC) URLs are endpoints that allow developers to interact with the blockchain. They are essential for sending transactions, querying blockchain data, and performing other interactions with the blockchain.

For the Bubs Testnet, you can connect to the following RPC URLs:

HTTPS

  • https://bubs-sepolia.rpc.caldera.xyz/http

WSS

  • wss://bubs-sepolia.rpc.caldera.xyz/ws

This URL serves as the entry point to the Bubs Testnet. You can use it in your applications to connect to the testnet and interact with the smart contracts you deploy there.

Remember, Bubs Testnet is a testing environment!

Bridge

Bridging is a process that enables the transfer of assets between different blockchains.

To bridge between Ethereum Sepolia and Bubs Testnet, visit the Bubs Testnet bridge.

Faucet

To visit the Bubs testnet faucet, go to https://bubs-sepolia.hub.caldera.xyz/ and click the "Faucet" tab.

Explorer

To visit the explorer, go to https://bubs-sepolia.explorer.caldera.xyz/.

Status

To see the status and uptime information for Bubs, visit the status page.

Next steps

Now that you have a better understanding of the Bubs Testnet and its integration of OP Stack and Celestia, you can start exploring its capabilities:

',32),n=[i];function l(h,c,d,p,u,b){return t(),a("div",null,n)}const g=e(r,[["render",l]]);export{f as __pageData,g as default}; diff --git a/assets/developers_bubs-testnet.md.50110685.lean.js b/assets/developers_bubs-testnet.md.50110685.lean.js new file mode 100644 index 00000000000..9a0432136cc --- /dev/null +++ b/assets/developers_bubs-testnet.md.50110685.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as s}from"./chunks/framework.51d6c45b.js";const o="/img/Celestia_Bubs_Testnet.jpg",f=JSON.parse('{"title":"Bubs testnet","description":"The first testnet built with OP Stack and Celestia.","frontmatter":{"description":"The first testnet built with OP Stack and Celestia.","head":[["meta",{"name":"og:title","content":"Bubs testnet | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/bubs-testnet.md","filePath":"developers/bubs-testnet.md","lastUpdated":1711392621000}'),r={name:"developers/bubs-testnet.md"},i=s("",32),n=[i];function l(h,c,d,p,u,b){return t(),a("div",null,n)}const g=e(r,[["render",l]]);export{f as __pageData,g as default}; diff --git a/assets/developers_build-whatever.md.1a071340.js b/assets/developers_build-whatever.md.1a071340.js new file mode 100644 index 00000000000..ac325c34bf1 --- /dev/null +++ b/assets/developers_build-whatever.md.1a071340.js @@ -0,0 +1 @@ +import{_ as d,o as l,c as i,k as t,t as r,e as p,H as e,Q as s,a as n}from"./chunks/framework.51d6c45b.js";const h="/img/da-and-validity.png";const m={name:"UrlImageButton",props:{url:String,imageSrc:String,text:String,notes:{type:String,default:""}}},g=["href"],b={class:"url-image-button"},f=["src"],y={class:"button-text"},k={key:0,class:"notes-text"};function w(c,u,o,z,O,B){return l(),i("a",{href:o.url,target:"_blank",rel:"noopener noreferrer",class:"button-link"},[t("button",b,[t("img",{src:o.imageSrc,alt:"",class:"button-icon"},null,8,f),t("span",y,r(o.text),1),o.notes?(l(),i("span",k,r(o.notes),1)):p("",!0)])],8,g)}const a=d(m,[["render",w],["__scopeId","data-v-21baf7e3"]]),_=s('

Build whatever

If you're a developer and want to know what the benefits of modular blockchains are for you, you’ve come to the right place. This page will give you the rundown on modular blockchains and their benefits for developers like you.

This section provides various guides and tutorials that cover different options for deploying rollups on Celestia.

Quickstart - Building on Celestia

Choose a framework

So, you’re ready to start experimenting and building on Celestia? Here are a few options that are currently available for developers.

',6),v={style:{display:"flex","flex-wrap":"wrap","justify-content":"center","align-items":"center","grid-template-columns":"repeat(auto-fill, minmax(150px, 1fr))",gap:"20px"}},x=t("h3",{id:"rollups-as-a-service",tabindex:"-1"},[n("Rollups-as-a-Service "),t("a",{class:"header-anchor",href:"#rollups-as-a-service","aria-label":'Permalink to "Rollups-as-a-Service"'},"​")],-1),S=t("p",null,"Deploy your rollup with a RaaS provider.",-1),C={style:{display:"flex","flex-wrap":"wrap","justify-content":"center","align-items":"center","grid-template-columns":"repeat(auto-fill, minmax(150px, 1fr))",gap:"20px"}},P=t("h3",{id:"smart-contracts",tabindex:"-1"},[n("Smart contracts "),t("a",{class:"header-anchor",href:"#smart-contracts","aria-label":'Permalink to "Smart contracts"'},"​")],-1),q=t("p",null,"Deploy your smart contracts on dedicated EVM-compatible rollups.",-1),A={style:{display:"flex","flex-wrap":"wrap","justify-content":"center","align-items":"center","grid-template-columns":"repeat(auto-fill, minmax(150px, 1fr))",gap:"20px"}},D=s('

What is a rollup?

A rollup is a type of blockchain that offloads some work to a layer 1, like Celestia. Rollups host applications and process user transactions. Once those transactions get processed, they are then published to layer 1. It’s layer 1s job to order those transactions and check that they are available, at minimum.

What is a modular blockchain?

With blockchains there are more or less four core functions that they do.

  • Execution: transaction execution and state update.
  • Settlement: finality and dispute resolution.
  • Consensus: agreement on transaction ordering.
  • Data availability: prove data was published to the network.

Modular blockchains specialize in one or two of these functions rather than doing all of them like a monolithic blockchain. You probably know about layer 1s and layer 2s. That’s the general idea.

A typical example of a modular blockchain you might’ve heard of is a rollup. Rollups host smart contracts and execute transactions, much like any monolithic chain. But, the data of those transactions get sent to a layer 1 blockchain to carry out the remaining functions.

If you want to brush up on your understanding of modular blockchains, head over to learn modular.

Benefits of modular blockchains

Ease of deploying a chain

One of the goals of modular blockchains is to make it as easy to deploy a blockchain as a smart contract. There are a few unique ways that modular blockchains can significantly reduce the cost of deploying a new blockchain.

  1. No validator set is required. Rollups can deploy without sourcing their own set of validators or sequencers.
  2. Inherit security from the start. Rollups don’t need to build all their security from scratch.
  3. Any part of the stack can be delegated. Development time can be reduced by outsourcing functions of the rollup to external providers.

All in all, builders will be able to outsource as much of the stack as they need. Deploying a new blockchain will be as simple as clicking a few options to initialize a production-ready rollup.

Scaling

Of course, a much higher scale is necessary if we want to support many more users. And modular blockchains use some new innovative technologies that can help us get there.

  • Data availability sampling enables modular blockchains like Celestia to scale data availability with the number of light nodes - that means more capacity for rollups.
  • Fraud and validity proofs make rollups vastly more efficient to verify. Nodes only need to verify a small proof of transaction validity (validity proof) or assume transactions are valid by default (fraud proof). This means rollups don’t require every node in the network to re-execute every transaction.

image

  • Decoupling execution from consensus lets developers define the VM that best fits the scaling needs of their application.
  • Separating applications across multiple rollups isolates congestion. If an application congests the execution capacity of one rollup, all other rollups remain unaffected in their execution capacity.

All these scaling properties combined make new types of applications and features possible, like onchain gaming, dynamic metadata, and ephemeral rollups, to name a few.

Customizability

By design, modular blockchains don’t lock in any feature set. They promote experimentation and customization.

Remember how decoupling execution from consensus enables VM customizability? Well, rollups are the execution component. Applications can run on their own rollup and adjust the VM to maximize their application's performance. Developers have that flexibility because Celestia's execution logic doesn't restrict rollups.

Basically, rollups can be customized to integrate any new or existing VM stack.

With existing rollup frameworks, developers can run rollup testnets using the EVM or Cosmos SDK. In the future, one can imagine a variety of VMs that rollup frameworks support, providing developers with more out-of-the-box options for their applications.

Some customizations that could be made to a rollup's VM include custom precompiles, changing transaction processing from sequential to parallel, or adding support for private smart contracts.

All of this only scratches the surface.

',26),R=JSON.parse('{"title":"Build whatever","description":"Advantages of building on modular blockchains like Celestia.","frontmatter":{"description":"Advantages of building on modular blockchains like Celestia.","head":[["meta",{"name":"og:title","content":"Build whatever | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/build-whatever.md","filePath":"developers/build-whatever.md","lastUpdated":1712870672000}'),V={name:"developers/build-whatever.md"},I=Object.assign(V,{setup(c){return(u,o)=>(l(),i("div",null,[_,t("div",v,[e(a,{url:"/developers/intro-to-op-stack",imageSrc:"/build/opstack.webp",text:"OP Stack",notes:"EVM",target:"_self",alt:"OP Stack logo","aria-label":"OP Stack"}),e(a,{url:"/developers/arbitrum-integration",imageSrc:"/build/arbitrum.webp",text:"Arbitrum Orbit",notes:"EVM",target:"_self",alt:"Arbitrum logo","aria-label":"Arbitrum"}),e(a,{url:"/developers/build-whatever",imageSrc:"/build/polygon.webp",text:"Polygon CDK",target:"_self",notes:"EVM: Coming soon",alt:"Polygon logo","aria-label":"Polygon"}),e(a,{url:"https://github.com/Sovereign-Labs/sovereign-sdk/tree/stable/examples/demo-rollup#demo-rollup",imageSrc:"/build/sovereign.webp",text:"Sovereign SDK",notes:"Sovereign",alt:"Sovereign logo","aria-label":"Sovereign"}),e(a,{url:"https://docs.dymension.xyz/build/overview/",imageSrc:"/build/dymension.webp",text:"Dymension",alt:"Dymension logo","aria-label":"Dymension"}),e(a,{url:"https://rollkit.dev",imageSrc:"/build/rollkit.webp",text:"Rollkit",notes:"Sovereign",alt:"Rollkit logo","aria-label":"Rollkit"})]),x,S,t("div",C,[e(a,{url:"https://altlayer.io/raas/",imageSrc:"/build/altlayer.webp",text:"AltLayer",notes:"Orbit, OP Stack, Polygon CDK",alt:"AltLayer logo","aria-label":"AltLayer"}),e(a,{url:"https://www.astria.org/",imageSrc:"/build/astria.webp",text:"Astria",alt:"Astria logo","aria-label":"Astria"}),e(a,{url:"https://www.caldera.xyz/",imageSrc:"/build/caldera.webp",text:"Caldera",notes:"Orbit, OP Stack",alt:"Caldera logo","aria-label":"Caldera"}),e(a,{url:"https://conduit.xyz/",imageSrc:"/build/conduit.webp",text:"Conduit",notes:"Orbit, OP Stack",alt:"Conduit logo","aria-label":"Conduit"}),e(a,{url:"https://gateway.fm/",imageSrc:"/build/gateway.webp",text:"Gateway",notes:"Polygon CDK",alt:"Gateway logo","aria-label":"Gateway"}),e(a,{url:"https://www.gelato.network/",imageSrc:"/build/gelato.webp",text:"Gelato",notes:"Orbit, OP Stack, Polygon CDK",alt:"Gelato logo","aria-label":"Gelato"}),e(a,{url:"https://www.karnot.xyz/",imageSrc:"/build/karnot.webp",text:"Karnot",notes:"Starknet",alt:"Karnot logo","aria-label":"Karnot"}),e(a,{url:"https://lumoz.org/",imageSrc:"/build/lumoz.webp",text:"Lumoz",notes:"Polygon CDK",alt:"Lumoz logo","aria-label":"Lumoz"}),e(a,{url:"https://snapchain.dev/",imageSrc:"/build/snapchain.webp",text:"Snapchain",notes:"Polygon CDK",alt:"Snapchain logo","aria-label":"Snapchain"}),e(a,{url:"https://docs.vistara.dev/",imageSrc:"/build/vistara.webp",text:"Vistara",alt:"Vistara logo","aria-label":"Vistara"}),e(a,{url:"https://www.zeeve.io/",imageSrc:"/build/zeeve.webp",text:"Zeeve",notes:"Orbit, OP Stack, Polygon CDK",alt:"Zeeve logo","aria-label":"Zeeve"})]),P,q,t("div",A,[e(a,{url:"https://bubstestnet.com/",imageSrc:"/build/caldera.webp",text:"Bubs testnet",notes:"OP Stack",alt:"Caldera logo","aria-label":"Caldera Bubs testnet"}),e(a,{url:"https://raas.gelato.network/rollups/details/public/opcelestia-raspberry",imageSrc:"/build/gelato.webp",text:"Raspberry testnet",notes:"OP Stack",alt:"Gelato logo","aria-label":"Gelato Raspberry testnet"})]),D]))}});export{R as __pageData,I as default}; diff --git a/assets/developers_build-whatever.md.1a071340.lean.js b/assets/developers_build-whatever.md.1a071340.lean.js new file mode 100644 index 00000000000..3ae3b8f2e70 --- /dev/null +++ b/assets/developers_build-whatever.md.1a071340.lean.js @@ -0,0 +1 @@ +import{_ as d,o as l,c as i,k as t,t as r,e as p,H as e,Q as s,a as n}from"./chunks/framework.51d6c45b.js";const h="/img/da-and-validity.png";const m={name:"UrlImageButton",props:{url:String,imageSrc:String,text:String,notes:{type:String,default:""}}},g=["href"],b={class:"url-image-button"},f=["src"],y={class:"button-text"},k={key:0,class:"notes-text"};function w(c,u,o,z,O,B){return l(),i("a",{href:o.url,target:"_blank",rel:"noopener noreferrer",class:"button-link"},[t("button",b,[t("img",{src:o.imageSrc,alt:"",class:"button-icon"},null,8,f),t("span",y,r(o.text),1),o.notes?(l(),i("span",k,r(o.notes),1)):p("",!0)])],8,g)}const a=d(m,[["render",w],["__scopeId","data-v-21baf7e3"]]),_=s("",6),v={style:{display:"flex","flex-wrap":"wrap","justify-content":"center","align-items":"center","grid-template-columns":"repeat(auto-fill, minmax(150px, 1fr))",gap:"20px"}},x=t("h3",{id:"rollups-as-a-service",tabindex:"-1"},[n("Rollups-as-a-Service "),t("a",{class:"header-anchor",href:"#rollups-as-a-service","aria-label":'Permalink to "Rollups-as-a-Service"'},"​")],-1),S=t("p",null,"Deploy your rollup with a RaaS provider.",-1),C={style:{display:"flex","flex-wrap":"wrap","justify-content":"center","align-items":"center","grid-template-columns":"repeat(auto-fill, minmax(150px, 1fr))",gap:"20px"}},P=t("h3",{id:"smart-contracts",tabindex:"-1"},[n("Smart contracts "),t("a",{class:"header-anchor",href:"#smart-contracts","aria-label":'Permalink to "Smart contracts"'},"​")],-1),q=t("p",null,"Deploy your smart contracts on dedicated EVM-compatible rollups.",-1),A={style:{display:"flex","flex-wrap":"wrap","justify-content":"center","align-items":"center","grid-template-columns":"repeat(auto-fill, minmax(150px, 1fr))",gap:"20px"}},D=s("",26),R=JSON.parse('{"title":"Build whatever","description":"Advantages of building on modular blockchains like Celestia.","frontmatter":{"description":"Advantages of building on modular blockchains like Celestia.","head":[["meta",{"name":"og:title","content":"Build whatever | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/build-whatever.md","filePath":"developers/build-whatever.md","lastUpdated":1712870672000}'),V={name:"developers/build-whatever.md"},I=Object.assign(V,{setup(c){return(u,o)=>(l(),i("div",null,[_,t("div",v,[e(a,{url:"/developers/intro-to-op-stack",imageSrc:"/build/opstack.webp",text:"OP Stack",notes:"EVM",target:"_self",alt:"OP Stack logo","aria-label":"OP Stack"}),e(a,{url:"/developers/arbitrum-integration",imageSrc:"/build/arbitrum.webp",text:"Arbitrum Orbit",notes:"EVM",target:"_self",alt:"Arbitrum logo","aria-label":"Arbitrum"}),e(a,{url:"/developers/build-whatever",imageSrc:"/build/polygon.webp",text:"Polygon CDK",target:"_self",notes:"EVM: Coming soon",alt:"Polygon logo","aria-label":"Polygon"}),e(a,{url:"https://github.com/Sovereign-Labs/sovereign-sdk/tree/stable/examples/demo-rollup#demo-rollup",imageSrc:"/build/sovereign.webp",text:"Sovereign SDK",notes:"Sovereign",alt:"Sovereign logo","aria-label":"Sovereign"}),e(a,{url:"https://docs.dymension.xyz/build/overview/",imageSrc:"/build/dymension.webp",text:"Dymension",alt:"Dymension logo","aria-label":"Dymension"}),e(a,{url:"https://rollkit.dev",imageSrc:"/build/rollkit.webp",text:"Rollkit",notes:"Sovereign",alt:"Rollkit logo","aria-label":"Rollkit"})]),x,S,t("div",C,[e(a,{url:"https://altlayer.io/raas/",imageSrc:"/build/altlayer.webp",text:"AltLayer",notes:"Orbit, OP Stack, Polygon CDK",alt:"AltLayer logo","aria-label":"AltLayer"}),e(a,{url:"https://www.astria.org/",imageSrc:"/build/astria.webp",text:"Astria",alt:"Astria logo","aria-label":"Astria"}),e(a,{url:"https://www.caldera.xyz/",imageSrc:"/build/caldera.webp",text:"Caldera",notes:"Orbit, OP Stack",alt:"Caldera logo","aria-label":"Caldera"}),e(a,{url:"https://conduit.xyz/",imageSrc:"/build/conduit.webp",text:"Conduit",notes:"Orbit, OP Stack",alt:"Conduit logo","aria-label":"Conduit"}),e(a,{url:"https://gateway.fm/",imageSrc:"/build/gateway.webp",text:"Gateway",notes:"Polygon CDK",alt:"Gateway logo","aria-label":"Gateway"}),e(a,{url:"https://www.gelato.network/",imageSrc:"/build/gelato.webp",text:"Gelato",notes:"Orbit, OP Stack, Polygon CDK",alt:"Gelato logo","aria-label":"Gelato"}),e(a,{url:"https://www.karnot.xyz/",imageSrc:"/build/karnot.webp",text:"Karnot",notes:"Starknet",alt:"Karnot logo","aria-label":"Karnot"}),e(a,{url:"https://lumoz.org/",imageSrc:"/build/lumoz.webp",text:"Lumoz",notes:"Polygon CDK",alt:"Lumoz logo","aria-label":"Lumoz"}),e(a,{url:"https://snapchain.dev/",imageSrc:"/build/snapchain.webp",text:"Snapchain",notes:"Polygon CDK",alt:"Snapchain logo","aria-label":"Snapchain"}),e(a,{url:"https://docs.vistara.dev/",imageSrc:"/build/vistara.webp",text:"Vistara",alt:"Vistara logo","aria-label":"Vistara"}),e(a,{url:"https://www.zeeve.io/",imageSrc:"/build/zeeve.webp",text:"Zeeve",notes:"Orbit, OP Stack, Polygon CDK",alt:"Zeeve logo","aria-label":"Zeeve"})]),P,q,t("div",A,[e(a,{url:"https://bubstestnet.com/",imageSrc:"/build/caldera.webp",text:"Bubs testnet",notes:"OP Stack",alt:"Caldera logo","aria-label":"Caldera Bubs testnet"}),e(a,{url:"https://raas.gelato.network/rollups/details/public/opcelestia-raspberry",imageSrc:"/build/gelato.webp",text:"Raspberry testnet",notes:"OP Stack",alt:"Gelato logo","aria-label":"Gelato Raspberry testnet"})]),D]))}});export{R as __pageData,I as default}; diff --git a/assets/developers_celestia-node-key.md.db3dee61.js b/assets/developers_celestia-node-key.md.db3dee61.js new file mode 100644 index 00000000000..d6a00078f10 --- /dev/null +++ b/assets/developers_celestia-node-key.md.db3dee61.js @@ -0,0 +1,63 @@ +import{m as o}from"./chunks/mocha_versions.3ba4a823.js";import{o as p,c as t,k as s,a as n,t as l,l as e,Q as a}from"./chunks/framework.51d6c45b.js";const c=a(`

Create a wallet with celestia-node

This tutorial will go over using the cel-key utility to generate a wallet on celestia-node.

While this tutorial will go over installation process of cel-key, it is recommended that you complete the following prerequisites first:

Once you completed the prerequisite, you can proceed with this tutorial.

Using the cel-key utility

Inside the celestia-node repository is a utility called cel-key that uses the key utility provided by Cosmos-SDK under the hood. The utility can be used to add, delete, and manage keys for any DA node type (bridge || full || light), or just keys in general.

Installation

You need to first pull down the celestia-node repository:

sh
git clone https://github.com/celestiaorg/celestia-node.git
+cd celestia-node/
git clone https://github.com/celestiaorg/celestia-node.git
+cd celestia-node/

It can be built using either of the following commands:

sh
# dumps binary in current working directory, accessible via \`./cel-key\`
+make cel-key
# dumps binary in current working directory, accessible via \`./cel-key\`
+make cel-key

or

sh
# installs binary in GOBIN path, accessible via \`cel-key\`
+make install-key
# installs binary in GOBIN path, accessible via \`cel-key\`
+make install-key

For the purpose of this guide, we will use the make cel-key command.

Steps for generating node keys

To generate a key for a Celestia node, select the tab for your node type:

TIP

You do not need to declare a network for Mainnet Beta. Refer to the chain ID section on the troubleshooting page for more information

bash
./cel-key add <key-name> --keyring-backend test --node.type bridge \\
+  --p2p.network <network>
./cel-key add <key-name> --keyring-backend test --node.type bridge \\
+  --p2p.network <network>
bash
./cel-key add <key-name> --keyring-backend test --node.type full \\
+  --p2p.network <network>
./cel-key add <key-name> --keyring-backend test --node.type full \\
+  --p2p.network <network>
bash
./cel-key add <key-name> --keyring-backend test --node.type light \\
+  --p2p.network <network>
./cel-key add <key-name> --keyring-backend test --node.type light \\
+  --p2p.network <network>

This will load the key <key-name> into the directory of the node.

Further flags you can use to customize your key are the following:

  • --node.store: Specifies a different directory you can use to save your node data and configurations. Expects a path to a directory.
  • --p2p.network: Specifies which network you want the key for. Values are arabica and mocha. Please note the default network will be mocha.

Keep in mind that your celestia-node will only pick up keys that are inside the node.store directory under /keys so you should make sure to point cel-key utility to the correct directory via the node.store or p2p.network flags if you have specified a custom directory or network other than Mocha.

Also keep in mind that if you do not specify a network with --p2p.network, the default one will always be mocha.

Steps for exporting node keys

You can export a private key from the local keyring in encrypted and ASCII-armored format.

bash
./cel-key export <key-name> --keyring-backend test --node.type bridge \\
+  --p2p.network <network>
./cel-key export <key-name> --keyring-backend test --node.type bridge \\
+  --p2p.network <network>
bash
./cel-key export <key-name> --keyring-backend test --node.type full \\
+  --p2p.network <network>
./cel-key export <key-name> --keyring-backend test --node.type full \\
+  --p2p.network <network>
bash
./cel-key export <key-name> --keyring-backend test --node.type light \\
+  --p2p.network <network>
./cel-key export <key-name> --keyring-backend test --node.type light \\
+  --p2p.network <network>

Steps for importing node keys

To import from a mnemonic, use the following command, then enter your bip39 mnemonic:

bash
./cel-key add <key-name> --recover --keyring-backend test \\
+  --node.type bridge --p2p.network <network>
./cel-key add <key-name> --recover --keyring-backend test \\
+  --node.type bridge --p2p.network <network>
bash
./cel-key add <key-name> --recover --keyring-backend test \\
+  --node.type full --p2p.network <network>
./cel-key add <key-name> --recover --keyring-backend test \\
+  --node.type full --p2p.network <network>
bash
./cel-key add <key-name> --recover --keyring-backend test \\
+  --node.type light --p2p.network <network>
./cel-key add <key-name> --recover --keyring-backend test \\
+  --node.type light --p2p.network <network>

View all options for cel-key

sh
./cel-key --help
./cel-key --help

Docker and cel-key

Prerequisites

Running your node

Run the Docker image (in this example, we are using a light node on Mocha testnet):

`,37),r={class:"language-bash vp-adaptive-theme"},y=s("button",{title:"Copy Code",class:"copy"},null,-1),i=s("span",{class:"lang"},"bash",-1),E={class:"shiki github-dark vp-code-dark"},d=a('docker run --name celestia-node -e NODE_TYPE=light -e P2P_NETWORK=mocha -p 26659:26659 \\',1),F={class:"line"},h={style:{color:"#E1E4E8"}},g=a('celestia light start \\',7),k=a('--core.ip rpc-mocha.pops.one --p2p.network mocha',1),u={class:"shiki github-light vp-code-light"},C=a('docker run --name celestia-node -e NODE_TYPE=light -e P2P_NETWORK=mocha -p 26659:26659 \\',1),b={class:"line"},v={style:{color:"#24292E"}},m=a('celestia light start \\',7),B=a('--core.ip rpc-mocha.pops.one --p2p.network mocha',1),_=a(`

TIP

Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

You do not need to declare a network for Mainnet Beta. Refer to the chain ID section on the troubleshooting page for more information

List active containers in another window with:

bash
docker ps
docker ps

The response will look like:

bash
CONTAINER ID   IMAGE           COMMAND                  CREATED          STATUS          PORTS      NAMES
+<container-id>   celestia-node   "/entrypoint.sh cele…"   22 seconds ago   Up 21 seconds   2121/tcp   docker-compose-test-celestia-1
CONTAINER ID   IMAGE           COMMAND                  CREATED          STATUS          PORTS      NAMES
+<container-id>   celestia-node   "/entrypoint.sh cele…"   22 seconds ago   Up 21 seconds   2121/tcp   docker-compose-test-celestia-1

Interact with the container by replacing <container-id> for the container ID:

bash
docker exec -ti <container-id> /bin/bash
docker exec -ti <container-id> /bin/bash

Now, interact with cel-key to check for the key that was autogenerated when you started the node:

bash
./cel-key list --keyring-backend test --node.type light
./cel-key list --keyring-backend test --node.type light

You can also export your key from the container. In the next section, you'll learn how to mount existing keys to the container.

Mounting existing keys to container

In this example, we'll be mounting an existing key to the container. We're also using an existing image called celestia-node. This will mount the entire /.celestia-light-<p2p-network>/keys directory to your image, or on Mainnet Beta the /.celestia-light/keys directory.

Write a docker-compose.yml to accomplish this:

yaml
version: "3.8"
+services:
+  celestia:
+    image: celestia-node
+    environment:
+      - NODE_TYPE=light
+    command: celestia light start --core.ip rpc-mocha.pops.one --p2p.network mocha --keyring.accname my_celes_key
+    volumes:
+      - \${PWD}/keys:/root/.celestia-light-mocha-4/keys
+    ports:
+      - 26659:26659
version: "3.8"
+services:
+  celestia:
+    image: celestia-node
+    environment:
+      - NODE_TYPE=light
+    command: celestia light start --core.ip rpc-mocha.pops.one --p2p.network mocha --keyring.accname my_celes_key
+    volumes:
+      - \${PWD}/keys:/root/.celestia-light-mocha-4/keys
+    ports:
+      - 26659:26659

Start the container by running the following command in the directory with your docker-compose.yml:

bash
docker-compose up
docker-compose up

List active containers in another window with:

bash
docker ps
docker ps

The response will look like:

bash
CONTAINER ID   IMAGE           COMMAND                  CREATED          STATUS          PORTS      NAMES
+<container-id>   celestia-node   "/entrypoint.sh cele…"   22 seconds ago   Up 21 seconds   2121/tcp   docker-compose-test-celestia-1
CONTAINER ID   IMAGE           COMMAND                  CREATED          STATUS          PORTS      NAMES
+<container-id>   celestia-node   "/entrypoint.sh cele…"   22 seconds ago   Up 21 seconds   2121/tcp   docker-compose-test-celestia-1

Interact with the container by replacing <container-id> for the container ID:

bash
docker exec -ti <container-id> /bin/bash
docker exec -ti <container-id> /bin/bash

Now, interact with cel-key to check your address matches the address you expect with the key you mounted:

bash
root@<container-id>:/# ./cel-key list --keyring-backend test --node.type light
+using directory:  ~/.celestia-light-mocha-4/keys
+- address: celestia1wkhyhr7ngf0ayqlpnsnxg4d72hfs5453dvunm9
+  name: my_celes_key
+  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"A1/NsoY0RGL7Hqt4VWLg441GQKJsZ2fBUnZXipgns8oV"}'
+  type: local
root@<container-id>:/# ./cel-key list --keyring-backend test --node.type light
+using directory:  ~/.celestia-light-mocha-4/keys
+- address: celestia1wkhyhr7ngf0ayqlpnsnxg4d72hfs5453dvunm9
+  name: my_celes_key
+  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"A1/NsoY0RGL7Hqt4VWLg441GQKJsZ2fBUnZXipgns8oV"}'
+  type: local
`,24),S=JSON.parse('{"title":"Create a wallet with celestia-node","description":"Use the cel-key utility to generate a wallet on celestia-node.","frontmatter":{"description":"Use the cel-key utility to generate a wallet on celestia-node.","prev":{"text":"Rollkit","link":"/developers/rollkit"},"head":[["meta",{"name":"og:title","content":"Create a wallet with celestia-node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/celestia-node-key.md","filePath":"developers/celestia-node-key.md","lastUpdated":1707231883000}'),f={name:"developers/celestia-node-key.md"},P=Object.assign(f,{setup(w){return(A,D)=>(p(),t("div",null,[c,s("div",r,[y,i,s("pre",E,[s("code",null,[d,n(` +`),s("span",F,[s("span",h,"ghcr.io/celestiaorg/celestia-node:"+l(e(o)["node-latest-tag"])+" ",1),g]),n(` +`),k])]),s("pre",u,[s("code",null,[C,n(` +`),s("span",b,[s("span",v,"ghcr.io/celestiaorg/celestia-node:"+l(e(o)["node-latest-tag"])+" ",1),m]),n(` +`),B])])]),_]))}});export{S as __pageData,P as default}; diff --git a/assets/developers_celestia-node-key.md.db3dee61.lean.js b/assets/developers_celestia-node-key.md.db3dee61.lean.js new file mode 100644 index 00000000000..4941e152733 --- /dev/null +++ b/assets/developers_celestia-node-key.md.db3dee61.lean.js @@ -0,0 +1,5 @@ +import{m as o}from"./chunks/mocha_versions.3ba4a823.js";import{o as p,c as t,k as s,a as n,t as l,l as e,Q as a}from"./chunks/framework.51d6c45b.js";const c=a("",37),r={class:"language-bash vp-adaptive-theme"},y=s("button",{title:"Copy Code",class:"copy"},null,-1),i=s("span",{class:"lang"},"bash",-1),E={class:"shiki github-dark vp-code-dark"},d=a("",1),F={class:"line"},h={style:{color:"#E1E4E8"}},g=a("",7),k=a("",1),u={class:"shiki github-light vp-code-light"},C=a("",1),b={class:"line"},v={style:{color:"#24292E"}},m=a("",7),B=a("",1),_=a("",24),S=JSON.parse('{"title":"Create a wallet with celestia-node","description":"Use the cel-key utility to generate a wallet on celestia-node.","frontmatter":{"description":"Use the cel-key utility to generate a wallet on celestia-node.","prev":{"text":"Rollkit","link":"/developers/rollkit"},"head":[["meta",{"name":"og:title","content":"Create a wallet with celestia-node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/celestia-node-key.md","filePath":"developers/celestia-node-key.md","lastUpdated":1707231883000}'),f={name:"developers/celestia-node-key.md"},P=Object.assign(f,{setup(w){return(A,D)=>(p(),t("div",null,[c,s("div",r,[y,i,s("pre",E,[s("code",null,[d,n(` +`),s("span",F,[s("span",h,"ghcr.io/celestiaorg/celestia-node:"+l(e(o)["node-latest-tag"])+" ",1),g]),n(` +`),k])]),s("pre",u,[s("code",null,[C,n(` +`),s("span",b,[s("span",v,"ghcr.io/celestiaorg/celestia-node:"+l(e(o)["node-latest-tag"])+" ",1),m]),n(` +`),B])])]),_]))}});export{S as __pageData,P as default}; diff --git a/assets/developers_deploy-on-bubs.md.c2d7ac48.js b/assets/developers_deploy-on-bubs.md.c2d7ac48.js new file mode 100644 index 00000000000..7d55db59ade --- /dev/null +++ b/assets/developers_deploy-on-bubs.md.c2d7ac48.js @@ -0,0 +1,137 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.51d6c45b.js";const F=JSON.parse('{"title":"Deploy a smart contract on Bubs testnet","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Deploy a smart contract on Bubs testnet | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/deploy-on-bubs.md","filePath":"developers/deploy-on-bubs.md","lastUpdated":1711388443000}'),p={name:"developers/deploy-on-bubs.md"},o=l(`

Deploy a smart contract on Bubs testnet

In this tutorial, we will deploy a smart contract to the Bubs testnet.

Dependencies

Setup

First, in your $HOME directory, set up a new project folder for this tutorial and init the project with npm:

bash
cd $HOME
+mkdir counter-project && cd counter-project && npm init -y
cd $HOME
+mkdir counter-project && cd counter-project && npm init -y

Next, initialize a Foundry project with the following command:

bash
forge init counter_contract
forge init counter_contract

Create your smart contract

Take a look at the Counter.sol file in your counter-project/counter_contract/src directory:

solidity
// SPDX-License-Identifier: UNLICENSED
+pragma solidity ^0.8.13;
+
+contract Counter {
+    uint256 public number;
+
+    function setNumber(uint256 newNumber) public {
+        number = newNumber;
+    }
+
+    function increment() public {
+        number++;
+    }
+}
// SPDX-License-Identifier: UNLICENSED
+pragma solidity ^0.8.13;
+
+contract Counter {
+    uint256 public number;
+
+    function setNumber(uint256 newNumber) public {
+        number = newNumber;
+    }
+
+    function increment() public {
+        number++;
+    }
+}

The contract contains a public unsigned integer variable named "number". There are two public functions in this contract. The setNumber function allows anyone to set a new value for the "number" variable, while the increment function increases the value of "number" by one each time it's called.

You can learn more about Solidity and smart contract programming.

To compile the contract, run the following forge command from the $HOME/counter-project/counter_contract/ directory:

bash
forge build
forge build

Your output should look similar to the following:

bash
[⠢] Compiling...
+[⠔] Compiling 21 files with 0.8.19
+[⠑] Solc 0.8.19 finished in 1.24s
+Compiler run successful
[⠢] Compiling...
+[⠔] Compiling 21 files with 0.8.19
+[⠑] Solc 0.8.19 finished in 1.24s
+Compiler run successful

Test your smart contract

Now, open the test/Counter.t.sol file:

solidity
// SPDX-License-Identifier: UNLICENSED
+pragma solidity ^0.8.13;
+
+import "forge-std/Test.sol";
+import "../src/Counter.sol";
+
+contract CounterTest is Test {
+    Counter public counter;
+
+    function setUp() public {
+        counter = new Counter();
+        counter.setNumber(0);
+    }
+
+    function testIncrement() public {
+        counter.increment();
+        assertEq(counter.number(), 1);
+    }
+
+    function testSetNumber(uint256 x) public {
+        counter.setNumber(x);
+        assertEq(counter.number(), x);
+    }
+}
// SPDX-License-Identifier: UNLICENSED
+pragma solidity ^0.8.13;
+
+import "forge-std/Test.sol";
+import "../src/Counter.sol";
+
+contract CounterTest is Test {
+    Counter public counter;
+
+    function setUp() public {
+        counter = new Counter();
+        counter.setNumber(0);
+    }
+
+    function testIncrement() public {
+        counter.increment();
+        assertEq(counter.number(), 1);
+    }
+
+    function testSetNumber(uint256 x) public {
+        counter.setNumber(x);
+        assertEq(counter.number(), x);
+    }
+}

This file performs unit testing on the contract we created in the previous section. Here's what the test is doing:

  • The contract includes a public "Counter" type variable called "counter". In the setUp function, it initializes a new instance of the "Counter" contract and sets the "number" variable to 0.

  • There are two test functions in the contract: testIncrement and testSetNumber.

  • The testIncrement function tests the "increment" function of the "Counter" contract by calling it and then asserting that the "number" in the "Counter" contract is 1. It verifies if the increment operation correctly increases the number by one.

  • The testSetNumber function is more generic. It takes an unsigned integer argument 'x' and tests the "setNumber" function of the "Counter" contract. After calling the "setNumber" function with 'x', it asserts that the "number" in the "Counter" contract is equal to 'x'. This verifies that the "setNumber" function correctly updates the "number" in the "Counter" contract.

Now, to test your code, run the following:

bash
forge test
forge test

If the test is successful, your output should be similar to this:

bash
[⠆] Compiling...
+No files changed, compilation skipped
+
+Running 2 tests for test/Counter.t.sol:CounterTest
+[PASS] testIncrement() (gas: 28334)
+[PASS] testSetNumber(uint256) (runs: 256, μ: 27709, ~: 28409)
+Test result: ok. 2 passed; 0 failed; finished in 8.96ms
[⠆] Compiling...
+No files changed, compilation skipped
+
+Running 2 tests for test/Counter.t.sol:CounterTest
+[PASS] testIncrement() (gas: 28334)
+[PASS] testSetNumber(uint256) (runs: 256, μ: 27709, ~: 28409)
+Test result: ok. 2 passed; 0 failed; finished in 8.96ms

Deploying your smart contract

Using Anvil

First, we'll test out our contract on a local devnet called "anvil". To start the local server, run:

bash
anvil
anvil

You'll see a local RPC endpoint (127.0.0.1:8545) and accounts to test with.

Let's deploy the contract now. First, set a private key from anvil:

bash
export PRIVATE_KEY=0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80
+export ANVIL_RPC_URL=http://localhost:8545
export PRIVATE_KEY=0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80
+export ANVIL_RPC_URL=http://localhost:8545

Now, deploy the contract:

bash
forge create --rpc-url $ANVIL_RPC_URL \\
+--private-key $PRIVATE_KEY \\
+src/Counter.sol:Counter
forge create --rpc-url $ANVIL_RPC_URL \\
+--private-key $PRIVATE_KEY \\
+src/Counter.sol:Counter

Using Bubs

First, set a private key from your funded Ethereum wallet and set the BUBS_RPC_URL variable with an RPC of your choosing:

bash
export BUBS_PRIVATE_KEY=0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80
+export BUBS_RPC_URL=https://bubs-sepolia.rpc.caldera.xyz/http
export BUBS_PRIVATE_KEY=0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80
+export BUBS_RPC_URL=https://bubs-sepolia.rpc.caldera.xyz/http

Now that we're ready to deploy the smart contract onto Bubs, we will run the forge create command.

bash
forge create --rpc-url $BUBS_RPC_URL \\
+--private-key $BUBS_PRIVATE_KEY \\
+src/Counter.sol:Counter
forge create --rpc-url $BUBS_RPC_URL \\
+--private-key $BUBS_PRIVATE_KEY \\
+src/Counter.sol:Counter

A successful deployment will return output similar to below:

bash
[⠆] Compiling...
+No files changed, compilation skipped
+Deployer: 0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266
+Deployed to: 0x5FbDB2315678afecb367f032d93F642f64180aa3
+Transaction hash: 0xf1a793a793cd9fc588f5132d99008565ea361eb3535d66499575e9e1908200b2
[⠆] Compiling...
+No files changed, compilation skipped
+Deployer: 0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266
+Deployed to: 0x5FbDB2315678afecb367f032d93F642f64180aa3
+Transaction hash: 0xf1a793a793cd9fc588f5132d99008565ea361eb3535d66499575e9e1908200b2

Once you've deployed the contract, you're ready to interact with it!

First, we'll set it as a variable:

bash
export CONTRACT_ADDRESS=0x5FbDB2315678afecb367f032d93F642f64180aa3
export CONTRACT_ADDRESS=0x5FbDB2315678afecb367f032d93F642f64180aa3

Interacting with your smart contract

Foundry uses cast, a CLI for performing Ethereum RPC calls.

To write to the contract, we'll use the cast send command:

bash
cast send $CONTRACT_ADDRESS "setNumber(uint256)" 10 --rpc-url $BUBS_RPC_URL --private-key $BUBS_PRIVATE_KEY
cast send $CONTRACT_ADDRESS "setNumber(uint256)" 10 --rpc-url $BUBS_RPC_URL --private-key $BUBS_PRIVATE_KEY

Your output will look similar:

bash
blockHash               0x131822bef6eb59656d7e1387c19b75be667e587006710365ec5cf58030786c42
+blockNumber             3
+contractAddress
+cumulativeGasUsed       43494
+effectiveGasPrice       3767182372
+gasUsed                 43494
+logs                    []
+logsBloom               0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+root
+status                  1
+transactionHash         0x8f15d6004598f0662dd673a9898dceef77be8cc28408cecc284b28d7be32307d
+transactionIndex        0
+type                    2
blockHash               0x131822bef6eb59656d7e1387c19b75be667e587006710365ec5cf58030786c42
+blockNumber             3
+contractAddress
+cumulativeGasUsed       43494
+effectiveGasPrice       3767182372
+gasUsed                 43494
+logs                    []
+logsBloom               0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+root
+status                  1
+transactionHash         0x8f15d6004598f0662dd673a9898dceef77be8cc28408cecc284b28d7be32307d
+transactionIndex        0
+type                    2

Now, we can make a read call to view the state of the number variable, using the cast call command:

bash
cast call $CONTRACT_ADDRESS "number()" --rpc-url $BUBS_RPC_URL
cast call $CONTRACT_ADDRESS "number()" --rpc-url $BUBS_RPC_URL

The result will look similar:

bash
0x000000000000000000000000000000000000000000000000000000000000000a
0x000000000000000000000000000000000000000000000000000000000000000a

Convert the result from hexadecimal to a base 10 value with:

bash
echo $((0x000000000000000000000000000000000000000000000000000000000000000a))
echo $((0x000000000000000000000000000000000000000000000000000000000000000a))

Next steps

Congratulations! You've learned how to deploy a smart contract to Bubs testnet.

What will you build next? Now, you're ready to check out the GM Portal tutorial.

`,61),e=[o];function t(c,r,i,y,E,d){return a(),n("div",null,e)}const h=s(p,[["render",t]]);export{F as __pageData,h as default}; diff --git a/assets/developers_deploy-on-bubs.md.c2d7ac48.lean.js b/assets/developers_deploy-on-bubs.md.c2d7ac48.lean.js new file mode 100644 index 00000000000..790d535d0fa --- /dev/null +++ b/assets/developers_deploy-on-bubs.md.c2d7ac48.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.51d6c45b.js";const F=JSON.parse('{"title":"Deploy a smart contract on Bubs testnet","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Deploy a smart contract on Bubs testnet | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/deploy-on-bubs.md","filePath":"developers/deploy-on-bubs.md","lastUpdated":1711388443000}'),p={name:"developers/deploy-on-bubs.md"},o=l("",61),e=[o];function t(c,r,i,y,E,d){return a(),n("div",null,e)}const h=s(p,[["render",t]]);export{F as __pageData,h as default}; diff --git a/assets/developers_ethereum-fallback.md.88698cb8.js b/assets/developers_ethereum-fallback.md.88698cb8.js new file mode 100644 index 00000000000..899c124ee6b --- /dev/null +++ b/assets/developers_ethereum-fallback.md.88698cb8.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as r}from"./chunks/framework.51d6c45b.js";const l="/img/Celestia_ethereum-fallback.jpg",b=JSON.parse('{"title":"Ethereum fallback","description":"The DA fallback mechanism to Ethereum for rollups.","frontmatter":{"description":"The DA fallback mechanism to Ethereum for rollups.","head":[["meta",{"name":"og:title","content":"Ethereum fallback | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/ethereum-fallback.md","filePath":"developers/ethereum-fallback.md","lastUpdated":1702493795000}'),n={name:"developers/ethereum-fallback.md"},o=r('

Ethereum fallback

Ethereum fallback is a fallback mechanism that enables Ethereum L2s (or L3s) to “fall back” to using Ethereum calldata for data availability in the event of downtime on Celestia Mainnet Beta. This feature is currently supported by Celestia integrations with:

In the case of Celestia downtime or temporary unavailability, L2s can fallback to posting transactions as calldata on Ethereum or another DA layer for data availability instead of posting to Celestia. This mechanism ensures users can continue to transact securely and seamlessly, preventing disruptions and helping to ensure user funds do not get stuck in the L2's bridge on Ethereum.

Ethereum fallback is triggered whenever the sequencer has an error sending the PayForBlobs transaction on Celestia. Fallback can be triggered due to a congested mempool or nonce error and can be simulated with an error such as low balance or incorrect sequence. Fallback can also be triggered in the event Blobstream stops relaying attestations.

Ethereum fallback

',6),s=[o];function i(c,m,h,u,d,p){return a(),t("div",null,s)}const _=e(n,[["render",i]]);export{b as __pageData,_ as default}; diff --git a/assets/developers_ethereum-fallback.md.88698cb8.lean.js b/assets/developers_ethereum-fallback.md.88698cb8.lean.js new file mode 100644 index 00000000000..f7fe685f880 --- /dev/null +++ b/assets/developers_ethereum-fallback.md.88698cb8.lean.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as r}from"./chunks/framework.51d6c45b.js";const l="/img/Celestia_ethereum-fallback.jpg",b=JSON.parse('{"title":"Ethereum fallback","description":"The DA fallback mechanism to Ethereum for rollups.","frontmatter":{"description":"The DA fallback mechanism to Ethereum for rollups.","head":[["meta",{"name":"og:title","content":"Ethereum fallback | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/ethereum-fallback.md","filePath":"developers/ethereum-fallback.md","lastUpdated":1702493795000}'),n={name:"developers/ethereum-fallback.md"},o=r("",6),s=[o];function i(c,m,h,u,d,p){return a(),t("div",null,s)}const _=e(n,[["render",i]]);export{b as __pageData,_ as default}; diff --git a/assets/developers_full-stack-modular-development-guide.md.9f203595.js b/assets/developers_full-stack-modular-development-guide.md.9f203595.js new file mode 100644 index 00000000000..759be2a8c64 --- /dev/null +++ b/assets/developers_full-stack-modular-development-guide.md.9f203595.js @@ -0,0 +1,893 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.51d6c45b.js";const u=JSON.parse('{"title":"Full stack modular blockchain development guide","description":"Learn to build a full stack modular dapp.","frontmatter":{"description":"Learn to build a full stack modular dapp.","head":[["meta",{"name":"og:title","content":"Full stack modular blockchain development guide | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/full-stack-modular-development-guide.md","filePath":"developers/full-stack-modular-development-guide.md","lastUpdated":1712870672000}'),p={name:"developers/full-stack-modular-development-guide.md"},o=l(`

Full stack modular blockchain development guide

Note

This tutorial needs to be updated

This guide will introduce you to modular blockchains like Celestia, explain their benefits, and show you how to build a full stack modular dapp with React, Vite, RainbowKit, Celestia, and Foundry.

Current blockchain architectures are not scalable and face challenges around accessibility. In order for blockchains and web3 to reach mass adoption, these challenges must be addressed.

Blockchains have evolved over time from application-specific networks like Bitcoin to shared smart contract platforms like Ethereum. This guide will cover how to build dapps on these newer, shared platforms.

If you're interested in learning more about modular blockchains, or are new to the Celestia ecosystem, we recommend you read the build whatever page first.

Getting started

Now that you’ve had an overview of what Celestia is, let’s start building!

The execution environment that we’ll be leveraging today is Ethermint, an EVM-compatible testnet that you will run locally for this tutorial.

Pre-requisites

Project setup

To get started, create a new Foundry project:

bash
forge init celestia-dapp
+cd celestia-dapp
forge init celestia-dapp
+cd celestia-dapp

Foundry has created an example smart contract located at src/Contract.sol.

Updating the contract and tests

Let's update the contracts to include a basic blog example. Create a new file in the src directory named Contract.sol with the following code:

solidity
// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+contract Blog {
+    string public name;
+    address public owner;
+
+    uint private _postId;
+
+    struct Post {
+      uint id;
+      string title;
+      string content;
+      bool published;
+    }
+    /* mappings can be seen as hash tables */
+    /* here we create lookups for posts by id and posts by ipfs hash */
+    mapping(uint => Post) private idToPost;
+    mapping(string => Post) private hashToPost;
+
+    /* events facilitate communication between smart contracts and their user interfaces  */
+    /* i.e. we can create listeners for events in the client and also use them in The Graph  */
+    event PostCreated(uint id, string title, string hash);
+    event PostUpdated(uint id, string title, string hash, bool published);
+
+    /* when the blog is deployed, give it a name */
+    /* also set the creator as the owner of the contract */
+    constructor(string memory _name) {
+        name = _name;
+        owner = msg.sender;
+    }
+
+    /* updates the blog name */
+    function updateName(string memory _name) public {
+        name = _name;
+    }
+
+    /* transfers ownership of the contract to another address */
+    function transferOwnership(address newOwner) public onlyOwner {
+        owner = newOwner;
+    }
+
+    /* fetches an individual post by the content hash */
+    function fetchPost(string memory hash) public view returns(Post memory){
+      return hashToPost[hash];
+    }
+
+    /* creates a new post */
+    function createPost(string memory title, string memory hash) public onlyOwner {
+        _postId = _postId + 1;
+        Post storage post = idToPost[_postId];
+        post.id = _postId;
+        post.title = title;
+        post.published = true;
+        post.content = hash;
+        hashToPost[hash] = post;
+        emit PostCreated(_postId, title, hash);
+    }
+
+    /* updates an existing post */
+    function updatePost(uint postId, string memory title, string memory hash, bool published) public onlyOwner {
+        Post storage post =  idToPost[postId];
+        post.title = title;
+        post.published = published;
+        post.content = hash;
+        idToPost[postId] = post;
+        hashToPost[hash] = post;
+        emit PostUpdated(post.id, title, hash, published);
+    }
+
+    /* fetches all posts */
+    function fetchPosts() public view returns (Post[] memory) {
+        uint itemCount = _postId;
+
+        Post[] memory posts = new Post[](itemCount);
+        for (uint i = 0; i < itemCount; i++) {
+            uint currentId = i + 1;
+            Post storage currentItem = idToPost[currentId];
+            posts[i] = currentItem;
+        }
+        return posts;
+    }
+
+    /* this modifier means only the contract owner can */
+    /* invoke the function */
+    modifier onlyOwner() {
+      require(msg.sender == owner);
+    _;
+  }
+}
// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+contract Blog {
+    string public name;
+    address public owner;
+
+    uint private _postId;
+
+    struct Post {
+      uint id;
+      string title;
+      string content;
+      bool published;
+    }
+    /* mappings can be seen as hash tables */
+    /* here we create lookups for posts by id and posts by ipfs hash */
+    mapping(uint => Post) private idToPost;
+    mapping(string => Post) private hashToPost;
+
+    /* events facilitate communication between smart contracts and their user interfaces  */
+    /* i.e. we can create listeners for events in the client and also use them in The Graph  */
+    event PostCreated(uint id, string title, string hash);
+    event PostUpdated(uint id, string title, string hash, bool published);
+
+    /* when the blog is deployed, give it a name */
+    /* also set the creator as the owner of the contract */
+    constructor(string memory _name) {
+        name = _name;
+        owner = msg.sender;
+    }
+
+    /* updates the blog name */
+    function updateName(string memory _name) public {
+        name = _name;
+    }
+
+    /* transfers ownership of the contract to another address */
+    function transferOwnership(address newOwner) public onlyOwner {
+        owner = newOwner;
+    }
+
+    /* fetches an individual post by the content hash */
+    function fetchPost(string memory hash) public view returns(Post memory){
+      return hashToPost[hash];
+    }
+
+    /* creates a new post */
+    function createPost(string memory title, string memory hash) public onlyOwner {
+        _postId = _postId + 1;
+        Post storage post = idToPost[_postId];
+        post.id = _postId;
+        post.title = title;
+        post.published = true;
+        post.content = hash;
+        hashToPost[hash] = post;
+        emit PostCreated(_postId, title, hash);
+    }
+
+    /* updates an existing post */
+    function updatePost(uint postId, string memory title, string memory hash, bool published) public onlyOwner {
+        Post storage post =  idToPost[postId];
+        post.title = title;
+        post.published = published;
+        post.content = hash;
+        idToPost[postId] = post;
+        hashToPost[hash] = post;
+        emit PostUpdated(post.id, title, hash, published);
+    }
+
+    /* fetches all posts */
+    function fetchPosts() public view returns (Post[] memory) {
+        uint itemCount = _postId;
+
+        Post[] memory posts = new Post[](itemCount);
+        for (uint i = 0; i < itemCount; i++) {
+            uint currentId = i + 1;
+            Post storage currentItem = idToPost[currentId];
+            posts[i] = currentItem;
+        }
+        return posts;
+    }
+
+    /* this modifier means only the contract owner can */
+    /* invoke the function */
+    modifier onlyOwner() {
+      require(msg.sender == owner);
+    _;
+  }
+}

Next, let's create a test for this contract.

Open test/Contract.t.sol and update the code with the following:

solidity
// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+import "forge-std/Test.sol";
+import "src/Contract.sol";
+
+contract ContractTest is Test {
+    Blog blog;
+
+    function setUp() public {
+        blog = new Blog("Celestia Blog");
+    }
+
+    function testCreatePost() public {
+        blog.createPost("My first post", "12345");
+        Blog.Post[] memory posts = blog.fetchPosts();
+        assertEq(posts.length, 1);
+    }
+
+    function testUpdatePost() public {
+        blog.createPost("My first post", "12345");
+        blog.updatePost(1, "My second post", "12345", true);
+        Blog.Post memory updatedPost = blog.fetchPost("12345");
+        assertEq(updatedPost.title, "My second post");
+    }
+
+    function testFetchPosts() public {
+        Blog.Post[] memory posts = blog.fetchPosts();
+        assertEq(posts.length, 0);
+        blog.createPost("My first post", "12345");
+        posts = blog.fetchPosts();
+        assertEq(posts.length, 1);
+    }
+
+    function testOnlyOwner() public {
+        blog.createPost("My first post", "12345");
+        address bob = address(0x1);
+        vm.startPrank(bob);
+        vm.expectRevert();
+        blog.updatePost(1, "My second post", "12345", true);
+    }
+}
// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+import "forge-std/Test.sol";
+import "src/Contract.sol";
+
+contract ContractTest is Test {
+    Blog blog;
+
+    function setUp() public {
+        blog = new Blog("Celestia Blog");
+    }
+
+    function testCreatePost() public {
+        blog.createPost("My first post", "12345");
+        Blog.Post[] memory posts = blog.fetchPosts();
+        assertEq(posts.length, 1);
+    }
+
+    function testUpdatePost() public {
+        blog.createPost("My first post", "12345");
+        blog.updatePost(1, "My second post", "12345", true);
+        Blog.Post memory updatedPost = blog.fetchPost("12345");
+        assertEq(updatedPost.title, "My second post");
+    }
+
+    function testFetchPosts() public {
+        Blog.Post[] memory posts = blog.fetchPosts();
+        assertEq(posts.length, 0);
+        blog.createPost("My first post", "12345");
+        posts = blog.fetchPosts();
+        assertEq(posts.length, 1);
+    }
+
+    function testOnlyOwner() public {
+        blog.createPost("My first post", "12345");
+        address bob = address(0x1);
+        vm.startPrank(bob);
+        vm.expectRevert();
+        blog.updatePost(1, "My second post", "12345", true);
+    }
+}

Foundry uses Dappsys Test to provide basic logging and assertion functionality. It's included in the Forge Standard Library.

Here, we are using assertEq to assert equality. You can view all of the assertion functions available.

Running the test

We can now run our tests to make sure our contract is working properly:

bash
forge test -vv
forge test -vv

Updating the deployment script

Now that we've tested the contract, let's try deploying it locally using Solidity Scripting.

To do so, update the deployment script at script/Contract.s.sol with the following code:

solidity
// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+import "forge-std/Script.sol";
+
+import {Blog} from "src/Contract.sol";
+
+contract ContractScript is Script {
+    function setUp() public {}
+
+    function run() public {
+        vm.startBroadcast();
+        new Blog("Celestia Blog");
+        vm.stopBroadcast();
+    }
+}
// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.13;
+
+import "forge-std/Script.sol";
+
+import {Blog} from "src/Contract.sol";
+
+contract ContractScript is Script {
+    function setUp() public {}
+
+    function run() public {
+        vm.startBroadcast();
+        new Blog("Celestia Blog");
+        vm.stopBroadcast();
+    }
+}

Now we can use this script to deploy our smart contract to either a live or test network.

Deploying locally

Next start Anvil, the local testnet:

bash
anvil --port 9545
anvil --port 9545

caution

We need to use port 9545, because Ethermint will use 8545.

Once started, Anvil will give you a local RPC endpoint as well as a handful of Private Keys and Accounts that you can use.

We can now use the local RPC along with one of the private keys to deploy locally:

bash
forge script script/Contract.s.sol:ContractScript --fork-url \\
+http://localhost:9545 --private-key $PRIVATE_KEY --broadcast
forge script script/Contract.s.sol:ContractScript --fork-url \\
+http://localhost:9545 --private-key $PRIVATE_KEY --broadcast

Once the contract has been deployed locally, Anvil will log out the contract address.

Take a note of this local contract address as we’ll be using it later in the frontend application.

Next, set the contract address as an environment variable:

bash
export CONTRACT_ADDRESS=<contract-address>
export CONTRACT_ADDRESS=<contract-address>

We can then test sending transactions to it with cast send.

bash
cast send $CONTRACT_ADDRESS \\
+"createPost(string,string)" "my first post" "12345" \\
+--private-key $PRIVATE_KEY
cast send $CONTRACT_ADDRESS \\
+"createPost(string,string)" "my first post" "12345" \\
+--private-key $PRIVATE_KEY

We can then perform read operations with cast call:

bash
cast call $CONTRACT_ADDRESS "fetchPosts()"
cast call $CONTRACT_ADDRESS "fetchPosts()"

Once the contract is deployed successfully, take a note of the contract address as we’ll also be needing it in just a moment when we test the live contract.

Deploying to the Ethermint Sovereign Rollup

First, we will need to follow the setup from the EVM tutorial.

Pre-requisites

It is required that you complete dependency setup, Rollkit installation, and Instantiating and EVM rollup from the EVM tutorial to complete the remainder of the tutorial.

Now that we've deployed and tested locally, we can deploy to our Ethermint chain.

First, we will need to export the private key generated by the ethermint init.sh script:

bash
PRIVATE_KEY=$(ethermintd keys unsafe-export-eth-key mykey --keyring-backend test)
PRIVATE_KEY=$(ethermintd keys unsafe-export-eth-key mykey --keyring-backend test)

NOTE

Here, the key name from init.sh is mykey but you can modify the init.sh to change the name of your key.

Now, we can start deploying the smart contract to our Ethermint chain.

To do so, run the following script in the celestia-dapp directory:

bash
forge script script/Contract.s.sol:ContractScript \\
+--rpc-url http://localhost:8545 --private-key $PRIVATE_KEY --broadcast
forge script script/Contract.s.sol:ContractScript \\
+--rpc-url http://localhost:8545 --private-key $PRIVATE_KEY --broadcast

Set the contract address in the output as the CONTRACT_ADDRESS variable:

bash
export CONTRACT_ADDRESS=<new-contract-address>
export CONTRACT_ADDRESS=<new-contract-address>

Once the contract has been deployed to the Ethermint rollup, we can use cast send to test sending transactions to it:

bash
cast send $CONTRACT_ADDRESS \\
+"createPost(string,string)" "my first post" "12345" \\
+--rpc-url http://localhost:8545 --private-key $PRIVATE_KEY
cast send $CONTRACT_ADDRESS \\
+"createPost(string,string)" "my first post" "12345" \\
+--rpc-url http://localhost:8545 --private-key $PRIVATE_KEY

We can then perform read operations with cast call:

bash
cast call $CONTRACT_ADDRESS "fetchPosts()" --rpc-url http://localhost:8545
cast call $CONTRACT_ADDRESS "fetchPosts()" --rpc-url http://localhost:8545

NOTE

You will want to redeploy the contract for your frontend, because the post is not uploaded to IPFS in the CLI.

Building the frontend

For the frontend project, we’ll be using the following libraries and frameworks:

React - JavaScript library for building user interfaces

Vite - Project generator / rapid development tool for modern web projects

Rainbowkit - Easy and beautiful library to connect a wallet

WAGMI - 20+ hooks for working with wallets, ENS, contracts, transactions, signing, etc

In the root of the Foundry project, create a new React.js application using Vite:

jsx
yarn create vite
+
+? Project name: › frontend
+? Select a framework › React
+? Select a variant > JavaScript
yarn create vite
+
+? Project name: › frontend
+? Select a framework › React
+? Select a variant > JavaScript

Next, copy the ABI that was created by Foundry into the frontend directory so that we can have it later (or manually copy it into a file named Blog.json in the frontend directory):

bash
cp out/Contract.sol/Blog.json frontend/
cp out/Contract.sol/Blog.json frontend/

Now, change into the frontend directory and install the node_modules:

bash
cd frontend
+yarn
cd frontend
+yarn

Configuring environment variables

Next we need to configure the environment variables for the Infura project ID and secret.

First, create an Infura account and new project for IPFS.

Create a file named .env.local in the frontend/ directory and add the following configuration with your own credentials:

txt
VITE_INFURA_ID=your-project-api-key
+VITE_INFURA_SECRET=your-project-api-key-secret
VITE_INFURA_ID=your-project-api-key
+VITE_INFURA_SECRET=your-project-api-key-secret

Now that the project is created, let’s install the additional dependencies using either NPM, Yarn, or PNPM:

jsx
npm install @rainbow-me/rainbowkit@0.8.0 wagmi@0.8.10 ethers ipfs-http-client react-markdown
npm install @rainbow-me/rainbowkit@0.8.0 wagmi@0.8.10 ethers ipfs-http-client react-markdown

Configuring the entrypoint

Next we’ll update the entrypoint at src/main.jsx.

The main things we’re doing here have to do with the configuration of Rainbowkit so that we can have a nice way for the user to connect their wallet.

Rainbowkit also allows a customizable array of network providers, so we’re creating a new network configuration for Ethermint.

jsx
import "./polyfills";
+import ReactDOM from "react-dom/client";
+import App from "./App";
+import "./index.css";
+import "@rainbow-me/rainbowkit/styles.css";
+import { RainbowKitProvider } from "@rainbow-me/rainbowkit";
+import { chain, configureChains, createClient, WagmiConfig } from "wagmi";
+import { publicProvider } from "wagmi/providers/public";
+import { injectedWallet, metaMaskWallet } from "@rainbow-me/rainbowkit/wallets";
+import { connectorsForWallets } from "@rainbow-me/rainbowkit";
+
+/* create configuration for Ethermint testnet */
+const ethermint = {
+  id: 9000,
+  name: "Ethermint",
+  network: "ethermint",
+  nativeCurrency: {
+    decimals: 18,
+    name: "Ethermint",
+    symbol: "CTE",
+  },
+  rpcUrls: {
+    default: {
+      http: ["http://localhost:8545/"],
+    },
+  },
+  testnet: true,
+};
+
+// remove chain.localhost or ethermint depending on which you want to connect to
+const { chains, provider } = configureChains(
+  [chain.localhost, ethermint],
+  [publicProvider()],
+);
+
+const connectors = connectorsForWallets([
+  {
+    groupName: "Recommended",
+    wallets: [metaMaskWallet({ chains }), injectedWallet({ chains })],
+  },
+]);
+
+const wagmiClient = createClient({
+  autoConnect: true,
+  connectors,
+  provider,
+});
+
+const containerStyle = {
+  width: "900px",
+  margin: "0 auto",
+};
+
+ReactDOM.createRoot(document.getElementById("root")).render(
+  <WagmiConfig client={wagmiClient}>
+    <RainbowKitProvider chains={chains}>
+      <div style={containerStyle}>
+        <App />
+      </div>
+    </RainbowKitProvider>
+  </WagmiConfig>,
+);
import "./polyfills";
+import ReactDOM from "react-dom/client";
+import App from "./App";
+import "./index.css";
+import "@rainbow-me/rainbowkit/styles.css";
+import { RainbowKitProvider } from "@rainbow-me/rainbowkit";
+import { chain, configureChains, createClient, WagmiConfig } from "wagmi";
+import { publicProvider } from "wagmi/providers/public";
+import { injectedWallet, metaMaskWallet } from "@rainbow-me/rainbowkit/wallets";
+import { connectorsForWallets } from "@rainbow-me/rainbowkit";
+
+/* create configuration for Ethermint testnet */
+const ethermint = {
+  id: 9000,
+  name: "Ethermint",
+  network: "ethermint",
+  nativeCurrency: {
+    decimals: 18,
+    name: "Ethermint",
+    symbol: "CTE",
+  },
+  rpcUrls: {
+    default: {
+      http: ["http://localhost:8545/"],
+    },
+  },
+  testnet: true,
+};
+
+// remove chain.localhost or ethermint depending on which you want to connect to
+const { chains, provider } = configureChains(
+  [chain.localhost, ethermint],
+  [publicProvider()],
+);
+
+const connectors = connectorsForWallets([
+  {
+    groupName: "Recommended",
+    wallets: [metaMaskWallet({ chains }), injectedWallet({ chains })],
+  },
+]);
+
+const wagmiClient = createClient({
+  autoConnect: true,
+  connectors,
+  provider,
+});
+
+const containerStyle = {
+  width: "900px",
+  margin: "0 auto",
+};
+
+ReactDOM.createRoot(document.getElementById("root")).render(
+  <WagmiConfig client={wagmiClient}>
+    <RainbowKitProvider chains={chains}>
+      <div style={containerStyle}>
+        <App />
+      </div>
+    </RainbowKitProvider>
+  </WagmiConfig>,
+);

Creating and reading posts

Now that the base configuration is set up we’ll create a view that allows users to create and view posts.

We’ll be using IPFS to upload the content of the post, then anchoring the hash of the post on chain. When we retrieve the post, we can then read the value from IPFS to view the post.

Update App.jsx with the following code:

jsx
import { useState, useEffect } from "react";
+import { ConnectButton } from "@rainbow-me/rainbowkit";
+import { ethers } from "ethers";
+import { create } from "ipfs-http-client";
+import { Buffer } from "buffer";
+import Blog from "../Blog.json";
+import { useAccount } from "wagmi";
+
+/* configure authorization for Infura and IPFS */
+const auth =
+  "Basic " +
+  Buffer.from(
+    import.meta.env.VITE_INFURA_ID + ":" + import.meta.env.VITE_INFURA_SECRET,
+  ).toString("base64");
+
+/* create an IPFS client */
+const client = create({
+  host: "ipfs.infura.io",
+  port: 5001,
+  protocol: "https",
+  headers: {
+    authorization: auth,
+  },
+});
+
+const contractAddress = "your-ethermint-contract-address";
+
+function App() {
+  useEffect(() => {
+    fetchPosts();
+  }, []);
+  const [viewState, setViewState] = useState("view-posts");
+  const [posts, setPosts] = useState([]);
+  const [title, setTitle] = useState("");
+  const [content, setContent] = useState("");
+  const { address } = useAccount();
+
+  /* when the component loads, useEffect will call this function */
+  async function fetchPosts() {
+    const provider = new ethers.providers.Web3Provider(window.ethereum);
+    const contract = new ethers.Contract(contractAddress, Blog.abi, provider);
+    let data = await contract.fetchPosts();
+    /* once the data is returned from the network we map over it and */
+    /* transform the data into a more readable format  */
+    data = data.map((d) => ({
+      content: d["content"],
+      title: d["title"],
+      published: d["published"],
+      id: d["id"].toString(),
+    }));
+
+    /* we then fetch the post content from IPFS and add it to the post objects */
+    data = await Promise.all(
+      data.map(async (d) => {
+        const endpoint = \`https://infura-ipfs.io/ipfs/\${d.content}\`;
+        const options = {
+          mode: "no-cors",
+        };
+        const response = await fetch(endpoint, options);
+        const value = await response.text();
+        d.postContent = value;
+        return d;
+      }),
+    );
+
+    setPosts(data);
+  }
+
+  async function createPost() {
+    const added = await client.add(content);
+    const provider = new ethers.providers.Web3Provider(window.ethereum);
+    const signer = provider.getSigner();
+
+    const contract = new ethers.Contract(contractAddress, Blog.abi, signer);
+    const tx = await contract.createPost(title, added.path);
+    await tx.wait();
+    setViewState("view-posts");
+  }
+
+  function toggleView(value) {
+    setViewState(value);
+    if (value === "view-posts") {
+      fetchPosts();
+    }
+  }
+
+  return (
+    <div style={outerContainerStyle}>
+      <div style={innerContainerStyle}>
+        <h1>Modular Rollup Blog</h1>
+        <p>
+          This allows users to securely create and share blog posts on the
+          blockchain without the need for a centralized server or authority.
+        </p>
+        {!address ? (
+          <div>
+            <h3>Getting Started</h3>
+            <p>
+              First, you will need to connect your Ethereum wallet to Ethermint
+              to display the posts from the smart contract and make posts.
+            </p>
+          </div>
+        ) : null}
+        <br />
+        <h3 style={{ justifyContent: "right", textAlign: "right" }}>
+          Connect your Ethereum wallet to begin ✨
+        </h3>
+        <div style={buttonContainerStyle}>
+          <ConnectButton />
+        </div>
+        {address ? (
+          <div style={buttonContainerStyle}>
+            <button
+              onClick={() => toggleView("view-posts")}
+              style={buttonStyle}
+            >
+              View Posts
+            </button>
+            <button
+              onClick={() => toggleView("create-post")}
+              style={buttonStyle}
+            >
+              Create Post
+            </button>
+          </div>
+        ) : null}
+        {viewState === "view-posts" && address && (
+          <div>
+            <div style={postContainerStyle}>
+              <h1>Posts</h1>
+              {posts.map((post, index) => (
+                <div key={index}>
+                  <h2>{post.title}</h2>
+                  <button
+                    style={{ fontSize: "16px" }}
+                    onClick={() =>
+                      window.open(\`https://infura-ipfs.io/ipfs/\${post.content}\`)
+                    }
+                  >
+                    Read on IPFS
+                  </button>
+                  {/* <ReactMarkdown>
+                    {post.postContent}
+                  </ReactMarkdown> */}
+                  <p style={mbidStyle}>GMID: {post.id}</p>
+                </div>
+              ))}
+            </div>
+          </div>
+        )}
+        {viewState === "create-post" && (
+          <div style={formContainerStyle}>
+            <h2>Create Post</h2>
+            <input
+              placeholder="Title"
+              onChange={(e) => setTitle(e.target.value)}
+              style={inputStyle}
+            />
+            <textarea
+              placeholder="Content"
+              onChange={(e) => setContent(e.target.value)}
+              style={inputStyle}
+            />
+            <button onClick={createPost}>Create Post</button>
+          </div>
+        )}
+      </div>
+    </div>
+  );
+}
+
+const outerContainerStyle = {
+  width: "90vw",
+  height: "100vh",
+  padding: "50px 0px",
+};
+
+const innerContainerStyle = {
+  width: "100%",
+  maxWidth: "800px",
+  margin: "0 auto",
+};
+
+const formContainerStyle = {
+  display: "flex",
+  flexDirection: "column",
+  alignItems: "center",
+};
+
+const inputStyle = {
+  width: "400px",
+  marginBottom: "10px",
+  padding: "10px",
+  height: "40px",
+};
+
+const postContainerStyle = {
+  margin: "0 auto",
+  padding: "1em",
+  width: "90%",
+  maxWidth: "800px",
+  display: "flex",
+  flexDirection: "column",
+  alignItems: "start",
+  justifyContent: "center",
+};
+
+const mbidStyle = {
+  fontSize: "10px",
+  textAlign: "start",
+};
+
+const buttonStyle = {
+  marginTop: 15,
+  marginRight: 5,
+  border: "1px solid rgba(255, 255, 255, .2)",
+};
+
+const buttonContainerStyle = {
+  marginTop: 15,
+  marginRight: 5,
+  display: "flex",
+  justifyContent: "right",
+};
+
+export default App;
import { useState, useEffect } from "react";
+import { ConnectButton } from "@rainbow-me/rainbowkit";
+import { ethers } from "ethers";
+import { create } from "ipfs-http-client";
+import { Buffer } from "buffer";
+import Blog from "../Blog.json";
+import { useAccount } from "wagmi";
+
+/* configure authorization for Infura and IPFS */
+const auth =
+  "Basic " +
+  Buffer.from(
+    import.meta.env.VITE_INFURA_ID + ":" + import.meta.env.VITE_INFURA_SECRET,
+  ).toString("base64");
+
+/* create an IPFS client */
+const client = create({
+  host: "ipfs.infura.io",
+  port: 5001,
+  protocol: "https",
+  headers: {
+    authorization: auth,
+  },
+});
+
+const contractAddress = "your-ethermint-contract-address";
+
+function App() {
+  useEffect(() => {
+    fetchPosts();
+  }, []);
+  const [viewState, setViewState] = useState("view-posts");
+  const [posts, setPosts] = useState([]);
+  const [title, setTitle] = useState("");
+  const [content, setContent] = useState("");
+  const { address } = useAccount();
+
+  /* when the component loads, useEffect will call this function */
+  async function fetchPosts() {
+    const provider = new ethers.providers.Web3Provider(window.ethereum);
+    const contract = new ethers.Contract(contractAddress, Blog.abi, provider);
+    let data = await contract.fetchPosts();
+    /* once the data is returned from the network we map over it and */
+    /* transform the data into a more readable format  */
+    data = data.map((d) => ({
+      content: d["content"],
+      title: d["title"],
+      published: d["published"],
+      id: d["id"].toString(),
+    }));
+
+    /* we then fetch the post content from IPFS and add it to the post objects */
+    data = await Promise.all(
+      data.map(async (d) => {
+        const endpoint = \`https://infura-ipfs.io/ipfs/\${d.content}\`;
+        const options = {
+          mode: "no-cors",
+        };
+        const response = await fetch(endpoint, options);
+        const value = await response.text();
+        d.postContent = value;
+        return d;
+      }),
+    );
+
+    setPosts(data);
+  }
+
+  async function createPost() {
+    const added = await client.add(content);
+    const provider = new ethers.providers.Web3Provider(window.ethereum);
+    const signer = provider.getSigner();
+
+    const contract = new ethers.Contract(contractAddress, Blog.abi, signer);
+    const tx = await contract.createPost(title, added.path);
+    await tx.wait();
+    setViewState("view-posts");
+  }
+
+  function toggleView(value) {
+    setViewState(value);
+    if (value === "view-posts") {
+      fetchPosts();
+    }
+  }
+
+  return (
+    <div style={outerContainerStyle}>
+      <div style={innerContainerStyle}>
+        <h1>Modular Rollup Blog</h1>
+        <p>
+          This allows users to securely create and share blog posts on the
+          blockchain without the need for a centralized server or authority.
+        </p>
+        {!address ? (
+          <div>
+            <h3>Getting Started</h3>
+            <p>
+              First, you will need to connect your Ethereum wallet to Ethermint
+              to display the posts from the smart contract and make posts.
+            </p>
+          </div>
+        ) : null}
+        <br />
+        <h3 style={{ justifyContent: "right", textAlign: "right" }}>
+          Connect your Ethereum wallet to begin ✨
+        </h3>
+        <div style={buttonContainerStyle}>
+          <ConnectButton />
+        </div>
+        {address ? (
+          <div style={buttonContainerStyle}>
+            <button
+              onClick={() => toggleView("view-posts")}
+              style={buttonStyle}
+            >
+              View Posts
+            </button>
+            <button
+              onClick={() => toggleView("create-post")}
+              style={buttonStyle}
+            >
+              Create Post
+            </button>
+          </div>
+        ) : null}
+        {viewState === "view-posts" && address && (
+          <div>
+            <div style={postContainerStyle}>
+              <h1>Posts</h1>
+              {posts.map((post, index) => (
+                <div key={index}>
+                  <h2>{post.title}</h2>
+                  <button
+                    style={{ fontSize: "16px" }}
+                    onClick={() =>
+                      window.open(\`https://infura-ipfs.io/ipfs/\${post.content}\`)
+                    }
+                  >
+                    Read on IPFS
+                  </button>
+                  {/* <ReactMarkdown>
+                    {post.postContent}
+                  </ReactMarkdown> */}
+                  <p style={mbidStyle}>GMID: {post.id}</p>
+                </div>
+              ))}
+            </div>
+          </div>
+        )}
+        {viewState === "create-post" && (
+          <div style={formContainerStyle}>
+            <h2>Create Post</h2>
+            <input
+              placeholder="Title"
+              onChange={(e) => setTitle(e.target.value)}
+              style={inputStyle}
+            />
+            <textarea
+              placeholder="Content"
+              onChange={(e) => setContent(e.target.value)}
+              style={inputStyle}
+            />
+            <button onClick={createPost}>Create Post</button>
+          </div>
+        )}
+      </div>
+    </div>
+  );
+}
+
+const outerContainerStyle = {
+  width: "90vw",
+  height: "100vh",
+  padding: "50px 0px",
+};
+
+const innerContainerStyle = {
+  width: "100%",
+  maxWidth: "800px",
+  margin: "0 auto",
+};
+
+const formContainerStyle = {
+  display: "flex",
+  flexDirection: "column",
+  alignItems: "center",
+};
+
+const inputStyle = {
+  width: "400px",
+  marginBottom: "10px",
+  padding: "10px",
+  height: "40px",
+};
+
+const postContainerStyle = {
+  margin: "0 auto",
+  padding: "1em",
+  width: "90%",
+  maxWidth: "800px",
+  display: "flex",
+  flexDirection: "column",
+  alignItems: "start",
+  justifyContent: "center",
+};
+
+const mbidStyle = {
+  fontSize: "10px",
+  textAlign: "start",
+};
+
+const buttonStyle = {
+  marginTop: 15,
+  marginRight: 5,
+  border: "1px solid rgba(255, 255, 255, .2)",
+};
+
+const buttonContainerStyle = {
+  marginTop: 15,
+  marginRight: 5,
+  display: "flex",
+  justifyContent: "right",
+};
+
+export default App;

Adding Ethermint Chain to MetaMask

Before we can test out our dapp, we'll need to configure the chains on MetaMask if we're deploying our rollup any

  1. Open your MetaMask wallet and click "Ethereum Mainnet" to open the dropdown.
  2. Select "Add network"
  3. Then "Add network manually"
  4. Enter the following details:
  • Network Name: Ethermint
  • New RPC URL: http://localhost:8545 or https://your.custom.ip.address:port
  • Chain ID: 9000
  • Currency symbol: CTE

Testing it out on Ethermint

Now we’re ready to run the app.

Right now, the app is configured to be using localhost:8545 using the Ethermint rollup we're running with Rollkit.

First, you'll need to install MetaMask.

To use the test account, you will need to import the private key from Ethermint to MetaMask. First, run the following command:

bash
PRIVATE_KEY=$(ethermintd keys unsafe-export-eth-key mykey --keyring-backend test)
+&& echo $PRIVATE_KEY | pbcopy
PRIVATE_KEY=$(ethermintd keys unsafe-export-eth-key mykey --keyring-backend test)
+&& echo $PRIVATE_KEY | pbcopy

Now, import the private key to MetaMask and switch to that account.

Next, let’s run it on your Ethermint rollup.

To do so, first update the contractAddress variable with the contract address deployed to Ethermint:

jsx
/* src/App.jsx */
+const contractAddress = "your-ethermint-contract-address";
/* src/App.jsx */
+const contractAddress = "your-ethermint-contract-address";

Next, run the React application:

bash
npm run dev
npm run dev

When you run the app, you should now be connected to and using the Ethermint rollup.

If you imported the address that started the chain, you'll see quite a large balance.

Now give it a spin 🌀

Now that you have your dapp running, go ahead and test out a new post on your Ethermint sovereign rollup. If you enjoyed this tutorial, be sure to share your example in our Discord!

`,113),e=[o];function t(c,r,E,y,i,F){return n(),a("div",null,e)}const h=s(p,[["render",t]]);export{u as __pageData,h as default}; diff --git a/assets/developers_full-stack-modular-development-guide.md.9f203595.lean.js b/assets/developers_full-stack-modular-development-guide.md.9f203595.lean.js new file mode 100644 index 00000000000..d6606abc159 --- /dev/null +++ b/assets/developers_full-stack-modular-development-guide.md.9f203595.lean.js @@ -0,0 +1 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.51d6c45b.js";const u=JSON.parse('{"title":"Full stack modular blockchain development guide","description":"Learn to build a full stack modular dapp.","frontmatter":{"description":"Learn to build a full stack modular dapp.","head":[["meta",{"name":"og:title","content":"Full stack modular blockchain development guide | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/full-stack-modular-development-guide.md","filePath":"developers/full-stack-modular-development-guide.md","lastUpdated":1712870672000}'),p={name:"developers/full-stack-modular-development-guide.md"},o=l("",113),e=[o];function t(c,r,E,y,i,F){return n(),a("div",null,e)}const h=s(p,[["render",t]]);export{u as __pageData,h as default}; diff --git a/assets/developers_gm-portal-bubs.md.230e5c37.js b/assets/developers_gm-portal-bubs.md.230e5c37.js new file mode 100644 index 00000000000..9a91f7a3414 --- /dev/null +++ b/assets/developers_gm-portal-bubs.md.230e5c37.js @@ -0,0 +1,21 @@ +import{_ as s,o as a,c as n,Q as o}from"./chunks/framework.51d6c45b.js";const t="/img/gm_contract.png",p="/img/gm_bubs.png",F=JSON.parse('{"title":"Deploying a dapp on Bubs testnet","description":"Make your own GM Portal dapp on the OP Stack.","frontmatter":{"description":"Make your own GM Portal dapp on the OP Stack.","head":[["meta",{"name":"og:title","content":"Deploying a dapp on Bubs testnet | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/gm-portal-bubs.md","filePath":"developers/gm-portal-bubs.md","lastUpdated":1711388443000}'),e={name:"developers/gm-portal-bubs.md"},l=o(`

Deploying a dapp on Bubs testnet

First, review the Bubs testnet page and the Deploy a smart contract to Bubs testnet tutorial.

You will need a funded account to deploy your smart contract.

Next, clone the gm-portal from Github and start the frontend:

bash
cd $HOME
+git clone https://github.com/jcstein/gm-portal.git
+cd gm-portal/frontend
+yarn && yarn dev
cd $HOME
+git clone https://github.com/jcstein/gm-portal.git
+cd gm-portal/frontend
+yarn && yarn dev

In a new terminal instance, set your private key for the faucet as a variable and the RPC URL you're using:

bash
export PRIVATE_KEY=ac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80
+export BUBS_RPC_URL=https://bubs-sepolia.rpc.caldera.xyz/http
export PRIVATE_KEY=ac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80
+export BUBS_RPC_URL=https://bubs-sepolia.rpc.caldera.xyz/http

Now, change into the gm-portal/contracts directory in the same terminal and deploy the contract using Foundry:

bash
cd $HOME/gm-portal/contracts
+forge script script/GmPortal.s.sol:GmPortalScript --rpc-url $BUBS_RPC_URL --private-key $PRIVATE_KEY --broadcast
cd $HOME/gm-portal/contracts
+forge script script/GmPortal.s.sol:GmPortalScript --rpc-url $BUBS_RPC_URL --private-key $PRIVATE_KEY --broadcast

gm-contract

In the output of the deployment, find the contract address and set it as a variable:

bash
export CONTRACT_ADDRESS=<your-contract-address-from-the-output-above>
export CONTRACT_ADDRESS=<your-contract-address-from-the-output-above>

Next, you're ready to interact with the contract from your terminal!

First, send a "gm" to the contract:

bash
cast send $CONTRACT_ADDRESS \\
+"gm(string)" "gm" \\
+--private-key $PRIVATE_KEY \\
+--rpc-url $BUBS_RPC_URL
cast send $CONTRACT_ADDRESS \\
+"gm(string)" "gm" \\
+--private-key $PRIVATE_KEY \\
+--rpc-url $BUBS_RPC_URL

Now that you've posted to the contract, you can read all "gms" (GMs) from the contract with this command:

bash
cast call $CONTRACT_ADDRESS "getAllGms()" --rpc-url $BUBS_RPC_URL
cast call $CONTRACT_ADDRESS "getAllGms()" --rpc-url $BUBS_RPC_URL

Next, query the total number of gms, which will be returned as a hex value:

bash
cast call $CONTRACT_ADDRESS "getTotalGms()" --rpc-url $BUBS_RPC_URL
cast call $CONTRACT_ADDRESS "getTotalGms()" --rpc-url $BUBS_RPC_URL

In order to interact with the contract on the frontend, you'll need to fund an account that you have in your Ethereum wallet. Transfer to an external account with this command:

bash
export RECEIVER=<receiver ETH address>
+cast send --private-key $PRIVATE_KEY $RECEIVER --value 1ether --rpc-url $BUBS_RPC_URL
export RECEIVER=<receiver ETH address>
+cast send --private-key $PRIVATE_KEY $RECEIVER --value 1ether --rpc-url $BUBS_RPC_URL

If you are in a different terminal than the one you set the private key in, you may need to set it again.

Update the frontend

Next, you will need to update a few things before you can interact with the contract on the frontend:

  1. Change the contract address on gm-portal/frontend/src/App.tsx to your contract address
  2. Match the chain info on gm-portal/frontend/src/main.tsx with the chain config of your L2
  3. If you changed the contract, update the ABI in gm-portal/frontend/GmPortal.json from gm-portal/contracts/out/GmPortal.sol/GmPortal.json. This can be done with:
bash
cd $HOME
+cp dev/gm-portal/contracts/out/GmPortal.sol/GmPortal.json dev/gm-portal/frontend
cd $HOME
+cp dev/gm-portal/contracts/out/GmPortal.sol/GmPortal.json dev/gm-portal/frontend

Interact with the frontend

Now, login with your wallet that you funded, and post a GM on your GM portal!

gm-bubs

Next steps

There are many possibilities of what could be built with this stack. These projects would be good to build on this stack:

  • onchain gaming
  • decentralized social media
  • an NFT ticketing rollup
  • Optimism on CelOPstia
  • OP Craft on Celestia
',32),c=[l];function r(i,y,d,E,h,u){return a(),n("div",null,c)}const b=s(e,[["render",r]]);export{F as __pageData,b as default}; diff --git a/assets/developers_gm-portal-bubs.md.230e5c37.lean.js b/assets/developers_gm-portal-bubs.md.230e5c37.lean.js new file mode 100644 index 00000000000..1c57e5c539b --- /dev/null +++ b/assets/developers_gm-portal-bubs.md.230e5c37.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as o}from"./chunks/framework.51d6c45b.js";const t="/img/gm_contract.png",p="/img/gm_bubs.png",F=JSON.parse('{"title":"Deploying a dapp on Bubs testnet","description":"Make your own GM Portal dapp on the OP Stack.","frontmatter":{"description":"Make your own GM Portal dapp on the OP Stack.","head":[["meta",{"name":"og:title","content":"Deploying a dapp on Bubs testnet | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/gm-portal-bubs.md","filePath":"developers/gm-portal-bubs.md","lastUpdated":1711388443000}'),e={name:"developers/gm-portal-bubs.md"},l=o("",32),c=[l];function r(i,y,d,E,h,u){return a(),n("div",null,c)}const b=s(e,[["render",r]]);export{F as __pageData,b as default}; diff --git a/assets/developers_integrate-celestia.md.02a50a7f.js b/assets/developers_integrate-celestia.md.02a50a7f.js new file mode 100644 index 00000000000..88e6bd4c018 --- /dev/null +++ b/assets/developers_integrate-celestia.md.02a50a7f.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"Integrate Celestia for service providers","description":"Learn how service providers can integrate with the Celestia network.","frontmatter":{"description":"Learn how service providers can integrate with the Celestia network.","prev":{"text":"Integrating Cosmostation for developers","link":"/developers/cosmostation"},"head":[["meta",{"name":"og:title","content":"Integrate Celestia for service providers | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/integrate-celestia.md","filePath":"developers/integrate-celestia.md","lastUpdated":1712870672000}'),s={name:"developers/integrate-celestia.md"},n=i('

Integrate Celestia for service providers

This document is for third-party service providers, such as custodians and explorers, integrating the Celestia network.

Getting started

When getting started Celestia, we recommend checking out these resources first:

Celestia service provider notes

Celestia is a fairly standard Cosmos-SDK based chain. We use the latest version of Tendermint and the Cosmos-SDK, with only minor modifications to each. This means that we are:

  • Using the default Cosmos-SDK modules: auth, bank, distribution, staking, slashing, mint, crisis, ibchost, genutil, evidence, ibctransfer, params, gov (limited in some TBD capacities), upgrade, vesting, feegrant, capability, and payment.
  • Use the standard digital keys schemes provided by the Cosmos-SDK and Tendermint, those being secp256k1 for user transactions, and tm-ed25519 for signing and verifying consensus messages.

While exactly which modules used is subject to change, Celestia aims to be as minimal as possible.

Custody and key management

Celestia supports many already existing key management systems, as we rely on the Cosmos-SDK and Tendermint libraries for signing and verifying transactions. Learn more in the Cosmos-SDK documentation

RPC and querying

In celestia-app, only the standard RPC endpoints for Tendermint and the Cosmos-SDK are exposed. We do not currently add or subtract any core functionality, but this could change in the future. The same goes for querying data from the chain.

In celestia-node, the Data Availability node client, there is a JSON-RPC API that allows you to interact directly with Celestia's Data Availability layer. Learn how to use the API in this tutorial.

Compatibility

Linux, particularly Ubuntu 20.04 LTS, is the most well tested. Potentially compatible with other OSs, but they are currently untested. Some of the cryptography libraries used for erasure data are not guaranteed to work on other platforms.

Syncing

Since we utilize Tendermint and the Cosmos-SDK, syncing the chain can be performed by any method that is supported by those libraries. This includes fast-sync, state sync, and quick sync.

Notable exceptions relative to other blockchains

Relative to other Tendermint based chains, Celestia will have significantly longer blocktimes of roughly 12* seconds. The reason behind this block time is to optimize the bandwidth used by light clients that are sampling the chain, and is not because we have modified Tendermint consensus in any meaningful way. Validators will likely download/upload relatively large blocks. It should be noted that while these blocks are large, very little typical blockchain state execution is actually occurring on Celestia. Meaning that the bandwidth requirements will likely be larger than that of a typical Cosmos-SDK based blockchain full node, the computing requirements should be similar in magnitude.

*Subject to Change

',21),r=[n];function o(l,d,c,h,u,m){return t(),a("div",null,r)}const b=e(s,[["render",o]]);export{g as __pageData,b as default}; diff --git a/assets/developers_integrate-celestia.md.02a50a7f.lean.js b/assets/developers_integrate-celestia.md.02a50a7f.lean.js new file mode 100644 index 00000000000..79a7f187f14 --- /dev/null +++ b/assets/developers_integrate-celestia.md.02a50a7f.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"Integrate Celestia for service providers","description":"Learn how service providers can integrate with the Celestia network.","frontmatter":{"description":"Learn how service providers can integrate with the Celestia network.","prev":{"text":"Integrating Cosmostation for developers","link":"/developers/cosmostation"},"head":[["meta",{"name":"og:title","content":"Integrate Celestia for service providers | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/integrate-celestia.md","filePath":"developers/integrate-celestia.md","lastUpdated":1712870672000}'),s={name:"developers/integrate-celestia.md"},n=i("",21),r=[n];function o(l,d,c,h,u,m){return t(),a("div",null,r)}const b=e(s,[["render",o]]);export{g as __pageData,b as default}; diff --git a/assets/developers_intro-to-op-stack.md.0252701c.js b/assets/developers_intro-to-op-stack.md.0252701c.js new file mode 100644 index 00000000000..d4231d20a88 --- /dev/null +++ b/assets/developers_intro-to-op-stack.md.0252701c.js @@ -0,0 +1 @@ +import{_ as t,o as e,c as a,Q as o}from"./chunks/framework.51d6c45b.js";const b=JSON.parse('{"title":"Introduction to OP Stack integration","description":"Learn about the integration of OP Stack with Celestia.","frontmatter":{"description":"Learn about the integration of OP Stack with Celestia.","head":[["meta",{"name":"og:title","content":"Introduction to OP Stack integration | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/intro-to-op-stack.md","filePath":"developers/intro-to-op-stack.md","lastUpdated":1700056188000}'),i={name:"developers/intro-to-op-stack.md"},s=o('

Introduction to OP Stack integration

Optimism is a low-cost and lightning-fast Ethereum L2 blockchain, built with the OP Stack.

Celestia is a modular consensus and data network, built to enable anyone to easily deploy their own blockchain with minimal overhead.

About the integration

Discover how to integrate existing blockchain frameworks like the OP Stack with Celestia in this category.

TIP

Tested on a machine with 8GB RAM, 160 GB SSD, Ubuntu 22.10, and a 4 core AMD CPU.

This is a beta integration and we are working on resolving open issues.

Optimism uses Ethereum as a data availability (DA) layer. Currently, settlement and DA for Optimism are on Ethereum, both onchain. op-batcher batches up rollup blocks and posts to Ethereum.

Table of contents of the category

Celestia and OP Stack repository

Find the repository for this integration at https://github.com/celestiaorg/optimism.

What are Optimism and the OP Stack?

Optimism, an Ethereum L2 blockchain, is powered by the OP Stack, which is also the foundation for the Optimism Collective committed to the impact=profit principle. This rewards individuals for their positive contributions to the collective.

Optimism addresses crypto ecosystem coordination failures, like funding public goods and infrastructure. The OP Stack fosters collaboration and prevents redundancy by creating a shared, open-source system for developing new L2 blockchains within the proposed Superchain ecosystem.

As Optimism evolves, the OP Stack will adapt to include components from blockchain infrastructure to governance systems. This software suite aims to simplify L2 blockchain creation and support the Optimism ecosystem's growth and development.

Learn more about Optimism.

What is Celestia?

Celestia is a modular consensus and data network, built to enable anyone to easily deploy their own blockchain with minimal overhead.

Celestia is a minimal blockchain that only orders and publishes transactions and does not execute them. By decoupling the consensus and application execution layers, Celestia modularizes the blockchain technology stack and unlocks new possibilities for decentralized application builders. Learn more at Celestia.org.

OP Stack and Celestia

This category will guide you through how to start your own devnet with a modified version of optimism-bedrock that uses Celestia as a DA layer.

The handling of data is accomplished in two ways. First, data is written to the data availability (DA) layer i.e. in this case Celestia, then the data commitment is written to the op-batcher. When reading op-node simply reads the data back from the DA layer by reading the data commitment from the op-batcher first, then reading the data from the DA layer using the data commitment. Hence, while previously op-node was reading from calldata on Ethereum, but now it reads data from Celestia.

There are a few tools involved in the data handling process. op-batcher batches up rollup blocks and posts them to Ethereum. op-geth handles execution, while op-proposer is responsible for state commitment submission.

By using Celestia as a DA layer, existing L2s can switch from posting their data as calldata on Ethereum, to posting to Celestia. The commitment to the block is posted on Celestia, which is purpose-built for data availability. This is a more scalable than the traditional method of posting this data as calldata on monolithic chains.

If you'd like to go modular, bedrock has made it easy to swap this out!

Next steps

Now that you understand the integration, you can start learning about the Bubs testnet, built with OP Stack and Celestia! This testnet is a great way to explore the possibilities of this integration and test your applications in a live environment.

',28),n=[s];function r(l,c,h,d,p,m){return e(),a("div",null,n)}const f=t(i,[["render",r]]);export{b as __pageData,f as default}; diff --git a/assets/developers_intro-to-op-stack.md.0252701c.lean.js b/assets/developers_intro-to-op-stack.md.0252701c.lean.js new file mode 100644 index 00000000000..7763dcb83fe --- /dev/null +++ b/assets/developers_intro-to-op-stack.md.0252701c.lean.js @@ -0,0 +1 @@ +import{_ as t,o as e,c as a,Q as o}from"./chunks/framework.51d6c45b.js";const b=JSON.parse('{"title":"Introduction to OP Stack integration","description":"Learn about the integration of OP Stack with Celestia.","frontmatter":{"description":"Learn about the integration of OP Stack with Celestia.","head":[["meta",{"name":"og:title","content":"Introduction to OP Stack integration | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/intro-to-op-stack.md","filePath":"developers/intro-to-op-stack.md","lastUpdated":1700056188000}'),i={name:"developers/intro-to-op-stack.md"},s=o("",28),n=[s];function r(l,c,h,d,p,m){return e(),a("div",null,n)}const f=t(i,[["render",r]]);export{b as __pageData,f as default}; diff --git a/assets/developers_node-api.md.df79f906.js b/assets/developers_node-api.md.df79f906.js new file mode 100644 index 00000000000..bffc84ad73c --- /dev/null +++ b/assets/developers_node-api.md.df79f906.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as r}from"./chunks/framework.51d6c45b.js";const w=JSON.parse('{"title":"Node API","description":"An overview of the celestia-node API.","frontmatter":{"description":"An overview of the celestia-node API.","head":[["meta",{"name":"og:title","content":"Node API | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/node-api.md","filePath":"developers/node-api.md","lastUpdated":1699288246000}'),o={name:"developers/node-api.md"},i=r('

Node API

The celestia-node API is made for interacting with celestia-node. There are two ways in which a user and developer can interact with the API, the RPC API and the Gateway API. View the API's documentation.

RPC API

The RPC API primarily focuses on developers and projects building on top of Celestia, who are willing to run their own DA nodes. The RPC API provides a richer set of features and a superior user experience. Unlike the Gateway API, the RPC API allows access to the internal wallet and keyring of the DA node, as well as other sensitive and administrative capabilities.

Library

The node can be used as a Golang library and designed for programmatic API access.

RPC

The RPC API is also exposed to OpenRPC(JSON-RPC 2.0) for users wanting to run their DA node as a separate DA service. It provides the same set of features as the library with an additional authentication system with different permissions levels to protect the wallet and signing + providing RPC-level DOS protection.

RPC API tutorial

The node tutorial, which uses the RPC CLI, is the recommended way to get started interacting with your Celestia node.

Gateway API

WARNING

The gateway endpoints have been deprecated and will be removed in the future. If you would like to use them anyway, you can find more details on GitHub.

The gateway API is a REST API which is meant to be deployed by infra providers to enable the public read-only gateway to the DA network for external users who don't want or can't run light nodes (like browsers currently) over HTTP. It has no wallet or signing functionality.

Gateway API tutorial

Check out the Prompt scavenger gateway API tutorial for more details.

',15),n=[i];function s(l,d,h,c,p,u){return a(),t("div",null,n)}const f=e(o,[["render",s]]);export{w as __pageData,f as default}; diff --git a/assets/developers_node-api.md.df79f906.lean.js b/assets/developers_node-api.md.df79f906.lean.js new file mode 100644 index 00000000000..53a06a71ed1 --- /dev/null +++ b/assets/developers_node-api.md.df79f906.lean.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as r}from"./chunks/framework.51d6c45b.js";const w=JSON.parse('{"title":"Node API","description":"An overview of the celestia-node API.","frontmatter":{"description":"An overview of the celestia-node API.","head":[["meta",{"name":"og:title","content":"Node API | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/node-api.md","filePath":"developers/node-api.md","lastUpdated":1699288246000}'),o={name:"developers/node-api.md"},i=r("",15),n=[i];function s(l,d,h,c,p,u){return a(),t("div",null,n)}const f=e(o,[["render",s]]);export{w as __pageData,f as default}; diff --git a/assets/developers_node-tutorial.md.8d60e9b7.js b/assets/developers_node-tutorial.md.8d60e9b7.js new file mode 100644 index 00000000000..2a27aeca882 --- /dev/null +++ b/assets/developers_node-tutorial.md.8d60e9b7.js @@ -0,0 +1,475 @@ +import{c as o}from"./chunks/constants.dc70dfe7.js";import{a as e}from"./chunks/arabica_versions.d7daf483.js";import{m as t}from"./chunks/mocha_versions.3ba4a823.js";import{m as c}from"./chunks/mainnet_versions.84d9fd1c.js";import{o as r,c as y,k as s,t as a,l as n,a as p,Q as l}from"./chunks/framework.51d6c45b.js";const E=l(`

Celestia-node RPC CLI tutorial

In this tutorial, we will cover how to use the celestia-node RPC API to submit and retrieve data (blobs) from the data availability layer by their namespace.

Introduction

Blobs

Data is posted to Celestia's DA layer by using MsgPayForBlobs transactions to the core network. Read more about MsgPayForBlobs.

Namespaces

Celestia partitions the block data into multiple namespaces, one for every application. This allows applications to only download their data, and not the data of other applications. Read more about Namespaced Merkle trees (NMTs).

TIP

If you already have a running and funded node, you can skip to the RPC CLI guide section.

WARNING

The gateway endpoints have been deprecated and will be removed in the future. If you would like to use them anyway, you can find more details on GitHub.

Hardware requirements

The following minimum hardware requirements are recommended for running a light node:

  • Memory: 500 MB RAM (minimum)
  • CPU: Single Core
  • Disk: 50 GB SSD Storage
  • Bandwidth: 56 Kbps for Download/56 Kbps for Upload

Setting up dependencies

This portion of the tutorial will go over setting up your development environment to run Celestia software. This environment can be used for development, building binaries, and running nodes.

In your terminal, set up dependencies needed to install and build celestia-node.

  1. If you are on Ubuntu, first update and upgrade your OS:

    bash
    sudo apt update && sudo apt upgrade -y
    sudo apt update && sudo apt upgrade -y
    bash
    sudo yum update
    sudo yum update
  2. Install essential packages that are necessary to execute many tasks like downloading files, compiling, and monitoring the node:

    bash
    sudo apt install curl tar wget clang pkg-config libssl-dev jq \\
    +build-essential git make ncdu -y
    sudo apt install curl tar wget clang pkg-config libssl-dev jq \\
    +build-essential git make ncdu -y
    bash
    sudo yum install curl tar wget clang pkg-config libssl-dev jq \\
    +build-essential git make ncdu -y
    sudo yum install curl tar wget clang pkg-config libssl-dev jq \\
    +build-essential git make ncdu -y
    bash
    # these commands are for installing Homebrew, wget and jq
    +# follow the instructions from the output after running this command
    +/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
    +
    +# then install wget & jq
    +brew install wget && brew install jq
    # these commands are for installing Homebrew, wget and jq
    +# follow the instructions from the output after running this command
    +/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
    +
    +# then install wget & jq
    +brew install wget && brew install jq

Install Golang

celestia-node is written in Golang so we must install Golang to build and run our node.

`,18),i=s("p",null,"Set the version for your desired network:",-1),d={class:"vp-code-group vp-adaptive-theme"},F=l('
',1),u={class:"blocks"},h={class:"language-bash vp-adaptive-theme active"},C=s("button",{title:"Copy Code",class:"copy"},null,-1),g=s("span",{class:"lang"},"bash",-1),b={class:"shiki github-dark vp-code-dark"},B={class:"line"},q=s("span",{style:{color:"#E1E4E8"}},"ver",-1),v=s("span",{style:{color:"#F97583"}},"=",-1),m={style:{color:"#9ECBFF"}},k={class:"shiki github-light vp-code-light"},A={class:"line"},_=s("span",{style:{color:"#24292E"}},"ver",-1),f=s("span",{style:{color:"#D73A49"}},"=",-1),D={style:{color:"#032F62"}},w={class:"language-bash vp-adaptive-theme"},T=s("button",{title:"Copy Code",class:"copy"},null,-1),x=s("span",{class:"lang"},"bash",-1),O={class:"shiki github-dark vp-code-dark"},R={class:"line"},P=s("span",{style:{color:"#E1E4E8"}},"ver",-1),I=s("span",{style:{color:"#F97583"}},"=",-1),S={style:{color:"#9ECBFF"}},N={class:"shiki github-light vp-code-light"},$={class:"line"},H=s("span",{style:{color:"#24292E"}},"ver",-1),U=s("span",{style:{color:"#D73A49"}},"=",-1),M={style:{color:"#032F62"}},j={class:"language-bash vp-adaptive-theme"},z=s("button",{title:"Copy Code",class:"copy"},null,-1),L=s("span",{class:"lang"},"bash",-1),V={class:"shiki github-dark vp-code-dark"},G={class:"line"},W=s("span",{style:{color:"#E1E4E8"}},"ver",-1),Y=s("span",{style:{color:"#F97583"}},"=",-1),K={style:{color:"#9ECBFF"}},X={class:"shiki github-light vp-code-light"},J={class:"line"},Q=s("span",{style:{color:"#24292E"}},"ver",-1),Z=s("span",{style:{color:"#D73A49"}},"=",-1),ss={style:{color:"#032F62"}},as=l(`
  • Download and install Golang:

    bash
    cd $HOME
    +wget "https://golang.org/dl/go$ver.linux-amd64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.linux-amd64.tar.gz"
    +rm "go$ver.linux-amd64.tar.gz"
    cd $HOME
    +wget "https://golang.org/dl/go$ver.linux-amd64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.linux-amd64.tar.gz"
    +rm "go$ver.linux-amd64.tar.gz"
    bash
    cd $HOME
    +wget "https://golang.org/dl/go$ver.linux-arm64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.linux-arm64.tar.gz"
    +rm "go$ver.linux-arm64.tar.gz"
    cd $HOME
    +wget "https://golang.org/dl/go$ver.linux-arm64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.linux-arm64.tar.gz"
    +rm "go$ver.linux-arm64.tar.gz"
    bash
    cd $HOME
    +wget "https://golang.org/dl/go$ver.darwin-arm64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.darwin-arm64.tar.gz"
    +rm "go$ver.darwin-arm64.tar.gz"
    cd $HOME
    +wget "https://golang.org/dl/go$ver.darwin-arm64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.darwin-arm64.tar.gz"
    +rm "go$ver.darwin-arm64.tar.gz"
    bash
    cd $HOME
    +wget "https://golang.org/dl/go$ver.darwin-amd64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.darwin-amd64.tar.gz"
    +rm "go$ver.darwin-amd64.tar.gz"
    cd $HOME
    +wget "https://golang.org/dl/go$ver.darwin-amd64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.darwin-amd64.tar.gz"
    +rm "go$ver.darwin-amd64.tar.gz"
  • Add your /usr/local/go/bin directory to your $PATH if you have not already:

    bash
    echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.bash_profile
    +source $HOME/.bash_profile
    echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.bash_profile
    +source $HOME/.bash_profile
    bash
    echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.zshrc
    +source $HOME/.zshrc
    echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.zshrc
    +source $HOME/.zshrc
  • To verify that the correct version of Go was installed correctly run:

    bash
    go version
    go version
  • `,3),ns=s("p",null,"The output will show the version installed.",-1),os=s("h2",{id:"celestia-node",tabindex:"-1"},[p("Celestia-node "),s("a",{class:"header-anchor",href:"#celestia-node","aria-label":'Permalink to "Celestia-node"'},"​")],-1),ls=s("h3",{id:"install-celestia-node",tabindex:"-1"},[p("Install celestia-node "),s("a",{class:"header-anchor",href:"#install-celestia-node","aria-label":'Permalink to "Install celestia-node"'},"​")],-1),ps=s("p",null,"Installing celestia-node for Arabica devnet or Mocha testnet means installing a specific version to be compatible with the network.",-1),es=s("p",null,"Install the celestia-node binary by running the following commands:",-1),ts=l(`
  • Remove any existing copy of celestia-node, clone the repository, and change into the directory.

    bash
    cd $HOME
    +rm -rf celestia-node
    +git clone https://github.com/celestiaorg/celestia-node.git
    +cd celestia-node/
    cd $HOME
    +rm -rf celestia-node
    +git clone https://github.com/celestiaorg/celestia-node.git
    +cd celestia-node/
  • `,1),cs=s("p",null,"Check out to the desired version, based on the network you will use:",-1),rs={class:"vp-code-group vp-adaptive-theme"},ys=l('
    ',1),Es={class:"blocks"},is={class:"language-bash vp-adaptive-theme active"},ds=s("button",{title:"Copy Code",class:"copy"},null,-1),Fs=s("span",{class:"lang"},"bash",-1),us={class:"shiki github-dark vp-code-dark"},hs={class:"line"},Cs=s("span",{style:{color:"#B392F0"}},"git",-1),gs=s("span",{style:{color:"#E1E4E8"}}," ",-1),bs=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),Bs=s("span",{style:{color:"#E1E4E8"}}," ",-1),qs={style:{color:"#9ECBFF"}},vs={class:"shiki github-light vp-code-light"},ms={class:"line"},ks=s("span",{style:{color:"#6F42C1"}},"git",-1),As=s("span",{style:{color:"#24292E"}}," ",-1),_s=s("span",{style:{color:"#032F62"}},"checkout",-1),fs=s("span",{style:{color:"#24292E"}}," ",-1),Ds={style:{color:"#032F62"}},ws={class:"language-bash vp-adaptive-theme"},Ts=s("button",{title:"Copy Code",class:"copy"},null,-1),xs=s("span",{class:"lang"},"bash",-1),Os={class:"shiki github-dark vp-code-dark"},Rs={class:"line"},Ps=s("span",{style:{color:"#B392F0"}},"git",-1),Is=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ss=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),Ns=s("span",{style:{color:"#E1E4E8"}}," ",-1),$s={style:{color:"#9ECBFF"}},Hs={class:"shiki github-light vp-code-light"},Us={class:"line"},Ms=s("span",{style:{color:"#6F42C1"}},"git",-1),js=s("span",{style:{color:"#24292E"}}," ",-1),zs=s("span",{style:{color:"#032F62"}},"checkout",-1),Ls=s("span",{style:{color:"#24292E"}}," ",-1),Vs={style:{color:"#032F62"}},Gs={class:"language-bash vp-adaptive-theme"},Ws=s("button",{title:"Copy Code",class:"copy"},null,-1),Ys=s("span",{class:"lang"},"bash",-1),Ks={class:"shiki github-dark vp-code-dark"},Xs={class:"line"},Js=s("span",{style:{color:"#B392F0"}},"git",-1),Qs=s("span",{style:{color:"#E1E4E8"}}," ",-1),Zs=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),sa=s("span",{style:{color:"#E1E4E8"}}," ",-1),aa={style:{color:"#9ECBFF"}},na={class:"shiki github-light vp-code-light"},oa={class:"line"},la=s("span",{style:{color:"#6F42C1"}},"git",-1),pa=s("span",{style:{color:"#24292E"}}," ",-1),ea=s("span",{style:{color:"#032F62"}},"checkout",-1),ta=s("span",{style:{color:"#24292E"}}," ",-1),ca={style:{color:"#032F62"}},ra=l('
  • Build the celestia binary:

    a. Standard build

    bash
    make build
    make build

    b. Experimental build

    OPTIONAL

    If you're a node operator comfortable with experimental features and seeking optimal performance with minimal RAM usage, this option is recommended for you.

    bash
    make build-jemalloc
    make build-jemalloc

    This build option enables CGO, and downloads and installs jemalloc. Learn more about the build command.

  • Install the binary:

    bash
    make install
    make install
  • Build the cel-key utility:

    bash
    make cel-key
    make cel-key
  • Verify that the binary is working and check the version:

    bash
    celestia version
    celestia version
  • ',4),ya=l(`

    The output will show the semantic version of celestia-node, commit hash, build date, system version, and Golang version.

    Instantiate a Celestia light node

    Now, let's instantiate a Celestia Light node:

    TIP

    RPC endpoints are exposed in all celestia-node types such as light, bridge and full nodes.

    bash
    celestia light init
    celestia light init
    bash
    celestia light init --p2p.network mocha
    celestia light init --p2p.network mocha
    bash
    celestia light init --p2p.network arabica
    celestia light init --p2p.network arabica

    Instantiating (or initializing) the node means setting up a node store on your machine. This is where the data and your keys will be stored.

    Connect to a public core endpoint

    Let's now run the Celestia Light node with a gRPC connection to an example public core endpoint.

    Note: You are also encouraged to find a community-run API endpoint and there are several in the Discord. This one is used for demonstration purposes. Check out the Mocha testnet page, or Arabica devnet page.

    bash
    celestia light start --core.ip <URI>
    celestia light start --core.ip <URI>
    bash
    celestia light start --core.ip <URI> --p2p.network mocha
    celestia light start --core.ip <URI> --p2p.network mocha
    bash
    celestia light start --core.ip <URI> --p2p.network arabica
    celestia light start --core.ip <URI> --p2p.network arabica

    TIP

    The --core.ip gRPC port defaults to 9090, so if you do not specify it in the command line, it will default to that port. You can add the port after the IP address or use the --core.grpc.port flag to specify another port if you prefer.

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    For example, your command along with an RPC endpoint might look like this:

    bash
    celestia light start --core.ip consensus.lunaroasis.net
    celestia light start --core.ip consensus.lunaroasis.net
    bash
    celestia light start --core.ip rpc-mocha.pops.one --p2p.network mocha
    celestia light start --core.ip rpc-mocha.pops.one --p2p.network mocha
    bash
    celestia light start --core.ip validator-1.celestia-arabica-11.com \\
    +  --p2p.network arabica
    celestia light start --core.ip validator-1.celestia-arabica-11.com \\
    +  --p2p.network arabica

    Keys and wallets

    You can create your key for your node by running the following command from the celestia-node directory:

    TIP

    You do not need to declare a network for Mainnet Beta. Refer to the chain ID section on the troubleshooting page for more information

    bash
    ./cel-key add <key-name> --keyring-backend test --node.type light \\
    +  --p2p.network <network>
    ./cel-key add <key-name> --keyring-backend test --node.type light \\
    +  --p2p.network <network>

    You can start your light node with the key created by running the following command:

    bash
    celestia light start --core.ip <URI> --keyring.accname <key-name>
    celestia light start --core.ip <URI> --keyring.accname <key-name>
    bash
    celestia light start --core.ip <URI> --keyring.accname <key-name> \\
    +  --p2p.network mocha
    celestia light start --core.ip <URI> --keyring.accname <key-name> \\
    +  --p2p.network mocha
    bash
    celestia light start --core.ip <URI> --keyring.accname <key-name> \\
    +  --p2p.network arabica
    celestia light start --core.ip <URI> --keyring.accname <key-name> \\
    +  --p2p.network arabica

    Once you start the light node, a wallet key will be generated for you. You will need to fund that address with Mocha testnet or Arabica devnet tokens to pay for PayForBlobs transactions.

    You can find the address by running the following command in the celestia-node directory:

    bash
    ./cel-key list --node.type light --keyring-backend test --p2p.network <network>
    ./cel-key list --node.type light --keyring-backend test --p2p.network <network>

    If you would like to fund your wallet with testnet tokens, head over to either the #mocha-faucet or #arabica-faucet channels on the Celestia Discord.

    You can request funds to your wallet address using the following command in Discord:

    text
    $request <CELESTIA-ADDRESS>
    $request <CELESTIA-ADDRESS>

    Where <CELESTIA-ADDRESS> is the celestia1****** address generated when you created the wallet.

    With your wallet funded, you can move on to the next step.

    RPC CLI guide

    This section of the tutorial will teach you how to interact with a Celestia node's remote procedure call (RPC) API using the command line interface (CLI).

    You will need to setup dependencies, install, and run celestia-node if you have not already.

    Command formatting

    The format for interacting with the RPC CLI methods is as follows:

    bash
    celestia <module> <method> [args...] [flags...]
    celestia <module> <method> [args...] [flags...]

    Where:

    • celestia is the main command to interact with the node.
    • <module> is the specific module in the node you want to interact with, such as blob, state, p2p, etc.
    • <method> is the specific method within the module that performs the action you want, such as blob.Submit, state.AccountAddress, p2p.Info, etc.
    • [args...] represents any additional arguments that the method might require.
    • [flags...] are parameters that modify the behavior of the command. They start with -- (e.g., --node.store, --token, or --url).

    For example, to submit a blob to Celestia, you can use this command once your node store is set:

    bash
    celestia blob submit 0x42690c204d39600fddd3 'gm' --node.store $NODE_STORE
    celestia blob submit 0x42690c204d39600fddd3 'gm' --node.store $NODE_STORE

    Alternatively, you could use the --token flag to set your auth token:

    bash
    celestia blob submit 0x42690c204d39600fddd3 'gm' --token $AUTH_TOKEN
    celestia blob submit 0x42690c204d39600fddd3 'gm' --token $AUTH_TOKEN

    Before you try that out, let's go over the basic flags that you will need to use when interacting with the RPC CLI. We'll also cover how to set your auth token and how to use the node store to set it.

    Basic flags

    All RPC CLI commands have basic flags that can be used to interact with the API.

    These include:

    • --node.store string - the path to root/home directory of your celestia-node store
    • --token string - authorization token for making requests
    • --url string - the address of the RPC, default is http://localhost:26658

    When running RPC CLI commands, you will need to set either the authentication token or set the node store, so the auth token can be retrieved from the store.

    The RPC CLI handles these flags in the following order:

    1. If user passes auth token, auth token is used.
    2. If user doesn't pass auth token, check node store flag, create token from node store, and use auth token from node store.

    Auth token 🔐

    In order to interact with the API using RPC CLI, you will need to set the authentication token.

    The --token string flag sets the authentication token. If a token is not found, authentication will not be set. And if authentication is not set, the request will fail.

    To set your authentication token, you can use the following command. Be sure to replace <node-type> with the type of node and <network> with the network that you are running your node on:

    bash
    export AUTH_TOKEN=$(celestia <node-type> auth admin --p2p.network <network>)
    export AUTH_TOKEN=$(celestia <node-type> auth admin --p2p.network <network>)

    Here's an example of how to set your auth token on a light node on Arabica:

    bash
    export AUTH_TOKEN=$(celestia light auth admin --p2p.network arabica)
    export AUTH_TOKEN=$(celestia light auth admin --p2p.network arabica)

    Node store

    In order to interact with the API using RPC CLI, you can also use your node store to set your auth token. This will allow you to interact with the API without setting an authentication token directly.

    `,56),Ea={class:"language-bash vp-adaptive-theme"},ia=s("button",{title:"Copy Code",class:"copy"},null,-1),da=s("span",{class:"lang"},"bash",-1),Fa={class:"shiki github-dark vp-code-dark"},ua={class:"line"},ha=s("span",{style:{color:"#F97583"}},"export",-1),Ca=s("span",{style:{color:"#E1E4E8"}}," NODE_STORE",-1),ga=s("span",{style:{color:"#F97583"}},"=",-1),ba=s("span",{style:{color:"#E1E4E8"}},"$HOME",-1),Ba={style:{color:"#9ECBFF"}},qa={class:"shiki github-light vp-code-light"},va={class:"line"},ma=s("span",{style:{color:"#D73A49"}},"export",-1),ka=s("span",{style:{color:"#24292E"}}," NODE_STORE",-1),Aa=s("span",{style:{color:"#D73A49"}},"=",-1),_a=s("span",{style:{color:"#24292E"}},"$HOME",-1),fa={style:{color:"#032F62"}},Da=l(`

    Then, set the --node.store flag to the $NODE_STORE variable to set the auth token from your node store:

    bash
    celestia <module> <method> [args...] --node.store $NODE_STORE
    celestia <module> <method> [args...] --node.store $NODE_STORE
    Auth token on custom or private network

    This section is for users who are using a CELESTIA_CUSTOM or private network.

    TIP

    If you are using a private and custom network, you will need to set the location of the node store in your auth command.

    bash
    --node.store $HOME/.celestia-light-private)
    --node.store $HOME/.celestia-light-private)

    The above is an example from the following custom network set up with:

    bash
    CELESTIA_CUSTOM=private celestia light init
    CELESTIA_CUSTOM=private celestia light init

    or

    bash
    celestia light init --p2p.network private
    celestia light init --p2p.network private

    As an example, this is what a completely custom network would look like:

    bash
    # Initialize node store
    +CELESTIA_CUSTOM=robusta-22 celestia light init
    +
    +# Set auth token
    +export AUTH_TOKEN=$(celestia light auth admin --p2p.network private \\
    +  --node.store $HOME/.celestia-light-robusta-22)
    # Initialize node store
    +CELESTIA_CUSTOM=robusta-22 celestia light init
    +
    +# Set auth token
    +export AUTH_TOKEN=$(celestia light auth admin --p2p.network private \\
    +  --node.store $HOME/.celestia-light-robusta-22)

    Submitting data

    In this example, we will be submitting a blob to the network with a blob.Submit transaction with our light node.

    Some things to consider:

    • The endpoint takes in namespace and data values.
      • The commitment will be generated by the node.
      • Share version is set by the node.
    • Namespace should be 10 bytes, prefixed by 0x if hex; otherwise use base64
    • Data can be hex-encoded (0x...), base64-encoded ("..."), or a plaintext string which will be encoded to base64 ('Hello There!')
    • Optionally, user can provide a gas fee and gas limit.

    We use the following namespace of 0x42690c204d39600fddd3 and the data value of 0x676d.

    Here is an example of the format of the blob.Submit transaction:

    bash
    celestia blob submit <hex-encoded namespace> <hex-encoded data> \\
    +  [optional: fee] [optional: gasLimit] [node store | auth token]
    celestia blob submit <hex-encoded namespace> <hex-encoded data> \\
    +  [optional: fee] [optional: gasLimit] [node store | auth token]

    We run the following to submit a blob to the network in hexadecimal format:

    bash
    celestia blob submit 0x42690c204d39600fddd3 0x676d \\
    +  --node.store $NODE_STORE
    celestia blob submit 0x42690c204d39600fddd3 0x676d \\
    +  --node.store $NODE_STORE

    We get the following output:

    json
    {
    +  "result": {
    +    "height": 252607,
    +    "commitment": "0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY="
    +  }
    +}
    {
    +  "result": {
    +    "height": 252607,
    +    "commitment": "0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY="
    +  }
    +}

    We can also use a string of text as the data value, which will be converted to base64. Here is an example of the format:

    bash
    celestia blob submit <hex-encoded namespace> <'data'> \\
    +  [optional: fee] [optional: gasLimit] [node store | auth token]
    celestia blob submit <hex-encoded namespace> <'data'> \\
    +  [optional: fee] [optional: gasLimit] [node store | auth token]

    And an example to submit "gm" as the plain-text data:

    bash
    celestia blob submit 0x42690c204d39600fddd3 'gm' --node.store $HOME/.celestia-light-mocha-4/
    celestia blob submit 0x42690c204d39600fddd3 'gm' --node.store $HOME/.celestia-light-mocha-4/

    Output:

    json
    {
    +  "result": {
    +    "height": 252614,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}
    {
    +  "result": {
    +    "height": 252614,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}

    If you notice from the above output, it returns a result of 252614 which we will use for the next command. The result corresponds to the height of the block in which the transaction was included.

    Optional: Submit with curl

    Refer to the submitting a blob using curl section.

    Retrieving data

    After submitting your PFB transaction, upon success, the node will return the block height for which the PFB transaction was included. You can then use that block height and the namespace ID with which you submitted your PFB transaction to get your message shares (data) returned to you. In this example, the block height we got was 252614 which we will use for the following command. Read more about shares in the Celestia Specs.

    Here is what an example of the format of the get command looks like:

    bash
    celestia blob get <block height> <hex-encoded namespace> \\
    +  <commitment from output above> <node store | auth>
    celestia blob get <block height> <hex-encoded namespace> \\
    +  <commitment from output above> <node store | auth>
    `,36),wa=l(`
    bash
    celestia blob get 252614 0x42690c204d39600fddd3 IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E= \\
    +  --node.store $NODE_STORE
    celestia blob get 252614 0x42690c204d39600fddd3 IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E= \\
    +  --node.store $NODE_STORE

    Will generate the following output:

    json
    {
    +  "result": {
    +    "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +    "data": "gm",
    +    "share_version": 0,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}
    {
    +  "result": {
    +    "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +    "data": "gm",
    +    "share_version": 0,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}

    The output here is base64 decoded to plain-text.

    To see the base64 response, use the --base64 flag set to TRUE (--base64=TRUE):

    bash
    celestia blob get 252614 0x42690c204d39600fddd3 IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E= \\
    +  --base64=TRUE --node.store $NODE_STORE
    celestia blob get 252614 0x42690c204d39600fddd3 IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E= \\
    +  --base64=TRUE --node.store $NODE_STORE

    The response will look similar to this:

    json
    {
    +  "result": {
    +    "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +    "data": "Z20=",
    +    "share_version": 0,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}
    {
    +  "result": {
    +    "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +    "data": "Z20=",
    +    "share_version": 0,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}

    To get all blobs in the namespace at the block height, use get-all instead of get:

    bash
    celestia blob get-all 252614 0x42690c204d39600fddd3 \\
    +  --node.store $NODE_STORE
    celestia blob get-all 252614 0x42690c204d39600fddd3 \\
    +  --node.store $NODE_STORE

    This will return the following:

    json
    {
    +  "result": [
    +    {
    +      "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +      "data": "gm",
    +      "share_version": 0,
    +      "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +    }
    +  ]
    +}
    {
    +  "result": [
    +    {
    +      "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +      "data": "gm",
    +      "share_version": 0,
    +      "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +    }
    +  ]
    +}

    To display the response in base64, use:

    bash
    celestia blob get-all 252614 0x42690c204d39600fddd3 \\
    +  --base64=TRUE --node.store $NODE_STORE
    celestia blob get-all 252614 0x42690c204d39600fddd3 \\
    +  --base64=TRUE --node.store $NODE_STORE

    Which will return:

    json
    {
    +  "result": [
    +    {
    +      "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +      "data": "gm",
    +      "share_version": 0,
    +      "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +    }
    +  ]
    +}
    {
    +  "result": [
    +    {
    +      "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +      "data": "gm",
    +      "share_version": 0,
    +      "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +    }
    +  ]
    +}

    Setting the gas price

    The --gas.price flag allows you to specify the gas price for the submission. If not specified, a default gas price will be used. The gas limit is automatically calculated based on the size of the blob being submitted.

    To set the gas price, you can use the --gas.price flag. The gas price will be set to default (0.002) if no value is passed.

    Learn more about gas fees and limits.

    To set a higher gas price of 0.004 utia, use the --gas.price 0.004 flag:

    bash
    celestia blob submit 0x42690c204d39600fddd3 'gm' --gas.price 0.004 \\
    +  --node.store $NODE_STORE
    celestia blob submit 0x42690c204d39600fddd3 'gm' --gas.price 0.004 \\
    +  --node.store $NODE_STORE

    You will receive the height and commitment of the block in which the transaction was included for these three examples:

    json
    {
    +  "result": {
    +    "height": 62562,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}
    {
    +  "result": {
    +    "height": 62562,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}

    Examples

    Check your balance

    Let's query our node for the balance of its default account (which is the account associated with the CELESTIA_NODE_AUTH_TOKEN key we generated above):

    bash
    celestia state balance --node.store $NODE_STORE
    celestia state balance --node.store $NODE_STORE

    The response will look similar to:

    json
    {
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "denom": "utia",
    +    "amount": "172118057"
    +  },
    +  "id": 1
    +}
    {
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "denom": "utia",
    +    "amount": "172118057"
    +  },
    +  "id": 1
    +}

    Check the balance of another address

    Here is an example of the format of the balance-for-address command:

    bash
    celestia state balance-for-address <address> \\
    +  --node.store $NODE_STORE
    celestia state balance-for-address <address> \\
    +  --node.store $NODE_STORE

    Let's query our node for the balance of another address:

    bash
    celestia state balance-for-address celestia10rtd9lhel2cuh6c659l25yncl6atcyt37umard \\
    +  --node.store $NODE_STORE
    celestia state balance-for-address celestia10rtd9lhel2cuh6c659l25yncl6atcyt37umard \\
    +  --node.store $NODE_STORE

    The response will be the balance of the address you queried:

    json
    {
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "denom": "utia",
    +    "amount": "1000000"
    +  },
    +  "id": 1
    +}
    {
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "denom": "utia",
    +    "amount": "1000000"
    +  },
    +  "id": 1
    +}

    Get your node ID

    This is an RPC call in order to get your node's peerId information:

    bash
    celestia p2p info --node.store $NODE_STORE
    celestia p2p info --node.store $NODE_STORE

    The node ID is in the ID value from the response:

    json
    {
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "ID": "12D3KooWFFhCaAqY56oEqY3pLZUdLsv4RYAfVWKATZRepUPdosLp",
    +    "Addrs": [
    +      "/ip4/10.0.0.171/tcp/2121",
    +      "/ip4/10.0.0.171/udp/2121/quic-v1",
    +      "/ip4/71.200.65.106/tcp/25630",
    +      "/ip4/71.200.65.106/udp/25630/quic-v1",
    +      "/ip6/::1/tcp/2121",
    +      "/ip6/::1/udp/2121/quic-v1"
    +    ]
    +  },
    +  "id": 1
    +}
    {
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "ID": "12D3KooWFFhCaAqY56oEqY3pLZUdLsv4RYAfVWKATZRepUPdosLp",
    +    "Addrs": [
    +      "/ip4/10.0.0.171/tcp/2121",
    +      "/ip4/10.0.0.171/udp/2121/quic-v1",
    +      "/ip4/71.200.65.106/tcp/25630",
    +      "/ip4/71.200.65.106/udp/25630/quic-v1",
    +      "/ip6/::1/tcp/2121",
    +      "/ip6/::1/udp/2121/quic-v1"
    +    ]
    +  },
    +  "id": 1
    +}

    Get your account address

    This is an RPC call in order to get your node's account address:

    bash
    celestia state account-address --node.store $NODE_STORE
    celestia state account-address --node.store $NODE_STORE

    Response:

    json
    {
    +  "jsonrpc": "2.0",
    +  "result": "celestia1znk24rh52pgcd9z5x2x42jztjh6raaaphuvrt3",
    +  "id": 1
    +}
    {
    +  "jsonrpc": "2.0",
    +  "result": "celestia1znk24rh52pgcd9z5x2x42jztjh6raaaphuvrt3",
    +  "id": 1
    +}

    Get block header by height

    Here is an example of the format of the GetByHeight command:

    bash
    celestia header get-by-height <height> \\
    +  --node.store $NODE_STORE
    celestia header get-by-height <height> \\
    +  --node.store $NODE_STORE

    Now, let's get the block header information.

    Here we will get the header from Block 1:

    bash
    celestia header get-by-height 1 \\
    +  --node.store $NODE_STORE\`\`\`
    +
    +It will output something like this:
    +
    +<!-- markdownlint-disable MD013 -->
    +
    +\`\`\`json
    +{
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "header": {
    +      "version": {
    +        "block": "11",
    +        "app": "1"
    +      },
    +      "chain_id": "arabica-11",
    +      "height": "1",
    +      "time": "2023-06-27T13:02:39.741743Z",
    +      "last_block_id": {
    +        "hash": "",
    +        "parts": {
    +          "total": 0,
    +          "hash": ""
    +        }
    +      },
    +      "last_commit_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "data_hash": "3D96B7D238E7E0456F6AF8E7CDF0A67BD6CF9C2089ECB559C659DCAA1F880353",
    +      "validators_hash": "6363C68770C200FD794445668F9B18F5B1DD1125180D6E8D5AB004F7DD7A0F48",
    +      "next_validators_hash": "6363C68770C200FD794445668F9B18F5B1DD1125180D6E8D5AB004F7DD7A0F48",
    +      "consensus_hash": "048091BC7DDC283F77BFBF91D73C44DA58C3DF8A9CBC867405D8B7F3DAADA22F",
    +      "app_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "last_results_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "evidence_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "proposer_address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86"
    +    },
    +    "validator_set": {
    +      "validators": [
    +        {
    +          "address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86",
    +          "pub_key": {
    +            "type": "tendermint/PubKeyEd25519",
    +            "value": "9aNBAxno1B4X5LR2qY5qWqwrMNOzejkctXwzq9BExsg="
    +          },
    +          "voting_power": "500000000",
    +          "proposer_priority": "0"
    +        }
    +      ],
    +      "proposer": {
    +        "address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86",
    +        "pub_key": {
    +          "type": "tendermint/PubKeyEd25519",
    +          "value": "9aNBAxno1B4X5LR2qY5qWqwrMNOzejkctXwzq9BExsg="
    +        },
    +        "voting_power": "500000000",
    +        "proposer_priority": "0"
    +      }
    +    },
    +    "commit": {
    +      "height": 1,
    +      "round": 0,
    +      "block_id": {
    +        "hash": "7A5FABB19713D732D967B1DA84FA0DF5E87A7B62302D783F78743E216C1A3550",
    +        "parts": {
    +          "total": 1,
    +          "hash": "D85C907CE660878A8203AC74BAA147CCC1F87114B45B568B72AD207B62AFE45E"
    +        }
    +      },
    +      "signatures": [
    +        {
    +          "block_id_flag": 2,
    +          "validator_address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86",
    +          "timestamp": "2023-06-30T08:40:19.299137127Z",
    +          "signature": "qmaEzrnbtgEXCRYc8pCvGRbS+uMuknIBoRAE4qyE7oSgWCRwBVYS/oPReXQLg9ER1oEY1De4MkWvMjlFnQOOCg=="
    +        }
    +      ]
    +    },
    +    "dah": {
    +      "row_roots": [
    +        "//////////////////////////////////////7//////////////////////////////////////huZWOTTDmD36N1F75A9BshxNlRasCnNpQiWqIhdVHcU",
    +        "/////////////////////////////////////////////////////////////////////////////5iieeroHBMfF+sER3JpvROIeEJZjbY+TRE0ntADQLL3"
    +      ],
    +      "column_roots": [
    +        "//////////////////////////////////////7//////////////////////////////////////huZWOTTDmD36N1F75A9BshxNlRasCnNpQiWqIhdVHcU",
    +        "/////////////////////////////////////////////////////////////////////////////5iieeroHBMfF+sER3JpvROIeEJZjbY+TRE0ntADQLL3"
    +      ]
    +    }
    +  },
    +  "id": 1
    +}
    celestia header get-by-height 1 \\
    +  --node.store $NODE_STORE\`\`\`
    +
    +It will output something like this:
    +
    +<!-- markdownlint-disable MD013 -->
    +
    +\`\`\`json
    +{
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "header": {
    +      "version": {
    +        "block": "11",
    +        "app": "1"
    +      },
    +      "chain_id": "arabica-11",
    +      "height": "1",
    +      "time": "2023-06-27T13:02:39.741743Z",
    +      "last_block_id": {
    +        "hash": "",
    +        "parts": {
    +          "total": 0,
    +          "hash": ""
    +        }
    +      },
    +      "last_commit_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "data_hash": "3D96B7D238E7E0456F6AF8E7CDF0A67BD6CF9C2089ECB559C659DCAA1F880353",
    +      "validators_hash": "6363C68770C200FD794445668F9B18F5B1DD1125180D6E8D5AB004F7DD7A0F48",
    +      "next_validators_hash": "6363C68770C200FD794445668F9B18F5B1DD1125180D6E8D5AB004F7DD7A0F48",
    +      "consensus_hash": "048091BC7DDC283F77BFBF91D73C44DA58C3DF8A9CBC867405D8B7F3DAADA22F",
    +      "app_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "last_results_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "evidence_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "proposer_address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86"
    +    },
    +    "validator_set": {
    +      "validators": [
    +        {
    +          "address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86",
    +          "pub_key": {
    +            "type": "tendermint/PubKeyEd25519",
    +            "value": "9aNBAxno1B4X5LR2qY5qWqwrMNOzejkctXwzq9BExsg="
    +          },
    +          "voting_power": "500000000",
    +          "proposer_priority": "0"
    +        }
    +      ],
    +      "proposer": {
    +        "address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86",
    +        "pub_key": {
    +          "type": "tendermint/PubKeyEd25519",
    +          "value": "9aNBAxno1B4X5LR2qY5qWqwrMNOzejkctXwzq9BExsg="
    +        },
    +        "voting_power": "500000000",
    +        "proposer_priority": "0"
    +      }
    +    },
    +    "commit": {
    +      "height": 1,
    +      "round": 0,
    +      "block_id": {
    +        "hash": "7A5FABB19713D732D967B1DA84FA0DF5E87A7B62302D783F78743E216C1A3550",
    +        "parts": {
    +          "total": 1,
    +          "hash": "D85C907CE660878A8203AC74BAA147CCC1F87114B45B568B72AD207B62AFE45E"
    +        }
    +      },
    +      "signatures": [
    +        {
    +          "block_id_flag": 2,
    +          "validator_address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86",
    +          "timestamp": "2023-06-30T08:40:19.299137127Z",
    +          "signature": "qmaEzrnbtgEXCRYc8pCvGRbS+uMuknIBoRAE4qyE7oSgWCRwBVYS/oPReXQLg9ER1oEY1De4MkWvMjlFnQOOCg=="
    +        }
    +      ]
    +    },
    +    "dah": {
    +      "row_roots": [
    +        "//////////////////////////////////////7//////////////////////////////////////huZWOTTDmD36N1F75A9BshxNlRasCnNpQiWqIhdVHcU",
    +        "/////////////////////////////////////////////////////////////////////////////5iieeroHBMfF+sER3JpvROIeEJZjbY+TRE0ntADQLL3"
    +      ],
    +      "column_roots": [
    +        "//////////////////////////////////////7//////////////////////////////////////huZWOTTDmD36N1F75A9BshxNlRasCnNpQiWqIhdVHcU",
    +        "/////////////////////////////////////////////////////////////////////////////5iieeroHBMfF+sER3JpvROIeEJZjbY+TRE0ntADQLL3"
    +      ]
    +    }
    +  },
    +  "id": 1
    +}

    Combined commands

    bash
    celestia share get-by-namespace "$(celestia header get-by-height 147105 \\
    +  --node.store  $NODE_STORE | jq '.result.dah' -r)" 0x42690c204d39600fddd3 \\
    +  --node.store $NODE_STORE
    celestia share get-by-namespace "$(celestia header get-by-height 147105 \\
    +  --node.store  $NODE_STORE | jq '.result.dah' -r)" 0x42690c204d39600fddd3 \\
    +  --node.store $NODE_STORE

    Get data availability sampler stats

    bash
    celestia das sampling-stats --node.store $NODE_STORE
    celestia das sampling-stats --node.store $NODE_STORE

    Transfer balance of utia to another account

    First, set your address as a variable:

    bash
    export ADDRESS=celestia1c425ckmve2489atttx022qpc02gxspa29wmh0d
    export ADDRESS=celestia1c425ckmve2489atttx022qpc02gxspa29wmh0d

    Then, transfer the amount of tokens that you would like, while setting the recipient's address, gas fee, and gasLimit. This is what the format will look like:

    bash
    celestia state transfer $ADDRESS <amount in utia> <gas fee in utia> <gas fee in utia> \\
    +  --node.store $NODE_STORE
    celestia state transfer $ADDRESS <amount in utia> <gas fee in utia> <gas fee in utia> \\
    +  --node.store $NODE_STORE

    Here is an example, sending 0.1 TIA, with a gas fee of 0.008 TIA, and a gas limit of 0.08:

    bash
    celestia state transfer $ADDRESS 100000 8000 80000 \\
    +  --node.store $NODE_STORE
    celestia state transfer $ADDRESS 100000 8000 80000 \\
    +  --node.store $NODE_STORE

    If you'd just like to return the transaction hash, you can use jq:

    bash
    celestia state transfer $ADDRESS 100000 8000 80000 --node.store $NODE_STORE | jq .result.txhash
    celestia state transfer $ADDRESS 100000 8000 80000 --node.store $NODE_STORE | jq .result.txhash

    API version

    To query your node's API version, you can use the following command:

    bash
    celestia node info --node.store $NODE_STORE
    celestia node info --node.store $NODE_STORE

    Help

    To get help and view the CLI menu, use the following command:

    bash
    celestia --help
    celestia --help

    To view help menu for a specific method, use the following command:

    bash
    celestia <module> <method> --node.store $NODE_STORE --help
    celestia <module> <method> --node.store $NODE_STORE --help

    Advanced example

    This example shows us using the jq command to parse the output of the celestia header get-by-height method to get the extended header used in celestia share get-by-namespace:

    bash
    celestia share get-by-namespace \\
    +  "$(celestia header get-by-height 252614 --node.store $NODE_STORE | jq '.result.dah' -r)" \\
    +  0x42690c204d39600fddd3 --node.store $NODE_STORE
    celestia share get-by-namespace \\
    +  "$(celestia header get-by-height 252614 --node.store $NODE_STORE | jq '.result.dah' -r)" \\
    +  0x42690c204d39600fddd3 --node.store $NODE_STORE

    Additional resources

    Submitting a blob using curl

    In order to post a blob using curl, you will need a light node running with the --core.ip string flag, providing access to a consensus endpoint. The flag indicates node to connect to the given core consensus node. Examples: 127.0.0.1 or subdomain.domain.tld. Using either IP or DNS assumes RPC port 26657 and gRPC port 9090 as default unless otherwise specified.

    1. In your terminal, set the auth token for the desired network. In this example, we will use Mainnet Beta.
    bash
    export CELESTIA_NODE_AUTH_TOKEN=$(celestia light auth admin --p2p.network celestia)
    export CELESTIA_NODE_AUTH_TOKEN=$(celestia light auth admin --p2p.network celestia)
    1. Post your blob with:
    bash
    curl -H "Content-Type: application/json" -H "Authorization: Bearer $CELESTIA_NODE_AUTH_TOKEN" -X POST --data '{"id": 1,
    +  "jsonrpc": "2.0",
    +  "method": "blob.Submit",
    +  "params": [
    +    [
    +      {
    +        "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAAECAwQFBgcICRA=",
    +        "data": "VGhpcyBpcyBhbiBleGFtcGxlIG9mIHNvbWUgYmxvYiBkYXRh",
    +        "share_version": 0,
    +        "commitment": "AD5EzbG0/EMvpw0p8NIjMVnoCP4Bv6K+V6gjmwdXUKU="
    +      }
    +    ],
    +    0.002
    +  ]
    +}' 127.0.0.1:26658
    curl -H "Content-Type: application/json" -H "Authorization: Bearer $CELESTIA_NODE_AUTH_TOKEN" -X POST --data '{"id": 1,
    +  "jsonrpc": "2.0",
    +  "method": "blob.Submit",
    +  "params": [
    +    [
    +      {
    +        "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAAECAwQFBgcICRA=",
    +        "data": "VGhpcyBpcyBhbiBleGFtcGxlIG9mIHNvbWUgYmxvYiBkYXRh",
    +        "share_version": 0,
    +        "commitment": "AD5EzbG0/EMvpw0p8NIjMVnoCP4Bv6K+V6gjmwdXUKU="
    +      }
    +    ],
    +    0.002
    +  ]
    +}' 127.0.0.1:26658
    1. Upon successful blob submission, the result will show the block height:
    bash
    {"jsonrpc":"2.0","result":362101,"id":1}
    {"jsonrpc":"2.0","result":362101,"id":1}

    The example transaction can be found on Celenium.

    Post an SVG as a PFB

    If you'd like to create your own SVG, post it to Celestia, and retrieve it, you can check out the Base64 SVG Tutorial.

    Golang guide

    If you're interested in interacting with the node's API in Go (client.go), you can use the da-rpc-client-tutorial repo.

    Troubleshooting

    If you encounter an error like:

    sh
    "rpc error: code = NotFound desc = account celestia1krkle0n547u0znz3unnln8paft2dq4z3rznv86 not found"
    "rpc error: code = NotFound desc = account celestia1krkle0n547u0znz3unnln8paft2dq4z3rznv86 not found"

    It is possible that the account you are trying to submit a PayForBlobs from doesn't have testnet tokens yet. Ensure the testnet faucet has funded your account with tokens and then try again.

    `,95),Ha=JSON.parse('{"title":"Celestia-node RPC CLI tutorial","description":"","frontmatter":{"next":{"text":"Prompt Scavenger","link":"/developers/prompt-scavenger"},"head":[["meta",{"name":"og:title","content":"Celestia-node RPC CLI tutorial | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/node-tutorial.md","filePath":"developers/node-tutorial.md","lastUpdated":1713191709000}'),Ta={name:"developers/node-tutorial.md"},Ua=Object.assign(Ta,{setup(xa){return(Oa,Ra)=>(r(),y("div",null,[E,s("ol",null,[s("li",null,[i,s("div",d,[F,s("div",u,[s("div",h,[C,g,s("pre",b,[s("code",null,[s("span",B,[q,v,s("span",m,'"'+a(n(o).golangNodeMainnet)+'"',1)])])]),s("pre",k,[s("code",null,[s("span",A,[_,f,s("span",D,'"'+a(n(o).golangNodeMainnet)+'"',1)])])])]),s("div",w,[T,x,s("pre",O,[s("code",null,[s("span",R,[P,I,s("span",S,'"'+a(n(o).golangNodeMocha)+'"',1)])])]),s("pre",N,[s("code",null,[s("span",$,[H,U,s("span",M,'"'+a(n(o).golangNodeMocha)+'"',1)])])])]),s("div",j,[z,L,s("pre",V,[s("code",null,[s("span",G,[W,Y,s("span",K,'"'+a(n(o).golangNodeArabica)+'"',1)])])]),s("pre",X,[s("code",null,[s("span",J,[Q,Z,s("span",ss,'"'+a(n(o).golangNodeArabica)+'"',1)])])])])])])]),as]),ns,os,ls,ps,es,s("ol",null,[ts,s("li",null,[cs,s("div",rs,[ys,s("div",Es,[s("div",is,[ds,Fs,s("pre",us,[s("code",null,[s("span",hs,[Cs,gs,bs,Bs,s("span",qs,"tags/"+a(n(c)["node-latest-tag"]),1)])])]),s("pre",vs,[s("code",null,[s("span",ms,[ks,As,_s,fs,s("span",Ds,"tags/"+a(n(c)["node-latest-tag"]),1)])])])]),s("div",ws,[Ts,xs,s("pre",Os,[s("code",null,[s("span",Rs,[Ps,Is,Ss,Ns,s("span",$s,"tags/"+a(n(t)["node-latest-tag"]),1)])])]),s("pre",Hs,[s("code",null,[s("span",Us,[Ms,js,zs,Ls,s("span",Vs,"tags/"+a(n(t)["node-latest-tag"]),1)])])])]),s("div",Gs,[Ws,Ys,s("pre",Ks,[s("code",null,[s("span",Xs,[Js,Qs,Zs,sa,s("span",aa,"tags/"+a(n(e)["node-latest-tag"]),1)])])]),s("pre",na,[s("code",null,[s("span",oa,[la,pa,ea,ta,s("span",ca,"tags/"+a(n(e)["node-latest-tag"]),1)])])])])])])]),ra]),ya,s("p",null,"To set your node store for a light node on "+a(n(o).mochaChainId)+", you can use the following command:",1),s("div",Ea,[ia,da,s("pre",Fa,[s("code",null,[s("span",ua,[ha,Ca,ga,ba,s("span",Ba,"/.celestia-light-"+a(n(o).mochaChainId),1)])])]),s("pre",qa,[s("code",null,[s("span",va,[ma,ka,Aa,_a,s("span",fa,"/.celestia-light-"+a(n(o).mochaChainId),1)])])])]),Da,s("p",null,[p("Here is an example command to retrieve the data from above, on "),s("code",null,a(n(o).arabicaChainId),1),p(":")]),wa]))}});export{Ha as __pageData,Ua as default}; diff --git a/assets/developers_node-tutorial.md.8d60e9b7.lean.js b/assets/developers_node-tutorial.md.8d60e9b7.lean.js new file mode 100644 index 00000000000..28b9de24385 --- /dev/null +++ b/assets/developers_node-tutorial.md.8d60e9b7.lean.js @@ -0,0 +1 @@ +import{c as o}from"./chunks/constants.dc70dfe7.js";import{a as e}from"./chunks/arabica_versions.d7daf483.js";import{m as t}from"./chunks/mocha_versions.3ba4a823.js";import{m as c}from"./chunks/mainnet_versions.84d9fd1c.js";import{o as r,c as y,k as s,t as a,l as n,a as p,Q as l}from"./chunks/framework.51d6c45b.js";const E=l("",18),i=s("p",null,"Set the version for your desired network:",-1),d={class:"vp-code-group vp-adaptive-theme"},F=l("",1),u={class:"blocks"},h={class:"language-bash vp-adaptive-theme active"},C=s("button",{title:"Copy Code",class:"copy"},null,-1),g=s("span",{class:"lang"},"bash",-1),b={class:"shiki github-dark vp-code-dark"},B={class:"line"},q=s("span",{style:{color:"#E1E4E8"}},"ver",-1),v=s("span",{style:{color:"#F97583"}},"=",-1),m={style:{color:"#9ECBFF"}},k={class:"shiki github-light vp-code-light"},A={class:"line"},_=s("span",{style:{color:"#24292E"}},"ver",-1),f=s("span",{style:{color:"#D73A49"}},"=",-1),D={style:{color:"#032F62"}},w={class:"language-bash vp-adaptive-theme"},T=s("button",{title:"Copy Code",class:"copy"},null,-1),x=s("span",{class:"lang"},"bash",-1),O={class:"shiki github-dark vp-code-dark"},R={class:"line"},P=s("span",{style:{color:"#E1E4E8"}},"ver",-1),I=s("span",{style:{color:"#F97583"}},"=",-1),S={style:{color:"#9ECBFF"}},N={class:"shiki github-light vp-code-light"},$={class:"line"},H=s("span",{style:{color:"#24292E"}},"ver",-1),U=s("span",{style:{color:"#D73A49"}},"=",-1),M={style:{color:"#032F62"}},j={class:"language-bash vp-adaptive-theme"},z=s("button",{title:"Copy Code",class:"copy"},null,-1),L=s("span",{class:"lang"},"bash",-1),V={class:"shiki github-dark vp-code-dark"},G={class:"line"},W=s("span",{style:{color:"#E1E4E8"}},"ver",-1),Y=s("span",{style:{color:"#F97583"}},"=",-1),K={style:{color:"#9ECBFF"}},X={class:"shiki github-light vp-code-light"},J={class:"line"},Q=s("span",{style:{color:"#24292E"}},"ver",-1),Z=s("span",{style:{color:"#D73A49"}},"=",-1),ss={style:{color:"#032F62"}},as=l("",3),ns=s("p",null,"The output will show the version installed.",-1),os=s("h2",{id:"celestia-node",tabindex:"-1"},[p("Celestia-node "),s("a",{class:"header-anchor",href:"#celestia-node","aria-label":'Permalink to "Celestia-node"'},"​")],-1),ls=s("h3",{id:"install-celestia-node",tabindex:"-1"},[p("Install celestia-node "),s("a",{class:"header-anchor",href:"#install-celestia-node","aria-label":'Permalink to "Install celestia-node"'},"​")],-1),ps=s("p",null,"Installing celestia-node for Arabica devnet or Mocha testnet means installing a specific version to be compatible with the network.",-1),es=s("p",null,"Install the celestia-node binary by running the following commands:",-1),ts=l("",1),cs=s("p",null,"Check out to the desired version, based on the network you will use:",-1),rs={class:"vp-code-group vp-adaptive-theme"},ys=l("",1),Es={class:"blocks"},is={class:"language-bash vp-adaptive-theme active"},ds=s("button",{title:"Copy Code",class:"copy"},null,-1),Fs=s("span",{class:"lang"},"bash",-1),us={class:"shiki github-dark vp-code-dark"},hs={class:"line"},Cs=s("span",{style:{color:"#B392F0"}},"git",-1),gs=s("span",{style:{color:"#E1E4E8"}}," ",-1),bs=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),Bs=s("span",{style:{color:"#E1E4E8"}}," ",-1),qs={style:{color:"#9ECBFF"}},vs={class:"shiki github-light vp-code-light"},ms={class:"line"},ks=s("span",{style:{color:"#6F42C1"}},"git",-1),As=s("span",{style:{color:"#24292E"}}," ",-1),_s=s("span",{style:{color:"#032F62"}},"checkout",-1),fs=s("span",{style:{color:"#24292E"}}," ",-1),Ds={style:{color:"#032F62"}},ws={class:"language-bash vp-adaptive-theme"},Ts=s("button",{title:"Copy Code",class:"copy"},null,-1),xs=s("span",{class:"lang"},"bash",-1),Os={class:"shiki github-dark vp-code-dark"},Rs={class:"line"},Ps=s("span",{style:{color:"#B392F0"}},"git",-1),Is=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ss=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),Ns=s("span",{style:{color:"#E1E4E8"}}," ",-1),$s={style:{color:"#9ECBFF"}},Hs={class:"shiki github-light vp-code-light"},Us={class:"line"},Ms=s("span",{style:{color:"#6F42C1"}},"git",-1),js=s("span",{style:{color:"#24292E"}}," ",-1),zs=s("span",{style:{color:"#032F62"}},"checkout",-1),Ls=s("span",{style:{color:"#24292E"}}," ",-1),Vs={style:{color:"#032F62"}},Gs={class:"language-bash vp-adaptive-theme"},Ws=s("button",{title:"Copy Code",class:"copy"},null,-1),Ys=s("span",{class:"lang"},"bash",-1),Ks={class:"shiki github-dark vp-code-dark"},Xs={class:"line"},Js=s("span",{style:{color:"#B392F0"}},"git",-1),Qs=s("span",{style:{color:"#E1E4E8"}}," ",-1),Zs=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),sa=s("span",{style:{color:"#E1E4E8"}}," ",-1),aa={style:{color:"#9ECBFF"}},na={class:"shiki github-light vp-code-light"},oa={class:"line"},la=s("span",{style:{color:"#6F42C1"}},"git",-1),pa=s("span",{style:{color:"#24292E"}}," ",-1),ea=s("span",{style:{color:"#032F62"}},"checkout",-1),ta=s("span",{style:{color:"#24292E"}}," ",-1),ca={style:{color:"#032F62"}},ra=l("",4),ya=l("",56),Ea={class:"language-bash vp-adaptive-theme"},ia=s("button",{title:"Copy Code",class:"copy"},null,-1),da=s("span",{class:"lang"},"bash",-1),Fa={class:"shiki github-dark vp-code-dark"},ua={class:"line"},ha=s("span",{style:{color:"#F97583"}},"export",-1),Ca=s("span",{style:{color:"#E1E4E8"}}," NODE_STORE",-1),ga=s("span",{style:{color:"#F97583"}},"=",-1),ba=s("span",{style:{color:"#E1E4E8"}},"$HOME",-1),Ba={style:{color:"#9ECBFF"}},qa={class:"shiki github-light vp-code-light"},va={class:"line"},ma=s("span",{style:{color:"#D73A49"}},"export",-1),ka=s("span",{style:{color:"#24292E"}}," NODE_STORE",-1),Aa=s("span",{style:{color:"#D73A49"}},"=",-1),_a=s("span",{style:{color:"#24292E"}},"$HOME",-1),fa={style:{color:"#032F62"}},Da=l("",36),wa=l("",95),Ha=JSON.parse('{"title":"Celestia-node RPC CLI tutorial","description":"","frontmatter":{"next":{"text":"Prompt Scavenger","link":"/developers/prompt-scavenger"},"head":[["meta",{"name":"og:title","content":"Celestia-node RPC CLI tutorial | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/node-tutorial.md","filePath":"developers/node-tutorial.md","lastUpdated":1713191709000}'),Ta={name:"developers/node-tutorial.md"},Ua=Object.assign(Ta,{setup(xa){return(Oa,Ra)=>(r(),y("div",null,[E,s("ol",null,[s("li",null,[i,s("div",d,[F,s("div",u,[s("div",h,[C,g,s("pre",b,[s("code",null,[s("span",B,[q,v,s("span",m,'"'+a(n(o).golangNodeMainnet)+'"',1)])])]),s("pre",k,[s("code",null,[s("span",A,[_,f,s("span",D,'"'+a(n(o).golangNodeMainnet)+'"',1)])])])]),s("div",w,[T,x,s("pre",O,[s("code",null,[s("span",R,[P,I,s("span",S,'"'+a(n(o).golangNodeMocha)+'"',1)])])]),s("pre",N,[s("code",null,[s("span",$,[H,U,s("span",M,'"'+a(n(o).golangNodeMocha)+'"',1)])])])]),s("div",j,[z,L,s("pre",V,[s("code",null,[s("span",G,[W,Y,s("span",K,'"'+a(n(o).golangNodeArabica)+'"',1)])])]),s("pre",X,[s("code",null,[s("span",J,[Q,Z,s("span",ss,'"'+a(n(o).golangNodeArabica)+'"',1)])])])])])])]),as]),ns,os,ls,ps,es,s("ol",null,[ts,s("li",null,[cs,s("div",rs,[ys,s("div",Es,[s("div",is,[ds,Fs,s("pre",us,[s("code",null,[s("span",hs,[Cs,gs,bs,Bs,s("span",qs,"tags/"+a(n(c)["node-latest-tag"]),1)])])]),s("pre",vs,[s("code",null,[s("span",ms,[ks,As,_s,fs,s("span",Ds,"tags/"+a(n(c)["node-latest-tag"]),1)])])])]),s("div",ws,[Ts,xs,s("pre",Os,[s("code",null,[s("span",Rs,[Ps,Is,Ss,Ns,s("span",$s,"tags/"+a(n(t)["node-latest-tag"]),1)])])]),s("pre",Hs,[s("code",null,[s("span",Us,[Ms,js,zs,Ls,s("span",Vs,"tags/"+a(n(t)["node-latest-tag"]),1)])])])]),s("div",Gs,[Ws,Ys,s("pre",Ks,[s("code",null,[s("span",Xs,[Js,Qs,Zs,sa,s("span",aa,"tags/"+a(n(e)["node-latest-tag"]),1)])])]),s("pre",na,[s("code",null,[s("span",oa,[la,pa,ea,ta,s("span",ca,"tags/"+a(n(e)["node-latest-tag"]),1)])])])])])])]),ra]),ya,s("p",null,"To set your node store for a light node on "+a(n(o).mochaChainId)+", you can use the following command:",1),s("div",Ea,[ia,da,s("pre",Fa,[s("code",null,[s("span",ua,[ha,Ca,ga,ba,s("span",Ba,"/.celestia-light-"+a(n(o).mochaChainId),1)])])]),s("pre",qa,[s("code",null,[s("span",va,[ma,ka,Aa,_a,s("span",fa,"/.celestia-light-"+a(n(o).mochaChainId),1)])])])]),Da,s("p",null,[p("Here is an example command to retrieve the data from above, on "),s("code",null,a(n(o).arabicaChainId),1),p(":")]),wa]))}});export{Ha as __pageData,Ua as default}; diff --git a/assets/developers_optimism-devnet.md.3b7c5762.js b/assets/developers_optimism-devnet.md.3b7c5762.js new file mode 100644 index 00000000000..a10f65fdbe7 --- /dev/null +++ b/assets/developers_optimism-devnet.md.3b7c5762.js @@ -0,0 +1,131 @@ +import{_ as s,o as a,c as n,Q as e}from"./chunks/framework.51d6c45b.js";const E=JSON.parse('{"title":"Deploy an OP Stack devnet","description":"Start your own devnet with a modified version of optimism-bedrock.","frontmatter":{"description":"Start your own devnet with a modified version of optimism-bedrock.","head":[["meta",{"name":"og:title","content":"Deploy an OP Stack devnet | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/optimism-devnet.md","filePath":"developers/optimism-devnet.md","lastUpdated":1713191709000}'),l={name:"developers/optimism-devnet.md"},o=e(`

    Deploy an OP Stack devnet

    This guide will show you how to run your own OP Stack devnet.

    Dependency setup

    Environment setup and Golang installation

    First, install dependencies for Celestia software and for OP Stack.

    Clone repository

    Next, clone the repo:

    bash
    cd $HOME
    +git clone https://github.com/celestiaorg/optimism
    +cd optimism
    cd $HOME
    +git clone https://github.com/celestiaorg/optimism
    +cd optimism

    Check out to the version for either the stable version or upstream version:

    bash
    git checkout tags/v1.2.0-OP_v1.7.0-CN_v0.12.4
    +git submodule update --init --recursive
    git checkout tags/v1.2.0-OP_v1.7.0-CN_v0.12.4
    +git submodule update --init --recursive
    bash
    git checkout celestia-develop
    +git submodule update --init --recursive
    git checkout celestia-develop
    +git submodule update --init --recursive

    Build devnet

    Build TypeScript definitions for TS dependencies:

    bash
    cd $HOME
    +cd optimism
    +make
    cd $HOME
    +cd optimism
    +make

    Set environment variables to start network:

    bash
    export SEQUENCER_BATCH_INBOX_ADDRESS=0xff00000000000000000000000000000000000000
    +export L2OO_ADDRESS=0x70997970C51812dc3A010C7d01b50e0d17dc79C8
    export SEQUENCER_BATCH_INBOX_ADDRESS=0xff00000000000000000000000000000000000000
    +export L2OO_ADDRESS=0x70997970C51812dc3A010C7d01b50e0d17dc79C8

    Start devnet

    Start the network by running:

    bash
    make devnet-up
    make devnet-up

    This starts up the layer 1 (ETH), layer 2 (op-geth), data availability layer (Celestia), the sequencer (op-node), batch submitter (op-batcher), state commitment service (op-proposer).

    View the logs of the devnet

    If you'd like to view the logs of the devnet, run the following command from the root of the Optimism directory:

    bash
    make devnet-logs
    make devnet-logs
    Optional: Docker tips

    Stop devnet

    If you'd like to start the network over, use the following command to safely shut down all of the containers:

    bash
    make devnet-down
    make devnet-down

    Then clean out the old config:

    bash
    make devnet-clean
    make devnet-clean

    Viewing containers

    To view the containers running, send:

    bash
    docker ps
    docker ps

    Find the container ID of the ops-bedrock_op-batcher_1 and run the following to follow the logs:

    bash
    docker logs -f <container-id>
    docker logs -f <container-id>

    In a new terminal, find the container ID of the ghcr.io/celestiaorg/local-celestia-devnet:main and run the following to follow the logs:

    bash
    docker logs -f <container-id>
    docker logs -f <container-id>

    You can do the same for other containers as you explore the stack.

    Find a transaction

    Now, we'll check for a recent transaction on the L1 with:

    bash
    cast block latest --rpc-url localhost:8545
    cast block latest --rpc-url localhost:8545

    Output of a block that contains a transaction will look like this:

    console
    baseFeePerGas        7
    +difficulty           2
    +extraData            0xd883010d04846765746888676f312e32312e33856c696e7578000000000000006b3afa42dce1f87f1f07a1ef569c4d43e41738ef93c865098bfa1458645f384e2e4498bcfe4ad9353ff1913a2e16162f496fafe5b0939a6c78fb5b503248d6da01
    +gasLimit             30000000
    +gasUsed              21568
    +hash                 0x1cb54d2369752ef73511c202ff9cdfd0eadf3a77b7aef0092bea63f2b5d57659
    +logsBloom            0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
    +miner                0x0000000000000000000000000000000000000000
    +mixHash              0x0000000000000000000000000000000000000000000000000000000000000000
    +nonce                0x0000000000000000
    +number               1141
    +parentHash           0x664bf4bb4a57dd5768a0a98991d77c58fb7a4e164c2581c79fb33ce9c3d4c250
    +receiptsRoot         0xaf8ff6af1180c8be9e4e8f3a5f882b3b227233f4abbefa479836d3721682a389
    +sealFields           []
    +sha3Uncles           0x1dcc4de8dec75d7aab85b567b6ccd41ad312451b948a7413f0a142fd40d49347
    +size                 767
    +stateRoot            0xd4b998a35d20d98ed3488221f0c161a0a9572d3de66399482553c8e3d2fae751
    +timestamp            1699638350
    +withdrawalsRoot
    +totalDifficulty      2283
    +transactions:        [
    +  0x79a0a7a1b4936aafe7a37dbfb07a6a9e55c145a4ed6fd54f962649b4b7db8de7
    +]
    baseFeePerGas        7
    +difficulty           2
    +extraData            0xd883010d04846765746888676f312e32312e33856c696e7578000000000000006b3afa42dce1f87f1f07a1ef569c4d43e41738ef93c865098bfa1458645f384e2e4498bcfe4ad9353ff1913a2e16162f496fafe5b0939a6c78fb5b503248d6da01
    +gasLimit             30000000
    +gasUsed              21568
    +hash                 0x1cb54d2369752ef73511c202ff9cdfd0eadf3a77b7aef0092bea63f2b5d57659
    +logsBloom            0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
    +miner                0x0000000000000000000000000000000000000000
    +mixHash              0x0000000000000000000000000000000000000000000000000000000000000000
    +nonce                0x0000000000000000
    +number               1141
    +parentHash           0x664bf4bb4a57dd5768a0a98991d77c58fb7a4e164c2581c79fb33ce9c3d4c250
    +receiptsRoot         0xaf8ff6af1180c8be9e4e8f3a5f882b3b227233f4abbefa479836d3721682a389
    +sealFields           []
    +sha3Uncles           0x1dcc4de8dec75d7aab85b567b6ccd41ad312451b948a7413f0a142fd40d49347
    +size                 767
    +stateRoot            0xd4b998a35d20d98ed3488221f0c161a0a9572d3de66399482553c8e3d2fae751
    +timestamp            1699638350
    +withdrawalsRoot
    +totalDifficulty      2283
    +transactions:        [
    +  0x79a0a7a1b4936aafe7a37dbfb07a6a9e55c145a4ed6fd54f962649b4b7db8de7
    +]

    Copy the transaction hash from transactions: <transaction-hash> and set it as a variable:

    bash
    export TX_HASH=0x79a0a7a1b4936aafe7a37dbfb07a6a9e55c145a4ed6fd54f962649b4b7db8de7
    export TX_HASH=0x79a0a7a1b4936aafe7a37dbfb07a6a9e55c145a4ed6fd54f962649b4b7db8de7

    Read the transaction call data

    Now read the transaction call data on the L1:

    bash
    cast tx $TX_HASH --rpc-url localhost:8545
    cast tx $TX_HASH --rpc-url localhost:8545

    The output will look similar to below:

    console
    blockHash            0x9f4dfae061b5ddd86f95a81be5daa0d7fe32e7f7f770f86dc375e0007d249bd2
    +blockNumber          24
    +from                 0x3C44CdDdB6a900fa2b585dd299e03d12FA4293BC
    +gas                  21572
    +gasPrice             1040676758
    +hash                 0xadd3a5dc0b8c605aeac891098e87cbaff43bb642896ebbf74f964c0690e46df2
    +input                0xce3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f
    +nonce                4
    +r                    0xaf5c1505c7dfcebca94d9a6a8c0caf99b6c87a8ed6d6c0b3161c9026f270a84f
    +s                    0x383ed2debf9f9055920cd7340418dda7e2bca6b989eb6992d83d123d4e322f2a
    +to                   0xFf00000000000000000000000000000000000901
    +transactionIndex     0
    +v                    0
    +value                0
    +yParity              0
    blockHash            0x9f4dfae061b5ddd86f95a81be5daa0d7fe32e7f7f770f86dc375e0007d249bd2
    +blockNumber          24
    +from                 0x3C44CdDdB6a900fa2b585dd299e03d12FA4293BC
    +gas                  21572
    +gasPrice             1040676758
    +hash                 0xadd3a5dc0b8c605aeac891098e87cbaff43bb642896ebbf74f964c0690e46df2
    +input                0xce3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f
    +nonce                4
    +r                    0xaf5c1505c7dfcebca94d9a6a8c0caf99b6c87a8ed6d6c0b3161c9026f270a84f
    +s                    0x383ed2debf9f9055920cd7340418dda7e2bca6b989eb6992d83d123d4e322f2a
    +to                   0xFf00000000000000000000000000000000000901
    +transactionIndex     0
    +v                    0
    +value                0
    +yParity              0

    TIP

    You are looking for a batcher transaction to the address 0xFf00000000000000000000000000000000000901.

    First, remove the prefix 0xce. Now, set the input as the INPUT variable and encode it as base64:

    bash
    export INPUT=ce3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f
    +export ENCODED_INPUT=$(echo "$INPUT" | xxd -r -p | base64)
    export INPUT=ce3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f
    +export ENCODED_INPUT=$(echo "$INPUT" | xxd -r -p | base64)

    TIP

    Remember to remove the 0xce prefix!

    Find the data on Celestia

    Clone the go-da repository:

    bash
    cd $HOME
    +git clone https://github.com/rollkit/go-da.git
    +cd go-da/proto/da
    cd $HOME
    +git clone https://github.com/rollkit/go-da.git
    +cd go-da/proto/da

    Now, from go-da/proto/da run:

    bash
    grpcurl -proto da.proto -plaintext -d "{\\"ids\\": [{\\"value\\": \\"$ENCODED_INPUT\\"}]}" 127.0.0.1:26650 da.DAService.Get
    grpcurl -proto da.proto -plaintext -d "{\\"ids\\": [{\\"value\\": \\"$ENCODED_INPUT\\"}]}" 127.0.0.1:26650 da.DAService.Get

    Your result will look similar to the below!

    console
    {
    +  "blobs": [
    +    {
    +      "value": "AKUumhJ8FnuyVrBs38FDKEIAAAAAAZB42trhw/DDc4GFAlv4klkv5Zh4E16mmO5fpNOS1f5wzpds8YK3S0Rvs4ULLJj13euw+Ovdv6Q23zuV1ShROEvk5aptIT7bGmZunvc1OiKwJTXVbN0BiGm6k2zNWq78cNsT2ez3+nzQq84Ds28or/aKz/o1w4NpV7w4caZtgJomX71w96m63+xzYnarXLu7WWvRrwbeb6cW8R93YHXt1r4+TXCBGVe76obzf5JLTNu22gksD2cL+83D8DGjX0FKcwZD0VofkGmboKY1uTddu8704s2MwgNNe09s1bzw+n9Fq6fKFw7pvwJL200eCS0oFJ3HfPAEywnlgyyGQc89dh+98GD5TrdU4aNql9afmW+sDzJtC9S0fzLWYROOS0bvK3W7EvNpmWXe5qrdzKlBmv1LZi4ofrrxLHGmbYOaJhHsEn+B81lGUh33HDet8K9nVKKSF2+W3Xul6uPSxydPBwsv2GHskR+yfUlDbvyl1ROTvtS1zXlpEPz0M1e/RIIt57fVj0Gm7TgACAAA//+Qdel2AQ=="
    +    }
    +  ]
    +}
    {
    +  "blobs": [
    +    {
    +      "value": "AKUumhJ8FnuyVrBs38FDKEIAAAAAAZB42trhw/DDc4GFAlv4klkv5Zh4E16mmO5fpNOS1f5wzpds8YK3S0Rvs4ULLJj13euw+Ovdv6Q23zuV1ShROEvk5aptIT7bGmZunvc1OiKwJTXVbN0BiGm6k2zNWq78cNsT2ez3+nzQq84Ds28or/aKz/o1w4NpV7w4caZtgJomX71w96m63+xzYnarXLu7WWvRrwbeb6cW8R93YHXt1r4+TXCBGVe76obzf5JLTNu22gksD2cL+83D8DGjX0FKcwZD0VofkGmboKY1uTddu8704s2MwgNNe09s1bzw+n9Fq6fKFw7pvwJL200eCS0oFJ3HfPAEywnlgyyGQc89dh+98GD5TrdU4aNql9afmW+sDzJtC9S0fzLWYROOS0bvK3W7EvNpmWXe5qrdzKlBmv1LZi4ofrrxLHGmbYOaJhHsEn+B81lGUh33HDet8K9nVKKSF2+W3Xul6uPSxydPBwsv2GHskR+yfUlDbvyl1ROTvtS1zXlpEPz0M1e/RIIt57fVj0Gm7TgACAAA//+Qdel2AQ=="
    +    }
    +  ]
    +}

    Ethereum fallback mechanism in OP Stack

    The Ethereum fallback mechanism allows rollups to "fall back" to Ethereum or another EVM chain in the case of downtime or errors submitting data to Celestia.

    Implementation of fallback

    The Ethereum fallback mechanism is implemented in the celestiaorg/optimism v1.1.0 release.

    The op-batcher/batcher/driver.go and op-node/rollup/derive/calldata_source.go files are part of the Ethereum fallback mechanism in the op-batcher and op-node respectively.

    In driver.go, the calldataTxCandidate function is responsible for the write path of the Ethereum fallback. This function creates and submits a transaction to the batch inbox address with the given data. It uses the underlying txmgr to handle transaction sending and gas price management.

    If the transaction data can be published as a blob to Celestia, it replaces the calldata with a blob identifier and sends the transaction with this data. If it cannot be published to Celestia, it falls back to Ethereum without any change to the transaction.

    The blob identifier starts with the special prefix 0xce, which was chosen as a mnemonic for Celestia, and indicates that the remaining data has to interpreted as a little-endian encoded Block Height (8 bytes) and Blob Commitment (32 bytes). The combination of these can later be used to retrieve the original calldata from Celestia.

    Prefix8 bytes32 bytes
    0xceBlock HeightBlob Commitment
    go
    func (l *BatchSubmitter) sendTransaction(
    +    txdata txData,
    +    queue *txmgr.Queue[txData],
    +    receiptsCh chan txmgr.TxReceipt[txData],
    +) {
    +    // ...
    +}
    func (l *BatchSubmitter) sendTransaction(
    +    txdata txData,
    +    queue *txmgr.Queue[txData],
    +    receiptsCh chan txmgr.TxReceipt[txData],
    +) {
    +    // ...
    +}

    In calldata_source.go, the DataFromEVMTransactions function defines the read path of the Ethereum fallback. This function filters all of the transactions and returns the calldata from transactions that are sent to the batch inbox address from the batch sender address.

    If the calldata matches the version prefix 0xce, it is decoded as a blob identifier, the original calldata is retrieved from Celestia and returned for derivation. If the calldata does not match the prefix, the entire calldata is returned for derivation.

    go
    func DataFromEVMTransactions(
    +    config *rollup.Config,
    +    batcherAddr common.Address,
    +    txs types.Transactions,
    +    log log.Logger
    +) ([]eth.Data, error) {
    +    // ...
    +}
    func DataFromEVMTransactions(
    +    config *rollup.Config,
    +    batcherAddr common.Address,
    +    txs types.Transactions,
    +    log log.Logger
    +) ([]eth.Data, error) {
    +    // ...
    +}

    These two functions work together to ensure that the Ethereum fallback mechanism operates correctly, allowing the rollup to continue functioning even during periods of downtime on Celestia.

    Testing the fallback

    Testing out the Ethereum fallback mechanism can be done with the go-da tool. Triggering a simultaneous blob transaction will cause the op-batcher blob transaction to fail, with an incorrect account sequence error, which triggers a fallback to Ethereum.

    To trigger the transaction, send this command from the same go/proto/da directory:

    bash
    grpcurl -proto da.proto -plaintext -d '{"blobs": [{"value": "SGVsbG8gd28ybGQh"}]}' 127.0.0.1:26650 da.DAService.Submit
    grpcurl -proto da.proto -plaintext -d '{"blobs": [{"value": "SGVsbG8gd28ybGQh"}]}' 127.0.0.1:26650 da.DAService.Submit

    Alternatively, you can shut off the local-celestia-devnet and see that the OP Stack devnet logs show that the rollup has fallen back to the L1, in this case Ethereum, for posting data.

    Span batches

    Span batches can be enabled by setting OP_BATCHER_BATCH_TYPE: 1 in your docker-compose.yml file.

    Note that this requires the Delta activation time to be configured. For your devnet, you should set "l2GenesisDeltaTimeOffset": "0x0", in devnetL1-template.json. This will enable span batches and can be tested by grepping docker compose logs -f | grep batch_type which should include batch_type=SpanBatch and batch_type=1.

    `,68),p=[o];function t(c,r,i,d,y,h){return a(),n("div",null,p)}const u=s(l,[["render",t]]);export{E as __pageData,u as default}; diff --git a/assets/developers_optimism-devnet.md.3b7c5762.lean.js b/assets/developers_optimism-devnet.md.3b7c5762.lean.js new file mode 100644 index 00000000000..6848b75f1d7 --- /dev/null +++ b/assets/developers_optimism-devnet.md.3b7c5762.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as e}from"./chunks/framework.51d6c45b.js";const E=JSON.parse('{"title":"Deploy an OP Stack devnet","description":"Start your own devnet with a modified version of optimism-bedrock.","frontmatter":{"description":"Start your own devnet with a modified version of optimism-bedrock.","head":[["meta",{"name":"og:title","content":"Deploy an OP Stack devnet | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/optimism-devnet.md","filePath":"developers/optimism-devnet.md","lastUpdated":1713191709000}'),l={name:"developers/optimism-devnet.md"},o=e("",68),p=[o];function t(c,r,i,d,y,h){return a(),n("div",null,p)}const u=s(l,[["render",t]]);export{E as __pageData,u as default}; diff --git a/assets/developers_optimism.md.8200414c.js b/assets/developers_optimism.md.8200414c.js new file mode 100644 index 00000000000..5f10599db7b --- /dev/null +++ b/assets/developers_optimism.md.8200414c.js @@ -0,0 +1,55 @@ +import{c as a}from"./chunks/constants.dc70dfe7.js";import{o as l,c as t,k as s,t as e,l as o,Q as n}from"./chunks/framework.51d6c45b.js";const p=n('

    Deploy an OP Stack devnet to Celestia

    In order to deploy a devnet to Celestia, you will need to have a modified version of optimism-bedrock. Refer to the steps to install dependencies and the modified version of OP Stack for your environment setup.

    Pick your deployment type

    Using Celestia and OP stack, you have the option to either run a light node of your own or a local-celestia-devnet, both of which will give you a local devnet to test things out with.

    Using a local devnet

    If you'd like to use the local-celestia-devnet, you're in luck! This is the default for the OP Stack + Celestia repository. Head to the previous page to get started.

    Using a light node

    This is a beta integration and we are working on resolving open issues.

    In order to allow your light node to post and retrieve data without errors, you will need to change UseShareExchange to false in:

    ',9),c={class:"vp-code-group vp-adaptive-theme"},i=n('
    ',1),r={class:"blocks"},d=n('
    bash
    $HOME/.celestia-light/config.toml
    $HOME/.celestia-light/config.toml
    ',1),h={class:"language-bash vp-adaptive-theme"},y=s("button",{title:"Copy Code",class:"copy"},null,-1),E=s("span",{class:"lang"},"bash",-1),u={class:"shiki github-dark vp-code-dark"},g={class:"line"},_={style:{color:"#E1E4E8"}},v={class:"shiki github-light vp-code-light"},f={class:"line"},m={style:{color:"#24292E"}},b={class:"language-bash vp-adaptive-theme"},k=s("button",{title:"Copy Code",class:"copy"},null,-1),C=s("span",{class:"lang"},"bash",-1),F={class:"shiki github-dark vp-code-dark"},T={class:"line"},w={style:{color:"#E1E4E8"}},B={class:"shiki github-light vp-code-light"},S={class:"line"},A={style:{color:"#24292E"}},P=n('

    If you choose to use your own node store, the light node must be fully synced and funded for you to be able to submit and retrieve PayForBlobs to a Celestia network.

    If it is not synced, you will run into errors similar to this.

    Visit the Arabica or Mocha pages to visit their faucets.

    In order to mount existing data, you must have a node store that is in this directory:

    ',4),q={class:"vp-code-group vp-adaptive-theme"},D=n('
    ',1),O={class:"blocks"},I=n('
    bash
    $HOME/.celestia-light
    $HOME/.celestia-light
    ',1),M={class:"language-bash vp-adaptive-theme"},H=s("button",{title:"Copy Code",class:"copy"},null,-1),R=s("span",{class:"lang"},"bash",-1),x={class:"shiki github-dark vp-code-dark"},N={class:"line"},V={style:{color:"#E1E4E8"}},$={class:"shiki github-light vp-code-light"},L={class:"line"},U={style:{color:"#24292E"}},G={class:"language-bash vp-adaptive-theme"},K=s("button",{title:"Copy Code",class:"copy"},null,-1),Q=s("span",{class:"lang"},"bash",-1),W={class:"shiki github-dark vp-code-dark"},X={class:"line"},Y={style:{color:"#E1E4E8"}},j={class:"shiki github-light vp-code-light"},Z={class:"line"},z={style:{color:"#24292E"}},J=n(`

    This is the default location of the node store when you initialize and run a new Celestia node.

    By default, the node will run with the account named my_celes_key.

    If you have your own setup you'd like to try, you can always edit optimism/ops-bedrock/docker-compose.yml to work with your setup.

    Using a RaaS provider

    If you'd like to use a Rollups as a Service (RaaS) provider, you can do so by going to the RaaS category in the menu.

    Build the devnet

    Build TypeScript definitions for TS dependencies:

    bash
    cd $HOME
    +cd optimism
    +make
    cd $HOME
    +cd optimism
    +make

    Set environment variables to start network:

    bash
    export SEQUENCER_BATCH_INBOX_ADDRESS=0xff00000000000000000000000000000000000000
    +export L2OO_ADDRESS=0x70997970C51812dc3A010C7d01b50e0d17dc79C8
    export SEQUENCER_BATCH_INBOX_ADDRESS=0xff00000000000000000000000000000000000000
    +export L2OO_ADDRESS=0x70997970C51812dc3A010C7d01b50e0d17dc79C8

    Start the devnet

    First, make sure your light node is synced and funded. It must not be running for this example to work.

    This example is for Mainnet Beta. You can modify the da: section of your $HOME/optimism/ops-bedrock/docker-compose.yml for your specific use, similarly to the example below:

    This setup will use celestia-da, which is celestia-node with a DA server on port 26650.

    For the P2P_NETWORK variable, you'll need to supply the network of choice, either celestia, mocha, or arabica. Using celestia, the volume path will be just .celestia-light instead of .celestia-light-<network>. You will also need to provide a core.ip RPC URL for the network you are using.

    yaml
    da:
    +  image: ghcr.io/rollkit/local-celestia-devnet:v0.12.1 
    +  image: ghcr.io/rollkit/celestia-da:v0.12.9 
    +  command: > 
    +    celestia-da light start 
    +    --p2p.network=<network> 
    +    --da.grpc.namespace=000008e5f679bf7116cb 
    +    --da.grpc.listen=0.0.0.0:26650 
    +    --core.ip <rpc-url> 
    +    --gateway 
    +  environment: 
    +      - NODE_TYPE=light 
    +      - P2P_NETWORK=<network> 
    +  ports:
    +    - "26650:26650"
    +    - "26658:26658"
    +    - "26659:26659"
    +  volumes: 
    +    - $HOME/.celestia-light-<network>/:/home/celestia/.celestia-light-<network>/ 
    +  healthcheck:
    +    test: ["CMD", "curl", "-f", "http://localhost:26659/header/1"]
    +    interval: 10s
    +    timeout: 5s
    +    retries: 5
    +    start_period: 30s
    da:
    +  image: ghcr.io/rollkit/local-celestia-devnet:v0.12.1 
    +  image: ghcr.io/rollkit/celestia-da:v0.12.9 
    +  command: > 
    +    celestia-da light start 
    +    --p2p.network=<network> 
    +    --da.grpc.namespace=000008e5f679bf7116cb 
    +    --da.grpc.listen=0.0.0.0:26650 
    +    --core.ip <rpc-url> 
    +    --gateway 
    +  environment: 
    +      - NODE_TYPE=light 
    +      - P2P_NETWORK=<network> 
    +  ports:
    +    - "26650:26650"
    +    - "26658:26658"
    +    - "26659:26659"
    +  volumes: 
    +    - $HOME/.celestia-light-<network>/:/home/celestia/.celestia-light-<network>/ 
    +  healthcheck:
    +    test: ["CMD", "curl", "-f", "http://localhost:26659/header/1"]
    +    interval: 10s
    +    timeout: 5s
    +    retries: 5
    +    start_period: 30s

    Now start the devnet:

    bash
    make devnet-up
    make devnet-up

    View the logs of the devnet

    If you'd like to view the logs of the devnet, run the following command from the root of the Optimism directory:

    bash
    make devnet-logs
    make devnet-logs

    Stop the devnet

    To safely stop the devnet, run the following command:

    bash
    make devnet-down
    make devnet-down

    Clean the devnet

    To remove all data from the devnet, run the following command:

    bash
    make devnet-clean
    make devnet-clean

    Deploying to an L1 (or L2)

    If you'd like to deploy to an EVM L1 or L2, reference the OP stack deployment guide.

    `,29),ts=JSON.parse('{"title":"Deploy an OP Stack devnet to Celestia","description":"Start your own devnet with a modified version of optimism-bedrock.","frontmatter":{"description":"Start your own devnet with a modified version of optimism-bedrock.","next":{"text":"Rollkit","link":"/developers/rollkit"},"head":[["meta",{"name":"og:title","content":"Deploy an OP Stack devnet to Celestia | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/optimism.md","filePath":"developers/optimism.md","lastUpdated":1713191709000}'),ss={name:"developers/optimism.md"},ps=Object.assign(ss,{setup(as){return(es,os)=>(l(),t("div",null,[p,s("div",c,[i,s("div",r,[d,s("div",h,[y,E,s("pre",u,[s("code",null,[s("span",g,[s("span",_,"$HOME/.celestia-light-"+e(o(a).mochaChainId)+"/config.toml",1)])])]),s("pre",v,[s("code",null,[s("span",f,[s("span",m,"$HOME/.celestia-light-"+e(o(a).mochaChainId)+"/config.toml",1)])])])]),s("div",b,[k,C,s("pre",F,[s("code",null,[s("span",T,[s("span",w,"$HOME/.celestia-light-"+e(o(a).arabicaChainId)+"/config.toml",1)])])]),s("pre",B,[s("code",null,[s("span",S,[s("span",A,"$HOME/.celestia-light-"+e(o(a).arabicaChainId)+"/config.toml",1)])])])])])]),P,s("div",q,[D,s("div",O,[I,s("div",M,[H,R,s("pre",x,[s("code",null,[s("span",N,[s("span",V,"$HOME/.celestia-light-"+e(o(a).mochaChainId),1)])])]),s("pre",$,[s("code",null,[s("span",L,[s("span",U,"$HOME/.celestia-light-"+e(o(a).mochaChainId),1)])])])]),s("div",G,[K,Q,s("pre",W,[s("code",null,[s("span",X,[s("span",Y,"$HOME/.celestia-light-"+e(o(a).arabicaChainId),1)])])]),s("pre",j,[s("code",null,[s("span",Z,[s("span",z,"$HOME/.celestia-light-"+e(o(a).arabicaChainId),1)])])])])])]),J]))}});export{ts as __pageData,ps as default}; diff --git a/assets/developers_optimism.md.8200414c.lean.js b/assets/developers_optimism.md.8200414c.lean.js new file mode 100644 index 00000000000..0b10a747018 --- /dev/null +++ b/assets/developers_optimism.md.8200414c.lean.js @@ -0,0 +1 @@ +import{c as a}from"./chunks/constants.dc70dfe7.js";import{o as l,c as t,k as s,t as e,l as o,Q as n}from"./chunks/framework.51d6c45b.js";const p=n("",9),c={class:"vp-code-group vp-adaptive-theme"},i=n("",1),r={class:"blocks"},d=n("",1),h={class:"language-bash vp-adaptive-theme"},y=s("button",{title:"Copy Code",class:"copy"},null,-1),E=s("span",{class:"lang"},"bash",-1),u={class:"shiki github-dark vp-code-dark"},g={class:"line"},_={style:{color:"#E1E4E8"}},v={class:"shiki github-light vp-code-light"},f={class:"line"},m={style:{color:"#24292E"}},b={class:"language-bash vp-adaptive-theme"},k=s("button",{title:"Copy Code",class:"copy"},null,-1),C=s("span",{class:"lang"},"bash",-1),F={class:"shiki github-dark vp-code-dark"},T={class:"line"},w={style:{color:"#E1E4E8"}},B={class:"shiki github-light vp-code-light"},S={class:"line"},A={style:{color:"#24292E"}},P=n("",4),q={class:"vp-code-group vp-adaptive-theme"},D=n("",1),O={class:"blocks"},I=n("",1),M={class:"language-bash vp-adaptive-theme"},H=s("button",{title:"Copy Code",class:"copy"},null,-1),R=s("span",{class:"lang"},"bash",-1),x={class:"shiki github-dark vp-code-dark"},N={class:"line"},V={style:{color:"#E1E4E8"}},$={class:"shiki github-light vp-code-light"},L={class:"line"},U={style:{color:"#24292E"}},G={class:"language-bash vp-adaptive-theme"},K=s("button",{title:"Copy Code",class:"copy"},null,-1),Q=s("span",{class:"lang"},"bash",-1),W={class:"shiki github-dark vp-code-dark"},X={class:"line"},Y={style:{color:"#E1E4E8"}},j={class:"shiki github-light vp-code-light"},Z={class:"line"},z={style:{color:"#24292E"}},J=n("",29),ts=JSON.parse('{"title":"Deploy an OP Stack devnet to Celestia","description":"Start your own devnet with a modified version of optimism-bedrock.","frontmatter":{"description":"Start your own devnet with a modified version of optimism-bedrock.","next":{"text":"Rollkit","link":"/developers/rollkit"},"head":[["meta",{"name":"og:title","content":"Deploy an OP Stack devnet to Celestia | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/optimism.md","filePath":"developers/optimism.md","lastUpdated":1713191709000}'),ss={name:"developers/optimism.md"},ps=Object.assign(ss,{setup(as){return(es,os)=>(l(),t("div",null,[p,s("div",c,[i,s("div",r,[d,s("div",h,[y,E,s("pre",u,[s("code",null,[s("span",g,[s("span",_,"$HOME/.celestia-light-"+e(o(a).mochaChainId)+"/config.toml",1)])])]),s("pre",v,[s("code",null,[s("span",f,[s("span",m,"$HOME/.celestia-light-"+e(o(a).mochaChainId)+"/config.toml",1)])])])]),s("div",b,[k,C,s("pre",F,[s("code",null,[s("span",T,[s("span",w,"$HOME/.celestia-light-"+e(o(a).arabicaChainId)+"/config.toml",1)])])]),s("pre",B,[s("code",null,[s("span",S,[s("span",A,"$HOME/.celestia-light-"+e(o(a).arabicaChainId)+"/config.toml",1)])])])])])]),P,s("div",q,[D,s("div",O,[I,s("div",M,[H,R,s("pre",x,[s("code",null,[s("span",N,[s("span",V,"$HOME/.celestia-light-"+e(o(a).mochaChainId),1)])])]),s("pre",$,[s("code",null,[s("span",L,[s("span",U,"$HOME/.celestia-light-"+e(o(a).mochaChainId),1)])])])]),s("div",G,[K,Q,s("pre",W,[s("code",null,[s("span",X,[s("span",Y,"$HOME/.celestia-light-"+e(o(a).arabicaChainId),1)])])]),s("pre",j,[s("code",null,[s("span",Z,[s("span",z,"$HOME/.celestia-light-"+e(o(a).arabicaChainId),1)])])])])])]),J]))}});export{ts as __pageData,ps as default}; diff --git a/assets/developers_prompt-scavenger.md.27b40ddb.js b/assets/developers_prompt-scavenger.md.27b40ddb.js new file mode 100644 index 00000000000..3d70b626d45 --- /dev/null +++ b/assets/developers_prompt-scavenger.md.27b40ddb.js @@ -0,0 +1,275 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.51d6c45b.js";const F=JSON.parse('{"title":"Prompt scavenger","description":"Learn how to interact with the Celestia Node API with this tutorial.","frontmatter":{"description":"Learn how to interact with the Celestia Node API with this tutorial.","prev":{"text":"Node RPC CLI tutorial","link":"/developers/node-tutorial"},"head":[["meta",{"name":"og:title","content":"Prompt scavenger | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/prompt-scavenger.md","filePath":"developers/prompt-scavenger.md","lastUpdated":1713191709000}'),p={name:"developers/prompt-scavenger.md"},o=l(`

    Prompt scavenger

    Note

    This tutorial needs to be updated.

    Welcome to the world of Prompt Scavenger, a game where you’ll be using Celestia’s Node API and OpenAI’s GPT-3.5 to decode hidden messages scattered throughout Celestia’s blockchain. In this tutorial, we’ll be using Golang to write the code for the game.

    Through this tutorial, you’ll gain experience using Celestia’s Node API to fetch data from the blockchain, process it, and submit new transactions with that data. You’ll also learn how to integrate OpenAI’s GPT-3.5 API to generate fun responses based on the data you’ve found.

    So if you’re ready to embark on an adventure that combines blockchain technology with the power of AI, and learn some Golang along the way, let’s get started!

    Dependencies

    The following dependencies are needed to be installed or obtained:

    Install Celestia Node and run a light node

    First, install the celestia-node binary.

    Let’s run the following commands to get our env vars setup (NOTE: For CORE_IP you can select from the list of available RPC endpoints on the Blockspace Race page:

    sh
    KEYNAME="scavenger_key"
    +NODETYPE="light"
    +NETWORK="blockspacerace"
    +AUTHTYPE="admin"
    +CORE_IP=""
    KEYNAME="scavenger_key"
    +NODETYPE="light"
    +NETWORK="blockspacerace"
    +AUTHTYPE="admin"
    +CORE_IP=""

    Next, let’s generate a wallet key for our light node:

    sh
    ./cel-key add $KEYNAME --keyring-backend test --node.type $NODETYPE --p2p.network $NETWORK
    ./cel-key add $KEYNAME --keyring-backend test --node.type $NODETYPE --p2p.network $NETWORK

    Be sure to save the mnemonics and your celestia1 public address in a safe place.

    Then head over to our Discord Server and request tokens from the #faucet channel under Blockspace Race.

    You can track receiving the funds on the explorer here: Interchain Explorer by Cosmostation

    Just make sure to paste your celestia1**** address in order to look it up.

    We will be running this version of Celestia Node with Blockspace Race test network. First, let’s initialize our node:

    sh
    celestia light init --p2p.network blockspacerace
    celestia light init --p2p.network blockspacerace

    Next, we will start our node:

    sh
    celestia light start --core.ip $CORE_IP --p2p.network $NETWORK --gateway.deprecated-endpoints --gateway --gateway.addr 127.0.0.1 --gateway.port 26659 --keyring.accname $KEYNAME
    celestia light start --core.ip $CORE_IP --p2p.network $NETWORK --gateway.deprecated-endpoints --gateway --gateway.addr 127.0.0.1 --gateway.port 26659 --keyring.accname $KEYNAME

    TIP

    The --core.ip gRPC port defaults to 9090, so if you do not specify it in the command line, it will default to that port. You can add the port after the IP address or use the --core.grpc.port flag to specify another port if you prefer.

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    You should now have a running light node on your machine. The rest of the tutorial will assume you will be building the script and running it where the light node is in your localhost.

    Node API authentication key

    On the same machine where your Celestia light node is running, run the following:

    sh
    export AUTH_TOKEN=$(celestia $NODETYPE auth $AUTHTYPE)
    export AUTH_TOKEN=$(celestia $NODETYPE auth $AUTHTYPE)

    Now run the following to get the auth token for your node:

    sh
    echo $AUTH_TOKEN
    echo $AUTH_TOKEN

    This will be used for the env var file we setup later.

    OpenAI key

    Make sure to go over to OpenAI in order to sign up for an account and generate an OpenAI API key. The key will be needed to communicate with OpenAI.

    Building the Prompt Scavenger

    We will first need to setup a .env file with the following pasted in:

    txt
    NODE_RPC_IP="http://localhost:26658"
    +NODE_JWT_TOKEN=""
    +OPENAI_KEY=""
    +NAMESPACE_ID="00000000ce1e571a"
    NODE_RPC_IP="http://localhost:26658"
    +NODE_JWT_TOKEN=""
    +OPENAI_KEY=""
    +NAMESPACE_ID="00000000ce1e571a"

    The OPENAI_KEY is the API key you got from Open AI. For NODE_RPC_IP, it is assumed to be the local host but it can point to a light node that is remote. NODE_JWT_TOKEN is the AUTH_TOKEN you generated earlier. For Namespace ID, we made a sample one for you to use but you can generate your own.

    Now, let’s build!

    Copy over Go files

    Run the following:

    sh
    mkdir test_scavenger
    +git clone https://github.com/celestiaorg/PromptScavenger.git
    +cp PromptScavenger/go.mod test_scavenger/
    +cp PromptScavenger/go.sum test_scavenger/
    +cd test_scavenger
    mkdir test_scavenger
    +git clone https://github.com/celestiaorg/PromptScavenger.git
    +cp PromptScavenger/go.mod test_scavenger/
    +cp PromptScavenger/go.sum test_scavenger/
    +cd test_scavenger

    This will copy over the required go.sum and go.mod files to a new directory we will use to import the Node API golang library.

    Build your import statements

    Inside the directory, create a main.gofile and setup the import statements:

    go
    package main
    +
    +import (
    +        "context"
    +        "fmt"
    +        "log"
    +        "os"
    +
    +        "github.com/celestiaorg/celestia-node/api/rpc/client"
    +        nodeheader "github.com/celestiaorg/celestia-node/header"
    +        "github.com/celestiaorg/nmt/namespace"
    +        "github.com/joho/godotenv"
    +        cosmosmath "cosmossdk.io/math"
    +        openai "github.com/sashabaranov/go-openai"
    +        "encoding/base64"
    +        "encoding/hex"
    +)
    +
    +func main() {
    +  // TODO
    +}
    package main
    +
    +import (
    +        "context"
    +        "fmt"
    +        "log"
    +        "os"
    +
    +        "github.com/celestiaorg/celestia-node/api/rpc/client"
    +        nodeheader "github.com/celestiaorg/celestia-node/header"
    +        "github.com/celestiaorg/nmt/namespace"
    +        "github.com/joho/godotenv"
    +        cosmosmath "cosmossdk.io/math"
    +        openai "github.com/sashabaranov/go-openai"
    +        "encoding/base64"
    +        "encoding/hex"
    +)
    +
    +func main() {
    +  // TODO
    +}

    Here we setup all required libraries we need to use plus the main function that we will use for our game.

    Helpful functions

    First we will need to create some helpful functions that you will need later.

    go
    // loadEnv loads environment variables from the .env file.
    +func loadEnv() {
    +        err := godotenv.Load(".env")
    +        if err != nil {
    +                log.Fatal("Error loading .env file")
    +        }
    +}
    // loadEnv loads environment variables from the .env file.
    +func loadEnv() {
    +        err := godotenv.Load(".env")
    +        if err != nil {
    +                log.Fatal("Error loading .env file")
    +        }
    +}

    The loadEnv function allows us to load up our .env file which has all the necessary env vars needed.

    Next, let’s create a helper function that allows us to load an instance of the Celestia Node client given the correct env vars passed to it:

    go
    // createClient initializes a new Celestia node client.
    +func createClient(ctx context.Context) *client.Client {
    +        nodeRPCIP := os.Getenv("NODE_RPC_IP")
    +        jwtToken := os.Getenv("NODE_JWT_TOKEN")
    +
    +        rpc, err := client.NewClient(ctx, nodeRPCIP, jwtToken)
    +        if err != nil {
    +                log.Fatalf("Error creating client: %v", err)
    +        }
    +
    +        return rpc
    +}
    // createClient initializes a new Celestia node client.
    +func createClient(ctx context.Context) *client.Client {
    +        nodeRPCIP := os.Getenv("NODE_RPC_IP")
    +        jwtToken := os.Getenv("NODE_JWT_TOKEN")
    +
    +        rpc, err := client.NewClient(ctx, nodeRPCIP, jwtToken)
    +        if err != nil {
    +                log.Fatalf("Error creating client: %v", err)
    +        }
    +
    +        return rpc
    +}

    As you can see, here the Celestia Node client takes in the Node RPC IP and the JWT Token we setup before.

    Now, if we go back to our main function, we can do the following to setup and load our env and client:

    go
    
    +func main() {
    +  ctx, cancel := context.WithCancel(context.Background())
    +  defer cancel()
    +  loadEnv()
    +
    +  // Close the client when you are finished
    +  client.Close()
    +}
    
    +func main() {
    +  ctx, cancel := context.WithCancel(context.Background())
    +  defer cancel()
    +  loadEnv()
    +
    +  // Close the client when you are finished
    +  client.Close()
    +}

    Here, we setup a workflow that allows us to load our env vars, instantiate the client with it, then close the client.

    Now, let’s build some more helpful functions:

    go
    func createNamespaceID() []byte {
    +        nIDString := os.Getenv("NAMESPACE_ID")
    +        data, err := hex.DecodeString(nIDString)
    +        if err != nil {
    +                log.Fatalf("Error decoding hex string:", err)
    +        }
    +        // Encode the byte array in Base64
    +        base64Str := base64.StdEncoding.EncodeToString(data)
    +        namespaceID, err := base64.StdEncoding.DecodeString(base64Str)
    +        if err != nil {
    +                log.Fatalf("Error decoding Base64 string:", err)
    +        }
    +        return namespaceID
    +}
    func createNamespaceID() []byte {
    +        nIDString := os.Getenv("NAMESPACE_ID")
    +        data, err := hex.DecodeString(nIDString)
    +        if err != nil {
    +                log.Fatalf("Error decoding hex string:", err)
    +        }
    +        // Encode the byte array in Base64
    +        base64Str := base64.StdEncoding.EncodeToString(data)
    +        namespaceID, err := base64.StdEncoding.DecodeString(base64Str)
    +        if err != nil {
    +                log.Fatalf("Error decoding Base64 string:", err)
    +        }
    +        return namespaceID
    +}

    Here, we are creating a helpful function called createNameSpaceID that given a string for a namespace ID, it can decode the hex string, encode it after to a byte array, then decode it as a base64 string which is needed by Node API.

    We will need to create just a few more functions before we wrap up things.

    go
    // postDataAndGetHeight submits a new transaction with the
    +// provided data to the Celestia node.
    +func postDataAndGetHeight(client *client.Client, namespaceID namespace.ID, payLoad [
    +]byte, fee cosmosmath.Int, gasLimit uint64) uint64 {
    +        response, err := client.State.SubmitPayForBlob(context.Background(), namespa
    +ceID, payLoad, fee, gasLimit)
    +        if err != nil {
    +                log.Fatalf("Error submitting pay for blob: %v", err)
    +        }
    +        fmt.Printf("Got output: %v", response)
    +        height := uint64(response.Height)
    +        fmt.Printf("Height that data was submitted at: %v", height)
    +        return height
    +}
    // postDataAndGetHeight submits a new transaction with the
    +// provided data to the Celestia node.
    +func postDataAndGetHeight(client *client.Client, namespaceID namespace.ID, payLoad [
    +]byte, fee cosmosmath.Int, gasLimit uint64) uint64 {
    +        response, err := client.State.SubmitPayForBlob(context.Background(), namespa
    +ceID, payLoad, fee, gasLimit)
    +        if err != nil {
    +                log.Fatalf("Error submitting pay for blob: %v", err)
    +        }
    +        fmt.Printf("Got output: %v", response)
    +        height := uint64(response.Height)
    +        fmt.Printf("Height that data was submitted at: %v", height)
    +        return height
    +}

    In the function postDataAndGetHeight, we show you how to submit a message to a specific namespace ID provided on Celestia. After a successful submission, the function returns back to you the block height it was submitted at.

    Next, implement the following function:

    go
    func getDataAsPrompt(client *client.Client, height uint64, namespaceID namespace.ID)
    + string {
    +        headerParam := getHeader(client, height)
    +        response, err := client.Share.GetSharesByNamespace(context.Background(), hea
    +derParam.DAH, namespaceID)
    +        if err != nil {
    +                log.Fatalf("Error getting shares by namespace data for block height:
    + %v. Error is %v", height, err)
    +        }
    +        var dataString string
    +        for _, shares := range response {
    +                for _, share := range shares.Shares {
    +                        dataString = string(share[8:])
    +                }
    +        }
    +        return dataString
    +}
    func getDataAsPrompt(client *client.Client, height uint64, namespaceID namespace.ID)
    + string {
    +        headerParam := getHeader(client, height)
    +        response, err := client.Share.GetSharesByNamespace(context.Background(), hea
    +derParam.DAH, namespaceID)
    +        if err != nil {
    +                log.Fatalf("Error getting shares by namespace data for block height:
    + %v. Error is %v", height, err)
    +        }
    +        var dataString string
    +        for _, shares := range response {
    +                for _, share := range shares.Shares {
    +                        dataString = string(share[8:])
    +                }
    +        }
    +        return dataString
    +}

    In the function getDataAsPrompt, we show a helpful function that given a particular block height and a namespace ID, it can return back the block data (called shares here) which we then convert to a string and return it back.

    Finally, we implement a GPT-3.5 helper function that given a prompt, it returns back a statement:

    go
    // gpt3 processes a given message using GPT-3 and prints the response.
    +func gpt3(msg string) {
    +        // Set the authentication header
    +        openAIKey := os.Getenv("OPENAI_KEY")
    +        client := openai.NewClient(openAIKey)
    +        resp, err := client.CreateChatCompletion(
    +                context.Background(),
    +                openai.ChatCompletionRequest{
    +                        Model: openai.GPT3Dot5Turbo,
    +                        Messages: []openai.ChatCompletionMessage{
    +                                {
    +                                        Role:    openai.ChatMessageRoleUser,
    +                                        Content: msg,
    +                                },
    +                        },
    +                },
    +        )
    +
    +        if err != nil {
    +                fmt.Printf("ChatCompletion error: %v\\n", err)
    +                return
    +        }
    +        fmt.Println(resp.Choices[0].Message.Content)
    +}
    // gpt3 processes a given message using GPT-3 and prints the response.
    +func gpt3(msg string) {
    +        // Set the authentication header
    +        openAIKey := os.Getenv("OPENAI_KEY")
    +        client := openai.NewClient(openAIKey)
    +        resp, err := client.CreateChatCompletion(
    +                context.Background(),
    +                openai.ChatCompletionRequest{
    +                        Model: openai.GPT3Dot5Turbo,
    +                        Messages: []openai.ChatCompletionMessage{
    +                                {
    +                                        Role:    openai.ChatMessageRoleUser,
    +                                        Content: msg,
    +                                },
    +                        },
    +                },
    +        )
    +
    +        if err != nil {
    +                fmt.Printf("ChatCompletion error: %v\\n", err)
    +                return
    +        }
    +        fmt.Println(resp.Choices[0].Message.Content)
    +}

    Wrapping up the functions

    Now, we will update our mainfunction to include the logic from the functions we built where we show you how to do the following:

    • Instantiate namespace ID, fee, gas limit and GPT prompt
    • Submit the GPT prompt as a PayForBlob transaction and then get back the Block Height
    • Get Back the Prompt from that Block Height as a Data Share and convert it to a string and return it
    • Submit that string to the GPT function to get a prompt output
    go
    func main() {
    +        ctx, cancel := context.WithCancel(context.Background())
    +        defer cancel()
    +        loadEnv()
    +        var namespaceID namespace.ID = createNamespaceID()
    +        client := createClient(ctx)
    +        var gasLimit uint64 = 6000000
    +        fee := cosmosmath.NewInt(10000)
    +        var gptPrompt string = "What are modular blockchains?"
    +        prompt := []byte{0x00, 0x01, 0x02}
    +        prompt = append(prompt, []byte(gptPrompt)...)
    +        height := postDataAndGetHeight(client, namespaceID, prompt, fee, gasLimit)
    +        promptString := getDataAsPrompt(client, height, namespaceID)
    +        gpt3(promptString)
    +        // Close the client when you are finished
    +        client.Close()
    +}
    func main() {
    +        ctx, cancel := context.WithCancel(context.Background())
    +        defer cancel()
    +        loadEnv()
    +        var namespaceID namespace.ID = createNamespaceID()
    +        client := createClient(ctx)
    +        var gasLimit uint64 = 6000000
    +        fee := cosmosmath.NewInt(10000)
    +        var gptPrompt string = "What are modular blockchains?"
    +        prompt := []byte{0x00, 0x01, 0x02}
    +        prompt = append(prompt, []byte(gptPrompt)...)
    +        height := postDataAndGetHeight(client, namespaceID, prompt, fee, gasLimit)
    +        promptString := getDataAsPrompt(client, height, namespaceID)
    +        gpt3(promptString)
    +        // Close the client when you are finished
    +        client.Close()
    +}

    And now you have the final version of the game!

    Run the golang script with the following command:

    sh
    go run main.go
    go run main.go

    After some time, it’ll post the output of the prompt you submitted to OpenAI that you pulled from Celestia’s blockchain.

    Next steps

    With this tutorial, you were able to construct a PFB transaction, submit it to Celestia, get it back from Celestia and decode its contents, then for added bonus, submit the message to GPT-3.5.

    For the next steps, we will be releasing quests to this tutorial where users will need to complete challenges that help familiarize them with the Celestia Data Availability layer.

    `,77),e=[o];function t(c,r,E,y,i,d){return n(),a("div",null,e)}const u=s(p,[["render",t]]);export{F as __pageData,u as default}; diff --git a/assets/developers_prompt-scavenger.md.27b40ddb.lean.js b/assets/developers_prompt-scavenger.md.27b40ddb.lean.js new file mode 100644 index 00000000000..9bab774b12f --- /dev/null +++ b/assets/developers_prompt-scavenger.md.27b40ddb.lean.js @@ -0,0 +1 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.51d6c45b.js";const F=JSON.parse('{"title":"Prompt scavenger","description":"Learn how to interact with the Celestia Node API with this tutorial.","frontmatter":{"description":"Learn how to interact with the Celestia Node API with this tutorial.","prev":{"text":"Node RPC CLI tutorial","link":"/developers/node-tutorial"},"head":[["meta",{"name":"og:title","content":"Prompt scavenger | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/prompt-scavenger.md","filePath":"developers/prompt-scavenger.md","lastUpdated":1713191709000}'),p={name:"developers/prompt-scavenger.md"},o=l("",77),e=[o];function t(c,r,E,y,i,d){return n(),a("div",null,e)}const u=s(p,[["render",t]]);export{F as __pageData,u as default}; diff --git a/assets/developers_requesting-data-commitment-ranges.md.3fe57c2b.js b/assets/developers_requesting-data-commitment-ranges.md.3fe57c2b.js new file mode 100644 index 00000000000..bd8fec13370 --- /dev/null +++ b/assets/developers_requesting-data-commitment-ranges.md.3fe57c2b.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as o}from"./chunks/framework.51d6c45b.js";const f=JSON.parse('{"title":"Requesting data commitment ranges","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Requesting data commitment ranges | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/requesting-data-commitment-ranges.md","filePath":"developers/requesting-data-commitment-ranges.md","lastUpdated":1709911265000}'),r={name:"developers/requesting-data-commitment-ranges.md"},n=o('

    Requesting data commitment ranges

    By default, the Blobstream X deployments on Ethereum will be updating every 4 hours, and on Arbitrum One and Base, updating every 1 hour. If you wish for the Blobstream X contract to be updated at a different cadence, then you have several different options for how to update the smart contract.

    To request proofs to be submitted to the Blobstream X contract at a different cadence, you can do one of the following:

    Run the Blobstream X operator with hosted proving on the Succinct platform, by running an operator script that pings the platform with proof requests at a specified cadence.

    Follow these instructions to run the operator script.

    Here are example values for the .env file:

    1. TENDERMINT_RPC_URL from the public Celestia list.
    2. SUCCINCT_RPC_URL = https://alpha.succinct.xyz/api
    3. Request for SUCCINCT_API_KEY from the Succinct team.
    4. CHAIN_ID is the chain ID of the deployed Blobstream X contract.
    5. CONTRACT_ADDRESS: Blobstream X proxy contract address.
    6. NEXT_HEADER_FUNCTION_ID & HEADER_RANGE_FUNCTION_ID: Get the functionId's from the Blobstream X contract by using the nextHeaderFunctionId and headerRangeFunctionId respectively, which are public storage variables.

    Local proving

    Run the Blobstream X operator with local proving.

    TIP

    Note: Requires a large cloud machine to run in a reasonable amount of time. EC2 r6a.16xlarge takes ~30 minutes to generate a header range proof.

    Request proof onchain

    Directly request a proof via the Blobstream X contract interface. Unlike the Blobstream X operator which handles requests off-chain, requesting on-chain requires gas, but the proof will be generated and relayed by the Succinct platform.

    1. Call requestHeaderRange(uint64 _targetBlock) with the end of the range you want a commitment for.

    2. A DataCommitmentStored(uint256, uint64, uint64, bytes32) will be emitted for the requested range when it is stored in the contract. Listen to this event to know that the proof has been generated successfully.

    ',14),i=[n];function c(s,l,d,h,m,p){return t(),a("div",null,i)}const g=e(r,[["render",c]]);export{f as __pageData,g as default}; diff --git a/assets/developers_requesting-data-commitment-ranges.md.3fe57c2b.lean.js b/assets/developers_requesting-data-commitment-ranges.md.3fe57c2b.lean.js new file mode 100644 index 00000000000..7d926d9fc9e --- /dev/null +++ b/assets/developers_requesting-data-commitment-ranges.md.3fe57c2b.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as o}from"./chunks/framework.51d6c45b.js";const f=JSON.parse('{"title":"Requesting data commitment ranges","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Requesting data commitment ranges | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/requesting-data-commitment-ranges.md","filePath":"developers/requesting-data-commitment-ranges.md","lastUpdated":1709911265000}'),r={name:"developers/requesting-data-commitment-ranges.md"},n=o("",14),i=[n];function c(s,l,d,h,m,p){return t(),a("div",null,i)}const g=e(r,[["render",c]]);export{f as __pageData,g as default}; diff --git a/assets/developers_rollkit.md.26b9c7c3.js b/assets/developers_rollkit.md.26b9c7c3.js new file mode 100644 index 00000000000..95a59ef2315 --- /dev/null +++ b/assets/developers_rollkit.md.26b9c7c3.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as l,Q as o}from"./chunks/framework.51d6c45b.js";const a="/img/rollkit.png",f=JSON.parse(`{"title":"Rollkit","description":"Learn how to build Cosmos-SDK applications that connect to Celestia's Data Availability Layer via Rollkit.","frontmatter":{"description":"Learn how to build Cosmos-SDK applications that connect to Celestia's Data Availability Layer via Rollkit.","prev":{"text":"Deploy an OP Stack devnet to Celestia","link":"/developers/optimism"},"next":{"text":"Create a wallet with celestia-node","link":"/developers/celestia-node-key"},"head":[["meta",{"name":"og:title","content":"Rollkit | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/rollkit.md","filePath":"developers/rollkit.md","lastUpdated":1707231883000}`),i={name:"developers/rollkit.md"},r=o('

    Rollkit

    Validator nodes allow you to participate in consensus in the Celestia network.

    rollkit

    Rollkit is an ABCI (Application Blockchain Interface) implementation for sovereign rollups to deploy on top of Celestia.

    It is built by replacing Tendermint, the Cosmos-SDK consensus layer, with a drop-in replacement that communicates directly with Celestia's Data Availability layer.

    It spins up a sovereign rollup, which collects transactions into blocks and posts them onto Celestia for consensus and data availability.

    The goal of Rollkit is to enable anyone to design and deploy a sovereign rollup on Celestia in minutes.

    Furthermore, while Rollkit allows you to build sovereign rollups on Celestia, it currently does not support fraud proofs yet and is therefore running in "pessimistic" mode, where nodes would need to re-execute the transactions to check the validity of the chain (i.e. a full node). Furthermore, Rollkit currently only supports a single sequencer.

    Tutorials

    The following tutorials will help you get started building Cosmos-SDK applications that connect to Celestia's Data Availability Layer via Rollkit. We call those chains Sovereign Rollups.

    You can get started with the following tutorials:

    ',12),s=[r];function n(p,c,d,h,u,k){return t(),l("div",null,s)}const _=e(i,[["render",n]]);export{f as __pageData,_ as default}; diff --git a/assets/developers_rollkit.md.26b9c7c3.lean.js b/assets/developers_rollkit.md.26b9c7c3.lean.js new file mode 100644 index 00000000000..90d99332564 --- /dev/null +++ b/assets/developers_rollkit.md.26b9c7c3.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as l,Q as o}from"./chunks/framework.51d6c45b.js";const a="/img/rollkit.png",f=JSON.parse(`{"title":"Rollkit","description":"Learn how to build Cosmos-SDK applications that connect to Celestia's Data Availability Layer via Rollkit.","frontmatter":{"description":"Learn how to build Cosmos-SDK applications that connect to Celestia's Data Availability Layer via Rollkit.","prev":{"text":"Deploy an OP Stack devnet to Celestia","link":"/developers/optimism"},"next":{"text":"Create a wallet with celestia-node","link":"/developers/celestia-node-key"},"head":[["meta",{"name":"og:title","content":"Rollkit | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/rollkit.md","filePath":"developers/rollkit.md","lastUpdated":1707231883000}`),i={name:"developers/rollkit.md"},r=o("",12),s=[r];function n(p,c,d,h,u,k){return t(),l("div",null,s)}const _=e(i,[["render",n]]);export{f as __pageData,_ as default}; diff --git a/assets/developers_submit-data.md.64bfc8e2.js b/assets/developers_submit-data.md.64bfc8e2.js new file mode 100644 index 00000000000..01de86355d9 --- /dev/null +++ b/assets/developers_submit-data.md.64bfc8e2.js @@ -0,0 +1,209 @@ +import{_ as o,o as a,c as n,k as s,a as t,Q as e}from"./chunks/framework.51d6c45b.js";const os=JSON.parse('{"title":"Submitting data blobs to Celestia","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Submitting data blobs to Celestia | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/submit-data.md","filePath":"developers/submit-data.md","lastUpdated":1713793066000}'),l={name:"developers/submit-data.md"},p=e('

    Submitting data blobs to Celestia

    To submit data to Celestia, users submit blob transactions (BlobTx). Blob transactions contain two components, a standard Cosmos-SDK transaction called MsgPayForBlobs and one or more Blobs of data.

    Fee market and mempool

    Celestia makes use of a standard gas-priced prioritized mempool. By default, transactions with gas prices higher than that of other transactions in the mempool will be prioritized by validators.

    Fees and gas limits

    As of version v1.0.0 of the application (celestia-app), there is no protocol enforced minimum fee (similar to EIP-1559 in Ethereum). Instead, each consensus node running a mempool uses a locally configured gas price threshold that must be met in order for that node to accept a transaction, either directly from a user or gossiped from another node, into its mempool.

    As of version v1.0.0 of the application (celestia-app), gas is not refunded. Instead, transaction fees are deducted by a flat fee, originally specified by the user in their tx (where fees = gasLimit * gasPrice). This means that users should use an accurate gas limit value if they do not wish to overpay.

    Under the hood, fees are currently handled by specifying and deducting a flat fee. However gas price is often specified by users instead of calculating the flat fee from the gas used and the gas price. Since the state machine does not refund users for unused gas, gas price is calculated by dividing the total fee by the gas limit.

    Estimating PFB gas

    Generally, the gas used by a PFB transaction involves a static fixed cost and a dynamic cost based on the size of each blob in the transaction.

    NOTE

    For a general use case of a normal account submitting a PFB, the static costs can be treated as such. However, due to the description above of how gas works in the Cosmos-SDK this is not always the case. Notably, if a vesting account or the feegrant modules are used, then these static costs change.

    The fixed cost is an approximation of the gas consumed by operations outside the function GasToConsume (for example, signature verification, tx size, read access to accounts), which has a default value of 65,000 gas.

    NOTE

    The first transaction sent by an account (sequence number == 0) has an additional one time gas cost of 10,000 gas. If this is the case, this should be accounted for.

    Each blob in the PFB contributes to the total gas cost based on its size. The function GasToConsume calculates the total gas consumed by all the blobs involved in a PFB, where each blob's gas cost is computed by first determining how many shares are needed to store the blob size. Then, it computes the product of the number of shares, the number of bytes per share, and the gasPerByte parameter. Finally, it adds a static amount per blob.

    The blob.GasPerBlobByte and auth.TxSizeCostPerByte are parameters that could potentially be adjusted through the system's governance mechanisms. Hence, actual costs may vary depending on the current state of these parameters.

    Gas fee calculation

    The total fee for a transaction is calculated as the product of the gas limit for the transaction and the gas price set by the user:

    ',17),r={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},c={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.186ex"},xmlns:"http://www.w3.org/2000/svg",width:"34.471ex",height:"1.781ex",role:"img",focusable:"false",viewBox:"0 -705 15236 787","aria-hidden":"true"},Q=e('',1),i=[Q],T=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("mtext",null,"Total Fee"),s("mo",null,"="),s("mtext",null,"Gas Limit"),s("mo",null,"×"),s("mtext",null,"Gas Price")])],-1),d=s("p",null,"The gas limit for a transaction is the maximum amount of gas that a user is willing to spend on a transaction. It is determined by both a static fixed cost (FC) and a variable dynamic cost based on the size of each blob involved in the transaction:",-1),y={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},h={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.777ex"},xmlns:"http://www.w3.org/2000/svg",width:"50.774ex",height:"2.563ex",role:"img",focusable:"false",viewBox:"0 -789.6 22442.1 1132.9","aria-hidden":"true"},E=e('',1),m=[E],u=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("mtext",null,"Gas Limit"),s("mo",null,"="),s("mi",null,"F"),s("mi",null,"C"),s("mo",null,"+"),s("munderover",null,[s("mo",{"data-mjx-texclass":"OP"},"∑"),s("mrow",{"data-mjx-texclass":"ORD"},[s("mi",null,"i"),s("mo",null,"="),s("mn",null,"1")]),s("mrow",{"data-mjx-texclass":"ORD"},[s("mi",null,"n")])]),s("mi",null,"S"),s("mi",null,"S"),s("mi",null,"N"),s("mo",{stretchy:"false"},"("),s("msub",null,[s("mi",null,"B"),s("mi",null,"i")]),s("mo",{stretchy:"false"},")"),s("mo",null,"×"),s("mi",null,"S"),s("mi",null,"S"),s("mo",null,"×"),s("mi",null,"G"),s("mi",null,"C"),s("mi",null,"P"),s("mi",null,"B"),s("mi",null,"B")])],-1),g=s("p",null,"Where:",-1),b={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},f={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.05ex"},xmlns:"http://www.w3.org/2000/svg",width:"3.414ex",height:"1.645ex",role:"img",focusable:"false",viewBox:"0 -705 1509 727","aria-hidden":"true"},H=e('',1),F=[H],w=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("mi",null,"F"),s("mi",null,"C")])],-1),x={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},V={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.777ex"},xmlns:"http://www.w3.org/2000/svg",width:"14.695ex",height:"2.563ex",role:"img",focusable:"false",viewBox:"0 -789.6 6495.3 1132.9","aria-hidden":"true"},k=e('',1),C=[k],L=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("munderover",null,[s("mo",{"data-mjx-texclass":"OP"},"∑"),s("mrow",{"data-mjx-texclass":"ORD"},[s("mi",null,"i"),s("mo",null,"="),s("mn",null,"1")]),s("mrow",{"data-mjx-texclass":"ORD"},[s("mi",null,"n")])]),s("mi",null,"S"),s("mi",null,"S"),s("mi",null,"N"),s("mo",{stretchy:"false"},"("),s("msub",null,[s("mi",null,"B"),s("mi",null,"i")]),s("mo",{stretchy:"false"},")")])],-1),_={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},D={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"0.781ex",height:"1.52ex",role:"img",focusable:"false",viewBox:"0 -661 345 672","aria-hidden":"true"},B=s("g",{stroke:"currentColor",fill:"currentColor","stroke-width":"0",transform:"scale(1,-1)"},[s("g",{"data-mml-node":"math"},[s("g",{"data-mml-node":"mi"},[s("path",{"data-c":"1D456",d:"M184 600Q184 624 203 642T247 661Q265 661 277 649T290 619Q290 596 270 577T226 557Q211 557 198 567T184 600ZM21 287Q21 295 30 318T54 369T98 420T158 442Q197 442 223 419T250 357Q250 340 236 301T196 196T154 83Q149 61 149 51Q149 26 166 26Q175 26 185 29T208 43T235 78T260 137Q263 149 265 151T282 153Q302 153 302 143Q302 135 293 112T268 61T223 11T161 -11Q129 -11 102 10T74 74Q74 91 79 106T122 220Q160 321 166 341T173 380Q173 404 156 404H154Q124 404 99 371T61 287Q60 286 59 284T58 281T56 279T53 278T49 278T41 278H27Q21 284 21 287Z",style:{"stroke-width":"3"}})])])],-1),M=[B],A=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("mi",null,"i")])],-1),v={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},Z={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"0.781ex",height:"1.52ex",role:"img",focusable:"false",viewBox:"0 -661 345 672","aria-hidden":"true"},q=s("g",{stroke:"currentColor",fill:"currentColor","stroke-width":"0",transform:"scale(1,-1)"},[s("g",{"data-mml-node":"math"},[s("g",{"data-mml-node":"mi"},[s("path",{"data-c":"1D456",d:"M184 600Q184 624 203 642T247 661Q265 661 277 649T290 619Q290 596 270 577T226 557Q211 557 198 567T184 600ZM21 287Q21 295 30 318T54 369T98 420T158 442Q197 442 223 419T250 357Q250 340 236 301T196 196T154 83Q149 61 149 51Q149 26 166 26Q175 26 185 29T208 43T235 78T260 137Q263 149 265 151T282 153Q302 153 302 143Q302 135 293 112T268 61T223 11T161 -11Q129 -11 102 10T74 74Q74 91 79 106T122 220Q160 321 166 341T173 380Q173 404 156 404H154Q124 404 99 371T61 287Q60 286 59 284T58 281T56 279T53 278T49 278T41 278H27Q21 284 21 287Z",style:{"stroke-width":"3"}})])])],-1),S=[q],P=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("mi",null,"i")])],-1),j={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},I={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.05ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.919ex",height:"1.645ex",role:"img",focusable:"false",viewBox:"0 -705 1290 727","aria-hidden":"true"},G=e('',1),O=[G],N=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("mi",null,"S"),s("mi",null,"S")])],-1),R={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},z={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.05ex"},xmlns:"http://www.w3.org/2000/svg",width:"8.631ex",height:"1.645ex",role:"img",focusable:"false",viewBox:"0 -705 3815 727","aria-hidden":"true"},J=e('',1),K=[J],U=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("mi",null,"G"),s("mi",null,"C"),s("mi",null,"P"),s("mi",null,"B"),s("mi",null,"B")])],-1),W=e(`

    The gas fee is set by the user when they submit a transaction. The fee is often specified by users directly. The total cost for the transaction is then calculated as the product of the estimated gas limit and the gas price. Since the state machine does not refund users for unused gas, it's important for users to estimate the gas limit accurately to avoid overpaying.

    For more details on how gas is calculated per blob, refer to the PayForBlobs function that consumes gas based on the blob sizes. This function uses the GasToConsume function to calculate the extra gas charged to pay for a set of blobs in a MsgPayForBlobs transaction. This function calculates the total shares used by all blobs and multiplies it by the ShareSize and gasPerByte to get the total gas to consume.

    For estimating the total gas required for a set of blobs, refer to the EstimateGas function. This function estimates the gas based on a linear model that is dependent on the governance parameters: gasPerByte and txSizeCost. It assumes other variables are constant, including the assumption that the MsgPayForBlobs is the only message in the transaction. The DefaultEstimateGas function runs EstimateGas with the system defaults.

    Estimating gas programmatically

    Users can estimate an efficient gas limit by using this function:

    go
    import (
    +    blobtypes "github.com/celestiaorg/celestia-app/x/blob/types"
    +)
    +gasLimit := blobtypes.DefaultEstimateGas([]uint32{uint32(sizeOfDataInBytes)})
    import (
    +    blobtypes "github.com/celestiaorg/celestia-app/x/blob/types"
    +)
    +gasLimit := blobtypes.DefaultEstimateGas([]uint32{uint32(sizeOfDataInBytes)})

    If using a celestia-node light client, then this function is automatically called for you when submitting a blob. This function works by breaking down the components of calculating gas for a blob transaction. These components consist of a flat costs for all PFBs, the size of each blob and how many shares each uses and the parameter for gas used per byte. More information about how gas is used can be found in the gas specs and the exact formula can be found in the blob module.

    Submitting multiple transactions in one block from the same account

    The mempool Celestia uses works by maintaining a fork of the canonical state each block. This means that each time we submit a transaction to it, it will update the sequence number (aka nonce) for the account that submitted the transaction. If users wish to submit a second transaction, they can, but must specify the nonce manually. If this is not done, the new transactions will not be able to be submitted until the first transaction is reaped from the mempool (i.e. included in a block), or dropped due to timing out.

    By default, nodes will drop a transaction if it does not get included in 5 blocks (roughly 1 minute). At this point, the user must resubmit their transaction if they want it to eventually be included.

    As of v1.0.0 of the application (celestia-app), users are unable to replace an existing transaction with a different one with higher fees. They must instead wait 5 blocks from the original submitted time and then resubmit the transaction. Again, community members have already suggested solutions and a willingness to accept changes to fix this issue.

    API

    Users can currently create and submit BlobTxs in four ways.

    The celestia-app consensus node CLI

    bash
    celestia-appd tx blob PayForBlobs <hex-encoded namespace> <hex-encoded data> [flags]
    celestia-appd tx blob PayForBlobs <hex-encoded namespace> <hex-encoded data> [flags]

    The celestia-node light node CLI

    Using blob.Submit:

    bash
    celestia blob submit <hex-encoded namespace> <hex-encoded data>
    celestia blob submit <hex-encoded namespace> <hex-encoded data>

    Learn more in the node tutorial.

    GRPC to a consensus node via the user package

    go
    import (
    +    "context"
    +    "fmt"
    +
    +    "github.com/celestiaorg/celestia-app/app"
    +    "github.com/celestiaorg/celestia-app/app/encoding"
    +    "github.com/celestiaorg/celestia-app/pkg/appconsts"
    +    "github.com/celestiaorg/celestia-app/pkg/namespace"
    +    "github.com/celestiaorg/celestia-app/pkg/user"
    +    blobtypes "github.com/celestiaorg/celestia-app/x/blob/types"
    +    "github.com/cosmos/cosmos-sdk/crypto/keyring"
    +    tmproto "github.com/tendermint/tendermint/proto/tendermint/types"
    +    "google.golang.org/grpc"
    +    "google.golang.org/grpc/credentials/insecure"
    +)
    +
    +// SubmitData is a demo function that shows how to use the signer to submit data
    +// to the blockchain directly via a celestia node. We can manage this keyring
    +// using the \`celestia-appd keys\` or \`celestia keys\` sub commands and load this
    +// keyring from a file and use it to programmatically sign transactions.
    +func DemoSubmitData(grpcAddr string, kr keyring.Keyring) error {
    +    // create an encoding config that can decode and encode all celestia-app
    +    // data structures.
    +    ecfg := encoding.MakeConfig(app.ModuleEncodingRegisters...)
    +
    +    // create a connection to the grpc server on the consensus node.
    +    conn, err := grpc.Dial(grpcAddr, grpc.WithTransportCredentials(insecure.NewCredentials()))
    +    if err != nil {
    +        return err
    +    }
    +    defer conn.Close()
    +
    +    // get the address of the account we want to use to sign transactions.
    +    rec, err := kr.Key("accountName")
    +    if err != nil {
    +        return err
    +    }
    +
    +    addr, err := rec.GetAddress()
    +    if err != nil {
    +        return err
    +    }
    +
    +    // Setup the signer. This function will automatically query the relevant
    +    // account information such as sequence (nonce) and account number.
    +    signer, err := user.SetupSigner(context.TODO(), kr, conn, addr, ecfg)
    +    if err != nil {
    +        return err
    +    }
    +
    +    ns := namespace.MustNewV0([]byte("1234567890"))
    +
    +    fmt.Println("namespace", len(ns.Bytes()))
    +
    +    blob, err := blobtypes.NewBlob(ns, []byte("some data"), appconsts.ShareVersionZero)
    +    if err != nil {
    +        return err
    +    }
    +
    +    gasLimit := blobtypes.DefaultEstimateGas([]uint32{uint32(len(blob.Data))})
    +
    +    options := []user.TxOption{
    +        // here we're setting estimating the gas limit from the above estimated
    +        // function, and then setting the gas price to 0.1utia per unit of gas.
    +        user.SetGasLimitAndFee(gasLimit, 0.1),
    +    }
    +
    +    // this function will submit the transaction and block until a timeout is
    +    // reached or the transaction is committed.
    +    resp, err := signer.SubmitPayForBlob(context.TODO(), []*tmproto.Blob{blob}, options...)
    +    if err != nil {
    +        return err
    +    }
    +
    +    // check the response code to see if the transaction was successful.
    +    if resp.Code != 0 {
    +        // handle code
    +        fmt.Println(resp.Code, resp.Codespace, resp.RawLog)
    +    }
    +
    +    // if we don't want to wait for the transaction to be confirmed, we can
    +    // manually sign and submit the transaction using the same package.
    +    blobTx, err := signer.CreatePayForBlob([]*tmproto.Blob{blob}, options...)
    +    if err != nil {
    +        return err
    +    }
    +
    +    resp, err = signer.BroadcastTx(context.TODO(), blobTx)
    +    if err != nil {
    +        return err
    +    }
    +
    +    // check the response code to see if the transaction was successful. Note
    +    // that this time we're not waiting for the transaction to be committed.
    +    // Therefore the code here is only from the consensus node's mempool.
    +    if resp.Code != 0 {
    +        // handle code
    +        fmt.Println(resp.Code, resp.Codespace, resp.RawLog)
    +    }
    +
    +    return err
    +}
    import (
    +    "context"
    +    "fmt"
    +
    +    "github.com/celestiaorg/celestia-app/app"
    +    "github.com/celestiaorg/celestia-app/app/encoding"
    +    "github.com/celestiaorg/celestia-app/pkg/appconsts"
    +    "github.com/celestiaorg/celestia-app/pkg/namespace"
    +    "github.com/celestiaorg/celestia-app/pkg/user"
    +    blobtypes "github.com/celestiaorg/celestia-app/x/blob/types"
    +    "github.com/cosmos/cosmos-sdk/crypto/keyring"
    +    tmproto "github.com/tendermint/tendermint/proto/tendermint/types"
    +    "google.golang.org/grpc"
    +    "google.golang.org/grpc/credentials/insecure"
    +)
    +
    +// SubmitData is a demo function that shows how to use the signer to submit data
    +// to the blockchain directly via a celestia node. We can manage this keyring
    +// using the \`celestia-appd keys\` or \`celestia keys\` sub commands and load this
    +// keyring from a file and use it to programmatically sign transactions.
    +func DemoSubmitData(grpcAddr string, kr keyring.Keyring) error {
    +    // create an encoding config that can decode and encode all celestia-app
    +    // data structures.
    +    ecfg := encoding.MakeConfig(app.ModuleEncodingRegisters...)
    +
    +    // create a connection to the grpc server on the consensus node.
    +    conn, err := grpc.Dial(grpcAddr, grpc.WithTransportCredentials(insecure.NewCredentials()))
    +    if err != nil {
    +        return err
    +    }
    +    defer conn.Close()
    +
    +    // get the address of the account we want to use to sign transactions.
    +    rec, err := kr.Key("accountName")
    +    if err != nil {
    +        return err
    +    }
    +
    +    addr, err := rec.GetAddress()
    +    if err != nil {
    +        return err
    +    }
    +
    +    // Setup the signer. This function will automatically query the relevant
    +    // account information such as sequence (nonce) and account number.
    +    signer, err := user.SetupSigner(context.TODO(), kr, conn, addr, ecfg)
    +    if err != nil {
    +        return err
    +    }
    +
    +    ns := namespace.MustNewV0([]byte("1234567890"))
    +
    +    fmt.Println("namespace", len(ns.Bytes()))
    +
    +    blob, err := blobtypes.NewBlob(ns, []byte("some data"), appconsts.ShareVersionZero)
    +    if err != nil {
    +        return err
    +    }
    +
    +    gasLimit := blobtypes.DefaultEstimateGas([]uint32{uint32(len(blob.Data))})
    +
    +    options := []user.TxOption{
    +        // here we're setting estimating the gas limit from the above estimated
    +        // function, and then setting the gas price to 0.1utia per unit of gas.
    +        user.SetGasLimitAndFee(gasLimit, 0.1),
    +    }
    +
    +    // this function will submit the transaction and block until a timeout is
    +    // reached or the transaction is committed.
    +    resp, err := signer.SubmitPayForBlob(context.TODO(), []*tmproto.Blob{blob}, options...)
    +    if err != nil {
    +        return err
    +    }
    +
    +    // check the response code to see if the transaction was successful.
    +    if resp.Code != 0 {
    +        // handle code
    +        fmt.Println(resp.Code, resp.Codespace, resp.RawLog)
    +    }
    +
    +    // if we don't want to wait for the transaction to be confirmed, we can
    +    // manually sign and submit the transaction using the same package.
    +    blobTx, err := signer.CreatePayForBlob([]*tmproto.Blob{blob}, options...)
    +    if err != nil {
    +        return err
    +    }
    +
    +    resp, err = signer.BroadcastTx(context.TODO(), blobTx)
    +    if err != nil {
    +        return err
    +    }
    +
    +    // check the response code to see if the transaction was successful. Note
    +    // that this time we're not waiting for the transaction to be committed.
    +    // Therefore the code here is only from the consensus node's mempool.
    +    if resp.Code != 0 {
    +        // handle code
    +        fmt.Println(resp.Code, resp.Codespace, resp.RawLog)
    +    }
    +
    +    return err
    +}

    RPC to a celestia-node

    Using the JSON RPC API, submit data using the following methods:

    Learn more in the celestia-node API docs.

    `,25);function X($,Y,ss,as,ns,es){return a(),n("div",null,[p,s("p",null,[s("mjx-container",r,[(a(),n("svg",c,i)),T])]),d,s("p",null,[s("mjx-container",y,[(a(),n("svg",h,m)),u])]),g,s("ul",null,[s("li",null,[s("mjx-container",b,[(a(),n("svg",f,F)),w]),t(" = Fixed Cost, is a static value (65,000 gas)")]),s("li",null,[s("mjx-container",x,[(a(),n("svg",V,C)),L]),t(" = SparseSharesNeeded for the "),s("mjx-container",_,[(a(),n("svg",D,M)),A]),t("th Blob, is the number of shares needed for the "),s("mjx-container",v,[(a(),n("svg",Z,S)),P]),t("th blob in the transaction")]),s("li",null,[s("mjx-container",j,[(a(),n("svg",I,O)),N]),t(" = Share Size, is the size of each share")]),s("li",null,[s("mjx-container",R,[(a(),n("svg",z,K)),U]),t(" = Gas Cost Per Blob Byte, is a parameter that could potentially be adjusted through the system's governance mechanisms.")])]),W])}const ls=o(l,[["render",X]]);export{os as __pageData,ls as default}; diff --git a/assets/developers_submit-data.md.64bfc8e2.lean.js b/assets/developers_submit-data.md.64bfc8e2.lean.js new file mode 100644 index 00000000000..c64aa25da84 --- /dev/null +++ b/assets/developers_submit-data.md.64bfc8e2.lean.js @@ -0,0 +1 @@ +import{_ as o,o as a,c as n,k as s,a as t,Q as e}from"./chunks/framework.51d6c45b.js";const os=JSON.parse('{"title":"Submitting data blobs to Celestia","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Submitting data blobs to Celestia | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/submit-data.md","filePath":"developers/submit-data.md","lastUpdated":1713793066000}'),l={name:"developers/submit-data.md"},p=e("",17),r={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},c={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.186ex"},xmlns:"http://www.w3.org/2000/svg",width:"34.471ex",height:"1.781ex",role:"img",focusable:"false",viewBox:"0 -705 15236 787","aria-hidden":"true"},Q=e("",1),i=[Q],T=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("mtext",null,"Total Fee"),s("mo",null,"="),s("mtext",null,"Gas Limit"),s("mo",null,"×"),s("mtext",null,"Gas Price")])],-1),d=s("p",null,"The gas limit for a transaction is the maximum amount of gas that a user is willing to spend on a transaction. It is determined by both a static fixed cost (FC) and a variable dynamic cost based on the size of each blob involved in the transaction:",-1),y={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},h={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.777ex"},xmlns:"http://www.w3.org/2000/svg",width:"50.774ex",height:"2.563ex",role:"img",focusable:"false",viewBox:"0 -789.6 22442.1 1132.9","aria-hidden":"true"},E=e("",1),m=[E],u=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("mtext",null,"Gas Limit"),s("mo",null,"="),s("mi",null,"F"),s("mi",null,"C"),s("mo",null,"+"),s("munderover",null,[s("mo",{"data-mjx-texclass":"OP"},"∑"),s("mrow",{"data-mjx-texclass":"ORD"},[s("mi",null,"i"),s("mo",null,"="),s("mn",null,"1")]),s("mrow",{"data-mjx-texclass":"ORD"},[s("mi",null,"n")])]),s("mi",null,"S"),s("mi",null,"S"),s("mi",null,"N"),s("mo",{stretchy:"false"},"("),s("msub",null,[s("mi",null,"B"),s("mi",null,"i")]),s("mo",{stretchy:"false"},")"),s("mo",null,"×"),s("mi",null,"S"),s("mi",null,"S"),s("mo",null,"×"),s("mi",null,"G"),s("mi",null,"C"),s("mi",null,"P"),s("mi",null,"B"),s("mi",null,"B")])],-1),g=s("p",null,"Where:",-1),b={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},f={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.05ex"},xmlns:"http://www.w3.org/2000/svg",width:"3.414ex",height:"1.645ex",role:"img",focusable:"false",viewBox:"0 -705 1509 727","aria-hidden":"true"},H=e("",1),F=[H],w=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("mi",null,"F"),s("mi",null,"C")])],-1),x={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},V={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.777ex"},xmlns:"http://www.w3.org/2000/svg",width:"14.695ex",height:"2.563ex",role:"img",focusable:"false",viewBox:"0 -789.6 6495.3 1132.9","aria-hidden":"true"},k=e("",1),C=[k],L=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("munderover",null,[s("mo",{"data-mjx-texclass":"OP"},"∑"),s("mrow",{"data-mjx-texclass":"ORD"},[s("mi",null,"i"),s("mo",null,"="),s("mn",null,"1")]),s("mrow",{"data-mjx-texclass":"ORD"},[s("mi",null,"n")])]),s("mi",null,"S"),s("mi",null,"S"),s("mi",null,"N"),s("mo",{stretchy:"false"},"("),s("msub",null,[s("mi",null,"B"),s("mi",null,"i")]),s("mo",{stretchy:"false"},")")])],-1),_={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},D={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"0.781ex",height:"1.52ex",role:"img",focusable:"false",viewBox:"0 -661 345 672","aria-hidden":"true"},B=s("g",{stroke:"currentColor",fill:"currentColor","stroke-width":"0",transform:"scale(1,-1)"},[s("g",{"data-mml-node":"math"},[s("g",{"data-mml-node":"mi"},[s("path",{"data-c":"1D456",d:"M184 600Q184 624 203 642T247 661Q265 661 277 649T290 619Q290 596 270 577T226 557Q211 557 198 567T184 600ZM21 287Q21 295 30 318T54 369T98 420T158 442Q197 442 223 419T250 357Q250 340 236 301T196 196T154 83Q149 61 149 51Q149 26 166 26Q175 26 185 29T208 43T235 78T260 137Q263 149 265 151T282 153Q302 153 302 143Q302 135 293 112T268 61T223 11T161 -11Q129 -11 102 10T74 74Q74 91 79 106T122 220Q160 321 166 341T173 380Q173 404 156 404H154Q124 404 99 371T61 287Q60 286 59 284T58 281T56 279T53 278T49 278T41 278H27Q21 284 21 287Z",style:{"stroke-width":"3"}})])])],-1),M=[B],A=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("mi",null,"i")])],-1),v={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},Z={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"0.781ex",height:"1.52ex",role:"img",focusable:"false",viewBox:"0 -661 345 672","aria-hidden":"true"},q=s("g",{stroke:"currentColor",fill:"currentColor","stroke-width":"0",transform:"scale(1,-1)"},[s("g",{"data-mml-node":"math"},[s("g",{"data-mml-node":"mi"},[s("path",{"data-c":"1D456",d:"M184 600Q184 624 203 642T247 661Q265 661 277 649T290 619Q290 596 270 577T226 557Q211 557 198 567T184 600ZM21 287Q21 295 30 318T54 369T98 420T158 442Q197 442 223 419T250 357Q250 340 236 301T196 196T154 83Q149 61 149 51Q149 26 166 26Q175 26 185 29T208 43T235 78T260 137Q263 149 265 151T282 153Q302 153 302 143Q302 135 293 112T268 61T223 11T161 -11Q129 -11 102 10T74 74Q74 91 79 106T122 220Q160 321 166 341T173 380Q173 404 156 404H154Q124 404 99 371T61 287Q60 286 59 284T58 281T56 279T53 278T49 278T41 278H27Q21 284 21 287Z",style:{"stroke-width":"3"}})])])],-1),S=[q],P=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("mi",null,"i")])],-1),j={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},I={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.05ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.919ex",height:"1.645ex",role:"img",focusable:"false",viewBox:"0 -705 1290 727","aria-hidden":"true"},G=e("",1),O=[G],N=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("mi",null,"S"),s("mi",null,"S")])],-1),R={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},z={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.05ex"},xmlns:"http://www.w3.org/2000/svg",width:"8.631ex",height:"1.645ex",role:"img",focusable:"false",viewBox:"0 -705 3815 727","aria-hidden":"true"},J=e("",1),K=[J],U=s("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[s("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[s("mi",null,"G"),s("mi",null,"C"),s("mi",null,"P"),s("mi",null,"B"),s("mi",null,"B")])],-1),W=e("",25);function X($,Y,ss,as,ns,es){return a(),n("div",null,[p,s("p",null,[s("mjx-container",r,[(a(),n("svg",c,i)),T])]),d,s("p",null,[s("mjx-container",y,[(a(),n("svg",h,m)),u])]),g,s("ul",null,[s("li",null,[s("mjx-container",b,[(a(),n("svg",f,F)),w]),t(" = Fixed Cost, is a static value (65,000 gas)")]),s("li",null,[s("mjx-container",x,[(a(),n("svg",V,C)),L]),t(" = SparseSharesNeeded for the "),s("mjx-container",_,[(a(),n("svg",D,M)),A]),t("th Blob, is the number of shares needed for the "),s("mjx-container",v,[(a(),n("svg",Z,S)),P]),t("th blob in the transaction")]),s("li",null,[s("mjx-container",j,[(a(),n("svg",I,O)),N]),t(" = Share Size, is the size of each share")]),s("li",null,[s("mjx-container",R,[(a(),n("svg",z,K)),U]),t(" = Gas Cost Per Blob Byte, is a parameter that could potentially be adjusted through the system's governance mechanisms.")])]),W])}const ls=o(l,[["render",X]]);export{os as __pageData,ls as default}; diff --git a/assets/developers_transaction-resubmission.md.96bd7ea9.js b/assets/developers_transaction-resubmission.md.96bd7ea9.js new file mode 100644 index 00000000000..8924d158f2d --- /dev/null +++ b/assets/developers_transaction-resubmission.md.96bd7ea9.js @@ -0,0 +1 @@ +import{_ as e,o as s,c as n,Q as i}from"./chunks/framework.51d6c45b.js";const b=JSON.parse('{"title":"Transaction resubmission","description":"This is a guide on transaction resubmission on Celestia.","frontmatter":{"description":"This is a guide on transaction resubmission on Celestia.","head":[["meta",{"name":"og:title","content":"Transaction resubmission | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/transaction-resubmission.md","filePath":"developers/transaction-resubmission.md","lastUpdated":1708339816000}'),t={name:"developers/transaction-resubmission.md"},a=i('

    Transaction resubmission

    In cases where transactions are not included within a 75-second window, resubmission is necessary. This is especially important during network congestion, as transactions with relatively low fees may not be processed even after the network clears up.

    Regardless of whether they originate from celestia-app or celestia-node, transactions will not be re-gossiped, except in the presence of a new peer.

    Monitoring and resubmission

    Monitor the status of your transactions. If a transaction is not included within a 75-second window, it should be resubmitted. This can be done manually or through automated processes.

    Changes introduced in celestiaorg/celestia-core#1089 may affect transaction gossiping and inclusion speed.

    Notes

    • All transactions, regardless of their origin, are subject to being sorted and pruned based on fees.
    • It is the user or developer's responsibility to monitor and possibly resubmit transactions if they are not included in a 75-second window.
    ',8),o=[a];function r(c,d,l,u,h,m){return s(),n("div",null,o)}const _=e(t,[["render",r]]);export{b as __pageData,_ as default}; diff --git a/assets/developers_transaction-resubmission.md.96bd7ea9.lean.js b/assets/developers_transaction-resubmission.md.96bd7ea9.lean.js new file mode 100644 index 00000000000..2b2658b9908 --- /dev/null +++ b/assets/developers_transaction-resubmission.md.96bd7ea9.lean.js @@ -0,0 +1 @@ +import{_ as e,o as s,c as n,Q as i}from"./chunks/framework.51d6c45b.js";const b=JSON.parse('{"title":"Transaction resubmission","description":"This is a guide on transaction resubmission on Celestia.","frontmatter":{"description":"This is a guide on transaction resubmission on Celestia.","head":[["meta",{"name":"og:title","content":"Transaction resubmission | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/transaction-resubmission.md","filePath":"developers/transaction-resubmission.md","lastUpdated":1708339816000}'),t={name:"developers/transaction-resubmission.md"},a=i("",8),o=[a];function r(c,d,l,u,h,m){return s(),n("div",null,o)}const _=e(t,[["render",r]]);export{b as __pageData,_ as default}; diff --git a/assets/developers_wallets.md.a10fbd85.js b/assets/developers_wallets.md.a10fbd85.js new file mode 100644 index 00000000000..8c2c39a4c4a --- /dev/null +++ b/assets/developers_wallets.md.a10fbd85.js @@ -0,0 +1,203 @@ +import{c as p}from"./chunks/constants.dc70dfe7.js";import{_ as e,o as t,c,k as s,a as n,t as l,Q as o}from"./chunks/framework.51d6c45b.js";const r={data(){return{constants:p}}},hn=JSON.parse('{"title":"Wallet integrations with Celestia","description":"How you can add Celestia network parameters to wallets such as Keplr, Leap, and Cosmostation.","frontmatter":{"description":"How you can add Celestia network parameters to wallets such as Keplr, Leap, and Cosmostation.","head":[["meta",{"name":"og:title","content":"Wallet integrations with Celestia | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/wallets.md","filePath":"developers/wallets.md","lastUpdated":1713191709000}'),i=o(`

    Wallet integrations with Celestia

    This page covers how developers can use Keplr and React to add Celestia network parameters to wallets, and how to add custom networks to Leap and Cosmostation.

    Add Celestia network parameters to Keplr with React

    Before we demonstrate how to export the specific parameters for Celestia's testnets, we need to create a ReactJS component that allows us to connect directly to Keplr and pass it the network parameters.

    In the following code, we show how you can export a component that detects whether Keplr is installed and sets the network params for it:

    jsx
    // @site/src/components/AddNetworkKeplr.js
    +import React from "react";
    +import styles from "./Keplr.module.css";
    +
    +export default function AddNetworkKeplr({ params }) {
    +  async function add() {
    +    if (!window.keplr) {
    +      alert("Please install keplr extension");
    +    } else {
    +      if (window.keplr.experimentalSuggestChain) {
    +        try {
    +          await window.keplr.experimentalSuggestChain({
    +            chainId: params.chainId,
    +            chainName: params.chainName,
    +            rpc: params.rpc,
    +            rest: params.rest,
    +            bip44: {
    +              coinType: 118,
    +            },
    +            bech32Config: {
    +              bech32PrefixAccAddr: "celestia",
    +              bech32PrefixAccPub: "celestia" + "pub",
    +              bech32PrefixValAddr: "celestia" + "valoper",
    +              bech32PrefixValPub: "celestia" + "valoperpub",
    +              bech32PrefixConsAddr: "celestia" + "valcons",
    +              bech32PrefixConsPub: "celestia" + "valconspub",
    +            },
    +            currencies: [
    +              {
    +                coinDenom: "TIA",
    +                coinMinimalDenom: "utia",
    +                coinDecimals: 6,
    +                coinGeckoId: "celestia",
    +              },
    +            ],
    +            feeCurrencies: [
    +              {
    +                coinDenom: "TIA",
    +                coinMinimalDenom: "utia",
    +                coinDecimals: 6,
    +                coinGeckoId: "celestia",
    +                gasPriceStep: {
    +                  low: 0.01,
    +                  average: 0.02,
    +                  high: 0.1,
    +                },
    +              },
    +            ],
    +            stakeCurrency: {
    +              coinDenom: "TIA",
    +              coinMinimalDenom: "utia",
    +              coinDecimals: 6,
    +              coinGeckoId: "celestia",
    +            },
    +          });
    +        } catch {
    +          alert("Failed to suggest the chain");
    +        }
    +      }
    +      const chainId = params.chainId;
    +      // Enabling before using the Keplr is recommended.
    +      // This method will ask the user whether to allow access if they haven't visited this website.
    +      // Also, it will request that the user unlock the wallet if the wallet is locked.
    +      await window.keplr.enable(chainId);
    +    }
    +  }
    +
    +  return (
    +    <div className={styles.center}>
    +      <button className={styles.keplrButton} onClick={add}>
    +        Add/switch To {params.chainName}
    +      </button>
    +    </div>
    +  );
    +}
    // @site/src/components/AddNetworkKeplr.js
    +import React from "react";
    +import styles from "./Keplr.module.css";
    +
    +export default function AddNetworkKeplr({ params }) {
    +  async function add() {
    +    if (!window.keplr) {
    +      alert("Please install keplr extension");
    +    } else {
    +      if (window.keplr.experimentalSuggestChain) {
    +        try {
    +          await window.keplr.experimentalSuggestChain({
    +            chainId: params.chainId,
    +            chainName: params.chainName,
    +            rpc: params.rpc,
    +            rest: params.rest,
    +            bip44: {
    +              coinType: 118,
    +            },
    +            bech32Config: {
    +              bech32PrefixAccAddr: "celestia",
    +              bech32PrefixAccPub: "celestia" + "pub",
    +              bech32PrefixValAddr: "celestia" + "valoper",
    +              bech32PrefixValPub: "celestia" + "valoperpub",
    +              bech32PrefixConsAddr: "celestia" + "valcons",
    +              bech32PrefixConsPub: "celestia" + "valconspub",
    +            },
    +            currencies: [
    +              {
    +                coinDenom: "TIA",
    +                coinMinimalDenom: "utia",
    +                coinDecimals: 6,
    +                coinGeckoId: "celestia",
    +              },
    +            ],
    +            feeCurrencies: [
    +              {
    +                coinDenom: "TIA",
    +                coinMinimalDenom: "utia",
    +                coinDecimals: 6,
    +                coinGeckoId: "celestia",
    +                gasPriceStep: {
    +                  low: 0.01,
    +                  average: 0.02,
    +                  high: 0.1,
    +                },
    +              },
    +            ],
    +            stakeCurrency: {
    +              coinDenom: "TIA",
    +              coinMinimalDenom: "utia",
    +              coinDecimals: 6,
    +              coinGeckoId: "celestia",
    +            },
    +          });
    +        } catch {
    +          alert("Failed to suggest the chain");
    +        }
    +      }
    +      const chainId = params.chainId;
    +      // Enabling before using the Keplr is recommended.
    +      // This method will ask the user whether to allow access if they haven't visited this website.
    +      // Also, it will request that the user unlock the wallet if the wallet is locked.
    +      await window.keplr.enable(chainId);
    +    }
    +  }
    +
    +  return (
    +    <div className={styles.center}>
    +      <button className={styles.keplrButton} onClick={add}>
    +        Add/switch To {params.chainName}
    +      </button>
    +    </div>
    +  );
    +}

    We still need to pass the Celestia network parameters to the AddNetworkKeplr function:

    `,7),E={class:"vp-code-group vp-adaptive-theme"},y=o('
    ',1),d={class:"blocks"},_={class:"language-js vp-adaptive-theme active"},h=s("button",{title:"Copy Code",class:"copy"},null,-1),u=s("span",{class:"lang"},"js",-1),F={class:"shiki github-dark vp-code-dark"},A=s("span",{class:"line"},[s("span",{style:{color:"#F97583"}},"import"),s("span",{style:{color:"#E1E4E8"}}," "),s("span",{style:{color:"#9ECBFF"}},"'@site/src/components/AddNetworkKeplr'")],-1),C=s("span",{class:"line"},null,-1),m=o('export const MAINNET_PARAMS = {`{',1),w={class:"line"},T={style:{color:"#9ECBFF"}},q=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}}," chainName: 'Celestia',")],-1),b={class:"line"},g={style:{color:"#9ECBFF"}},D={class:"line"},k={style:{color:"#9ECBFF"}},B=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}},"}`"),s("span",{style:{color:"#E1E4E8"}},"}")],-1),I=s("span",{class:"line"},null,-1),P=o('{<AddNetworkKeplr params={MAINNET_PARAMS}/>}',1),S={class:"shiki github-light vp-code-light"},v=s("span",{class:"line"},[s("span",{style:{color:"#D73A49"}},"import"),s("span",{style:{color:"#24292E"}}," "),s("span",{style:{color:"#032F62"}},"'@site/src/components/AddNetworkKeplr'")],-1),f=s("span",{class:"line"},null,-1),N=o('export const MAINNET_PARAMS = {`{',1),R={class:"line"},x={style:{color:"#032F62"}},V=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}}," chainName: 'Celestia',")],-1),M={class:"line"},K={style:{color:"#032F62"}},U={class:"line"},L={style:{color:"#032F62"}},j=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}},"}`"),s("span",{style:{color:"#24292E"}},"}")],-1),W=s("span",{class:"line"},null,-1),G=o('{<AddNetworkKeplr params={MAINNET_PARAMS}/>}',1),J={class:"language-js vp-adaptive-theme"},H=s("button",{title:"Copy Code",class:"copy"},null,-1),Y=s("span",{class:"lang"},"js",-1),O={class:"shiki github-dark vp-code-dark"},Q=s("span",{class:"line"},[s("span",{style:{color:"#F97583"}},"import"),s("span",{style:{color:"#E1E4E8"}}," "),s("span",{style:{color:"#9ECBFF"}},"'@site/src/components/AddNetworkKeplr'")],-1),z=s("span",{class:"line"},null,-1),X=o('export const MOCHA_PARAMS = {`{',1),Z={class:"line"},$={style:{color:"#9ECBFF"}},ss=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}}," chainName: 'Mocha testnet',")],-1),ns={class:"line"},as={style:{color:"#9ECBFF"}},ls={class:"line"},os={style:{color:"#9ECBFF"}},ps=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}},"}`"),s("span",{style:{color:"#E1E4E8"}},"}")],-1),es=s("span",{class:"line"},null,-1),ts=o('{<AddNetworkKeplr params={MOCHA_PARAMS}/>}',1),cs={class:"shiki github-light vp-code-light"},rs=s("span",{class:"line"},[s("span",{style:{color:"#D73A49"}},"import"),s("span",{style:{color:"#24292E"}}," "),s("span",{style:{color:"#032F62"}},"'@site/src/components/AddNetworkKeplr'")],-1),is=s("span",{class:"line"},null,-1),Es=o('export const MOCHA_PARAMS = {`{',1),ys={class:"line"},ds={style:{color:"#032F62"}},_s=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}}," chainName: 'Mocha testnet',")],-1),hs={class:"line"},us={style:{color:"#032F62"}},Fs={class:"line"},As={style:{color:"#032F62"}},Cs=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}},"}`"),s("span",{style:{color:"#24292E"}},"}")],-1),ms=s("span",{class:"line"},null,-1),ws=o('{<AddNetworkKeplr params={MOCHA_PARAMS}/>}',1),Ts={class:"language-js vp-adaptive-theme"},qs=s("button",{title:"Copy Code",class:"copy"},null,-1),bs=s("span",{class:"lang"},"js",-1),gs={class:"shiki github-dark vp-code-dark"},Ds=s("span",{class:"line"},[s("span",{style:{color:"#F97583"}},"import"),s("span",{style:{color:"#E1E4E8"}}," "),s("span",{style:{color:"#9ECBFF"}},"'@site/src/components/AddNetworkKeplr'")],-1),ks=s("span",{class:"line"},null,-1),Bs=o('export const ARABICA_PARAMS = {`{',1),Is={class:"line"},Ps={style:{color:"#9ECBFF"}},Ss=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}}," chainName: 'Arabica devnet',")],-1),vs={class:"line"},fs={style:{color:"#9ECBFF"}},Ns={class:"line"},Rs={style:{color:"#9ECBFF"}},xs=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}},"}`"),s("span",{style:{color:"#E1E4E8"}},"}")],-1),Vs=s("span",{class:"line"},null,-1),Ms=o('{<AddNetworkKeplr params={ARABICA_PARAMS}/>}',1),Ks={class:"shiki github-light vp-code-light"},Us=s("span",{class:"line"},[s("span",{style:{color:"#D73A49"}},"import"),s("span",{style:{color:"#24292E"}}," "),s("span",{style:{color:"#032F62"}},"'@site/src/components/AddNetworkKeplr'")],-1),Ls=s("span",{class:"line"},null,-1),js=o('export const ARABICA_PARAMS = {`{',1),Ws={class:"line"},Gs={style:{color:"#032F62"}},Js=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}}," chainName: 'Arabica devnet',")],-1),Hs={class:"line"},Ys={style:{color:"#032F62"}},Os={class:"line"},Qs={style:{color:"#032F62"}},zs=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}},"}`"),s("span",{style:{color:"#24292E"}},"}")],-1),Xs=s("span",{class:"line"},null,-1),Zs=o('{<AddNetworkKeplr params={ARABICA_PARAMS}/>}',1),$s=s("p",null,"Now, we can connect to the network that you would like to use in Keplr wallet.",-1),sn=s("h2",{id:"adding-a-custom-chain-to-leap",tabindex:"-1"},[n("Adding a custom chain to Leap "),s("a",{class:"header-anchor",href:"#adding-a-custom-chain-to-leap","aria-label":'Permalink to "Adding a custom chain to Leap"'},"​")],-1),nn=s("p",null,"If you want to add a custom chain to Leap, you can do so by:",-1),an=s("ol",null,[s("li",null,"Clicking the Cosmos logo in the top corner of Leap wallet"),s("li",null,'Scrolling down and clicking "Add new chain"')],-1),ln=s("p",null,"You can then add the following parameters:",-1),on=o("
  • Chain Name: Arabica devnet
  • New RPC URL: https://rpc.celestia-arabica-11.com/
  • New REST URL: https://api.celestia-arabica-11.com
  • Address Prefix: celestia
  • Native Denom: utia
  • Coin Type: 118
  • Decimals: 6
  • Block explorer URL (optional): https://explorer.celestia-arabica-10.com
  • ",8),pn=o('

    Now, click Add chain and you will be able to view your Arabica account balance and transactions in Leap wallet.

    You'll see that you're connected to Arabica Devnet.

    Adding a custom chain to Cosmostation

    Click the hamburger menu icon in the top corner of Cosmostation wallet. Scroll down and click "Add Custom Chain"

    You can then add the following parameters:

    • Custom Chain name: Mocha testnet
    • Rest URL: https://api-mocha.pops.one
    • New RPC URL: https://rpc-mocha.pops.one
    • Currency symbol: TIA
    • Address prefix: celestia
    • Denom: utia
    • Symbol image URL (optional): https://raw.githubusercontent.com/cosmos/chain-registry/master/testnets/celestiatestnet/images/celestia.svg
    • Explorer URL (optional): https://testnet.mintscan.io/celestia-testnet
    • Coin Type: 118
    • Decimals: 6
    • Gas rate Tiny: 0.1
    • Gas rate Low: 0.25
    • Gas rate Average: 0.5

    Now, click Add a custom chain and you will be able to view your Celestia account balance and transactions in Cosmostation wallet.

    Switch chains to "Mocha testnet" and you'll see that you're connected to Celestia's Mocha testnet!

    ',8);function en(tn,cn,rn,En,a,yn){return t(),c("div",null,[i,s("div",E,[y,s("div",d,[s("div",_,[h,u,s("pre",F,[s("code",null,[A,n(` +`),C,n(` +`),m,n(` +`),s("span",w,[s("span",T," chainId: '"+l(a.constants.mainnetChainId)+"',",1)]),n(` +`),q,n(` +`),s("span",b,[s("span",g," rpc: '"+l(a.constants.mainnetRpcUrl)+"',",1)]),n(` +`),s("span",D,[s("span",k," rest: '"+l(a.constants.mainnetRestUrl)+"'",1)]),n(` +`),B,n(` +`),I,n(` +`),P])]),s("pre",S,[s("code",null,[v,n(` +`),f,n(` +`),N,n(` +`),s("span",R,[s("span",x," chainId: '"+l(a.constants.mainnetChainId)+"',",1)]),n(` +`),V,n(` +`),s("span",M,[s("span",K," rpc: '"+l(a.constants.mainnetRpcUrl)+"',",1)]),n(` +`),s("span",U,[s("span",L," rest: '"+l(a.constants.mainnetRestUrl)+"'",1)]),n(` +`),j,n(` +`),W,n(` +`),G])])]),s("div",J,[H,Y,s("pre",O,[s("code",null,[Q,n(` +`),z,n(` +`),X,n(` +`),s("span",Z,[s("span",$," chainId: '"+l(a.constants.mochaChainId)+"',",1)]),n(` +`),ss,n(` +`),s("span",ns,[s("span",as," rpc: '"+l(a.constants.mochaRpcUrl)+"',",1)]),n(` +`),s("span",ls,[s("span",os," rest: '"+l(a.constants.mochaRestUrl)+"'",1)]),n(` +`),ps,n(` +`),es,n(` +`),ts])]),s("pre",cs,[s("code",null,[rs,n(` +`),is,n(` +`),Es,n(` +`),s("span",ys,[s("span",ds," chainId: '"+l(a.constants.mochaChainId)+"',",1)]),n(` +`),_s,n(` +`),s("span",hs,[s("span",us," rpc: '"+l(a.constants.mochaRpcUrl)+"',",1)]),n(` +`),s("span",Fs,[s("span",As," rest: '"+l(a.constants.mochaRestUrl)+"'",1)]),n(` +`),Cs,n(` +`),ms,n(` +`),ws])])]),s("div",Ts,[qs,bs,s("pre",gs,[s("code",null,[Ds,n(` +`),ks,n(` +`),Bs,n(` +`),s("span",Is,[s("span",Ps," chainId: '"+l(a.constants.arabicaChainId)+"',",1)]),n(` +`),Ss,n(` +`),s("span",vs,[s("span",fs," rpc: '"+l(a.constants.arabicaRpcUrl)+"',",1)]),n(` +`),s("span",Ns,[s("span",Rs," rest: '"+l(a.constants.arabicaRestUrl)+"'",1)]),n(` +`),xs,n(` +`),Vs,n(` +`),Ms])]),s("pre",Ks,[s("code",null,[Us,n(` +`),Ls,n(` +`),js,n(` +`),s("span",Ws,[s("span",Gs," chainId: '"+l(a.constants.arabicaChainId)+"',",1)]),n(` +`),Js,n(` +`),s("span",Hs,[s("span",Ys," rpc: '"+l(a.constants.arabicaRpcUrl)+"',",1)]),n(` +`),s("span",Os,[s("span",Qs," rest: '"+l(a.constants.arabicaRestUrl)+"'",1)]),n(` +`),zs,n(` +`),Xs,n(` +`),Zs])])])])]),$s,sn,nn,an,ln,s("ul",null,[s("li",null,[n("Chain Id: "),s("code",null,l(a.constants.arabicaChainId),1)]),on]),pn])}const un=e(r,[["render",en]]);export{hn as __pageData,un as default}; diff --git a/assets/developers_wallets.md.a10fbd85.lean.js b/assets/developers_wallets.md.a10fbd85.lean.js new file mode 100644 index 00000000000..ffa366afe83 --- /dev/null +++ b/assets/developers_wallets.md.a10fbd85.lean.js @@ -0,0 +1,55 @@ +import{c as p}from"./chunks/constants.dc70dfe7.js";import{_ as e,o as t,c,k as s,a as n,t as l,Q as o}from"./chunks/framework.51d6c45b.js";const r={data(){return{constants:p}}},hn=JSON.parse('{"title":"Wallet integrations with Celestia","description":"How you can add Celestia network parameters to wallets such as Keplr, Leap, and Cosmostation.","frontmatter":{"description":"How you can add Celestia network parameters to wallets such as Keplr, Leap, and Cosmostation.","head":[["meta",{"name":"og:title","content":"Wallet integrations with Celestia | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/wallets.md","filePath":"developers/wallets.md","lastUpdated":1713191709000}'),i=o("",7),E={class:"vp-code-group vp-adaptive-theme"},y=o("",1),d={class:"blocks"},_={class:"language-js vp-adaptive-theme active"},h=s("button",{title:"Copy Code",class:"copy"},null,-1),u=s("span",{class:"lang"},"js",-1),F={class:"shiki github-dark vp-code-dark"},A=s("span",{class:"line"},[s("span",{style:{color:"#F97583"}},"import"),s("span",{style:{color:"#E1E4E8"}}," "),s("span",{style:{color:"#9ECBFF"}},"'@site/src/components/AddNetworkKeplr'")],-1),C=s("span",{class:"line"},null,-1),m=o("",1),w={class:"line"},T={style:{color:"#9ECBFF"}},q=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}}," chainName: 'Celestia',")],-1),b={class:"line"},g={style:{color:"#9ECBFF"}},D={class:"line"},k={style:{color:"#9ECBFF"}},B=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}},"}`"),s("span",{style:{color:"#E1E4E8"}},"}")],-1),I=s("span",{class:"line"},null,-1),P=o("",1),S={class:"shiki github-light vp-code-light"},v=s("span",{class:"line"},[s("span",{style:{color:"#D73A49"}},"import"),s("span",{style:{color:"#24292E"}}," "),s("span",{style:{color:"#032F62"}},"'@site/src/components/AddNetworkKeplr'")],-1),f=s("span",{class:"line"},null,-1),N=o("",1),R={class:"line"},x={style:{color:"#032F62"}},V=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}}," chainName: 'Celestia',")],-1),M={class:"line"},K={style:{color:"#032F62"}},U={class:"line"},L={style:{color:"#032F62"}},j=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}},"}`"),s("span",{style:{color:"#24292E"}},"}")],-1),W=s("span",{class:"line"},null,-1),G=o("",1),J={class:"language-js vp-adaptive-theme"},H=s("button",{title:"Copy Code",class:"copy"},null,-1),Y=s("span",{class:"lang"},"js",-1),O={class:"shiki github-dark vp-code-dark"},Q=s("span",{class:"line"},[s("span",{style:{color:"#F97583"}},"import"),s("span",{style:{color:"#E1E4E8"}}," "),s("span",{style:{color:"#9ECBFF"}},"'@site/src/components/AddNetworkKeplr'")],-1),z=s("span",{class:"line"},null,-1),X=o("",1),Z={class:"line"},$={style:{color:"#9ECBFF"}},ss=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}}," chainName: 'Mocha testnet',")],-1),ns={class:"line"},as={style:{color:"#9ECBFF"}},ls={class:"line"},os={style:{color:"#9ECBFF"}},ps=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}},"}`"),s("span",{style:{color:"#E1E4E8"}},"}")],-1),es=s("span",{class:"line"},null,-1),ts=o("",1),cs={class:"shiki github-light vp-code-light"},rs=s("span",{class:"line"},[s("span",{style:{color:"#D73A49"}},"import"),s("span",{style:{color:"#24292E"}}," "),s("span",{style:{color:"#032F62"}},"'@site/src/components/AddNetworkKeplr'")],-1),is=s("span",{class:"line"},null,-1),Es=o("",1),ys={class:"line"},ds={style:{color:"#032F62"}},_s=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}}," chainName: 'Mocha testnet',")],-1),hs={class:"line"},us={style:{color:"#032F62"}},Fs={class:"line"},As={style:{color:"#032F62"}},Cs=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}},"}`"),s("span",{style:{color:"#24292E"}},"}")],-1),ms=s("span",{class:"line"},null,-1),ws=o("",1),Ts={class:"language-js vp-adaptive-theme"},qs=s("button",{title:"Copy Code",class:"copy"},null,-1),bs=s("span",{class:"lang"},"js",-1),gs={class:"shiki github-dark vp-code-dark"},Ds=s("span",{class:"line"},[s("span",{style:{color:"#F97583"}},"import"),s("span",{style:{color:"#E1E4E8"}}," "),s("span",{style:{color:"#9ECBFF"}},"'@site/src/components/AddNetworkKeplr'")],-1),ks=s("span",{class:"line"},null,-1),Bs=o("",1),Is={class:"line"},Ps={style:{color:"#9ECBFF"}},Ss=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}}," chainName: 'Arabica devnet',")],-1),vs={class:"line"},fs={style:{color:"#9ECBFF"}},Ns={class:"line"},Rs={style:{color:"#9ECBFF"}},xs=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}},"}`"),s("span",{style:{color:"#E1E4E8"}},"}")],-1),Vs=s("span",{class:"line"},null,-1),Ms=o("",1),Ks={class:"shiki github-light vp-code-light"},Us=s("span",{class:"line"},[s("span",{style:{color:"#D73A49"}},"import"),s("span",{style:{color:"#24292E"}}," "),s("span",{style:{color:"#032F62"}},"'@site/src/components/AddNetworkKeplr'")],-1),Ls=s("span",{class:"line"},null,-1),js=o("",1),Ws={class:"line"},Gs={style:{color:"#032F62"}},Js=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}}," chainName: 'Arabica devnet',")],-1),Hs={class:"line"},Ys={style:{color:"#032F62"}},Os={class:"line"},Qs={style:{color:"#032F62"}},zs=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}},"}`"),s("span",{style:{color:"#24292E"}},"}")],-1),Xs=s("span",{class:"line"},null,-1),Zs=o("",1),$s=s("p",null,"Now, we can connect to the network that you would like to use in Keplr wallet.",-1),sn=s("h2",{id:"adding-a-custom-chain-to-leap",tabindex:"-1"},[n("Adding a custom chain to Leap "),s("a",{class:"header-anchor",href:"#adding-a-custom-chain-to-leap","aria-label":'Permalink to "Adding a custom chain to Leap"'},"​")],-1),nn=s("p",null,"If you want to add a custom chain to Leap, you can do so by:",-1),an=s("ol",null,[s("li",null,"Clicking the Cosmos logo in the top corner of Leap wallet"),s("li",null,'Scrolling down and clicking "Add new chain"')],-1),ln=s("p",null,"You can then add the following parameters:",-1),on=o("",8),pn=o("",8);function en(tn,cn,rn,En,a,yn){return t(),c("div",null,[i,s("div",E,[y,s("div",d,[s("div",_,[h,u,s("pre",F,[s("code",null,[A,n(` +`),C,n(` +`),m,n(` +`),s("span",w,[s("span",T," chainId: '"+l(a.constants.mainnetChainId)+"',",1)]),n(` +`),q,n(` +`),s("span",b,[s("span",g," rpc: '"+l(a.constants.mainnetRpcUrl)+"',",1)]),n(` +`),s("span",D,[s("span",k," rest: '"+l(a.constants.mainnetRestUrl)+"'",1)]),n(` +`),B,n(` +`),I,n(` +`),P])]),s("pre",S,[s("code",null,[v,n(` +`),f,n(` +`),N,n(` +`),s("span",R,[s("span",x," chainId: '"+l(a.constants.mainnetChainId)+"',",1)]),n(` +`),V,n(` +`),s("span",M,[s("span",K," rpc: '"+l(a.constants.mainnetRpcUrl)+"',",1)]),n(` +`),s("span",U,[s("span",L," rest: '"+l(a.constants.mainnetRestUrl)+"'",1)]),n(` +`),j,n(` +`),W,n(` +`),G])])]),s("div",J,[H,Y,s("pre",O,[s("code",null,[Q,n(` +`),z,n(` +`),X,n(` +`),s("span",Z,[s("span",$," chainId: '"+l(a.constants.mochaChainId)+"',",1)]),n(` +`),ss,n(` +`),s("span",ns,[s("span",as," rpc: '"+l(a.constants.mochaRpcUrl)+"',",1)]),n(` +`),s("span",ls,[s("span",os," rest: '"+l(a.constants.mochaRestUrl)+"'",1)]),n(` +`),ps,n(` +`),es,n(` +`),ts])]),s("pre",cs,[s("code",null,[rs,n(` +`),is,n(` +`),Es,n(` +`),s("span",ys,[s("span",ds," chainId: '"+l(a.constants.mochaChainId)+"',",1)]),n(` +`),_s,n(` +`),s("span",hs,[s("span",us," rpc: '"+l(a.constants.mochaRpcUrl)+"',",1)]),n(` +`),s("span",Fs,[s("span",As," rest: '"+l(a.constants.mochaRestUrl)+"'",1)]),n(` +`),Cs,n(` +`),ms,n(` +`),ws])])]),s("div",Ts,[qs,bs,s("pre",gs,[s("code",null,[Ds,n(` +`),ks,n(` +`),Bs,n(` +`),s("span",Is,[s("span",Ps," chainId: '"+l(a.constants.arabicaChainId)+"',",1)]),n(` +`),Ss,n(` +`),s("span",vs,[s("span",fs," rpc: '"+l(a.constants.arabicaRpcUrl)+"',",1)]),n(` +`),s("span",Ns,[s("span",Rs," rest: '"+l(a.constants.arabicaRestUrl)+"'",1)]),n(` +`),xs,n(` +`),Vs,n(` +`),Ms])]),s("pre",Ks,[s("code",null,[Us,n(` +`),Ls,n(` +`),js,n(` +`),s("span",Ws,[s("span",Gs," chainId: '"+l(a.constants.arabicaChainId)+"',",1)]),n(` +`),Js,n(` +`),s("span",Hs,[s("span",Ys," rpc: '"+l(a.constants.arabicaRpcUrl)+"',",1)]),n(` +`),s("span",Os,[s("span",Qs," rest: '"+l(a.constants.arabicaRestUrl)+"'",1)]),n(` +`),zs,n(` +`),Xs,n(` +`),Zs])])])])]),$s,sn,nn,an,ln,s("ul",null,[s("li",null,[n("Chain Id: "),s("code",null,l(a.constants.arabicaChainId),1)]),on]),pn])}const un=e(r,[["render",en]]);export{hn as __pageData,un as default}; diff --git a/assets/index.md.10ada84c.js b/assets/index.md.10ada84c.js new file mode 100644 index 00000000000..0a46c179bb4 --- /dev/null +++ b/assets/index.md.10ada84c.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a}from"./chunks/framework.51d6c45b.js";const u=JSON.parse('{"title":"","titleTemplate":":title","description":"","frontmatter":{"layout":"home","titleTemplate":":title","hero":{"name":"Celestia","text":"The first modular blockchain network","tagline":"Celestia is a modular data availability network that securely scales with the number of users, making it easy for anyone to launch their own blockchain.","image":{"src":"/modular.svg","alt":"Celestia"},"actions":[{"theme":"brand","text":"Build whatever","link":"/developers/build-whatever"},{"theme":"alt","text":"Introduction","link":"/learn/how-celestia-works/overview"}]},"features":[{"title":"Learn","details":"Celestia allows you to deploy your own blockchain in minutes, as easily as a smart contract.","link":"/learn/how-celestia-works/overview","icon":"🏗️"},{"title":"Run a node","details":"Access the dynamic scaling unlocked by data availability sampling, where scale increases with the number of users.","link":"/nodes/overview","icon":"📈"},{"title":"Developers","details":"Create applications using your favorite VM or define your own. Build sovereign rollups, a new type of self-governing blockchain with minimal platform risk.","link":"/developers/build-whatever","icon":"⚙️"},{"title":"Community","details":"Join the Celestia community to connect, collaborate, and contribute to the future of modular blockchains.","link":"/community/overview","icon":"🏰"}],"head":[["meta",{"name":"og:title","content":"Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"index.md","filePath":"index.md","lastUpdated":1712870672000}'),i={name:"index.md"};function o(n,l,s,r,c,d){return t(),a("div")}const h=e(i,[["render",o]]);export{u as __pageData,h as default}; diff --git a/assets/index.md.10ada84c.lean.js b/assets/index.md.10ada84c.lean.js new file mode 100644 index 00000000000..0a46c179bb4 --- /dev/null +++ b/assets/index.md.10ada84c.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a}from"./chunks/framework.51d6c45b.js";const u=JSON.parse('{"title":"","titleTemplate":":title","description":"","frontmatter":{"layout":"home","titleTemplate":":title","hero":{"name":"Celestia","text":"The first modular blockchain network","tagline":"Celestia is a modular data availability network that securely scales with the number of users, making it easy for anyone to launch their own blockchain.","image":{"src":"/modular.svg","alt":"Celestia"},"actions":[{"theme":"brand","text":"Build whatever","link":"/developers/build-whatever"},{"theme":"alt","text":"Introduction","link":"/learn/how-celestia-works/overview"}]},"features":[{"title":"Learn","details":"Celestia allows you to deploy your own blockchain in minutes, as easily as a smart contract.","link":"/learn/how-celestia-works/overview","icon":"🏗️"},{"title":"Run a node","details":"Access the dynamic scaling unlocked by data availability sampling, where scale increases with the number of users.","link":"/nodes/overview","icon":"📈"},{"title":"Developers","details":"Create applications using your favorite VM or define your own. Build sovereign rollups, a new type of self-governing blockchain with minimal platform risk.","link":"/developers/build-whatever","icon":"⚙️"},{"title":"Community","details":"Join the Celestia community to connect, collaborate, and contribute to the future of modular blockchains.","link":"/community/overview","icon":"🏰"}],"head":[["meta",{"name":"og:title","content":"Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"index.md","filePath":"index.md","lastUpdated":1712870672000}'),i={name:"index.md"};function o(n,l,s,r,c,d){return t(),a("div")}const h=e(i,[["render",o]]);export{u as __pageData,h as default}; diff --git a/assets/inter-italic-cyrillic-ext.33bd5a8e.woff2 b/assets/inter-italic-cyrillic-ext.33bd5a8e.woff2 new file mode 100644 index 00000000000..2a687296748 Binary files /dev/null and b/assets/inter-italic-cyrillic-ext.33bd5a8e.woff2 differ diff --git a/assets/inter-italic-cyrillic.ea42a392.woff2 b/assets/inter-italic-cyrillic.ea42a392.woff2 new file mode 100644 index 00000000000..f64035158d7 Binary files /dev/null and b/assets/inter-italic-cyrillic.ea42a392.woff2 differ diff --git a/assets/inter-italic-greek-ext.4fbe9427.woff2 b/assets/inter-italic-greek-ext.4fbe9427.woff2 new file mode 100644 index 00000000000..00218960325 Binary files /dev/null and b/assets/inter-italic-greek-ext.4fbe9427.woff2 differ diff --git a/assets/inter-italic-greek.8f4463c4.woff2 b/assets/inter-italic-greek.8f4463c4.woff2 new file mode 100644 index 00000000000..71c265f85c9 Binary files /dev/null and b/assets/inter-italic-greek.8f4463c4.woff2 differ diff --git a/assets/inter-italic-latin-ext.bd8920cc.woff2 b/assets/inter-italic-latin-ext.bd8920cc.woff2 new file mode 100644 index 00000000000..9c1b9440ed4 Binary files /dev/null and b/assets/inter-italic-latin-ext.bd8920cc.woff2 differ diff --git a/assets/inter-italic-latin.bd3b6f56.woff2 b/assets/inter-italic-latin.bd3b6f56.woff2 new file mode 100644 index 00000000000..01fcf20724f Binary files /dev/null and b/assets/inter-italic-latin.bd3b6f56.woff2 differ diff --git a/assets/inter-italic-vietnamese.6ce511fb.woff2 b/assets/inter-italic-vietnamese.6ce511fb.woff2 new file mode 100644 index 00000000000..e4f788ee02b Binary files /dev/null and b/assets/inter-italic-vietnamese.6ce511fb.woff2 differ diff --git a/assets/inter-roman-cyrillic-ext.e75737ce.woff2 b/assets/inter-roman-cyrillic-ext.e75737ce.woff2 new file mode 100644 index 00000000000..28593ccb8a4 Binary files /dev/null and b/assets/inter-roman-cyrillic-ext.e75737ce.woff2 differ diff --git a/assets/inter-roman-cyrillic.5f2c6c8c.woff2 b/assets/inter-roman-cyrillic.5f2c6c8c.woff2 new file mode 100644 index 00000000000..a20adc161f4 Binary files /dev/null and b/assets/inter-roman-cyrillic.5f2c6c8c.woff2 differ diff --git a/assets/inter-roman-greek-ext.ab0619bc.woff2 b/assets/inter-roman-greek-ext.ab0619bc.woff2 new file mode 100644 index 00000000000..e3b0be76db2 Binary files /dev/null and b/assets/inter-roman-greek-ext.ab0619bc.woff2 differ diff --git a/assets/inter-roman-greek.d5a6d92a.woff2 b/assets/inter-roman-greek.d5a6d92a.woff2 new file mode 100644 index 00000000000..f790e047daa Binary files /dev/null and b/assets/inter-roman-greek.d5a6d92a.woff2 differ diff --git a/assets/inter-roman-latin-ext.0030eebd.woff2 b/assets/inter-roman-latin-ext.0030eebd.woff2 new file mode 100644 index 00000000000..715bd903b9b Binary files /dev/null and b/assets/inter-roman-latin-ext.0030eebd.woff2 differ diff --git a/assets/inter-roman-latin.2ed14f66.woff2 b/assets/inter-roman-latin.2ed14f66.woff2 new file mode 100644 index 00000000000..a540b7afe7a Binary files /dev/null and b/assets/inter-roman-latin.2ed14f66.woff2 differ diff --git a/assets/inter-roman-vietnamese.14ce25a6.woff2 b/assets/inter-roman-vietnamese.14ce25a6.woff2 new file mode 100644 index 00000000000..5a9f9cb9ca0 Binary files /dev/null and b/assets/inter-roman-vietnamese.14ce25a6.woff2 differ diff --git a/assets/learn_how-celestia-works_data-availability-faq.md.3d484283.js b/assets/learn_how-celestia-works_data-availability-faq.md.3d484283.js new file mode 100644 index 00000000000..56e26039d16 --- /dev/null +++ b/assets/learn_how-celestia-works_data-availability-faq.md.3d484283.js @@ -0,0 +1 @@ +import{_ as a,o as t,c as e,Q as i}from"./chunks/framework.51d6c45b.js";const o="/img/learn/data-availability-faq/Data-availability.png",s="/img/learn/data-availability-faq/Data-storage.png",m=JSON.parse('{"title":"Data availability FAQ","description":"Frequently asked questions related to Data Availability.","frontmatter":{"description":"Frequently asked questions related to Data Availability.","next":{"text":"Overview of TIA","link":"/learn/tia"},"head":[["meta",{"name":"og:title","content":"Data availability FAQ | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/how-celestia-works/data-availability-faq.md","filePath":"learn/how-celestia-works/data-availability-faq.md","lastUpdated":1700234482000}'),r={name:"learn/how-celestia-works/data-availability-faq.md"},l=i('

    Data availability FAQ

    What is data availability?

    Data availability answers the question, has this data been published? Specifically, a node will verify data availability when it receives a new block that is getting added to the chain. The node will attempt to download all the transaction data for the new block to verify availability. If the node can download all the transaction data, then it successfully verified data availability, proving that the block data was actually published to the network.

    Modular VS Monolithic

    As you’ll see, modular blockchains like Celestia employ other primitives that allow nodes to verify data availability more efficiently. Data availability is critical to the security of any blockchain because it ensures that anyone can inspect the ledger of transactions and verify it. Data availability becomes particularly problematic when scaling blockchains. As the blocks get bigger, it becomes impractical for normal users to download all the data, and therefore users can no longer verify the chain.

    What is the data availability problem?

    The problem with data availability occurs when the transaction data for a newly proposed block cannot be downloaded and verified. This type of attack by a block producer is called a data withholding attack, which sees the block producer withhold transaction data of a new block.

    Since transaction data is withheld, nodes cannot update to the latest state. Such an attack can have numerous consequences, from halting a chain to gaining the ability to steal funds. The severity of the consequences will depend on the type of blockchain (L1 or L2) and whether data availability is kept onchain or offchain. The data availability problem commonly arises around L2 scaling solutions like rollups and validiums.

    How do nodes verify data availability in Celestia?

    In most blockchains, nodes that verify data availability do so by downloading all transaction data for a block. If they are able to download all the data, they have verified its availability. In Celestia, light nodes have access to a new mechanism to verify data availability without needing to download all the data for a block. This new primitive for verifying data availability is called data availability sampling.

    What is data availability sampling?

    Data availability sampling is a mechanism for light nodes to verify data availability without having to download all data for a block. Data availability sampling (DAS) works by having light nodes conduct multiple rounds of random sampling for small portions of block data. As a light node completes more rounds of sampling for block data, it increases its confidence that data is available. Once the light node successfully reaches a predetermined confidence level (e.g. 99%) it will consider the block data as available.

    Want a simpler explanation? Check out this thread on how data availability sampling is like flipping a coin.

    What are some of the security assumptions that Celestia makes for data availability sampling?

    Celestia assumes that there is a minimum number of light nodes that are conducting data availability sampling for a given block size. This assumption is necessary so that a full node can reconstruct an entire block from the portions of data light nodes sampled and stored. The amount of light nodes that are needed will depend on the block size - for bigger blocks more light nodes are assumed to be running.

    A second notable assumption that is made by light nodes is that they are connected to at least one honest full node. This ensures that they can receive fraud proofs for incorrectly erasure coded blocks. If a light node is not connected to an honest full node, such as during an eclipse attack, it can’t verify that the block is improperly constructed.

    Why is block reconstruction necessary for security?

    In Celestia, blocks need to be erasure coded so that there is redundant data to aid the data availability sampling process. However, nodes tasked with erasure coding the data could do so incorrectly. Since Celestia uses fraud proofs to verify that erasure coding is incorrect, the full block data is needed to generate a bad encoding fraud proof.

    There could be a situation where validators only provide data to light nodes and not full nodes. If the full nodes don’t have the ability to reconstruct the full block from the portions of data stored by light nodes, they wouldn’t be able to generate a bad encoding fraud proof.

    What is data storage?

    Data storage is concerned with the ability to store and access past transaction data.

    Modular VS Monolithic

    Data storage and retrieval is needed for multiple purposes, such as:

    • Reading the information of a previous transaction
    • Syncing a node
    • Indexing and serving transaction data
    • Retrieving NFT information

    What is the problem around data storage?

    The issue with data storage is whether past transaction data can be stored and successfully retrieved at a later time. The inability to retrieve historical transaction data can cause problems, such as users being unable to access information about their past transactions or nodes that cannot sync from genesis. Luckily, the assumptions around storing and accessing past data are weak. Only a single copy of a blockchain’s history needs to be accessible for users to gain access to historical transaction data. In other words, data storage security is a 1 of N honesty assumption.

    What is the difference between data availability and data storage?

    Data availability is about verifying that transaction data for a new block is public and available. In contrast, data storage involves storing and accessing past transaction data from old blocks.

    Where does blockchain state fit into this?

    Up until now it’s been all about transaction data, but blockchain state is a related topic. The state is different from transaction data. Specifically, the state is like a current snapshot of the network, which includes information about account balances, smart contract balances, and validator set info. Problems that arise from the size of the state are different in nature than those around data availability and retrievability.

    Why doesn’t Celestia incentivize storage of historical data?

    Most blockchains don’t incentivize storage of data because it shouldn’t be the responsibility of a blockchain to guarantee past data will be retrievable forever. In addition, the data storage problem only requires a single party to store and provide the data for users, which is not a strong problem. As such, Celestia’s purpose is to provide a secure and scalable way to verify the availability of data. Once data has been verified as available, the job of storing and retrieving historical data is left up to other entities that require the data. Luckily, there are natural incentives for outside parties to store and serve historical data to users.

    Who may store historical data if there is no reward?

    There are multiple types of actors that may be likely to store historical data. Some of those include:

    • Block explorers that provide access to past transaction data.
    • Indexers that provide API queries for past data.
    • Applications or rollups that require historical data for certain processes.
    • Users that want to guarantee that they will have access to their transaction history.

    What are some things blockchains can do to provide stronger assurances of data retrievability?

    • Reward nodes based on the amount of transaction data they store and requests for data they serve (this is the case with some data storage blockchains, like Filecoin).
    • Publish transaction data onto a data storage blockchain that incentivizes storing and serving requests for historical data.
    ',37),n=[l];function d(h,c,b,u,f,p){return t(),e("div",null,n)}const v=a(r,[["render",d]]);export{m as __pageData,v as default}; diff --git a/assets/learn_how-celestia-works_data-availability-faq.md.3d484283.lean.js b/assets/learn_how-celestia-works_data-availability-faq.md.3d484283.lean.js new file mode 100644 index 00000000000..290717d97c0 --- /dev/null +++ b/assets/learn_how-celestia-works_data-availability-faq.md.3d484283.lean.js @@ -0,0 +1 @@ +import{_ as a,o as t,c as e,Q as i}from"./chunks/framework.51d6c45b.js";const o="/img/learn/data-availability-faq/Data-availability.png",s="/img/learn/data-availability-faq/Data-storage.png",m=JSON.parse('{"title":"Data availability FAQ","description":"Frequently asked questions related to Data Availability.","frontmatter":{"description":"Frequently asked questions related to Data Availability.","next":{"text":"Overview of TIA","link":"/learn/tia"},"head":[["meta",{"name":"og:title","content":"Data availability FAQ | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/how-celestia-works/data-availability-faq.md","filePath":"learn/how-celestia-works/data-availability-faq.md","lastUpdated":1700234482000}'),r={name:"learn/how-celestia-works/data-availability-faq.md"},l=i("",37),n=[l];function d(h,c,b,u,f,p){return t(),e("div",null,n)}const v=a(r,[["render",d]]);export{m as __pageData,v as default}; diff --git a/assets/learn_how-celestia-works_data-availability-layer.md.c6171925.js b/assets/learn_how-celestia-works_data-availability-layer.md.c6171925.js new file mode 100644 index 00000000000..14b5ad77629 --- /dev/null +++ b/assets/learn_how-celestia-works_data-availability-layer.md.c6171925.js @@ -0,0 +1 @@ +import{_ as l,o as a,c as o,k as e,a as t,Q as s}from"./chunks/framework.51d6c45b.js";const i="/img/learn/reed-solomon-encoding.png",n="/img/learn/nmt.png",r="/img/learn/celestia-app.png",g4=JSON.parse(`{"title":"Celestia's data availability layer","description":"Celestia's Data Availability layer and its key features.","frontmatter":{"description":"Celestia's Data Availability layer and its key features.","head":[["meta",{"name":"og:title","content":"Celestia's data availability layer | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/how-celestia-works/data-availability-layer.md","filePath":"learn/how-celestia-works/data-availability-layer.md","lastUpdated":1709673776000}`),T={name:"learn/how-celestia-works/data-availability-layer.md"},d=s('

    Celestia's data availability layer

    Celestia is a data availability (DA) layer that provides a scalable solution to the data availability problem. Due to the permissionless nature of the blockchain networks, a DA layer must provide a mechanism for the execution and settlement layers to check in a trust-minimized way whether transaction data is indeed available.

    Two key features of Celestia's DA layer are data availability sampling (DAS) and Namespaced Merkle trees (NMTs). Both features are novel blockchain scaling solutions: DAS enables light nodes to verify data availability without needing to download an entire block; NMTs enable execution and settlement layers on Celestia to download transactions that are only relevant to them.

    Data availability sampling (DAS)

    In general, light nodes download only block headers that contain commitments (i.e., Merkle roots) of the block data (i.e., the list of transactions).

    ',5),Q={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},h={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"5.123ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 2264.4 705","aria-hidden":"true"},c=s('',1),m=[c],p=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k"),e("mo",null,"×"),e("mi",null,"k")])],-1),g={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},x={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"5.123ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 2264.4 705","aria-hidden":"true"},u=s('',1),_=[u],w=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k"),e("mo",null,"×"),e("mi",null,"k")])],-1),f={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},b={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"7.386ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 3264.4 705","aria-hidden":"true"},k=s('',1),y=[k],v=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"2"),e("mi",null,"k"),e("mo",null,"×"),e("mn",null,"2"),e("mi",null,"k")])],-1),L={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},M={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.31ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 1021 705","aria-hidden":"true"},H=s('',1),V=[H],S=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"4"),e("mi",null,"k")])],-1),C=e("p",null,[e("img",{src:i,alt:"2D Reed-Soloman (RS) Encoding"})],-1),D={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},A={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"7.386ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 3264.4 705","aria-hidden":"true"},j=s('',1),Z=[j],P=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"2"),e("mi",null,"k"),e("mo",null,"×"),e("mn",null,"2"),e("mi",null,"k")])],-1),N=e("p",null,[t("Every light node randomly chooses a set of unique coordinates in the extended matrix and queries full nodes for the data shares and the corresponding Merkle proofs at those coordinates. If light nodes receive a valid response for each sampling query, then there is a "),e("a",{href:"https://github.com/celestiaorg/celestia-node/issues/805#issuecomment-1150081075",target:"_blank",rel:"noreferrer"},"high probability guarantee"),t(" that the whole block's data is available.")],-1),I=e("em",null,"i.e.",-1),B={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},R={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"5.123ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 2264.4 705","aria-hidden":"true"},q=s('',1),E=[q],z=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k"),e("mo",null,"×"),e("mi",null,"k")])],-1),G=e("p",null,[t("For more details on DAS, take a look at the "),e("a",{href:"https://arxiv.org/abs/1809.09044",target:"_blank",rel:"noreferrer"},"original paper"),t(".")],-1),J=e("h3",{id:"scalability",tabindex:"-1"},[t("Scalability "),e("a",{class:"header-anchor",href:"#scalability","aria-label":'Permalink to "Scalability"'},"​")],-1),F=e("p",null,"DAS enables Celestia to scale the DA layer. DAS can be performed by resource-limited light nodes since each light node only samples a small portion of the block data. The more light nodes there are in the network, the more data they can collectively download and store.",-1),O=e("em",null,"i.e.",-1),$={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},W={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.31ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 1021 705","aria-hidden":"true"},K=s('',1),U=[K],X=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"4"),e("mi",null,"k")])],-1),Y={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},e1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.345ex",height:"1.912ex",role:"img",focusable:"false",viewBox:"0 -833.9 1036.6 844.9","aria-hidden":"true"},t1=s('',1),a1=[t1],o1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("msup",null,[e("mi",null,"n"),e("mn",null,"2")])])],-1),s1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},l1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.566ex"},xmlns:"http://www.w3.org/2000/svg",width:"4.844ex",height:"2.262ex",role:"img",focusable:"false",viewBox:"0 -750 2141 1000","aria-hidden":"true"},i1=s('',1),n1=[i1],r1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"O"),e("mo",{stretchy:"false"},"("),e("mi",null,"n"),e("mo",{stretchy:"false"},")")])],-1),T1=e("h3",{id:"fraud-proofs-of-incorrectly-extended-data",tabindex:"-1"},[t("Fraud proofs of incorrectly extended data "),e("a",{class:"header-anchor",href:"#fraud-proofs-of-incorrectly-extended-data","aria-label":'Permalink to "Fraud proofs of incorrectly extended data"'},"​")],-1),d1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},Q1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.31ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 1021 705","aria-hidden":"true"},h1=s('',1),c1=[h1],m1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"4"),e("mi",null,"k")])],-1),p1=e("em",null,"i.e.",-1),g1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},x1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"1.179ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 521 705","aria-hidden":"true"},u1=e("g",{stroke:"currentColor",fill:"currentColor","stroke-width":"0",transform:"scale(1,-1)"},[e("g",{"data-mml-node":"math"},[e("g",{"data-mml-node":"mi"},[e("path",{"data-c":"1D458",d:"M121 647Q121 657 125 670T137 683Q138 683 209 688T282 694Q294 694 294 686Q294 679 244 477Q194 279 194 272Q213 282 223 291Q247 309 292 354T362 415Q402 442 438 442Q468 442 485 423T503 369Q503 344 496 327T477 302T456 291T438 288Q418 288 406 299T394 328Q394 353 410 369T442 390L458 393Q446 405 434 405H430Q398 402 367 380T294 316T228 255Q230 254 243 252T267 246T293 238T320 224T342 206T359 180T365 147Q365 130 360 106T354 66Q354 26 381 26Q429 26 459 145Q461 153 479 153H483Q499 153 499 144Q499 139 496 130Q455 -11 378 -11Q333 -11 305 15T277 90Q277 108 280 121T283 145Q283 167 269 183T234 206T200 217T182 220H180Q168 178 159 139T145 81T136 44T129 20T122 7T111 -2Q98 -11 83 -11Q66 -11 57 -1T48 16Q48 26 85 176T158 471L195 616Q196 629 188 632T149 637H144Q134 637 131 637T124 640T121 647Z",style:{"stroke-width":"3"}})])])],-1),_1=[u1],w1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k")])],-1),f1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},b1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"1.179ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 521 705","aria-hidden":"true"},k1=e("g",{stroke:"currentColor",fill:"currentColor","stroke-width":"0",transform:"scale(1,-1)"},[e("g",{"data-mml-node":"math"},[e("g",{"data-mml-node":"mi"},[e("path",{"data-c":"1D458",d:"M121 647Q121 657 125 670T137 683Q138 683 209 688T282 694Q294 694 294 686Q294 679 244 477Q194 279 194 272Q213 282 223 291Q247 309 292 354T362 415Q402 442 438 442Q468 442 485 423T503 369Q503 344 496 327T477 302T456 291T438 288Q418 288 406 299T394 328Q394 353 410 369T442 390L458 393Q446 405 434 405H430Q398 402 367 380T294 316T228 255Q230 254 243 252T267 246T293 238T320 224T342 206T359 180T365 147Q365 130 360 106T354 66Q354 26 381 26Q429 26 459 145Q461 153 479 153H483Q499 153 499 144Q499 139 496 130Q455 -11 378 -11Q333 -11 305 15T277 90Q277 108 280 121T283 145Q283 167 269 183T234 206T200 217T182 220H180Q168 178 159 139T145 81T136 44T129 20T122 7T111 -2Q98 -11 83 -11Q66 -11 57 -1T48 16Q48 26 85 176T158 471L195 616Q196 629 188 632T149 637H144Q134 637 131 637T124 640T121 647Z",style:{"stroke-width":"3"}})])])],-1),y1=[k1],v1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k")])],-1),L1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},M1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.31ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 1021 705","aria-hidden":"true"},H1=s('',1),V1=[H1],S1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"2"),e("mi",null,"k")])],-1),C1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},D1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.566ex"},xmlns:"http://www.w3.org/2000/svg",width:"4.844ex",height:"2.262ex",role:"img",focusable:"false",viewBox:"0 -750 2141 1000","aria-hidden":"true"},A1=s('',1),j1=[A1],Z1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"O"),e("mo",{stretchy:"false"},"("),e("mi",null,"n"),e("mo",{stretchy:"false"},")")])],-1),P1=e("p",null,"The downside of the standard Reed-Solomon encoding is dealing with malicious block producers that generate the extended data incorrectly.",-1),N1=e("strong",null,[t("Celestia does not require a majority of the consensus ("),e("em",null,"i.e."),t(", block producers) to be honest to guarantee data availability.")],-1),I1=e("em",null,"i.e.",-1),B1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},R1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"1.179ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 521 705","aria-hidden":"true"},q1=e("g",{stroke:"currentColor",fill:"currentColor","stroke-width":"0",transform:"scale(1,-1)"},[e("g",{"data-mml-node":"math"},[e("g",{"data-mml-node":"mi"},[e("path",{"data-c":"1D458",d:"M121 647Q121 657 125 670T137 683Q138 683 209 688T282 694Q294 694 294 686Q294 679 244 477Q194 279 194 272Q213 282 223 291Q247 309 292 354T362 415Q402 442 438 442Q468 442 485 423T503 369Q503 344 496 327T477 302T456 291T438 288Q418 288 406 299T394 328Q394 353 410 369T442 390L458 393Q446 405 434 405H430Q398 402 367 380T294 316T228 255Q230 254 243 252T267 246T293 238T320 224T342 206T359 180T365 147Q365 130 360 106T354 66Q354 26 381 26Q429 26 459 145Q461 153 479 153H483Q499 153 499 144Q499 139 496 130Q455 -11 378 -11Q333 -11 305 15T277 90Q277 108 280 121T283 145Q283 167 269 183T234 206T200 217T182 220H180Q168 178 159 139T145 81T136 44T129 20T122 7T111 -2Q98 -11 83 -11Q66 -11 57 -1T48 16Q48 26 85 176T158 471L195 616Q196 629 188 632T149 637H144Q134 637 131 637T124 640T121 647Z",style:{"stroke-width":"3"}})])])],-1),E1=[q1],z1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k")])],-1),G1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},J1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"5.123ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 2264.4 705","aria-hidden":"true"},F1=s('',1),O1=[F1],$1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k"),e("mo",null,"×"),e("mi",null,"k")])],-1),W1=e("em",null,"Fraud Proofs of Incorrectly Generated Extended Data",-1),K1=e("em",null,"i.e.",-1),U1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},X1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.345ex",height:"1.912ex",role:"img",focusable:"false",viewBox:"0 -833.9 1036.6 844.9","aria-hidden":"true"},Y1=s('',1),e4=[Y1],t4=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("msup",null,[e("mi",null,"n"),e("mn",null,"2")])])],-1),a4={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},o4={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.566ex"},xmlns:"http://www.w3.org/2000/svg",width:"4.844ex",height:"2.262ex",role:"img",focusable:"false",viewBox:"0 -750 2141 1000","aria-hidden":"true"},s4=s('',1),l4=[s4],i4=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"O"),e("mo",{stretchy:"false"},"("),e("mi",null,"n"),e("mo",{stretchy:"false"},")")])],-1),n4=s('

    Namespaced Merkle trees (NMTs)

    Celestia partitions the block data into multiple namespaces, one for every application (e.g., rollup) using the DA layer. As a result, every application needs to download only its own data and can ignore the data of other applications.

    For this to work, the DA layer must be able to prove that the provided data is complete, i.e., all the data for a given namespace is returned. To this end, Celestia is using Namespaced Merkle trees (NMTs).

    An NMT is a Merkle tree with the leafs ordered by the namespace identifiers and the hash function modified so that every node in the tree includes the range of namespaces of all its descendants. The following figure shows an example of an NMT with height three (i.e., eight data shares). The data is partitioned into three namespaces.

    Namespaced Merkle Tree

    When an application requests the data for namespace 2, the DA layer must provide the data shares D3, D4, D5, and D6 and the nodes N2, N8 and N7 as proof (note that the application already has the root N14 from the block header).

    As a result, the application is able to check that the provided data is part of the block data. Furthermore, the application can verify that all the data for namespace 2 was provided. If the DA layer provides for example only the data shares D4 and D5, it must also provide nodes N12 and N11 as proofs. However, the application can identify that the data is incomplete by checking the namespace range of the two nodes, i.e., both N12 and N11 have descendants part of namespace 2.

    For more details on NMTs, refer to the original paper.

    Building a PoS blockchain for DA

    Providing data availability

    The Celestia DA layer consists of a PoS blockchain. Celestia is dubbing this blockchain as the celestia-app, an application that provides transactions to facilitate the DA layer and is built using Cosmos SDK. The following figure shows the main components of celestia-app.

    Main components of celestia-app

    celestia-app is built on top of celestia-core, a modified version of the Tendermint consensus algorithm. Among the more important changes to vanilla Tendermint, celestia-core:

    • Enables the erasure coding of block data (using the 2-dimensional Reed-Solomon encoding scheme).
    • Replaces the regular Merkle tree used by Tendermint to store block data with a Namespaced Merkle tree that enables the above layers (i.e., execution and settlement) to only download the needed data (for more details, see the section below describing use cases).

    For more details on the changes to Tendermint, take a look at the ADRs. Notice that celestia-core nodes are still using the Tendermint p2p network.

    Similarly to Tendermint, celestia-core is connected to the application layer (i.e., the state machine) by ABCI++, a major evolution of ABCI (Application Blockchain Interface).

    The celestia-app state machine is necessary to execute the PoS logic and to enable the governance of the DA layer.

    However, the celestia-app is data-agnostic -- the state machine neither validates nor stores the data that is made available by the celestia-app.

    ',18);function r4(T4,d4,Q4,h4,c4,m4){return a(),o("div",null,[d,e("p",null,[t("To make DAS possible, Celestia uses a 2-dimensional Reed-Solomon encoding scheme to encode the block data: every block data is split into "),e("mjx-container",Q,[(a(),o("svg",h,m)),p]),t(" shares, arranged in a "),e("mjx-container",g,[(a(),o("svg",x,_)),w]),t(" matrix, and extended with parity data into a "),e("mjx-container",f,[(a(),o("svg",b,y)),v]),t(" extended matrix by applying multiple times Reed-Solomon encoding.")]),e("p",null,[t("Then, "),e("mjx-container",L,[(a(),o("svg",M,V)),S]),t(" separate Merkle roots are computed for the rows and columns of the extended matrix; the Merkle root of these Merkle roots is used as the block data commitment in the block header.")]),C,e("p",null,[t("To verify that the data is available, Celestia light nodes are sampling the "),e("mjx-container",D,[(a(),o("svg",A,Z)),P]),t(" data shares.")]),N,e("p",null,[t("Additionally, every received data share with a correct Merkle proof is gossiped to the network. As a result, as long as the Celestia light nodes are sampling together enough data shares ("),I,t(", at least "),e("mjx-container",B,[(a(),o("svg",R,E)),z]),t(" unique shares), the full block can be recovered by honest full nodes.")]),G,J,F,e("p",null,[t("This means that increasing the number of light nodes performing DAS allows for larger blocks ("),O,t(", with more transactions), while still keeping DAS feasible for resource-limited light nodes. However, in order to validate block headers, Celestia light nodes need to download the "),e("mjx-container",$,[(a(),o("svg",W,U)),X]),t(" intermediate Merkle roots.")]),e("p",null,[t("For a block data size of "),e("mjx-container",Y,[(a(),o("svg",e1,a1)),o1]),t(" bytes, this means that every light node must download "),e("mjx-container",s1,[(a(),o("svg",l1,n1)),r1]),t(" bytes. Therefore, any improvement in the bandwidth capacity of Celestia light nodes has a quadratic effect on the throughput of Celestia's DA layer.")]),T1,e("p",null,[t("The requirement of downloading the "),e("mjx-container",d1,[(a(),o("svg",Q1,c1)),m1]),t(" intermediate Merkle roots is a consequence of using a 2-dimensional Reed-Solomon encoding scheme. Alternatively, DAS could be designed with a standard ("),p1,t(", 1-dimensional) Reed-Solomon encoding, where the original data is split into "),e("mjx-container",g1,[(a(),o("svg",x1,_1)),w1]),t(" shares and extended with "),e("mjx-container",f1,[(a(),o("svg",b1,y1)),v1]),t(" additional shares of parity data. Since the block data commitment is the Merkle root of the "),e("mjx-container",L1,[(a(),o("svg",M1,V1)),S1]),t(" resulting data shares, light nodes no longer need to download "),e("mjx-container",C1,[(a(),o("svg",D1,j1)),Z1]),t(" bytes to validate block headers.")]),P1,e("p",null,[t("This is possible as "),N1,t(" Thus, if the extended data is invalid, the original data might not be recoverable, even if the light nodes are sampling sufficient unique shares ("),I1,t(", at least "),e("mjx-container",B1,[(a(),o("svg",R1,E1)),z1]),t(" for a standard encoding and "),e("mjx-container",G1,[(a(),o("svg",J1,O1)),$1]),t(" for a 2-dimensional encoding).")]),e("p",null,[t("As a solution, "),W1,t(" enable light nodes to reject blocks with invalid extended data. Such proofs require reconstructing the encoding and verifying the mismatch. With standard Reed-Solomon encoding, this entails downloading the original data, "),K1,t(", "),e("mjx-container",U1,[(a(),o("svg",X1,e4)),t4]),t(" bytes. Contrastingly, with 2-dimensional Reed-Solomon encoding, only "),e("mjx-container",a4,[(a(),o("svg",o4,l4)),i4]),t(" bytes are required as it is sufficient to verify only one row or one column of the extended matrix.")]),n4])}const x4=l(T,[["render",r4]]);export{g4 as __pageData,x4 as default}; diff --git a/assets/learn_how-celestia-works_data-availability-layer.md.c6171925.lean.js b/assets/learn_how-celestia-works_data-availability-layer.md.c6171925.lean.js new file mode 100644 index 00000000000..ffbae39612a --- /dev/null +++ b/assets/learn_how-celestia-works_data-availability-layer.md.c6171925.lean.js @@ -0,0 +1 @@ +import{_ as l,o as a,c as o,k as e,a as t,Q as s}from"./chunks/framework.51d6c45b.js";const i="/img/learn/reed-solomon-encoding.png",n="/img/learn/nmt.png",r="/img/learn/celestia-app.png",g4=JSON.parse(`{"title":"Celestia's data availability layer","description":"Celestia's Data Availability layer and its key features.","frontmatter":{"description":"Celestia's Data Availability layer and its key features.","head":[["meta",{"name":"og:title","content":"Celestia's data availability layer | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/how-celestia-works/data-availability-layer.md","filePath":"learn/how-celestia-works/data-availability-layer.md","lastUpdated":1709673776000}`),T={name:"learn/how-celestia-works/data-availability-layer.md"},d=s("",5),Q={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},h={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"5.123ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 2264.4 705","aria-hidden":"true"},c=s("",1),m=[c],p=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k"),e("mo",null,"×"),e("mi",null,"k")])],-1),g={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},x={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"5.123ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 2264.4 705","aria-hidden":"true"},u=s("",1),_=[u],w=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k"),e("mo",null,"×"),e("mi",null,"k")])],-1),f={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},b={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"7.386ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 3264.4 705","aria-hidden":"true"},k=s("",1),y=[k],v=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"2"),e("mi",null,"k"),e("mo",null,"×"),e("mn",null,"2"),e("mi",null,"k")])],-1),L={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},M={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.31ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 1021 705","aria-hidden":"true"},H=s("",1),V=[H],S=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"4"),e("mi",null,"k")])],-1),C=e("p",null,[e("img",{src:i,alt:"2D Reed-Soloman (RS) Encoding"})],-1),D={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},A={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"7.386ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 3264.4 705","aria-hidden":"true"},j=s("",1),Z=[j],P=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"2"),e("mi",null,"k"),e("mo",null,"×"),e("mn",null,"2"),e("mi",null,"k")])],-1),N=e("p",null,[t("Every light node randomly chooses a set of unique coordinates in the extended matrix and queries full nodes for the data shares and the corresponding Merkle proofs at those coordinates. If light nodes receive a valid response for each sampling query, then there is a "),e("a",{href:"https://github.com/celestiaorg/celestia-node/issues/805#issuecomment-1150081075",target:"_blank",rel:"noreferrer"},"high probability guarantee"),t(" that the whole block's data is available.")],-1),I=e("em",null,"i.e.",-1),B={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},R={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"5.123ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 2264.4 705","aria-hidden":"true"},q=s("",1),E=[q],z=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k"),e("mo",null,"×"),e("mi",null,"k")])],-1),G=e("p",null,[t("For more details on DAS, take a look at the "),e("a",{href:"https://arxiv.org/abs/1809.09044",target:"_blank",rel:"noreferrer"},"original paper"),t(".")],-1),J=e("h3",{id:"scalability",tabindex:"-1"},[t("Scalability "),e("a",{class:"header-anchor",href:"#scalability","aria-label":'Permalink to "Scalability"'},"​")],-1),F=e("p",null,"DAS enables Celestia to scale the DA layer. DAS can be performed by resource-limited light nodes since each light node only samples a small portion of the block data. The more light nodes there are in the network, the more data they can collectively download and store.",-1),O=e("em",null,"i.e.",-1),$={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},W={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.31ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 1021 705","aria-hidden":"true"},K=s("",1),U=[K],X=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"4"),e("mi",null,"k")])],-1),Y={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},e1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.345ex",height:"1.912ex",role:"img",focusable:"false",viewBox:"0 -833.9 1036.6 844.9","aria-hidden":"true"},t1=s("",1),a1=[t1],o1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("msup",null,[e("mi",null,"n"),e("mn",null,"2")])])],-1),s1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},l1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.566ex"},xmlns:"http://www.w3.org/2000/svg",width:"4.844ex",height:"2.262ex",role:"img",focusable:"false",viewBox:"0 -750 2141 1000","aria-hidden":"true"},i1=s("",1),n1=[i1],r1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"O"),e("mo",{stretchy:"false"},"("),e("mi",null,"n"),e("mo",{stretchy:"false"},")")])],-1),T1=e("h3",{id:"fraud-proofs-of-incorrectly-extended-data",tabindex:"-1"},[t("Fraud proofs of incorrectly extended data "),e("a",{class:"header-anchor",href:"#fraud-proofs-of-incorrectly-extended-data","aria-label":'Permalink to "Fraud proofs of incorrectly extended data"'},"​")],-1),d1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},Q1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.31ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 1021 705","aria-hidden":"true"},h1=s("",1),c1=[h1],m1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"4"),e("mi",null,"k")])],-1),p1=e("em",null,"i.e.",-1),g1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},x1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"1.179ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 521 705","aria-hidden":"true"},u1=e("g",{stroke:"currentColor",fill:"currentColor","stroke-width":"0",transform:"scale(1,-1)"},[e("g",{"data-mml-node":"math"},[e("g",{"data-mml-node":"mi"},[e("path",{"data-c":"1D458",d:"M121 647Q121 657 125 670T137 683Q138 683 209 688T282 694Q294 694 294 686Q294 679 244 477Q194 279 194 272Q213 282 223 291Q247 309 292 354T362 415Q402 442 438 442Q468 442 485 423T503 369Q503 344 496 327T477 302T456 291T438 288Q418 288 406 299T394 328Q394 353 410 369T442 390L458 393Q446 405 434 405H430Q398 402 367 380T294 316T228 255Q230 254 243 252T267 246T293 238T320 224T342 206T359 180T365 147Q365 130 360 106T354 66Q354 26 381 26Q429 26 459 145Q461 153 479 153H483Q499 153 499 144Q499 139 496 130Q455 -11 378 -11Q333 -11 305 15T277 90Q277 108 280 121T283 145Q283 167 269 183T234 206T200 217T182 220H180Q168 178 159 139T145 81T136 44T129 20T122 7T111 -2Q98 -11 83 -11Q66 -11 57 -1T48 16Q48 26 85 176T158 471L195 616Q196 629 188 632T149 637H144Q134 637 131 637T124 640T121 647Z",style:{"stroke-width":"3"}})])])],-1),_1=[u1],w1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k")])],-1),f1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},b1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"1.179ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 521 705","aria-hidden":"true"},k1=e("g",{stroke:"currentColor",fill:"currentColor","stroke-width":"0",transform:"scale(1,-1)"},[e("g",{"data-mml-node":"math"},[e("g",{"data-mml-node":"mi"},[e("path",{"data-c":"1D458",d:"M121 647Q121 657 125 670T137 683Q138 683 209 688T282 694Q294 694 294 686Q294 679 244 477Q194 279 194 272Q213 282 223 291Q247 309 292 354T362 415Q402 442 438 442Q468 442 485 423T503 369Q503 344 496 327T477 302T456 291T438 288Q418 288 406 299T394 328Q394 353 410 369T442 390L458 393Q446 405 434 405H430Q398 402 367 380T294 316T228 255Q230 254 243 252T267 246T293 238T320 224T342 206T359 180T365 147Q365 130 360 106T354 66Q354 26 381 26Q429 26 459 145Q461 153 479 153H483Q499 153 499 144Q499 139 496 130Q455 -11 378 -11Q333 -11 305 15T277 90Q277 108 280 121T283 145Q283 167 269 183T234 206T200 217T182 220H180Q168 178 159 139T145 81T136 44T129 20T122 7T111 -2Q98 -11 83 -11Q66 -11 57 -1T48 16Q48 26 85 176T158 471L195 616Q196 629 188 632T149 637H144Q134 637 131 637T124 640T121 647Z",style:{"stroke-width":"3"}})])])],-1),y1=[k1],v1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k")])],-1),L1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},M1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.31ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 1021 705","aria-hidden":"true"},H1=s("",1),V1=[H1],S1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"2"),e("mi",null,"k")])],-1),C1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},D1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.566ex"},xmlns:"http://www.w3.org/2000/svg",width:"4.844ex",height:"2.262ex",role:"img",focusable:"false",viewBox:"0 -750 2141 1000","aria-hidden":"true"},A1=s("",1),j1=[A1],Z1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"O"),e("mo",{stretchy:"false"},"("),e("mi",null,"n"),e("mo",{stretchy:"false"},")")])],-1),P1=e("p",null,"The downside of the standard Reed-Solomon encoding is dealing with malicious block producers that generate the extended data incorrectly.",-1),N1=e("strong",null,[t("Celestia does not require a majority of the consensus ("),e("em",null,"i.e."),t(", block producers) to be honest to guarantee data availability.")],-1),I1=e("em",null,"i.e.",-1),B1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},R1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"1.179ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 521 705","aria-hidden":"true"},q1=e("g",{stroke:"currentColor",fill:"currentColor","stroke-width":"0",transform:"scale(1,-1)"},[e("g",{"data-mml-node":"math"},[e("g",{"data-mml-node":"mi"},[e("path",{"data-c":"1D458",d:"M121 647Q121 657 125 670T137 683Q138 683 209 688T282 694Q294 694 294 686Q294 679 244 477Q194 279 194 272Q213 282 223 291Q247 309 292 354T362 415Q402 442 438 442Q468 442 485 423T503 369Q503 344 496 327T477 302T456 291T438 288Q418 288 406 299T394 328Q394 353 410 369T442 390L458 393Q446 405 434 405H430Q398 402 367 380T294 316T228 255Q230 254 243 252T267 246T293 238T320 224T342 206T359 180T365 147Q365 130 360 106T354 66Q354 26 381 26Q429 26 459 145Q461 153 479 153H483Q499 153 499 144Q499 139 496 130Q455 -11 378 -11Q333 -11 305 15T277 90Q277 108 280 121T283 145Q283 167 269 183T234 206T200 217T182 220H180Q168 178 159 139T145 81T136 44T129 20T122 7T111 -2Q98 -11 83 -11Q66 -11 57 -1T48 16Q48 26 85 176T158 471L195 616Q196 629 188 632T149 637H144Q134 637 131 637T124 640T121 647Z",style:{"stroke-width":"3"}})])])],-1),E1=[q1],z1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k")])],-1),G1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},J1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"5.123ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 2264.4 705","aria-hidden":"true"},F1=s("",1),O1=[F1],$1=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k"),e("mo",null,"×"),e("mi",null,"k")])],-1),W1=e("em",null,"Fraud Proofs of Incorrectly Generated Extended Data",-1),K1=e("em",null,"i.e.",-1),U1={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},X1={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.345ex",height:"1.912ex",role:"img",focusable:"false",viewBox:"0 -833.9 1036.6 844.9","aria-hidden":"true"},Y1=s("",1),e4=[Y1],t4=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("msup",null,[e("mi",null,"n"),e("mn",null,"2")])])],-1),a4={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},o4={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.566ex"},xmlns:"http://www.w3.org/2000/svg",width:"4.844ex",height:"2.262ex",role:"img",focusable:"false",viewBox:"0 -750 2141 1000","aria-hidden":"true"},s4=s("",1),l4=[s4],i4=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"O"),e("mo",{stretchy:"false"},"("),e("mi",null,"n"),e("mo",{stretchy:"false"},")")])],-1),n4=s("",18);function r4(T4,d4,Q4,h4,c4,m4){return a(),o("div",null,[d,e("p",null,[t("To make DAS possible, Celestia uses a 2-dimensional Reed-Solomon encoding scheme to encode the block data: every block data is split into "),e("mjx-container",Q,[(a(),o("svg",h,m)),p]),t(" shares, arranged in a "),e("mjx-container",g,[(a(),o("svg",x,_)),w]),t(" matrix, and extended with parity data into a "),e("mjx-container",f,[(a(),o("svg",b,y)),v]),t(" extended matrix by applying multiple times Reed-Solomon encoding.")]),e("p",null,[t("Then, "),e("mjx-container",L,[(a(),o("svg",M,V)),S]),t(" separate Merkle roots are computed for the rows and columns of the extended matrix; the Merkle root of these Merkle roots is used as the block data commitment in the block header.")]),C,e("p",null,[t("To verify that the data is available, Celestia light nodes are sampling the "),e("mjx-container",D,[(a(),o("svg",A,Z)),P]),t(" data shares.")]),N,e("p",null,[t("Additionally, every received data share with a correct Merkle proof is gossiped to the network. As a result, as long as the Celestia light nodes are sampling together enough data shares ("),I,t(", at least "),e("mjx-container",B,[(a(),o("svg",R,E)),z]),t(" unique shares), the full block can be recovered by honest full nodes.")]),G,J,F,e("p",null,[t("This means that increasing the number of light nodes performing DAS allows for larger blocks ("),O,t(", with more transactions), while still keeping DAS feasible for resource-limited light nodes. However, in order to validate block headers, Celestia light nodes need to download the "),e("mjx-container",$,[(a(),o("svg",W,U)),X]),t(" intermediate Merkle roots.")]),e("p",null,[t("For a block data size of "),e("mjx-container",Y,[(a(),o("svg",e1,a1)),o1]),t(" bytes, this means that every light node must download "),e("mjx-container",s1,[(a(),o("svg",l1,n1)),r1]),t(" bytes. Therefore, any improvement in the bandwidth capacity of Celestia light nodes has a quadratic effect on the throughput of Celestia's DA layer.")]),T1,e("p",null,[t("The requirement of downloading the "),e("mjx-container",d1,[(a(),o("svg",Q1,c1)),m1]),t(" intermediate Merkle roots is a consequence of using a 2-dimensional Reed-Solomon encoding scheme. Alternatively, DAS could be designed with a standard ("),p1,t(", 1-dimensional) Reed-Solomon encoding, where the original data is split into "),e("mjx-container",g1,[(a(),o("svg",x1,_1)),w1]),t(" shares and extended with "),e("mjx-container",f1,[(a(),o("svg",b1,y1)),v1]),t(" additional shares of parity data. Since the block data commitment is the Merkle root of the "),e("mjx-container",L1,[(a(),o("svg",M1,V1)),S1]),t(" resulting data shares, light nodes no longer need to download "),e("mjx-container",C1,[(a(),o("svg",D1,j1)),Z1]),t(" bytes to validate block headers.")]),P1,e("p",null,[t("This is possible as "),N1,t(" Thus, if the extended data is invalid, the original data might not be recoverable, even if the light nodes are sampling sufficient unique shares ("),I1,t(", at least "),e("mjx-container",B1,[(a(),o("svg",R1,E1)),z1]),t(" for a standard encoding and "),e("mjx-container",G1,[(a(),o("svg",J1,O1)),$1]),t(" for a 2-dimensional encoding).")]),e("p",null,[t("As a solution, "),W1,t(" enable light nodes to reject blocks with invalid extended data. Such proofs require reconstructing the encoding and verifying the mismatch. With standard Reed-Solomon encoding, this entails downloading the original data, "),K1,t(", "),e("mjx-container",U1,[(a(),o("svg",X1,e4)),t4]),t(" bytes. Contrastingly, with 2-dimensional Reed-Solomon encoding, only "),e("mjx-container",a4,[(a(),o("svg",o4,l4)),i4]),t(" bytes are required as it is sufficient to verify only one row or one column of the extended matrix.")]),n4])}const x4=l(T,[["render",r4]]);export{g4 as __pageData,x4 as default}; diff --git a/assets/learn_how-celestia-works_monolithic-vs-modular.md.db0fb9d5.js b/assets/learn_how-celestia-works_monolithic-vs-modular.md.db0fb9d5.js new file mode 100644 index 00000000000..f5a1e86b0ec --- /dev/null +++ b/assets/learn_how-celestia-works_monolithic-vs-modular.md.db0fb9d5.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.51d6c45b.js";const n="/img/learn/monolithic-modular.png",g=JSON.parse('{"title":"Monolithic vs. modular blockchains","description":"Comparison between monolithic and modular blockchains.","frontmatter":{"description":"Comparison between monolithic and modular blockchains.","head":[["meta",{"name":"og:title","content":"Monolithic vs. modular blockchains | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/how-celestia-works/monolithic-vs-modular.md","filePath":"learn/how-celestia-works/monolithic-vs-modular.md","lastUpdated":1711980671000}'),i={name:"learn/how-celestia-works/monolithic-vs-modular.md"},s=a('

    Monolithic vs. modular blockchains

    Blockchains instantiate replicated state machines: the nodes in a permissionless distributed network apply an ordered sequence of deterministic transactions to an initial state resulting in a common final state.

    In other words, this means that nodes in a network all follow the same set of rules (i.e., an ordered sequence of transactions) to go from a starting point (i.e., an initial state) to an ending point (i.e., a common final state). This process ensures that all nodes in the network agree on the final state of the blockchain, even though they operate independently.

    This means blockchains require the following four functions:

    • Execution entails executing transactions that update the state correctly. Thus, execution must ensure that only valid transactions are executed, i.e., transactions that result in valid state machine transitions.
    • Settlement entails an environment for execution layers to verify proofs, resolve fraud disputes, and bridge between other execution layers.
    • Consensus entails agreeing on the order of the transactions.
    • Data Availability (DA) entails making the transaction data available. Note that execution, settlement, and consensus require DA.

    Traditional blockchains, i.e. monolithic blockchains, implement all four functions together in a single base consensus layer. The problem with monolithic blockchains is that the consensus layer must perform numerous different tasks, and it cannot be optimized for only one of these functions. As a result, the monolithic paradigm limits the throughput of the system.

    Modular VS Monolithic

    As a solution, modular blockchains decouple these functions among multiple specialized layers as part of a modular stack. Due to the flexibility that specialization provides, there are many possibilities in which that stack can be arranged. For example, one such arrangement is the separation of the four functions into three specialized layers.

    The base layer consists of DA and consensus and thus, is referred to as the Consensus and DA layer (or for brevity, the DA layer), while both settlement and execution are moved on top in their own layers. As a result, every layer can be specialized to optimally perform only its function, and thus, increase the throughput of the system. Furthermore, this modular paradigm enables multiple execution layers, i.e., rollups, to use the same settlement and DA layers.

    ',9),r=[s];function l(c,h,m,d,u,p){return t(),o("div",null,r)}const b=e(i,[["render",l]]);export{g as __pageData,b as default}; diff --git a/assets/learn_how-celestia-works_monolithic-vs-modular.md.db0fb9d5.lean.js b/assets/learn_how-celestia-works_monolithic-vs-modular.md.db0fb9d5.lean.js new file mode 100644 index 00000000000..1f972a59c6c --- /dev/null +++ b/assets/learn_how-celestia-works_monolithic-vs-modular.md.db0fb9d5.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.51d6c45b.js";const n="/img/learn/monolithic-modular.png",g=JSON.parse('{"title":"Monolithic vs. modular blockchains","description":"Comparison between monolithic and modular blockchains.","frontmatter":{"description":"Comparison between monolithic and modular blockchains.","head":[["meta",{"name":"og:title","content":"Monolithic vs. modular blockchains | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/how-celestia-works/monolithic-vs-modular.md","filePath":"learn/how-celestia-works/monolithic-vs-modular.md","lastUpdated":1711980671000}'),i={name:"learn/how-celestia-works/monolithic-vs-modular.md"},s=a("",9),r=[s];function l(c,h,m,d,u,p){return t(),o("div",null,r)}const b=e(i,[["render",l]]);export{g as __pageData,b as default}; diff --git a/assets/learn_how-celestia-works_overview.md.3d86106d.js b/assets/learn_how-celestia-works_overview.md.3d86106d.js new file mode 100644 index 00000000000..fff4371904b --- /dev/null +++ b/assets/learn_how-celestia-works_overview.md.3d86106d.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as r}from"./chunks/framework.51d6c45b.js";const f=JSON.parse('{"title":"Introduction","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Introduction | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/how-celestia-works/overview.md","filePath":"learn/how-celestia-works/overview.md","lastUpdated":1709673776000}'),i={name:"learn/how-celestia-works/overview.md"},o=r('

    Introduction

    Celestia is a modular data availability network that securely scales with the number of users, making it easy for anyone to launch their own blockchain.

    Celestia enables the next generation of scalable blockchain architectures - modular blockchains. Celestia scales by decoupling execution from consensus and introducing a new primitive, data availability sampling.

    The former entails that Celestia is only responsible for ordering transactions and guaranteeing their data availability; this is similar to reducing consensus to atomic broadcast.

    The latter provides an efficient solution to the data availability problem by only requiring resource-limited light nodes to sample a small number of random shares from each block to verify data availability.

    Interestingly, more light nodes that participate in sampling increases the amount of data that the network can safely handle, enabling the block size to increase without equally increasing the cost to verify the chain.

    ',6),n=[o];function s(l,c,h,d,p,m){return a(),t("div",null,n)}const _=e(i,[["render",s]]);export{f as __pageData,_ as default}; diff --git a/assets/learn_how-celestia-works_overview.md.3d86106d.lean.js b/assets/learn_how-celestia-works_overview.md.3d86106d.lean.js new file mode 100644 index 00000000000..d94536a6e4f --- /dev/null +++ b/assets/learn_how-celestia-works_overview.md.3d86106d.lean.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as r}from"./chunks/framework.51d6c45b.js";const f=JSON.parse('{"title":"Introduction","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Introduction | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/how-celestia-works/overview.md","filePath":"learn/how-celestia-works/overview.md","lastUpdated":1709673776000}'),i={name:"learn/how-celestia-works/overview.md"},o=r("",6),n=[o];function s(l,c,h,d,p,m){return a(),t("div",null,n)}const _=e(i,[["render",s]]);export{f as __pageData,_ as default}; diff --git a/assets/learn_how-celestia-works_transaction-lifecycle.md.c6f24695.js b/assets/learn_how-celestia-works_transaction-lifecycle.md.c6f24695.js new file mode 100644 index 00000000000..cbd24ddf394 --- /dev/null +++ b/assets/learn_how-celestia-works_transaction-lifecycle.md.c6f24695.js @@ -0,0 +1 @@ +import{_ as i}from"./chunks/tx-lifecycle.92aed1e6.js";import{_ as n,o as a,c as o,k as e,a as t,Q as s}from"./chunks/framework.51d6c45b.js";const l="/img/learn/consensus-da.png",K=JSON.parse('{"title":"The lifecycle of a celestia-app transaction","description":"Learn what is the lifecycle of a celestia-app transaction.","frontmatter":{"description":"Learn what is the lifecycle of a celestia-app transaction.","head":[["meta",{"name":"og:title","content":"The lifecycle of a celestia-app transaction | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/how-celestia-works/transaction-lifecycle.md","filePath":"learn/how-celestia-works/transaction-lifecycle.md","lastUpdated":1709673776000}'),r={name:"learn/how-celestia-works/transaction-lifecycle.md"},c=s('

    The lifecycle of a celestia-app transaction

    Users request the celestia-app to make data available by sending PayForBlobs transactions. Every such transaction consists of the identity of the sender, the data to be made available, also referred to as the message, the data size, the namespace, and a signature. Every block producer batches multiple PayForBlobs transactions into a block.

    Before proposing the block though, the producer passes it to the state machine via ABCI++, where each PayForBlobs transaction is split into a namespaced message (denoted by Msg in the figure below), i.e., the data together with the namespace ID, and an executable transaction (denoted by e-Tx in the figure below) that does not contain the data, but only a commitment that can be used at a later time to prove that the data was indeed made available.

    Thus, the block data consists of data partitioned into namespaces and executable transactions. Note that only these transactions are executed by the Celestia state machine once the block is committed.

    Lifecycle of a celestia-app Transaction

    ',5),h=e("a",{href:"./data-availability-layer"},`described in the "Celestia's data availability layer" page`,-1),d={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},T={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.31ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 1021 705","aria-hidden":"true"},Q=s('',1),p=[Q],m=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"4"),e("mi",null,"k")])],-1),u=e("li",null,"It splits the executable transactions and the namespaced data into shares. Every share consists of some bytes prefixed by a namespace. To this end, the executable transactions are associated with a reserved namespace.",-1),_={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},g={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"5.123ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 2264.4 705","aria-hidden":"true"},f=s('',1),x=[f],w=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k"),e("mo",null,"×"),e("mi",null,"k")])],-1),b={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},k={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"7.386ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 3264.4 705","aria-hidden":"true"},y=s('',1),v=[y],L=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"2"),e("mi",null,"k"),e("mo",null,"×"),e("mn",null,"2"),e("mi",null,"k")])],-1),H=e("li",null,"It computes a commitment for every row and column of the extended matrix using the NMTs described above.",-1),V=e("p",null,"Thus, the commitment of the block data is the root of a Merkle tree with the leaves the roots of a forest of Namespaced Merkle subtrees, one for every row and column of the extended matrix.",-1),M=e("h2",{id:"checking-data-availability",tabindex:"-1"},[t("Checking data availability "),e("a",{class:"header-anchor",href:"#checking-data-availability","aria-label":'Permalink to "Checking data availability"'},"​")],-1),A=e("p",null,[e("img",{src:l,alt:"DA network"})],-1),C=e("p",null,[t("To enhance connectivity, the celestia-node augments the celestia-app with a separate libp2p network, "),e("em",null,"i.e."),t(", the so-called "),e("em",null,"DA network"),t(", that serves DAS requests.")],-1),D={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},S={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.31ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 1021 705","aria-hidden":"true"},I=s('',1),P=[I],N=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"4"),e("mi",null,"k")])],-1),j=e("p",null,[t("Note that although it is recommended, performing DAS is optional -- light nodes could just trust that the data corresponding to the commitments in the block headers was indeed made available by the Celestia DA layer. In addition, light nodes can also submit transactions to the celestia-app, i.e., "),e("code",null,"PayForBlobs"),t(" transactions.")],-1),B=e("p",null,"While performing DAS for a block header, every light node queries Celestia Nodes for a number of random data shares from the extended matrix and the corresponding Merkle proofs. If all the queries are successful, then the light node accepts the block header as valid (from a DA perspective).",-1),Z=e("p",null,"If at least one of the queries fails (i.e., either the data share is not received or the Merkle proof is invalid), then the light node rejects the block header and tries again later. The retrial is necessary to deal with false negatives, i.e., block headers being rejected although the block data is available. This may happen due to network congestion for example.",-1),q=e("p",null,[t("Alternatively, light nodes may accept a block header although the data is not available, i.e., a "),e("em",null,"false positive"),t(". This is possible since the soundness property (i.e., if an honest light node accepts a block as available, then at least one honest full node will eventually have the entire block data) is probabilistically guaranteed (for more details, take a look at the "),e("a",{href:"https://arxiv.org/abs/1809.09044",target:"_blank",rel:"noreferrer"},"original paper"),t(").")],-1),E=e("p",null,"By fine tuning Celestia's parameters (e.g., the number of data shares sampled by each light node) the likelihood of false positives can be sufficiently reduced such that block producers have no incentive to withhold the block data.",-1);function z(R,J,F,G,$,U){return a(),o("div",null,[c,e("p",null,[t("Next, the block producer adds to the block header a commitment of the block data. As "),h,t(", the commitment is the Merkle root of the "),e("mjx-container",d,[(a(),o("svg",T,p)),m]),t(" intermediate Merkle roots (i.e., one for each row and column of the extended matrix). To compute this commitment, the block producer performs the following operations:")]),e("ul",null,[u,e("li",null,[t("It arranges these shares into a square matrix (row-wise). Note that the shares are padded to the next power of two. The outcome square of size "),e("mjx-container",_,[(a(),o("svg",g,x)),w]),t(" is referred to as the original data.")]),e("li",null,[t("It extends the original data to a "),e("mjx-container",b,[(a(),o("svg",k,v)),L]),t(" square matrix using the 2-dimensional Reed-Solomon encoding scheme described above. The extended shares (i.e., containing erasure data) are associated with another reserved namespace.")]),H]),V,M,A,C,e("p",null,[t("Light nodes connect to a celestia-node in the DA network, listen to extended block headers (i.e., the block headers together with the relevant DA metadata, such as the "),e("mjx-container",D,[(a(),o("svg",S,P)),N]),t(" intermediate Merkle roots), and perform DAS on the received headers (i.e., ask for random data shares).")]),j,B,Z,q,E])}const X=n(r,[["render",z]]);export{K as __pageData,X as default}; diff --git a/assets/learn_how-celestia-works_transaction-lifecycle.md.c6f24695.lean.js b/assets/learn_how-celestia-works_transaction-lifecycle.md.c6f24695.lean.js new file mode 100644 index 00000000000..cbfc53c3995 --- /dev/null +++ b/assets/learn_how-celestia-works_transaction-lifecycle.md.c6f24695.lean.js @@ -0,0 +1 @@ +import{_ as i}from"./chunks/tx-lifecycle.92aed1e6.js";import{_ as n,o as a,c as o,k as e,a as t,Q as s}from"./chunks/framework.51d6c45b.js";const l="/img/learn/consensus-da.png",K=JSON.parse('{"title":"The lifecycle of a celestia-app transaction","description":"Learn what is the lifecycle of a celestia-app transaction.","frontmatter":{"description":"Learn what is the lifecycle of a celestia-app transaction.","head":[["meta",{"name":"og:title","content":"The lifecycle of a celestia-app transaction | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/how-celestia-works/transaction-lifecycle.md","filePath":"learn/how-celestia-works/transaction-lifecycle.md","lastUpdated":1709673776000}'),r={name:"learn/how-celestia-works/transaction-lifecycle.md"},c=s("",5),h=e("a",{href:"./data-availability-layer"},`described in the "Celestia's data availability layer" page`,-1),d={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},T={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.31ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 1021 705","aria-hidden":"true"},Q=s("",1),p=[Q],m=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"4"),e("mi",null,"k")])],-1),u=e("li",null,"It splits the executable transactions and the namespaced data into shares. Every share consists of some bytes prefixed by a namespace. To this end, the executable transactions are associated with a reserved namespace.",-1),_={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},g={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"5.123ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 2264.4 705","aria-hidden":"true"},f=s("",1),x=[f],w=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mi",null,"k"),e("mo",null,"×"),e("mi",null,"k")])],-1),b={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},k={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"7.386ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 3264.4 705","aria-hidden":"true"},y=s("",1),v=[y],L=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"2"),e("mi",null,"k"),e("mo",null,"×"),e("mn",null,"2"),e("mi",null,"k")])],-1),H=e("li",null,"It computes a commitment for every row and column of the extended matrix using the NMTs described above.",-1),V=e("p",null,"Thus, the commitment of the block data is the root of a Merkle tree with the leaves the roots of a forest of Namespaced Merkle subtrees, one for every row and column of the extended matrix.",-1),M=e("h2",{id:"checking-data-availability",tabindex:"-1"},[t("Checking data availability "),e("a",{class:"header-anchor",href:"#checking-data-availability","aria-label":'Permalink to "Checking data availability"'},"​")],-1),A=e("p",null,[e("img",{src:l,alt:"DA network"})],-1),C=e("p",null,[t("To enhance connectivity, the celestia-node augments the celestia-app with a separate libp2p network, "),e("em",null,"i.e."),t(", the so-called "),e("em",null,"DA network"),t(", that serves DAS requests.")],-1),D={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},S={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.025ex"},xmlns:"http://www.w3.org/2000/svg",width:"2.31ex",height:"1.595ex",role:"img",focusable:"false",viewBox:"0 -694 1021 705","aria-hidden":"true"},I=s("",1),P=[I],N=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mn",null,"4"),e("mi",null,"k")])],-1),j=e("p",null,[t("Note that although it is recommended, performing DAS is optional -- light nodes could just trust that the data corresponding to the commitments in the block headers was indeed made available by the Celestia DA layer. In addition, light nodes can also submit transactions to the celestia-app, i.e., "),e("code",null,"PayForBlobs"),t(" transactions.")],-1),B=e("p",null,"While performing DAS for a block header, every light node queries Celestia Nodes for a number of random data shares from the extended matrix and the corresponding Merkle proofs. If all the queries are successful, then the light node accepts the block header as valid (from a DA perspective).",-1),Z=e("p",null,"If at least one of the queries fails (i.e., either the data share is not received or the Merkle proof is invalid), then the light node rejects the block header and tries again later. The retrial is necessary to deal with false negatives, i.e., block headers being rejected although the block data is available. This may happen due to network congestion for example.",-1),q=e("p",null,[t("Alternatively, light nodes may accept a block header although the data is not available, i.e., a "),e("em",null,"false positive"),t(". This is possible since the soundness property (i.e., if an honest light node accepts a block as available, then at least one honest full node will eventually have the entire block data) is probabilistically guaranteed (for more details, take a look at the "),e("a",{href:"https://arxiv.org/abs/1809.09044",target:"_blank",rel:"noreferrer"},"original paper"),t(").")],-1),E=e("p",null,"By fine tuning Celestia's parameters (e.g., the number of data shares sampled by each light node) the likelihood of false positives can be sufficiently reduced such that block producers have no incentive to withhold the block data.",-1);function z(R,J,F,G,$,U){return a(),o("div",null,[c,e("p",null,[t("Next, the block producer adds to the block header a commitment of the block data. As "),h,t(", the commitment is the Merkle root of the "),e("mjx-container",d,[(a(),o("svg",T,p)),m]),t(" intermediate Merkle roots (i.e., one for each row and column of the extended matrix). To compute this commitment, the block producer performs the following operations:")]),e("ul",null,[u,e("li",null,[t("It arranges these shares into a square matrix (row-wise). Note that the shares are padded to the next power of two. The outcome square of size "),e("mjx-container",_,[(a(),o("svg",g,x)),w]),t(" is referred to as the original data.")]),e("li",null,[t("It extends the original data to a "),e("mjx-container",b,[(a(),o("svg",k,v)),L]),t(" square matrix using the 2-dimensional Reed-Solomon encoding scheme described above. The extended shares (i.e., containing erasure data) are associated with another reserved namespace.")]),H]),V,M,A,C,e("p",null,[t("Light nodes connect to a celestia-node in the DA network, listen to extended block headers (i.e., the block headers together with the relevant DA metadata, such as the "),e("mjx-container",D,[(a(),o("svg",S,P)),N]),t(" intermediate Merkle roots), and perform DAS on the received headers (i.e., ask for random data shares).")]),j,B,Z,q,E])}const X=n(r,[["render",z]]);export{K as __pageData,X as default}; diff --git a/assets/learn_how-to-stake-tia.md.4ed3b67d.js b/assets/learn_how-to-stake-tia.md.4ed3b67d.js new file mode 100644 index 00000000000..a8bdd49797c --- /dev/null +++ b/assets/learn_how-to-stake-tia.md.4ed3b67d.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as o}from"./chunks/framework.51d6c45b.js";const r="/img/keplr.png",l="/img/leap.png",i="/img/keplr/keplr1.gif",n="/img/keplr/keplr2.gif",s="/img/keplr/keplr3.gif",p="/img/keplr/keplr4.gif",c="/img/leap/leap1.gif",d="/img/leap/leap2.gif",h="/img/leap/leap3.gif",A=JSON.parse('{"title":"How to stake TIA","description":"This tutorial covers how to stake TIA with Keplr or Leap wallet","frontmatter":{"sidebar_label":"How to stake TIA","description":"This tutorial covers how to stake TIA with Keplr or Leap wallet","head":[["meta",{"name":"og:title","content":"How to stake TIA | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/how-to-stake-tia.md","filePath":"learn/how-to-stake-tia.md","lastUpdated":1707231883000}'),k={name:"learn/how-to-stake-tia.md"},w=o('

    How to stake TIA

    Celestia is a proof-of-stake blockchain based on the Cosmos SDK.

    Staking TIA as a delegator allows you to secure the Celestia network. This means that you can stake the native token TIA and vote on governance proposals.

    In this tutorial, you will learn how to stake TIA tokens via Keplr and Leap wallets.

    Select your preferred wallet

    Keplr
    Leap

    Stake TIA with Keplr wallet

    1️⃣ Open your Keplr browser extension

    Navigate to Staked and select Stake with Keplr Dashboard.

    This will open the Keplr dashboard in a new browser page.

    Keplr1

    2️⃣ Select Celestia network and search for a validator

    In the Keplr dashboard, select the Celestia network and pick a validator of your choice.

    Keplr1

    3️⃣ Stake your TIA tokens

    On the following screen enter amount of TIA tokens and select Stake.

    A Keplr popup will appear, requesting your approval for the transaction. Select Approve.

    Keplr1

    4️⃣ Confirm and manage your TIA

    After the transaction is confirmed, you will see the following overview dashboard where you can claim rewards, unstake, redelegate, or stake additional tokens.

    Keplr1

    Stake TIA with Leap wallet

    1️⃣ Open your Leap browser extension

    In top right select Celestia network and navigate to Stake.

    Similarly to previous step, select the +Stake button.

    Keplr1

    2️⃣ Select a validator and stake TIA

    On the following screen choose a validator of your choice, enter the desired amount, and click Review.

    Following that, review the transaction details and select Stake, then wait for the transaction to finalize.

    Keplr1

    3️⃣ Confirm and manage your TIA

    After the transaction is confirmed, you will see the following overview dashboard where you can claim rewards, unstake, redelegate, or stake additional tokens.

    Keplr1

    ',34),m=[w];function u(f,g,_,b,y,T){return a(),t("div",null,m)}const I=e(k,[["render",u]]);export{A as __pageData,I as default}; diff --git a/assets/learn_how-to-stake-tia.md.4ed3b67d.lean.js b/assets/learn_how-to-stake-tia.md.4ed3b67d.lean.js new file mode 100644 index 00000000000..59dd336e2fd --- /dev/null +++ b/assets/learn_how-to-stake-tia.md.4ed3b67d.lean.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as o}from"./chunks/framework.51d6c45b.js";const r="/img/keplr.png",l="/img/leap.png",i="/img/keplr/keplr1.gif",n="/img/keplr/keplr2.gif",s="/img/keplr/keplr3.gif",p="/img/keplr/keplr4.gif",c="/img/leap/leap1.gif",d="/img/leap/leap2.gif",h="/img/leap/leap3.gif",A=JSON.parse('{"title":"How to stake TIA","description":"This tutorial covers how to stake TIA with Keplr or Leap wallet","frontmatter":{"sidebar_label":"How to stake TIA","description":"This tutorial covers how to stake TIA with Keplr or Leap wallet","head":[["meta",{"name":"og:title","content":"How to stake TIA | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/how-to-stake-tia.md","filePath":"learn/how-to-stake-tia.md","lastUpdated":1707231883000}'),k={name:"learn/how-to-stake-tia.md"},w=o("",34),m=[w];function u(f,g,_,b,y,T){return a(),t("div",null,m)}const I=e(k,[["render",u]]);export{A as __pageData,I as default}; diff --git a/assets/learn_paying-for-blobspace.md.50c0da43.js b/assets/learn_paying-for-blobspace.md.50c0da43.js new file mode 100644 index 00000000000..7c79e4cd95f --- /dev/null +++ b/assets/learn_paying-for-blobspace.md.50c0da43.js @@ -0,0 +1 @@ +import{_ as e}from"./chunks/tx-lifecycle.92aed1e6.js";import{_ as a,o as t,c as o,Q as i}from"./chunks/framework.51d6c45b.js";const u=JSON.parse('{"title":"Paying for blobspace","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Paying for blobspace | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/paying-for-blobspace.md","filePath":"learn/paying-for-blobspace.md","lastUpdated":1698091500000}'),s={name:"learn/paying-for-blobspace.md"},n=i('

    Paying for blobspace

    PayForBlobs transactions

    To publish data on Celestia, developers can submit PayForBlobs transactions. A PayForBlobs transaction consists of the identity of the sender, the data to be made available, the data size, the namespace, and a signature.

    Each PayForBlobs transaction is split into two parts: the blob or blobs which include the data to be made available along with the namespace, and the executable payment transaction which includes a commitment to the data.

    Both the blobs and executable payment transactions are put into the block within the appropriate namespace. The block data is extended using erasure coding and then Merkelized into a data root commitment included in the block header.

    Lifecycle of a celestia-app Transaction

    See the detailed life cycle of a Celestia transaction.

    Learn how to submit data to Celestia’s data availability layer.

    Fee market overview

    Celestia uses a standard gas-price prioritised mempool. This means that transactions with higher fees will be prioritised by validators. Fees are comprised of a flat fee per transaction and then a variable fee based on the size of each blob in the transaction.

    Understand how fees are calculated on Celestia in the overview on submitting PFB transactions.

    ',11),r=[n];function c(l,d,p,h,b,m){return t(),o("div",null,r)}const y=a(s,[["render",c]]);export{u as __pageData,y as default}; diff --git a/assets/learn_paying-for-blobspace.md.50c0da43.lean.js b/assets/learn_paying-for-blobspace.md.50c0da43.lean.js new file mode 100644 index 00000000000..b017199cccf --- /dev/null +++ b/assets/learn_paying-for-blobspace.md.50c0da43.lean.js @@ -0,0 +1 @@ +import{_ as e}from"./chunks/tx-lifecycle.92aed1e6.js";import{_ as a,o as t,c as o,Q as i}from"./chunks/framework.51d6c45b.js";const u=JSON.parse('{"title":"Paying for blobspace","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Paying for blobspace | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/paying-for-blobspace.md","filePath":"learn/paying-for-blobspace.md","lastUpdated":1698091500000}'),s={name:"learn/paying-for-blobspace.md"},n=i("",11),r=[n];function c(l,d,p,h,b,m){return t(),o("div",null,r)}const y=a(s,[["render",c]]);export{u as __pageData,y as default}; diff --git a/assets/learn_retrievability.md.c6864d62.js b/assets/learn_retrievability.md.c6864d62.js new file mode 100644 index 00000000000..af78b14fb50 --- /dev/null +++ b/assets/learn_retrievability.md.c6864d62.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as i}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"Data retrievability and pruning","description":"Practices and expectations for data retrievability and pruning on Celestia.","frontmatter":{"sidebar_label":"Data retrievability and pruning","description":"Practices and expectations for data retrievability and pruning on Celestia.","head":[["meta",{"name":"og:title","content":"Data retrievability and pruning | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/retrievability.md","filePath":"learn/retrievability.md","lastUpdated":1708603850000}'),o={name:"learn/retrievability.md"},r=i('

    Data retrievability and pruning

    The purpose of data availability layers such as Celestia is to ensure that block data is provably published, so that applications and rollups can know what the state of their chain is, and store that data. Once the data is published, data availability layers do not inherently guarantee that historical data will be permanently stored and remain retrievable.

    In this document, we discuss the state of data retrievability and pruning in Celestia, as well as some tips for rollup developers in order to ensure that syncing new rollup nodes is possible.

    Data retrievability and pruning in celestia-node

    As of version v0.13.0, celestia-node has implemented a light node sampling window of 30 days, as specified in CIP-4. This means that once pruning is implemented, light nodes will now only sample blocks within a 30-day window instead of sampling all blocks from genesis. This change introduces the concept of pruning to celestia-node, where data outside of the 30-day window may not be stored by light nodes, marking a significant update in how data retrievability and storage are managed within the network (v0.13.0 release notes).

    Data blobs older than the recency window will be pruned by default on light nodes, after pruning is fully implemented, but will continue to be stored by archival nodes that do not prune data. Light nodes will be able to query historic blob data in namespaces from archival nodes, as long as archival nodes exist on the public network.

    Once pruning is fully implemented, light nodes will only perform data availability sampling for blocks within the data recency window of 30 days.

    Suggested practices for rollups

    Rollups may need to access historic data in order to allow new rollup nodes to reconstruct the latest state by replaying historic blocks. Once data has been published on Celestia and guaranteed to have been made available, rollups and applications are responsible for storing their historical data.

    While it is possible to continue to do this by using the GetAll API method in celestia-node on historic blocks as long as archival nodes exist on the public Celestia network, rollup developers should not rely on this as the only method to access historical data, as archival nodes serving requests for historical data for free is not guaranteed. Below are some other suggested methods to access historical data.

    • Use professional archival node or data providers. It is expected that professional infrastructure providers will provide paid access to archival nodes, where historical data can be retrieved, for example using the GetAll API method. This provides better guarantees than solely relying on free archival nodes on the public Celestia network.
    • Share snapshots of rollup nodes. Rollups could share snapshots of their data directories which can be downloaded manually by users bootstrapping new nodes. These snapshots could contain the latest state of the rollup, and/or all the historical blocks.
    • Add peer-to-peer support for historical block sync. A less manual version of sharing snapshots, where rollup nodes could implement built-in support for block sync, where rollup nodes download historical block data from each other over a peer-to-peer network.
      • Namespace pinning. In the future, celestia-node is expected to allow nodes to choose to "pin" data from selected namespaces that they wish to store and make available for other nodes. This will allow rollup nodes to be responsible for storing their data, without needing to implement their own peer-to-peer historical block sync mechanism.
    ',11),s=[r];function n(l,d,c,h,p,u){return a(),t("div",null,s)}const f=e(o,[["render",n]]);export{g as __pageData,f as default}; diff --git a/assets/learn_retrievability.md.c6864d62.lean.js b/assets/learn_retrievability.md.c6864d62.lean.js new file mode 100644 index 00000000000..b64dde145b5 --- /dev/null +++ b/assets/learn_retrievability.md.c6864d62.lean.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as i}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"Data retrievability and pruning","description":"Practices and expectations for data retrievability and pruning on Celestia.","frontmatter":{"sidebar_label":"Data retrievability and pruning","description":"Practices and expectations for data retrievability and pruning on Celestia.","head":[["meta",{"name":"og:title","content":"Data retrievability and pruning | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/retrievability.md","filePath":"learn/retrievability.md","lastUpdated":1708603850000}'),o={name:"learn/retrievability.md"},r=i("",11),s=[r];function n(l,d,c,h,p,u){return a(),t("div",null,s)}const f=e(o,[["render",n]]);export{g as __pageData,f as default}; diff --git a/assets/learn_staking-governance-supply.md.b7f1bae9.js b/assets/learn_staking-governance-supply.md.b7f1bae9.js new file mode 100644 index 00000000000..afa0cb106d8 --- /dev/null +++ b/assets/learn_staking-governance-supply.md.b7f1bae9.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as o}from"./chunks/framework.51d6c45b.js";const n="/img/learn/Celestia_TIA_Inflation.png",r="/img/learn/Celestia_TIA_Allocation_at_Genesis.png",s="/img/learn/Celestia_TIA_Available_Supply.png",k=JSON.parse('{"title":"Staking, governance, & supply","description":"Learn about proof-of-stake on Celestia","frontmatter":{"description":"Learn about proof-of-stake on Celestia","head":[["meta",{"name":"og:title","content":"Staking, governance, & supply | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/staking-governance-supply.md","filePath":"learn/staking-governance-supply.md","lastUpdated":1713191709000}'),i={name:"learn/staking-governance-supply.md"},l=o('

    Staking, governance, & supply

    Proof-of-stake on Celestia

    Celestia is a proof-of-stake blockchain based on CometBFT and the Cosmos SDK. Celestia supports in-protocol delegation and will start with an initial validator set of 100.

    Staking TIA as a validator or delegator enables you to earn staking rewards from the network. Validators charge a fee to delegators which gives them a percentage of staking rewards.

    Learn how proof of stake works on Cosmos SDK chains like Celestia.

    Consensus mechanismProof-of-stake
    Blockchain frameworkCosmos SDK
    Validator set size100
    Delegation supportYes

    Learn how to stake on your own at the community dashboards.

    Inflation

    TIA inflation starts at 8% annually and decreases by 10% every year until it reaches the long term issuance rate of 1.5%. Exact annual inflation rates can be found in the diagram below.

    inflation diagram

    The annual provisions for inflation are calculated based on the total supply of TIA at the beginning of each year. To calculate how many TIA to issue per block, Celestia uses the block timestamp rather than the block height since the time between blocks can vary and cause actual issuance to be higher than the target.

    For an in-depth understanding, refer to ADR019.

    Decentralised governance

    Network parameters

    TIA holders (not just stakers) can propose and vote on governance proposals to change a subset of network parameters. To learn more, see a complete list of both the changeable and non-changeable parameters and their values. Additionally, learn how to submit and vote on governance proposals.

    Community pool

    Starting at genesis, Celestia’s community pool receives 2% of all Celestia block rewards. TIA stakers may vote to fund ecosystem initiatives as in many other Cosmos SDK chains.

    Learn how to submit a governance proposal to spend community pool funds.

    TIA allocation at genesis

    Celestia will have a total supply of 1,000,000,000 TIA at genesis, split across five categories described in the chart and table below.

    allocation diagram

    CategoryDescription%
    Public AllocationGenesis Drop and Incentivized Testnet: 7.4%
    Future initiatives: 12.6%
    20%
    R&D & EcosystemTokens allocated to the Celestia Foundation and core devs for research, development, and ecosystem initiatives including:
    - Protocol maintenance and development
    - Programs for rollup developers, infrastructure, and node operators
    26.8%
    Early Backers: Series A&BEarly supporters of Celestia19.7%
    Early Backers: SeedEarly supporters of Celestia15.9%
    Initial Core ContributorsMembers of Celestia Labs, the first core contributor to Celestia17.6%

    Unlocks

    Celestia’s 1 billion TIA supply at genesis will be subject to several different unlock schedules. All tokens, locked or unlocked, may be staked, but staking rewards are unlocked upon receipt and will add to the circulating supply.

    Circulating supply is defined as the amount of TIA tokens in general circulation without onchain transfer restrictions.

    Available supply is defined as the amount of TIA tokens that are either part of the circulating supply or are unlocked but subject to some form of governance to determine when the tokens are allocated. This includes the unlocked portion of the R&D & Ecosystem tokens and the tokens set aside for future initiatives.

    The definitions for circulating and available supply were adapted from Optimism’s definitions.

    supply diagram

    Unlock schedule by category is described in the table below.

    CategoryUnlock Schedule
    Public AllocationFully unlocked at launch.
    R&D & Ecosystem25% unlocked at launch.
    Remaining 75% unlocks continuously from year 1 to year 4.
    Initial Core Contributors33% unlocked at year 1.
    Remaining 67% unlocks continuously from year 1 to year 3.
    Early Backers: Seed33% unlocked at year 1.
    Remaining 67% unlocks continuously from year 1 to year 2.
    Early Backers: Series A&B33% unlocked at year 1.
    Remaining 67% unlocks continuously from year 1 to year 2.
    ',30),d=[l];function c(p,h,u,m,f,g){return t(),a("div",null,d)}const y=e(i,[["render",c]]);export{k as __pageData,y as default}; diff --git a/assets/learn_staking-governance-supply.md.b7f1bae9.lean.js b/assets/learn_staking-governance-supply.md.b7f1bae9.lean.js new file mode 100644 index 00000000000..b1306a73b2d --- /dev/null +++ b/assets/learn_staking-governance-supply.md.b7f1bae9.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as o}from"./chunks/framework.51d6c45b.js";const n="/img/learn/Celestia_TIA_Inflation.png",r="/img/learn/Celestia_TIA_Allocation_at_Genesis.png",s="/img/learn/Celestia_TIA_Available_Supply.png",k=JSON.parse('{"title":"Staking, governance, & supply","description":"Learn about proof-of-stake on Celestia","frontmatter":{"description":"Learn about proof-of-stake on Celestia","head":[["meta",{"name":"og:title","content":"Staking, governance, & supply | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/staking-governance-supply.md","filePath":"learn/staking-governance-supply.md","lastUpdated":1713191709000}'),i={name:"learn/staking-governance-supply.md"},l=o("",30),d=[l];function c(p,h,u,m,f,g){return t(),a("div",null,d)}const y=e(i,[["render",c]]);export{k as __pageData,y as default}; diff --git a/assets/learn_staking.md.d69d2c40.js b/assets/learn_staking.md.d69d2c40.js new file mode 100644 index 00000000000..1db09bd1175 --- /dev/null +++ b/assets/learn_staking.md.d69d2c40.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as n}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"Staking on Celestia","description":"Learn how to stake your tokens on the Celestia network.","frontmatter":{"description":"Learn how to stake your tokens on the Celestia network.","head":[["meta",{"name":"og:title","content":"Staking on Celestia | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/staking.md","filePath":"learn/staking.md","lastUpdated":1707231883000}'),s={name:"learn/staking.md"},r=n('

    Staking on Celestia

    Engage with the Celestia network at a deeper level through staking. An essential mechanism to a proof-of-stake network, users can secure the network by delegating to a validator and receive a share of its staking rewards.

    Mainnet Beta

    Currently, the following staking interfaces exist for the Mainnet Beta.

    Just connect your wallet to get started!

    Mocha testnet

    Currently, the following staking interfaces exist for the Mocha testnet.

    Just connect your wallet to get started!

    ',10),l=[r];function i(o,h,c,p,k,d){return t(),a("div",null,l)}const m=e(s,[["render",i]]);export{g as __pageData,m as default}; diff --git a/assets/learn_staking.md.d69d2c40.lean.js b/assets/learn_staking.md.d69d2c40.lean.js new file mode 100644 index 00000000000..c9a140bf8b5 --- /dev/null +++ b/assets/learn_staking.md.d69d2c40.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as n}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"Staking on Celestia","description":"Learn how to stake your tokens on the Celestia network.","frontmatter":{"description":"Learn how to stake your tokens on the Celestia network.","head":[["meta",{"name":"og:title","content":"Staking on Celestia | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/staking.md","filePath":"learn/staking.md","lastUpdated":1707231883000}'),s={name:"learn/staking.md"},r=n("",10),l=[r];function i(o,h,c,p,k,d){return t(),a("div",null,l)}const m=e(s,[["render",i]]);export{g as __pageData,m as default}; diff --git a/assets/learn_tia.md.fc5441b2.js b/assets/learn_tia.md.fc5441b2.js new file mode 100644 index 00000000000..cf8a06254da --- /dev/null +++ b/assets/learn_tia.md.fc5441b2.js @@ -0,0 +1 @@ +import{_ as s,o as t,c as e,k as a,a as o,Q as n}from"./chunks/framework.51d6c45b.js";const A=JSON.parse('{"title":"Overview of TIA","description":"","frontmatter":{"prev":{"text":"Data availability FAQ","link":"/learn/how-celestia-works/data-availability-faq"},"head":[["meta",{"name":"og:title","content":"Overview of TIA | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/tia.md","filePath":"learn/tia.md","lastUpdated":1702318874000}'),i={name:"learn/tia.md"},r=a("h1",{id:"overview-of-tia",tabindex:"-1"},[o("Overview of TIA "),a("a",{class:"header-anchor",href:"#overview-of-tia","aria-label":'Permalink to "Overview of TIA"'},"​")],-1),l=a("h2",{id:"tia-at-a-glance",tabindex:"-1"},[o("TIA at a glance "),a("a",{class:"header-anchor",href:"#tia-at-a-glance","aria-label":'Permalink to "TIA at a glance"'},"​")],-1),d=a("thead",null,[a("tr",null,[a("th",null,"Property"),a("th",null,"Details")])],-1),c=a("tr",null,[a("td",null,"Abbreviation"),a("td",null,"TIA")],-1),h=a("tr",null,[a("td",null,"Total supply at genesis"),a("td",null,"1,000,000,000 TIA")],-1),Q=a("tr",null,[a("td",null,"Inflation schedule"),a("td",null,"8% in the first year, decreasing 10% per year until reaching an inflation floor of 1.5% annually")],-1),p=a("tr",null,[a("td",null,"Decimals"),a("td",null,"6")],-1),T=a("td",null,"Conversion",-1),m={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},u={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.186ex"},xmlns:"http://www.w3.org/2000/svg",width:"21.526ex",height:"2.14ex",role:"img",focusable:"false",viewBox:"0 -864 9514.7 946","aria-hidden":"true"},f=n('',1),g=[f],H=a("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[a("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[a("mtext",null,"1 uTIA"),a("mo",null,"="),a("mtext",null,"TIA"),a("mo",null,"×"),a("msup",null,[a("mn",null,"10"),a("mrow",{"data-mjx-texclass":"ORD"},[a("mo",null,"−"),a("mn",null,"6")])])])],-1),k=n('

    Role of TIA

    Paying for blobspace

    Celestia’s native asset, TIA, is an essential part of how developers build on the first modular blockchain network. To use Celestia for data availability, rollup developers submit PayForBlobs transactions on the network for a fee, denominated in TIA.

    Bootstrapping new rollups

    A core part of the Celestia vision is that deploying a blockchain should be as easy as deploying a smart contract. In the modular era, developers no longer need to issue a token to launch their own blockchain.

    Similarly to ETH on Ethereum-based rollups, developers may opt to bootstrap their chain quickly by using TIA as a gas token and currency, in addition to paying for data availability. In this mode, developers can focus on creating their application or execution layer, instead of issuing a token right away.

    Proof-of-stake

    As a permissionless network built with Cosmos SDK, Celestia uses proof-of-stake to secure its own consensus. Like in other Cosmos networks, any user can help secure the network by delegating their TIA to a Celestia validator for a portion of their validator’s staking rewards.

    Learn how proof-of-stake works in Cosmos.

    Decentralised governance

    TIA staking also allows the community to play a critical role in decentralised governance over key parts of Celestia, such as voting on network parameters through governance proposals, and governing the community pool, which receives 2% of block rewards.

    Learn more about Celestia’s decentralised governance model.

    Denominations

    TIA: display token

    TIA is the DisplayDenom that you will typically see in wallets and user interfaces.

    utia: staking denomination

    utia is the BondDenom and stands for "micro TIA", with 1 TIA = 1,000,000 utia. This is the native staking denomination.

    In staking operations or transactions, if no denomination is specified, utia is assumed.

    microtia: staking denomination alias

    microtia is the BondDenomAlias, an alias for utia.

    ',20);function b(_,w,V,y,v,x){return t(),e("div",null,[r,l,a("table",null,[d,a("tbody",null,[c,h,Q,p,a("tr",null,[T,a("td",null,[a("mjx-container",m,[(t(),e("svg",u,g)),H])])])])]),k])}const I=s(i,[["render",b]]);export{A as __pageData,I as default}; diff --git a/assets/learn_tia.md.fc5441b2.lean.js b/assets/learn_tia.md.fc5441b2.lean.js new file mode 100644 index 00000000000..2755fba516d --- /dev/null +++ b/assets/learn_tia.md.fc5441b2.lean.js @@ -0,0 +1 @@ +import{_ as s,o as t,c as e,k as a,a as o,Q as n}from"./chunks/framework.51d6c45b.js";const A=JSON.parse('{"title":"Overview of TIA","description":"","frontmatter":{"prev":{"text":"Data availability FAQ","link":"/learn/how-celestia-works/data-availability-faq"},"head":[["meta",{"name":"og:title","content":"Overview of TIA | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"learn/tia.md","filePath":"learn/tia.md","lastUpdated":1702318874000}'),i={name:"learn/tia.md"},r=a("h1",{id:"overview-of-tia",tabindex:"-1"},[o("Overview of TIA "),a("a",{class:"header-anchor",href:"#overview-of-tia","aria-label":'Permalink to "Overview of TIA"'},"​")],-1),l=a("h2",{id:"tia-at-a-glance",tabindex:"-1"},[o("TIA at a glance "),a("a",{class:"header-anchor",href:"#tia-at-a-glance","aria-label":'Permalink to "TIA at a glance"'},"​")],-1),d=a("thead",null,[a("tr",null,[a("th",null,"Property"),a("th",null,"Details")])],-1),c=a("tr",null,[a("td",null,"Abbreviation"),a("td",null,"TIA")],-1),h=a("tr",null,[a("td",null,"Total supply at genesis"),a("td",null,"1,000,000,000 TIA")],-1),Q=a("tr",null,[a("td",null,"Inflation schedule"),a("td",null,"8% in the first year, decreasing 10% per year until reaching an inflation floor of 1.5% annually")],-1),p=a("tr",null,[a("td",null,"Decimals"),a("td",null,"6")],-1),T=a("td",null,"Conversion",-1),m={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},u={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.186ex"},xmlns:"http://www.w3.org/2000/svg",width:"21.526ex",height:"2.14ex",role:"img",focusable:"false",viewBox:"0 -864 9514.7 946","aria-hidden":"true"},f=n("",1),g=[f],H=a("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[a("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[a("mtext",null,"1 uTIA"),a("mo",null,"="),a("mtext",null,"TIA"),a("mo",null,"×"),a("msup",null,[a("mn",null,"10"),a("mrow",{"data-mjx-texclass":"ORD"},[a("mo",null,"−"),a("mn",null,"6")])])])],-1),k=n("",20);function b(_,w,V,y,v,x){return t(),e("div",null,[r,l,a("table",null,[d,a("tbody",null,[c,h,Q,p,a("tr",null,[T,a("td",null,[a("mjx-container",m,[(t(),e("svg",u,g)),H])])])])]),k])}const I=s(i,[["render",b]]);export{A as __pageData,I as default}; diff --git a/assets/nodes_arabica-devnet.md.1651e52a.js b/assets/nodes_arabica-devnet.md.1651e52a.js new file mode 100644 index 00000000000..5bc1426ce28 --- /dev/null +++ b/assets/nodes_arabica-devnet.md.1651e52a.js @@ -0,0 +1,5 @@ +import{A as i}from"./chunks/ArabicaVersionTags.e649ff5a.js";import{c as d}from"./chunks/constants.dc70dfe7.js";import{_ as h,o,c as n,k as e,t as a,H as s,Q as r,a as u}from"./chunks/framework.51d6c45b.js";import"./chunks/arabica_versions.d7daf483.js";const b="/img/arabica-devnet.png",y={name:"ArabicaDevnetDetails",data(){return{constants:d}}},E=e("tr",null,[e("th",null,"Detail"),e("th",null,"Value")],-1),g=e("td",null,"Chain ID",-1),f=e("tr",null,[e("td",null,"Genesis hash"),e("td",null,[e("code",null,"27122593765E07329BC348E8D16E92DCB4C75B34CCCB35C640FD7A4484D4C711")])],-1),m=e("td",null,"Genesis file",-1),v=["href"],_=e("td",null,"Peers file",-1),k=["href"],C=e("tr",null,[e("td",null,"Validators"),e("td",null," 4 ")],-1);function A(l,c,p,N,t,I){return o(),n("table",null,[E,e("tr",null,[g,e("td",null,[e("code",null,a(t.constants.arabicaChainId),1)])]),f,e("tr",null,[m,e("td",null,[e("a",{href:`https://github.com/celestiaorg/networks/blob/master/${t.constants.arabicaChainId}/genesis.json`,target:"_blank",rel:"noopener noreferrer"}," https://github.com/celestiaorg/networks/blob/master/"+a(t.constants.arabicaChainId)+"/genesis.json ",9,v)])]),e("tr",null,[_,e("td",null,[e("a",{href:`https://github.com/celestiaorg/networks/blob/master/${t.constants.arabicaChainId}/peers.txt`,target:"_blank",rel:"noopener noreferrer"}," https://github.com/celestiaorg/networks/blob/master/"+a(t.constants.arabicaChainId)+"/peers.txt ",9,k)])]),C])}const F=h(y,[["render",A]]),w=r('

    Arabica devnet

    arabica-devnet

    Arabica devnet is a testnet from Celestia Labs that is focused exclusively on providing developers with enhanced performance and the latest upgrades for testing their rollups and applications.

    Arabica does not focus on validator or consensus-level testing, rather, that is what Mocha testnet is used for. If you are a validator, we recommend testing your validator operations on the Mocha testnet.

    Network stability and upgrades

    Arabica has the latest updates from all Celestia's products deployed on it, it can be subject to many changes. Therefore, as a fair warning, Arabica can break unexpectedly, but given it will be continuously updated, it is a useful way to keep testing the latest changes in the software.

    Developers can still deploy on Mocha testnet their sovereign rollups if they chose to do so, it just will always lag behind Arabica devnet until Mocha undergoes Hardfork Upgrades in coordination with Validators.

    Network details

    ',8),D=e("h3",{id:"software-version-numbers",tabindex:"-1"},[u("Software version numbers "),e("a",{class:"header-anchor",href:"#software-version-numbers","aria-label":'Permalink to "Software version numbers"'},"​")],-1),T=r(`

    Integrations

    This guide contains the relevant sections for how to connect to Arabica devnet, depending on the type of node you are running. Your best approach to participating is to first determine which node you would like to run. Each node’s guide will link to the relevant network in order to show you how to connect to them. Learn about the different endpoint types in the Cosmos SDK documentation.

    RPC endpoints

    RPC endpoints and types of nodes you can run in order to participate in Arabica devnet:

    Node typeEndpoint typeEndpoint
    Consensus nodes (full)Consensus RPChttps://rpc.celestia-arabica-11.com
    APIhttps://api.celestia-arabica-11.com
    gRPCgrpc.celestia-arabica-11.com:443
    Direct endpoints with open portsOpen ports: 26656 (p2p), 26657 (RPC), 1317 (API), 9090 (GRPC)
    validator-1.celestia-arabica-11.com
    validator-2.celestia-arabica-11.com
    validator-3.celestia-arabica-11.com
    validator-4.celestia-arabica-11.com
    Data availability nodesDA Bridge Node Endpoints/dns4/da-bridge-1.celestia-arabica-11.com/tcp/2121/p2p/12D3KooWGqwzdEqM54Dce6LXzfFr97Bnhvm6rN7KM7MFwdomfm4S
    (light, bridge, full)/dns4/da-bridge-2.celestia-arabica-11.com/tcp/2121/p2p/12D3KooWCMGM5eZWVfCN9ZLAViGfLUWAfXP5pCm78NFKb9jpBtua
    /dns4/da-bridge-3.celestia-arabica-11.com/tcp/2121/p2p/12D3KooWEWuqrjULANpukDFGVoHW3RoeUU53Ec9t9v5cwW3MkVdQ
    /dns4/da-bridge-4.celestia-arabica-11.com/tcp/2121/p2p/12D3KooWLT1ysSrD7XWSBjh7tU1HQanF5M64dHV6AuM6cYEJxMPk
    --core.ip string endpointsRefer to "Direct endpoints with open ports" above

    You can find the status of these endpoints.

    Using consensus endpoints with DA nodes

    Consensus RPC endpoints are used to provide DA nodes with state access for querying the chain’s state and broadcasting transactions (balances, blobs, etc.) to the Celestia network.

    Developers will need to provide a –core.ip string from a consensus node’s URI or an IP that populates 2 ports for 2 types (RPC and gRPC, at ports 26657 and 9090, respectively) to their respective DA node.

    EXAMPLE

    bash
    celestia <da_type> start –core.ip <url> –core.rpc.port <port> \\
    +    –core.grpc.port <port> \\
    celestia <da_type> start –core.ip <url> –core.rpc.port <port> \\
    +    –core.grpc.port <port> \\

    RPCs for DA nodes to initialise or start your celestia-node to Arabica devnet with can be found in the table in the "Direct endpoints with open ports" section above.

    As an example, this command will work to start a light node with state access, using default ports:

    bash
    celestia light start --p2p.network arabica \\
    +  --core.ip validator-1.celestia-arabica-11.com
    celestia light start --p2p.network arabica \\
    +  --core.ip validator-1.celestia-arabica-11.com

    Bridge node runners

    Not all of the RPC endpoints do not guarantee the full block history. Find an archive endpoint on the community dashboard or run your own full consensus node with no pruning for your bridge node.

    Arabica devnet faucet

    WARNING

    USING THIS FAUCET DOES NOT ENTITLE YOU TO ANY AIRDROP OR OTHER DISTRIBUTION OF MAINNET CELESTIA TOKENS. THERE ARE NO PUBLIC SALES OF ANY MAINNET CELESTIA TOKENS.

    Discord

    You can request from Arabica devnet Faucet on the #arabica-faucet channel on Celestia's Discord server with the following command:

    text
    $request <CELESTIA-ADDRESS>
    $request <CELESTIA-ADDRESS>

    Where <CELESTIA-ADDRESS> is a celestia1****** generated address.

    NOTE

    Faucet has a limit of 10 tokens per week per address/Discord ID.

    Web

    The web faucet is available at https://faucet.celestia-arabica-11.com/.

    Explorers

    There are multiple explorers you can use for Arabica:

    Network upgrades

    Join our Telegram announcement channel for network upgrades.

    `,28),R=JSON.parse('{"title":"Arabica devnet","description":"A guide to Arabica devnet.","frontmatter":{"description":"A guide to Arabica devnet.","head":[["meta",{"name":"og:title","content":"Arabica devnet | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/arabica-devnet.md","filePath":"nodes/arabica-devnet.md","lastUpdated":1712705756000}'),P={name:"nodes/arabica-devnet.md"},L=Object.assign(P,{setup(l){return(c,p)=>(o(),n("div",null,[w,s(F),D,s(i),T]))}});export{R as __pageData,L as default}; diff --git a/assets/nodes_arabica-devnet.md.1651e52a.lean.js b/assets/nodes_arabica-devnet.md.1651e52a.lean.js new file mode 100644 index 00000000000..fa3c11a0533 --- /dev/null +++ b/assets/nodes_arabica-devnet.md.1651e52a.lean.js @@ -0,0 +1 @@ +import{A as i}from"./chunks/ArabicaVersionTags.e649ff5a.js";import{c as d}from"./chunks/constants.dc70dfe7.js";import{_ as h,o,c as n,k as e,t as a,H as s,Q as r,a as u}from"./chunks/framework.51d6c45b.js";import"./chunks/arabica_versions.d7daf483.js";const b="/img/arabica-devnet.png",y={name:"ArabicaDevnetDetails",data(){return{constants:d}}},E=e("tr",null,[e("th",null,"Detail"),e("th",null,"Value")],-1),g=e("td",null,"Chain ID",-1),f=e("tr",null,[e("td",null,"Genesis hash"),e("td",null,[e("code",null,"27122593765E07329BC348E8D16E92DCB4C75B34CCCB35C640FD7A4484D4C711")])],-1),m=e("td",null,"Genesis file",-1),v=["href"],_=e("td",null,"Peers file",-1),k=["href"],C=e("tr",null,[e("td",null,"Validators"),e("td",null," 4 ")],-1);function A(l,c,p,N,t,I){return o(),n("table",null,[E,e("tr",null,[g,e("td",null,[e("code",null,a(t.constants.arabicaChainId),1)])]),f,e("tr",null,[m,e("td",null,[e("a",{href:`https://github.com/celestiaorg/networks/blob/master/${t.constants.arabicaChainId}/genesis.json`,target:"_blank",rel:"noopener noreferrer"}," https://github.com/celestiaorg/networks/blob/master/"+a(t.constants.arabicaChainId)+"/genesis.json ",9,v)])]),e("tr",null,[_,e("td",null,[e("a",{href:`https://github.com/celestiaorg/networks/blob/master/${t.constants.arabicaChainId}/peers.txt`,target:"_blank",rel:"noopener noreferrer"}," https://github.com/celestiaorg/networks/blob/master/"+a(t.constants.arabicaChainId)+"/peers.txt ",9,k)])]),C])}const F=h(y,[["render",A]]),w=r("",8),D=e("h3",{id:"software-version-numbers",tabindex:"-1"},[u("Software version numbers "),e("a",{class:"header-anchor",href:"#software-version-numbers","aria-label":'Permalink to "Software version numbers"'},"​")],-1),T=r("",28),R=JSON.parse('{"title":"Arabica devnet","description":"A guide to Arabica devnet.","frontmatter":{"description":"A guide to Arabica devnet.","head":[["meta",{"name":"og:title","content":"Arabica devnet | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/arabica-devnet.md","filePath":"nodes/arabica-devnet.md","lastUpdated":1712705756000}'),P={name:"nodes/arabica-devnet.md"},L=Object.assign(P,{setup(l){return(c,p)=>(o(),n("div",null,[w,s(F),D,s(i),T]))}});export{R as __pageData,L as default}; diff --git a/assets/nodes_bridge-node.md.c77c0ff2.js b/assets/nodes_bridge-node.md.c77c0ff2.js new file mode 100644 index 00000000000..0f0534ec43d --- /dev/null +++ b/assets/nodes_bridge-node.md.c77c0ff2.js @@ -0,0 +1,7 @@ +import{_ as s,o as a,c as e,Q as o}from"./chunks/framework.51d6c45b.js";const n="/img/nodes/BridgeNodes.png",u=JSON.parse('{"title":"Setting up a Celestia bridge node","description":"","frontmatter":{"\\\\description":"Learn how to set up your Celestia bridge node.","head":[["meta",{"name":"og:title","content":"Setting up a Celestia bridge node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/bridge-node.md","filePath":"nodes/bridge-node.md","lastUpdated":1709131711000}'),l={name:"nodes/bridge-node.md"},t=o('

    Setting up a Celestia bridge node

    This tutorial will go over the steps to setting up your Celestia bridge node.

    Bridge nodes connect the data availability layer and the consensus layer.

    Overview of bridge nodes

    A Celestia bridge node has the following properties:

    1. Import and process “raw” headers & blocks from a trusted core process (meaning a trusted RPC connection to a celestia-core node) in the Consensus network. Bridge nodes can run this core process internally (embedded) or simply connect to a remote endpoint. Bridge nodes also have the option of being an active validator in the consensus network.
    2. Validate and erasure code the “raw” blocks
    3. Supply block shares with data availability headers to light nodes in the DA network.

    bridge-node-diagram

    From an implementation perspective, Bridge nodes run two separate processes:

    1. celestia-app with celestia-core (see repo)

      • celestia-app is the state machine where the application and the proof-of-stake logic is run. celestia-app is built on Cosmos SDK and also encompasses celestia-core.
      • celestia-core is the state interaction, consensus and block production layer. celestia-core is built on Tendermint Core, modified to store data roots of erasure coded blocks among other changes (see ADRs).
    2. celestia-node (see repo)

      • celestia-node augments the above with a separate libp2p network that serves data availability sampling requests. The team sometimes refers to this as the “halo” network.

    Hardware requirements

    The following hardware minimum requirements are recommended for running the bridge node:

    • Memory: 4 GB RAM (minimum)
    • CPU: 6 cores
    • Disk: 10 TB SSD Storage
    • Bandwidth: 1 Gbps for Download/1 Gbps for Upload

    Setting up your bridge node

    The following tutorial is done on an Ubuntu Linux 20.04 (LTS) x64 instance machine.

    Setup the dependencies

    Follow the tutorial for installing the dependencies.

    Deploy the Celestia bridge node

    Install Celestia Node

    Install the celestia-node binary, which will be used to run the bridge node.

    Follow the tutorial for installing celestia-node.

    Initialize the bridge node

    Run the following:

    sh
    celestia bridge init --core.ip <URI>
    celestia bridge init --core.ip <URI>

    The --core.ip gRPC port defaults to 9090, so if you do not specify it in the command line, it will default to that port. You can add the port after the IP address or use the --core.grpc.port flag to specify another port if you prefer.

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    Using an RPC of your own, or one from the list on the Mocha testnet page or list on the Arabica devnet page, start your node.

    Here is an example of initializing the bridge node:

    sh
    celestia bridge init --core.ip <URI>
    celestia bridge init --core.ip <URI>
    sh
    celestia bridge init --core.ip <URI> --p2p.network mocha
    celestia bridge init --core.ip <URI> --p2p.network mocha
    sh
    celestia bridge init --core.ip <URI> --p2p.network arabica
    celestia bridge init --core.ip <URI> --p2p.network arabica

    Run the bridge node

    Start the bridge node with a connection to a validator node's gRPC endpoint (which is usually exposed on port 9090):

    sh
    celestia bridge start --core.ip <URI>
    celestia bridge start --core.ip <URI>

    Here is an example of starting the bridge node on Mocha:

    sh
    celestia bridge start --core.ip rpc-mocha.pops.one:26657 --p2p.network mocha
    celestia bridge start --core.ip rpc-mocha.pops.one:26657 --p2p.network mocha

    And on Arabica:

    sh
    celestia bridge start --core.ip validator-1.celestia-arabica-11.com \\
    +  --p2p.network arabica
    celestia bridge start --core.ip validator-1.celestia-arabica-11.com \\
    +  --p2p.network arabica

    You can create your key for your node by following the cel-key instructions.

    Once you start the bridge node, a wallet key will be generated for you. You will need to fund that address with Testnet tokens to pay for PayForBlob transactions. You can find the address by running the following command:

    sh
    ./cel-key list --node.type bridge --keyring-backend test --p2p.network <network>
    ./cel-key list --node.type bridge --keyring-backend test --p2p.network <network>

    TIP

    You do not need to declare a network for Mainnet Beta. Refer to the chain ID section on the troubleshooting page for more information

    You can get testnet tokens from:

    NOTE

    If you are running a bridge node for your validator it is highly recommended to request Mocha testnet tokens as this is the testnet used to test out validator operations.

    Optional: run the bridge node with a custom key

    In order to run a bridge node using a custom key:

    1. The custom key must exist inside the celestia bridge node directory at the correct path (default: ~/.celestia-bridge/keys/keyring-test)
    2. The name of the custom key must be passed upon start, like so:
    sh
    celestia bridge start --core.ip <URI> --keyring.accname <name-of-custom-key>
    celestia bridge start --core.ip <URI> --keyring.accname <name-of-custom-key>
    sh
    celestia bridge start --core.ip <URI> --keyring.accname <name-of-custom-key> \\
    +  --p2p.network mocha
    celestia bridge start --core.ip <URI> --keyring.accname <name-of-custom-key> \\
    +  --p2p.network mocha
    sh
    celestia bridge start --core.ip <URI> --keyring.accname <name-of-custom-key> \\
    +  --p2p.network arabica
    celestia bridge start --core.ip <URI> --keyring.accname <name-of-custom-key> \\
    +  --p2p.network arabica

    Optional: start the bridge node with SystemD

    Follow the tutorial on setting up the bridge node as a background process with SystemD.

    You have successfully set up a bridge node that is syncing with the network.

    `,49),p=[t];function r(c,i,d,y,E,h){return a(),e("div",null,p)}const F=s(l,[["render",r]]);export{u as __pageData,F as default}; diff --git a/assets/nodes_bridge-node.md.c77c0ff2.lean.js b/assets/nodes_bridge-node.md.c77c0ff2.lean.js new file mode 100644 index 00000000000..a8f1e106e54 --- /dev/null +++ b/assets/nodes_bridge-node.md.c77c0ff2.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as e,Q as o}from"./chunks/framework.51d6c45b.js";const n="/img/nodes/BridgeNodes.png",u=JSON.parse('{"title":"Setting up a Celestia bridge node","description":"","frontmatter":{"\\\\description":"Learn how to set up your Celestia bridge node.","head":[["meta",{"name":"og:title","content":"Setting up a Celestia bridge node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/bridge-node.md","filePath":"nodes/bridge-node.md","lastUpdated":1709131711000}'),l={name:"nodes/bridge-node.md"},t=o("",49),p=[t];function r(c,i,d,y,E,h){return a(),e("div",null,p)}const F=s(l,[["render",r]]);export{u as __pageData,F as default}; diff --git a/assets/nodes_celestia-app-commands.md.63e455f7.js b/assets/nodes_celestia-app-commands.md.63e455f7.js new file mode 100644 index 00000000000..f7099882b45 --- /dev/null +++ b/assets/nodes_celestia-app-commands.md.63e455f7.js @@ -0,0 +1,213 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.51d6c45b.js";const u=JSON.parse('{"title":"Helpful CLI commands","description":"Some of the most helpful celestia-app CLI commands.","frontmatter":{"description":"Some of the most helpful celestia-app CLI commands.","next":{"text":"SystemD","link":"/nodes/systemd"},"head":[["meta",{"name":"og:title","content":"Helpful CLI commands | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-commands.md","filePath":"nodes/celestia-app-commands.md","lastUpdated":1702318874000}'),p={name:"nodes/celestia-app-commands.md"},o=l(`

    Helpful CLI commands

    View all options:

    console
    $ celestia-appd --help
    +Start celestia-app
    +
    +Usage:
    +  celestia-appd [command]
    +
    +Available Commands:
    +  add-genesis-account Add a genesis account to genesis.json
    +  collect-gentxs      Collect genesis txs and output a genesis.json file
    +  config              Create or query an application CLI configuration file
    +  debug               Tool for helping with debugging your application
    +  export              Export state to JSON
    +  gentx               Generate a genesis tx carrying a self delegation
    +  help                Help about any command
    +  init                Initialize private validator, p2p, genesis,
    +  and application configuration files
    +  keys                Manage your application's keys
    +  migrate             Migrate genesis to a specified target version
    +  query               Querying subcommands
    +  rollback            rollback tendermint state by one height
    +  rollback            rollback cosmos-sdk and tendermint state by one height
    +  start               Run the full node
    +  status              Query remote node for status
    +  tendermint          Tendermint subcommands
    +  tx                  Transactions subcommands
    +  validate-genesis    validates the genesis file at the default
    +  location or at the location passed as an arg
    +  version             Print the application binary version information
    $ celestia-appd --help
    +Start celestia-app
    +
    +Usage:
    +  celestia-appd [command]
    +
    +Available Commands:
    +  add-genesis-account Add a genesis account to genesis.json
    +  collect-gentxs      Collect genesis txs and output a genesis.json file
    +  config              Create or query an application CLI configuration file
    +  debug               Tool for helping with debugging your application
    +  export              Export state to JSON
    +  gentx               Generate a genesis tx carrying a self delegation
    +  help                Help about any command
    +  init                Initialize private validator, p2p, genesis,
    +  and application configuration files
    +  keys                Manage your application's keys
    +  migrate             Migrate genesis to a specified target version
    +  query               Querying subcommands
    +  rollback            rollback tendermint state by one height
    +  rollback            rollback cosmos-sdk and tendermint state by one height
    +  start               Run the full node
    +  status              Query remote node for status
    +  tendermint          Tendermint subcommands
    +  tx                  Transactions subcommands
    +  validate-genesis    validates the genesis file at the default
    +  location or at the location passed as an arg
    +  version             Print the application binary version information

    Creating a wallet

    sh
    celestia-appd config keyring-backend test
    celestia-appd config keyring-backend test

    keyring-backend configures the keyring's backend, where the keys are stored.

    Options are: os|file|kwallet|pass|test|memory.

    You can learn more on the Cosmos documentation or Go Package documentation.

    Key management

    sh
    # listing keys
    +celestia-appd keys list
    +
    +# adding keys
    +celestia-appd keys add <KEY_NAME>
    +
    +# deleting keys
    +celestia-appd keys delete <KEY_NAME>
    +
    +# renaming keys
    +celestia-appd keys rename <CURRENT_KEY_NAME> <NEW_KEY_NAME>
    # listing keys
    +celestia-appd keys list
    +
    +# adding keys
    +celestia-appd keys add <KEY_NAME>
    +
    +# deleting keys
    +celestia-appd keys delete <KEY_NAME>
    +
    +# renaming keys
    +celestia-appd keys rename <CURRENT_KEY_NAME> <NEW_KEY_NAME>

    Importing and exporting keys

    Import an encrypted and ASCII-armored private key into the local keybase.

    sh
    celestia-appd keys import <KEY_NAME> <KEY_FILE>
    celestia-appd keys import <KEY_NAME> <KEY_FILE>

    Example usage:

    sh
    celestia-appd keys import amanda ./keyfile.txt
    celestia-appd keys import amanda ./keyfile.txt

    Export a private key from the local keyring in encrypted and ASCII-armored format:

    sh
    celestia-appd keys export <KEY_NAME>
    +
    +# you will then be prompted to set a password for the encrypted private key:
    +Enter passphrase to encrypt the exported key:
    celestia-appd keys export <KEY_NAME>
    +
    +# you will then be prompted to set a password for the encrypted private key:
    +Enter passphrase to encrypt the exported key:

    After you set a password, your encrypted key will be displayed.

    Querying subcommands

    Usage:

    sh
    celestia-appd query <FLAGS> | <COMMAND>
    +
    +# alias q
    +celestia-appd q <FLAGS> | <COMMAND>
    celestia-appd query <FLAGS> | <COMMAND>
    +
    +# alias q
    +celestia-appd q <FLAGS> | <COMMAND>

    To see all options:

    sh
    celestia-appd q --help
    celestia-appd q --help

    Token management

    Get token balances:

    sh
    celestia-appd q bank balances <ADDRESS> --node <NODE_URI>
    celestia-appd q bank balances <ADDRESS> --node <NODE_URI>

    Example usage:

    sh
    celestia-appd q bank balances celestia1czpgn3hdh9sodm06d5qk23xzgpq2uyc8ggdqgw \\
    +    --node https://rpc-mocha.pops.one
    celestia-appd q bank balances celestia1czpgn3hdh9sodm06d5qk23xzgpq2uyc8ggdqgw \\
    +    --node https://rpc-mocha.pops.one

    Transfer tokens from one wallet to another:

    sh
    celestia-appd tx bank send <FROM_ADDRESS> <TO_ADDRESS> \\
    +    <amount> --node <NODE_URI> --chain-id <CHAIN_ID>
    celestia-appd tx bank send <FROM_ADDRESS> <TO_ADDRESS> \\
    +    <amount> --node <NODE_URI> --chain-id <CHAIN_ID>

    Example usage:

    sh
    celestia-appd tx bank send <FROM_ADDRESS> <TO_ADDRESS> \\
    +    19000000utia --node https://rpc-mocha.pops.one/ --chain-id mocha
    celestia-appd tx bank send <FROM_ADDRESS> <TO_ADDRESS> \\
    +    19000000utia --node https://rpc-mocha.pops.one/ --chain-id mocha

    To see options:

    sh
    celestia-appd tx bank send --help
    celestia-appd tx bank send --help

    Governance

    Governance proposals on Celestia are limited as there are no text proposals, upgrades occur via social consensus, and some params are not modifiable. However, one can submit governance proposals to change certain parameters and spend community funds. More detailed information on this topic can be found in the cosmos-sdk documentation for submitting proposals, the list of parameter defaults in the specs, and the x/paramfilter module specs.

    Viewing the available proposals can be done with the query command:

    sh
    celestia-appd q gov proposals
    celestia-appd q gov proposals

    There are four options when voting "yes", "no", "no_with_veto" and "abstain". The "no_with_veto" vote is different from the "no" vote in that the submitter of the proposer's deposit will get burned, and a minority of stake (1/3) can stop a proposal that might otherwise pass quorum. You can use those options to vote on a governance proposal with the following command:

    sh
    celestia-appd tx gov vote <proposal id> <option> \\
    +    --from <wallet> --chain-id <chain-id>
    celestia-appd tx gov vote <proposal id> <option> \\
    +    --from <wallet> --chain-id <chain-id>

    To submit a proposal, there are two commands that can be used. The first is the legacy command, which is the recommended way to submit a proposal.

    To change the max validators to 105, one would first save this JSON file:

    json
    {
    +  "title": "Staking Param Change",
    +  "description": "Update max validators",
    +  "changes": [
    +    {
    +      "subspace": "staking",
    +      "key": "MaxValidators",
    +      "value": 105
    +    }
    +  ],
    +  "deposit": "1000000000utia"
    +}
    {
    +  "title": "Staking Param Change",
    +  "description": "Update max validators",
    +  "changes": [
    +    {
    +      "subspace": "staking",
    +      "key": "MaxValidators",
    +      "value": 105
    +    }
    +  ],
    +  "deposit": "1000000000utia"
    +}

    Then you can submit the proposal with:

    sh
    celestia-appd tx gov submit-legacy-proposal \\
    +    parameter-change <path to json file> \\
    +    --from <wallet> --chain-id <chain-id>
    celestia-appd tx gov submit-legacy-proposal \\
    +    parameter-change <path to json file> \\
    +    --from <wallet> --chain-id <chain-id>

    If we want to use the newer api, we can submit a proposal by first saving the sdk.Msg proposal in the json encoded format to a json.

    json
    {
    +  "messages": [
    +    {
    +      "@type": "/cosmos.gov.v1beta1.MsgSubmitProposal",
    +      "content": {
    +        "@type": "/cosmos.params.v1beta1.ParameterChangeProposal",
    +        "title": "title",
    +        "description": "description",
    +        "changes": [
    +          { "subspace": "staking", "key": "MaxValidators", "value": "103" }
    +        ]
    +      },
    +      "initial_deposit": [{ "denom": "utia", "amount": "1000000000" }],
    +      "proposer": "celestia10d07y265gmmuvt4z0w9aw880jnsr700jtgz4v7"
    +    }
    +  ]
    +}
    {
    +  "messages": [
    +    {
    +      "@type": "/cosmos.gov.v1beta1.MsgSubmitProposal",
    +      "content": {
    +        "@type": "/cosmos.params.v1beta1.ParameterChangeProposal",
    +        "title": "title",
    +        "description": "description",
    +        "changes": [
    +          { "subspace": "staking", "key": "MaxValidators", "value": "103" }
    +        ]
    +      },
    +      "initial_deposit": [{ "denom": "utia", "amount": "1000000000" }],
    +      "proposer": "celestia10d07y265gmmuvt4z0w9aw880jnsr700jtgz4v7"
    +    }
    +  ]
    +}

    Note that the proposer here must be the gov module account. That account can be found by using this command:

    sh
    celestia-appd q auth module-account gov
    celestia-appd q auth module-account gov

    Then one can submit the proposal with:

    sh
    celestia-appd tx gov submit-proposal <path to json file> \\
    +    --from <wallet> --chain-id <chain-id>
    celestia-appd tx gov submit-proposal <path to json file> \\
    +    --from <wallet> --chain-id <chain-id>

    Community Pool

    A percentage of the block rewards are allocated to the community pool. Community members can submit governance proposals to spend the community pool funds, and token holders can vote on these proposals. The proposals to spend are arbitrary in nature in that they can only contain text and some address to send funds to. To view the community pool balance, use the following command:

    sh
    celestia-appd q distribution community-pool
    celestia-appd q distribution community-pool

    To submit a proposal to spend the community pool funds, first create a JSON file that contains the proposal.

    json
    {
    +  "title": "Community Pool Spend",
    +  "description": "Fund an open source project.",
    +  "recipient": "celestia17adsjkuecgjheugrdrwdqv9uh3qkrfmj9xzawx",
    +  "amount": "100000000000utia",
    +  "deposit": "1000000000utia"
    +}
    {
    +  "title": "Community Pool Spend",
    +  "description": "Fund an open source project.",
    +  "recipient": "celestia17adsjkuecgjheugrdrwdqv9uh3qkrfmj9xzawx",
    +  "amount": "100000000000utia",
    +  "deposit": "1000000000utia"
    +}

    The json file can be submitted using a similar proposal submission command as above:

    sh
    celestia-appd tx gov submit-legacy-proposal \\
    +    community-pool-spend <path to json file> \\
    +    --from <wallet>
    celestia-appd tx gov submit-legacy-proposal \\
    +    community-pool-spend <path to json file> \\
    +    --from <wallet>

    Claim validator rewards

    You can claim your validator rewards with the following command:

    sh
    celestia-appd tx distribution withdraw-rewards <validator valoper> \\
    +    --commission --from=<validator wallet> --chain-id <chain-id> \\
    +    --gas auto -y
    celestia-appd tx distribution withdraw-rewards <validator valoper> \\
    +    --commission --from=<validator wallet> --chain-id <chain-id> \\
    +    --gas auto -y

    Delegate & undelegate tokens

    You can delegate your tokens to a validator with the following command:

    sh
    celestia-appd tx staking delegate <validator valoper> <amount>\\
    +    --from <wallet> --chain-id <chain-id>
    celestia-appd tx staking delegate <validator valoper> <amount>\\
    +    --from <wallet> --chain-id <chain-id>

    You can undelegate tokens to a validator with the unbond command:

    sh
    celestia-appd tx staking unbond <validator valoper> <amount>\\
    +    --from <wallet> --chain-id <chain-id>
    celestia-appd tx staking unbond <validator valoper> <amount>\\
    +    --from <wallet> --chain-id <chain-id>

    Unjailing the validator

    You can unjail your validator with the following command:

    sh
    celestia-appd tx slashing unjail --from <validator wallet>\\
    +    --chain-id <chain-id> --gas auto -y
    celestia-appd tx slashing unjail --from <validator wallet>\\
    +    --chain-id <chain-id> --gas auto -y

    How to export logs with SystemD

    You can export your logs if you are running a SystemD service with the following command:

    sh
    sudo journalctl -u <your systemd service> -S yesterday > node_logs.txt
    +sudo journalctl -u <your systemd service> -S today > node_logs.txt
    +# This command outputs the last 1 million lines!
    +sudo journalctl -u <your systemd service> -n 1000000 > node_logs.txt
    sudo journalctl -u <your systemd service> -S yesterday > node_logs.txt
    +sudo journalctl -u <your systemd service> -S today > node_logs.txt
    +# This command outputs the last 1 million lines!
    +sudo journalctl -u <your systemd service> -n 1000000 > node_logs.txt

    Signing genesis for a new network

    You can first run the following commands:

    sh
    VALIDATOR_NAME=validator1
    +CHAIN_ID=testnet
    +celestia-appd init $VALIDATOR_NAME --chain-id $CHAIN_ID
    +MONIKER=validator_name
    VALIDATOR_NAME=validator1
    +CHAIN_ID=testnet
    +celestia-appd init $VALIDATOR_NAME --chain-id $CHAIN_ID
    +MONIKER=validator_name

    Next create a wallet:

    sh
    KEY_NAME=validator
    +celestia-appd keys add $KEY_NAME
    KEY_NAME=validator
    +celestia-appd keys add $KEY_NAME

    Create or assign an EVM address:

    sh
    EVM_ADDRESS=<EVM_ADDRESS>
    EVM_ADDRESS=<EVM_ADDRESS>

    Then add genesis account:

    sh
    CELES_AMOUNT="5000100000000utia"
    +celestia-appd add-genesis-account $KEY_NAME $CELES_AMOUNT
    CELES_AMOUNT="5000100000000utia"
    +celestia-appd add-genesis-account $KEY_NAME $CELES_AMOUNT

    Then generate your gentx:

    sh
    STAKING_AMOUNT=5000000000000utia
    +celestia-appd gentx $KEY_NAME $STAKING_AMOUNT --chain-id $CHAIN_ID \\
    +    --pubkey=$(celestia-appd tendermint show-validator) \\
    +    --moniker=$MONIKER \\
    +    --commission-rate=0.1 \\
    +    --commission-max-rate=0.2 \\
    +    --commission-max-change-rate=0.01 \\
    +    --min-self-delegation=1 \\
    +    --evm-address=$EVM_ADDRESS \\
    STAKING_AMOUNT=5000000000000utia
    +celestia-appd gentx $KEY_NAME $STAKING_AMOUNT --chain-id $CHAIN_ID \\
    +    --pubkey=$(celestia-appd tendermint show-validator) \\
    +    --moniker=$MONIKER \\
    +    --commission-rate=0.1 \\
    +    --commission-max-rate=0.2 \\
    +    --commission-max-change-rate=0.01 \\
    +    --min-self-delegation=1 \\
    +    --evm-address=$EVM_ADDRESS \\

    You can then share your gentx JSON file on the networks repo in the respective network directory you are participating in.

    `,84),e=[o];function t(c,r,y,E,i,F){return a(),n("div",null,e)}const g=s(p,[["render",t]]);export{u as __pageData,g as default}; diff --git a/assets/nodes_celestia-app-commands.md.63e455f7.lean.js b/assets/nodes_celestia-app-commands.md.63e455f7.lean.js new file mode 100644 index 00000000000..2f367c2f12d --- /dev/null +++ b/assets/nodes_celestia-app-commands.md.63e455f7.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.51d6c45b.js";const u=JSON.parse('{"title":"Helpful CLI commands","description":"Some of the most helpful celestia-app CLI commands.","frontmatter":{"description":"Some of the most helpful celestia-app CLI commands.","next":{"text":"SystemD","link":"/nodes/systemd"},"head":[["meta",{"name":"og:title","content":"Helpful CLI commands | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-commands.md","filePath":"nodes/celestia-app-commands.md","lastUpdated":1702318874000}'),p={name:"nodes/celestia-app-commands.md"},o=l("",84),e=[o];function t(c,r,y,E,i,F){return a(),n("div",null,e)}const g=s(p,[["render",t]]);export{u as __pageData,g as default}; diff --git a/assets/nodes_celestia-app-metrics.md.74ef2ef7.js b/assets/nodes_celestia-app-metrics.md.74ef2ef7.js new file mode 100644 index 00000000000..10fae6c6d97 --- /dev/null +++ b/assets/nodes_celestia-app-metrics.md.74ef2ef7.js @@ -0,0 +1,83 @@ +import{_ as s,o as a,c as n,Q as e}from"./chunks/framework.51d6c45b.js";const m=JSON.parse('{"title":"Metrics","description":"A guide on how to monitor the health and performance of your system.","frontmatter":{"description":"A guide on how to monitor the health and performance of your system.","prev":{"text":"Troubleshooting","link":"/nodes/celestia-node-troubleshooting"},"head":[["meta",{"name":"og:title","content":"Metrics | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-metrics.md","filePath":"nodes/celestia-app-metrics.md","lastUpdated":1707231883000}'),o={name:"nodes/celestia-app-metrics.md"},l=e(`

    Metrics

    Metrics are a powerful tool for monitoring the health and performance of a system. Celestia provides support for metrics to make sure, as an operator, your system continues to remain up and running. Metrics can also provide critical insight into how Celestia is used and how it can be improved.

    Setup

    Celestia uses Prometheus to publish metrics. It can be enabled through the config.toml file.

    bash
    #######################################################
    +###       Instrumentation Configuration Options     ###
    +#######################################################
    +[instrumentation]
    +
    +# When true, Prometheus metrics are served under /metrics on
    +# PrometheusListenAddr.
    +# Check out the documentation for the list of available metrics.
    +prometheus = true
    +
    +# Address to listen for Prometheus collector(s) connections
    +prometheus_listen_addr = ":26660"
    +
    +# Maximum number of simultaneous connections.
    +# If you want to accept a larger number than the default, make sure
    +# you increase your OS limits.
    +# 0 - unlimited.
    +max_open_connections = 3
    +
    +# Instrumentation namespace
    +namespace = "celestia"
    #######################################################
    +###       Instrumentation Configuration Options     ###
    +#######################################################
    +[instrumentation]
    +
    +# When true, Prometheus metrics are served under /metrics on
    +# PrometheusListenAddr.
    +# Check out the documentation for the list of available metrics.
    +prometheus = true
    +
    +# Address to listen for Prometheus collector(s) connections
    +prometheus_listen_addr = ":26660"
    +
    +# Maximum number of simultaneous connections.
    +# If you want to accept a larger number than the default, make sure
    +# you increase your OS limits.
    +# 0 - unlimited.
    +max_open_connections = 3
    +
    +# Instrumentation namespace
    +namespace = "celestia"

    If you restart your node, you can check to see it's working by running:

    bash
    curl localhost:26660/metrics
    curl localhost:26660/metrics

    Visualization

    Now your nodes are publishing metrics, we need something to scrape it and a visualizer to create a dashboard. Commonly, Prometheus is paired with Grafana.

    First, you will need to install Prometheus either from their downloads page or through a package manager like brew.

    Next, create a config file $HOME/.celestia-app/config/prometheus.yml and fill out some basic settings as follows:

    yml
    global:
    +  scrape_interval: 15s # By default, scrape targets every 15 seconds.
    +
    +  # Attach these labels to any time series or alerts when communicating
    +  # with external systems (federation, remote storage, Alertmanager).
    +  external_labels:
    +    monitor: "codelab-monitor"
    +
    +# A scrape configuration containing exactly one endpoint to scrape:
    +# Here it's Prometheus itself.
    +scrape_configs:
    +  # The job name is added as a label \`job=<job_name>\` to any timeseries
    +  # scraped from this config.
    +  - job_name: "prometheus"
    +
    +    # Override the global default and scrape targets from this job every
    +    # 5 seconds.
    +    scrape_interval: 5s
    +
    +    static_configs:
    +      # Point to the same endpoint that Celestia is publishing on
    +      - targets: ["localhost:26660"]
    global:
    +  scrape_interval: 15s # By default, scrape targets every 15 seconds.
    +
    +  # Attach these labels to any time series or alerts when communicating
    +  # with external systems (federation, remote storage, Alertmanager).
    +  external_labels:
    +    monitor: "codelab-monitor"
    +
    +# A scrape configuration containing exactly one endpoint to scrape:
    +# Here it's Prometheus itself.
    +scrape_configs:
    +  # The job name is added as a label \`job=<job_name>\` to any timeseries
    +  # scraped from this config.
    +  - job_name: "prometheus"
    +
    +    # Override the global default and scrape targets from this job every
    +    # 5 seconds.
    +    scrape_interval: 5s
    +
    +    static_configs:
    +      # Point to the same endpoint that Celestia is publishing on
    +      - targets: ["localhost:26660"]

    Note, that Prometheus by default runs its server on :9090. If you are running this on the same machine as your consensus node, it will collide with gRPC which runs on the same port. To avoid this, either switch off gRPC (if it's not needed), change the gRPC port in app.toml, or run Prometheus on a different port e.g. --web.listen-address="0.0.0.0:8000"

    To run the prometheus server:

    bash
    prometheus --config.file="$HOME/.celestia-app/config/prometheus.yml"
    prometheus --config.file="$HOME/.celestia-app/config/prometheus.yml"

    A prometheus server can scrape metrics from multiple nodes at once; a good way of bringing together information from many machines to one place.

    To visualize the information, you can use Grafana: either with their cloud option or run the open source code yourself.

    Once setup, run:

    bash
    grafana server
    grafana server

    which will begin a server on localhost:3000. If you open the url on your browser you will see the Grafana login page. Use admin for both the user and password to log in.

    You will need to link the prometheus server as a data source. To do that go to "Configuration" in the sidebar and then "Data Sources". Add a new data source specifying the URL of the Prometheus instance (default at localhost:9090). Click "Save & test" to confirm.

    Lastly, you will need to setup a dashboard. You can choose to do this yourself, handpicking the metrics that are important or you can simply export an existing design. Fortunately the cosmos ecosystem has conjured a "Cosmos Dashboard". On the sidebar, click "Dashboards" and then "import". Enter the following dashboard ID: 11036 and then link it to the "Prometheus" data source you just set up. Finally click the "Import" button and the "Cosmos Dashboard" should appear.

    Node exporter

    Celestia's metrics include a plethora of application specific trackers but it's also important to keep an eye on system level metrics such as memory usage and disk space. This can be best achieved by running Node Exporter. Follow the guide in the link to get set up, adding the port number to the prometheus.yml file.

    Alerts

    The final cherry on the cake is to integrate your monitoring system with a mechanism for producing alerts to warn you if your node has crashed or is no longer able to stay at the head of the chain.

    Since we're already using Grafana, we can install the Grafana OnCall plugin. OnCall allows you to setup integrations. It could be a webhook or a direct integration into Telegram or Slack. You can find more information on Grafana's Docs Page.

    `,27),t=[l];function p(r,c,i,y,u,h){return a(),n("div",null,t)}const E=s(o,[["render",p]]);export{m as __pageData,E as default}; diff --git a/assets/nodes_celestia-app-metrics.md.74ef2ef7.lean.js b/assets/nodes_celestia-app-metrics.md.74ef2ef7.lean.js new file mode 100644 index 00000000000..c8105670030 --- /dev/null +++ b/assets/nodes_celestia-app-metrics.md.74ef2ef7.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as e}from"./chunks/framework.51d6c45b.js";const m=JSON.parse('{"title":"Metrics","description":"A guide on how to monitor the health and performance of your system.","frontmatter":{"description":"A guide on how to monitor the health and performance of your system.","prev":{"text":"Troubleshooting","link":"/nodes/celestia-node-troubleshooting"},"head":[["meta",{"name":"og:title","content":"Metrics | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-metrics.md","filePath":"nodes/celestia-app-metrics.md","lastUpdated":1707231883000}'),o={name:"nodes/celestia-app-metrics.md"},l=e("",27),t=[l];function p(r,c,i,y,u,h){return a(),n("div",null,t)}const E=s(o,[["render",p]]);export{m as __pageData,E as default}; diff --git a/assets/nodes_celestia-app-multisig.md.e4b4f44c.js b/assets/nodes_celestia-app-multisig.md.e4b4f44c.js new file mode 100644 index 00000000000..423998146c5 --- /dev/null +++ b/assets/nodes_celestia-app-multisig.md.e4b4f44c.js @@ -0,0 +1,103 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.51d6c45b.js";const d=JSON.parse('{"title":"Multisig","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Multisig | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-multisig.md","filePath":"nodes/celestia-app-multisig.md","lastUpdated":1704813470000}'),p={name:"nodes/celestia-app-multisig.md"},o=l(`

    Multisig

    Celestia inherits support for multisig accounts from the Cosmos SDK. Multisig accounts behave similarly to regular accounts with the added requirement that a threshold of signatures is needed to authorize a transaction.

    Multisig accounts can be created from the command line or using a graphical interface such as Keplr.

    Command line

    bash
    #!/bin/sh
    +
    +# Prerequisite: prior to running this script, start a single node devnet with ./scripts/single-node.sh
    +CHAIN_ID="private"
    +KEY_NAME="validator"
    +KEYRING_BACKEND="test"
    +BROADCAST_MODE="block"
    +
    +# Create 3 test keys
    +celestia-appd keys add test1
    +celestia-appd keys add test2
    +celestia-appd keys add test3
    +# Create the multisig account
    +celestia-appd keys add multisig \\
    +    --multisig test1,test2,test3 \\
    +    --multisig-threshold 2
    +
    +# Send some funds from the validator account to the multisig account
    +celestia-appd tx bank send $VALIDATOR $MULTISIG 100000utia \\
    +    --from $VALIDATOR \\
    +    --fees 1000utia \\
    +    --chain-id $CHAIN_ID \\
    +    --keyring-backend $KEYRING_BACKEND \\
    +    --broadcast-mode $BROADCAST_MODE \\
    +    --yes
    +
    +# Send some funds from the multisig account to the validator account.
    +# Note this transaction will need to be signed by at least 2 of the 3 test accounts.
    +celestia-appd tx bank send $MULTISIG $VALIDATOR 1utia \\
    +    --from $MULTISIG \\
    +    --fees 1000utia \\
    +    --chain-id $CHAIN_ID \\
    +    --keyring-backend $KEYRING_BACKEND \\
    +    --generate-only > unsignedTx.json
    +
    +# Sign from test1 and test2
    +celestia-appd tx sign unsignedTx.json \\
    +    --multisig $MULTISIG \\
    +    --from test1 \\
    +    --output-document test1sig.json \\
    +    --chain-id $CHAIN_ID
    +celestia-appd tx sign unsignedTx.json \\
    +    --multisig $MULTISIG \\
    +    --from test2 \\
    +    --output-document test2sig.json \\
    +    --chain-id $CHAIN_ID
    +
    +# Generate the final signed transaction
    +celestia-appd tx multisign unsignedTx.json multisig \\
    +    test1sig.json test2sig.json \\
    +    --output-document signedTx.json \\
    +    --chain-id $CHAIN_ID
    #!/bin/sh
    +
    +# Prerequisite: prior to running this script, start a single node devnet with ./scripts/single-node.sh
    +CHAIN_ID="private"
    +KEY_NAME="validator"
    +KEYRING_BACKEND="test"
    +BROADCAST_MODE="block"
    +
    +# Create 3 test keys
    +celestia-appd keys add test1
    +celestia-appd keys add test2
    +celestia-appd keys add test3
    +# Create the multisig account
    +celestia-appd keys add multisig \\
    +    --multisig test1,test2,test3 \\
    +    --multisig-threshold 2
    +
    +# Send some funds from the validator account to the multisig account
    +celestia-appd tx bank send $VALIDATOR $MULTISIG 100000utia \\
    +    --from $VALIDATOR \\
    +    --fees 1000utia \\
    +    --chain-id $CHAIN_ID \\
    +    --keyring-backend $KEYRING_BACKEND \\
    +    --broadcast-mode $BROADCAST_MODE \\
    +    --yes
    +
    +# Send some funds from the multisig account to the validator account.
    +# Note this transaction will need to be signed by at least 2 of the 3 test accounts.
    +celestia-appd tx bank send $MULTISIG $VALIDATOR 1utia \\
    +    --from $MULTISIG \\
    +    --fees 1000utia \\
    +    --chain-id $CHAIN_ID \\
    +    --keyring-backend $KEYRING_BACKEND \\
    +    --generate-only > unsignedTx.json
    +
    +# Sign from test1 and test2
    +celestia-appd tx sign unsignedTx.json \\
    +    --multisig $MULTISIG \\
    +    --from test1 \\
    +    --output-document test1sig.json \\
    +    --chain-id $CHAIN_ID
    +celestia-appd tx sign unsignedTx.json \\
    +    --multisig $MULTISIG \\
    +    --from test2 \\
    +    --output-document test2sig.json \\
    +    --chain-id $CHAIN_ID
    +
    +# Generate the final signed transaction
    +celestia-appd tx multisign unsignedTx.json multisig \\
    +    test1sig.json test2sig.json \\
    +    --output-document signedTx.json \\
    +    --chain-id $CHAIN_ID

    Resources

    `,7),e=[o];function t(c,r,y,E,i,F){return n(),a("div",null,e)}const u=s(p,[["render",t]]);export{d as __pageData,u as default}; diff --git a/assets/nodes_celestia-app-multisig.md.e4b4f44c.lean.js b/assets/nodes_celestia-app-multisig.md.e4b4f44c.lean.js new file mode 100644 index 00000000000..98e082561d1 --- /dev/null +++ b/assets/nodes_celestia-app-multisig.md.e4b4f44c.lean.js @@ -0,0 +1 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.51d6c45b.js";const d=JSON.parse('{"title":"Multisig","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Multisig | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-multisig.md","filePath":"nodes/celestia-app-multisig.md","lastUpdated":1704813470000}'),p={name:"nodes/celestia-app-multisig.md"},o=l("",7),e=[o];function t(c,r,y,E,i,F){return n(),a("div",null,e)}const u=s(p,[["render",t]]);export{d as __pageData,u as default}; diff --git a/assets/nodes_celestia-app-slashing.md.4d50cd08.js b/assets/nodes_celestia-app-slashing.md.4d50cd08.js new file mode 100644 index 00000000000..97e5e65dba3 --- /dev/null +++ b/assets/nodes_celestia-app-slashing.md.4d50cd08.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as o}from"./chunks/framework.51d6c45b.js";const f=JSON.parse('{"title":"Slashing on Celestia","description":"This section covers the slashing mechanics for validators in Celestia.","frontmatter":{"description":"This section covers the slashing mechanics for validators in Celestia.","head":[["meta",{"name":"og:title","content":"Slashing on Celestia | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-slashing.md","filePath":"nodes/celestia-app-slashing.md","lastUpdated":1706730069000}'),s={name:"nodes/celestia-app-slashing.md"},i=o('

    Slashing on Celestia

    Slashing is a mechanism employed in proof of stake blockchains that is used to deter and punish malicious behavior. It functions by removing a percentage of a validator's stake each time they act harmfully towards the network.

    Celestia is built with the Cosmos SDK and uses the x/slashing module.

    If a validator gets slashed, delegators bonded to that validator will also have the same percentage of their delegated funds slashed.

    The following are the conditions for a validator to get slashed:

    1. Downtime: If a validator is offline for more than 25% of a rolling window of the last 5,000 blocks, they will be jailed for 1 minute. During this period, the validator is removed from the validator set temporarily, and will be unable to propose new blocks or earn rewards. After the jail period, the validator can send an unjail request to rejoin the validator set.

    2. Double signing: This is a more severe offense and results in a larger percentage loss. If a validator engages in double signing, the validator will lose 2% of their stake and the remainder of their stake will be returned to them. The validator will be permanently removed from the validator set and will not have the ability to unjail. Delegators bonded to that validator will automatically enter the unbonding period for 21 days, and can delegate to another validator after they have been unbonded.

    For more details on the slashing parameters, refer to the celestia-app specifications page.

    ',7),l=[i];function n(r,d,h,c,p,g){return a(),t("div",null,l)}const u=e(s,[["render",n]]);export{f as __pageData,u as default}; diff --git a/assets/nodes_celestia-app-slashing.md.4d50cd08.lean.js b/assets/nodes_celestia-app-slashing.md.4d50cd08.lean.js new file mode 100644 index 00000000000..b07022fc383 --- /dev/null +++ b/assets/nodes_celestia-app-slashing.md.4d50cd08.lean.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as o}from"./chunks/framework.51d6c45b.js";const f=JSON.parse('{"title":"Slashing on Celestia","description":"This section covers the slashing mechanics for validators in Celestia.","frontmatter":{"description":"This section covers the slashing mechanics for validators in Celestia.","head":[["meta",{"name":"og:title","content":"Slashing on Celestia | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-slashing.md","filePath":"nodes/celestia-app-slashing.md","lastUpdated":1706730069000}'),s={name:"nodes/celestia-app-slashing.md"},i=o("",7),l=[i];function n(r,d,h,c,p,g){return a(),t("div",null,l)}const u=e(s,[["render",n]]);export{f as __pageData,u as default}; diff --git a/assets/nodes_celestia-app-upgrade-monitor.md.91d63439.js b/assets/nodes_celestia-app-upgrade-monitor.md.91d63439.js new file mode 100644 index 00000000000..07ee06d6abd --- /dev/null +++ b/assets/nodes_celestia-app-upgrade-monitor.md.91d63439.js @@ -0,0 +1 @@ +import{_ as o,o as a,c as r,k as e,a as t}from"./chunks/framework.51d6c45b.js";const f=JSON.parse('{"title":"Upgrade Monitor","description":"upgrade-monitor is a tool to monitor upgrades on a Celestia network.","frontmatter":{"description":"upgrade-monitor is a tool to monitor upgrades on a Celestia network.","head":[["meta",{"name":"og:title","content":"Upgrade Monitor | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-upgrade-monitor.md","filePath":"nodes/celestia-app-upgrade-monitor.md","lastUpdated":1702584616000}'),n={name:"nodes/celestia-app-upgrade-monitor.md"},s=e("h1",{id:"upgrade-monitor",tabindex:"-1"},[t("Upgrade Monitor "),e("a",{class:"header-anchor",href:"#upgrade-monitor","aria-label":'Permalink to "Upgrade Monitor"'},"​")],-1),i=e("p",null,[t("Upgrade monitor is a binary that monitors that status of upgrades on a Celestia network. See the "),e("a",{href:"https://github.com/celestiaorg/upgrade-monitor",target:"_blank",rel:"noreferrer"},"README"),t(" for instructions on how to install and use the binary.")],-1),d=[s,i];function p(c,l,m,g,u,h){return a(),r("div",null,d)}const k=o(n,[["render",p]]);export{f as __pageData,k as default}; diff --git a/assets/nodes_celestia-app-upgrade-monitor.md.91d63439.lean.js b/assets/nodes_celestia-app-upgrade-monitor.md.91d63439.lean.js new file mode 100644 index 00000000000..07ee06d6abd --- /dev/null +++ b/assets/nodes_celestia-app-upgrade-monitor.md.91d63439.lean.js @@ -0,0 +1 @@ +import{_ as o,o as a,c as r,k as e,a as t}from"./chunks/framework.51d6c45b.js";const f=JSON.parse('{"title":"Upgrade Monitor","description":"upgrade-monitor is a tool to monitor upgrades on a Celestia network.","frontmatter":{"description":"upgrade-monitor is a tool to monitor upgrades on a Celestia network.","head":[["meta",{"name":"og:title","content":"Upgrade Monitor | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-upgrade-monitor.md","filePath":"nodes/celestia-app-upgrade-monitor.md","lastUpdated":1702584616000}'),n={name:"nodes/celestia-app-upgrade-monitor.md"},s=e("h1",{id:"upgrade-monitor",tabindex:"-1"},[t("Upgrade Monitor "),e("a",{class:"header-anchor",href:"#upgrade-monitor","aria-label":'Permalink to "Upgrade Monitor"'},"​")],-1),i=e("p",null,[t("Upgrade monitor is a binary that monitors that status of upgrades on a Celestia network. See the "),e("a",{href:"https://github.com/celestiaorg/upgrade-monitor",target:"_blank",rel:"noreferrer"},"README"),t(" for instructions on how to install and use the binary.")],-1),d=[s,i];function p(c,l,m,g,u,h){return a(),r("div",null,d)}const k=o(n,[["render",p]]);export{f as __pageData,k as default}; diff --git a/assets/nodes_celestia-app-vesting.md.0867750b.js b/assets/nodes_celestia-app-vesting.md.0867750b.js new file mode 100644 index 00000000000..087777e863f --- /dev/null +++ b/assets/nodes_celestia-app-vesting.md.0867750b.js @@ -0,0 +1,215 @@ +import{_ as s,o as a,c as n,Q as o}from"./chunks/framework.51d6c45b.js";const u=JSON.parse('{"title":"How to create a vesting account with celestia-app","description":"Learn how to generate a vesting account using celestia-app.","frontmatter":{"description":"Learn how to generate a vesting account using celestia-app.","head":[["meta",{"name":"og:title","content":"How to create a vesting account with celestia-app | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-vesting.md","filePath":"nodes/celestia-app-vesting.md","lastUpdated":1711744858000}'),l={name:"nodes/celestia-app-vesting.md"},p=o(`

    How to create a vesting account with celestia-app

    In this guide, we will learn how to create a vesting account using celestia-app for both a local devnet and on Mocha testnet.

    note

    The instructions for this tutorial are for a continuous vesting account, if you'd like to make a delayed vesting account, just add the --delayed flag to your vesting transaction.

    Local devnet

    First, download and install celestia-app, selecting the network and corresponding version that you would like to use.

    Setting up the local devnet

    Run the devnet

    Next, change into the $HOME/celestia-app directory and run the single-node-devnet script.

    bash
    cd $HOME/celestia-app
    +./scripts/build-run-single-node.sh
    cd $HOME/celestia-app
    +./scripts/build-run-single-node.sh

    Save the home directory path

    At the top of the output, you will see a path to the "Home directory", find yours from the output (it will be unique every time):

    bash
    ./scripts/build-run-single-node.sh
    +Home directory: /var/folders/_8/ljj6hspn0kn09qf9fy8kdyh40000gn/T/celestia_app_XXXXXXXXXXXXX.XV92a3qx
    +--> Updating go.mod
    ./scripts/build-run-single-node.sh
    +Home directory: /var/folders/_8/ljj6hspn0kn09qf9fy8kdyh40000gn/T/celestia_app_XXXXXXXXXXXXX.XV92a3qx
    +--> Updating go.mod

    And set the location as the CElESTIA_APP_HOME variable. We will use this for the remainder of the devnet section.

    bash
    export CElESTIA_APP_HOME=/var/folders/_8/ljj6hspn0kn09qf9fy8kdyh40000gn/T/celestia_app_XXXXXXXXXXXXX.XV92a3qx
    export CElESTIA_APP_HOME=/var/folders/_8/ljj6hspn0kn09qf9fy8kdyh40000gn/T/celestia_app_XXXXXXXXXXXXX.XV92a3qx

    note

    This does not replace the celestia-appd binary that was installed with celestia-appd, but builds and runs one in the $HOME/celestia-app/build directory.

    Check the version of the devnet

    If you'd like to check the version of your local devnet, you can use:

    bash
    cd $HOME/celestia-app/build
    +./celestia-appd version
    cd $HOME/celestia-app/build
    +./celestia-appd version

    Next steps

    Congratulations! You now have a private devnet running locally on your machine. The devnet is made up of one validator that is creating new blocks. This is the Celestia consensus network on your machine! The key that was created to run the validator also lives in a temporary directory for the devnet.

    Now you are ready to test creating a vesting account on our devnet before going to Mocha, a live testnet.

    Setting up vesting account on devnet

    You already have one key setup, but you will need one more to create a vesting account.

    Create a new key

    First, create a vesting key:

    bash
    cd $HOME/celestia-app/build
    +./celestia-appd keys add vesting-key --home $CElESTIA_APP_HOME
    cd $HOME/celestia-app/build
    +./celestia-appd keys add vesting-key --home $CElESTIA_APP_HOME

    You will see the address, mnemonic, and more details about your key in the output:

    bash
    - address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +  name: vesting-key
    +  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"A5JF/we+s5gFt6g944XbKVVYgQB9OY+U/l5dhZjLDczO"}'
    +  type: local
    +
    +
    +**Important** write this mnemonic phrase in a safe place.
    +It is the only way to recover your account if you ever forget your password.
    +
    +index enter egg broken ostrich duty bitter blind all car hollow coral youth early verify point void anger daring sausage decline net shove oil
    - address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +  name: vesting-key
    +  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"A5JF/we+s5gFt6g944XbKVVYgQB9OY+U/l5dhZjLDczO"}'
    +  type: local
    +
    +
    +**Important** write this mnemonic phrase in a safe place.
    +It is the only way to recover your account if you ever forget your password.
    +
    +index enter egg broken ostrich duty bitter blind all car hollow coral youth early verify point void anger daring sausage decline net shove oil

    List your keys

    bash
    ./celestia-appd keys list --home $CElESTIA_APP_HOME
    ./celestia-appd keys list --home $CElESTIA_APP_HOME

    Output:

    bash
    - address: celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +  name: validator
    +  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"Ahzu6yr9XMPIxLquhgBhj9xL3wIaOz6PE3CvML/oPQym"}'
    +  type: local
    +- address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +  name: vesting-key
    +  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"A5JF/we+s5gFt6g944XbKVVYgQB9OY+U/l5dhZjLDczO"}'
    +  type: local
    - address: celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +  name: validator
    +  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"Ahzu6yr9XMPIxLquhgBhj9xL3wIaOz6PE3CvML/oPQym"}'
    +  type: local
    +- address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +  name: vesting-key
    +  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"A5JF/we+s5gFt6g944XbKVVYgQB9OY+U/l5dhZjLDczO"}'
    +  type: local

    Set variables

    Set the keys as variables, using the validator address as the FROM_ADDRESS and the vesting-key as the TO_ADDRESS.

    bash
    export FROM_ADDRESS=celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +export TO_ADDRESS=celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    export FROM_ADDRESS=celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +export TO_ADDRESS=celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5

    Create your devnet vesting account

    Create the vesting account with the following command:

    note

    The remainder of the instructions are for a continuous vesting account, if you'd like to make a delayed vesting account, use the --delayed flag.

    For example, the command to create a delayed vesting account would look like:

    bash
    ./celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas auto --fees 100000utia --chain-id private --home $CElESTIA_APP_HOME --delayed
    ./celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas auto --fees 100000utia --chain-id private --home $CElESTIA_APP_HOME --delayed
    bash
    ./celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas auto --fees 100000utia --chain-id private --home $CElESTIA_APP_HOME
    ./celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas auto --fees 100000utia --chain-id private --home $CElESTIA_APP_HOME

    Select "Y" to choose "yes".

    Optional

    If you'd like to run the command with the -y flag, it will execute the transaction without needing to provide the "y" answer as above.

    bash
    ./celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas auto --fees 100000utia --chain-id private --home $CElESTIA_APP_HOME -y
    ./celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas auto --fees 100000utia --chain-id private --home $CElESTIA_APP_HOME -y

    Output:

    bash
    gas estimate: 96112
    +auth_info:
    +  fee:
    +    amount:
    +    - amount: "100000"
    +      denom: utia
    +    gas_limit: "96112"
    +    granter: ""
    +    payer: ""
    +  signer_infos: []
    +  tip: null
    +body:
    +  extension_options: []
    +  memo: ""
    +  messages:
    +  - '@type': /cosmos.vesting.v1beta1.MsgCreateVestingAccount
    +    amount:
    +    - amount: "100000"
    +      denom: utia
    +    delayed: false
    +    end_time: "1686748051"
    +    from_address: celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +    to_address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +  non_critical_extension_options: []
    +  timeout_height: "0"
    +signatures: []
    +confirm transaction before signing and broadcasting [y/N]: y
    +code: 0
    +codespace: ""
    +data: ""
    +events: []
    +gas_used: "0"
    +gas_wanted: "0"
    +height: "0"
    +info: ""
    +logs: []
    +raw_log: '[]'
    +timestamp: ""
    +tx: null
    +txhash: 6093DF76DBA90F04FF63D197FC1569F04ED3DBE64081A0BBA9BAD4E69AA570D2
    gas estimate: 96112
    +auth_info:
    +  fee:
    +    amount:
    +    - amount: "100000"
    +      denom: utia
    +    gas_limit: "96112"
    +    granter: ""
    +    payer: ""
    +  signer_infos: []
    +  tip: null
    +body:
    +  extension_options: []
    +  memo: ""
    +  messages:
    +  - '@type': /cosmos.vesting.v1beta1.MsgCreateVestingAccount
    +    amount:
    +    - amount: "100000"
    +      denom: utia
    +    delayed: false
    +    end_time: "1686748051"
    +    from_address: celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +    to_address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +  non_critical_extension_options: []
    +  timeout_height: "0"
    +signatures: []
    +confirm transaction before signing and broadcasting [y/N]: y
    +code: 0
    +codespace: ""
    +data: ""
    +events: []
    +gas_used: "0"
    +gas_wanted: "0"
    +height: "0"
    +info: ""
    +logs: []
    +raw_log: '[]'
    +timestamp: ""
    +tx: null
    +txhash: 6093DF76DBA90F04FF63D197FC1569F04ED3DBE64081A0BBA9BAD4E69AA570D2

    The timestamp for the previous command is in the past, so once you create the vesting account, the tokens will vest. You can check your account balances to verify this.

    Query the devnet vesting account details

    Check that the account has been created and works as expected by querying the TO_ADDRESS account details:

    bash
    ./celestia-appd query account $TO_ADDRESS --home $CElESTIA_APP_HOME
    ./celestia-appd query account $TO_ADDRESS --home $CElESTIA_APP_HOME

    In the output, you will notice that the account type is a ContinuousVestingAccount:

    bash
    '@type': /cosmos.vesting.v1beta1.ContinuousVestingAccount
    +base_vesting_account:
    +  base_account:
    +    account_number: "7"
    +    address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +    pub_key: null
    +    sequence: "0"
    +  delegated_free: []
    +  delegated_vesting: []
    +  end_time: "1686748051"
    +  original_vesting:
    +  - amount: "100000"
    +    denom: utia
    +start_time: "1687908352"
    '@type': /cosmos.vesting.v1beta1.ContinuousVestingAccount
    +base_vesting_account:
    +  base_account:
    +    account_number: "7"
    +    address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +    pub_key: null
    +    sequence: "0"
    +  delegated_free: []
    +  delegated_vesting: []
    +  end_time: "1686748051"
    +  original_vesting:
    +  - amount: "100000"
    +    denom: utia
    +start_time: "1687908352"

    Query the devnet base account details

    Check the FROM_ADDRESS account details:

    bash
    ./celestia-appd query account $FROM_ADDRESS --home $CElESTIA_APP_HOME
    ./celestia-appd query account $FROM_ADDRESS --home $CElESTIA_APP_HOME

    In the output, you will notice the account type is BaseAccount:

    bash
    '@type': /cosmos.auth.v1beta1.BaseAccount
    +account_number: "0"
    +address: celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +pub_key:
    +  '@type': /cosmos.crypto.secp256k1.PubKey
    +  key: Ahzu6yr9XMPIxLquhgBhj9xL3wIaOz6PE3CvML/oPQym
    +sequence: "2"
    '@type': /cosmos.auth.v1beta1.BaseAccount
    +account_number: "0"
    +address: celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +pub_key:
    +  '@type': /cosmos.crypto.secp256k1.PubKey
    +  key: Ahzu6yr9XMPIxLquhgBhj9xL3wIaOz6PE3CvML/oPQym
    +sequence: "2"

    Query the balances of the devnet accounts

    Next, we can check the balance of the accounts:

    bash
    ./celestia-appd query bank balances $TO_ADDRESS --home $CElESTIA_APP_HOME
    ./celestia-appd query bank balances $TO_ADDRESS --home $CElESTIA_APP_HOME

    Output will show you the balance of the vesting account:

    bash
    balances:
    +- amount: "100000"
    +  denom: utia
    +pagination:
    +  next_key: null
    +  total: "0"
    balances:
    +- amount: "100000"
    +  denom: utia
    +pagination:
    +  next_key: null
    +  total: "0"
    bash
    ./celestia-appd query bank balances $FROM_ADDRESS --home $CElESTIA_APP_HOME
    ./celestia-appd query bank balances $FROM_ADDRESS --home $CElESTIA_APP_HOME

    The output will show the remaining balance of the validator:

    bash
    balances:
    +- amount: "999994999800000"
    +  denom: utia
    +pagination:
    +  next_key: null
    +  total: "0"
    balances:
    +- amount: "999994999800000"
    +  denom: utia
    +pagination:
    +  next_key: null
    +  total: "0"

    Congratulations! You've now made your own vesting account on a local devnet. Next, you can learn how to create a vesting account on Mocha testnet.

    Mocha

    In the previous section of this tutorial, we learned how to create a vesting account on a local devnet. In this portion of the tutorial, we'll cover how to set up a full consensus node and set up a vesting account on Mocha testnet.

    First, be sure that you have installed celestia-app for the latest version for Mocha testnet.

    Create a wallet

    Set the keyring backend, so you don't need to use the flag for every command:

    bash
    celestia-appd config keyring-backend test
    celestia-appd config keyring-backend test

    Add a new key for a full node and one for a vesting account:

    bash
    celestia-appd keys add origin && celestia-appd keys add vesting
    celestia-appd keys add origin && celestia-appd keys add vesting

    List the keys:

    bash
    celestia-appd keys list
    celestia-appd keys list

    Set your keys as variables:

    bash
    export FROM_ADDRESS=address_of_origin_account
    +export TO_ADDRESS=address_of_vesting_account
    export FROM_ADDRESS=address_of_origin_account
    +export TO_ADDRESS=address_of_vesting_account

    Fund your account

    Head to the faucet, and fund your origin address.

    Create a vesting account on Mocha

    To create a vesting account on Mocha, you will need an RPC URL to send the transaction to. You can find the RPC endpoints on the Mocha testnet page.

    Set your RPC URL:

    bash
    export RPC_URL=https://rpc-mocha.pops.one:443
    export RPC_URL=https://rpc-mocha.pops.one:443

    We will use a few flags in our vesting command that are different than the devnet version. Since we aren't using our own validator or full node, we will use an RPC URL.

    We also need to declare the chain ID as mocha.

    View the help menu for vesting to understand these flags more:

    bash
    celestia-appd tx vesting --help
    celestia-appd tx vesting --help

    Here's an example command to set up the vesting account:

    bash
    celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas 100000 --fees 100000utia --node $RPC_URL --chain-id mocha --delayed
    celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas 100000 --fees 100000utia --node $RPC_URL --chain-id mocha --delayed

    Optional: Set up a full consensus node or validator

    Running a full consensus node or validator will prevent you from needing to use an RPC.

    You can set up a validator or full consensus node for the previous portion of the tutorial.

    Note: this may take some time depending on how you choose to sync the state of the chain.

    Optional: Change your client.toml

    If you edit your client configuration in client.toml, you can set both the chain ID and the node RPC URL. This will prevent you from needing to run each flag for every command line that you use.

    toml
    # This is a TOML config file.
    +# For more information, see https://github.com/toml-lang/toml
    +
    +###############################################################################
    +###                           Client Configuration                            ###
    +###############################################################################
    +
    +# The network chain ID
    +chain-id = "mocha"
    +# The keyring's backend, where the keys are stored (os|file|kwallet|pass|test|memory)
    +keyring-backend = "test"
    +# CLI output format (text|json)
    +output = "text"
    +# <host>:<port> to Tendermint RPC interface for this chain
    +node = "tcp://rpc-mocha.pops.one:443"
    +# Transaction broadcasting mode (sync|async|block)
    +broadcast-mode = "sync"
    # This is a TOML config file.
    +# For more information, see https://github.com/toml-lang/toml
    +
    +###############################################################################
    +###                           Client Configuration                            ###
    +###############################################################################
    +
    +# The network chain ID
    +chain-id = "mocha"
    +# The keyring's backend, where the keys are stored (os|file|kwallet|pass|test|memory)
    +keyring-backend = "test"
    +# CLI output format (text|json)
    +output = "text"
    +# <host>:<port> to Tendermint RPC interface for this chain
    +node = "tcp://rpc-mocha.pops.one:443"
    +# Transaction broadcasting mode (sync|async|block)
    +broadcast-mode = "sync"

    Notes

    Not all vesting accounts can be created with a message, some need to be set at genesis. You can learn more in the Cosmos Network documentation.

    Conclusion

    Congratulations! You've learned how to create a local devnet, create a vesting account on it, and how to make a vesting account on the Mocha testnet!

    `,98),e=[p];function t(c,r,y,E,i,F){return a(),n("div",null,e)}const h=s(l,[["render",t]]);export{u as __pageData,h as default}; diff --git a/assets/nodes_celestia-app-vesting.md.0867750b.lean.js b/assets/nodes_celestia-app-vesting.md.0867750b.lean.js new file mode 100644 index 00000000000..6103fddea4a --- /dev/null +++ b/assets/nodes_celestia-app-vesting.md.0867750b.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as o}from"./chunks/framework.51d6c45b.js";const u=JSON.parse('{"title":"How to create a vesting account with celestia-app","description":"Learn how to generate a vesting account using celestia-app.","frontmatter":{"description":"Learn how to generate a vesting account using celestia-app.","head":[["meta",{"name":"og:title","content":"How to create a vesting account with celestia-app | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-vesting.md","filePath":"nodes/celestia-app-vesting.md","lastUpdated":1711744858000}'),l={name:"nodes/celestia-app-vesting.md"},p=o("",98),e=[p];function t(c,r,y,E,i,F){return a(),n("div",null,e)}const h=s(l,[["render",t]]);export{u as __pageData,h as default}; diff --git a/assets/nodes_celestia-app-wallet.md.e787539c.js b/assets/nodes_celestia-app-wallet.md.e787539c.js new file mode 100644 index 00000000000..7dfcdb1f00d --- /dev/null +++ b/assets/nodes_celestia-app-wallet.md.e787539c.js @@ -0,0 +1,3 @@ +import{_ as a,o as e,c as s,Q as t}from"./chunks/framework.51d6c45b.js";const u=JSON.parse('{"title":"Create a wallet with celestia-app","description":"Learn how to generate a Celestia wallet using celestia-app.","frontmatter":{"description":"Learn how to generate a Celestia wallet using celestia-app.","head":[["meta",{"name":"og:title","content":"Create a wallet with celestia-app | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-wallet.md","filePath":"nodes/celestia-app-wallet.md","lastUpdated":1707231883000}'),l={name:"nodes/celestia-app-wallet.md"},o=t(`

    Create a wallet with celestia-app

    For this guide, we will go over how you can generate a Celestia wallet using celestia-app.

    Prerequisites

    Note, you do not need to install celestia-node for this tutorial.

    Create a wallet

    First, create an application CLI configuration file:

    sh
    celestia-appd config keyring-backend test
    celestia-appd config keyring-backend test

    You can pick whatever wallet name you want. For our example we used "validator" as the wallet name:

    sh
    celestia-appd keys add validator --interactive
    celestia-appd keys add validator --interactive

    Save the mnemonic output as this is the only way to recover your validator wallet in case you lose it!

    To check all your wallets you can run:

    sh
    celestia-appd keys list
    celestia-appd keys list

    Fund a wallet

    For the public celestia address, you can fund the previously created wallet via Discord by sending this message to either the #mocha-faucet or #arabica-faucet channel:

    text
    $request celestia1xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
    $request celestia1xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx

    Wait to see if you get a confirmation that the tokens have been successfully sent. To check if tokens have arrived successfully to the destination wallet run the command below replacing the public address with your own:

    sh
    celestia-appd start
    +celestia-appd query bank balances celestia1xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
    celestia-appd start
    +celestia-appd query bank balances celestia1xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx

    TIP

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    `,19),n=[o];function p(c,r,i,x,d,h){return e(),s("div",null,n)}const E=a(l,[["render",p]]);export{u as __pageData,E as default}; diff --git a/assets/nodes_celestia-app-wallet.md.e787539c.lean.js b/assets/nodes_celestia-app-wallet.md.e787539c.lean.js new file mode 100644 index 00000000000..391d7cfd0db --- /dev/null +++ b/assets/nodes_celestia-app-wallet.md.e787539c.lean.js @@ -0,0 +1 @@ +import{_ as a,o as e,c as s,Q as t}from"./chunks/framework.51d6c45b.js";const u=JSON.parse('{"title":"Create a wallet with celestia-app","description":"Learn how to generate a Celestia wallet using celestia-app.","frontmatter":{"description":"Learn how to generate a Celestia wallet using celestia-app.","head":[["meta",{"name":"og:title","content":"Create a wallet with celestia-app | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-wallet.md","filePath":"nodes/celestia-app-wallet.md","lastUpdated":1707231883000}'),l={name:"nodes/celestia-app-wallet.md"},o=t("",19),n=[o];function p(c,r,i,x,d,h){return e(),s("div",null,n)}const E=a(l,[["render",p]]);export{u as __pageData,E as default}; diff --git a/assets/nodes_celestia-app.md.1b04d1e9.js b/assets/nodes_celestia-app.md.1b04d1e9.js new file mode 100644 index 00000000000..40e923f07f3 --- /dev/null +++ b/assets/nodes_celestia-app.md.1b04d1e9.js @@ -0,0 +1,7 @@ +import{a}from"./chunks/arabica_versions.d7daf483.js";import{m as l}from"./chunks/mocha_versions.3ba4a823.js";import{m as n}from"./chunks/mainnet_versions.84d9fd1c.js";import{o as c,c as p,k as s,t,l as e,Q as o}from"./chunks/framework.51d6c45b.js";const i=o('

    Install celestia-app

    This tutorial will guide you through installing celestia-app, both from source and with a pre-built binary

    Building binary from source

    This section of the tutorial presumes you completed the steps in setting up your own environment.

    The steps below will create a binary file named celestia-appd inside $HOME/go/bin folder which will be used later to run the node. Be sure to select the correct network to install the binary for.

    ',5),r=o(`
  • Remove any existing copy of celestia-app, clone the repository, and change into the directory:

    bash
    cd $HOME
    +rm -rf celestia-app
    +git clone https://github.com/celestiaorg/celestia-app.git
    +cd celestia-app
    cd $HOME
    +rm -rf celestia-app
    +git clone https://github.com/celestiaorg/celestia-app.git
    +cd celestia-app
  • `,1),d=s("p",null,"Check out to the desired version, based on the network you will use:",-1),h={class:"vp-code-group vp-adaptive-theme"},_=o('
    ',1),y={class:"blocks"},u={class:"language-bash vp-adaptive-theme active"},b=s("button",{title:"Copy Code",class:"copy"},null,-1),g=s("span",{class:"lang"},"bash",-1),E={class:"shiki github-dark vp-code-dark"},F={class:"line"},m=s("span",{style:{color:"#B392F0"}},"git",-1),C=s("span",{style:{color:"#E1E4E8"}}," ",-1),v=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),f=s("span",{style:{color:"#E1E4E8"}}," ",-1),k={style:{color:"#9ECBFF"}},B=s("span",{style:{color:"#E1E4E8"}}," ",-1),T=s("span",{style:{color:"#79B8FF"}},"-b",-1),w=s("span",{style:{color:"#E1E4E8"}}," ",-1),P={style:{color:"#9ECBFF"}},I={class:"shiki github-light vp-code-light"},A={class:"line"},S=s("span",{style:{color:"#6F42C1"}},"git",-1),V=s("span",{style:{color:"#24292E"}}," ",-1),q=s("span",{style:{color:"#032F62"}},"checkout",-1),x=s("span",{style:{color:"#24292E"}}," ",-1),D={style:{color:"#032F62"}},L=s("span",{style:{color:"#24292E"}}," ",-1),M=s("span",{style:{color:"#005CC5"}},"-b",-1),N=s("span",{style:{color:"#24292E"}}," ",-1),R={style:{color:"#032F62"}},O={class:"language-bash vp-adaptive-theme"},$=s("button",{title:"Copy Code",class:"copy"},null,-1),H=s("span",{class:"lang"},"bash",-1),Q={class:"shiki github-dark vp-code-dark"},Y={class:"line"},K=s("span",{style:{color:"#B392F0"}},"git",-1),U=s("span",{style:{color:"#E1E4E8"}}," ",-1),j=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),J=s("span",{style:{color:"#E1E4E8"}}," ",-1),W={style:{color:"#9ECBFF"}},z=s("span",{style:{color:"#E1E4E8"}}," ",-1),G=s("span",{style:{color:"#79B8FF"}},"-b",-1),X=s("span",{style:{color:"#E1E4E8"}}," ",-1),Z={style:{color:"#9ECBFF"}},ss={class:"shiki github-light vp-code-light"},ts={class:"line"},es=s("span",{style:{color:"#6F42C1"}},"git",-1),os=s("span",{style:{color:"#24292E"}}," ",-1),as=s("span",{style:{color:"#032F62"}},"checkout",-1),ls=s("span",{style:{color:"#24292E"}}," ",-1),ns={style:{color:"#032F62"}},cs=s("span",{style:{color:"#24292E"}}," ",-1),ps=s("span",{style:{color:"#005CC5"}},"-b",-1),is=s("span",{style:{color:"#24292E"}}," ",-1),rs={style:{color:"#032F62"}},ds={class:"language-bash vp-adaptive-theme"},hs=s("button",{title:"Copy Code",class:"copy"},null,-1),_s=s("span",{class:"lang"},"bash",-1),ys={class:"shiki github-dark vp-code-dark"},us={class:"line"},bs=s("span",{style:{color:"#B392F0"}},"git",-1),gs=s("span",{style:{color:"#E1E4E8"}}," ",-1),Es=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),Fs=s("span",{style:{color:"#E1E4E8"}}," ",-1),ms={style:{color:"#9ECBFF"}},Cs=s("span",{style:{color:"#E1E4E8"}}," ",-1),vs=s("span",{style:{color:"#79B8FF"}},"-b",-1),fs=s("span",{style:{color:"#E1E4E8"}}," ",-1),ks={style:{color:"#9ECBFF"}},Bs={class:"shiki github-light vp-code-light"},Ts={class:"line"},ws=s("span",{style:{color:"#6F42C1"}},"git",-1),Ps=s("span",{style:{color:"#24292E"}}," ",-1),Is=s("span",{style:{color:"#032F62"}},"checkout",-1),As=s("span",{style:{color:"#24292E"}}," ",-1),Ss={style:{color:"#032F62"}},Vs=s("span",{style:{color:"#24292E"}}," ",-1),qs=s("span",{style:{color:"#005CC5"}},"-b",-1),xs=s("span",{style:{color:"#24292E"}}," ",-1),Ds={style:{color:"#032F62"}},Ls=o('
  • Build and install the celestia-appd binary:

    bash
    make install
    make install
  • To check if the binary was successfully installed you can run the binary using the --help flag:

    sh
    celestia-appd --help
    celestia-appd --help
  • ',2),Ms=o('

    You will see an output with the menu for celestia-appd. Learn more on the helpful CLI commands page

    Installing a pre-built binary

    Installing a pre-built binary is the fastest way to get started with your Celestia consensus node. Releases after celestia-app v1.3.0 should have these binaries available.

    The steps below will download a binary file named celestia-appd. Depending on the setup that you choose during installation, the celestia-appd binary will be available at either:

    • $HOME/celestia-app-temp/celestia-appd
    • /usr/local/bin/celestia-appd

    Pre-built binaries are available for:

    • Operating systems: Darwin (Apple), Linux
    • Architectures: x86_64 (amd64), arm64

    To install the latest pre-built binary you can run this command in your terminal:

    bash
    bash -c "$(curl -sL https://docs.celestia.org/celestia-app.sh)"
    bash -c "$(curl -sL https://docs.celestia.org/celestia-app.sh)"

    Follow the instructions in the terminal output to choose your installation preferences.

    You will see an output with the menu for celestia-appd. Learn more on the helpful CLI commands page

    View the script to learn more about what it is doing.

    Ports

    When interacting with a consensus node, you may need to open ports on your machine to allow communication between nodes, such as bridge nodes. It is essential that specific ports are accessible. Make sure that your firewall allows connections to the correct ports.

    If you run a node on a cloud server, make sure that the ports are open on the server's firewall. If you run a node at home, make sure that your router allows connections to the correct ports.

    For example, validator ports 9090 and 26657 need to be accessible by the bridge, and port 2121 is required for P2P connections for all node types.

    The following ports are used by Celestia app nodes:

    PortProtocolAddressDescriptionEnabled by default on nodeFlag
    2121TCP/UDPlocalhostP2PtrueN/A
    9090HTTP0.0.0.0gRPCtrue--grpc.address string
    26657TCPlocalhostRPCfalse (only open to localhost)--rpc.laddr string
    ',18),Us=JSON.parse('{"title":"Install celestia-app","description":"Learn how you can build celestia-app.","frontmatter":{"description":"Learn how you can build celestia-app.","head":[["meta",{"name":"og:title","content":"Install celestia-app | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app.md","filePath":"nodes/celestia-app.md","lastUpdated":1712949880000}'),Ns={name:"nodes/celestia-app.md"},js=Object.assign(Ns,{setup(Rs){return(Os,$s)=>(c(),p("div",null,[i,s("ol",null,[r,s("li",null,[d,s("div",h,[_,s("div",y,[s("div",u,[b,g,s("pre",E,[s("code",null,[s("span",F,[m,C,v,f,s("span",k,"tags/"+t(e(n)["app-latest-tag"]),1),B,T,w,s("span",P,t(e(n)["app-latest-tag"]),1)])])]),s("pre",I,[s("code",null,[s("span",A,[S,V,q,x,s("span",D,"tags/"+t(e(n)["app-latest-tag"]),1),L,M,N,s("span",R,t(e(n)["app-latest-tag"]),1)])])])]),s("div",O,[$,H,s("pre",Q,[s("code",null,[s("span",Y,[K,U,j,J,s("span",W,"tags/"+t(e(l)["app-latest-tag"]),1),z,G,X,s("span",Z,t(e(l)["app-latest-tag"]),1)])])]),s("pre",ss,[s("code",null,[s("span",ts,[es,os,as,ls,s("span",ns,"tags/"+t(e(l)["app-latest-tag"]),1),cs,ps,is,s("span",rs,t(e(l)["app-latest-tag"]),1)])])])]),s("div",ds,[hs,_s,s("pre",ys,[s("code",null,[s("span",us,[bs,gs,Es,Fs,s("span",ms,"tags/"+t(e(a)["app-latest-tag"]),1),Cs,vs,fs,s("span",ks,t(e(a)["app-latest-tag"]),1)])])]),s("pre",Bs,[s("code",null,[s("span",Ts,[ws,Ps,Is,As,s("span",Ss,"tags/"+t(e(a)["app-latest-tag"]),1),Vs,qs,xs,s("span",Ds,t(e(a)["app-latest-tag"]),1)])])])])])])]),Ls]),Ms]))}});export{Us as __pageData,js as default}; diff --git a/assets/nodes_celestia-app.md.1b04d1e9.lean.js b/assets/nodes_celestia-app.md.1b04d1e9.lean.js new file mode 100644 index 00000000000..5cb67a98eee --- /dev/null +++ b/assets/nodes_celestia-app.md.1b04d1e9.lean.js @@ -0,0 +1 @@ +import{a}from"./chunks/arabica_versions.d7daf483.js";import{m as l}from"./chunks/mocha_versions.3ba4a823.js";import{m as n}from"./chunks/mainnet_versions.84d9fd1c.js";import{o as c,c as p,k as s,t,l as e,Q as o}from"./chunks/framework.51d6c45b.js";const i=o("",5),r=o("",1),d=s("p",null,"Check out to the desired version, based on the network you will use:",-1),h={class:"vp-code-group vp-adaptive-theme"},_=o("",1),y={class:"blocks"},u={class:"language-bash vp-adaptive-theme active"},b=s("button",{title:"Copy Code",class:"copy"},null,-1),g=s("span",{class:"lang"},"bash",-1),E={class:"shiki github-dark vp-code-dark"},F={class:"line"},m=s("span",{style:{color:"#B392F0"}},"git",-1),C=s("span",{style:{color:"#E1E4E8"}}," ",-1),v=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),f=s("span",{style:{color:"#E1E4E8"}}," ",-1),k={style:{color:"#9ECBFF"}},B=s("span",{style:{color:"#E1E4E8"}}," ",-1),T=s("span",{style:{color:"#79B8FF"}},"-b",-1),w=s("span",{style:{color:"#E1E4E8"}}," ",-1),P={style:{color:"#9ECBFF"}},I={class:"shiki github-light vp-code-light"},A={class:"line"},S=s("span",{style:{color:"#6F42C1"}},"git",-1),V=s("span",{style:{color:"#24292E"}}," ",-1),q=s("span",{style:{color:"#032F62"}},"checkout",-1),x=s("span",{style:{color:"#24292E"}}," ",-1),D={style:{color:"#032F62"}},L=s("span",{style:{color:"#24292E"}}," ",-1),M=s("span",{style:{color:"#005CC5"}},"-b",-1),N=s("span",{style:{color:"#24292E"}}," ",-1),R={style:{color:"#032F62"}},O={class:"language-bash vp-adaptive-theme"},$=s("button",{title:"Copy Code",class:"copy"},null,-1),H=s("span",{class:"lang"},"bash",-1),Q={class:"shiki github-dark vp-code-dark"},Y={class:"line"},K=s("span",{style:{color:"#B392F0"}},"git",-1),U=s("span",{style:{color:"#E1E4E8"}}," ",-1),j=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),J=s("span",{style:{color:"#E1E4E8"}}," ",-1),W={style:{color:"#9ECBFF"}},z=s("span",{style:{color:"#E1E4E8"}}," ",-1),G=s("span",{style:{color:"#79B8FF"}},"-b",-1),X=s("span",{style:{color:"#E1E4E8"}}," ",-1),Z={style:{color:"#9ECBFF"}},ss={class:"shiki github-light vp-code-light"},ts={class:"line"},es=s("span",{style:{color:"#6F42C1"}},"git",-1),os=s("span",{style:{color:"#24292E"}}," ",-1),as=s("span",{style:{color:"#032F62"}},"checkout",-1),ls=s("span",{style:{color:"#24292E"}}," ",-1),ns={style:{color:"#032F62"}},cs=s("span",{style:{color:"#24292E"}}," ",-1),ps=s("span",{style:{color:"#005CC5"}},"-b",-1),is=s("span",{style:{color:"#24292E"}}," ",-1),rs={style:{color:"#032F62"}},ds={class:"language-bash vp-adaptive-theme"},hs=s("button",{title:"Copy Code",class:"copy"},null,-1),_s=s("span",{class:"lang"},"bash",-1),ys={class:"shiki github-dark vp-code-dark"},us={class:"line"},bs=s("span",{style:{color:"#B392F0"}},"git",-1),gs=s("span",{style:{color:"#E1E4E8"}}," ",-1),Es=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),Fs=s("span",{style:{color:"#E1E4E8"}}," ",-1),ms={style:{color:"#9ECBFF"}},Cs=s("span",{style:{color:"#E1E4E8"}}," ",-1),vs=s("span",{style:{color:"#79B8FF"}},"-b",-1),fs=s("span",{style:{color:"#E1E4E8"}}," ",-1),ks={style:{color:"#9ECBFF"}},Bs={class:"shiki github-light vp-code-light"},Ts={class:"line"},ws=s("span",{style:{color:"#6F42C1"}},"git",-1),Ps=s("span",{style:{color:"#24292E"}}," ",-1),Is=s("span",{style:{color:"#032F62"}},"checkout",-1),As=s("span",{style:{color:"#24292E"}}," ",-1),Ss={style:{color:"#032F62"}},Vs=s("span",{style:{color:"#24292E"}}," ",-1),qs=s("span",{style:{color:"#005CC5"}},"-b",-1),xs=s("span",{style:{color:"#24292E"}}," ",-1),Ds={style:{color:"#032F62"}},Ls=o("",2),Ms=o("",18),Us=JSON.parse('{"title":"Install celestia-app","description":"Learn how you can build celestia-app.","frontmatter":{"description":"Learn how you can build celestia-app.","head":[["meta",{"name":"og:title","content":"Install celestia-app | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app.md","filePath":"nodes/celestia-app.md","lastUpdated":1712949880000}'),Ns={name:"nodes/celestia-app.md"},js=Object.assign(Ns,{setup(Rs){return(Os,$s)=>(c(),p("div",null,[i,s("ol",null,[r,s("li",null,[d,s("div",h,[_,s("div",y,[s("div",u,[b,g,s("pre",E,[s("code",null,[s("span",F,[m,C,v,f,s("span",k,"tags/"+t(e(n)["app-latest-tag"]),1),B,T,w,s("span",P,t(e(n)["app-latest-tag"]),1)])])]),s("pre",I,[s("code",null,[s("span",A,[S,V,q,x,s("span",D,"tags/"+t(e(n)["app-latest-tag"]),1),L,M,N,s("span",R,t(e(n)["app-latest-tag"]),1)])])])]),s("div",O,[$,H,s("pre",Q,[s("code",null,[s("span",Y,[K,U,j,J,s("span",W,"tags/"+t(e(l)["app-latest-tag"]),1),z,G,X,s("span",Z,t(e(l)["app-latest-tag"]),1)])])]),s("pre",ss,[s("code",null,[s("span",ts,[es,os,as,ls,s("span",ns,"tags/"+t(e(l)["app-latest-tag"]),1),cs,ps,is,s("span",rs,t(e(l)["app-latest-tag"]),1)])])])]),s("div",ds,[hs,_s,s("pre",ys,[s("code",null,[s("span",us,[bs,gs,Es,Fs,s("span",ms,"tags/"+t(e(a)["app-latest-tag"]),1),Cs,vs,fs,s("span",ks,t(e(a)["app-latest-tag"]),1)])])]),s("pre",Bs,[s("code",null,[s("span",Ts,[ws,Ps,Is,As,s("span",Ss,"tags/"+t(e(a)["app-latest-tag"]),1),Vs,qs,xs,s("span",Ds,t(e(a)["app-latest-tag"]),1)])])])])])])]),Ls]),Ms]))}});export{Us as __pageData,js as default}; diff --git a/assets/nodes_celestia-node-custom-networks.md.804b9f53.js b/assets/nodes_celestia-node-custom-networks.md.804b9f53.js new file mode 100644 index 00000000000..222848dc4f3 --- /dev/null +++ b/assets/nodes_celestia-node-custom-networks.md.804b9f53.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as o,Q as n}from"./chunks/framework.51d6c45b.js";const F=JSON.parse('{"title":"Custom networks and values","description":"Learn about custom networks and values on celestia-node.","frontmatter":{"description":"Learn about custom networks and values on celestia-node.","head":[["meta",{"name":"og:title","content":"Custom networks and values | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-node-custom-networks.md","filePath":"nodes/celestia-node-custom-networks.md","lastUpdated":1713793066000}'),p={name:"nodes/celestia-node-custom-networks.md"},l=n('

    Custom networks and values

    This section will cover importing bootstrapper IDs, chain ID, and network ID. This will allow you to import custom values for a chain that is not in the default configuration.

    If you have a custom network you can export CELESTIA_CUSTOM, which will look something like:

    bash
    export BRIDGE="/ip4/<ip-address>/tcp/2121/p2p/<node-ID>"\nexport GENESIS_HASH=<genesis-hash>\nexport NETWORK=<network-name>\nexport CELESTIA_CUSTOM="${NETWORK}:${GENESIS_HASH}:${BRIDGE}"
    export BRIDGE="/ip4/<ip-address>/tcp/2121/p2p/<node-ID>"\nexport GENESIS_HASH=<genesis-hash>\nexport NETWORK=<network-name>\nexport CELESTIA_CUSTOM="${NETWORK}:${GENESIS_HASH}:${BRIDGE}"

    Query your node ID using the RPC CLI. These values with examples would look like:

    bash
    export BRIDGE="/ip4/151.115.14.33/tcp/2121/p2p/12D3KooWKEeRtzVMPUdxYsZo2edqps6mS67n6LT5mPdULSkPSxBQ"\nexport GENESIS_HASH=580B3DFF8A7C716968161D91116A1E171F486298D582874E93714E489C9E6E88\nexport NETWORK=custom\nexport CELESTIA_CUSTOM="${NETWORK}:${GENESIS_HASH}:${BRIDGE}"
    export BRIDGE="/ip4/151.115.14.33/tcp/2121/p2p/12D3KooWKEeRtzVMPUdxYsZo2edqps6mS67n6LT5mPdULSkPSxBQ"\nexport GENESIS_HASH=580B3DFF8A7C716968161D91116A1E171F486298D582874E93714E489C9E6E88\nexport NETWORK=custom\nexport CELESTIA_CUSTOM="${NETWORK}:${GENESIS_HASH}:${BRIDGE}"

    Then, start your node with:

    bash
    celestia <node-type> start [flags...]
    celestia <node-type> start [flags...]
    ',8),e=[l];function t(c,r,E,y,i,d){return a(),o("div",null,e)}const h=s(p,[["render",t]]);export{F as __pageData,h as default}; diff --git a/assets/nodes_celestia-node-custom-networks.md.804b9f53.lean.js b/assets/nodes_celestia-node-custom-networks.md.804b9f53.lean.js new file mode 100644 index 00000000000..130fe416ae2 --- /dev/null +++ b/assets/nodes_celestia-node-custom-networks.md.804b9f53.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as o,Q as n}from"./chunks/framework.51d6c45b.js";const F=JSON.parse('{"title":"Custom networks and values","description":"Learn about custom networks and values on celestia-node.","frontmatter":{"description":"Learn about custom networks and values on celestia-node.","head":[["meta",{"name":"og:title","content":"Custom networks and values | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-node-custom-networks.md","filePath":"nodes/celestia-node-custom-networks.md","lastUpdated":1713793066000}'),p={name:"nodes/celestia-node-custom-networks.md"},l=n("",8),e=[l];function t(c,r,E,y,i,d){return a(),o("div",null,e)}const h=s(p,[["render",t]]);export{F as __pageData,h as default}; diff --git a/assets/nodes_celestia-node-metrics.md.cfabfbdc.js b/assets/nodes_celestia-node-metrics.md.cfabfbdc.js new file mode 100644 index 00000000000..a41d20d4be2 --- /dev/null +++ b/assets/nodes_celestia-node-metrics.md.cfabfbdc.js @@ -0,0 +1,29 @@ +import{_ as s,o as n,c as a,Q as o}from"./chunks/framework.51d6c45b.js";const h=JSON.parse('{"title":"celestia-node metrics","description":"A guide on how to run metrics for your celestia-node DA instance.","frontmatter":{"description":"A guide on how to run metrics for your celestia-node DA instance.","head":[["meta",{"name":"og:title","content":"celestia-node metrics | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-node-metrics.md","filePath":"nodes/celestia-node-metrics.md","lastUpdated":1709566764000}'),l={name:"nodes/celestia-node-metrics.md"},e=o(`

    celestia-node metrics

    This tutorial is for running metrics for your celestia-node data availability instance. This tutorial will focus on running metrics for a light node.

    This tutorial assumes you have already setup your light node by following the tutorial in the celestia-node API tutorial.

    Running metrics flags

    You can enable the celestia-node metric flags with the following command:

    sh
    celestia <node-type> start --metrics.tls=<boolean> \\
    +    --metrics --metrics.endpoint <URI> \\
    +    --p2p.network <network> --core.ip <URI>
    celestia <node-type> start --metrics.tls=<boolean> \\
    +    --metrics --metrics.endpoint <URI> \\
    +    --p2p.network <network> --core.ip <URI>

    Add metrics flags to your node start command and restart your node to apply it. The metrics endpoint will gather your node's data to track your uptime.

    Note that the --metrics flag enables metrics and expects an input into --metrics.endpoint.

    We will go over what the endpoint will need to be in the metrics endpoint design considerations section.

    Mainnet Beta

    Here is an example for Mainnet Beta:

    sh
    celestia <node-type> start --metrics.tls=true \\
    +    --metrics --metrics.endpoint otel.celestia.observer \\
    +    --core.ip <URI>
    celestia <node-type> start --metrics.tls=true \\
    +    --metrics --metrics.endpoint otel.celestia.observer \\
    +    --core.ip <URI>

    Mocha testnet

    Here is an example for Mocha testnet:

    sh
    celestia <node-type> start --metrics.tls=true \\
    +    --metrics --metrics.endpoint otel.celestia-mocha.com \\
    +    --core.ip <URI> --p2p.network mocha
    celestia <node-type> start --metrics.tls=true \\
    +    --metrics --metrics.endpoint otel.celestia-mocha.com \\
    +    --core.ip <URI> --p2p.network mocha

    TLS connections

    The --metrics.tls flag enables or disables a TLS connection to the OpenTelemetry Protocol metrics backend. You need to choose a boolean value (true or false) for this flag.

    It's also common to set this flag to false when spinning up a local collector to check the metrics locally.

    However, if the collector is hosted in the cloud as a separate entity (like in a DevOps environment), enabling TLS is a necessity for secure communication.

    Here are examples of how to use it:

    bash
    # To enable TLS connection
    +celestia <node-type> start --metrics.tls=true --metrics \\
    +    --metrics.endpoint <URI> \\
    +    --p2p.network <network> --core.ip <URI>
    +
    +# To disable TLS connection
    +celestia <node-type> start --metrics.tls=false --metrics \\
    +    --metrics.endpoint <URI> \\
    +    --p2p.network <network> --core.ip <URI>
    # To enable TLS connection
    +celestia <node-type> start --metrics.tls=true --metrics \\
    +    --metrics.endpoint <URI> \\
    +    --p2p.network <network> --core.ip <URI>
    +
    +# To disable TLS connection
    +celestia <node-type> start --metrics.tls=false --metrics \\
    +    --metrics.endpoint <URI> \\
    +    --p2p.network <network> --core.ip <URI>

    Metrics endpoint design considerations

    At the moment, the architecture of celestia-node metrics works as specified in the following ADR #010.

    Essentially, the design considerations here will necessitate running an OpenTelemetry (OTEL) collector that connects to Celestia light node.

    For an overview of OTEL, check out the guide.

    The ADR and the OTEL docs will help you run your collector on the metrics endpoint. This will then allow you to process the data in the collector on a Prometheus server which can then be viewed on a Grafana dashboard.

    In the future, we do want to open-source some developer toolings around this infrastructure to allow for node operators to be able to monitor their data availability nodes.

    `,27),p=[e];function t(c,r,y,i,E,d){return n(),a("div",null,p)}const m=s(l,[["render",t]]);export{h as __pageData,m as default}; diff --git a/assets/nodes_celestia-node-metrics.md.cfabfbdc.lean.js b/assets/nodes_celestia-node-metrics.md.cfabfbdc.lean.js new file mode 100644 index 00000000000..a0dbd00aa77 --- /dev/null +++ b/assets/nodes_celestia-node-metrics.md.cfabfbdc.lean.js @@ -0,0 +1 @@ +import{_ as s,o as n,c as a,Q as o}from"./chunks/framework.51d6c45b.js";const h=JSON.parse('{"title":"celestia-node metrics","description":"A guide on how to run metrics for your celestia-node DA instance.","frontmatter":{"description":"A guide on how to run metrics for your celestia-node DA instance.","head":[["meta",{"name":"og:title","content":"celestia-node metrics | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-node-metrics.md","filePath":"nodes/celestia-node-metrics.md","lastUpdated":1709566764000}'),l={name:"nodes/celestia-node-metrics.md"},e=o("",27),p=[e];function t(c,r,y,i,E,d){return n(),a("div",null,p)}const m=s(l,[["render",t]]);export{h as __pageData,m as default}; diff --git a/assets/nodes_celestia-node-troubleshooting.md.b993a166.js b/assets/nodes_celestia-node-troubleshooting.md.b993a166.js new file mode 100644 index 00000000000..00a9ca394f5 --- /dev/null +++ b/assets/nodes_celestia-node-troubleshooting.md.b993a166.js @@ -0,0 +1,31 @@ +import{c as o}from"./chunks/constants.dc70dfe7.js";import{o as p,c as t,k as s,a,t as n,l,Q as e}from"./chunks/framework.51d6c45b.js";const r=e(`

    Troubleshooting

    Network selection

    Note: If you do not select a network, the default network will be Mainnet Beta.

    sh
    celestia <node-type> init --p2p.network <network>
    +celestia <node-type> start --p2p.network <network> --core.ip <URI>
    celestia <node-type> init --p2p.network <network>
    +celestia <node-type> start --p2p.network <network> --core.ip <URI>

    TIP

    Refer to the ports section of this page for information on which ports are required to be open on your machine.

    NOTE

    It is advised before switching networks to reinitialize your node via init command. This is due to an old config being present. Re-initialisation will reset the config.

    Chain ID

    `,7),c=s("em",null,"i.e.",-1),i=s("code",null,"--p2p.network string",-1),y=s("thead",null,[s("tr",null,[s("th",null,"Network"),s("th",null,"Chain ID"),s("th",null,[s("code",null,"--p2p.network string")])])],-1),E=s("td",null,"Mainnet Beta",-1),d=s("td",null,[a("not required ("),s("code",null,"--p2p.network celestia"),a(")")],-1),h=s("td",null,"Mocha",-1),F=s("td",null,[s("code",null,"--p2p.network mocha")],-1),g=s("td",null,"Arabica",-1),u=s("td",null,[s("code",null,"--p2p.network arabica")],-1),C=e(`

    Ports

    When interacting with a Celestia node, you may need to open ports on your machine to allow communication between nodes, such as bridge nodes. It is essential that specific ports are accessible. Make sure that your firewall allows connections to the correct ports.

    If you run a node on a cloud server, make sure that the ports are open on the server's firewall. If you run a node at home, make sure that your router allows connections to the correct ports.

    For example, validator ports 9090 and 26657 need to be accessible by the bridge, and port 2121 is required for P2P connections for all node types.

    The following ports are used by Celestia nodes:

    PortProtocolAddressDescriptionEnabled by default on nodeFlag
    2121TCP/UDPlocalhostP2PtrueN/A
    26658HTTPlocalhostRPCtrue--rpc.port string
    26659HTTPlocalhostREST Gatewayfalse--gateway.port string

    WARNING

    The gateway endpoints have been deprecated and will be removed in the future. If you would like to use them anyway, you can find more details on GitHub.

    Changing the location of your node store

    In this section, we'll guide you through starting your node using a node store in a different location than you originally started with.

    First, stop your node safely using control + C.

    Then, init your node again with a new node store:

    bash
    celestia <node-type> init --node.store /home/user/celestia-<node-type>-location/ \\
    +    --p2p.network mocha
    celestia <node-type> init --node.store /home/user/celestia-<node-type>-location/ \\
    +    --p2p.network mocha

    Next, start your node:

    bash
    celestia full start --core.ip rpc-mocha.pops.one --p2p.network mocha \\
    +    --node.store /home/user/celestia-<node-type>-location/
    celestia full start --core.ip rpc-mocha.pops.one --p2p.network mocha \\
    +    --node.store /home/user/celestia-<node-type>-location/

    If you choose to change the location of your node store, you will need to execute each command on your node with the following flag:

    bash
    --node.store /home/user/celestia-<node-type>-location/
    --node.store /home/user/celestia-<node-type>-location/

    When using cel-key, the process is different. To show the keys you should add --keyring-dir like this example:

    bash
    ./cel-key list --p2p.network mocha --node.type full \\
    +    --keyring-dir /home/user/celestia-<node-type>-location/keys/
    ./cel-key list --p2p.network mocha --node.type full \\
    +    --keyring-dir /home/user/celestia-<node-type>-location/keys/

    Resetting your config

    If you an encounter an error, it is likely that an old config file is present:

    sh
    Error: nodebuilder/share: interval must be positive; nodebuilder/core: invalid IP addr given:
    +
    +# or
    +
    +Error: nodebuilder/share: interval must be positive
    Error: nodebuilder/share: interval must be positive; nodebuilder/core: invalid IP addr given:
    +
    +# or
    +
    +Error: nodebuilder/share: interval must be positive

    You can re-initialize your node's config with the following commands:

    TIP

    Save your config so custom values are not lost.

    Run the following command to update your config:

    bash
    celestia <node-type> config-update --p2p.network <network>
    celestia <node-type> config-update --p2p.network <network>

    This will pull in any new values from new configuration and merge them into the existing configuration.

    TIP

    After using the config-update command, it is encouraged to double-check that your custom values are preserved.

    Then, to start your node again:

    bash
    celestia <node-type> start --p2p.network <network>
    celestia <node-type> start --p2p.network <network>

    Clearing the data store

    For bridge, full, and light nodes, remove the data store with this command:

    bash
    celestia <node-type> unsafe-reset-store --p2p.network <network>
    celestia <node-type> unsafe-reset-store --p2p.network <network>
    bash
    celestia light unsafe-reset-store --p2p.network mocha
    celestia light unsafe-reset-store --p2p.network mocha

    FATAL headers given to the heightSub are in the wrong order

    If you observe a FATAL log line like:

    bash
    FATAL   header/store   store/heightsub.go:87    PLEASE FILE A BUG REPORT: headers given to the heightSub are in the wrong order"
    FATAL   header/store   store/heightsub.go:87    PLEASE FILE A BUG REPORT: headers given to the heightSub are in the wrong order"

    then it is possible the celestia-node data/ directory contains headers from a previous instance of the network that you are currently trying to run against. One resolution strategy is to delete the existing celestia-node config for the target network and re-initialize it:

    sh
    # rm -rf ~/.celestia-<node-type>-<network>
    +rm -rf ~/.celestia-bridge-private
    +
    +# celestia <node-type> init --p2p.network <network>
    +celestia bridge init --p2p.network private
    # rm -rf ~/.celestia-<node-type>-<network>
    +rm -rf ~/.celestia-bridge-private
    +
    +# celestia <node-type> init --p2p.network <network>
    +celestia bridge init --p2p.network private

    Error: "too many open files"

    When running a Celestia bridge node, you may encounter an error in the logs similar to this:

    bash
    Error while creating log file in valueLog.open error: while opening file: /opt/celestia/.celestia-bridge/data/003442.vlog error: open /opt/celestia/.celestia-bridge/data/003442.vlog: too many open files
    Error while creating log file in valueLog.open error: while opening file: /opt/celestia/.celestia-bridge/data/003442.vlog error: open /opt/celestia/.celestia-bridge/data/003442.vlog: too many open files

    This error indicates that the Celestia application is trying to open more files than the operating system's limit allows. To fix this, you will need to edit the Celestia bridge service file to increase the number of file descriptors that the service can open.

    1. Open the service file for editing:
    bash
    nano /etc/systemd/system/celestia-bridge.service
    nano /etc/systemd/system/celestia-bridge.service
    1. Modify the LimitNOFILE parameter:

    In the service file, find the LimitNOFILE parameter under the [Service] section and set its value to 1400000. It should look like this:

    ini
    [Service]
    +...
    +LimitNOFILE=1400000
    +...
    [Service]
    +...
    +LimitNOFILE=1400000
    +...

    NOTE

    Be cautious when increasing file descriptor limits. Setting this value too high might affect system performance. Ensure the value is appropriate for your system's capabilities.

    1. Reload daemon and restart bridge service:
    bash
    sudo systemctl daemon-reload
    sudo systemctl daemon-reload
    bash
    sudo systemctl restart celestia-bridge
    sudo systemctl restart celestia-bridge
    `,51),w=JSON.parse('{"title":"Troubleshooting","description":"A guide to troubleshooting common issues with Celestia Node.","frontmatter":{"description":"A guide to troubleshooting common issues with Celestia Node.","next":{"text":"Metrics, visualization, and alerts","link":"nodes/celestia-app-metrics"},"head":[["meta",{"name":"og:title","content":"Troubleshooting | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-node-troubleshooting.md","filePath":"nodes/celestia-node-troubleshooting.md","lastUpdated":1709217618000}'),b={name:"nodes/celestia-node-troubleshooting.md"},A=Object.assign(b,{setup(v){return(k,m)=>(p(),t("div",null,[r,s("p",null,[a("When interacting with celestia-node, it is important to take into account the different chain IDs for different networks. For Mainnet Beta, there is no need to declare a chain ID, as the default is "+n(l(o).mainnetChainId)+", ",1),c,a(" no "),i,a(" flag is required for Mainnet Beta.")]),s("table",null,[y,s("tbody",null,[s("tr",null,[E,s("td",null,n(l(o).mainnetChainId),1),d]),s("tr",null,[h,s("td",null,n(l(o).mochaChainId),1),F]),s("tr",null,[g,s("td",null,n(l(o).arabicaChainId),1),u])])]),C]))}});export{w as __pageData,A as default}; diff --git a/assets/nodes_celestia-node-troubleshooting.md.b993a166.lean.js b/assets/nodes_celestia-node-troubleshooting.md.b993a166.lean.js new file mode 100644 index 00000000000..e06b6000f67 --- /dev/null +++ b/assets/nodes_celestia-node-troubleshooting.md.b993a166.lean.js @@ -0,0 +1 @@ +import{c as o}from"./chunks/constants.dc70dfe7.js";import{o as p,c as t,k as s,a,t as n,l,Q as e}from"./chunks/framework.51d6c45b.js";const r=e("",7),c=s("em",null,"i.e.",-1),i=s("code",null,"--p2p.network string",-1),y=s("thead",null,[s("tr",null,[s("th",null,"Network"),s("th",null,"Chain ID"),s("th",null,[s("code",null,"--p2p.network string")])])],-1),E=s("td",null,"Mainnet Beta",-1),d=s("td",null,[a("not required ("),s("code",null,"--p2p.network celestia"),a(")")],-1),h=s("td",null,"Mocha",-1),F=s("td",null,[s("code",null,"--p2p.network mocha")],-1),g=s("td",null,"Arabica",-1),u=s("td",null,[s("code",null,"--p2p.network arabica")],-1),C=e("",51),w=JSON.parse('{"title":"Troubleshooting","description":"A guide to troubleshooting common issues with Celestia Node.","frontmatter":{"description":"A guide to troubleshooting common issues with Celestia Node.","next":{"text":"Metrics, visualization, and alerts","link":"nodes/celestia-app-metrics"},"head":[["meta",{"name":"og:title","content":"Troubleshooting | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-node-troubleshooting.md","filePath":"nodes/celestia-node-troubleshooting.md","lastUpdated":1709217618000}'),b={name:"nodes/celestia-node-troubleshooting.md"},A=Object.assign(b,{setup(v){return(k,m)=>(p(),t("div",null,[r,s("p",null,[a("When interacting with celestia-node, it is important to take into account the different chain IDs for different networks. For Mainnet Beta, there is no need to declare a chain ID, as the default is "+n(l(o).mainnetChainId)+", ",1),c,a(" no "),i,a(" flag is required for Mainnet Beta.")]),s("table",null,[y,s("tbody",null,[s("tr",null,[E,s("td",null,n(l(o).mainnetChainId),1),d]),s("tr",null,[h,s("td",null,n(l(o).mochaChainId),1),F]),s("tr",null,[g,s("td",null,n(l(o).arabicaChainId),1),u])])]),C]))}});export{w as __pageData,A as default}; diff --git a/assets/nodes_celestia-node.md.791b437d.js b/assets/nodes_celestia-node.md.791b437d.js new file mode 100644 index 00000000000..ece7ef44295 --- /dev/null +++ b/assets/nodes_celestia-node.md.791b437d.js @@ -0,0 +1,7 @@ +import{a as t}from"./chunks/arabica_versions.d7daf483.js";import{m as l}from"./chunks/mocha_versions.3ba4a823.js";import{m as n}from"./chunks/mainnet_versions.84d9fd1c.js";import{o as i,c,k as s,t as e,l as a,Q as o}from"./chunks/framework.51d6c45b.js";const p=o('

    Install celestia-node

    Installing from source

    This section goes over building and installing celestia-node. This tutorial assumes you completed the steps in setting up your development environment.

    Install the celestia-node binary by running the following commands:

    ',4),r=o(`
  • Remove any existing copy of celestia-node, clone the repository, and change into the directory:

    bash
    cd $HOME
    +rm -rf celestia-node
    +git clone https://github.com/celestiaorg/celestia-node.git
    +cd celestia-node/
    cd $HOME
    +rm -rf celestia-node
    +git clone https://github.com/celestiaorg/celestia-node.git
    +cd celestia-node/
  • `,1),d=s("p",null,"Check out to the desired version, based on the network you will use:",-1),h={class:"vp-code-group vp-adaptive-theme"},u=o('
    ',1),_={class:"blocks"},y={class:"language-bash vp-adaptive-theme active"},g=s("button",{title:"Copy Code",class:"copy"},null,-1),b=s("span",{class:"lang"},"bash",-1),m={class:"shiki github-dark vp-code-dark"},E={class:"line"},v=s("span",{style:{color:"#B392F0"}},"git",-1),F=s("span",{style:{color:"#E1E4E8"}}," ",-1),k=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),C=s("span",{style:{color:"#E1E4E8"}}," ",-1),f={style:{color:"#9ECBFF"}},B={class:"shiki github-light vp-code-light"},T={class:"line"},w=s("span",{style:{color:"#6F42C1"}},"git",-1),I=s("span",{style:{color:"#24292E"}}," ",-1),A=s("span",{style:{color:"#032F62"}},"checkout",-1),S=s("span",{style:{color:"#24292E"}}," ",-1),P={style:{color:"#032F62"}},q={class:"language-bash vp-adaptive-theme"},x=s("button",{title:"Copy Code",class:"copy"},null,-1),V=s("span",{class:"lang"},"bash",-1),N={class:"shiki github-dark vp-code-dark"},O={class:"line"},D=s("span",{style:{color:"#B392F0"}},"git",-1),R=s("span",{style:{color:"#E1E4E8"}}," ",-1),j=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),L=s("span",{style:{color:"#E1E4E8"}}," ",-1),M={style:{color:"#9ECBFF"}},$={class:"shiki github-light vp-code-light"},H={class:"line"},J=s("span",{style:{color:"#6F42C1"}},"git",-1),U=s("span",{style:{color:"#24292E"}}," ",-1),Y=s("span",{style:{color:"#032F62"}},"checkout",-1),G=s("span",{style:{color:"#24292E"}}," ",-1),Z={style:{color:"#032F62"}},Q={class:"language-bash vp-adaptive-theme"},z=s("button",{title:"Copy Code",class:"copy"},null,-1),K=s("span",{class:"lang"},"bash",-1),W={class:"shiki github-dark vp-code-dark"},X={class:"line"},ss=s("span",{style:{color:"#B392F0"}},"git",-1),es=s("span",{style:{color:"#E1E4E8"}}," ",-1),as=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),os=s("span",{style:{color:"#E1E4E8"}}," ",-1),ts={style:{color:"#9ECBFF"}},ls={class:"shiki github-light vp-code-light"},ns={class:"line"},is=s("span",{style:{color:"#6F42C1"}},"git",-1),cs=s("span",{style:{color:"#24292E"}}," ",-1),ps=s("span",{style:{color:"#032F62"}},"checkout",-1),rs=s("span",{style:{color:"#24292E"}}," ",-1),ds={style:{color:"#032F62"}},hs=o('
  • Build the celestia binary:

    a. Standard build

    bash
    make build
    make build

    b. Experimental build

    OPTIONAL

    If you're a node operator comfortable with experimental features and seeking optimal performance with minimal RAM usage, this option is recommended for you.

    bash
    make build-jemalloc
    make build-jemalloc

    This build option enables CGO, and downloads and installs jemalloc. Learn more about the build command.

  • Install the binary:

    bash
    make install
    make install
  • Build the cel-key utility:

    bash
    make cel-key
    make cel-key
  • Verify that the binary is working and check the version:

    bash
    celestia version
    celestia version
  • ',4),us=o('

    The output will show the semantic version of celestia-node, commit hash, build date, system version, and Golang version.

    Installing a pre-built binary

    Installing a pre-built binary is the fastest way to get started with your Celestia data availability node. Releases after celestia-node v0.13.3 should have these binaries available.

    The steps below will download a binary file named celestia. Depending on the setup that you choose during installation, the celestia binary will be available at either:

    • $HOME/celestia-node-temp/celestia
    • /usr/local/bin/celestia

    Pre-built binaries are available for:

    • Operating systems: Darwin (Apple), Linux
    • Architectures: x86_64 (amd64), arm64

    To install the latest pre-built binary you can run this command in your terminal:

    bash
    bash -c "$(curl -sL https://docs.celestia.org/celestia-node.sh)"
    bash -c "$(curl -sL https://docs.celestia.org/celestia-node.sh)"

    Follow the instructions in the terminal output to choose your installation preferences.

    You will see an output with the menu for celestia.

    View the script to learn more about what it is doing.

    Next steps

    First, we recommend reading the overview of our node types, if you haven't yet.

    Now that you've installed Celestia Node, it's time to pick your node type and run your node!

    If you're planning to run a light node, we recommend the node RPC CLI tutorial.

    Upgrading your binary

    To upgrade your binary, you can install the latest version from the instructions above and restart your node. If you run into any issues, Refer to the troubleshooting section.

    ',18),ks=JSON.parse('{"title":"Install celestia-node","description":"Learn to build and install celestia-node.","frontmatter":{"description":"Learn to build and install celestia-node.","head":[["meta",{"name":"og:title","content":"Install celestia-node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-node.md","filePath":"nodes/celestia-node.md","lastUpdated":1712949997000}'),_s={name:"nodes/celestia-node.md"},Cs=Object.assign(_s,{setup(ys){return(gs,bs)=>(i(),c("div",null,[p,s("ol",null,[r,s("li",null,[d,s("div",h,[u,s("div",_,[s("div",y,[g,b,s("pre",m,[s("code",null,[s("span",E,[v,F,k,C,s("span",f,"tags/"+e(a(n)["node-latest-tag"]),1)])])]),s("pre",B,[s("code",null,[s("span",T,[w,I,A,S,s("span",P,"tags/"+e(a(n)["node-latest-tag"]),1)])])])]),s("div",q,[x,V,s("pre",N,[s("code",null,[s("span",O,[D,R,j,L,s("span",M,"tags/"+e(a(l)["node-latest-tag"]),1)])])]),s("pre",$,[s("code",null,[s("span",H,[J,U,Y,G,s("span",Z,"tags/"+e(a(l)["node-latest-tag"]),1)])])])]),s("div",Q,[z,K,s("pre",W,[s("code",null,[s("span",X,[ss,es,as,os,s("span",ts,"tags/"+e(a(t)["node-latest-tag"]),1)])])]),s("pre",ls,[s("code",null,[s("span",ns,[is,cs,ps,rs,s("span",ds,"tags/"+e(a(t)["node-latest-tag"]),1)])])])])])])]),hs]),us]))}});export{ks as __pageData,Cs as default}; diff --git a/assets/nodes_celestia-node.md.791b437d.lean.js b/assets/nodes_celestia-node.md.791b437d.lean.js new file mode 100644 index 00000000000..04ce96d7518 --- /dev/null +++ b/assets/nodes_celestia-node.md.791b437d.lean.js @@ -0,0 +1 @@ +import{a as t}from"./chunks/arabica_versions.d7daf483.js";import{m as l}from"./chunks/mocha_versions.3ba4a823.js";import{m as n}from"./chunks/mainnet_versions.84d9fd1c.js";import{o as i,c,k as s,t as e,l as a,Q as o}from"./chunks/framework.51d6c45b.js";const p=o("",4),r=o("",1),d=s("p",null,"Check out to the desired version, based on the network you will use:",-1),h={class:"vp-code-group vp-adaptive-theme"},u=o("",1),_={class:"blocks"},y={class:"language-bash vp-adaptive-theme active"},g=s("button",{title:"Copy Code",class:"copy"},null,-1),b=s("span",{class:"lang"},"bash",-1),m={class:"shiki github-dark vp-code-dark"},E={class:"line"},v=s("span",{style:{color:"#B392F0"}},"git",-1),F=s("span",{style:{color:"#E1E4E8"}}," ",-1),k=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),C=s("span",{style:{color:"#E1E4E8"}}," ",-1),f={style:{color:"#9ECBFF"}},B={class:"shiki github-light vp-code-light"},T={class:"line"},w=s("span",{style:{color:"#6F42C1"}},"git",-1),I=s("span",{style:{color:"#24292E"}}," ",-1),A=s("span",{style:{color:"#032F62"}},"checkout",-1),S=s("span",{style:{color:"#24292E"}}," ",-1),P={style:{color:"#032F62"}},q={class:"language-bash vp-adaptive-theme"},x=s("button",{title:"Copy Code",class:"copy"},null,-1),V=s("span",{class:"lang"},"bash",-1),N={class:"shiki github-dark vp-code-dark"},O={class:"line"},D=s("span",{style:{color:"#B392F0"}},"git",-1),R=s("span",{style:{color:"#E1E4E8"}}," ",-1),j=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),L=s("span",{style:{color:"#E1E4E8"}}," ",-1),M={style:{color:"#9ECBFF"}},$={class:"shiki github-light vp-code-light"},H={class:"line"},J=s("span",{style:{color:"#6F42C1"}},"git",-1),U=s("span",{style:{color:"#24292E"}}," ",-1),Y=s("span",{style:{color:"#032F62"}},"checkout",-1),G=s("span",{style:{color:"#24292E"}}," ",-1),Z={style:{color:"#032F62"}},Q={class:"language-bash vp-adaptive-theme"},z=s("button",{title:"Copy Code",class:"copy"},null,-1),K=s("span",{class:"lang"},"bash",-1),W={class:"shiki github-dark vp-code-dark"},X={class:"line"},ss=s("span",{style:{color:"#B392F0"}},"git",-1),es=s("span",{style:{color:"#E1E4E8"}}," ",-1),as=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),os=s("span",{style:{color:"#E1E4E8"}}," ",-1),ts={style:{color:"#9ECBFF"}},ls={class:"shiki github-light vp-code-light"},ns={class:"line"},is=s("span",{style:{color:"#6F42C1"}},"git",-1),cs=s("span",{style:{color:"#24292E"}}," ",-1),ps=s("span",{style:{color:"#032F62"}},"checkout",-1),rs=s("span",{style:{color:"#24292E"}}," ",-1),ds={style:{color:"#032F62"}},hs=o("",4),us=o("",18),ks=JSON.parse('{"title":"Install celestia-node","description":"Learn to build and install celestia-node.","frontmatter":{"description":"Learn to build and install celestia-node.","head":[["meta",{"name":"og:title","content":"Install celestia-node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-node.md","filePath":"nodes/celestia-node.md","lastUpdated":1712949997000}'),_s={name:"nodes/celestia-node.md"},Cs=Object.assign(_s,{setup(ys){return(gs,bs)=>(i(),c("div",null,[p,s("ol",null,[r,s("li",null,[d,s("div",h,[u,s("div",_,[s("div",y,[g,b,s("pre",m,[s("code",null,[s("span",E,[v,F,k,C,s("span",f,"tags/"+e(a(n)["node-latest-tag"]),1)])])]),s("pre",B,[s("code",null,[s("span",T,[w,I,A,S,s("span",P,"tags/"+e(a(n)["node-latest-tag"]),1)])])])]),s("div",q,[x,V,s("pre",N,[s("code",null,[s("span",O,[D,R,j,L,s("span",M,"tags/"+e(a(l)["node-latest-tag"]),1)])])]),s("pre",$,[s("code",null,[s("span",H,[J,U,Y,G,s("span",Z,"tags/"+e(a(l)["node-latest-tag"]),1)])])])]),s("div",Q,[z,K,s("pre",W,[s("code",null,[s("span",X,[ss,es,as,os,s("span",ts,"tags/"+e(a(t)["node-latest-tag"]),1)])])]),s("pre",ls,[s("code",null,[s("span",ns,[is,cs,ps,rs,s("span",ds,"tags/"+e(a(t)["node-latest-tag"]),1)])])])])])])]),hs]),us]))}});export{ks as __pageData,Cs as default}; diff --git a/assets/nodes_config-toml.md.2c893551.js b/assets/nodes_config-toml.md.2c893551.js new file mode 100644 index 00000000000..979592bd49d --- /dev/null +++ b/assets/nodes_config-toml.md.2c893551.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"config.toml guide","description":"A guide to the contents of the Config.toml file.","frontmatter":{"description":"A guide to the contents of the Config.toml file.","head":[["meta",{"name":"og:title","content":"config.toml guide | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/config-toml.md","filePath":"nodes/config-toml.md","lastUpdated":1698091500000}'),i={name:"nodes/config-toml.md"},r=a('

    config.toml guide

    Pre-requisites

    Please, make sure that you have installed and initialized celestia-node

    Understanding config.toml

    After initialization, for any type of node, you will find a config.toml in the following path (default location):

    • $HOME/.celestia-bridge/config.toml for bridge node
    • $HOME/.celestia-light/config.toml for light node
    • $HOME/.celestia-full/config.toml for a full DA node

    Let's break down some of the most used sections.

    Core

    This section is needed for the Celestia bridge node. By default, Remote = false. Still for devnet, we are going to use the remote core option and this can also be set by the command line flag --core.remote.

    P2P

    Bootstrap

    Bootstrappers help new nodes to find peers faster in the network. By default, the Bootstrapper = false and the BootstrapPeers is empty. If you want your node to be a bootstrapper, then activate Bootstrapper = true. BootstrapPeers are already provided by default during initialisation. If you want to add your own manually, you need to provide the multiaddresses of the peers.

    Mutual peers

    The purpose of this config is to set up a bidirectional communication. This is usually the case for Celestia bridge nodes. In addition, you need to change the field PeerExchange from false to true.

    Services

    TrustedHash and TrustedPeer

    TrustedHash is needed to properly initialize a Celestia bridge node with an already-running Remote celestia-core node. Celestia light node will take a genesis hash as the trusted one, if no hash is manually provided during initialization phase.

    TrustedPeers is the array of bridge nodes' peers that Celestia light node trusts. By default, bootstrap peers becomes trusted peers for Celestia light nodes if a user is not setting the trusted peer params in config file.

    Any Celestia bridge node can be a trusted peer for the light one. However, the light node by design can not be a trusted peer for another light node.

    ',19),n=[r];function d(s,l,c,h,u,f){return t(),o("div",null,n)}const m=e(i,[["render",d]]);export{g as __pageData,m as default}; diff --git a/assets/nodes_config-toml.md.2c893551.lean.js b/assets/nodes_config-toml.md.2c893551.lean.js new file mode 100644 index 00000000000..29c184e2293 --- /dev/null +++ b/assets/nodes_config-toml.md.2c893551.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.51d6c45b.js";const g=JSON.parse('{"title":"config.toml guide","description":"A guide to the contents of the Config.toml file.","frontmatter":{"description":"A guide to the contents of the Config.toml file.","head":[["meta",{"name":"og:title","content":"config.toml guide | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/config-toml.md","filePath":"nodes/config-toml.md","lastUpdated":1698091500000}'),i={name:"nodes/config-toml.md"},r=a("",19),n=[r];function d(s,l,c,h,u,f){return t(),o("div",null,n)}const m=e(i,[["render",d]]);export{g as __pageData,m as default}; diff --git a/assets/nodes_decide-node.md.03a657c5.js b/assets/nodes_decide-node.md.03a657c5.js new file mode 100644 index 00000000000..e4e828c1340 --- /dev/null +++ b/assets/nodes_decide-node.md.03a657c5.js @@ -0,0 +1 @@ +import{_ as e,o as n,c as o,Q as a}from"./chunks/framework.51d6c45b.js";const _=JSON.parse('{"title":"Deciding which node to run","description":"Guide on helping you decide which type of node to run.","frontmatter":{"description":"Guide on helping you decide which type of node to run.","head":[["meta",{"name":"og:title","content":"Deciding which node to run | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/decide-node.md","filePath":"nodes/decide-node.md","lastUpdated":1711744858000}'),t={name:"nodes/decide-node.md"},i=a('

    Deciding which node to run

    Now that you have installed the basic dependencies, you can start exploring which nodes to run!

    Beginner

    It is highly recommended if you are a beginner to get started with running a Data-Availability light node.

    In order to get started, you can proceed to the light node section.

    You can also play around with the Data Availability API in this tutorial for posting and retrieving data with a light node.

    Advanced

    If you are looking to run a consensus node, please follow the tutorial for running a consensus node.

    Note that running a validator means you must also run a bridge node, which is covered in this section.

    ',9),d=[i];function r(c,h,s,l,u,p){return n(),o("div",null,d)}const f=e(t,[["render",r]]);export{_ as __pageData,f as default}; diff --git a/assets/nodes_decide-node.md.03a657c5.lean.js b/assets/nodes_decide-node.md.03a657c5.lean.js new file mode 100644 index 00000000000..9e2bb1a8b9e --- /dev/null +++ b/assets/nodes_decide-node.md.03a657c5.lean.js @@ -0,0 +1 @@ +import{_ as e,o as n,c as o,Q as a}from"./chunks/framework.51d6c45b.js";const _=JSON.parse('{"title":"Deciding which node to run","description":"Guide on helping you decide which type of node to run.","frontmatter":{"description":"Guide on helping you decide which type of node to run.","head":[["meta",{"name":"og:title","content":"Deciding which node to run | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/decide-node.md","filePath":"nodes/decide-node.md","lastUpdated":1711744858000}'),t={name:"nodes/decide-node.md"},i=a("",9),d=[i];function r(c,h,s,l,u,p){return n(),o("div",null,d)}const f=e(t,[["render",r]]);export{_ as __pageData,f as default}; diff --git a/assets/nodes_docker-images.md.a7ae248f.js b/assets/nodes_docker-images.md.a7ae248f.js new file mode 100644 index 00000000000..36ca80e71ee --- /dev/null +++ b/assets/nodes_docker-images.md.a7ae248f.js @@ -0,0 +1,59 @@ +import{a as l}from"./chunks/arabica_versions.d7daf483.js";import{m as t}from"./chunks/mocha_versions.3ba4a823.js";import{m as p}from"./chunks/mainnet_versions.84d9fd1c.js";import{o as c,c as r,k as s,a,t as n,l as e,Q as o}from"./chunks/framework.51d6c45b.js";const _=o('

    🐳 Docker setup

    This page has instructions to run celestia-node using Docker. If you are looking for instructions to run celestia-node using a binary, please refer to the celestia-node page.

    Using Docker is the easiest way to run celestia-node for most users. Docker is a containerization platform that allows you to run celestia-node in an isolated environment.

    This means that you can run celestia-node on your machine without having to worry about installing and configuring all of the dependencies required to run the node.

    If you would like to learn more about key management in Docker, visit the Docker and cel-key section.

    The easiest way to install Docker is to use the Docker Desktop installer or Ubuntu. You can follow the instructions for your operating system.

    Prerequisites

    Quick start

    ',9),i=o('
  • Set the network you would like to run your node on:

    bash
    export NETWORK=celestia
    export NETWORK=celestia
    bash
    export NETWORK=mocha
    export NETWORK=mocha
    bash
    export NETWORK=arabica
    export NETWORK=arabica
  • Set the node type

    bash
    export NODE_TYPE=light
    export NODE_TYPE=light
    bash
    export NODE_TYPE=bridge
    export NODE_TYPE=bridge
    bash
    export NODE_TYPE=full
    export NODE_TYPE=full
  • Set an RPC endpoint for either Mainnet Beta, Mocha, or Arabica using the bare URL (without http or https):

    bash
    export RPC_URL=this-is-an-rpc-url.com
    export RPC_URL=this-is-an-rpc-url.com
  • ',3),E=s("p",null,"Run the image from the command line:",-1),y={class:"vp-code-group vp-adaptive-theme"},d=o('
    ',1),h={class:"blocks"},T={class:"language-bash vp-adaptive-theme active"},F=s("button",{title:"Copy Code",class:"copy"},null,-1),C=s("span",{class:"lang"},"bash",-1),u={class:"shiki github-dark vp-code-dark"},g=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),b={class:"line"},v=s("span",{style:{color:"#E1E4E8"}}," ",-1),P={style:{color:"#9ECBFF"}},k=s("span",{style:{color:"#E1E4E8"}}," ",-1),m=s("span",{style:{color:"#79B8FF"}},"\\",-1),A=o(' celestia $NODE_TYPE start --core.ip $RPC_URL --p2p.network $NETWORK',1),S={class:"shiki github-light vp-code-light"},N=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),B={class:"line"},D=s("span",{style:{color:"#24292E"}}," ",-1),R={style:{color:"#032F62"}},O=s("span",{style:{color:"#24292E"}}," ",-1),I=s("span",{style:{color:"#005CC5"}},"\\",-1),V=o(' celestia $NODE_TYPE start --core.ip $RPC_URL --p2p.network $NETWORK',1),f={class:"language-bash vp-adaptive-theme"},$=s("button",{title:"Copy Code",class:"copy"},null,-1),W=s("span",{class:"lang"},"bash",-1),Y={class:"shiki github-dark vp-code-dark"},w=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),K={class:"line"},M=s("span",{style:{color:"#E1E4E8"}}," ",-1),x={style:{color:"#9ECBFF"}},L=s("span",{style:{color:"#E1E4E8"}}," ",-1),q=s("span",{style:{color:"#79B8FF"}},"\\",-1),H=o(' celestia $NODE_TYPE start --core.ip $RPC_URL --p2p.network $NETWORK',1),U={class:"shiki github-light vp-code-light"},Q=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),j={class:"line"},G=s("span",{style:{color:"#24292E"}}," ",-1),z={style:{color:"#032F62"}},Z=s("span",{style:{color:"#24292E"}}," ",-1),J=s("span",{style:{color:"#005CC5"}},"\\",-1),X=o(' celestia $NODE_TYPE start --core.ip $RPC_URL --p2p.network $NETWORK',1),ss={class:"language-bash vp-adaptive-theme"},os=s("button",{title:"Copy Code",class:"copy"},null,-1),as=s("span",{class:"lang"},"bash",-1),ns={class:"shiki github-dark vp-code-dark"},es=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),ls={class:"line"},ts=s("span",{style:{color:"#E1E4E8"}}," ",-1),ps={style:{color:"#9ECBFF"}},cs=s("span",{style:{color:"#E1E4E8"}}," ",-1),rs=s("span",{style:{color:"#79B8FF"}},"\\",-1),_s=o(' celestia $NODE_TYPE start --core.ip $RPC_URL --p2p.network $NETWORK',1),is={class:"shiki github-light vp-code-light"},Es=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),ys={class:"line"},ds=s("span",{style:{color:"#24292E"}}," ",-1),hs={style:{color:"#032F62"}},Ts=s("span",{style:{color:"#24292E"}}," ",-1),Fs=s("span",{style:{color:"#005CC5"}},"\\",-1),Cs=o(' celestia $NODE_TYPE start --core.ip $RPC_URL --p2p.network $NETWORK',1),us=o(`

    Congratulations! You now have a celestia-node running!

    If you would like to run the node with custom flags, you can refer to the celestia-node tutorial page. Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    Light node setup with persistent storage

    If you delete a container that you started above, all data will be lost. To avoid this, you can mount a volume to the container. This will allow you to persist data even after the container is deleted.

    First, you will need to create a directory on your host machine. This directory will be used to store the data for the container. Create a directory on your host machine and give it a name. For example, you can name it my-node-store:

    bash
    cd $HOME
    +mkdir my-node-store
    cd $HOME
    +mkdir my-node-store

    Now, you can mount this directory to the container. Before mounting a volume, you may need to set permissions for the user on the host machine by running:

    bash
    sudo chown 10001:10001 $HOME/my-node-store
    sudo chown 10001:10001 $HOME/my-node-store
    bash
    # you're good to go 😎
    # you're good to go 😎

    Initialize the node store and key

    In order to mount a volume to the container, you need to specify the path to the volume. When you run your container, you can specify the path to the volume using the --volume (or -v for short) flag. In this command, we'll create our key and initialize the node store, using the variables we set in the quick start section:

    bash
    # --volume == -v [local path]:[container path]
    +docker run [args...] -v $HOME/my-node-store:/home/celestia \\
    +    celestia $NODE_TYPE init [args...]
    # --volume == -v [local path]:[container path]
    +docker run [args...] -v $HOME/my-node-store:/home/celestia \\
    +    celestia $NODE_TYPE init [args...]

    An example init command will look similar to below:

    `,12),gs={class:"vp-code-group vp-adaptive-theme"},bs=o('
    ',1),vs={class:"blocks"},Ps={class:"language-bash vp-adaptive-theme active"},ks=s("button",{title:"Copy Code",class:"copy"},null,-1),ms=s("span",{class:"lang"},"bash",-1),As={class:"shiki github-dark vp-code-dark"},Ss=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),Ns=o(' -v $HOME/my-node-store:/home/celestia \\',1),Bs={class:"line"},Ds=s("span",{style:{color:"#E1E4E8"}}," ",-1),Rs={style:{color:"#9ECBFF"}},Os=s("span",{style:{color:"#E1E4E8"}}," ",-1),Is=s("span",{style:{color:"#79B8FF"}},"\\",-1),Vs=o(' celestia light init --p2p.network $NETWORK',1),fs={class:"shiki github-light vp-code-light"},$s=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),Ws=o(' -v $HOME/my-node-store:/home/celestia \\',1),Ys={class:"line"},ws=s("span",{style:{color:"#24292E"}}," ",-1),Ks={style:{color:"#032F62"}},Ms=s("span",{style:{color:"#24292E"}}," ",-1),xs=s("span",{style:{color:"#005CC5"}},"\\",-1),Ls=o(' celestia light init --p2p.network $NETWORK',1),qs={class:"language-bash vp-adaptive-theme"},Hs=s("button",{title:"Copy Code",class:"copy"},null,-1),Us=s("span",{class:"lang"},"bash",-1),Qs={class:"shiki github-dark vp-code-dark"},js=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),Gs=o(' -v $HOME/my-node-store:/home/celestia \\',1),zs={class:"line"},Zs=s("span",{style:{color:"#E1E4E8"}}," ",-1),Js={style:{color:"#9ECBFF"}},Xs=s("span",{style:{color:"#E1E4E8"}}," ",-1),so=s("span",{style:{color:"#79B8FF"}},"\\",-1),oo=o(' celestia light init --p2p.network $NETWORK',1),ao={class:"shiki github-light vp-code-light"},no=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),eo=o(' -v $HOME/my-node-store:/home/celestia \\',1),lo={class:"line"},to=s("span",{style:{color:"#24292E"}}," ",-1),po={style:{color:"#032F62"}},co=s("span",{style:{color:"#24292E"}}," ",-1),ro=s("span",{style:{color:"#005CC5"}},"\\",-1),_o=o(' celestia light init --p2p.network $NETWORK',1),io={class:"language-bash vp-adaptive-theme"},Eo=s("button",{title:"Copy Code",class:"copy"},null,-1),yo=s("span",{class:"lang"},"bash",-1),ho={class:"shiki github-dark vp-code-dark"},To=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),Fo=o(' -v $HOME/my-node-store:/home/celestia \\',1),Co={class:"line"},uo=s("span",{style:{color:"#E1E4E8"}}," ",-1),go={style:{color:"#9ECBFF"}},bo=s("span",{style:{color:"#E1E4E8"}}," ",-1),vo=s("span",{style:{color:"#79B8FF"}},"\\",-1),Po=o(' celestia light init --p2p.network $NETWORK',1),ko={class:"shiki github-light vp-code-light"},mo=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),Ao=o(' -v $HOME/my-node-store:/home/celestia \\',1),So={class:"line"},No=s("span",{style:{color:"#24292E"}}," ",-1),Bo={style:{color:"#032F62"}},Do=s("span",{style:{color:"#24292E"}}," ",-1),Ro=s("span",{style:{color:"#005CC5"}},"\\",-1),Oo=o(' celestia light init --p2p.network $NETWORK',1),Io=o(`

    Start the node

    Run the following command to start the node:

    bash
    # --volume == -v [local path]:[container path]
    +docker run [...args] -v $HOME/my-node-store:/home/celestia \\
    +    celestia <node-type> start [...args]
    # --volume == -v [local path]:[container path]
    +docker run [...args] -v $HOME/my-node-store:/home/celestia \\
    +    celestia <node-type> start [...args]

    A full start command will look similar to below.

    `,4),Vo={class:"vp-code-group vp-adaptive-theme"},fo=o('
    ',1),$o={class:"blocks"},Wo={class:"language-bash vp-adaptive-theme active"},Yo=s("button",{title:"Copy Code",class:"copy"},null,-1),wo=s("span",{class:"lang"},"bash",-1),Ko={class:"shiki github-dark vp-code-dark"},Mo=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),xo=o(' -v $HOME/my-node-store:/home/celestia \\',1),Lo={class:"line"},qo=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ho={style:{color:"#9ECBFF"}},Uo=s("span",{style:{color:"#E1E4E8"}}," ",-1),Qo=s("span",{style:{color:"#79B8FF"}},"\\",-1),jo=o(' celestia light start --core.ip $RPC_URL',1),Go={class:"shiki github-light vp-code-light"},zo=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),Zo=o(' -v $HOME/my-node-store:/home/celestia \\',1),Jo={class:"line"},Xo=s("span",{style:{color:"#24292E"}}," ",-1),sa={style:{color:"#032F62"}},oa=s("span",{style:{color:"#24292E"}}," ",-1),aa=s("span",{style:{color:"#005CC5"}},"\\",-1),na=o(' celestia light start --core.ip $RPC_URL',1),ea={class:"language-bash vp-adaptive-theme"},la=s("button",{title:"Copy Code",class:"copy"},null,-1),ta=s("span",{class:"lang"},"bash",-1),pa={class:"shiki github-dark vp-code-dark"},ca=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),ra=o(' -v $HOME/my-node-store:/home/celestia \\',1),_a={class:"line"},ia=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ea={style:{color:"#9ECBFF"}},ya=s("span",{style:{color:"#E1E4E8"}}," ",-1),da=s("span",{style:{color:"#79B8FF"}},"\\",-1),ha=o(' celestia light start --core.ip $RPC_URL',1),Ta={class:"shiki github-light vp-code-light"},Fa=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),Ca=o(' -v $HOME/my-node-store:/home/celestia \\',1),ua={class:"line"},ga=s("span",{style:{color:"#24292E"}}," ",-1),ba={style:{color:"#032F62"}},va=s("span",{style:{color:"#24292E"}}," ",-1),Pa=s("span",{style:{color:"#005CC5"}},"\\",-1),ka=o(' celestia light start --core.ip $RPC_URL',1),ma={class:"language-bash vp-adaptive-theme"},Aa=s("button",{title:"Copy Code",class:"copy"},null,-1),Sa=s("span",{class:"lang"},"bash",-1),Na={class:"shiki github-dark vp-code-dark"},Ba=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),Da=o(' -v $HOME/my-node-store:/home/celestia \\',1),Ra={class:"line"},Oa=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ia={style:{color:"#9ECBFF"}},Va=s("span",{style:{color:"#E1E4E8"}}," ",-1),fa=s("span",{style:{color:"#79B8FF"}},"\\",-1),$a=o(' celestia light start --core.ip $RPC_URL',1),Wa={class:"shiki github-light vp-code-light"},Ya=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),wa=o(' -v $HOME/my-node-store:/home/celestia \\',1),Ka={class:"line"},Ma=s("span",{style:{color:"#24292E"}}," ",-1),xa={style:{color:"#032F62"}},La=s("span",{style:{color:"#24292E"}}," ",-1),qa=s("span",{style:{color:"#005CC5"}},"\\",-1),Ha=o(' celestia light start --core.ip $RPC_URL',1),Ua=s("p",null,"Congratulations! You now have a node running with persistent storage.",-1),Qa=s("h2",{id:"video-walkthrough",tabindex:"-1"},[a("Video walkthrough "),s("a",{class:"header-anchor",href:"#video-walkthrough","aria-label":'Permalink to "Video walkthrough"'},"​")],-1),ja=s("div",{class:"youtube-wrapper"},[s("iframe",{class:"youtube-video",title:"Running a Celestia light node",src:"https://youtube.com/embed/WFubhQc8tGk",allowfullscreen:""})],-1),Ga=s("h3",{id:"_2-5-minute-version",tabindex:"-1"},[a("2.5 minute version "),s("a",{class:"header-anchor",href:"#_2-5-minute-version","aria-label":'Permalink to "2.5 minute version"'},"​")],-1),za=s("div",{class:"youtube-wrapper"},[s("iframe",{class:"youtube-video",title:"Running a Celestia light node",src:"https://youtube.com/embed/ROZv871Q7RM",allowfullscreen:""})],-1),Za=s("h2",{id:"troubleshooting",tabindex:"-1"},[a("Troubleshooting "),s("a",{class:"header-anchor",href:"#troubleshooting","aria-label":'Permalink to "Troubleshooting"'},"​")],-1),Ja=s("p",null,[a("For security purposes Celestia expects to interact with the your node's keys in a read-only manner. This is enforced using linux style permissions on the filesystem. Windows NTFS does not support these types of permissions. As a result the recommended path for Windows users to mount a persisted volume is to do so within WSL. You can find "),s("a",{href:"https://learn.microsoft.com/en-us/windows/wsl/install",target:"_blank",rel:"noreferrer"},"instructions for installing WSL"),a(".")],-1),pn=JSON.parse('{"title":"🐳 Docker setup","description":"Running Celestia Node using Docker.","frontmatter":{"sidebar_label":"Docker images","description":"Running Celestia Node using Docker.","head":[["meta",{"name":"og:title","content":"🐳 Docker setup | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/docker-images.md","filePath":"nodes/docker-images.md","lastUpdated":1700056188000}'),Xa={name:"nodes/docker-images.md"},cn=Object.assign(Xa,{setup(sn){return(on,an)=>(c(),r("div",null,[_,s("ol",null,[i,s("li",null,[E,s("div",y,[d,s("div",h,[s("div",T,[F,C,s("pre",u,[s("code",null,[g,a(` +`),s("span",b,[v,s("span",P,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),k,m]),a(` +`),A])]),s("pre",S,[s("code",null,[N,a(` +`),s("span",B,[D,s("span",R,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),O,I]),a(` +`),V])])]),s("div",f,[$,W,s("pre",Y,[s("code",null,[w,a(` +`),s("span",K,[M,s("span",x,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),L,q]),a(` +`),H])]),s("pre",U,[s("code",null,[Q,a(` +`),s("span",j,[G,s("span",z,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),Z,J]),a(` +`),X])])]),s("div",ss,[os,as,s("pre",ns,[s("code",null,[es,a(` +`),s("span",ls,[ts,s("span",ps,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),cs,rs]),a(` +`),_s])]),s("pre",is,[s("code",null,[Es,a(` +`),s("span",ys,[ds,s("span",hs,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),Ts,Fs]),a(` +`),Cs])])])])])])]),us,s("div",gs,[bs,s("div",vs,[s("div",Ps,[ks,ms,s("pre",As,[s("code",null,[Ss,a(` +`),Ns,a(` +`),s("span",Bs,[Ds,s("span",Rs,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),Os,Is]),a(` +`),Vs])]),s("pre",fs,[s("code",null,[$s,a(` +`),Ws,a(` +`),s("span",Ys,[ws,s("span",Ks,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),Ms,xs]),a(` +`),Ls])])]),s("div",qs,[Hs,Us,s("pre",Qs,[s("code",null,[js,a(` +`),Gs,a(` +`),s("span",zs,[Zs,s("span",Js,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),Xs,so]),a(` +`),oo])]),s("pre",ao,[s("code",null,[no,a(` +`),eo,a(` +`),s("span",lo,[to,s("span",po,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),co,ro]),a(` +`),_o])])]),s("div",io,[Eo,yo,s("pre",ho,[s("code",null,[To,a(` +`),Fo,a(` +`),s("span",Co,[uo,s("span",go,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),bo,vo]),a(` +`),Po])]),s("pre",ko,[s("code",null,[mo,a(` +`),Ao,a(` +`),s("span",So,[No,s("span",Bo,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),Do,Ro]),a(` +`),Oo])])])])]),Io,s("div",Vo,[fo,s("div",$o,[s("div",Wo,[Yo,wo,s("pre",Ko,[s("code",null,[Mo,a(` +`),xo,a(` +`),s("span",Lo,[qo,s("span",Ho,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),Uo,Qo]),a(` +`),jo])]),s("pre",Go,[s("code",null,[zo,a(` +`),Zo,a(` +`),s("span",Jo,[Xo,s("span",sa,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),oa,aa]),a(` +`),na])])]),s("div",ea,[la,ta,s("pre",pa,[s("code",null,[ca,a(` +`),ra,a(` +`),s("span",_a,[ia,s("span",Ea,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),ya,da]),a(` +`),ha])]),s("pre",Ta,[s("code",null,[Fa,a(` +`),Ca,a(` +`),s("span",ua,[ga,s("span",ba,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),va,Pa]),a(` +`),ka])])]),s("div",ma,[Aa,Sa,s("pre",Na,[s("code",null,[Ba,a(` +`),Da,a(` +`),s("span",Ra,[Oa,s("span",Ia,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),Va,fa]),a(` +`),$a])]),s("pre",Wa,[s("code",null,[Ya,a(` +`),wa,a(` +`),s("span",Ka,[Ma,s("span",xa,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),La,qa]),a(` +`),Ha])])])])]),Ua,Qa,ja,Ga,za,Za,Ja]))}});export{pn as __pageData,cn as default}; diff --git a/assets/nodes_docker-images.md.a7ae248f.lean.js b/assets/nodes_docker-images.md.a7ae248f.lean.js new file mode 100644 index 00000000000..cb794b18558 --- /dev/null +++ b/assets/nodes_docker-images.md.a7ae248f.lean.js @@ -0,0 +1,49 @@ +import{a as l}from"./chunks/arabica_versions.d7daf483.js";import{m as t}from"./chunks/mocha_versions.3ba4a823.js";import{m as p}from"./chunks/mainnet_versions.84d9fd1c.js";import{o as c,c as r,k as s,a,t as n,l as e,Q as o}from"./chunks/framework.51d6c45b.js";const _=o("",9),i=o("",3),E=s("p",null,"Run the image from the command line:",-1),y={class:"vp-code-group vp-adaptive-theme"},d=o("",1),h={class:"blocks"},T={class:"language-bash vp-adaptive-theme active"},F=s("button",{title:"Copy Code",class:"copy"},null,-1),C=s("span",{class:"lang"},"bash",-1),u={class:"shiki github-dark vp-code-dark"},g=o("",1),b={class:"line"},v=s("span",{style:{color:"#E1E4E8"}}," ",-1),P={style:{color:"#9ECBFF"}},k=s("span",{style:{color:"#E1E4E8"}}," ",-1),m=s("span",{style:{color:"#79B8FF"}},"\\",-1),A=o("",1),S={class:"shiki github-light vp-code-light"},N=o("",1),B={class:"line"},D=s("span",{style:{color:"#24292E"}}," ",-1),R={style:{color:"#032F62"}},O=s("span",{style:{color:"#24292E"}}," ",-1),I=s("span",{style:{color:"#005CC5"}},"\\",-1),V=o("",1),f={class:"language-bash vp-adaptive-theme"},$=s("button",{title:"Copy Code",class:"copy"},null,-1),W=s("span",{class:"lang"},"bash",-1),Y={class:"shiki github-dark vp-code-dark"},w=o("",1),K={class:"line"},M=s("span",{style:{color:"#E1E4E8"}}," ",-1),x={style:{color:"#9ECBFF"}},L=s("span",{style:{color:"#E1E4E8"}}," ",-1),q=s("span",{style:{color:"#79B8FF"}},"\\",-1),H=o("",1),U={class:"shiki github-light vp-code-light"},Q=o("",1),j={class:"line"},G=s("span",{style:{color:"#24292E"}}," ",-1),z={style:{color:"#032F62"}},Z=s("span",{style:{color:"#24292E"}}," ",-1),J=s("span",{style:{color:"#005CC5"}},"\\",-1),X=o("",1),ss={class:"language-bash vp-adaptive-theme"},os=s("button",{title:"Copy Code",class:"copy"},null,-1),as=s("span",{class:"lang"},"bash",-1),ns={class:"shiki github-dark vp-code-dark"},es=o("",1),ls={class:"line"},ts=s("span",{style:{color:"#E1E4E8"}}," ",-1),ps={style:{color:"#9ECBFF"}},cs=s("span",{style:{color:"#E1E4E8"}}," ",-1),rs=s("span",{style:{color:"#79B8FF"}},"\\",-1),_s=o("",1),is={class:"shiki github-light vp-code-light"},Es=o("",1),ys={class:"line"},ds=s("span",{style:{color:"#24292E"}}," ",-1),hs={style:{color:"#032F62"}},Ts=s("span",{style:{color:"#24292E"}}," ",-1),Fs=s("span",{style:{color:"#005CC5"}},"\\",-1),Cs=o("",1),us=o("",12),gs={class:"vp-code-group vp-adaptive-theme"},bs=o("",1),vs={class:"blocks"},Ps={class:"language-bash vp-adaptive-theme active"},ks=s("button",{title:"Copy Code",class:"copy"},null,-1),ms=s("span",{class:"lang"},"bash",-1),As={class:"shiki github-dark vp-code-dark"},Ss=o("",1),Ns=o("",1),Bs={class:"line"},Ds=s("span",{style:{color:"#E1E4E8"}}," ",-1),Rs={style:{color:"#9ECBFF"}},Os=s("span",{style:{color:"#E1E4E8"}}," ",-1),Is=s("span",{style:{color:"#79B8FF"}},"\\",-1),Vs=o("",1),fs={class:"shiki github-light vp-code-light"},$s=o("",1),Ws=o("",1),Ys={class:"line"},ws=s("span",{style:{color:"#24292E"}}," ",-1),Ks={style:{color:"#032F62"}},Ms=s("span",{style:{color:"#24292E"}}," ",-1),xs=s("span",{style:{color:"#005CC5"}},"\\",-1),Ls=o("",1),qs={class:"language-bash vp-adaptive-theme"},Hs=s("button",{title:"Copy Code",class:"copy"},null,-1),Us=s("span",{class:"lang"},"bash",-1),Qs={class:"shiki github-dark vp-code-dark"},js=o("",1),Gs=o("",1),zs={class:"line"},Zs=s("span",{style:{color:"#E1E4E8"}}," ",-1),Js={style:{color:"#9ECBFF"}},Xs=s("span",{style:{color:"#E1E4E8"}}," ",-1),so=s("span",{style:{color:"#79B8FF"}},"\\",-1),oo=o("",1),ao={class:"shiki github-light vp-code-light"},no=o("",1),eo=o("",1),lo={class:"line"},to=s("span",{style:{color:"#24292E"}}," ",-1),po={style:{color:"#032F62"}},co=s("span",{style:{color:"#24292E"}}," ",-1),ro=s("span",{style:{color:"#005CC5"}},"\\",-1),_o=o("",1),io={class:"language-bash vp-adaptive-theme"},Eo=s("button",{title:"Copy Code",class:"copy"},null,-1),yo=s("span",{class:"lang"},"bash",-1),ho={class:"shiki github-dark vp-code-dark"},To=o("",1),Fo=o("",1),Co={class:"line"},uo=s("span",{style:{color:"#E1E4E8"}}," ",-1),go={style:{color:"#9ECBFF"}},bo=s("span",{style:{color:"#E1E4E8"}}," ",-1),vo=s("span",{style:{color:"#79B8FF"}},"\\",-1),Po=o("",1),ko={class:"shiki github-light vp-code-light"},mo=o("",1),Ao=o("",1),So={class:"line"},No=s("span",{style:{color:"#24292E"}}," ",-1),Bo={style:{color:"#032F62"}},Do=s("span",{style:{color:"#24292E"}}," ",-1),Ro=s("span",{style:{color:"#005CC5"}},"\\",-1),Oo=o("",1),Io=o("",4),Vo={class:"vp-code-group vp-adaptive-theme"},fo=o("",1),$o={class:"blocks"},Wo={class:"language-bash vp-adaptive-theme active"},Yo=s("button",{title:"Copy Code",class:"copy"},null,-1),wo=s("span",{class:"lang"},"bash",-1),Ko={class:"shiki github-dark vp-code-dark"},Mo=o("",1),xo=o("",1),Lo={class:"line"},qo=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ho={style:{color:"#9ECBFF"}},Uo=s("span",{style:{color:"#E1E4E8"}}," ",-1),Qo=s("span",{style:{color:"#79B8FF"}},"\\",-1),jo=o("",1),Go={class:"shiki github-light vp-code-light"},zo=o("",1),Zo=o("",1),Jo={class:"line"},Xo=s("span",{style:{color:"#24292E"}}," ",-1),sa={style:{color:"#032F62"}},oa=s("span",{style:{color:"#24292E"}}," ",-1),aa=s("span",{style:{color:"#005CC5"}},"\\",-1),na=o("",1),ea={class:"language-bash vp-adaptive-theme"},la=s("button",{title:"Copy Code",class:"copy"},null,-1),ta=s("span",{class:"lang"},"bash",-1),pa={class:"shiki github-dark vp-code-dark"},ca=o("",1),ra=o("",1),_a={class:"line"},ia=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ea={style:{color:"#9ECBFF"}},ya=s("span",{style:{color:"#E1E4E8"}}," ",-1),da=s("span",{style:{color:"#79B8FF"}},"\\",-1),ha=o("",1),Ta={class:"shiki github-light vp-code-light"},Fa=o("",1),Ca=o("",1),ua={class:"line"},ga=s("span",{style:{color:"#24292E"}}," ",-1),ba={style:{color:"#032F62"}},va=s("span",{style:{color:"#24292E"}}," ",-1),Pa=s("span",{style:{color:"#005CC5"}},"\\",-1),ka=o("",1),ma={class:"language-bash vp-adaptive-theme"},Aa=s("button",{title:"Copy Code",class:"copy"},null,-1),Sa=s("span",{class:"lang"},"bash",-1),Na={class:"shiki github-dark vp-code-dark"},Ba=o("",1),Da=o("",1),Ra={class:"line"},Oa=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ia={style:{color:"#9ECBFF"}},Va=s("span",{style:{color:"#E1E4E8"}}," ",-1),fa=s("span",{style:{color:"#79B8FF"}},"\\",-1),$a=o("",1),Wa={class:"shiki github-light vp-code-light"},Ya=o("",1),wa=o("",1),Ka={class:"line"},Ma=s("span",{style:{color:"#24292E"}}," ",-1),xa={style:{color:"#032F62"}},La=s("span",{style:{color:"#24292E"}}," ",-1),qa=s("span",{style:{color:"#005CC5"}},"\\",-1),Ha=o("",1),Ua=s("p",null,"Congratulations! You now have a node running with persistent storage.",-1),Qa=s("h2",{id:"video-walkthrough",tabindex:"-1"},[a("Video walkthrough "),s("a",{class:"header-anchor",href:"#video-walkthrough","aria-label":'Permalink to "Video walkthrough"'},"​")],-1),ja=s("div",{class:"youtube-wrapper"},[s("iframe",{class:"youtube-video",title:"Running a Celestia light node",src:"https://youtube.com/embed/WFubhQc8tGk",allowfullscreen:""})],-1),Ga=s("h3",{id:"_2-5-minute-version",tabindex:"-1"},[a("2.5 minute version "),s("a",{class:"header-anchor",href:"#_2-5-minute-version","aria-label":'Permalink to "2.5 minute version"'},"​")],-1),za=s("div",{class:"youtube-wrapper"},[s("iframe",{class:"youtube-video",title:"Running a Celestia light node",src:"https://youtube.com/embed/ROZv871Q7RM",allowfullscreen:""})],-1),Za=s("h2",{id:"troubleshooting",tabindex:"-1"},[a("Troubleshooting "),s("a",{class:"header-anchor",href:"#troubleshooting","aria-label":'Permalink to "Troubleshooting"'},"​")],-1),Ja=s("p",null,[a("For security purposes Celestia expects to interact with the your node's keys in a read-only manner. This is enforced using linux style permissions on the filesystem. Windows NTFS does not support these types of permissions. As a result the recommended path for Windows users to mount a persisted volume is to do so within WSL. You can find "),s("a",{href:"https://learn.microsoft.com/en-us/windows/wsl/install",target:"_blank",rel:"noreferrer"},"instructions for installing WSL"),a(".")],-1),pn=JSON.parse('{"title":"🐳 Docker setup","description":"Running Celestia Node using Docker.","frontmatter":{"sidebar_label":"Docker images","description":"Running Celestia Node using Docker.","head":[["meta",{"name":"og:title","content":"🐳 Docker setup | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/docker-images.md","filePath":"nodes/docker-images.md","lastUpdated":1700056188000}'),Xa={name:"nodes/docker-images.md"},cn=Object.assign(Xa,{setup(sn){return(on,an)=>(c(),r("div",null,[_,s("ol",null,[i,s("li",null,[E,s("div",y,[d,s("div",h,[s("div",T,[F,C,s("pre",u,[s("code",null,[g,a(` +`),s("span",b,[v,s("span",P,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),k,m]),a(` +`),A])]),s("pre",S,[s("code",null,[N,a(` +`),s("span",B,[D,s("span",R,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),O,I]),a(` +`),V])])]),s("div",f,[$,W,s("pre",Y,[s("code",null,[w,a(` +`),s("span",K,[M,s("span",x,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),L,q]),a(` +`),H])]),s("pre",U,[s("code",null,[Q,a(` +`),s("span",j,[G,s("span",z,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),Z,J]),a(` +`),X])])]),s("div",ss,[os,as,s("pre",ns,[s("code",null,[es,a(` +`),s("span",ls,[ts,s("span",ps,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),cs,rs]),a(` +`),_s])]),s("pre",is,[s("code",null,[Es,a(` +`),s("span",ys,[ds,s("span",hs,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),Ts,Fs]),a(` +`),Cs])])])])])])]),us,s("div",gs,[bs,s("div",vs,[s("div",Ps,[ks,ms,s("pre",As,[s("code",null,[Ss,a(` +`),Ns,a(` +`),s("span",Bs,[Ds,s("span",Rs,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),Os,Is]),a(` +`),Vs])]),s("pre",fs,[s("code",null,[$s,a(` +`),Ws,a(` +`),s("span",Ys,[ws,s("span",Ks,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),Ms,xs]),a(` +`),Ls])])]),s("div",qs,[Hs,Us,s("pre",Qs,[s("code",null,[js,a(` +`),Gs,a(` +`),s("span",zs,[Zs,s("span",Js,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),Xs,so]),a(` +`),oo])]),s("pre",ao,[s("code",null,[no,a(` +`),eo,a(` +`),s("span",lo,[to,s("span",po,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),co,ro]),a(` +`),_o])])]),s("div",io,[Eo,yo,s("pre",ho,[s("code",null,[To,a(` +`),Fo,a(` +`),s("span",Co,[uo,s("span",go,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),bo,vo]),a(` +`),Po])]),s("pre",ko,[s("code",null,[mo,a(` +`),Ao,a(` +`),s("span",So,[No,s("span",Bo,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),Do,Ro]),a(` +`),Oo])])])])]),Io,s("div",Vo,[fo,s("div",$o,[s("div",Wo,[Yo,wo,s("pre",Ko,[s("code",null,[Mo,a(` +`),xo,a(` +`),s("span",Lo,[qo,s("span",Ho,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),Uo,Qo]),a(` +`),jo])]),s("pre",Go,[s("code",null,[zo,a(` +`),Zo,a(` +`),s("span",Jo,[Xo,s("span",sa,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),oa,aa]),a(` +`),na])])]),s("div",ea,[la,ta,s("pre",pa,[s("code",null,[ca,a(` +`),ra,a(` +`),s("span",_a,[ia,s("span",Ea,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),ya,da]),a(` +`),ha])]),s("pre",Ta,[s("code",null,[Fa,a(` +`),Ca,a(` +`),s("span",ua,[ga,s("span",ba,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),va,Pa]),a(` +`),ka])])]),s("div",ma,[Aa,Sa,s("pre",Na,[s("code",null,[Ba,a(` +`),Da,a(` +`),s("span",Ra,[Oa,s("span",Ia,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),Va,fa]),a(` +`),$a])]),s("pre",Wa,[s("code",null,[Ya,a(` +`),wa,a(` +`),s("span",Ka,[Ma,s("span",xa,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),La,qa]),a(` +`),Ha])])])])]),Ua,Qa,ja,Ga,za,Za,Ja]))}});export{pn as __pageData,cn as default}; diff --git a/assets/nodes_environment.md.31fdd4c4.js b/assets/nodes_environment.md.31fdd4c4.js new file mode 100644 index 00000000000..c39267391a6 --- /dev/null +++ b/assets/nodes_environment.md.31fdd4c4.js @@ -0,0 +1,51 @@ +import{c as a}from"./chunks/constants.dc70dfe7.js";import{o as p,c as e,k as s,t as n,l as o,Q as l}from"./chunks/framework.51d6c45b.js";const t=l(`

    Development environment

    This page will go over setting up your development environment to run Celestia software. This environment can be used for development, building binaries, and running nodes.

    Install dependencies

    1. If you are on Ubuntu, first update and upgrade your OS:

      bash
      sudo apt update && sudo apt upgrade -y
      sudo apt update && sudo apt upgrade -y
      bash
      sudo yum update
      sudo yum update
    2. Install essential packages that are necessary to execute many tasks like downloading files, compiling, and monitoring the node:

      bash
      sudo apt install curl tar wget aria2 clang pkg-config libssl-dev jq build-essential \\
      +git make ncdu -y
      sudo apt install curl tar wget aria2 clang pkg-config libssl-dev jq build-essential \\
      +git make ncdu -y
      bash
      sudo yum install curl tar wget aria2 clang pkg-config libssl-dev jq build-essential \\
      +git make ncdu -y
      sudo yum install curl tar wget aria2 clang pkg-config libssl-dev jq build-essential \\
      +git make ncdu -y
      bash
      # these commands are for installing Homebrew, wget and jq
      +# follow the instructions from the output after running this command
      +/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
      +
      +# then install wget & jq
      +brew install wget && brew install jq
      # these commands are for installing Homebrew, wget and jq
      +# follow the instructions from the output after running this command
      +/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
      +
      +# then install wget & jq
      +brew install wget && brew install jq

    Install Golang

    celestia-node is written in Golang so we must install Golang to build and run our node.

    `,6),c=s("p",null,"Set the version for your desired network:",-1),r={class:"vp-code-group vp-adaptive-theme"},y=l('
    ',1),i={class:"blocks"},E={class:"language-bash vp-adaptive-theme active"},d=s("button",{title:"Copy Code",class:"copy"},null,-1),F=s("span",{class:"lang"},"bash",-1),u={class:"shiki github-dark vp-code-dark"},g={class:"line"},h=s("span",{style:{color:"#E1E4E8"}},"ver",-1),C=s("span",{style:{color:"#F97583"}},"=",-1),b={style:{color:"#9ECBFF"}},v={class:"shiki github-light vp-code-light"},m={class:"line"},B=s("span",{style:{color:"#24292E"}},"ver",-1),_=s("span",{style:{color:"#D73A49"}},"=",-1),k={style:{color:"#032F62"}},q={class:"language-bash vp-adaptive-theme"},f=s("button",{title:"Copy Code",class:"copy"},null,-1),w=s("span",{class:"lang"},"bash",-1),$={class:"shiki github-dark vp-code-dark"},A={class:"line"},z=s("span",{style:{color:"#E1E4E8"}},"ver",-1),H=s("span",{style:{color:"#F97583"}},"=",-1),D={style:{color:"#9ECBFF"}},T={class:"shiki github-light vp-code-light"},M={class:"line"},O=s("span",{style:{color:"#24292E"}},"ver",-1),x=s("span",{style:{color:"#D73A49"}},"=",-1),P={style:{color:"#032F62"}},I={class:"language-bash vp-adaptive-theme"},j=s("button",{title:"Copy Code",class:"copy"},null,-1),N=s("span",{class:"lang"},"bash",-1),S={class:"shiki github-dark vp-code-dark"},U={class:"line"},V=s("span",{style:{color:"#E1E4E8"}},"ver",-1),R=s("span",{style:{color:"#F97583"}},"=",-1),G={style:{color:"#9ECBFF"}},J={class:"shiki github-light vp-code-light"},L={class:"line"},X=s("span",{style:{color:"#24292E"}},"ver",-1),Q=s("span",{style:{color:"#D73A49"}},"=",-1),K={style:{color:"#032F62"}},Y=l(`
  • Download and install Golang:

    bash
    cd $HOME
    +wget "https://golang.org/dl/go$ver.linux-amd64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.linux-amd64.tar.gz"
    +rm "go$ver.linux-amd64.tar.gz"
    cd $HOME
    +wget "https://golang.org/dl/go$ver.linux-amd64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.linux-amd64.tar.gz"
    +rm "go$ver.linux-amd64.tar.gz"
    bash
    cd $HOME
    +wget "https://golang.org/dl/go$ver.linux-arm64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.linux-arm64.tar.gz"
    +rm "go$ver.linux-arm64.tar.gz"
    cd $HOME
    +wget "https://golang.org/dl/go$ver.linux-arm64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.linux-arm64.tar.gz"
    +rm "go$ver.linux-arm64.tar.gz"
    bash
    cd $HOME
    +wget "https://golang.org/dl/go$ver.darwin-arm64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.darwin-arm64.tar.gz"
    +rm "go$ver.darwin-arm64.tar.gz"
    cd $HOME
    +wget "https://golang.org/dl/go$ver.darwin-arm64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.darwin-arm64.tar.gz"
    +rm "go$ver.darwin-arm64.tar.gz"
    bash
    cd $HOME
    +wget "https://golang.org/dl/go$ver.darwin-amd64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.darwin-amd64.tar.gz"
    +rm "go$ver.darwin-amd64.tar.gz"
    cd $HOME
    +wget "https://golang.org/dl/go$ver.darwin-amd64.tar.gz"
    +sudo rm -rf /usr/local/go
    +sudo tar -C /usr/local -xzf "go$ver.darwin-amd64.tar.gz"
    +rm "go$ver.darwin-amd64.tar.gz"
  • Add your /usr/local/go/bin directory to your $PATH if you have not already:

    bash
    echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.bash_profile
    +source $HOME/.bash_profile
    echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.bash_profile
    +source $HOME/.bash_profile
    bash
    echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.zshrc
    +source $HOME/.zshrc
    echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.zshrc
    +source $HOME/.zshrc
  • To verify that the correct version of Go was installed correctly run:

    bash
    go version
    go version
  • `,3),Z=s("p",null,"The output will show the version installed.",-1),ps=JSON.parse('{"title":"Development environment","description":"Learn to set up your development environment to run Celestia software.","frontmatter":{"description":"Learn to set up your development environment to run Celestia software.","head":[["meta",{"name":"og:title","content":"Development environment | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/environment.md","filePath":"nodes/environment.md","lastUpdated":1707425339000}'),W={name:"nodes/environment.md"},es=Object.assign(W,{setup(ss){return(as,ns)=>(p(),e("div",null,[t,s("ol",null,[s("li",null,[c,s("div",r,[y,s("div",i,[s("div",E,[d,F,s("pre",u,[s("code",null,[s("span",g,[h,C,s("span",b,'"'+n(o(a).golangNodeMainnet)+'"',1)])])]),s("pre",v,[s("code",null,[s("span",m,[B,_,s("span",k,'"'+n(o(a).golangNodeMainnet)+'"',1)])])])]),s("div",q,[f,w,s("pre",$,[s("code",null,[s("span",A,[z,H,s("span",D,'"'+n(o(a).golangNodeMocha)+'"',1)])])]),s("pre",T,[s("code",null,[s("span",M,[O,x,s("span",P,'"'+n(o(a).golangNodeMocha)+'"',1)])])])]),s("div",I,[j,N,s("pre",S,[s("code",null,[s("span",U,[V,R,s("span",G,'"'+n(o(a).golangNodeArabica)+'"',1)])])]),s("pre",J,[s("code",null,[s("span",L,[X,Q,s("span",K,'"'+n(o(a).golangNodeArabica)+'"',1)])])])])])])]),Y]),Z]))}});export{ps as __pageData,es as default}; diff --git a/assets/nodes_environment.md.31fdd4c4.lean.js b/assets/nodes_environment.md.31fdd4c4.lean.js new file mode 100644 index 00000000000..c72a42046bd --- /dev/null +++ b/assets/nodes_environment.md.31fdd4c4.lean.js @@ -0,0 +1 @@ +import{c as a}from"./chunks/constants.dc70dfe7.js";import{o as p,c as e,k as s,t as n,l as o,Q as l}from"./chunks/framework.51d6c45b.js";const t=l("",6),c=s("p",null,"Set the version for your desired network:",-1),r={class:"vp-code-group vp-adaptive-theme"},y=l("",1),i={class:"blocks"},E={class:"language-bash vp-adaptive-theme active"},d=s("button",{title:"Copy Code",class:"copy"},null,-1),F=s("span",{class:"lang"},"bash",-1),u={class:"shiki github-dark vp-code-dark"},g={class:"line"},h=s("span",{style:{color:"#E1E4E8"}},"ver",-1),C=s("span",{style:{color:"#F97583"}},"=",-1),b={style:{color:"#9ECBFF"}},v={class:"shiki github-light vp-code-light"},m={class:"line"},B=s("span",{style:{color:"#24292E"}},"ver",-1),_=s("span",{style:{color:"#D73A49"}},"=",-1),k={style:{color:"#032F62"}},q={class:"language-bash vp-adaptive-theme"},f=s("button",{title:"Copy Code",class:"copy"},null,-1),w=s("span",{class:"lang"},"bash",-1),$={class:"shiki github-dark vp-code-dark"},A={class:"line"},z=s("span",{style:{color:"#E1E4E8"}},"ver",-1),H=s("span",{style:{color:"#F97583"}},"=",-1),D={style:{color:"#9ECBFF"}},T={class:"shiki github-light vp-code-light"},M={class:"line"},O=s("span",{style:{color:"#24292E"}},"ver",-1),x=s("span",{style:{color:"#D73A49"}},"=",-1),P={style:{color:"#032F62"}},I={class:"language-bash vp-adaptive-theme"},j=s("button",{title:"Copy Code",class:"copy"},null,-1),N=s("span",{class:"lang"},"bash",-1),S={class:"shiki github-dark vp-code-dark"},U={class:"line"},V=s("span",{style:{color:"#E1E4E8"}},"ver",-1),R=s("span",{style:{color:"#F97583"}},"=",-1),G={style:{color:"#9ECBFF"}},J={class:"shiki github-light vp-code-light"},L={class:"line"},X=s("span",{style:{color:"#24292E"}},"ver",-1),Q=s("span",{style:{color:"#D73A49"}},"=",-1),K={style:{color:"#032F62"}},Y=l("",3),Z=s("p",null,"The output will show the version installed.",-1),ps=JSON.parse('{"title":"Development environment","description":"Learn to set up your development environment to run Celestia software.","frontmatter":{"description":"Learn to set up your development environment to run Celestia software.","head":[["meta",{"name":"og:title","content":"Development environment | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/environment.md","filePath":"nodes/environment.md","lastUpdated":1707425339000}'),W={name:"nodes/environment.md"},es=Object.assign(W,{setup(ss){return(as,ns)=>(p(),e("div",null,[t,s("ol",null,[s("li",null,[c,s("div",r,[y,s("div",i,[s("div",E,[d,F,s("pre",u,[s("code",null,[s("span",g,[h,C,s("span",b,'"'+n(o(a).golangNodeMainnet)+'"',1)])])]),s("pre",v,[s("code",null,[s("span",m,[B,_,s("span",k,'"'+n(o(a).golangNodeMainnet)+'"',1)])])])]),s("div",q,[f,w,s("pre",$,[s("code",null,[s("span",A,[z,H,s("span",D,'"'+n(o(a).golangNodeMocha)+'"',1)])])]),s("pre",T,[s("code",null,[s("span",M,[O,x,s("span",P,'"'+n(o(a).golangNodeMocha)+'"',1)])])])]),s("div",I,[j,N,s("pre",S,[s("code",null,[s("span",U,[V,R,s("span",G,'"'+n(o(a).golangNodeArabica)+'"',1)])])]),s("pre",J,[s("code",null,[s("span",L,[X,Q,s("span",K,'"'+n(o(a).golangNodeArabica)+'"',1)])])])])])])]),Y]),Z]))}});export{ps as __pageData,es as default}; diff --git a/assets/nodes_full-consensus-node.md.efda1850.js b/assets/nodes_full-consensus-node.md.efda1850.js new file mode 100644 index 00000000000..273e2777712 --- /dev/null +++ b/assets/nodes_full-consensus-node.md.efda1850.js @@ -0,0 +1,77 @@ +import{c as n}from"./chunks/constants.dc70dfe7.js";import{o as l,c as p,k as s,t as e,l as t,a,Q as o}from"./chunks/framework.51d6c45b.js";const c="/img/nodes/full-consensus-node.png",i=o('

    Setting up a Celestia full consensus node

    This guide covers how to set up a full consensus node on Celestia. Full consensus nodes allow you to sync blockchain history in the Celestia consensus layer.

    full consensus node

    Hardware requirements

    The following hardware minimum requirements are recommended for running a full consensus node:

    • Memory: 8 GB RAM
    • CPU: Quad-Core
    • Disk: 250 GB SSD Storage
    • Bandwidth: 1 Gbps for Download/1 Gbps for Upload

    Running a full consensus node requires significant storage capacity to store the entire blockchain history. As of the latest recommendation, it is advisable to have at least 250 GB of SSD storage for a Celestia full consensus node if you are using pruning. If you are not using pruning, you are running an archive node, and it is recommended to have 500 GB of SSD storage. Please ensure that your storage meets this requirement to ensure smooth syncing and operation of the node.

    Setting up a full consensus node

    The following tutorial is done on an Ubuntu Linux 20.04 (LTS) x64 instance machine.

    Set up the dependencies

    Follow the instructions on installing dependencies.

    Install celestia-app

    Follow the tutorial on installing celestia-app.

    Set up the P2P networks

    Now we will set up the P2P Networks by cloning the networks repository:

    sh
    cd $HOME
    +rm -rf networks
    +git clone https://github.com/celestiaorg/networks.git
    cd $HOME
    +rm -rf networks
    +git clone https://github.com/celestiaorg/networks.git

    To initialize the network, pick a "node-name" that describes your node. Keep in mind that this might change if a new testnet is deployed.

    `,17),r={class:"vp-code-group vp-adaptive-theme"},_=o('
    ',1),d={class:"blocks"},h={class:"language-bash vp-adaptive-theme active"},y=s("button",{title:"Copy Code",class:"copy"},null,-1),E=s("span",{class:"lang"},"bash",-1),u={class:"shiki github-dark vp-code-dark"},F={class:"line"},C=o('celestia-appd init "node-name" --chain-id ',8),g={style:{color:"#9ECBFF"}},T={class:"shiki github-light vp-code-light"},b={class:"line"},S=o('celestia-appd init "node-name" --chain-id ',8),m={style:{color:"#032F62"}},A={class:"language-bash vp-adaptive-theme"},B=s("button",{title:"Copy Code",class:"copy"},null,-1),f=s("span",{class:"lang"},"bash",-1),k={class:"shiki github-dark vp-code-dark"},v={class:"line"},P=o('celestia-appd init "node-name" --chain-id ',8),q={style:{color:"#9ECBFF"}},I={class:"shiki github-light vp-code-light"},V={class:"line"},R=o('celestia-appd init "node-name" --chain-id ',8),D={style:{color:"#032F62"}},N={class:"language-bash vp-adaptive-theme"},w=s("button",{title:"Copy Code",class:"copy"},null,-1),x=s("span",{class:"lang"},"bash",-1),$={class:"shiki github-dark vp-code-dark"},M={class:"line"},O=o('celestia-appd init "node-name" --chain-id ',8),H={style:{color:"#9ECBFF"}},L={class:"shiki github-light vp-code-light"},Q={class:"line"},j=o('celestia-appd init "node-name" --chain-id ',8),G={style:{color:"#032F62"}},Y=s("p",null,[a("Download the "),s("code",null,"genesis.json"),a(" file:")],-1),J={class:"vp-code-group vp-adaptive-theme"},W=o('
    ',1),U={class:"blocks"},X={class:"language-bash vp-adaptive-theme active"},K=s("button",{title:"Copy Code",class:"copy"},null,-1),Z=s("span",{class:"lang"},"bash",-1),z={class:"shiki github-dark vp-code-dark"},ss={class:"line"},os=s("span",{style:{color:"#B392F0"}},"celestia-appd",-1),as=s("span",{style:{color:"#E1E4E8"}}," ",-1),ns=s("span",{style:{color:"#9ECBFF"}},"download-genesis",-1),es=s("span",{style:{color:"#E1E4E8"}}," ",-1),ts={style:{color:"#9ECBFF"}},ls={class:"shiki github-light vp-code-light"},ps={class:"line"},cs=s("span",{style:{color:"#6F42C1"}},"celestia-appd",-1),is=s("span",{style:{color:"#24292E"}}," ",-1),rs=s("span",{style:{color:"#032F62"}},"download-genesis",-1),_s=s("span",{style:{color:"#24292E"}}," ",-1),ds={style:{color:"#032F62"}},hs={class:"language-bash vp-adaptive-theme"},ys=s("button",{title:"Copy Code",class:"copy"},null,-1),Es=s("span",{class:"lang"},"bash",-1),us={class:"shiki github-dark vp-code-dark"},Fs={class:"line"},Cs=s("span",{style:{color:"#B392F0"}},"celestia-appd",-1),gs=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ts=s("span",{style:{color:"#9ECBFF"}},"download-genesis",-1),bs=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ss={style:{color:"#9ECBFF"}},ms={class:"shiki github-light vp-code-light"},As={class:"line"},Bs=s("span",{style:{color:"#6F42C1"}},"celestia-appd",-1),fs=s("span",{style:{color:"#24292E"}}," ",-1),ks=s("span",{style:{color:"#032F62"}},"download-genesis",-1),vs=s("span",{style:{color:"#24292E"}}," ",-1),Ps={style:{color:"#032F62"}},qs={class:"language-bash vp-adaptive-theme"},Is=s("button",{title:"Copy Code",class:"copy"},null,-1),Vs=s("span",{class:"lang"},"bash",-1),Rs={class:"shiki github-dark vp-code-dark"},Ds={class:"line"},Ns=s("span",{style:{color:"#B392F0"}},"celestia-appd",-1),ws=s("span",{style:{color:"#E1E4E8"}}," ",-1),xs=s("span",{style:{color:"#9ECBFF"}},"download-genesis",-1),$s=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ms={style:{color:"#9ECBFF"}},Os={class:"shiki github-light vp-code-light"},Hs={class:"line"},Ls=s("span",{style:{color:"#6F42C1"}},"celestia-appd",-1),Qs=s("span",{style:{color:"#24292E"}}," ",-1),js=s("span",{style:{color:"#032F62"}},"download-genesis",-1),Gs=s("span",{style:{color:"#24292E"}}," ",-1),Ys={style:{color:"#032F62"}},Js=s("p",null,[a("Set seeds in the "),s("code",null,"$HOME/.celestia-app/config/config.toml"),a(" file:")],-1),Ws={class:"vp-code-group vp-adaptive-theme"},Us=o('
    ',1),Xs={class:"blocks"},Ks={class:"language-bash vp-adaptive-theme active"},Zs=s("button",{title:"Copy Code",class:"copy"},null,-1),zs=s("span",{class:"lang"},"bash",-1),so={class:"shiki github-dark vp-code-dark"},oo={class:"line"},ao=o('SEEDS=$(curl -sL',6),no={style:{color:"#9ECBFF"}},eo=s("span",{style:{color:"#F97583"}},"|",-1),to=s("span",{style:{color:"#9ECBFF"}}," ",-1),lo=s("span",{style:{color:"#B392F0"}},"tr",-1),po=s("span",{style:{color:"#9ECBFF"}}," '\\n' ',')",-1),co=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $SEEDS")],-1),io=o('sed -i.bak -e "s/^seeds *=.*/seeds = \\"$SEEDS\\"/" $HOME/.celestia-app/config/config.toml',1),ro={class:"shiki github-light vp-code-light"},_o={class:"line"},ho=o('SEEDS=$(curl -sL',6),yo={style:{color:"#032F62"}},Eo=s("span",{style:{color:"#D73A49"}},"|",-1),uo=s("span",{style:{color:"#032F62"}}," ",-1),Fo=s("span",{style:{color:"#6F42C1"}},"tr",-1),Co=s("span",{style:{color:"#032F62"}}," '\\n' ',')",-1),go=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $SEEDS")],-1),To=o('sed -i.bak -e "s/^seeds *=.*/seeds = \\"$SEEDS\\"/" $HOME/.celestia-app/config/config.toml',1),bo={class:"language-bash vp-adaptive-theme"},So=s("button",{title:"Copy Code",class:"copy"},null,-1),mo=s("span",{class:"lang"},"bash",-1),Ao={class:"shiki github-dark vp-code-dark"},Bo={class:"line"},fo=o('SEEDS=$(curl -sL',6),ko={style:{color:"#9ECBFF"}},vo=s("span",{style:{color:"#F97583"}},"|",-1),Po=s("span",{style:{color:"#9ECBFF"}}," ",-1),qo=s("span",{style:{color:"#B392F0"}},"tr",-1),Io=s("span",{style:{color:"#9ECBFF"}}," '\\n' ',')",-1),Vo=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $SEEDS")],-1),Ro=o('sed -i.bak -e "s/^seeds *=.*/seeds = \\"$SEEDS\\"/" $HOME/.celestia-app/config/config.toml',1),Do={class:"shiki github-light vp-code-light"},No={class:"line"},wo=o('SEEDS=$(curl -sL',6),xo={style:{color:"#032F62"}},$o=s("span",{style:{color:"#D73A49"}},"|",-1),Mo=s("span",{style:{color:"#032F62"}}," ",-1),Oo=s("span",{style:{color:"#6F42C1"}},"tr",-1),Ho=s("span",{style:{color:"#032F62"}}," '\\n' ',')",-1),Lo=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $SEEDS")],-1),Qo=o('sed -i.bak -e "s/^seeds *=.*/seeds = \\"$SEEDS\\"/" $HOME/.celestia-app/config/config.toml',1),jo=o(`
    bash
    # For Arabica, you can set seeds manually in the
    +# \`$HOME/.celestia-app/config/config.toml\` file:
    +# Comma separated list of seed nodes to connect to
    +seeds = ""
    # For Arabica, you can set seeds manually in the
    +# \`$HOME/.celestia-app/config/config.toml\` file:
    +# Comma separated list of seed nodes to connect to
    +seeds = ""
    `,1),Go=s("p",null,[s("strong",null,"Optionally"),a(", you can set persistent peers in your "),s("code",null,"config.toml"),a(" file. You can get the persistent peers from the networks repository with the following commands:")],-1),Yo=s("p",null,"Setting persistent peers is advised only if you are running a sentry node.",-1),Jo={class:"vp-code-group vp-adaptive-theme"},Wo=o('
    ',1),Uo={class:"blocks"},Xo={class:"language-bash vp-adaptive-theme active"},Ko=s("button",{title:"Copy Code",class:"copy"},null,-1),Zo=s("span",{class:"lang"},"bash",-1),zo={class:"shiki github-dark vp-code-dark"},sa={class:"line"},oa=o('PERSISTENT_PEERS=$(curl -sL',6),aa={style:{color:"#9ECBFF"}},na=s("span",{style:{color:"#F97583"}},"|",-1),ea=s("span",{style:{color:"#9ECBFF"}}," ",-1),ta=s("span",{style:{color:"#B392F0"}},"tr",-1),la=s("span",{style:{color:"#9ECBFF"}}," '\\n' ',')",-1),pa=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $PERSISTENT_PEERS")],-1),ca=o('sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \\"$PERSISTENT_PEERS\\"/" $HOME/.celestia-app/config/config.toml',1),ia={class:"shiki github-light vp-code-light"},ra={class:"line"},_a=o('PERSISTENT_PEERS=$(curl -sL',6),da={style:{color:"#032F62"}},ha=s("span",{style:{color:"#D73A49"}},"|",-1),ya=s("span",{style:{color:"#032F62"}}," ",-1),Ea=s("span",{style:{color:"#6F42C1"}},"tr",-1),ua=s("span",{style:{color:"#032F62"}}," '\\n' ',')",-1),Fa=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $PERSISTENT_PEERS")],-1),Ca=o('sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \\"$PERSISTENT_PEERS\\"/" $HOME/.celestia-app/config/config.toml',1),ga={class:"language-bash vp-adaptive-theme"},Ta=s("button",{title:"Copy Code",class:"copy"},null,-1),ba=s("span",{class:"lang"},"bash",-1),Sa={class:"shiki github-dark vp-code-dark"},ma={class:"line"},Aa=o('PERSISTENT_PEERS=$(curl -sL',6),Ba={style:{color:"#9ECBFF"}},fa=s("span",{style:{color:"#F97583"}},"|",-1),ka=s("span",{style:{color:"#9ECBFF"}}," ",-1),va=s("span",{style:{color:"#B392F0"}},"tr",-1),Pa=s("span",{style:{color:"#9ECBFF"}}," '\\n' ',')",-1),qa=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $PERSISTENT_PEERS")],-1),Ia=o('sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \\"$PERSISTENT_PEERS\\"/" $HOME/.celestia-app/config/config.toml',1),Va={class:"shiki github-light vp-code-light"},Ra={class:"line"},Da=o('PERSISTENT_PEERS=$(curl -sL',6),Na={style:{color:"#032F62"}},wa=s("span",{style:{color:"#D73A49"}},"|",-1),xa=s("span",{style:{color:"#032F62"}}," ",-1),$a=s("span",{style:{color:"#6F42C1"}},"tr",-1),Ma=s("span",{style:{color:"#032F62"}}," '\\n' ',')",-1),Oa=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $PERSISTENT_PEERS")],-1),Ha=o('sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \\"$PERSISTENT_PEERS\\"/" $HOME/.celestia-app/config/config.toml',1),La={class:"language-bash vp-adaptive-theme"},Qa=s("button",{title:"Copy Code",class:"copy"},null,-1),ja=s("span",{class:"lang"},"bash",-1),Ga={class:"shiki github-dark vp-code-dark"},Ya={class:"line"},Ja=o('PERSISTENT_PEERS=$(curl -sL',6),Wa={style:{color:"#9ECBFF"}},Ua=s("span",{style:{color:"#F97583"}},"|",-1),Xa=s("span",{style:{color:"#9ECBFF"}}," ",-1),Ka=s("span",{style:{color:"#B392F0"}},"tr",-1),Za=s("span",{style:{color:"#9ECBFF"}}," '\\n' ',')",-1),za=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $PERSISTENT_PEERS")],-1),sn=o('sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \\"$PERSISTENT_PEERS\\"/" $HOME/.celestia-app/config/config.toml',1),on={class:"shiki github-light vp-code-light"},an={class:"line"},nn=o('PERSISTENT_PEERS=$(curl -sL',6),en={style:{color:"#032F62"}},tn=s("span",{style:{color:"#D73A49"}},"|",-1),ln=s("span",{style:{color:"#032F62"}}," ",-1),pn=s("span",{style:{color:"#6F42C1"}},"tr",-1),cn=s("span",{style:{color:"#032F62"}}," '\\n' ',')",-1),rn=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $PERSISTENT_PEERS")],-1),_n=o('sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \\"$PERSISTENT_PEERS\\"/" $HOME/.celestia-app/config/config.toml',1),dn=o(`

    Storage and pruning configurations

    Connecting a consensus node to a bridge node

    If your consensus node is being connected to a celestia-node bridge node, you will need to enable transaction indexing and retain all block data. This can be achieved with the following settings in your config.toml.

    Enable transaction indexing

    toml
    indexer = "kv"
    indexer = "kv"

    Retain all block data

    And in your app.toml, min-retain-blocks should remain as the default setting of 0:

    toml
    min-retain-blocks = 0 # retain all block data, this is default setting
    min-retain-blocks = 0 # retain all block data, this is default setting

    Querying transactions by hash

    To query transactions using their hash, transaction indexing must be turned on. Set the indexer to "kv" in your config.toml:

    toml
    indexer = "kv"
    indexer = "kv"

    Accessing historical state

    If you want to query the historical state — for example, you might want to know the balance of a Celestia wallet at a given height in the past — you should run an archive node with pruning = "nothing" in your app.toml. Note that this configuration is resource-intensive and will require significant storage:

    toml
    pruning = "nothing"
    pruning = "nothing"

    Saving on storage requirements

    If you want to save on storage requirements, consider using pruning = "everything" in your app.toml to prune everything. If you select "everything" or "default", but still want to keep the block data, you can do so by not changing the default value of min-retain-blocks = 0 in your app.toml. A value of 0 for min-retain-blocks will keep all block data. This will prune snapshots of the state, but it will keep block data:

    toml
    pruning = "everything"
    +min-retain-blocks = 0 # this is the default setting
    pruning = "everything"
    +min-retain-blocks = 0 # this is the default setting

    Syncing

    By default, a consensus node will sync using block sync; that is request, validate and execute every block up to the head of the blockchain. This is the most secure mechanism yet the slowest (taking up to days depending on the height of the blockchain).

    There are two alternatives for quicker syncing.

    State sync

    State sync uses light client verification to verify state snapshots from peers and then apply them. State sync relies on weak subjectivity; a trusted header (specifically the hash and height) must be provided. This can be found by querying a trusted RPC endpoint (/block). RPC endpoints are also required for retrieving light blocks. These can be found in the docs here under the respective networks or from the chain-registry.

    In $HOME/.celestia-app/config/config.toml, set

    toml
    rpc_servers = ""
    +trust_height = 0
    +trust_hash = ""
    rpc_servers = ""
    +trust_height = 0
    +trust_hash = ""

    To their respective fields. At least two different rpc endpoints should be provided. The more, the greater the chance of detecting any fraudulent behavior.

    Once setup, you should be ready to start the node as normal. In the logs, you should see: Discovering snapshots. This may take a few minutes before snapshots are found depending on the network topology.

    Quick sync

    Quick sync effectively downloads the entire data directory from a third-party provider meaning the node has all the application and blockchain state as the node it was copied from.

    Run the following command to quick-sync from a snapshot:

    `,29),hn={class:"vp-code-group vp-adaptive-theme"},yn=o('
    ',1),En={class:"blocks"},un={class:"language-bash vp-adaptive-theme active"},Fn=s("button",{title:"Copy Code",class:"copy"},null,-1),Cn=s("span",{class:"lang"},"bash",-1),gn={class:"shiki github-dark vp-code-dark"},Tn=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"cd"),s("span",{style:{color:"#E1E4E8"}}," $HOME")],-1),bn=o('rm -rf ~/.celestia-app/data',1),Sn=o('mkdir -p ~/.celestia-app/data',1),mn=o('SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \\',1),An={class:"line"},Bn=s("span",{style:{color:"#9ECBFF"}}," ",-1),fn=s("span",{style:{color:"#B392F0"}},"egrep",-1),kn=s("span",{style:{color:"#9ECBFF"}}," ",-1),vn=s("span",{style:{color:"#79B8FF"}},"-o",-1),Pn={style:{color:"#9ECBFF"}},qn=o('| tr -d ">")',6),In=o('aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"',1),Vn=o('tar xf celestia-snap.tar -C ~/.celestia-app/data/',1),Rn={class:"shiki github-light vp-code-light"},Dn=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"cd"),s("span",{style:{color:"#24292E"}}," $HOME")],-1),Nn=o('rm -rf ~/.celestia-app/data',1),wn=o('mkdir -p ~/.celestia-app/data',1),xn=o('SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \\',1),$n={class:"line"},Mn=s("span",{style:{color:"#032F62"}}," ",-1),On=s("span",{style:{color:"#6F42C1"}},"egrep",-1),Hn=s("span",{style:{color:"#032F62"}}," ",-1),Ln=s("span",{style:{color:"#005CC5"}},"-o",-1),Qn={style:{color:"#032F62"}},jn=o('| tr -d ">")',6),Gn=o('aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"',1),Yn=o('tar xf celestia-snap.tar -C ~/.celestia-app/data/',1),Jn={class:"language-bash vp-adaptive-theme"},Wn=s("button",{title:"Copy Code",class:"copy"},null,-1),Un=s("span",{class:"lang"},"bash",-1),Xn={class:"shiki github-dark vp-code-dark"},Kn=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"cd"),s("span",{style:{color:"#E1E4E8"}}," $HOME")],-1),Zn=o('rm -rf ~/.celestia-app/data',1),zn=o('mkdir -p ~/.celestia-app/data',1),se=o('SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \\',1),oe={class:"line"},ae=s("span",{style:{color:"#9ECBFF"}}," ",-1),ne=s("span",{style:{color:"#B392F0"}},"egrep",-1),ee=s("span",{style:{color:"#9ECBFF"}}," ",-1),te=s("span",{style:{color:"#79B8FF"}},"-o",-1),le={style:{color:"#9ECBFF"}},pe=o('| tr -d ">")',6),ce=o('aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"',1),ie=o('tar xf celestia-snap.tar -C ~/.celestia-app/data/',1),re={class:"shiki github-light vp-code-light"},_e=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"cd"),s("span",{style:{color:"#24292E"}}," $HOME")],-1),de=o('rm -rf ~/.celestia-app/data',1),he=o('mkdir -p ~/.celestia-app/data',1),ye=o('SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \\',1),Ee={class:"line"},ue=s("span",{style:{color:"#032F62"}}," ",-1),Fe=s("span",{style:{color:"#6F42C1"}},"egrep",-1),Ce=s("span",{style:{color:"#032F62"}}," ",-1),ge=s("span",{style:{color:"#005CC5"}},"-o",-1),Te={style:{color:"#032F62"}},be=o('| tr -d ">")',6),Se=o('aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"',1),me=o('tar xf celestia-snap.tar -C ~/.celestia-app/data/',1),Ae={class:"language-bash vp-adaptive-theme"},Be=s("button",{title:"Copy Code",class:"copy"},null,-1),fe=s("span",{class:"lang"},"bash",-1),ke={class:"shiki github-dark vp-code-dark"},ve=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"cd"),s("span",{style:{color:"#E1E4E8"}}," $HOME")],-1),Pe=o('rm -rf ~/.celestia-app/data',1),qe=o('mkdir -p ~/.celestia-app/data',1),Ie=o('SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \\',1),Ve={class:"line"},Re=s("span",{style:{color:"#9ECBFF"}}," ",-1),De=s("span",{style:{color:"#B392F0"}},"egrep",-1),Ne=s("span",{style:{color:"#9ECBFF"}}," ",-1),we=s("span",{style:{color:"#79B8FF"}},"-o",-1),xe={style:{color:"#9ECBFF"}},$e=o('| tr -d ">")',6),Me=o('aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"',1),Oe=o('tar xf celestia-snap.tar -C ~/.celestia-app/data/',1),He={class:"shiki github-light vp-code-light"},Le=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"cd"),s("span",{style:{color:"#24292E"}}," $HOME")],-1),Qe=o('rm -rf ~/.celestia-app/data',1),je=o('mkdir -p ~/.celestia-app/data',1),Ge=o('SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \\',1),Ye={class:"line"},Je=s("span",{style:{color:"#032F62"}}," ",-1),We=s("span",{style:{color:"#6F42C1"}},"egrep",-1),Ue=s("span",{style:{color:"#032F62"}}," ",-1),Xe=s("span",{style:{color:"#005CC5"}},"-o",-1),Ke={style:{color:"#032F62"}},Ze=o('| tr -d ">")',6),ze=o('aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"',1),st=o('tar xf celestia-snap.tar -C ~/.celestia-app/data/',1),ot=o(`

    Start the consensus node

    In order to start your full consensus node, run the following:

    sh
    celestia-appd start
    celestia-appd start

    Optional: If you would like celestia-app to run as a background process, you can follow the SystemD tutorial.

    TIP

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    Extra resources for consensus nodes

    Optional: Reset network

    This will delete all data folders so we can start fresh:

    sh
    celestia-appd tendermint unsafe-reset-all --home $HOME/.celestia-app
    celestia-appd tendermint unsafe-reset-all --home $HOME/.celestia-app

    Optional: Configuring an RPC endpoint

    You can configure your full consensus node to be a public RPC endpoint. This allows it to accept connections from data availability nodes and serve requests for the data availability API.

    Expose RPC

    By default, the RPC service listens on localhost which means it can't be accessed from other machines. To make the RPC service available publicly, you need to bind it to a public IP or 0.0.0.0 (which means listening on all available network interfaces).

    You can do this by editing the config.toml file:

    sh
    sed -i 's#"tcp://127.0.0.1:26657"#"tcp://0.0.0.0:26657"#g' ~/.celestia-app/config/config.toml
    sed -i 's#"tcp://127.0.0.1:26657"#"tcp://0.0.0.0:26657"#g' ~/.celestia-app/config/config.toml

    This command replaces the localhost IP address with 0.0.0.0, making the RPC service listen on all available network interfaces.

    Note on external-address

    The external-address field in the configuration is used when your node is behind a NAT and you need to advertise a different address for peers to dial. Populating this field is not necessary for making the RPC endpoint public.

    sh
    EXTERNAL-ADDRESS=$(wget -qO- eth0.me)
    +sed -i.bak -e "s/^external-address = ""/external-address = "$EXTERNAL-ADDRESS:26656"/" \\
    +    $HOME/.celestia-app/config/config.toml
    EXTERNAL-ADDRESS=$(wget -qO- eth0.me)
    +sed -i.bak -e "s/^external-address = ""/external-address = "$EXTERNAL-ADDRESS:26656"/" \\
    +    $HOME/.celestia-app/config/config.toml

    Restart the node

    After making these changes, restart celestia-appd to load the new configurations.

    Optional: Transaction indexer configuration options

    This section guides you on how to configure your config.toml file in celestia-app to select which transactions to index. Depending on the application's configuration, a node operator may decide which transactions to index.

    The available options are:

    1. null: This option disables indexing. If you don't need to query transactions, you can choose this option to save space.
    2. kv (default): This is the simplest indexer, backed by key-value storage (defaults to levelDB; see DBBackend). When kv is chosen, tx.height and tx.hash will always be indexed. This option is suitable for basic queries on transactions.
    3. psql: This indexer is backed by PostgreSQL. When psql is chosen, tx.height and tx.hash will always be indexed. This option is suitable for complex queries on transactions.

    An example to set the value to kv in config.toml is:

    toml
    indexer = "kv"
    indexer = "kv"

    Remember to restart celestia-appd after making changes to the configuration to load the new settings.

    Optional: Discard ABCI responses configuration

    This section will guide you on how to configure your config.toml file in celestia-app to manage the storage of ABCI responses. ABCI responses are the results of executing transactions and are used for /block_results RPC queries and to reindex events in the command-line tool.

    The discard_abci_responses option allows you to control whether these responses are persisted in the state store:

    • false (default): ABCI responses are stored in the state store. This ensures that ABCI responses are available for /block_results RPC queries and for reindexing events. However, it can consume a significant amount of disk space.
    • true: ABCI responses are not stored in the state store. This can save a considerable amount of disk space, but /block_results RPC queries and event reindexing will not be available.

    An example to set the value to false in config.toml is:

    toml
    discard_abci_responses = false
    discard_abci_responses = false

    Remember to restart celestia-appd after making changes to the configuration to load the new settings.

    `,35),ct=JSON.parse('{"title":"Setting up a Celestia full consensus node","description":"Learn how to set up a Celestia full consensus node.","frontmatter":{"description":"Learn how to set up a Celestia full consensus node.","outline":"deep","head":[["meta",{"name":"og:title","content":"Setting up a Celestia full consensus node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/full-consensus-node.md","filePath":"nodes/full-consensus-node.md","lastUpdated":1711744858000}'),at={name:"nodes/full-consensus-node.md"},it=Object.assign(at,{setup(nt){return(et,tt)=>(l(),p("div",null,[i,s("div",r,[_,s("div",d,[s("div",h,[y,E,s("pre",u,[s("code",null,[s("span",F,[C,s("span",g,e(t(n).mainnetChainId),1)])])]),s("pre",T,[s("code",null,[s("span",b,[S,s("span",m,e(t(n).mainnetChainId),1)])])])]),s("div",A,[B,f,s("pre",k,[s("code",null,[s("span",v,[P,s("span",q,e(t(n).mochaChainId),1)])])]),s("pre",I,[s("code",null,[s("span",V,[R,s("span",D,e(t(n).mochaChainId),1)])])])]),s("div",N,[w,x,s("pre",$,[s("code",null,[s("span",M,[O,s("span",H,e(t(n).arabicaChainId),1)])])]),s("pre",L,[s("code",null,[s("span",Q,[j,s("span",G,e(t(n).arabicaChainId),1)])])])])])]),Y,s("div",J,[W,s("div",U,[s("div",X,[K,Z,s("pre",z,[s("code",null,[s("span",ss,[os,as,ns,es,s("span",ts,e(t(n).mainnetChainId),1)])])]),s("pre",ls,[s("code",null,[s("span",ps,[cs,is,rs,_s,s("span",ds,e(t(n).mainnetChainId),1)])])])]),s("div",hs,[ys,Es,s("pre",us,[s("code",null,[s("span",Fs,[Cs,gs,Ts,bs,s("span",Ss,e(t(n).mochaChainId),1)])])]),s("pre",ms,[s("code",null,[s("span",As,[Bs,fs,ks,vs,s("span",Ps,e(t(n).mochaChainId),1)])])])]),s("div",qs,[Is,Vs,s("pre",Rs,[s("code",null,[s("span",Ds,[Ns,ws,xs,$s,s("span",Ms,e(t(n).arabicaChainId),1)])])]),s("pre",Os,[s("code",null,[s("span",Hs,[Ls,Qs,js,Gs,s("span",Ys,e(t(n).arabicaChainId),1)])])])])])]),Js,s("div",Ws,[Us,s("div",Xs,[s("div",Ks,[Zs,zs,s("pre",so,[s("code",null,[s("span",oo,[ao,s("span",no," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mainnetChainId)+"/seeds.txt ",1),eo,to,lo,po]),a(` +`),co,a(` +`),io])]),s("pre",ro,[s("code",null,[s("span",_o,[ho,s("span",yo," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mainnetChainId)+"/seeds.txt ",1),Eo,uo,Fo,Co]),a(` +`),go,a(` +`),To])])]),s("div",bo,[So,mo,s("pre",Ao,[s("code",null,[s("span",Bo,[fo,s("span",ko," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mochaChainId)+"/seeds.txt ",1),vo,Po,qo,Io]),a(` +`),Vo,a(` +`),Ro])]),s("pre",Do,[s("code",null,[s("span",No,[wo,s("span",xo," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mochaChainId)+"/seeds.txt ",1),$o,Mo,Oo,Ho]),a(` +`),Lo,a(` +`),Qo])])]),jo])]),Go,Yo,s("div",Jo,[Wo,s("div",Uo,[s("div",Xo,[Ko,Zo,s("pre",zo,[s("code",null,[s("span",sa,[oa,s("span",aa," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mainnetChainId)+"/peers.txt ",1),na,ea,ta,la]),a(` +`),pa,a(` +`),ca])]),s("pre",ia,[s("code",null,[s("span",ra,[_a,s("span",da," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mainnetChainId)+"/peers.txt ",1),ha,ya,Ea,ua]),a(` +`),Fa,a(` +`),Ca])])]),s("div",ga,[Ta,ba,s("pre",Sa,[s("code",null,[s("span",ma,[Aa,s("span",Ba," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mochaChainId)+"/peers.txt ",1),fa,ka,va,Pa]),a(` +`),qa,a(` +`),Ia])]),s("pre",Va,[s("code",null,[s("span",Ra,[Da,s("span",Na," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mochaChainId)+"/peers.txt ",1),wa,xa,$a,Ma]),a(` +`),Oa,a(` +`),Ha])])]),s("div",La,[Qa,ja,s("pre",Ga,[s("code",null,[s("span",Ya,[Ja,s("span",Wa," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).arabicaChainId)+"/peers.txt ",1),Ua,Xa,Ka,Za]),a(` +`),za,a(` +`),sn])]),s("pre",on,[s("code",null,[s("span",an,[nn,s("span",en," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).arabicaChainId)+"/peers.txt ",1),tn,ln,pn,cn]),a(` +`),rn,a(` +`),_n])])])])]),dn,s("div",hn,[yn,s("div",En,[s("div",un,[Fn,Cn,s("pre",gn,[s("code",null,[Tn,a(` +`),bn,a(` +`),Sn,a(` +`),mn,a(` +`),s("span",An,[Bn,fn,kn,vn,s("span",Pn,' ">'+e(t(n).mainnetChainId)+'.*tar" ',1),qn]),a(` +`),In,a(` +`),Vn])]),s("pre",Rn,[s("code",null,[Dn,a(` +`),Nn,a(` +`),wn,a(` +`),xn,a(` +`),s("span",$n,[Mn,On,Hn,Ln,s("span",Qn,' ">'+e(t(n).mainnetChainId)+'.*tar" ',1),jn]),a(` +`),Gn,a(` +`),Yn])])]),s("div",Jn,[Wn,Un,s("pre",Xn,[s("code",null,[Kn,a(` +`),Zn,a(` +`),zn,a(` +`),se,a(` +`),s("span",oe,[ae,ne,ee,te,s("span",le,' ">'+e(t(n).mochaChainId)+'.*tar" ',1),pe]),a(` +`),ce,a(` +`),ie])]),s("pre",re,[s("code",null,[_e,a(` +`),de,a(` +`),he,a(` +`),ye,a(` +`),s("span",Ee,[ue,Fe,Ce,ge,s("span",Te,' ">'+e(t(n).mochaChainId)+'.*tar" ',1),be]),a(` +`),Se,a(` +`),me])])]),s("div",Ae,[Be,fe,s("pre",ke,[s("code",null,[ve,a(` +`),Pe,a(` +`),qe,a(` +`),Ie,a(` +`),s("span",Ve,[Re,De,Ne,we,s("span",xe,' ">'+e(t(n).arabicaChainId)+'.*tar" ',1),$e]),a(` +`),Me,a(` +`),Oe])]),s("pre",He,[s("code",null,[Le,a(` +`),Qe,a(` +`),je,a(` +`),Ge,a(` +`),s("span",Ye,[Je,We,Ue,Xe,s("span",Ke,' ">'+e(t(n).arabicaChainId)+'.*tar" ',1),Ze]),a(` +`),ze,a(` +`),st])])])])]),ot]))}});export{ct as __pageData,it as default}; diff --git a/assets/nodes_full-consensus-node.md.efda1850.lean.js b/assets/nodes_full-consensus-node.md.efda1850.lean.js new file mode 100644 index 00000000000..44340113371 --- /dev/null +++ b/assets/nodes_full-consensus-node.md.efda1850.lean.js @@ -0,0 +1,57 @@ +import{c as n}from"./chunks/constants.dc70dfe7.js";import{o as l,c as p,k as s,t as e,l as t,a,Q as o}from"./chunks/framework.51d6c45b.js";const c="/img/nodes/full-consensus-node.png",i=o("",17),r={class:"vp-code-group vp-adaptive-theme"},_=o("",1),d={class:"blocks"},h={class:"language-bash vp-adaptive-theme active"},y=s("button",{title:"Copy Code",class:"copy"},null,-1),E=s("span",{class:"lang"},"bash",-1),u={class:"shiki github-dark vp-code-dark"},F={class:"line"},C=o("",8),g={style:{color:"#9ECBFF"}},T={class:"shiki github-light vp-code-light"},b={class:"line"},S=o("",8),m={style:{color:"#032F62"}},A={class:"language-bash vp-adaptive-theme"},B=s("button",{title:"Copy Code",class:"copy"},null,-1),f=s("span",{class:"lang"},"bash",-1),k={class:"shiki github-dark vp-code-dark"},v={class:"line"},P=o("",8),q={style:{color:"#9ECBFF"}},I={class:"shiki github-light vp-code-light"},V={class:"line"},R=o("",8),D={style:{color:"#032F62"}},N={class:"language-bash vp-adaptive-theme"},w=s("button",{title:"Copy Code",class:"copy"},null,-1),x=s("span",{class:"lang"},"bash",-1),$={class:"shiki github-dark vp-code-dark"},M={class:"line"},O=o("",8),H={style:{color:"#9ECBFF"}},L={class:"shiki github-light vp-code-light"},Q={class:"line"},j=o("",8),G={style:{color:"#032F62"}},Y=s("p",null,[a("Download the "),s("code",null,"genesis.json"),a(" file:")],-1),J={class:"vp-code-group vp-adaptive-theme"},W=o("",1),U={class:"blocks"},X={class:"language-bash vp-adaptive-theme active"},K=s("button",{title:"Copy Code",class:"copy"},null,-1),Z=s("span",{class:"lang"},"bash",-1),z={class:"shiki github-dark vp-code-dark"},ss={class:"line"},os=s("span",{style:{color:"#B392F0"}},"celestia-appd",-1),as=s("span",{style:{color:"#E1E4E8"}}," ",-1),ns=s("span",{style:{color:"#9ECBFF"}},"download-genesis",-1),es=s("span",{style:{color:"#E1E4E8"}}," ",-1),ts={style:{color:"#9ECBFF"}},ls={class:"shiki github-light vp-code-light"},ps={class:"line"},cs=s("span",{style:{color:"#6F42C1"}},"celestia-appd",-1),is=s("span",{style:{color:"#24292E"}}," ",-1),rs=s("span",{style:{color:"#032F62"}},"download-genesis",-1),_s=s("span",{style:{color:"#24292E"}}," ",-1),ds={style:{color:"#032F62"}},hs={class:"language-bash vp-adaptive-theme"},ys=s("button",{title:"Copy Code",class:"copy"},null,-1),Es=s("span",{class:"lang"},"bash",-1),us={class:"shiki github-dark vp-code-dark"},Fs={class:"line"},Cs=s("span",{style:{color:"#B392F0"}},"celestia-appd",-1),gs=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ts=s("span",{style:{color:"#9ECBFF"}},"download-genesis",-1),bs=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ss={style:{color:"#9ECBFF"}},ms={class:"shiki github-light vp-code-light"},As={class:"line"},Bs=s("span",{style:{color:"#6F42C1"}},"celestia-appd",-1),fs=s("span",{style:{color:"#24292E"}}," ",-1),ks=s("span",{style:{color:"#032F62"}},"download-genesis",-1),vs=s("span",{style:{color:"#24292E"}}," ",-1),Ps={style:{color:"#032F62"}},qs={class:"language-bash vp-adaptive-theme"},Is=s("button",{title:"Copy Code",class:"copy"},null,-1),Vs=s("span",{class:"lang"},"bash",-1),Rs={class:"shiki github-dark vp-code-dark"},Ds={class:"line"},Ns=s("span",{style:{color:"#B392F0"}},"celestia-appd",-1),ws=s("span",{style:{color:"#E1E4E8"}}," ",-1),xs=s("span",{style:{color:"#9ECBFF"}},"download-genesis",-1),$s=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ms={style:{color:"#9ECBFF"}},Os={class:"shiki github-light vp-code-light"},Hs={class:"line"},Ls=s("span",{style:{color:"#6F42C1"}},"celestia-appd",-1),Qs=s("span",{style:{color:"#24292E"}}," ",-1),js=s("span",{style:{color:"#032F62"}},"download-genesis",-1),Gs=s("span",{style:{color:"#24292E"}}," ",-1),Ys={style:{color:"#032F62"}},Js=s("p",null,[a("Set seeds in the "),s("code",null,"$HOME/.celestia-app/config/config.toml"),a(" file:")],-1),Ws={class:"vp-code-group vp-adaptive-theme"},Us=o("",1),Xs={class:"blocks"},Ks={class:"language-bash vp-adaptive-theme active"},Zs=s("button",{title:"Copy Code",class:"copy"},null,-1),zs=s("span",{class:"lang"},"bash",-1),so={class:"shiki github-dark vp-code-dark"},oo={class:"line"},ao=o("",6),no={style:{color:"#9ECBFF"}},eo=s("span",{style:{color:"#F97583"}},"|",-1),to=s("span",{style:{color:"#9ECBFF"}}," ",-1),lo=s("span",{style:{color:"#B392F0"}},"tr",-1),po=s("span",{style:{color:"#9ECBFF"}}," '\\n' ',')",-1),co=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $SEEDS")],-1),io=o("",1),ro={class:"shiki github-light vp-code-light"},_o={class:"line"},ho=o("",6),yo={style:{color:"#032F62"}},Eo=s("span",{style:{color:"#D73A49"}},"|",-1),uo=s("span",{style:{color:"#032F62"}}," ",-1),Fo=s("span",{style:{color:"#6F42C1"}},"tr",-1),Co=s("span",{style:{color:"#032F62"}}," '\\n' ',')",-1),go=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $SEEDS")],-1),To=o("",1),bo={class:"language-bash vp-adaptive-theme"},So=s("button",{title:"Copy Code",class:"copy"},null,-1),mo=s("span",{class:"lang"},"bash",-1),Ao={class:"shiki github-dark vp-code-dark"},Bo={class:"line"},fo=o("",6),ko={style:{color:"#9ECBFF"}},vo=s("span",{style:{color:"#F97583"}},"|",-1),Po=s("span",{style:{color:"#9ECBFF"}}," ",-1),qo=s("span",{style:{color:"#B392F0"}},"tr",-1),Io=s("span",{style:{color:"#9ECBFF"}}," '\\n' ',')",-1),Vo=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $SEEDS")],-1),Ro=o("",1),Do={class:"shiki github-light vp-code-light"},No={class:"line"},wo=o("",6),xo={style:{color:"#032F62"}},$o=s("span",{style:{color:"#D73A49"}},"|",-1),Mo=s("span",{style:{color:"#032F62"}}," ",-1),Oo=s("span",{style:{color:"#6F42C1"}},"tr",-1),Ho=s("span",{style:{color:"#032F62"}}," '\\n' ',')",-1),Lo=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $SEEDS")],-1),Qo=o("",1),jo=o("",1),Go=s("p",null,[s("strong",null,"Optionally"),a(", you can set persistent peers in your "),s("code",null,"config.toml"),a(" file. You can get the persistent peers from the networks repository with the following commands:")],-1),Yo=s("p",null,"Setting persistent peers is advised only if you are running a sentry node.",-1),Jo={class:"vp-code-group vp-adaptive-theme"},Wo=o("",1),Uo={class:"blocks"},Xo={class:"language-bash vp-adaptive-theme active"},Ko=s("button",{title:"Copy Code",class:"copy"},null,-1),Zo=s("span",{class:"lang"},"bash",-1),zo={class:"shiki github-dark vp-code-dark"},sa={class:"line"},oa=o("",6),aa={style:{color:"#9ECBFF"}},na=s("span",{style:{color:"#F97583"}},"|",-1),ea=s("span",{style:{color:"#9ECBFF"}}," ",-1),ta=s("span",{style:{color:"#B392F0"}},"tr",-1),la=s("span",{style:{color:"#9ECBFF"}}," '\\n' ',')",-1),pa=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $PERSISTENT_PEERS")],-1),ca=o("",1),ia={class:"shiki github-light vp-code-light"},ra={class:"line"},_a=o("",6),da={style:{color:"#032F62"}},ha=s("span",{style:{color:"#D73A49"}},"|",-1),ya=s("span",{style:{color:"#032F62"}}," ",-1),Ea=s("span",{style:{color:"#6F42C1"}},"tr",-1),ua=s("span",{style:{color:"#032F62"}}," '\\n' ',')",-1),Fa=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $PERSISTENT_PEERS")],-1),Ca=o("",1),ga={class:"language-bash vp-adaptive-theme"},Ta=s("button",{title:"Copy Code",class:"copy"},null,-1),ba=s("span",{class:"lang"},"bash",-1),Sa={class:"shiki github-dark vp-code-dark"},ma={class:"line"},Aa=o("",6),Ba={style:{color:"#9ECBFF"}},fa=s("span",{style:{color:"#F97583"}},"|",-1),ka=s("span",{style:{color:"#9ECBFF"}}," ",-1),va=s("span",{style:{color:"#B392F0"}},"tr",-1),Pa=s("span",{style:{color:"#9ECBFF"}}," '\\n' ',')",-1),qa=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $PERSISTENT_PEERS")],-1),Ia=o("",1),Va={class:"shiki github-light vp-code-light"},Ra={class:"line"},Da=o("",6),Na={style:{color:"#032F62"}},wa=s("span",{style:{color:"#D73A49"}},"|",-1),xa=s("span",{style:{color:"#032F62"}}," ",-1),$a=s("span",{style:{color:"#6F42C1"}},"tr",-1),Ma=s("span",{style:{color:"#032F62"}}," '\\n' ',')",-1),Oa=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $PERSISTENT_PEERS")],-1),Ha=o("",1),La={class:"language-bash vp-adaptive-theme"},Qa=s("button",{title:"Copy Code",class:"copy"},null,-1),ja=s("span",{class:"lang"},"bash",-1),Ga={class:"shiki github-dark vp-code-dark"},Ya={class:"line"},Ja=o("",6),Wa={style:{color:"#9ECBFF"}},Ua=s("span",{style:{color:"#F97583"}},"|",-1),Xa=s("span",{style:{color:"#9ECBFF"}}," ",-1),Ka=s("span",{style:{color:"#B392F0"}},"tr",-1),Za=s("span",{style:{color:"#9ECBFF"}}," '\\n' ',')",-1),za=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $PERSISTENT_PEERS")],-1),sn=o("",1),on={class:"shiki github-light vp-code-light"},an={class:"line"},nn=o("",6),en={style:{color:"#032F62"}},tn=s("span",{style:{color:"#D73A49"}},"|",-1),ln=s("span",{style:{color:"#032F62"}}," ",-1),pn=s("span",{style:{color:"#6F42C1"}},"tr",-1),cn=s("span",{style:{color:"#032F62"}}," '\\n' ',')",-1),rn=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $PERSISTENT_PEERS")],-1),_n=o("",1),dn=o("",29),hn={class:"vp-code-group vp-adaptive-theme"},yn=o("",1),En={class:"blocks"},un={class:"language-bash vp-adaptive-theme active"},Fn=s("button",{title:"Copy Code",class:"copy"},null,-1),Cn=s("span",{class:"lang"},"bash",-1),gn={class:"shiki github-dark vp-code-dark"},Tn=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"cd"),s("span",{style:{color:"#E1E4E8"}}," $HOME")],-1),bn=o("",1),Sn=o("",1),mn=o("",1),An={class:"line"},Bn=s("span",{style:{color:"#9ECBFF"}}," ",-1),fn=s("span",{style:{color:"#B392F0"}},"egrep",-1),kn=s("span",{style:{color:"#9ECBFF"}}," ",-1),vn=s("span",{style:{color:"#79B8FF"}},"-o",-1),Pn={style:{color:"#9ECBFF"}},qn=o("",6),In=o("",1),Vn=o("",1),Rn={class:"shiki github-light vp-code-light"},Dn=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"cd"),s("span",{style:{color:"#24292E"}}," $HOME")],-1),Nn=o("",1),wn=o("",1),xn=o("",1),$n={class:"line"},Mn=s("span",{style:{color:"#032F62"}}," ",-1),On=s("span",{style:{color:"#6F42C1"}},"egrep",-1),Hn=s("span",{style:{color:"#032F62"}}," ",-1),Ln=s("span",{style:{color:"#005CC5"}},"-o",-1),Qn={style:{color:"#032F62"}},jn=o("",6),Gn=o("",1),Yn=o("",1),Jn={class:"language-bash vp-adaptive-theme"},Wn=s("button",{title:"Copy Code",class:"copy"},null,-1),Un=s("span",{class:"lang"},"bash",-1),Xn={class:"shiki github-dark vp-code-dark"},Kn=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"cd"),s("span",{style:{color:"#E1E4E8"}}," $HOME")],-1),Zn=o("",1),zn=o("",1),se=o("",1),oe={class:"line"},ae=s("span",{style:{color:"#9ECBFF"}}," ",-1),ne=s("span",{style:{color:"#B392F0"}},"egrep",-1),ee=s("span",{style:{color:"#9ECBFF"}}," ",-1),te=s("span",{style:{color:"#79B8FF"}},"-o",-1),le={style:{color:"#9ECBFF"}},pe=o("",6),ce=o("",1),ie=o("",1),re={class:"shiki github-light vp-code-light"},_e=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"cd"),s("span",{style:{color:"#24292E"}}," $HOME")],-1),de=o("",1),he=o("",1),ye=o("",1),Ee={class:"line"},ue=s("span",{style:{color:"#032F62"}}," ",-1),Fe=s("span",{style:{color:"#6F42C1"}},"egrep",-1),Ce=s("span",{style:{color:"#032F62"}}," ",-1),ge=s("span",{style:{color:"#005CC5"}},"-o",-1),Te={style:{color:"#032F62"}},be=o("",6),Se=o("",1),me=o("",1),Ae={class:"language-bash vp-adaptive-theme"},Be=s("button",{title:"Copy Code",class:"copy"},null,-1),fe=s("span",{class:"lang"},"bash",-1),ke={class:"shiki github-dark vp-code-dark"},ve=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"cd"),s("span",{style:{color:"#E1E4E8"}}," $HOME")],-1),Pe=o("",1),qe=o("",1),Ie=o("",1),Ve={class:"line"},Re=s("span",{style:{color:"#9ECBFF"}}," ",-1),De=s("span",{style:{color:"#B392F0"}},"egrep",-1),Ne=s("span",{style:{color:"#9ECBFF"}}," ",-1),we=s("span",{style:{color:"#79B8FF"}},"-o",-1),xe={style:{color:"#9ECBFF"}},$e=o("",6),Me=o("",1),Oe=o("",1),He={class:"shiki github-light vp-code-light"},Le=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"cd"),s("span",{style:{color:"#24292E"}}," $HOME")],-1),Qe=o("",1),je=o("",1),Ge=o("",1),Ye={class:"line"},Je=s("span",{style:{color:"#032F62"}}," ",-1),We=s("span",{style:{color:"#6F42C1"}},"egrep",-1),Ue=s("span",{style:{color:"#032F62"}}," ",-1),Xe=s("span",{style:{color:"#005CC5"}},"-o",-1),Ke={style:{color:"#032F62"}},Ze=o("",6),ze=o("",1),st=o("",1),ot=o("",35),ct=JSON.parse('{"title":"Setting up a Celestia full consensus node","description":"Learn how to set up a Celestia full consensus node.","frontmatter":{"description":"Learn how to set up a Celestia full consensus node.","outline":"deep","head":[["meta",{"name":"og:title","content":"Setting up a Celestia full consensus node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/full-consensus-node.md","filePath":"nodes/full-consensus-node.md","lastUpdated":1711744858000}'),at={name:"nodes/full-consensus-node.md"},it=Object.assign(at,{setup(nt){return(et,tt)=>(l(),p("div",null,[i,s("div",r,[_,s("div",d,[s("div",h,[y,E,s("pre",u,[s("code",null,[s("span",F,[C,s("span",g,e(t(n).mainnetChainId),1)])])]),s("pre",T,[s("code",null,[s("span",b,[S,s("span",m,e(t(n).mainnetChainId),1)])])])]),s("div",A,[B,f,s("pre",k,[s("code",null,[s("span",v,[P,s("span",q,e(t(n).mochaChainId),1)])])]),s("pre",I,[s("code",null,[s("span",V,[R,s("span",D,e(t(n).mochaChainId),1)])])])]),s("div",N,[w,x,s("pre",$,[s("code",null,[s("span",M,[O,s("span",H,e(t(n).arabicaChainId),1)])])]),s("pre",L,[s("code",null,[s("span",Q,[j,s("span",G,e(t(n).arabicaChainId),1)])])])])])]),Y,s("div",J,[W,s("div",U,[s("div",X,[K,Z,s("pre",z,[s("code",null,[s("span",ss,[os,as,ns,es,s("span",ts,e(t(n).mainnetChainId),1)])])]),s("pre",ls,[s("code",null,[s("span",ps,[cs,is,rs,_s,s("span",ds,e(t(n).mainnetChainId),1)])])])]),s("div",hs,[ys,Es,s("pre",us,[s("code",null,[s("span",Fs,[Cs,gs,Ts,bs,s("span",Ss,e(t(n).mochaChainId),1)])])]),s("pre",ms,[s("code",null,[s("span",As,[Bs,fs,ks,vs,s("span",Ps,e(t(n).mochaChainId),1)])])])]),s("div",qs,[Is,Vs,s("pre",Rs,[s("code",null,[s("span",Ds,[Ns,ws,xs,$s,s("span",Ms,e(t(n).arabicaChainId),1)])])]),s("pre",Os,[s("code",null,[s("span",Hs,[Ls,Qs,js,Gs,s("span",Ys,e(t(n).arabicaChainId),1)])])])])])]),Js,s("div",Ws,[Us,s("div",Xs,[s("div",Ks,[Zs,zs,s("pre",so,[s("code",null,[s("span",oo,[ao,s("span",no," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mainnetChainId)+"/seeds.txt ",1),eo,to,lo,po]),a(` +`),co,a(` +`),io])]),s("pre",ro,[s("code",null,[s("span",_o,[ho,s("span",yo," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mainnetChainId)+"/seeds.txt ",1),Eo,uo,Fo,Co]),a(` +`),go,a(` +`),To])])]),s("div",bo,[So,mo,s("pre",Ao,[s("code",null,[s("span",Bo,[fo,s("span",ko," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mochaChainId)+"/seeds.txt ",1),vo,Po,qo,Io]),a(` +`),Vo,a(` +`),Ro])]),s("pre",Do,[s("code",null,[s("span",No,[wo,s("span",xo," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mochaChainId)+"/seeds.txt ",1),$o,Mo,Oo,Ho]),a(` +`),Lo,a(` +`),Qo])])]),jo])]),Go,Yo,s("div",Jo,[Wo,s("div",Uo,[s("div",Xo,[Ko,Zo,s("pre",zo,[s("code",null,[s("span",sa,[oa,s("span",aa," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mainnetChainId)+"/peers.txt ",1),na,ea,ta,la]),a(` +`),pa,a(` +`),ca])]),s("pre",ia,[s("code",null,[s("span",ra,[_a,s("span",da," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mainnetChainId)+"/peers.txt ",1),ha,ya,Ea,ua]),a(` +`),Fa,a(` +`),Ca])])]),s("div",ga,[Ta,ba,s("pre",Sa,[s("code",null,[s("span",ma,[Aa,s("span",Ba," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mochaChainId)+"/peers.txt ",1),fa,ka,va,Pa]),a(` +`),qa,a(` +`),Ia])]),s("pre",Va,[s("code",null,[s("span",Ra,[Da,s("span",Na," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).mochaChainId)+"/peers.txt ",1),wa,xa,$a,Ma]),a(` +`),Oa,a(` +`),Ha])])]),s("div",La,[Qa,ja,s("pre",Ga,[s("code",null,[s("span",Ya,[Ja,s("span",Wa," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).arabicaChainId)+"/peers.txt ",1),Ua,Xa,Ka,Za]),a(` +`),za,a(` +`),sn])]),s("pre",on,[s("code",null,[s("span",an,[nn,s("span",en," https://raw.githubusercontent.com/celestiaorg/networks/master/"+e(t(n).arabicaChainId)+"/peers.txt ",1),tn,ln,pn,cn]),a(` +`),rn,a(` +`),_n])])])])]),dn,s("div",hn,[yn,s("div",En,[s("div",un,[Fn,Cn,s("pre",gn,[s("code",null,[Tn,a(` +`),bn,a(` +`),Sn,a(` +`),mn,a(` +`),s("span",An,[Bn,fn,kn,vn,s("span",Pn,' ">'+e(t(n).mainnetChainId)+'.*tar" ',1),qn]),a(` +`),In,a(` +`),Vn])]),s("pre",Rn,[s("code",null,[Dn,a(` +`),Nn,a(` +`),wn,a(` +`),xn,a(` +`),s("span",$n,[Mn,On,Hn,Ln,s("span",Qn,' ">'+e(t(n).mainnetChainId)+'.*tar" ',1),jn]),a(` +`),Gn,a(` +`),Yn])])]),s("div",Jn,[Wn,Un,s("pre",Xn,[s("code",null,[Kn,a(` +`),Zn,a(` +`),zn,a(` +`),se,a(` +`),s("span",oe,[ae,ne,ee,te,s("span",le,' ">'+e(t(n).mochaChainId)+'.*tar" ',1),pe]),a(` +`),ce,a(` +`),ie])]),s("pre",re,[s("code",null,[_e,a(` +`),de,a(` +`),he,a(` +`),ye,a(` +`),s("span",Ee,[ue,Fe,Ce,ge,s("span",Te,' ">'+e(t(n).mochaChainId)+'.*tar" ',1),be]),a(` +`),Se,a(` +`),me])])]),s("div",Ae,[Be,fe,s("pre",ke,[s("code",null,[ve,a(` +`),Pe,a(` +`),qe,a(` +`),Ie,a(` +`),s("span",Ve,[Re,De,Ne,we,s("span",xe,' ">'+e(t(n).arabicaChainId)+'.*tar" ',1),$e]),a(` +`),Me,a(` +`),Oe])]),s("pre",He,[s("code",null,[Le,a(` +`),Qe,a(` +`),je,a(` +`),Ge,a(` +`),s("span",Ye,[Je,We,Ue,Xe,s("span",Ke,' ">'+e(t(n).arabicaChainId)+'.*tar" ',1),Ze]),a(` +`),ze,a(` +`),st])])])])]),ot]))}});export{ct as __pageData,it as default}; diff --git a/assets/nodes_full-storage-node.md.03ad71bf.js b/assets/nodes_full-storage-node.md.03ad71bf.js new file mode 100644 index 00000000000..35f6736c5d5 --- /dev/null +++ b/assets/nodes_full-storage-node.md.03ad71bf.js @@ -0,0 +1,11 @@ +import{_ as s,o as a,c as e,Q as o}from"./chunks/framework.51d6c45b.js";const l="/img/nodes/full-storage-node.png",g=JSON.parse('{"title":"Setting up a Celestia full storage Node","description":"Set up a Celestia full storage node.","frontmatter":{"description":"Set up a Celestia full storage node.","head":[["meta",{"name":"og:title","content":"Setting up a Celestia full storage Node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/full-storage-node.md","filePath":"nodes/full-storage-node.md","lastUpdated":1702574981000}'),n={name:"nodes/full-storage-node.md"},t=o('

    Setting up a Celestia full storage Node

    This tutorial will guide you through setting up a Celestia full storage node, which is a celestia-node that doesn't connect to celestia-app (hence not a full consensus node), but stores all the data.

    Overview of full storage nodes

    Full storage nodes are Celestia nodes that store all the data. Full storage nodes send block shares, headers, and fraud proofs to light nodes. The light nodes gossip headers, fraud proofs, and sometimes block shares, between one another.

    Full storage node

    Hardware requirements

    The following hardware minimum requirements are recommended for running the full storage node:

    • Memory: 4 GB RAM (minimum)
    • CPU: 6 cores
    • Disk: 10 TB SSD Storage
    • Bandwidth: 1 Gbps for Download/1 Gbps for Upload

    Setting up your full storage node

    The following tutorial is done on an Ubuntu Linux 20.04 (LTS) x64 instance machine.

    Setup the dependencies

    You can follow the tutorial for setting up your dependencies

    Install celestia-node

    You can follow the tutorial for installing celestia-node

    Run the full storage node

    Initialize the full storage node

    Run the following command:

    sh
    celestia full init
    celestia full init
    sh
    celestia full init --p2p.network mocha
    celestia full init --p2p.network mocha
    sh
    celestia full init --p2p.network arabica
    celestia full init --p2p.network arabica

    Start the full storage node

    Start the full storage node with a connection to a validator node's gRPC endpoint (which is usually exposed on port 9090):

    In order for access to the ability to get/submit state-related information, such as the ability to submit PayForBlob transactions, or query for the node's account balance, a gRPC endpoint of a validator (core) node must be passed as directed below.

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    sh
    celestia full start --core.ip <URI>
    celestia full start --core.ip <URI>

    Using an RPC of your own, or one from the list on the Mocha testnet page or list on the Arabica devnet page, start your node.

    You can create your key for your node by following the cel-key instructions

    Once you start the full storage node, a wallet key will be generated for you. You will need to fund that address with testnet tokens to pay for PayForBlob transactions. You can find the address by running the following command:

    sh
    ./cel-key list --node.type full --keyring-backend test --p2p.network <network>
    ./cel-key list --node.type full --keyring-backend test --p2p.network <network>

    TIP

    You do not need to declare a network for Mainnet Beta. Refer to the chain ID section on the troubleshooting page for more information

    You can get testnet tokens from:

    NOTE

    If you are running a full-storage node for your sovereign rollup, it is highly recommended to request Arabica devnet tokens as Arabica has the latest changes that can be used to test for developing your sovereign rollup. You can still use Mocha testnet as well, it is just mostly used for validator operations.

    Optional: run the full storage node with a custom key

    In order to run a full storage node using a custom key:

    1. The custom key must exist inside the celestia full storage node directory at the correct path (default: ~/.celestia-full/keys/keyring-test)
    2. The name of the custom key must be passed upon start, like so:
    sh
    celestia full start --core.ip <URI> \\
    +  --keyring.accname <name-of-custom-key> \\
    celestia full start --core.ip <URI> \\
    +  --keyring.accname <name-of-custom-key> \\
    sh
    celestia full start --core.ip <URI> \\
    +  --keyring.accname <name-of-custom-key> \\
    +  --p2p.network mocha
    celestia full start --core.ip <URI> \\
    +  --keyring.accname <name-of-custom-key> \\
    +  --p2p.network mocha
    sh
    celestia full start --core.ip <URI> \\
    +  --keyring.accname <name-of-custom-key> \\
    +  --p2p.network arabica
    celestia full start --core.ip <URI> \\
    +  --keyring.accname <name-of-custom-key> \\
    +  --p2p.network arabica

    Optional: start the full storage node with SystemD

    If you would like to run the full storage node as a background process, follow the SystemD tutorial.

    With that, you are now running a Celestia full storage node.

    Stop the full storage node

    In order to gracefully stop the full storage node, use Control + C in the terminal window where the node is running. Be sure to only do this once as the shutdown will not be instantaneous.

    `,40),p=[t];function r(c,i,d,y,u,h){return a(),e("div",null,p)}const f=s(n,[["render",r]]);export{g as __pageData,f as default}; diff --git a/assets/nodes_full-storage-node.md.03ad71bf.lean.js b/assets/nodes_full-storage-node.md.03ad71bf.lean.js new file mode 100644 index 00000000000..b9e5b256df1 --- /dev/null +++ b/assets/nodes_full-storage-node.md.03ad71bf.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as e,Q as o}from"./chunks/framework.51d6c45b.js";const l="/img/nodes/full-storage-node.png",g=JSON.parse('{"title":"Setting up a Celestia full storage Node","description":"Set up a Celestia full storage node.","frontmatter":{"description":"Set up a Celestia full storage node.","head":[["meta",{"name":"og:title","content":"Setting up a Celestia full storage Node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/full-storage-node.md","filePath":"nodes/full-storage-node.md","lastUpdated":1702574981000}'),n={name:"nodes/full-storage-node.md"},t=o("",40),p=[t];function r(c,i,d,y,u,h){return a(),e("div",null,p)}const f=s(n,[["render",r]]);export{g as __pageData,f as default}; diff --git a/assets/nodes_hardfork-process.md.431e0126.js b/assets/nodes_hardfork-process.md.431e0126.js new file mode 100644 index 00000000000..5410d348dcf --- /dev/null +++ b/assets/nodes_hardfork-process.md.431e0126.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as o,Q as t}from"./chunks/framework.51d6c45b.js";const k=JSON.parse('{"title":"Celestia hardfork process","description":"Overview of the Celestia hardfork process.","frontmatter":{"description":"Overview of the Celestia hardfork process.","head":[["meta",{"name":"og:title","content":"Celestia hardfork process | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/hardfork-process.md","filePath":"nodes/hardfork-process.md","lastUpdated":1702583650000}'),r={name:"nodes/hardfork-process.md"},s=t('

    Celestia hardfork process

    Blockchain networks often times need to upgrade with new features which require coordination work among the validators prior to activating the upgrades.

    This process is called a hardfork or a network upgrade. In those events, the Celestia Labs team will be coordinating with the validators on what they need to do in order to be ready for an upcoming hardfork.

    Hardforks are not backward-compatible with older versions of the network software which is why it is important that validators upgrade their software to continue validating on the network after the network upgrades.

    General process

    The general process can be broken down into several components:

    • Hardfork specifications and features (defined by description of features and code implementation of those features).
    • Binary used to add those features (a new binary release with those features will be provided by Celestia team in order for validators to upgrade their nodes to the new binary).
    • A block number for when the network upgrades (even if validators upgrade their binary to be hardfork ready, the network upgrade does not happen right away, but some short time in the future at a specific block number).
    • Testing of the features (happens on testnets first prior to activating on mainnet in order to ensure the network can upgrade securely).

    The two testnets were hardforks are deployed on are:

    Mocha hardfork

    Celestia is planning the Mocha Hardfork upgrade on the Mamaki Testnet. This hardfork is unique as it will reset the Mamaki network to block 0 while maintaining the existing state and also will rename Mamaki to Mocha.

    The new chain-id will be mocha.

    You can find the release logs for consensus nodes on the celestia-app releases page.

    The most exciting feature included is setting the stage for Blobstream on Mocha.

    Validators will need to generate 2 new keys in order to be Blobstream-ready. Note that for the Mocha Hardfork, Blobstream will not launch yet so you can swap those keys after for new ones if needed. The keys needed are:

    • 1 EVM key
    • 1 Celestia key

    So, in order for this to happen, validators will need to maintain two new keys in order to have a successful upgrade.

    Those two keys will need to be added to 2 new flags on celestia-app:

    • --evm-address: This flag should contain a 0x EVM address.
    ',19),i=[s];function n(d,l,h,c,p,f){return a(),o("div",null,i)}const w=e(r,[["render",n]]);export{k as __pageData,w as default}; diff --git a/assets/nodes_hardfork-process.md.431e0126.lean.js b/assets/nodes_hardfork-process.md.431e0126.lean.js new file mode 100644 index 00000000000..5e15edb8f8d --- /dev/null +++ b/assets/nodes_hardfork-process.md.431e0126.lean.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as o,Q as t}from"./chunks/framework.51d6c45b.js";const k=JSON.parse('{"title":"Celestia hardfork process","description":"Overview of the Celestia hardfork process.","frontmatter":{"description":"Overview of the Celestia hardfork process.","head":[["meta",{"name":"og:title","content":"Celestia hardfork process | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/hardfork-process.md","filePath":"nodes/hardfork-process.md","lastUpdated":1702583650000}'),r={name:"nodes/hardfork-process.md"},s=t("",19),i=[s];function n(d,l,h,c,p,f){return a(),o("div",null,i)}const w=e(r,[["render",n]]);export{k as __pageData,w as default}; diff --git a/assets/nodes_ibc-relayer.md.b3114711.js b/assets/nodes_ibc-relayer.md.b3114711.js new file mode 100644 index 00000000000..51d8419a097 --- /dev/null +++ b/assets/nodes_ibc-relayer.md.b3114711.js @@ -0,0 +1,455 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.51d6c45b.js";const h=JSON.parse('{"title":"IBC relaying guide","description":"Learn how to establish IBC connections and relay packets.","frontmatter":{"description":"Learn how to establish IBC connections and relay packets.","next":{"text":"Metrics","link":"/nodes/celestia-node-metrics"},"head":[["meta",{"name":"og:title","content":"IBC relaying guide | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/ibc-relayer.md","filePath":"nodes/ibc-relayer.md","lastUpdated":1707231883000}'),p={name:"nodes/ibc-relayer.md"},o=l(`

    IBC relaying guide

    Celestia uses IBC (Inter-Blockchain Communication protocol) to enable cross-chain transfer of tokens. To support this capability it relies on relayers, processes that can be run by anyone which constantly scan for outbound packets on one chain and submits these packets alongside corresponding proofs on the destination chain. This section describes how one can setup a relayer and create new connections between chains. There are two standard implementations:

    The following guide explains how to establish IBC connections and relay packets between Mocha testnet and Cosmos hub testnet networks by using the Hermes relayer.

    Check the latest celestia-app release's go.mod for the version of ibc-go that is currently used.

    Hermes

    Hermes is an open-source Rust implementation of an IBC relayer released as part of the ibc-relayer-cli crate. It includes a CLI for relaying packets between Cosmos SDK chains, as well as Prometheus metrics and a REST API.

    Please follow the steps at Hermes Quick Start to install Hermes. Before proceeding, verify that Hermes is installed correctly by running hermes version.

    TIP

    Hermes currently doesn't support configuring the Tendermint CompatMode in chain config (see hermes#3623). Until that issue is resolved, please use Hermes v1.7.0+ because it falls back to Tendermint CompatMode v0.34 (see hermes#3663) which is compatible with Celestia.

    Configuration

    After you have successfully installed Hermes and created the necessary folders, you now have to edit config.toml and add the appropriate configurations for the chains you want to relay between.

    For this tutorial, we will be using the following chains:

    • Celestia's mocha-4 testnet
    • Cosmos Hub's theta-testnet-001 testnet

    Edit the Hermes configuration.

    bash
    vim $HOME/.hermes/config.toml
    vim $HOME/.hermes/config.toml
    toml
    [global]
    +log_level = "info"
    +
    +[mode.clients]
    +enabled = true
    +refresh = true
    +misbehaviour = true
    +
    +[mode.connections]
    +enabled = false
    +
    +[mode.channels]
    +enabled = false
    +
    +[mode.packets]
    +enabled = true
    +clear_interval = 100
    +clear_on_start = true
    +tx_confirmation = false
    +auto_register_counterparty_payee = false
    +
    +[rest]
    +enabled = false
    +host = "127.0.0.1"
    +port = 3000
    +
    +[telemetry]
    +enabled = false
    +host = "127.0.0.1"
    +port = 3001
    +
    +[telemetry.buckets.latency_submitted]
    +start = 500
    +end = 20000
    +buckets = 10
    +
    +[telemetry.buckets.latency_confirmed]
    +start = 1000
    +end = 30000
    +buckets = 10
    +
    +[[chains]]
    +id = "theta-testnet-001"
    +type = "CosmosSdk"
    +rpc_addr = "https://rpc.sentry-01.theta-testnet.polypore.xyz"
    +grpc_addr = "https://grpc.sentry-01.theta-testnet.polypore.xyz"
    +rpc_timeout = "10s"
    +trusted_node = false
    +account_prefix = "cosmos"
    +key_name = "key-cosmos"
    +key_store_type = "Test"
    +store_prefix = "ibc"
    +default_gas = 100000
    +max_gas = 400000
    +gas_multiplier = 1.5
    +max_msg_num = 30
    +max_tx_size = 180000
    +max_grpc_decoding_size = 33554432
    +clock_drift = "5s"
    +max_block_time = "30s"
    +ccv_consumer_chain = false
    +memo_prefix = ""
    +sequential_batch_tx = false
    +
    +[chains.event_source]
    +mode = "push"
    +url = "ws://rpc.sentry-01.theta-testnet.polypore.xyz:26657/websocket"
    +batch_delay = "500ms"
    +
    +[chains.trust_threshold]
    +numerator = "1"
    +denominator = "3"
    +
    +[chains.gas_price]
    +price = 0.025
    +denom = "uatom"
    +
    +[chains.packet_filter]
    +policy = "allow"
    +list = [["transfer", "channel-3108"]]
    +
    +[chains.packet_filter.min_fees]
    +
    +[chains.address_type]
    +derivation = "cosmos"
    +
    +[[chains]]
    +id = "mocha-4"
    +type = "CosmosSdk"
    +rpc_addr = "https://rpc-celestia-mocha.architectnodes.com"
    +grpc_addr = "https://grpc.celestia-mocha.com:443"
    +rpc_timeout = "10s"
    +trusted_node = false
    +account_prefix = "celestia"
    +key_name = "celestia-key"
    +key_store_type = "Test"
    +store_prefix = "ibc"
    +default_gas = 100000
    +max_gas = 400000
    +gas_multiplier = 1.5
    +max_msg_num = 30
    +max_tx_size = 180000
    +max_grpc_decoding_size = 33554432
    +clock_drift = "5s"
    +max_block_time = "30s"
    +ccv_consumer_chain = false
    +memo_prefix = ""
    +sequential_batch_tx = false
    +
    +[chains.event_source]
    +mode = "push"
    +url = "ws://rpc-mocha.pops.one:26657/websocket"
    +batch_delay = "500ms"
    +
    +[chains.trust_threshold]
    +numerator = "1"
    +denominator = "3"
    +
    +[chains.gas_price]
    +price = 0.1
    +denom = "utia"
    +
    +[chains.packet_filter]
    +policy = "allow"
    +list = [["transfer", "channel-0"]]
    +
    +[chains.packet_filter.min_fees]
    +
    +[chains.address_type]
    +derivation = "cosmos"
    [global]
    +log_level = "info"
    +
    +[mode.clients]
    +enabled = true
    +refresh = true
    +misbehaviour = true
    +
    +[mode.connections]
    +enabled = false
    +
    +[mode.channels]
    +enabled = false
    +
    +[mode.packets]
    +enabled = true
    +clear_interval = 100
    +clear_on_start = true
    +tx_confirmation = false
    +auto_register_counterparty_payee = false
    +
    +[rest]
    +enabled = false
    +host = "127.0.0.1"
    +port = 3000
    +
    +[telemetry]
    +enabled = false
    +host = "127.0.0.1"
    +port = 3001
    +
    +[telemetry.buckets.latency_submitted]
    +start = 500
    +end = 20000
    +buckets = 10
    +
    +[telemetry.buckets.latency_confirmed]
    +start = 1000
    +end = 30000
    +buckets = 10
    +
    +[[chains]]
    +id = "theta-testnet-001"
    +type = "CosmosSdk"
    +rpc_addr = "https://rpc.sentry-01.theta-testnet.polypore.xyz"
    +grpc_addr = "https://grpc.sentry-01.theta-testnet.polypore.xyz"
    +rpc_timeout = "10s"
    +trusted_node = false
    +account_prefix = "cosmos"
    +key_name = "key-cosmos"
    +key_store_type = "Test"
    +store_prefix = "ibc"
    +default_gas = 100000
    +max_gas = 400000
    +gas_multiplier = 1.5
    +max_msg_num = 30
    +max_tx_size = 180000
    +max_grpc_decoding_size = 33554432
    +clock_drift = "5s"
    +max_block_time = "30s"
    +ccv_consumer_chain = false
    +memo_prefix = ""
    +sequential_batch_tx = false
    +
    +[chains.event_source]
    +mode = "push"
    +url = "ws://rpc.sentry-01.theta-testnet.polypore.xyz:26657/websocket"
    +batch_delay = "500ms"
    +
    +[chains.trust_threshold]
    +numerator = "1"
    +denominator = "3"
    +
    +[chains.gas_price]
    +price = 0.025
    +denom = "uatom"
    +
    +[chains.packet_filter]
    +policy = "allow"
    +list = [["transfer", "channel-3108"]]
    +
    +[chains.packet_filter.min_fees]
    +
    +[chains.address_type]
    +derivation = "cosmos"
    +
    +[[chains]]
    +id = "mocha-4"
    +type = "CosmosSdk"
    +rpc_addr = "https://rpc-celestia-mocha.architectnodes.com"
    +grpc_addr = "https://grpc.celestia-mocha.com:443"
    +rpc_timeout = "10s"
    +trusted_node = false
    +account_prefix = "celestia"
    +key_name = "celestia-key"
    +key_store_type = "Test"
    +store_prefix = "ibc"
    +default_gas = 100000
    +max_gas = 400000
    +gas_multiplier = 1.5
    +max_msg_num = 30
    +max_tx_size = 180000
    +max_grpc_decoding_size = 33554432
    +clock_drift = "5s"
    +max_block_time = "30s"
    +ccv_consumer_chain = false
    +memo_prefix = ""
    +sequential_batch_tx = false
    +
    +[chains.event_source]
    +mode = "push"
    +url = "ws://rpc-mocha.pops.one:26657/websocket"
    +batch_delay = "500ms"
    +
    +[chains.trust_threshold]
    +numerator = "1"
    +denominator = "3"
    +
    +[chains.gas_price]
    +price = 0.1
    +denom = "utia"
    +
    +[chains.packet_filter]
    +policy = "allow"
    +list = [["transfer", "channel-0"]]
    +
    +[chains.packet_filter.min_fees]
    +
    +[chains.address_type]
    +derivation = "cosmos"

    Add relayer wallets

    Now that we have successfully configured our relaying chains, we need to import the wallets that will be used for relaying. Please note that both wallets need to be funded with the native tokens of each chain.

    You can get testnet tokens from faucets for bot testnets via Discord:

    Add your seed phrase to a file and upload it to the server. Do not use wallets for anything else but relaying to avoid running into account sequence errors.

    Follow the steps at adding-keys-to-hermes to add keys for each chain

    bash
    hermes keys add --chain mocha-4 --mnemonic-file <seed-file>
    +hermes keys add --chain theta-testnet-001 --mnemonic-file <seed-file>
    hermes keys add --chain mocha-4 --mnemonic-file <seed-file>
    +hermes keys add --chain theta-testnet-001 --mnemonic-file <seed-file>

    Verify configuration files

    After editing config.toml and adding wallet keys, it’s time to test the configurations and ensure the system is healthy. Run the following:

    bash
    hermes health-check
    +hermes config validate
    hermes health-check
    +hermes config validate

    If everything was set up correctly, you should see output like:

    bash
    SUCCESS performed health check for all chains in the config
    +SUCCESS "configuration is valid"
    SUCCESS performed health check for all chains in the config
    +SUCCESS "configuration is valid"

    Create a connection between 2 chains

    If you’re attempting to create new connections, verify that the chains in question don’t already have connections and clients in place and use the existing ones if they do. In that case you can skip this step and go to Configure channels in Hermes section.

    In this example, we are creating new clients and a new connection between mocha-4 and theta-testnet-001 networks.

    Create clients

    To learn if a client already exists, you can use the following command:

    bash
    hermes query clients --host-chain mocha-4 --reference-chain theta-testnet-001
    hermes query clients --host-chain mocha-4 --reference-chain theta-testnet-001

    To create a new client, use the create-client command:

    bash
    hermes create client --host-chain mocha-4 --reference-chain theta-testnet-001
    hermes create client --host-chain mocha-4 --reference-chain theta-testnet-001

    Create a second client:

    bash
    hermes create client --host-chain theta-testnet-001 --reference-chain mocha-4
    hermes create client --host-chain theta-testnet-001 --reference-chain mocha-4

    Open connection over new clients

    To create a new connection over clients, use the following command:

    bash
    hermes create connection --a-chain mocha-4 --b-chain theta-testnet-001
    hermes create connection --a-chain mocha-4 --b-chain theta-testnet-001

    You should be seeing a similar output to this:

    bash
    SUCCESS Connection {
    +    delay_period: 0ns,
    +    a_side: ConnectionSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "theta-testnet-001",
    +                version: 0,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-2382",
    +        ),
    +        connection_id: Some(
    +            ConnectionId(
    +                "connection-2727",
    +            ),
    +        ),
    +    },
    +    b_side: ConnectionSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "mocha-4",
    +                version: 4,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-0",
    +        ),
    +        connection_id: Some(
    +            ConnectionId(
    +                "connection-0",
    +            ),
    +        ),
    +    },
    +}
    SUCCESS Connection {
    +    delay_period: 0ns,
    +    a_side: ConnectionSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "theta-testnet-001",
    +                version: 0,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-2382",
    +        ),
    +        connection_id: Some(
    +            ConnectionId(
    +                "connection-2727",
    +            ),
    +        ),
    +    },
    +    b_side: ConnectionSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "mocha-4",
    +                version: 4,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-0",
    +        ),
    +        connection_id: Some(
    +            ConnectionId(
    +                "connection-0",
    +            ),
    +        ),
    +    },
    +}

    Now that the connection has been established over the clients, we need to create a new channel, by leveraging an existing connection:

    bash
    hermes create channel --a-chain theta-testnet-001 --a-connection connection-2727 --a-port transfer --b-port transfer
    hermes create channel --a-chain theta-testnet-001 --a-connection connection-2727 --a-port transfer --b-port transfer

    You should be seeing a similar output to this:

    bash
    SUCCESS Channel {
    +    ordering: Unordered,
    +    a_side: ChannelSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "theta-testnet-001",
    +                version: 0,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-2382",
    +        ),
    +        connection_id: ConnectionId(
    +            "connection-2727",
    +        ),
    +        port_id: PortId(
    +            "transfer",
    +        ),
    +        channel_id: Some(
    +            ChannelId(
    +                "channel-3152",
    +            ),
    +        ),
    +        version: None,
    +    },
    +    b_side: ChannelSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "mocha-4",
    +                version: 4,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-0",
    +        ),
    +        connection_id: ConnectionId(
    +            "connection-0",
    +        ),
    +        port_id: PortId(
    +            "transfer",
    +        ),
    +        channel_id: Some(
    +            ChannelId(
    +                "channel-0",
    +            ),
    +        ),
    +        version: None,
    +    },
    +    connection_delay: 0ns,
    +}
    SUCCESS Channel {
    +    ordering: Unordered,
    +    a_side: ChannelSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "theta-testnet-001",
    +                version: 0,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-2382",
    +        ),
    +        connection_id: ConnectionId(
    +            "connection-2727",
    +        ),
    +        port_id: PortId(
    +            "transfer",
    +        ),
    +        channel_id: Some(
    +            ChannelId(
    +                "channel-3152",
    +            ),
    +        ),
    +        version: None,
    +    },
    +    b_side: ChannelSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "mocha-4",
    +                version: 4,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-0",
    +        ),
    +        connection_id: ConnectionId(
    +            "connection-0",
    +        ),
    +        port_id: PortId(
    +            "transfer",
    +        ),
    +        channel_id: Some(
    +            ChannelId(
    +                "channel-0",
    +            ),
    +        ),
    +        version: None,
    +    },
    +    connection_delay: 0ns,
    +}

    Congratulations!

    You have successfully created a new IBC connection between two networks.

    Configure channels in Hermes

    Now that we have created new connections and opened channels, we need to edit config.toml again and add the newly created channels, or use the already existing ones.

    For mocha-4 add:

    bash
    [chains.packet_filter]
    +policy = 'allow'
    +list = [
    +  ['transfer', 'channel-0'], # theta-testnet-001
    +]
    [chains.packet_filter]
    +policy = 'allow'
    +list = [
    +  ['transfer', 'channel-0'], # theta-testnet-001
    +]

    For theta-testnet-001 add:

    bash
    [chains.packet_filter]
    +policy = 'allow'
    +list = [
    +  ['transfer', 'channel-3108'], # mocha-4
    +]
    [chains.packet_filter]
    +policy = 'allow'
    +list = [
    +  ['transfer', 'channel-3108'], # mocha-4
    +]

    Start the relayer

    Start the relayer via hermes start

    Transfer

    The Celestia state machine is built with the IBC transfer module, allowing for the native Celestia token to be transferred to any other IBC enabled chain. Transfer can be initialized through the celestia-appd CLI. Information can be found via the help label as follows:

    bash
    celestia-appd tx ibc-transfer transfer --help
    celestia-appd tx ibc-transfer transfer --help

    Token filter

    The transfer module uses a token filter middleware which serves to prevent non-native Celestia tokens from being on Celestia. If a user is to try to send a token from another chain across, it will be simply rejected and the token returned back to the user.

    `,61),e=[o];function t(c,r,E,y,i,F){return n(),a("div",null,e)}const u=s(p,[["render",t]]);export{h as __pageData,u as default}; diff --git a/assets/nodes_ibc-relayer.md.b3114711.lean.js b/assets/nodes_ibc-relayer.md.b3114711.lean.js new file mode 100644 index 00000000000..50e243412fe --- /dev/null +++ b/assets/nodes_ibc-relayer.md.b3114711.lean.js @@ -0,0 +1 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.51d6c45b.js";const h=JSON.parse('{"title":"IBC relaying guide","description":"Learn how to establish IBC connections and relay packets.","frontmatter":{"description":"Learn how to establish IBC connections and relay packets.","next":{"text":"Metrics","link":"/nodes/celestia-node-metrics"},"head":[["meta",{"name":"og:title","content":"IBC relaying guide | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/ibc-relayer.md","filePath":"nodes/ibc-relayer.md","lastUpdated":1707231883000}'),p={name:"nodes/ibc-relayer.md"},o=l("",61),e=[o];function t(c,r,E,y,i,F){return n(),a("div",null,e)}const u=s(p,[["render",t]]);export{h as __pageData,u as default}; diff --git a/assets/nodes_instantiate-testnet.md.db07a97a.js b/assets/nodes_instantiate-testnet.md.db07a97a.js new file mode 100644 index 00000000000..fbc13593b9c --- /dev/null +++ b/assets/nodes_instantiate-testnet.md.db07a97a.js @@ -0,0 +1,21 @@ +import{_ as e,o as s,c as a,Q as n}from"./chunks/framework.51d6c45b.js";const E=JSON.parse('{"title":"Celestia App network instantiation guide","description":"A guide that helps you instantiate a new testnetwork with Celestia App.","frontmatter":{"description":"A guide that helps you instantiate a new testnetwork with Celestia App.","head":[["meta",{"name":"og:title","content":"Celestia App network instantiation guide | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/instantiate-testnet.md","filePath":"nodes/instantiate-testnet.md","lastUpdated":1711744858000}'),o={name:"nodes/instantiate-testnet.md"},t=n(`

    Celestia App network instantiation guide

    This guide is for helping instantiate a new testnetwork and following the correct steps to do so with Celestia App. You should only follow this guide if you want to experiment with your own Celestia test network (testnet) or if you want to test out new features to build as a core developer.

    Hardware requirements

    You will need to follow hardware requirements.

    Setup dependencies

    You will need to setup dependencies by following the guide.

    celestia-app installation

    You will need to install celestia-app by following the guide.

    Spin up a Celestia testnet

    If you want to spin up a quick testnet with your friends, you can follow these steps. Unless otherwise noted, every step must be done by everyone who wants to participate in this testnet.

    Optional: Reset working directory

    If you have already initialized a working directory for celestia-appd in the past, you must clean up before reinitializing a new directory. You can do so by running the following command:

    sh
    celestia-appd tendermint unsafe-reset-all --home $HOME/.celestia-app
    celestia-appd tendermint unsafe-reset-all --home $HOME/.celestia-app

    Initialize a working directory

    Run the following command:

    sh
    VALIDATOR_NAME=validator1
    +CHAIN_ID=testnet
    +celestia-appd init $VALIDATOR_NAME --chain-id $CHAIN_ID
    VALIDATOR_NAME=validator1
    +CHAIN_ID=testnet
    +celestia-appd init $VALIDATOR_NAME --chain-id $CHAIN_ID
    • The value we will use for $VALIDATOR_NAME is validator1 but you should choose your own node name.
    • The value we will use for $CHAIN_ID is testnet. The $CHAIN_ID must remain the same for everyone participating in this network.

    Create a new key

    Next, run the following command:

    sh
    KEY_NAME=validator
    +celestia-appd keys add $KEY_NAME --keyring-backend test
    KEY_NAME=validator
    +celestia-appd keys add $KEY_NAME --keyring-backend test

    This will create a new key, with a name of your choosing. Save the output of this command somewhere; you'll need the address generated here later. Here, we set the value of our key $KEY_NAME to validator for demonstration.

    Add genesis account KeyName

    Run the following command:

    sh
    TIA_AMOUNT="10000000utia"
    +celestia-appd add-genesis-account $KEY_NAME $TIA_AMOUNT --keyring-backend test
    TIA_AMOUNT="10000000utia"
    +celestia-appd add-genesis-account $KEY_NAME $TIA_AMOUNT --keyring-backend test

    Here $KEY_NAME is the same key name as before.

    Optional: Adding other validators

    If other participants in your testnet also want to be validators, repeat the command above with the specific amount for their public keys.

    Once all the validators are added, the genesis.json file is created. You need to share it with all other validators in your testnet in order for everyone to proceed with the following step.

    You can find the genesis.json at $HOME/.celestia-app/config/genesis.json

    Create the genesis transaction for new chain

    Run the following command:

    sh
    STAKING_AMOUNT=9000000utia
    +celestia-appd gentx $KEY_NAME $STAKING_AMOUNT --chain-id $CHAIN_ID \\
    +  --keyring-backend test
    STAKING_AMOUNT=9000000utia
    +celestia-appd gentx $KEY_NAME $STAKING_AMOUNT --chain-id $CHAIN_ID \\
    +  --keyring-backend test

    This will create the genesis transaction for your new chain. Here $STAKING_AMOUNT should be at least 9000000utia. If you provide too much or too little, you will encounter an error when starting your node.

    You will find the generated gentx JSON file inside $HOME/.celestia-app/config/gentx/gentx-$KEY_NAME.json

    NOTE

    If you have other validators in your network, they need to also run the above command with the genesis.json file you shared with them in the previous step.

    Creating the genesis JSON file

    Once all participants have submitted their gentx JSON files to you, you will pull all those gentx files inside the following directory: $HOME/.celestia-appd/config/gentx and use them to create the final genesis.json file.

    Once you added the gentx files of all the participants, run the following command:

    sh
    celestia-appd collect-gentxs
    celestia-appd collect-gentxs

    This command will look for the gentx files in this repo which should be moved to the following directory $HOME/.celestia-app/config/gentx.

    It will update the genesis.json file after in this location $HOME/.celestia-app/config/genesis.json which now includes the gentx of other participants.

    You should then share this final genesis.json file with all the other participants who must add it to their $HOME/.celestia-app/config directory.

    Everyone must ensure that they replace their existing genesis.json file with this new one created.

    Modify your config file

    Open the following file $HOME/.celestia-app/config/config.toml to modify it.

    Inside the file, add the other participants by modifying the following line to include other participants as persistent peers:

    text
    # Comma separated list of nodes to keep persistent connections to
    +persistent_peers = "[validator_address]@[ip_address]:[port],[validator_address]@[ip_address]:[port]"
    # Comma separated list of nodes to keep persistent connections to
    +persistent_peers = "[validator_address]@[ip_address]:[port],[validator_address]@[ip_address]:[port]"

    Add your node as a persistent peer

    The following allows you to share your node as a persistent peer that you can share in the networks repo or with others so other participants can peer with you.

    Run the following command:

    sh
    IP_ADDRESS=$(curl ifconfig.me)
    +NODE_ID=$(celestia-appd tendermint show-node-id)
    +PORT_NUMBER=26656
    IP_ADDRESS=$(curl ifconfig.me)
    +NODE_ID=$(celestia-appd tendermint show-node-id)
    +PORT_NUMBER=26656

    Note that the default port is 26656

    Now you can run the following command to output your validator node address:

    sh
    PEER="$NODE_ID@$IP_ADDRESS:$PORT_NUMBER"
    +echo $PEER
    PEER="$NODE_ID@$IP_ADDRESS:$PORT_NUMBER"
    +echo $PEER

    The output is your validator node address which you can share with other validators so they can peer with you.

    Instantiate the network

    You can start your node by running the following command:

    sh
    celestia-appd start
    celestia-appd start

    TIP

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    Now you have a new Celestia testnet to play around with!

    `,60),l=[t];function p(i,r,c,d,h,y){return s(),a("div",null,l)}const g=e(o,[["render",p]]);export{E as __pageData,g as default}; diff --git a/assets/nodes_instantiate-testnet.md.db07a97a.lean.js b/assets/nodes_instantiate-testnet.md.db07a97a.lean.js new file mode 100644 index 00000000000..b900ee6d192 --- /dev/null +++ b/assets/nodes_instantiate-testnet.md.db07a97a.lean.js @@ -0,0 +1 @@ +import{_ as e,o as s,c as a,Q as n}from"./chunks/framework.51d6c45b.js";const E=JSON.parse('{"title":"Celestia App network instantiation guide","description":"A guide that helps you instantiate a new testnetwork with Celestia App.","frontmatter":{"description":"A guide that helps you instantiate a new testnetwork with Celestia App.","head":[["meta",{"name":"og:title","content":"Celestia App network instantiation guide | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/instantiate-testnet.md","filePath":"nodes/instantiate-testnet.md","lastUpdated":1711744858000}'),o={name:"nodes/instantiate-testnet.md"},t=n("",60),l=[t];function p(i,r,c,d,h,y){return s(),a("div",null,l)}const g=e(o,[["render",p]]);export{E as __pageData,g as default}; diff --git a/assets/nodes_light-node.md.92409172.js b/assets/nodes_light-node.md.92409172.js new file mode 100644 index 00000000000..f950d5fa0fe --- /dev/null +++ b/assets/nodes_light-node.md.92409172.js @@ -0,0 +1,25 @@ +import{_ as s,o as a,c as n,Q as o}from"./chunks/framework.51d6c45b.js";const l="/img/nodes/LightNodes.png",F=JSON.parse('{"title":"Setting up a Celestia light node","description":"This tutorial covers setting up a Celestia light node.","frontmatter":{"sidebar_label":"Light node","description":"This tutorial covers setting up a Celestia light node.","head":[["meta",{"name":"og:title","content":"Setting up a Celestia light node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/light-node.md","filePath":"nodes/light-node.md","lastUpdated":1707523743000}'),e={name:"nodes/light-node.md"},p=o('

    Setting up a Celestia light node

    This tutorial will guide you through setting up a Celestia light node, which will allow you to perform data availability sampling (DAS) on Celestia's data availability (DA) network.

    Overview of light nodes

    Light nodes ensure data availability. This is the most common way to interact with Celestia networks.

    light-node

    Light nodes have the following behavior:

    1. They listen for ExtendedHeaders, i.e. wrapped “raw” headers, that notify Celestia nodes of new block headers and relevant DA metadata.
    2. They perform DAS on the received headers

    Hardware requirements

    The following minimum hardware requirements are recommended for running a light node:

    • Memory: 500 MB RAM (minimum)
    • CPU: Single Core
    • Disk: 50 GB SSD Storage
    • Bandwidth: 56 Kbps for Download/56 Kbps for Upload

    Setting up your light node

    This tutorial was performed on an Ubuntu Linux 20.04 (LTS) x64 instance machine.

    Set up dependencies on the setting up environment page.

    Install celestia-node

    Install the celestia binary by building and installing celestia-node.

    Initialize the light node

    Run the following command:

    sh
    celestia light init
    celestia light init
    sh
    celestia light init --p2p.network mocha
    celestia light init --p2p.network mocha
    sh
    celestia light init --p2p.network arabica
    celestia light init --p2p.network arabica

    The output in your terminal will show the location of your node store and config. It will also show confirmation that the node store has been initialized.

    Start the light node

    Start the light node with a connection to a validator node's gRPC endpoint (which is usually exposed on port 9090):

    In order for access to the ability to get and submit state-related information, such as the ability to submit PayForBlobs transactions, or query for the node's account balance, a gRPC endpoint of a validator (core) node must be passed as directed below.

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    To start the light node with a connection to a validator node's gRPC endpoint (which is usually exposed on port 9090):

    sh
    celestia light start --core.ip <URI> --p2p.network <network>
    celestia light start --core.ip <URI> --p2p.network <network>

    TIP

    You do not need to declare a network for Mainnet Beta. Refer to the chain ID section on the troubleshooting page for more information

    Using an RPC of your own, or one from the list on the Mocha testnet page or list on the Arabica devnet page, start your node.

    For example, your command might look something like this for Mocha:

    sh
    celestia light start --core.ip rpc-mocha.pops.one --p2p.network mocha
    celestia light start --core.ip rpc-mocha.pops.one --p2p.network mocha

    Or for Arabica:

    sh
    celestia light start --core.ip validator-1.celestia-arabica-11.com \\
    +    --p2p.network arabica
    celestia light start --core.ip validator-1.celestia-arabica-11.com \\
    +    --p2p.network arabica

    Keys and wallets

    You can create your key for your node by running the following command with the cel-key utility in the celestia-node directory:

    sh
    ./cel-key add <key-name> --keyring-backend test \\
    +    --node.type light --p2p.network <network>
    ./cel-key add <key-name> --keyring-backend test \\
    +    --node.type light --p2p.network <network>

    You can start your light node with the key created above by running the following command:

    sh
    celestia light start --keyring.accname my_celes_key \\
    +    --core.ip consensus.lunaroasis.net
    celestia light start --keyring.accname my_celes_key \\
    +    --core.ip consensus.lunaroasis.net
    sh
    celestia light start --keyring.accname my_celes_key \\
    +    --core.ip rpc-mocha.pops.one --p2p.network mocha
    celestia light start --keyring.accname my_celes_key \\
    +    --core.ip rpc-mocha.pops.one --p2p.network mocha
    sh
    celestia light start --keyring.accname my_celes_key \\
    +    --core.ip validator-1.celestia-arabica-11.com \\
    +    --p2p.network arabica
    celestia light start --keyring.accname my_celes_key \\
    +    --core.ip validator-1.celestia-arabica-11.com \\
    +    --p2p.network arabica

    Once you start the light node, a wallet key will be generated for you. You will need to fund that address with testnet tokens to pay for PayForBlob transactions.

    You can find the address using the RPC CLI or by running the following command in the celestia-node directory:

    sh
    ./cel-key list --node.type light --keyring-backend test \\
    +    --p2p.network <network>
    ./cel-key list --node.type light --keyring-backend test \\
    +    --p2p.network <network>

    Testnet tokens

    You have two networks to get testnet tokens from:

    You can request funds to your wallet address using the following command in Discord:

    console
    $request <CELESTIA-ADDRESS>
    $request <CELESTIA-ADDRESS>

    Where <CELESTIA-ADDRESS> is the celestia1****** address generated when you created the wallet.

    Optional: run the light node with a custom key

    In order to run a light node using a custom key:

    1. The custom key must exist inside the celestia light node directory at the correct path (default: ~/.celestia-light/keys/keyring-test)
    2. The name of the custom key must be passed upon start, like so:
    sh
    celestia light start --core.ip <URI> \\
    +    --keyring.accname <name-of-custom-key> \\
    celestia light start --core.ip <URI> \\
    +    --keyring.accname <name-of-custom-key> \\
    sh
    celestia light start --core.ip <URI> \\
    +    --keyring.accname <name-of-custom-key> \\
    +    --p2p.network arabica
    celestia light start --core.ip <URI> \\
    +    --keyring.accname <name-of-custom-key> \\
    +    --p2p.network arabica
    sh
    celestia light start --core.ip <URI> \\
    +    --keyring.accname <name-of-custom-key> \\
    +    --p2p.network mocha
    celestia light start --core.ip <URI> \\
    +    --keyring.accname <name-of-custom-key> \\
    +    --p2p.network mocha

    Optional: start light node with SystemD

    Follow the tutorial on setting up the light node as a background process with SystemD.

    Data availability sampling

    With your light node running, you can check out this tutorial on submitting PayForBlob transactions.

    `,53),t=[p];function c(r,i,y,E,d,h){return a(),n("div",null,t)}const u=s(e,[["render",c]]);export{F as __pageData,u as default}; diff --git a/assets/nodes_light-node.md.92409172.lean.js b/assets/nodes_light-node.md.92409172.lean.js new file mode 100644 index 00000000000..5638e457fa0 --- /dev/null +++ b/assets/nodes_light-node.md.92409172.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as o}from"./chunks/framework.51d6c45b.js";const l="/img/nodes/LightNodes.png",F=JSON.parse('{"title":"Setting up a Celestia light node","description":"This tutorial covers setting up a Celestia light node.","frontmatter":{"sidebar_label":"Light node","description":"This tutorial covers setting up a Celestia light node.","head":[["meta",{"name":"og:title","content":"Setting up a Celestia light node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/light-node.md","filePath":"nodes/light-node.md","lastUpdated":1707523743000}'),e={name:"nodes/light-node.md"},p=o("",53),t=[p];function c(r,i,y,E,d,h){return a(),n("div",null,t)}const u=s(e,[["render",c]]);export{F as __pageData,u as default}; diff --git a/assets/nodes_mainnet.md.dd077830.js b/assets/nodes_mainnet.md.dd077830.js new file mode 100644 index 00000000000..3ad974ba2fb --- /dev/null +++ b/assets/nodes_mainnet.md.dd077830.js @@ -0,0 +1,3 @@ +import{M as T}from"./chunks/MainnetVersionTags.4372f6f0.js";import{c as d}from"./chunks/constants.dc70dfe7.js";import{_ as Q,o,c as l,k as e,t as s,H as r,Q as a,a as p}from"./chunks/framework.51d6c45b.js";import"./chunks/mainnet_versions.84d9fd1c.js";const h="/img/Mainnet-Beta.png",m="/grove/grove-sandbox.png",u={name:"MainnetBetaDetails",data(){return{constants:d}}},g=e("tr",null,[e("th",null,"Detail"),e("th",null,"Value")],-1),H=e("td",null,"Chain ID",-1),y=e("tr",null,[e("td",null,"Genesis hash"),e("td",null,[e("code",null,"6BE39EFD10BA412A9DB5288488303F5DD32CF386707A5BEF33617F4C43301872")])],-1),f=e("td",null,"Genesis file",-1),V=["href"],b=e("td",null,"Peers file",-1),k=["href"],L=e("tr",null,[e("td",null,"Validators"),e("td",null," 100 ")],-1);function w(n,i,c,D,t,q){return o(),l("table",null,[g,e("tr",null,[H,e("td",null,[e("code",null,s(t.constants.mainnetChainId),1)])]),y,e("tr",null,[f,e("td",null,[e("a",{href:`https://github.com/celestiaorg/networks/blob/master/${t.constants.mainnetChainId}/genesis.json`,target:"_blank",rel:"noopener noreferrer"}," https://github.com/celestiaorg/networks/blob/master/"+s(t.constants.mainnetChainId)+"/genesis.json ",9,V)])]),e("tr",null,[b,e("td",null,[e("a",{href:`https://github.com/celestiaorg/networks/blob/master/${t.constants.mainnetChainId}/peers.txt`,target:"_blank",rel:"noopener noreferrer"}," https://github.com/celestiaorg/networks/blob/master/"+s(t.constants.mainnetChainId)+"/peers.txt ",9,k)])]),L])}const _=Q(u,[["render",w]]),v=a('

    Mainnet Beta

    Mainnet Beta

    Welcome to the guide for Celestia’s Mainnet Beta, the production network that marks the pinnacle of Celestia’s evolution since its inception in 2019. This network is where all components of the Celestia ecosystem come to life in a real-world environment.

    Mainnet Beta is the culmination of rigorous community testing, upgrades, and feedback. It serves as the platform for deploying mainnet rollups and applications.

    Network stability and upgrades

    Mainnet Beta is a stable network, but will still receive updates and improvements. Any changes or upgrades will be coordinated with node operators and the broader Celestia community to ensure seamless integration and minimal service interruptions.

    As we step into unexplored territories with groundbreaking technologies like data availability sampling, it's crucial to remember that Mainnet Beta remains experimental at this stage. While the network is live and functional, users may encounter occasional instability or reduced performance.

    Network details

    ',8),M=e("h2",{id:"software-version-numbers",tabindex:"-1"},[p("Software version numbers "),e("a",{class:"header-anchor",href:"#software-version-numbers","aria-label":'Permalink to "Software version numbers"'},"​")],-1),x=a('

    Network parameters

    Full network parameters, such as max bytes, can be found in the celestia-app specifications.

    CIP-13 has been drafted to create a living document for these parameters as a part of the CIP process.

    Maximum bytes

    There is a hard limit on the total blob size in a transaction, which is determined by the effective maximum square size. Given that the current governance maximum square size is 64, the total blob size in a transaction must be slightly less than ~2 MiB, or 1,973,786 bytes to be exact.

    The following provides an approximation of the maximum block size:

    • The maximum square size is 64x64, which gives us 4096 shares.
    • One share is reserved for the PFB transaction, leaving us with 4095 shares.
    • The first sparse share has 478 bytes available, and the remaining sparse shares have 482 bytes each.

    This can be calculated as follows:

    ',8),Z={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},C={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.566ex"},xmlns:"http://www.w3.org/2000/svg",width:"68.588ex",height:"2.262ex",role:"img",focusable:"false",viewBox:"0 -750 30315.8 1000","aria-hidden":"true"},E=a('',1),P=[E],F=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mtext",null,"Total Bytes"),e("mo",null,"="),e("mo",{stretchy:"false"},"("),e("mn",null,"1"),e("mo",null,"×"),e("mn",null,"478"),e("mstyle",{scriptlevel:"0"},[e("mspace",{width:"0.167em"})]),e("mtext",null,"bytes"),e("mo",{stretchy:"false"},")"),e("mo",null,"+"),e("mo",{stretchy:"false"},"("),e("mn",null,"4094"),e("mo",null,"×"),e("mn",null,"482"),e("mstyle",{scriptlevel:"0"},[e("mspace",{width:"0.167em"})]),e("mtext",null,"bytes"),e("mo",{stretchy:"false"},")"),e("mo",null,"="),e("mn",null,"1"),e("mo",null,","),e("mn",null,"973"),e("mo",null,","),e("mn",null,"786"),e("mstyle",{scriptlevel:"0"},[e("mspace",{width:"0.167em"})]),e("mtext",null,"bytes")])],-1),A=a(`

    Please note that there isn't a precise upper bound on the maximum total blob size. It depends on several factors:

    • The maximum square size, which is determined by a governance parameter and a versioned constant.
    • The maximum bytes in a block, which is determined by a governance parameter and a hard-coded constant in CometBFT.
    • The number of shares occupied by the PFB transaction share.

    These factors can cause the maximum total blob size that can be included in one block to vary.

    Integrations

    This guide contains the relevant sections for how to connect to Mainnet Beta, depending on the type of node you are running. Your best approach to participating is to first determine which node you would like to run. Each node’s guide will link to the relevant network in order to show you how to connect to them. Learn about the different endpoint types in the Cosmos SDK documentation.

    Here is a list of options of the types of nodes you can run in order to participate in Mainnet Beta:

    Consensus nodes

    Consensus RPC endpoints

    • public-celestia-rpc.numia.xyz
    • celestia-rpc.mesa.newmetric.xyz
    • rpc.celestia.pops.one
    • rpc.lunaroasis.net
    • rpc.celestia.nodestake.top
    • celestia-rpc.brightlystake.com
    • celestia-rpc.spidey.services
    • rpc-celestia.contributiondao.com
    • celestia.rpc.stakin-nodes.com
    • celestia.cumulo.org.es
    • rpc-celestia.mzonder.com
    • rpc-celestia-01.stakeflow.io
    • rpc-celestia.alphab.ai
    • rpc-celestia-full.avril14th.org
    • celestia-rpc.easy2stake.com
    • celestia.rpc.kjnodes.com
    • celestia-rpc.0xcryptovestor.com
    • rpc-celestia-mainnet.trusted-point.com
    • celestia.rpc.archives.validao.xyz
    • rpc-archive.celestia.bitszn.com
    • celestia-rpc.f5nodes.com
    • celestia-rpc.chainode.tech:33373
    • https://rpc-celestia.staker.space/
    • https://celestia-rpc.noders.services/

    API endpoints

    • public-celestia-lcd.numia.xyz
    • celestia-rest.mesa.newmetric.xyz
    • api.celestia.pops.one
    • api.lunaroasis.net
    • api.celestia.nodestake.top
    • celestia-rpc.brightlystake.com/api
    • celestia-api.spidey.services
    • api-celestia.contributiondao.com
    • celestia.rest.stakin-nodes.com
    • celestia.api.cumulo.org.es
    • api-celestia.mzonder.com
    • api-celestia-01.stakeflow.io
    • api-celestia.alphab.ai
    • api-celestia-full.avril14th.org
    • celestia-lcd.easy2stake.com
    • celestia.api.kjnodes.com
    • api-celestia-mainnet.trusted-point.com
    • celestia.rest.archives.validao.xyz
    • api-archive.celestia.bitszn.com
    • celestia-api.f5nodes.com
    • https://celestia-api.chainode.tech
    • https://api-celestia.staker.space/
    • https://celestia-api.noders.services/

    gRPC endpoints

    • public-celestia-grpc.numia.xyz
    • celestia-grpc.mesa.newmetric.xyz
    • grpc.celestia.pops.one
    • grpc.lunaroasis.net:443
    • grpc.celestia.nodestake.top
    • celestia-rpc.brightlystake.com:9090
    • celestia-grpc.spidey.services
    • grpc-celestia.contributiondao.com
    • celestia.grpc.stakin-nodes.com:443
    • celestia.grpc.cumulo.org.es:443
    • grpc-celestia.mzonder.com:443
    • grpc-celestia-01.stakeflow.io:15002
    • rpc-celestia.alphab.ai:9090
    • grpc-celestia-full.avril14th.org
    • celestia.grpc.kjnodes.com:443
    • grpc-celestia-mainnet.trusted-point.com:9095
    • celestia.grpc.archives.validao.xyz:9090
    • gprc-archive.celestia.bitszn.com
    • celestia-grpc.f5nodes.com:9390
    • celestia-grpc.chainode.tech:443
    • https://grpc-celestia.staker.space/
    • celestia-grpc.noders.services:11090

    WebSocket endpoints

    • wss://celestia-ws.chainode.tech:33373/websocket

    Data availability nodes

    Data availability (DA) RPC endpoints

    These RPC endpoints for DA nodes are to provide state access for querying the chain’s state and broadcasting transactions (balances, blobs, etc.) to the Celestia network. For users, they will need to provide a –core.ip string from a consensus node’s URL or IP that populates 2 ports for 2 types (RPC and gRPC, at ports 26657 and 9090, respectively) to their respective DA node.

    TIP

    bash
    celestia <da_type> start --core.ip <url> –core.rpc.port <port> \\
    +    –core.grpc.port <port>
    celestia <da_type> start --core.ip <url> –core.rpc.port <port> \\
    +    –core.grpc.port <port>

    Bridge nodes

    Not all RPC endpoints guarantee the full block history. Find an archive endpoint on the community dashboard or run your own full consensus node with no pruning for your bridge node.

    RPCs for DA nodes to initialise or start your celestia-node to Mainnet Beta with:

    • public-celestia-consensus.numia.xyz
      • gRPC: port 9090
      • RPC: port 26657
    • celestia-consensus.mesa.newmetric.xyz
      • gRPC: port 9090
      • RPC: port 26657
    • rpc.celestia.pops.one
      • gRPC: port 9090
      • RPC: port 26657
    • consensus.lunaroasis.net
      • gRPC: port 9090
      • RPC: port 26657
    • rpc-celestia.alphab.ai
      • gRPC: port 9090
      • RPC: port 26657

    DA full and light nodes might have troubles connecting to the networks, so you can check out this Grafana dashboard to see health/uptime status of DA bootstrappers (now celestia network only).

    You can find the status of these endpoints.

    Archival DA RPC endpoints

    By default, light nodes prune recent data to save on storage space. Archival data availability (DA) nodes store the entire history of the chain without pruning any data so all data available data is retrievable. You can read more about light vs archival nodes.

    Grove archival endpoints

    You can provision your own Celestia Archival endpoint on Grove. Learn more about Celestia on Grove, or find the fully supported spec.

    There is a sandbox you can leverage for testing straight in your browser:

    grove-sandbox

    Explorers

    There are multiple explorers you can use for Mainnet Beta:

    Analytics

    The following websites provide analytics for Celestia:

    Network upgrades

    Join the Community Telegram announcement channel for network upgrades.

    ',40),N=JSON.parse('{"title":"Mainnet Beta","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Mainnet Beta | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/mainnet.md","filePath":"nodes/mainnet.md","lastUpdated":1713191731000}'),B={name:"nodes/mainnet.md"},j=Object.assign(B,{setup(n){return(i,c)=>(o(),l("div",null,[v,r(_),M,r(T),x,e("p",null,[e("mjx-container",Z,[(o(),l("svg",C,P)),F])]),A]))}});export{N as __pageData,j as default}; diff --git a/assets/nodes_mainnet.md.dd077830.lean.js b/assets/nodes_mainnet.md.dd077830.lean.js new file mode 100644 index 00000000000..e0d8c0a16df --- /dev/null +++ b/assets/nodes_mainnet.md.dd077830.lean.js @@ -0,0 +1 @@ +import{M as T}from"./chunks/MainnetVersionTags.4372f6f0.js";import{c as d}from"./chunks/constants.dc70dfe7.js";import{_ as Q,o,c as l,k as e,t as s,H as r,Q as a,a as p}from"./chunks/framework.51d6c45b.js";import"./chunks/mainnet_versions.84d9fd1c.js";const h="/img/Mainnet-Beta.png",m="/grove/grove-sandbox.png",u={name:"MainnetBetaDetails",data(){return{constants:d}}},g=e("tr",null,[e("th",null,"Detail"),e("th",null,"Value")],-1),H=e("td",null,"Chain ID",-1),y=e("tr",null,[e("td",null,"Genesis hash"),e("td",null,[e("code",null,"6BE39EFD10BA412A9DB5288488303F5DD32CF386707A5BEF33617F4C43301872")])],-1),f=e("td",null,"Genesis file",-1),V=["href"],b=e("td",null,"Peers file",-1),k=["href"],L=e("tr",null,[e("td",null,"Validators"),e("td",null," 100 ")],-1);function w(n,i,c,D,t,q){return o(),l("table",null,[g,e("tr",null,[H,e("td",null,[e("code",null,s(t.constants.mainnetChainId),1)])]),y,e("tr",null,[f,e("td",null,[e("a",{href:`https://github.com/celestiaorg/networks/blob/master/${t.constants.mainnetChainId}/genesis.json`,target:"_blank",rel:"noopener noreferrer"}," https://github.com/celestiaorg/networks/blob/master/"+s(t.constants.mainnetChainId)+"/genesis.json ",9,V)])]),e("tr",null,[b,e("td",null,[e("a",{href:`https://github.com/celestiaorg/networks/blob/master/${t.constants.mainnetChainId}/peers.txt`,target:"_blank",rel:"noopener noreferrer"}," https://github.com/celestiaorg/networks/blob/master/"+s(t.constants.mainnetChainId)+"/peers.txt ",9,k)])]),L])}const _=Q(u,[["render",w]]),v=a("",8),M=e("h2",{id:"software-version-numbers",tabindex:"-1"},[p("Software version numbers "),e("a",{class:"header-anchor",href:"#software-version-numbers","aria-label":'Permalink to "Software version numbers"'},"​")],-1),x=a("",8),Z={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},C={style:{overflow:"visible","min-height":"1px","min-width":"1px","vertical-align":"-0.566ex"},xmlns:"http://www.w3.org/2000/svg",width:"68.588ex",height:"2.262ex",role:"img",focusable:"false",viewBox:"0 -750 30315.8 1000","aria-hidden":"true"},E=a("",1),P=[E],F=e("mjx-assistive-mml",{unselectable:"on",display:"inline",style:{top:"0px",left:"0px",clip:"rect(1px, 1px, 1px, 1px)","-webkit-touch-callout":"none","-webkit-user-select":"none","-khtml-user-select":"none","-moz-user-select":"none","-ms-user-select":"none","user-select":"none",position:"absolute",padding:"1px 0px 0px 0px",border:"0px",display:"block",width:"auto",overflow:"hidden"}},[e("math",{xmlns:"http://www.w3.org/1998/Math/MathML"},[e("mtext",null,"Total Bytes"),e("mo",null,"="),e("mo",{stretchy:"false"},"("),e("mn",null,"1"),e("mo",null,"×"),e("mn",null,"478"),e("mstyle",{scriptlevel:"0"},[e("mspace",{width:"0.167em"})]),e("mtext",null,"bytes"),e("mo",{stretchy:"false"},")"),e("mo",null,"+"),e("mo",{stretchy:"false"},"("),e("mn",null,"4094"),e("mo",null,"×"),e("mn",null,"482"),e("mstyle",{scriptlevel:"0"},[e("mspace",{width:"0.167em"})]),e("mtext",null,"bytes"),e("mo",{stretchy:"false"},")"),e("mo",null,"="),e("mn",null,"1"),e("mo",null,","),e("mn",null,"973"),e("mo",null,","),e("mn",null,"786"),e("mstyle",{scriptlevel:"0"},[e("mspace",{width:"0.167em"})]),e("mtext",null,"bytes")])],-1),A=a("",40),N=JSON.parse('{"title":"Mainnet Beta","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Mainnet Beta | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/mainnet.md","filePath":"nodes/mainnet.md","lastUpdated":1713191731000}'),B={name:"nodes/mainnet.md"},j=Object.assign(B,{setup(n){return(i,c)=>(o(),l("div",null,[v,r(_),M,r(T),x,e("p",null,[e("mjx-container",Z,[(o(),l("svg",C,P)),F])]),A]))}});export{N as __pageData,j as default}; diff --git a/assets/nodes_mocha-testnet.md.d00a2606.js b/assets/nodes_mocha-testnet.md.d00a2606.js new file mode 100644 index 00000000000..2b61550418d --- /dev/null +++ b/assets/nodes_mocha-testnet.md.d00a2606.js @@ -0,0 +1 @@ +import{M as d}from"./chunks/MochaVersionTags.139813c5.js";import{c as h}from"./chunks/constants.dc70dfe7.js";import{_ as p,o as s,c as l,k as e,t as o,H as a,Q as i,a as u}from"./chunks/framework.51d6c45b.js";import"./chunks/mocha_versions.3ba4a823.js";const m="/img/mocha.jpg",f={name:"MochaTestnetDetails",data(){return{constants:h}}},g=e("tr",null,[e("th",null,"Detail"),e("th",null,"Value")],-1),b=e("td",null,"Chain ID",-1),_=e("tr",null,[e("td",null,"Genesis hash"),e("td",null,[e("code",null,"B93BBE20A0FBFDF955811B6420F8433904664D45DB4BF51022BE4200C1A1680D")])],-1),k=e("td",null,"Genesis file",-1),w=["href"],T=e("td",null,"Peers file",-1),v=["href"],C=e("tr",null,[e("td",null,"Validators"),e("td",null," 100 ")],-1);function y(n,c,r,x,t,I){return s(),l("table",null,[g,e("tr",null,[b,e("td",null,[e("code",null,o(t.constants.mochaChainId),1)])]),_,e("tr",null,[k,e("td",null,[e("a",{href:`https://github.com/celestiaorg/networks/blob/master/${t.constants.mochaChainId}/genesis.json`,target:"_blank",rel:"noopener noreferrer"}," https://github.com/celestiaorg/networks/blob/master/"+o(t.constants.mochaChainId)+"/genesis.json ",9,w)])]),e("tr",null,[T,e("td",null,[e("a",{href:`https://github.com/celestiaorg/networks/blob/master/${t.constants.mochaChainId}/peers.txt`,target:"_blank",rel:"noopener noreferrer"}," https://github.com/celestiaorg/networks/blob/master/"+o(t.constants.mochaChainId)+"/peers.txt ",9,v)])]),C])}const P=p(f,[["render",y]]),E=i('

    Mocha testnet

    mocha-testnet

    This guide contains the relevant sections for how to connect to Mocha, depending on the type of node you are running. Mocha testnet is designed to help validators test out their infrastructure and node software. Developers are encouraged to deploy their sovereign rollups on Mocha, but we also recommend Arabica devnet for that as it is designed for development purposes.

    Mocha is a milestone in Celestia, allowing everyone to test out core functionalities on the network. Read the announcement. Your best approach to participating is to first determine which node you would like to run. Each node's guide will link to the relevant networks, to show you how to connect to them.

    You have a list of options on the types of nodes you can run to participate in Mocha:

    Consensus:

    Data Availability:

    Select the type of node you would like to run and follow the instructions on each respective page. Whenever you are asked to select the type of network you want to connect to in those guides, select Mocha to refer to the correct instructions on this page on how to connect to Mocha.

    Network details

    ',11),S=e("h2",{id:"software-version-numbers",tabindex:"-1"},[u("Software version numbers "),e("a",{class:"header-anchor",href:"#software-version-numbers","aria-label":'Permalink to "Software version numbers"'},"​")],-1),A=i('

    RPC for DA bridge, full, and light nodes

    These RPC endpoints for DA nodes are to provide state access for querying the chain’s state and broadcasting transactions (balances, blobs, etc.) to the Celestia network. For users, they will need to provide a –core.ip string from a consensus node’s URL or IP that populates 2 ports for 2 types (RPC and gRPC, at ports 26657 and 9090, respectively) to their respective DA node.

    Bridge nodes

    Mentioned below RPC endpoints do not guarantee you the download of full blocks from them. We advise that if you are running a bridge node, that you also run a local full consensus node in order to download full blocks from it.

    • public-celestia-mocha4-consensus.numia.xyz
    • mocha-4-consensus.mesa.newmetric.xyz
    • full.consensus.mocha-4.celestia-mocha.com
    • consensus-full-mocha-4.celestia-mocha.com
    • rpc-mocha.pops.one

    RPC endpoints

    The RPC endpoint is to allow users to interact with Celestia's nodes by querying the node's state and broadcasting transactions on the Celestia network. The default port is 26657.

    • public-celestia-mocha4-consensus.numia.xyz:26657
    • mocha-4-consensus.mesa.newmetric.xyz:26657
    • rpc.celestia-mocha.com
    • rpc-2.celestia-mocha.com
    • celestia-testnet-rpc.f5nodes.com
    • celestia-testnet.brightlystake.com
    • rpc-celestia-mocha.architectnodes.com
    • rpc-celestia-mocha.trusted-point.com
    • rpc-celestia-testnet-01.stakeflow.io
    • mocha.celestia.rpc.cumulo.me
    • rpc-mocha-4.spidey.services
    • rpc-mocha-full.avril14th.org
    • rpc.mocha.bitszn.com
    • celestia-t-rpc.noders.services/

    API endpoints

    The API endpoint is to allow users to interact with the REST API in Cosmos SDK which is implemented using gRPC-gateway, which exposes gRPC endpoints as REST endpoints. This allows for communication with the node using REST calls, which can be useful if the client does not support gRPC or HTTP2. The default port is 1317.

    gRPC endpoints

    The gRPC endpoint is to allow users to interact with a Celestia Node using gRPC, a modern open-source and high-performance RPC framework. The default port is 9090. In the Cosmos SDK, gRPC is used to define state queries and broadcast transactions.

    • public-celestia-mocha4-consensus.numia.xyz:9090
    • mocha-4-consensus.mesa.newmetric.xyz:9090
    • grpc-mocha.pops.one
    • grpc.celestia-mocha.com:443
    • grpc-2.celestia-mocha.com:443
    • full.consensus.mocha-4.celestia-mocha.com:9090
    • consensus-full-mocha-4.celestia-mocha.com:9090
    • celestia-testnet-grpc.f5nodes.com
    • celestia-testnet.brightlystake.com:9390
    • grpc-celestia-mocha.architectnodes.com:1443
    • grpc-celestia-mocha.trusted-point.com:9099
    • grpc-celestia-testnet-01.stakeflow.io:16002
    • mocha.grpc.cumulo.me:443
    • grpc-mocha-4.spidey.services
    • grpc-mocha-full.avril14th.org
    • grpc.mocha.bitszn.com
    • celestia-grpc.noders.services:21090

    Bridge and full node endpoints

    The endpoints below are for bridge and full nodes only. They can be used to find bootstrapper peers in the p2p network.

    Bridge node 1:

    • da-bridge-mocha-4.celestia-mocha.com
    • bridge-mocha-4.da.celestia-mocha.com

    Bridge node 2:

    • da-bridge-mocha-4-2.celestia-mocha.com
    • bridge-mocha-4-2.da.celestia-mocha.com

    Full node 1:

    • da-full-1-mocha-4.celestia-mocha.com
    • full-1-mocha-4.da.celestia-mocha.com

    Full node 2:

    • da-full-2-mocha-4.celestia-mocha.com
    • full-2-mocha-4.da.celestia-mocha.com

    Mocha testnet faucet

    WARNING

    USING THIS FAUCET DOES NOT ENTITLE YOU TO ANY AIRDROP OR OTHER DISTRIBUTION OF MAINNET CELESTIA TOKENS. THERE ARE NO PUBLIC SALES OF ANY MAINNET CELESTIA TOKENS.

    You can request from Mocha testnet Faucet on the #mocha-faucet channel on Celestia's Discord server with the following command:

    text
    $request <CELESTIA-ADDRESS>
    $request <CELESTIA-ADDRESS>

    Where <CELESTIA-ADDRESS> is a celestia1****** generated address.

    NOTE

    Faucet has a limit of 10 tokens per week per address/Discord ID.

    Explorers

    There are several explorers you can use for Mocha:

    Network upgrades

    Join our Telegram announcement channel for network upgrades.

    ',34),B=JSON.parse('{"title":"Mocha testnet","description":"Learn how to connect to the Mocha network.","frontmatter":{"description":"Learn how to connect to the Mocha network.","head":[["meta",{"name":"og:title","content":"Mocha testnet | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/mocha-testnet.md","filePath":"nodes/mocha-testnet.md","lastUpdated":1713191769000}'),R={name:"nodes/mocha-testnet.md"},F=Object.assign(R,{setup(n){return(c,r)=>(s(),l("div",null,[E,a(P),S,a(d),A]))}});export{B as __pageData,F as default}; diff --git a/assets/nodes_mocha-testnet.md.d00a2606.lean.js b/assets/nodes_mocha-testnet.md.d00a2606.lean.js new file mode 100644 index 00000000000..ad6461c3bc4 --- /dev/null +++ b/assets/nodes_mocha-testnet.md.d00a2606.lean.js @@ -0,0 +1 @@ +import{M as d}from"./chunks/MochaVersionTags.139813c5.js";import{c as h}from"./chunks/constants.dc70dfe7.js";import{_ as p,o as s,c as l,k as e,t as o,H as a,Q as i,a as u}from"./chunks/framework.51d6c45b.js";import"./chunks/mocha_versions.3ba4a823.js";const m="/img/mocha.jpg",f={name:"MochaTestnetDetails",data(){return{constants:h}}},g=e("tr",null,[e("th",null,"Detail"),e("th",null,"Value")],-1),b=e("td",null,"Chain ID",-1),_=e("tr",null,[e("td",null,"Genesis hash"),e("td",null,[e("code",null,"B93BBE20A0FBFDF955811B6420F8433904664D45DB4BF51022BE4200C1A1680D")])],-1),k=e("td",null,"Genesis file",-1),w=["href"],T=e("td",null,"Peers file",-1),v=["href"],C=e("tr",null,[e("td",null,"Validators"),e("td",null," 100 ")],-1);function y(n,c,r,x,t,I){return s(),l("table",null,[g,e("tr",null,[b,e("td",null,[e("code",null,o(t.constants.mochaChainId),1)])]),_,e("tr",null,[k,e("td",null,[e("a",{href:`https://github.com/celestiaorg/networks/blob/master/${t.constants.mochaChainId}/genesis.json`,target:"_blank",rel:"noopener noreferrer"}," https://github.com/celestiaorg/networks/blob/master/"+o(t.constants.mochaChainId)+"/genesis.json ",9,w)])]),e("tr",null,[T,e("td",null,[e("a",{href:`https://github.com/celestiaorg/networks/blob/master/${t.constants.mochaChainId}/peers.txt`,target:"_blank",rel:"noopener noreferrer"}," https://github.com/celestiaorg/networks/blob/master/"+o(t.constants.mochaChainId)+"/peers.txt ",9,v)])]),C])}const P=p(f,[["render",y]]),E=i("",11),S=e("h2",{id:"software-version-numbers",tabindex:"-1"},[u("Software version numbers "),e("a",{class:"header-anchor",href:"#software-version-numbers","aria-label":'Permalink to "Software version numbers"'},"​")],-1),A=i("",34),B=JSON.parse('{"title":"Mocha testnet","description":"Learn how to connect to the Mocha network.","frontmatter":{"description":"Learn how to connect to the Mocha network.","head":[["meta",{"name":"og:title","content":"Mocha testnet | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/mocha-testnet.md","filePath":"nodes/mocha-testnet.md","lastUpdated":1713191769000}'),R={name:"nodes/mocha-testnet.md"},F=Object.assign(R,{setup(n){return(c,r)=>(s(),l("div",null,[E,a(P),S,a(d),A]))}});export{B as __pageData,F as default}; diff --git a/assets/nodes_overview.md.b302dbe6.js b/assets/nodes_overview.md.b302dbe6.js new file mode 100644 index 00000000000..047ad48c04a --- /dev/null +++ b/assets/nodes_overview.md.b302dbe6.js @@ -0,0 +1 @@ +import{_ as e,o,c as t,Q as n}from"./chunks/framework.51d6c45b.js";const a="/img/node-requirements.jpg",_=JSON.parse('{"title":"Overview to running nodes on Celestia","description":"An overview on how to participate in the Celestia network.","frontmatter":{"description":"An overview on how to participate in the Celestia network.","head":[["meta",{"name":"og:title","content":"Overview to running nodes on Celestia | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/overview.md","filePath":"nodes/overview.md","lastUpdated":1711744858000}'),i={name:"nodes/overview.md"},s=n('

    Overview to running nodes on Celestia

    There are many ways you can participate in the Celestia networks.

    Celestia node operators can run several options on the network.

    Consensus:

    • Validator node: This type of node participates in consensus by producing and voting on blocks.
    • Full consensus node: A celestia-app Full node to sync blockchain history.

    Data Availability:

    • Bridge node: This node bridges blocks between the Data-Availability network and the Consensus network.
    • Full storage node: This node stores all the data but does not connect to Consensus.
    • Light node: Light clients conduct data availability sampling on the Data Availability network.

    You can learn more about how to set up each different node by going through each tutorial guide.

    Banner

    Please provide any feedback on the tutorials and guides. If you notice a bug or issue, feel free to make a pull request or write up a Github issue!

    ',10),r=[s];function l(d,c,p,u,h,v){return o(),t("div",null,r)}const f=e(i,[["render",l]]);export{_ as __pageData,f as default}; diff --git a/assets/nodes_overview.md.b302dbe6.lean.js b/assets/nodes_overview.md.b302dbe6.lean.js new file mode 100644 index 00000000000..1dcaefd785e --- /dev/null +++ b/assets/nodes_overview.md.b302dbe6.lean.js @@ -0,0 +1 @@ +import{_ as e,o,c as t,Q as n}from"./chunks/framework.51d6c45b.js";const a="/img/node-requirements.jpg",_=JSON.parse('{"title":"Overview to running nodes on Celestia","description":"An overview on how to participate in the Celestia network.","frontmatter":{"description":"An overview on how to participate in the Celestia network.","head":[["meta",{"name":"og:title","content":"Overview to running nodes on Celestia | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/overview.md","filePath":"nodes/overview.md","lastUpdated":1711744858000}'),i={name:"nodes/overview.md"},s=n("",10),r=[s];function l(d,c,p,u,h,v){return o(),t("div",null,r)}const f=e(i,[["render",l]]);export{_ as __pageData,f as default}; diff --git a/assets/nodes_participate.md.4b9d8d8b.js b/assets/nodes_participate.md.4b9d8d8b.js new file mode 100644 index 00000000000..40058266632 --- /dev/null +++ b/assets/nodes_participate.md.4b9d8d8b.js @@ -0,0 +1 @@ +import{A as r}from"./chunks/ArabicaVersionTags.e649ff5a.js";import{M as n}from"./chunks/MochaVersionTags.139813c5.js";import{M as i}from"./chunks/MainnetVersionTags.4372f6f0.js";import{o as s,c,H as a,Q as o,k as e,a as t}from"./chunks/framework.51d6c45b.js";import"./chunks/arabica_versions.d7daf483.js";import"./chunks/constants.dc70dfe7.js";import"./chunks/mocha_versions.3ba4a823.js";import"./chunks/mainnet_versions.84d9fd1c.js";const l=o('

    Participate in the Celestia networks

    Mainnet Beta

    Celestia’s Mainnet Beta is the production network for deploying mainnet rollups and applications. This marks the culmination of years of development and community testing. While the network is stable and continues to receive updates, it remains experimental and users may experience occasional instability or reduced performance.

    Compatible software versions for Mainnet Beta

    ',4),d=o('

    Testnets

    Celestia currently has two existing testnets that you can participate in:

    Arabica Devnet

    Arabica devnet is a devnet focused on developers who want to deploy sovereign rollups on the latest changes from Celestia's codebase. Arabica will be updated frequently and might be unstable at times given new updates. Validators won't be able to validate on Arabica as it is not designed for validators to participate.

    Compatible software versions for Arabica devnet

    ',5),h=o('

    Mocha testnet

    Mocha testnet is a testnet focused on enabling validators to test out their infrastructure by running nodes connected to the network. Developers can also deploy sovereign rollups on Mocha, it just will always be behind Arabica as Mocha upgrades are slower given they need to be done via hardforks in coordination with the validator community on Mocha.

    Compatible software versions for Mocha testnet

    ',3),p=e("h2",{id:"network-upgrades",tabindex:"-1"},[t("Network upgrades "),e("a",{class:"header-anchor",href:"#network-upgrades","aria-label":'Permalink to "Network upgrades"'},"​")],-1),m=e("p",null,[t("Join our "),e("a",{href:"https://t.me/+smSFIA7XXLU4MjJh",target:"_blank",rel:"noreferrer"},"Telegram announcement channel"),t(" for network upgrades.")],-1),M=JSON.parse('{"title":"Participate in the Celestia networks","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Participate in the Celestia networks | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/participate.md","filePath":"nodes/participate.md","lastUpdated":1710176202000}'),b={name:"nodes/participate.md"},q=Object.assign(b,{setup(f){return(_,u)=>(s(),c("div",null,[l,a(i),d,a(r),h,a(n),p,m]))}});export{M as __pageData,q as default}; diff --git a/assets/nodes_participate.md.4b9d8d8b.lean.js b/assets/nodes_participate.md.4b9d8d8b.lean.js new file mode 100644 index 00000000000..b93e7065216 --- /dev/null +++ b/assets/nodes_participate.md.4b9d8d8b.lean.js @@ -0,0 +1 @@ +import{A as r}from"./chunks/ArabicaVersionTags.e649ff5a.js";import{M as n}from"./chunks/MochaVersionTags.139813c5.js";import{M as i}from"./chunks/MainnetVersionTags.4372f6f0.js";import{o as s,c,H as a,Q as o,k as e,a as t}from"./chunks/framework.51d6c45b.js";import"./chunks/arabica_versions.d7daf483.js";import"./chunks/constants.dc70dfe7.js";import"./chunks/mocha_versions.3ba4a823.js";import"./chunks/mainnet_versions.84d9fd1c.js";const l=o("",4),d=o("",5),h=o("",3),p=e("h2",{id:"network-upgrades",tabindex:"-1"},[t("Network upgrades "),e("a",{class:"header-anchor",href:"#network-upgrades","aria-label":'Permalink to "Network upgrades"'},"​")],-1),m=e("p",null,[t("Join our "),e("a",{href:"https://t.me/+smSFIA7XXLU4MjJh",target:"_blank",rel:"noreferrer"},"Telegram announcement channel"),t(" for network upgrades.")],-1),M=JSON.parse('{"title":"Participate in the Celestia networks","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Participate in the Celestia networks | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/participate.md","filePath":"nodes/participate.md","lastUpdated":1710176202000}'),b={name:"nodes/participate.md"},q=Object.assign(b,{setup(f){return(_,u)=>(s(),c("div",null,[l,a(i),d,a(r),h,a(n),p,m]))}});export{M as __pageData,q as default}; diff --git a/assets/nodes_quick-start.md.05f1f426.js b/assets/nodes_quick-start.md.05f1f426.js new file mode 100644 index 00000000000..7cf253347ed --- /dev/null +++ b/assets/nodes_quick-start.md.05f1f426.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as o}from"./chunks/framework.51d6c45b.js";const f=JSON.parse('{"title":"Quick start guide","description":"Learn how to get started and run your first node on Celestia.","frontmatter":{"description":"Learn how to get started and run your first node on Celestia.","head":[["meta",{"name":"og:title","content":"Quick start guide | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/quick-start.md","filePath":"nodes/quick-start.md","lastUpdated":1698417238000}'),i={name:"nodes/quick-start.md"},n=o('

    Quick start guide

    In this section, we show you how to get started with installing the needed libraries and packages in Celestia to help you run a node on Celestia.

    Celestia Node

    Install celestia-node allows you to get started with running a light node and do data availability sampling.

    Light nodes are the best nodes to test out initially if you are new to participating in Celestia.

    celestia-node client also allows you to run other types of data availability (DA) nodes like bridge and full DA storage nodes, which will be covered in later sections.

    Celestia App

    Install celestia-app allows you to get started running a consensus node.

    celestia-app is the software that allows you to run validator nodes and also provide RPC endpoints.

    celestia-app covers the consensus layer, while celestia-node covers the DA layer.

    Getting started

    As covered in the previous section, Celestia offers two different test networks, Arabica devnet and Mocha testnet.

    If you are planning to run a light node, it is recommended to use Arabica, which you will find options to connecting to in the later sections.

    If you plan on running a validator, your only option is to run your node on Mocha.

    In this quick start guide, we will go over installing both of the software clients: celestia-node and celestia-app.

    NOTE

    If you just want to run a light node, you don't need to install celestia-app and can skip that part.

    Proceed to the next section in order to get started.

    ',17),s=[n];function r(l,d,c,p,u,h){return t(),a("div",null,s)}const _=e(i,[["render",r]]);export{f as __pageData,_ as default}; diff --git a/assets/nodes_quick-start.md.05f1f426.lean.js b/assets/nodes_quick-start.md.05f1f426.lean.js new file mode 100644 index 00000000000..8522063af9a --- /dev/null +++ b/assets/nodes_quick-start.md.05f1f426.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as o}from"./chunks/framework.51d6c45b.js";const f=JSON.parse('{"title":"Quick start guide","description":"Learn how to get started and run your first node on Celestia.","frontmatter":{"description":"Learn how to get started and run your first node on Celestia.","head":[["meta",{"name":"og:title","content":"Quick start guide | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/quick-start.md","filePath":"nodes/quick-start.md","lastUpdated":1698417238000}'),i={name:"nodes/quick-start.md"},n=o("",17),s=[n];function r(l,d,c,p,u,h){return t(),a("div",null,s)}const _=e(i,[["render",r]]);export{f as __pageData,_ as default}; diff --git a/assets/nodes_systemd.md.459598f1.js b/assets/nodes_systemd.md.459598f1.js new file mode 100644 index 00000000000..59bbeeff741 --- /dev/null +++ b/assets/nodes_systemd.md.459598f1.js @@ -0,0 +1,125 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.51d6c45b.js";const h=JSON.parse('{"title":"Setting up your node as a background process with SystemD","description":"Learn how to setup your node as a background process with SystemD.","frontmatter":{"description":"Learn how to setup your node as a background process with SystemD.","head":[["meta",{"name":"og:title","content":"Setting up your node as a background process with SystemD | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/systemd.md","filePath":"nodes/systemd.md","lastUpdated":1707833567000}'),e={name:"nodes/systemd.md"},p=l(`

    Setting up your node as a background process with SystemD

    SystemD is a daemon service useful for running applications as background processes.

    Consensus nodes

    If you are running a validator or full consensus node, here are the steps to setting up celestia-appd as a background process.

    Start the celestia-app with SystemD

    SystemD is a daemon service useful for running applications as background processes.

    Create Celestia-App systemd file:

    sh
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-appd.service
    +[Unit]
    +Description=celestia-appd Cosmos daemon
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia-appd) start
    +Restart=on-failure
    +RestartSec=3
    +LimitNOFILE=65535
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-appd.service
    +[Unit]
    +Description=celestia-appd Cosmos daemon
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia-appd) start
    +Restart=on-failure
    +RestartSec=3
    +LimitNOFILE=65535
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF

    If the file was created successfully you will be able to see its content:

    sh
    cat /etc/systemd/system/celestia-appd.service
    cat /etc/systemd/system/celestia-appd.service

    Enable and start celestia-appd daemon:

    sh
    sudo systemctl enable celestia-appd
    +sudo systemctl start celestia-appd
    sudo systemctl enable celestia-appd
    +sudo systemctl start celestia-appd

    Check if daemon has been started correctly:

    sh
    sudo systemctl status celestia-appd
    sudo systemctl status celestia-appd

    Check daemon logs in real time:

    sh
    sudo journalctl -u celestia-appd.service -f
    sudo journalctl -u celestia-appd.service -f

    To check if your node is in sync before going forward:

    sh
    curl -s localhost:26657/status | jq .result | jq .sync_info
    curl -s localhost:26657/status | jq .result | jq .sync_info

    Make sure that you have "catching_up": false, otherwise leave it running until it is in sync.

    Data availability nodes

    Celestia full storage node

    Create Celestia full storage node systemd file:

    sh
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-full.service
    +[Unit]
    +Description=celestia-full Cosmos daemon
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia) full start
    +Restart=on-failure
    +RestartSec=3
    +LimitNOFILE=1400000
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-full.service
    +[Unit]
    +Description=celestia-full Cosmos daemon
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia) full start
    +Restart=on-failure
    +RestartSec=3
    +LimitNOFILE=1400000
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF

    If the file was created successfully you will be able to see its content:

    sh
    cat /etc/systemd/system/celestia-full.service
    cat /etc/systemd/system/celestia-full.service

    Enable and start celestia-full daemon:

    sh
    sudo systemctl enable celestia-full
    +sudo systemctl start celestia-full && sudo journalctl -u \\
    +celestia-full.service -f
    sudo systemctl enable celestia-full
    +sudo systemctl start celestia-full && sudo journalctl -u \\
    +celestia-full.service -f

    You should be seeing logs coming through of the full storage node syncing.

    Celestia bridge node

    Create Celestia Bridge systemd file:

    sh
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-bridge.service
    +[Unit]
    +Description=celestia-bridge Cosmos daemon
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia) bridge start
    +Restart=on-failure
    +RestartSec=3
    +LimitNOFILE=1400000
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-bridge.service
    +[Unit]
    +Description=celestia-bridge Cosmos daemon
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia) bridge start
    +Restart=on-failure
    +RestartSec=3
    +LimitNOFILE=1400000
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF

    If the file was created successfully you will be able to see its content:

    sh
    cat /etc/systemd/system/celestia-bridge.service
    cat /etc/systemd/system/celestia-bridge.service

    Enable and start celestia-bridge daemon:

    sh
    sudo systemctl enable celestia-bridge
    +sudo systemctl start celestia-bridge && sudo journalctl -u \\
    +celestia-bridge.service -f
    sudo systemctl enable celestia-bridge
    +sudo systemctl start celestia-bridge && sudo journalctl -u \\
    +celestia-bridge.service -f

    Now, the Celestia bridge node will start syncing headers and storing blocks from celestia-app.

    NOTE

    At startup, we can see the multiaddress from Celestia bridge node. This is needed for future light node connections and communication between Celestia Bridge Nodes

    Example:

    sh
    NODE_IP=<URI>]
    +/ip4/$NODE_IP/tcp/2121/p2p/12D3KooWD5wCBJXKQuDjhXFjTFMrZoysGVLtVht5hMoVbSLCbV22
    NODE_IP=<URI>]
    +/ip4/$NODE_IP/tcp/2121/p2p/12D3KooWD5wCBJXKQuDjhXFjTFMrZoysGVLtVht5hMoVbSLCbV22

    You should be seeing logs coming through of the bridge node syncing.

    Celestia light node

    Start the light node as daemon process in the background

    sh
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-lightd.service
    +[Unit]
    +Description=celestia-lightd light node
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia) light start --core.ip <URI>
    +Restart=on-failure
    +RestartSec=3
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-lightd.service
    +[Unit]
    +Description=celestia-lightd light node
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia) light start --core.ip <URI>
    +Restart=on-failure
    +RestartSec=3
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF

    If the file was created successfully you will be able to see its content:

    sh
    cat /etc/systemd/system/celestia-lightd.service
    cat /etc/systemd/system/celestia-lightd.service

    Enable and start celestia-lightd daemon:

    sh
    sudo systemctl enable celestia-lightd
    +sudo systemctl start celestia-lightd
    sudo systemctl enable celestia-lightd
    +sudo systemctl start celestia-lightd

    Check if daemon has been started correctly:

    sh
    sudo systemctl status celestia-lightd
    sudo systemctl status celestia-lightd

    Check daemon logs in real time:

    sh
    sudo journalctl -u celestia-lightd.service -f
    sudo journalctl -u celestia-lightd.service -f

    Now, the Celestia light node will start syncing headers. After sync is finished, light node will do Data Availability Sampling (DAS) from the bridge node.

    `,52),o=[p];function t(c,r,i,y,d,E){return a(),n("div",null,o)}const u=s(e,[["render",t]]);export{h as __pageData,u as default}; diff --git a/assets/nodes_systemd.md.459598f1.lean.js b/assets/nodes_systemd.md.459598f1.lean.js new file mode 100644 index 00000000000..c7fffb5b101 --- /dev/null +++ b/assets/nodes_systemd.md.459598f1.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.51d6c45b.js";const h=JSON.parse('{"title":"Setting up your node as a background process with SystemD","description":"Learn how to setup your node as a background process with SystemD.","frontmatter":{"description":"Learn how to setup your node as a background process with SystemD.","head":[["meta",{"name":"og:title","content":"Setting up your node as a background process with SystemD | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/systemd.md","filePath":"nodes/systemd.md","lastUpdated":1707833567000}'),e={name:"nodes/systemd.md"},p=l("",52),o=[p];function t(c,r,i,y,d,E){return a(),n("div",null,o)}const u=s(e,[["render",t]]);export{h as __pageData,u as default}; diff --git a/assets/nodes_validator-node.md.6e656228.js b/assets/nodes_validator-node.md.6e656228.js new file mode 100644 index 00000000000..fd0c9ee5916 --- /dev/null +++ b/assets/nodes_validator-node.md.6e656228.js @@ -0,0 +1,83 @@ +import{c as o}from"./chunks/constants.dc70dfe7.js";import{o as t,c as p,k as s,a,t as e,l,Q as n}from"./chunks/framework.51d6c45b.js";const c="/img/nodes/validator.png",i=n('

    Setting up a Celestia validator node

    This tutorial will guide you through setting up a validator node on Celestia. Validator nodes allow you to participate in consensus in the Celestia network.

    validator node

    Hardware requirements

    The following hardware minimum requirements are recommended for running a validator node:

    • Memory: 8 GB RAM
    • CPU: 6 cores
    • Disk: 500 GB SSD Storage
    • Bandwidth: 1 Gbps for Download/1 Gbps for Upload

    Setting up a validator node

    The following tutorial is done on an Ubuntu Linux 20.04 (LTS) x64 instance machine.

    First, follow the instructions on setting up a full consensus node.

    Wallet

    Follow the tutorial on creating a wallet.

    Delegate stake to a validator

    Create an environment variable for the address:

    bash
    VALIDATOR_WALLET=<validator-wallet-name>
    VALIDATOR_WALLET=<validator-wallet-name>

    If you want to delegate more stake to any validator, including your own you will need the celesvaloper address of the validator in question. You can run the command below to get the celesvaloper of your local validator wallet in case you want to delegate more to it:

    bash
    celestia-appd keys show $VALIDATOR_WALLET --bech val -a
    celestia-appd keys show $VALIDATOR_WALLET --bech val -a

    After entering the wallet passphrase you should see a similar output:

    bash
    Enter keyring passphrase:
    +celesvaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u43cv6hd
    Enter keyring passphrase:
    +celesvaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u43cv6hd

    To delegate tokens to the celestiavaloper validator, as an example you can run:

    `,19),r={class:"language-bash vp-adaptive-theme"},d=s("button",{title:"Copy Code",class:"copy"},null,-1),_=s("span",{class:"lang"},"bash",-1),h={class:"shiki github-dark vp-code-dark"},y=n('celestia-appd tx staking delegate \\',1),u=n('celestiavaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u4q4gx4p 1000000utia \\',1),E={class:"line"},C=s("span",{style:{color:"#E1E4E8"}},"--from=$VALIDATOR_WALLET ",-1),g={style:{color:"#79B8FF"}},F=s("span",{style:{color:"#E1E4E8"}}," ",-1),T=s("span",{style:{color:"#79B8FF"}},"\\",-1),b=s("span",{class:"line"},[s("span",{style:{color:"#E1E4E8"}},"--fees=21000utia")],-1),m={class:"shiki github-light vp-code-light"},v=n('celestia-appd tx staking delegate \\',1),A=n('celestiavaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u4q4gx4p 1000000utia \\',1),f={class:"line"},k=s("span",{style:{color:"#24292E"}},"--from=$VALIDATOR_WALLET ",-1),S={style:{color:"#005CC5"}},q=s("span",{style:{color:"#24292E"}}," ",-1),B=s("span",{style:{color:"#005CC5"}},"\\",-1),I=s("span",{class:"line"},[s("span",{style:{color:"#24292E"}},"--fees=21000utia")],-1),w=n(`

    If successful, you should see a similar output as:

    console
    code: 0
    +codespace: ""
    +data: ""
    +gas_used: "0"
    +gas_wanted: "0"
    +height: "0"
    +info: ""
    +logs: []
    +raw_log: '[]'
    +timestamp: ""
    +tx: null
    +txhash: <tx-hash>
    code: 0
    +codespace: ""
    +data: ""
    +gas_used: "0"
    +gas_wanted: "0"
    +height: "0"
    +info: ""
    +logs: []
    +raw_log: '[]'
    +timestamp: ""
    +tx: null
    +txhash: <tx-hash>

    You can check if the TX hash went through using the block explorer by inputting the txhash ID that was returned.

    Optional: Deploy the celestia-node

    Running a bridge node is critical to the Celestia network as it enables the data availability and consensus nodes to communicate with one another. It is recommended to support the data availability network, but is not required for celestia-app.

    If you are not running a bridge node, you can skip to run a validator node.

    This section describes part 2 of Celestia validator node setup: running a Celestia bridge node daemon.

    Install celestia-node

    You can follow the tutorial for installing celestia-node

    Initialize the bridge node

    Run the following:

    bash
    celestia bridge init --core.ip <URI>
    celestia bridge init --core.ip <URI>

    TIP

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    If you need a list of RPC endpoints to connect to, you can find the list on the Mocha testnet page or list on the Arabica devnet page.

    Run the bridge node

    Run the following:

    bash
    celestia bridge start
    celestia bridge start

    Optional: start the bridge node with SystemD

    Follow the tutorial on setting up the bridge node as a background process with SystemD.

    You have successfully set up a bridge node that is syncing with the network.

    Run the validator node

    In order to start your validator node, run the following:

    bash
    celestia-appd start
    celestia-appd start

    After completing all the necessary steps, you are now ready to run a validator! In order to create your validator onchain, follow the instructions below. Keep in mind that these steps are necessary ONLY if you want to participate in the consensus.

    Pick a moniker name of your choice! This is the validator name that will show up on public dashboards and explorers. VALIDATOR_WALLET must be the same you defined previously. Parameter --min-self-delegation=1000000 defines the amount of tokens that are self delegated from your validator wallet.

    Now, connect to the network of your choice.

    You have the following option of connecting to list of networks shown below:

    Continuing the validator tutorial, here are the steps to connect your validator to Mocha:

    `,28),P={class:"language-bash vp-adaptive-theme"},V=s("button",{title:"Copy Code",class:"copy"},null,-1),x=s("span",{class:"lang"},"bash",-1),D={class:"shiki github-dark vp-code-dark"},R=s("span",{class:"line"},[s("span",{style:{color:"#E1E4E8"}},"MONIKER"),s("span",{style:{color:"#F97583"}},"="),s("span",{style:{color:"#9ECBFF"}},'"your_moniker"')],-1),N=s("span",{class:"line"},[s("span",{style:{color:"#E1E4E8"}},"VALIDATOR_WALLET"),s("span",{style:{color:"#F97583"}},"="),s("span",{style:{color:"#9ECBFF"}},'"validator"')],-1),L=s("span",{class:"line"},null,-1),O=n('celestia-appd tx staking create-validator \\',1),W=n(' --amount=1000000utia \\',1),$=n(' --pubkey=$(celestia-appd tendermint show-validator) \\',1),z=n(' --moniker=$MONIKER \\',1),M={class:"line"},Y=s("span",{style:{color:"#E1E4E8"}}," ",-1),U={style:{color:"#79B8FF"}},K=s("span",{style:{color:"#E1E4E8"}}," ",-1),G=s("span",{style:{color:"#79B8FF"}},"\\",-1),H=n(' --commission-rate=0.1 \\',1),j=n(' --commission-max-rate=0.2 \\',1),J=n(' --commission-max-change-rate=0.01 \\',1),Q=n(' --min-self-delegation=1000000 \\',1),X=n(' --from=$VALIDATOR_WALLET \\',1),Z=n(' --keyring-backend=test \\',1),ss=n(' --fees=21000utia \\',1),as=s("span",{class:"line"},[s("span",{style:{color:"#E1E4E8"}}," "),s("span",{style:{color:"#79B8FF"}},"--gas=220000")],-1),ns={class:"shiki github-light vp-code-light"},os=s("span",{class:"line"},[s("span",{style:{color:"#24292E"}},"MONIKER"),s("span",{style:{color:"#D73A49"}},"="),s("span",{style:{color:"#032F62"}},'"your_moniker"')],-1),es=s("span",{class:"line"},[s("span",{style:{color:"#24292E"}},"VALIDATOR_WALLET"),s("span",{style:{color:"#D73A49"}},"="),s("span",{style:{color:"#032F62"}},'"validator"')],-1),ls=s("span",{class:"line"},null,-1),ts=n('celestia-appd tx staking create-validator \\',1),ps=n(' --amount=1000000utia \\',1),cs=n(' --pubkey=$(celestia-appd tendermint show-validator) \\',1),is=n(' --moniker=$MONIKER \\',1),rs={class:"line"},ds=s("span",{style:{color:"#24292E"}}," ",-1),_s={style:{color:"#005CC5"}},hs=s("span",{style:{color:"#24292E"}}," ",-1),ys=s("span",{style:{color:"#005CC5"}},"\\",-1),us=n(' --commission-rate=0.1 \\',1),Es=n(' --commission-max-rate=0.2 \\',1),Cs=n(' --commission-max-change-rate=0.01 \\',1),gs=n(' --min-self-delegation=1000000 \\',1),Fs=n(' --from=$VALIDATOR_WALLET \\',1),Ts=n(' --keyring-backend=test \\',1),bs=n(' --fees=21000utia \\',1),ms=s("span",{class:"line"},[s("span",{style:{color:"#24292E"}}," "),s("span",{style:{color:"#005CC5"}},"--gas=220000")],-1),vs=n(`

    You will be prompted to confirm the transaction:

    console
    confirm transaction before signing and broadcasting [y/N]: y
    confirm transaction before signing and broadcasting [y/N]: y

    Inputting y should provide an output similar to:

    console
    code: 0
    +codespace: ""
    +data: ""
    +gas_used: "0"
    +gas_wanted: "0"
    +height: "0"
    +info: ""
    +logs: []
    +raw_log: '[]'
    +timestamp: ""
    +tx: null
    +txhash: <tx-hash>
    code: 0
    +codespace: ""
    +data: ""
    +gas_used: "0"
    +gas_wanted: "0"
    +height: "0"
    +info: ""
    +logs: []
    +raw_log: '[]'
    +timestamp: ""
    +tx: null
    +txhash: <tx-hash>

    You should now be able to see your validator from a block explorer

    Submit your validator information

    After starting your node, please submit your node as a seed and peer to the networks repository.

    Optional: Transaction indexer configuration options

    Follow the instructions under transaction indexer configuration options to configure your config.toml file to select which transactions to index.

    Additional resources

    For additional resources, refer to the extra resources for consensus nodessection of the full consensus node page .

    `,11),Is=JSON.parse('{"title":"Setting up a Celestia validator node","description":"Learn how to set up a Celestia validator node.","frontmatter":{"description":"Learn how to set up a Celestia validator node.","outline":"deep","head":[["meta",{"name":"og:title","content":"Setting up a Celestia validator node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/validator-node.md","filePath":"nodes/validator-node.md","lastUpdated":1711744858000}'),As={name:"nodes/validator-node.md"},ws=Object.assign(As,{setup(fs){return(ks,Ss)=>(t(),p("div",null,[i,s("div",r,[d,_,s("pre",h,[s("code",null,[y,a(` +`),u,a(` +`),s("span",E,[C,s("span",g,"--chain-id="+e(l(o).mochaChainId),1),F,T]),a(` +`),b])]),s("pre",m,[s("code",null,[v,a(` +`),A,a(` +`),s("span",f,[k,s("span",S,"--chain-id="+e(l(o).mochaChainId),1),q,B]),a(` +`),I])])]),w,s("div",P,[V,x,s("pre",D,[s("code",null,[R,a(` +`),N,a(` +`),L,a(` +`),O,a(` +`),W,a(` +`),$,a(` +`),z,a(` +`),s("span",M,[Y,s("span",U,"--chain-id="+e(l(o).mochaChainId),1),K,G]),a(` +`),H,a(` +`),j,a(` +`),J,a(` +`),Q,a(` +`),X,a(` +`),Z,a(` +`),ss,a(` +`),as])]),s("pre",ns,[s("code",null,[os,a(` +`),es,a(` +`),ls,a(` +`),ts,a(` +`),ps,a(` +`),cs,a(` +`),is,a(` +`),s("span",rs,[ds,s("span",_s,"--chain-id="+e(l(o).mochaChainId),1),hs,ys]),a(` +`),us,a(` +`),Es,a(` +`),Cs,a(` +`),gs,a(` +`),Fs,a(` +`),Ts,a(` +`),bs,a(` +`),ms])])]),vs]))}});export{Is as __pageData,ws as default}; diff --git a/assets/nodes_validator-node.md.6e656228.lean.js b/assets/nodes_validator-node.md.6e656228.lean.js new file mode 100644 index 00000000000..8dc8d2c524b --- /dev/null +++ b/assets/nodes_validator-node.md.6e656228.lean.js @@ -0,0 +1,37 @@ +import{c as o}from"./chunks/constants.dc70dfe7.js";import{o as t,c as p,k as s,a,t as e,l,Q as n}from"./chunks/framework.51d6c45b.js";const c="/img/nodes/validator.png",i=n("",19),r={class:"language-bash vp-adaptive-theme"},d=s("button",{title:"Copy Code",class:"copy"},null,-1),_=s("span",{class:"lang"},"bash",-1),h={class:"shiki github-dark vp-code-dark"},y=n("",1),u=n("",1),E={class:"line"},C=s("span",{style:{color:"#E1E4E8"}},"--from=$VALIDATOR_WALLET ",-1),g={style:{color:"#79B8FF"}},F=s("span",{style:{color:"#E1E4E8"}}," ",-1),T=s("span",{style:{color:"#79B8FF"}},"\\",-1),b=s("span",{class:"line"},[s("span",{style:{color:"#E1E4E8"}},"--fees=21000utia")],-1),m={class:"shiki github-light vp-code-light"},v=n("",1),A=n("",1),f={class:"line"},k=s("span",{style:{color:"#24292E"}},"--from=$VALIDATOR_WALLET ",-1),S={style:{color:"#005CC5"}},q=s("span",{style:{color:"#24292E"}}," ",-1),B=s("span",{style:{color:"#005CC5"}},"\\",-1),I=s("span",{class:"line"},[s("span",{style:{color:"#24292E"}},"--fees=21000utia")],-1),w=n("",28),P={class:"language-bash vp-adaptive-theme"},V=s("button",{title:"Copy Code",class:"copy"},null,-1),x=s("span",{class:"lang"},"bash",-1),D={class:"shiki github-dark vp-code-dark"},R=s("span",{class:"line"},[s("span",{style:{color:"#E1E4E8"}},"MONIKER"),s("span",{style:{color:"#F97583"}},"="),s("span",{style:{color:"#9ECBFF"}},'"your_moniker"')],-1),N=s("span",{class:"line"},[s("span",{style:{color:"#E1E4E8"}},"VALIDATOR_WALLET"),s("span",{style:{color:"#F97583"}},"="),s("span",{style:{color:"#9ECBFF"}},'"validator"')],-1),L=s("span",{class:"line"},null,-1),O=n("",1),W=n("",1),$=n("",1),z=n("",1),M={class:"line"},Y=s("span",{style:{color:"#E1E4E8"}}," ",-1),U={style:{color:"#79B8FF"}},K=s("span",{style:{color:"#E1E4E8"}}," ",-1),G=s("span",{style:{color:"#79B8FF"}},"\\",-1),H=n("",1),j=n("",1),J=n("",1),Q=n("",1),X=n("",1),Z=n("",1),ss=n("",1),as=s("span",{class:"line"},[s("span",{style:{color:"#E1E4E8"}}," "),s("span",{style:{color:"#79B8FF"}},"--gas=220000")],-1),ns={class:"shiki github-light vp-code-light"},os=s("span",{class:"line"},[s("span",{style:{color:"#24292E"}},"MONIKER"),s("span",{style:{color:"#D73A49"}},"="),s("span",{style:{color:"#032F62"}},'"your_moniker"')],-1),es=s("span",{class:"line"},[s("span",{style:{color:"#24292E"}},"VALIDATOR_WALLET"),s("span",{style:{color:"#D73A49"}},"="),s("span",{style:{color:"#032F62"}},'"validator"')],-1),ls=s("span",{class:"line"},null,-1),ts=n("",1),ps=n("",1),cs=n("",1),is=n("",1),rs={class:"line"},ds=s("span",{style:{color:"#24292E"}}," ",-1),_s={style:{color:"#005CC5"}},hs=s("span",{style:{color:"#24292E"}}," ",-1),ys=s("span",{style:{color:"#005CC5"}},"\\",-1),us=n("",1),Es=n("",1),Cs=n("",1),gs=n("",1),Fs=n("",1),Ts=n("",1),bs=n("",1),ms=s("span",{class:"line"},[s("span",{style:{color:"#24292E"}}," "),s("span",{style:{color:"#005CC5"}},"--gas=220000")],-1),vs=n("",11),Is=JSON.parse('{"title":"Setting up a Celestia validator node","description":"Learn how to set up a Celestia validator node.","frontmatter":{"description":"Learn how to set up a Celestia validator node.","outline":"deep","head":[["meta",{"name":"og:title","content":"Setting up a Celestia validator node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/validator-node.md","filePath":"nodes/validator-node.md","lastUpdated":1711744858000}'),As={name:"nodes/validator-node.md"},ws=Object.assign(As,{setup(fs){return(ks,Ss)=>(t(),p("div",null,[i,s("div",r,[d,_,s("pre",h,[s("code",null,[y,a(` +`),u,a(` +`),s("span",E,[C,s("span",g,"--chain-id="+e(l(o).mochaChainId),1),F,T]),a(` +`),b])]),s("pre",m,[s("code",null,[v,a(` +`),A,a(` +`),s("span",f,[k,s("span",S,"--chain-id="+e(l(o).mochaChainId),1),q,B]),a(` +`),I])])]),w,s("div",P,[V,x,s("pre",D,[s("code",null,[R,a(` +`),N,a(` +`),L,a(` +`),O,a(` +`),W,a(` +`),$,a(` +`),z,a(` +`),s("span",M,[Y,s("span",U,"--chain-id="+e(l(o).mochaChainId),1),K,G]),a(` +`),H,a(` +`),j,a(` +`),J,a(` +`),Q,a(` +`),X,a(` +`),Z,a(` +`),ss,a(` +`),as])]),s("pre",ns,[s("code",null,[os,a(` +`),es,a(` +`),ls,a(` +`),ts,a(` +`),ps,a(` +`),cs,a(` +`),is,a(` +`),s("span",rs,[ds,s("span",_s,"--chain-id="+e(l(o).mochaChainId),1),hs,ys]),a(` +`),us,a(` +`),Es,a(` +`),Cs,a(` +`),gs,a(` +`),Fs,a(` +`),Ts,a(` +`),bs,a(` +`),ms])])]),vs]))}});export{Is as __pageData,ws as default}; diff --git a/assets/style.442f57ad.css b/assets/style.442f57ad.css new file mode 100644 index 00000000000..3b4ccf1449c --- /dev/null +++ b/assets/style.442f57ad.css @@ -0,0 +1 @@ +@font-face{font-family:Inter var;font-weight:100 900;font-display:swap;font-style:normal;font-named-instance:"Regular";src:url(/assets/inter-roman-cyrillic.5f2c6c8c.woff2) format("woff2");unicode-range:U+0301,U+0400-045F,U+0490-0491,U+04B0-04B1,U+2116}@font-face{font-family:Inter var;font-weight:100 900;font-display:swap;font-style:normal;font-named-instance:"Regular";src:url(/assets/inter-roman-cyrillic-ext.e75737ce.woff2) format("woff2");unicode-range:U+0460-052F,U+1C80-1C88,U+20B4,U+2DE0-2DFF,U+A640-A69F,U+FE2E-FE2F}@font-face{font-family:Inter var;font-weight:100 900;font-display:swap;font-style:normal;font-named-instance:"Regular";src:url(/assets/inter-roman-greek.d5a6d92a.woff2) format("woff2");unicode-range:U+0370-03FF}@font-face{font-family:Inter var;font-weight:100 900;font-display:swap;font-style:normal;font-named-instance:"Regular";src:url(/assets/inter-roman-greek-ext.ab0619bc.woff2) format("woff2");unicode-range:U+1F00-1FFF}@font-face{font-family:Inter var;font-weight:100 900;font-display:swap;font-style:normal;font-named-instance:"Regular";src:url(/assets/inter-roman-latin.2ed14f66.woff2) format("woff2");unicode-range:U+0000-00FF,U+0131,U+0152-0153,U+02BB-02BC,U+02C6,U+02DA,U+02DC,U+2000-206F,U+2074,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:Inter var;font-weight:100 900;font-display:swap;font-style:normal;font-named-instance:"Regular";src:url(/assets/inter-roman-latin-ext.0030eebd.woff2) format("woff2");unicode-range:U+0100-024F,U+0259,U+1E00-1EFF,U+2020,U+20A0-20AB,U+20AD-20CF,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:Inter var;font-weight:100 900;font-display:swap;font-style:normal;font-named-instance:"Regular";src:url(/assets/inter-roman-vietnamese.14ce25a6.woff2) format("woff2");unicode-range:U+0102-0103,U+0110-0111,U+0128-0129,U+0168-0169,U+01A0-01A1,U+01AF-01B0,U+1EA0-1EF9,U+20AB}@font-face{font-family:Inter var;font-weight:100 900;font-display:swap;font-style:italic;font-named-instance:"Italic";src:url(/assets/inter-italic-cyrillic.ea42a392.woff2) format("woff2");unicode-range:U+0301,U+0400-045F,U+0490-0491,U+04B0-04B1,U+2116}@font-face{font-family:Inter var;font-weight:100 900;font-display:swap;font-style:italic;font-named-instance:"Italic";src:url(/assets/inter-italic-cyrillic-ext.33bd5a8e.woff2) format("woff2");unicode-range:U+0460-052F,U+1C80-1C88,U+20B4,U+2DE0-2DFF,U+A640-A69F,U+FE2E-FE2F}@font-face{font-family:Inter var;font-weight:100 900;font-display:swap;font-style:italic;font-named-instance:"Italic";src:url(/assets/inter-italic-greek.8f4463c4.woff2) format("woff2");unicode-range:U+0370-03FF}@font-face{font-family:Inter var;font-weight:100 900;font-display:swap;font-style:italic;font-named-instance:"Italic";src:url(/assets/inter-italic-greek-ext.4fbe9427.woff2) format("woff2");unicode-range:U+1F00-1FFF}@font-face{font-family:Inter var;font-weight:100 900;font-display:swap;font-style:italic;font-named-instance:"Italic";src:url(/assets/inter-italic-latin.bd3b6f56.woff2) format("woff2");unicode-range:U+0000-00FF,U+0131,U+0152-0153,U+02BB-02BC,U+02C6,U+02DA,U+02DC,U+2000-206F,U+2074,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:Inter var;font-weight:100 900;font-display:swap;font-style:italic;font-named-instance:"Italic";src:url(/assets/inter-italic-latin-ext.bd8920cc.woff2) format("woff2");unicode-range:U+0100-024F,U+0259,U+1E00-1EFF,U+2020,U+20A0-20AB,U+20AD-20CF,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:Inter var;font-weight:100 900;font-display:swap;font-style:italic;font-named-instance:"Italic";src:url(/assets/inter-italic-vietnamese.6ce511fb.woff2) format("woff2");unicode-range:U+0102-0103,U+0110-0111,U+0128-0129,U+0168-0169,U+01A0-01A1,U+01AF-01B0,U+1EA0-1EF9,U+20AB}@font-face{font-family:Chinese Quotes;src:local("PingFang SC Regular"),local("PingFang SC"),local("SimHei"),local("Source Han Sans SC");unicode-range:U+2018,U+2019,U+201C,U+201D}:root{--vp-c-white: #ffffff;--vp-c-black: #000000;--vp-c-neutral: var(--vp-c-black);--vp-c-neutral-inverse: var(--vp-c-white)}.dark{--vp-c-neutral: var(--vp-c-white);--vp-c-neutral-inverse: var(--vp-c-black)}:root{--vp-c-gray-1: #dddde3;--vp-c-gray-2: #e4e4e9;--vp-c-gray-3: #ebebef;--vp-c-gray-soft: rgba(142, 150, 170, .14);--vp-c-indigo-1: #3451b2;--vp-c-indigo-2: #3a5ccc;--vp-c-indigo-3: #5672cd;--vp-c-indigo-soft: rgba(100, 108, 255, .14);--vp-c-green-1: #18794e;--vp-c-green-2: #299764;--vp-c-green-3: #30a46c;--vp-c-green-soft: rgba(16, 185, 129, .14);--vp-c-yellow-1: #915930;--vp-c-yellow-2: #946300;--vp-c-yellow-3: #9f6a00;--vp-c-yellow-soft: rgba(234, 179, 8, .14);--vp-c-red-1: #b8272c;--vp-c-red-2: #d5393e;--vp-c-red-3: #e0575b;--vp-c-red-soft: rgba(244, 63, 94, .14);--vp-c-sponsor: #db2777}.dark{--vp-c-gray-1: #515c67;--vp-c-gray-2: #414853;--vp-c-gray-3: #32363f;--vp-c-gray-soft: rgba(101, 117, 133, .16);--vp-c-indigo-1: #a8b1ff;--vp-c-indigo-2: #5c73e7;--vp-c-indigo-3: #3e63dd;--vp-c-indigo-soft: rgba(100, 108, 255, .16);--vp-c-green-1: #3dd68c;--vp-c-green-2: #30a46c;--vp-c-green-3: #298459;--vp-c-green-soft: rgba(16, 185, 129, .16);--vp-c-yellow-1: #f9b44e;--vp-c-yellow-2: #da8b17;--vp-c-yellow-3: #a46a0a;--vp-c-yellow-soft: rgba(234, 179, 8, .16);--vp-c-red-1: #f66f81;--vp-c-red-2: #f14158;--vp-c-red-3: #b62a3c;--vp-c-red-soft: rgba(244, 63, 94, .16)}:root{--vp-c-bg: #ffffff;--vp-c-bg-alt: #f6f6f7;--vp-c-bg-elv: #ffffff;--vp-c-bg-soft: #f6f6f7}.dark{--vp-c-bg: #1b1b1f;--vp-c-bg-alt: #161618;--vp-c-bg-elv: #202127;--vp-c-bg-soft: #202127}:root{--vp-c-border: #c2c2c4;--vp-c-divider: #e2e2e3;--vp-c-gutter: #e2e2e3}.dark{--vp-c-border: #3c3f44;--vp-c-divider: #2e2e32;--vp-c-gutter: #000000}:root{--vp-c-text-1: rgba(60, 60, 67);--vp-c-text-2: rgba(60, 60, 67, .78);--vp-c-text-3: rgba(60, 60, 67, .56)}.dark{--vp-c-text-1: rgba(255, 255, 245, .86);--vp-c-text-2: rgba(235, 235, 245, .6);--vp-c-text-3: rgba(235, 235, 245, .38)}:root{--vp-c-default-1: var(--vp-c-gray-1);--vp-c-default-2: var(--vp-c-gray-2);--vp-c-default-3: var(--vp-c-gray-3);--vp-c-default-soft: var(--vp-c-gray-soft);--vp-c-brand-1: var(--vp-c-indigo-1);--vp-c-brand-2: var(--vp-c-indigo-2);--vp-c-brand-3: var(--vp-c-indigo-3);--vp-c-brand-soft: var(--vp-c-indigo-soft);--vp-c-brand: var(--vp-c-brand-1);--vp-c-tip-1: var(--vp-c-brand-1);--vp-c-tip-2: var(--vp-c-brand-2);--vp-c-tip-3: var(--vp-c-brand-3);--vp-c-tip-soft: var(--vp-c-brand-soft);--vp-c-warning-1: var(--vp-c-yellow-1);--vp-c-warning-2: var(--vp-c-yellow-2);--vp-c-warning-3: var(--vp-c-yellow-3);--vp-c-warning-soft: var(--vp-c-yellow-soft);--vp-c-danger-1: var(--vp-c-red-1);--vp-c-danger-2: var(--vp-c-red-2);--vp-c-danger-3: var(--vp-c-red-3);--vp-c-danger-soft: var(--vp-c-red-soft)}:root{--vp-font-family-base: "Chinese Quotes", "Inter var", "Inter", ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Helvetica, Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";--vp-font-family-mono: ui-monospace, SFMono-Regular, "SF Mono", Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace}:root{--vp-shadow-1: 0 1px 2px rgba(0, 0, 0, .04), 0 1px 2px rgba(0, 0, 0, .06);--vp-shadow-2: 0 3px 12px rgba(0, 0, 0, .07), 0 1px 4px rgba(0, 0, 0, .07);--vp-shadow-3: 0 12px 32px rgba(0, 0, 0, .1), 0 2px 6px rgba(0, 0, 0, .08);--vp-shadow-4: 0 14px 44px rgba(0, 0, 0, .12), 0 3px 9px rgba(0, 0, 0, .12);--vp-shadow-5: 0 18px 56px rgba(0, 0, 0, .16), 0 4px 12px rgba(0, 0, 0, .16)}:root{--vp-z-index-footer: 10;--vp-z-index-local-nav: 20;--vp-z-index-nav: 30;--vp-z-index-layout-top: 40;--vp-z-index-backdrop: 50;--vp-z-index-sidebar: 60}:root{--vp-icon-copy: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' height='20' width='20' stroke='rgba(128,128,128,1)' stroke-width='2' viewBox='0 0 24 24'%3E%3Cpath stroke-linecap='round' stroke-linejoin='round' d='M9 5H7a2 2 0 0 0-2 2v12a2 2 0 0 0 2 2h10a2 2 0 0 0 2-2V7a2 2 0 0 0-2-2h-2M9 5a2 2 0 0 0 2 2h2a2 2 0 0 0 2-2M9 5a2 2 0 0 1 2-2h2a2 2 0 0 1 2 2'/%3E%3C/svg%3E");--vp-icon-copied: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' height='20' width='20' stroke='rgba(128,128,128,1)' stroke-width='2' viewBox='0 0 24 24'%3E%3Cpath stroke-linecap='round' stroke-linejoin='round' d='M9 5H7a2 2 0 0 0-2 2v12a2 2 0 0 0 2 2h10a2 2 0 0 0 2-2V7a2 2 0 0 0-2-2h-2M9 5a2 2 0 0 0 2 2h2a2 2 0 0 0 2-2M9 5a2 2 0 0 1 2-2h2a2 2 0 0 1 2 2m-6 9 2 2 4-4'/%3E%3C/svg%3E")}:root{--vp-layout-max-width: 1440px}:root{--vp-header-anchor-symbol: "#"}:root{--vp-code-line-height: 1.7;--vp-code-font-size: .875em;--vp-code-color: var(--vp-c-brand-1);--vp-code-link-color: var(--vp-c-brand-1);--vp-code-link-hover-color: var(--vp-c-brand-2);--vp-code-bg: var(--vp-c-default-soft);--vp-code-block-color: var(--vp-c-text-2);--vp-code-block-bg: var(--vp-c-bg-alt);--vp-code-block-divider-color: var(--vp-c-gutter);--vp-code-lang-color: var(--vp-c-text-3);--vp-code-line-highlight-color: var(--vp-c-default-soft);--vp-code-line-number-color: var(--vp-c-text-3);--vp-code-line-diff-add-color: var(--vp-c-green-soft);--vp-code-line-diff-add-symbol-color: var(--vp-c-green-1);--vp-code-line-diff-remove-color: var(--vp-c-red-soft);--vp-code-line-diff-remove-symbol-color: var(--vp-c-red-1);--vp-code-line-warning-color: var(--vp-c-yellow-soft);--vp-code-line-error-color: var(--vp-c-red-soft);--vp-code-copy-code-border-color: var(--vp-c-divider);--vp-code-copy-code-bg: var(--vp-c-bg-soft);--vp-code-copy-code-hover-border-color: var(--vp-c-divider);--vp-code-copy-code-hover-bg: var(--vp-c-bg);--vp-code-copy-code-active-text: var(--vp-c-text-2);--vp-code-copy-copied-text-content: "Copied";--vp-code-tab-divider: var(--vp-code-block-divider-color);--vp-code-tab-text-color: var(--vp-c-text-2);--vp-code-tab-bg: var(--vp-code-block-bg);--vp-code-tab-hover-text-color: var(--vp-c-text-1);--vp-code-tab-active-text-color: var(--vp-c-text-1);--vp-code-tab-active-bar-color: var(--vp-c-brand-1)}:root{--vp-button-brand-border: transparent;--vp-button-brand-text: var(--vp-c-white);--vp-button-brand-bg: var(--vp-c-brand-3);--vp-button-brand-hover-border: transparent;--vp-button-brand-hover-text: var(--vp-c-white);--vp-button-brand-hover-bg: var(--vp-c-brand-2);--vp-button-brand-active-border: transparent;--vp-button-brand-active-text: var(--vp-c-white);--vp-button-brand-active-bg: var(--vp-c-brand-1);--vp-button-alt-border: transparent;--vp-button-alt-text: var(--vp-c-text-1);--vp-button-alt-bg: var(--vp-c-default-3);--vp-button-alt-hover-border: transparent;--vp-button-alt-hover-text: var(--vp-c-text-1);--vp-button-alt-hover-bg: var(--vp-c-default-2);--vp-button-alt-active-border: transparent;--vp-button-alt-active-text: var(--vp-c-text-1);--vp-button-alt-active-bg: var(--vp-c-default-1);--vp-button-sponsor-border: var(--vp-c-text-2);--vp-button-sponsor-text: var(--vp-c-text-2);--vp-button-sponsor-bg: transparent;--vp-button-sponsor-hover-border: var(--vp-c-sponsor);--vp-button-sponsor-hover-text: var(--vp-c-sponsor);--vp-button-sponsor-hover-bg: transparent;--vp-button-sponsor-active-border: var(--vp-c-sponsor);--vp-button-sponsor-active-text: var(--vp-c-sponsor);--vp-button-sponsor-active-bg: transparent}:root{--vp-custom-block-font-size: 14px;--vp-custom-block-code-font-size: 13px;--vp-custom-block-info-border: transparent;--vp-custom-block-info-text: var(--vp-c-text-1);--vp-custom-block-info-bg: var(--vp-c-default-soft);--vp-custom-block-info-code-bg: var(--vp-c-default-soft);--vp-custom-block-tip-border: transparent;--vp-custom-block-tip-text: var(--vp-c-text-1);--vp-custom-block-tip-bg: var(--vp-c-brand-soft);--vp-custom-block-tip-code-bg: var(--vp-c-brand-soft);--vp-custom-block-warning-border: transparent;--vp-custom-block-warning-text: var(--vp-c-text-1);--vp-custom-block-warning-bg: var(--vp-c-warning-soft);--vp-custom-block-warning-code-bg: var(--vp-c-warning-soft);--vp-custom-block-danger-border: transparent;--vp-custom-block-danger-text: var(--vp-c-text-1);--vp-custom-block-danger-bg: var(--vp-c-danger-soft);--vp-custom-block-danger-code-bg: var(--vp-c-danger-soft);--vp-custom-block-details-border: var(--vp-custom-block-info-border);--vp-custom-block-details-text: var(--vp-custom-block-info-text);--vp-custom-block-details-bg: var(--vp-custom-block-info-bg);--vp-custom-block-details-code-bg: var(--vp-custom-block-info-code-bg)}:root{--vp-input-border-color: var(--vp-c-border);--vp-input-bg-color: var(--vp-c-bg-alt);--vp-input-switch-bg-color: var(--vp-c-gray-soft)}:root{--vp-nav-height: 64px;--vp-nav-bg-color: var(--vp-c-bg);--vp-nav-screen-bg-color: var(--vp-c-bg);--vp-nav-logo-height: 24px}.hide-nav{--vp-nav-height: 0px}.hide-nav .VPSidebar{--vp-nav-height: 22px}:root{--vp-local-nav-bg-color: var(--vp-c-bg)}:root{--vp-sidebar-width: 272px;--vp-sidebar-bg-color: var(--vp-c-bg-alt)}:root{--vp-backdrop-bg-color: rgba(0, 0, 0, .6)}:root{--vp-home-hero-name-color: var(--vp-c-brand-1);--vp-home-hero-name-background: transparent;--vp-home-hero-image-background-image: none;--vp-home-hero-image-filter: none}:root{--vp-badge-info-border: transparent;--vp-badge-info-text: var(--vp-c-text-2);--vp-badge-info-bg: var(--vp-c-default-soft);--vp-badge-tip-border: transparent;--vp-badge-tip-text: var(--vp-c-brand-1);--vp-badge-tip-bg: var(--vp-c-brand-soft);--vp-badge-warning-border: transparent;--vp-badge-warning-text: var(--vp-c-warning-1);--vp-badge-warning-bg: var(--vp-c-warning-soft);--vp-badge-danger-border: transparent;--vp-badge-danger-text: var(--vp-c-danger-1);--vp-badge-danger-bg: var(--vp-c-danger-soft)}:root{--vp-carbon-ads-text-color: var(--vp-c-text-1);--vp-carbon-ads-poweredby-color: var(--vp-c-text-2);--vp-carbon-ads-bg-color: var(--vp-c-bg-soft);--vp-carbon-ads-hover-text-color: var(--vp-c-brand-1);--vp-carbon-ads-hover-poweredby-color: var(--vp-c-text-1)}:root{--vp-local-search-bg: var(--vp-c-bg);--vp-local-search-result-bg: var(--vp-c-bg);--vp-local-search-result-border: var(--vp-c-divider);--vp-local-search-result-selected-bg: var(--vp-c-bg);--vp-local-search-result-selected-border: var(--vp-c-brand-1);--vp-local-search-highlight-bg: var(--vp-c-brand-1);--vp-local-search-highlight-text: var(--vp-c-neutral-inverse)}@media (prefers-reduced-motion: reduce){*,:before,:after{animation-delay:-1ms!important;animation-duration:1ms!important;animation-iteration-count:1!important;background-attachment:initial!important;scroll-behavior:auto!important;transition-duration:0s!important;transition-delay:0s!important}}*,:before,:after{box-sizing:border-box}html{line-height:1.4;font-size:16px;-webkit-text-size-adjust:100%}html.dark{color-scheme:dark}body{margin:0;width:100%;min-width:320px;min-height:100vh;line-height:24px;font-family:var(--vp-font-family-base);font-size:16px;font-weight:400;color:var(--vp-c-text-1);background-color:var(--vp-c-bg);direction:ltr;font-synthesis:style;text-rendering:optimizeLegibility;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}main{display:block}h1,h2,h3,h4,h5,h6{margin:0;line-height:24px;font-size:16px;font-weight:400}p{margin:0}strong,b{font-weight:600}a,area,button,[role=button],input,label,select,summary,textarea{touch-action:manipulation}a{color:inherit;text-decoration:inherit}ol,ul{list-style:none;margin:0;padding:0}blockquote{margin:0}pre,code,kbd,samp{font-family:var(--vp-font-family-mono)}img,svg,video,canvas,audio,iframe,embed,object{display:block}figure{margin:0}img,video{max-width:100%;height:auto}button,input,optgroup,select,textarea{border:0;padding:0;line-height:inherit;color:inherit}button{padding:0;font-family:inherit;background-color:transparent;background-image:none}button:enabled,[role=button]:enabled{cursor:pointer}button:focus,button:focus-visible{outline:1px dotted;outline:4px auto -webkit-focus-ring-color}button:focus:not(:focus-visible){outline:none!important}input:focus,textarea:focus,select:focus{outline:none}table{border-collapse:collapse}input{background-color:transparent}input:-ms-input-placeholder,textarea:-ms-input-placeholder{color:var(--vp-c-text-3)}input::-ms-input-placeholder,textarea::-ms-input-placeholder{color:var(--vp-c-text-3)}input::placeholder,textarea::placeholder{color:var(--vp-c-text-3)}input::-webkit-outer-spin-button,input::-webkit-inner-spin-button{-webkit-appearance:none;margin:0}input[type=number]{-moz-appearance:textfield}textarea{resize:vertical}select{-webkit-appearance:none}fieldset{margin:0;padding:0}h1,h2,h3,h4,h5,h6,li,p{overflow-wrap:break-word}vite-error-overlay{z-index:9999}mjx-container{display:inline-block;margin:auto 2px -2px}mjx-container>svg{margin:auto}.visually-hidden{position:absolute;width:1px;height:1px;white-space:nowrap;clip:rect(0 0 0 0);clip-path:inset(50%);overflow:hidden}.custom-block{border:1px solid transparent;border-radius:8px;padding:16px 16px 8px;line-height:24px;font-size:var(--vp-custom-block-font-size);color:var(--vp-c-text-2)}.custom-block.info{border-color:var(--vp-custom-block-info-border);color:var(--vp-custom-block-info-text);background-color:var(--vp-custom-block-info-bg)}.custom-block.info a,.custom-block.info code{color:var(--vp-c-brand-1)}.custom-block.info a:hover{color:var(--vp-c-brand-2)}.custom-block.info code{background-color:var(--vp-custom-block-info-code-bg)}.custom-block.tip{border-color:var(--vp-custom-block-tip-border);color:var(--vp-custom-block-tip-text);background-color:var(--vp-custom-block-tip-bg)}.custom-block.tip a,.custom-block.tip code{color:var(--vp-c-brand-1)}.custom-block.tip a:hover{color:var(--vp-c-brand-2)}.custom-block.tip code{background-color:var(--vp-custom-block-tip-code-bg)}.custom-block.warning{border-color:var(--vp-custom-block-warning-border);color:var(--vp-custom-block-warning-text);background-color:var(--vp-custom-block-warning-bg)}.custom-block.warning a,.custom-block.warning code{color:var(--vp-c-warning-1)}.custom-block.warning a:hover{color:var(--vp-c-warning-2)}.custom-block.warning code{background-color:var(--vp-custom-block-warning-code-bg)}.custom-block.danger{border-color:var(--vp-custom-block-danger-border);color:var(--vp-custom-block-danger-text);background-color:var(--vp-custom-block-danger-bg)}.custom-block.danger a,.custom-block.danger code{color:var(--vp-c-danger-1)}.custom-block.danger a:hover{color:var(--vp-c-danger-2)}.custom-block.danger code{background-color:var(--vp-custom-block-danger-code-bg)}.custom-block.details{border-color:var(--vp-custom-block-details-border);color:var(--vp-custom-block-details-text);background-color:var(--vp-custom-block-details-bg)}.custom-block.details a{color:var(--vp-c-brand-1)}.custom-block.details a:hover{color:var(--vp-c-brand-2)}.custom-block.details code{background-color:var(--vp-custom-block-details-code-bg)}.custom-block-title{font-weight:600}.custom-block p+p{margin:8px 0}.custom-block.details summary{margin:0 0 8px;font-weight:700;cursor:pointer}.custom-block.details summary+p{margin:8px 0}.custom-block a{color:inherit;font-weight:600;text-decoration:underline;text-underline-offset:2px;transition:opacity .25s}.custom-block a:hover{opacity:.75}.custom-block code{font-size:var(--vp-custom-block-code-font-size)}.custom-block.custom-block th,.custom-block.custom-block blockquote>p{font-size:var(--vp-custom-block-font-size);color:inherit}.dark .vp-code-light{display:none}html:not(.dark) .vp-code-dark{display:none}.vp-code-group{margin-top:16px}.vp-code-group .tabs{position:relative;display:flex;margin-right:-24px;margin-left:-24px;padding:0 12px;background-color:var(--vp-code-tab-bg);overflow-x:auto;overflow-y:hidden;box-shadow:inset 0 -1px var(--vp-code-tab-divider)}@media (min-width: 640px){.vp-code-group .tabs{margin-right:0;margin-left:0;border-radius:8px 8px 0 0}}.vp-code-group .tabs input{position:fixed;opacity:0;pointer-events:none}.vp-code-group .tabs label{position:relative;display:inline-block;border-bottom:1px solid transparent;padding:0 12px;line-height:48px;font-size:14px;font-weight:500;color:var(--vp-code-tab-text-color);white-space:nowrap;cursor:pointer;transition:color .25s}.vp-code-group .tabs label:after{position:absolute;right:8px;bottom:-1px;left:8px;z-index:1;height:2px;border-radius:2px;content:"";background-color:transparent;transition:background-color .25s}.vp-code-group label:hover{color:var(--vp-code-tab-hover-text-color)}.vp-code-group input:checked+label{color:var(--vp-code-tab-active-text-color)}.vp-code-group input:checked+label:after{background-color:var(--vp-code-tab-active-bar-color)}.vp-code-group div[class*=language-],.vp-block{display:none;margin-top:0!important;border-top-left-radius:0!important;border-top-right-radius:0!important}.vp-code-group div[class*=language-].active,.vp-block.active{display:block}.vp-block{padding:20px 24px}.vp-doc h1,.vp-doc h2,.vp-doc h3,.vp-doc h4,.vp-doc h5,.vp-doc h6{position:relative;font-weight:600;outline:none}.vp-doc h1{letter-spacing:-.02em;line-height:40px;font-size:28px}.vp-doc h2{margin:48px 0 16px;border-top:1px solid var(--vp-c-divider);padding-top:24px;letter-spacing:-.02em;line-height:32px;font-size:24px}.vp-doc h3{margin:32px 0 0;letter-spacing:-.01em;line-height:28px;font-size:20px}.vp-doc .header-anchor{position:absolute;top:0;left:0;margin-left:-.87em;font-weight:500;-webkit-user-select:none;user-select:none;opacity:0;text-decoration:none;transition:color .25s,opacity .25s}.vp-doc .header-anchor:before{content:var(--vp-header-anchor-symbol)}.vp-doc h1:hover .header-anchor,.vp-doc h1 .header-anchor:focus,.vp-doc h2:hover .header-anchor,.vp-doc h2 .header-anchor:focus,.vp-doc h3:hover .header-anchor,.vp-doc h3 .header-anchor:focus,.vp-doc h4:hover .header-anchor,.vp-doc h4 .header-anchor:focus,.vp-doc h5:hover .header-anchor,.vp-doc h5 .header-anchor:focus,.vp-doc h6:hover .header-anchor,.vp-doc h6 .header-anchor:focus{opacity:1}@media (min-width: 768px){.vp-doc h1{letter-spacing:-.02em;line-height:40px;font-size:32px}}.vp-doc h2 .header-anchor{top:24px}.vp-doc p,.vp-doc summary{margin:16px 0}.vp-doc p{line-height:28px}.vp-doc blockquote{margin:16px 0;border-left:2px solid var(--vp-c-divider);padding-left:16px;transition:border-color .5s}.vp-doc blockquote>p{margin:0;font-size:16px;color:var(--vp-c-text-2);transition:color .5s}.vp-doc a{font-weight:500;color:var(--vp-c-brand-1);text-decoration:underline;text-underline-offset:2px;transition:color .25s,opacity .25s}.vp-doc a:hover{color:var(--vp-c-brand-2)}.vp-doc strong{font-weight:600}.vp-doc ul,.vp-doc ol{padding-left:1.25rem;margin:16px 0}.vp-doc ul{list-style:disc}.vp-doc ol{list-style:decimal}.vp-doc li+li{margin-top:8px}.vp-doc li>ol,.vp-doc li>ul{margin:8px 0 0}.vp-doc table{display:block;border-collapse:collapse;margin:20px 0;overflow-x:auto}.vp-doc tr{border-top:1px solid var(--vp-c-divider);transition:background-color .5s}.vp-doc tr:nth-child(2n){background-color:var(--vp-c-bg-soft)}.vp-doc th,.vp-doc td{border:1px solid var(--vp-c-divider);padding:8px 16px}.vp-doc th{text-align:left;font-size:14px;font-weight:600;color:var(--vp-c-text-2);background-color:var(--vp-c-bg-soft)}.vp-doc td{font-size:14px}.vp-doc hr{margin:16px 0;border:none;border-top:1px solid var(--vp-c-divider)}.vp-doc .custom-block{margin:16px 0}.vp-doc .custom-block p{margin:8px 0;line-height:24px}.vp-doc .custom-block p:first-child{margin:0}.vp-doc .custom-block div[class*=language-]{margin:8px 0;border-radius:8px}.vp-doc .custom-block div[class*=language-] code{font-weight:400;background-color:transparent}.vp-doc .custom-block .vp-code-group .tabs{margin:0;border-radius:8px 8px 0 0}.vp-doc :not(pre,h1,h2,h3,h4,h5,h6)>code{font-size:var(--vp-code-font-size);color:var(--vp-code-color)}.vp-doc :not(pre)>code{border-radius:4px;padding:3px 6px;background-color:var(--vp-code-bg);transition:color .25s,background-color .5s}.vp-doc a>code{color:var(--vp-code-link-color)}.vp-doc a:hover>code{color:var(--vp-code-link-hover-color)}.vp-doc h1>code,.vp-doc h2>code,.vp-doc h3>code{font-size:.9em}.vp-doc div[class*=language-],.vp-block{position:relative;margin:16px -24px;background-color:var(--vp-code-block-bg);overflow-x:auto;transition:background-color .5s}@media (min-width: 640px){.vp-doc div[class*=language-],.vp-block{border-radius:8px;margin:16px 0}}@media (max-width: 639px){.vp-doc li div[class*=language-]{border-radius:8px 0 0 8px}}.vp-doc div[class*=language-]+div[class*=language-],.vp-doc div[class$=-api]+div[class*=language-],.vp-doc div[class*=language-]+div[class$=-api]>div[class*=language-]{margin-top:-8px}.vp-doc [class*=language-] pre,.vp-doc [class*=language-] code{direction:ltr;text-align:left;white-space:pre;word-spacing:normal;word-break:normal;word-wrap:normal;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-hyphens:none;-moz-hyphens:none;-ms-hyphens:none;hyphens:none}.vp-doc [class*=language-] pre{position:relative;z-index:1;margin:0;padding:20px 0;background:transparent;overflow-x:auto}.vp-doc [class*=language-] code{display:block;padding:0 24px;width:fit-content;min-width:100%;line-height:var(--vp-code-line-height);font-size:var(--vp-code-font-size);color:var(--vp-code-block-color);transition:color .5s}.vp-doc [class*=language-] code .highlighted{background-color:var(--vp-code-line-highlight-color);transition:background-color .5s;margin:0 -24px;padding:0 24px;width:calc(100% + 48px);display:inline-block}.vp-doc [class*=language-] code .highlighted.error{background-color:var(--vp-code-line-error-color)}.vp-doc [class*=language-] code .highlighted.warning{background-color:var(--vp-code-line-warning-color)}.vp-doc [class*=language-] code .diff{transition:background-color .5s;margin:0 -24px;padding:0 24px;width:calc(100% + 48px);display:inline-block}.vp-doc [class*=language-] code .diff:before{position:absolute;left:10px}.vp-doc [class*=language-] .has-focused-lines .line:not(.has-focus){filter:blur(.095rem);opacity:.4;transition:filter .35s,opacity .35s}.vp-doc [class*=language-] .has-focused-lines .line:not(.has-focus){opacity:.7;transition:filter .35s,opacity .35s}.vp-doc [class*=language-]:hover .has-focused-lines .line:not(.has-focus){filter:blur(0);opacity:1}.vp-doc [class*=language-] code .diff.remove{background-color:var(--vp-code-line-diff-remove-color);opacity:.7}.vp-doc [class*=language-] code .diff.remove:before{content:"-";color:var(--vp-code-line-diff-remove-symbol-color)}.vp-doc [class*=language-] code .diff.add{background-color:var(--vp-code-line-diff-add-color)}.vp-doc [class*=language-] code .diff.add:before{content:"+";color:var(--vp-code-line-diff-add-symbol-color)}.vp-doc div[class*=language-].line-numbers-mode{padding-left:32px}.vp-doc .line-numbers-wrapper{position:absolute;top:0;bottom:0;left:0;z-index:3;border-right:1px solid var(--vp-code-block-divider-color);padding-top:20px;width:32px;text-align:center;font-family:var(--vp-font-family-mono);line-height:var(--vp-code-line-height);font-size:var(--vp-code-font-size);color:var(--vp-code-line-number-color);transition:border-color .5s,color .5s}.vp-doc [class*=language-]>button.copy{direction:ltr;position:absolute;top:12px;right:12px;z-index:3;border:1px solid var(--vp-code-copy-code-border-color);border-radius:4px;width:40px;height:40px;background-color:var(--vp-code-copy-code-bg);opacity:0;cursor:pointer;background-image:var(--vp-icon-copy);background-position:50%;background-size:20px;background-repeat:no-repeat;transition:border-color .25s,background-color .25s,opacity .25s}.vp-doc [class*=language-]:hover>button.copy,.vp-doc [class*=language-]>button.copy:focus{opacity:1}.vp-doc [class*=language-]>button.copy:hover,.vp-doc [class*=language-]>button.copy.copied{border-color:var(--vp-code-copy-code-hover-border-color);background-color:var(--vp-code-copy-code-hover-bg)}.vp-doc [class*=language-]>button.copy.copied,.vp-doc [class*=language-]>button.copy:hover.copied{border-radius:0 4px 4px 0;background-color:var(--vp-code-copy-code-hover-bg);background-image:var(--vp-icon-copied)}.vp-doc [class*=language-]>button.copy.copied:before,.vp-doc [class*=language-]>button.copy:hover.copied:before{position:relative;top:-1px;transform:translate(calc(-100% - 1px));display:flex;justify-content:center;align-items:center;border:1px solid var(--vp-code-copy-code-hover-border-color);border-right:0;border-radius:4px 0 0 4px;padding:0 10px;width:fit-content;height:40px;text-align:center;font-size:12px;font-weight:500;color:var(--vp-code-copy-code-active-text);background-color:var(--vp-code-copy-code-hover-bg);white-space:nowrap;content:var(--vp-code-copy-copied-text-content)}.vp-doc [class*=language-]>span.lang{position:absolute;top:2px;right:8px;z-index:2;font-size:12px;font-weight:500;color:var(--vp-code-lang-color);transition:color .4s,opacity .4s}.vp-doc [class*=language-]:hover>button.copy+span.lang,.vp-doc [class*=language-]>button.copy:focus+span.lang{opacity:0}.vp-doc .VPTeamMembers{margin-top:24px}.vp-doc .VPTeamMembers.small.count-1 .container{margin:0!important;max-width:calc((100% - 24px)/2)!important}.vp-doc .VPTeamMembers.small.count-2 .container,.vp-doc .VPTeamMembers.small.count-3 .container{max-width:100%!important}.vp-doc .VPTeamMembers.medium.count-1 .container{margin:0!important;max-width:calc((100% - 24px)/2)!important}:is(.vp-external-link-icon,.vp-doc a[href*="://"],.vp-doc a[target=_blank]):not(.no-icon):after{display:inline-block;margin-top:-1px;margin-left:4px;width:11px;height:11px;background:currentColor;color:var(--vp-c-text-3);flex-shrink:0;--icon: url("data:image/svg+xml, %3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' %3E%3Cpath d='M0 0h24v24H0V0z' fill='none' /%3E%3Cpath d='M9 5v2h6.59L4 18.59 5.41 20 17 8.41V15h2V5H9z' /%3E%3C/svg%3E");-webkit-mask-image:var(--icon);mask-image:var(--icon)}.vp-external-link-icon:after{content:""}.vp-sponsor{border-radius:16px;overflow:hidden}.vp-sponsor.aside{border-radius:12px}.vp-sponsor-section+.vp-sponsor-section{margin-top:4px}.vp-sponsor-tier{margin-bottom:4px;text-align:center;letter-spacing:1px;line-height:24px;width:100%;font-weight:600;color:var(--vp-c-text-2);background-color:var(--vp-c-bg-soft)}.vp-sponsor.normal .vp-sponsor-tier{padding:13px 0 11px;font-size:14px}.vp-sponsor.aside .vp-sponsor-tier{padding:9px 0 7px;font-size:12px}.vp-sponsor-grid+.vp-sponsor-tier{margin-top:4px}.vp-sponsor-grid{display:flex;flex-wrap:wrap;gap:4px}.vp-sponsor-grid.xmini .vp-sponsor-grid-link{height:64px}.vp-sponsor-grid.xmini .vp-sponsor-grid-image{max-width:64px;max-height:22px}.vp-sponsor-grid.mini .vp-sponsor-grid-link{height:72px}.vp-sponsor-grid.mini .vp-sponsor-grid-image{max-width:96px;max-height:24px}.vp-sponsor-grid.small .vp-sponsor-grid-link{height:96px}.vp-sponsor-grid.small .vp-sponsor-grid-image{max-width:96px;max-height:24px}.vp-sponsor-grid.medium .vp-sponsor-grid-link{height:112px}.vp-sponsor-grid.medium .vp-sponsor-grid-image{max-width:120px;max-height:36px}.vp-sponsor-grid.big .vp-sponsor-grid-link{height:184px}.vp-sponsor-grid.big .vp-sponsor-grid-image{max-width:192px;max-height:56px}.vp-sponsor-grid[data-vp-grid="2"] .vp-sponsor-grid-item{width:calc((100% - 4px)/2)}.vp-sponsor-grid[data-vp-grid="3"] .vp-sponsor-grid-item{width:calc((100% - 4px * 2) / 3)}.vp-sponsor-grid[data-vp-grid="4"] .vp-sponsor-grid-item{width:calc((100% - 12px)/4)}.vp-sponsor-grid[data-vp-grid="5"] .vp-sponsor-grid-item{width:calc((100% - 16px)/5)}.vp-sponsor-grid[data-vp-grid="6"] .vp-sponsor-grid-item{width:calc((100% - 4px * 5) / 6)}.vp-sponsor-grid-item{flex-shrink:0;width:100%;background-color:var(--vp-c-bg-soft);transition:background-color .25s}.vp-sponsor-grid-item:hover{background-color:var(--vp-c-default-soft)}.vp-sponsor-grid-item:hover .vp-sponsor-grid-image{filter:grayscale(0) invert(0)}.vp-sponsor-grid-item.empty:hover{background-color:var(--vp-c-bg-soft)}.dark .vp-sponsor-grid-item:hover{background-color:var(--vp-c-white)}.dark .vp-sponsor-grid-item.empty:hover{background-color:var(--vp-c-bg-soft)}.vp-sponsor-grid-link{display:flex}.vp-sponsor-grid-box{display:flex;justify-content:center;align-items:center;width:100%}.vp-sponsor-grid-image{max-width:100%;filter:grayscale(1);transition:filter .25s}.dark .vp-sponsor-grid-image{filter:grayscale(1) invert(1)}.VPBadge[data-v-ea5b2908]{display:inline-block;margin-left:2px;border:1px solid transparent;border-radius:12px;padding:0 10px;line-height:22px;font-size:12px;font-weight:500;transform:translateY(-2px)}.vp-doc h1>.VPBadge[data-v-ea5b2908]{margin-top:4px;vertical-align:top}.vp-doc h2>.VPBadge[data-v-ea5b2908]{margin-top:3px;padding:0 8px;vertical-align:top}.vp-doc h3>.VPBadge[data-v-ea5b2908]{vertical-align:middle}.vp-doc h4>.VPBadge[data-v-ea5b2908],.vp-doc h5>.VPBadge[data-v-ea5b2908],.vp-doc h6>.VPBadge[data-v-ea5b2908]{vertical-align:middle;line-height:18px}.VPBadge.info[data-v-ea5b2908]{border-color:var(--vp-badge-info-border);color:var(--vp-badge-info-text);background-color:var(--vp-badge-info-bg)}.VPBadge.tip[data-v-ea5b2908]{border-color:var(--vp-badge-tip-border);color:var(--vp-badge-tip-text);background-color:var(--vp-badge-tip-bg)}.VPBadge.warning[data-v-ea5b2908]{border-color:var(--vp-badge-warning-border);color:var(--vp-badge-warning-text);background-color:var(--vp-badge-warning-bg)}.VPBadge.danger[data-v-ea5b2908]{border-color:var(--vp-badge-danger-border);color:var(--vp-badge-danger-text);background-color:var(--vp-badge-danger-bg)}.VPBackdrop[data-v-54a304ca]{position:fixed;top:0;right:0;bottom:0;left:0;z-index:var(--vp-z-index-backdrop);background:var(--vp-backdrop-bg-color);transition:opacity .5s}.VPBackdrop.fade-enter-from[data-v-54a304ca],.VPBackdrop.fade-leave-to[data-v-54a304ca]{opacity:0}.VPBackdrop.fade-leave-active[data-v-54a304ca]{transition-duration:.25s}@media (min-width: 1280px){.VPBackdrop[data-v-54a304ca]{display:none}}.NotFound[data-v-b9c0c15a]{padding:64px 24px 96px;text-align:center}@media (min-width: 768px){.NotFound[data-v-b9c0c15a]{padding:96px 32px 168px}}.code[data-v-b9c0c15a]{line-height:64px;font-size:64px;font-weight:600}.title[data-v-b9c0c15a]{padding-top:12px;letter-spacing:2px;line-height:20px;font-size:20px;font-weight:700}.divider[data-v-b9c0c15a]{margin:24px auto 18px;width:64px;height:1px;background-color:var(--vp-c-divider)}.quote[data-v-b9c0c15a]{margin:0 auto;max-width:256px;font-size:14px;font-weight:500;color:var(--vp-c-text-2)}.action[data-v-b9c0c15a]{padding-top:20px}.link[data-v-b9c0c15a]{display:inline-block;border:1px solid var(--vp-c-brand-1);border-radius:16px;padding:3px 16px;font-size:14px;font-weight:500;color:var(--vp-c-brand-1);transition:border-color .25s,color .25s}.link[data-v-b9c0c15a]:hover{border-color:var(--vp-c-brand-2);color:var(--vp-c-brand-2)}.root[data-v-463da30f]{position:relative;z-index:1}.nested[data-v-463da30f]{padding-left:16px}.outline-link[data-v-463da30f]{display:block;line-height:28px;color:var(--vp-c-text-2);white-space:nowrap;overflow:hidden;text-overflow:ellipsis;transition:color .5s;font-weight:400}.outline-link[data-v-463da30f]:hover,.outline-link.active[data-v-463da30f]{color:var(--vp-c-text-1);transition:color .25s}.outline-link.nested[data-v-463da30f]{padding-left:13px}.VPDocAsideOutline[data-v-3a6c4994]{display:none}.VPDocAsideOutline.has-outline[data-v-3a6c4994]{display:block}.content[data-v-3a6c4994]{position:relative;border-left:1px solid var(--vp-c-divider);padding-left:16px;font-size:13px;font-weight:500}.outline-marker[data-v-3a6c4994]{position:absolute;top:32px;left:-1px;z-index:0;opacity:0;width:2px;border-radius:2px;height:18px;background-color:var(--vp-c-brand-1);transition:top .25s cubic-bezier(0,1,.5,1),background-color .5s,opacity .25s}.outline-title[data-v-3a6c4994]{letter-spacing:.4px;line-height:28px;font-size:13px;font-weight:600}.VPDocAside[data-v-cb998dce]{display:flex;flex-direction:column;flex-grow:1}.spacer[data-v-cb998dce]{flex-grow:1}.VPDocAside[data-v-cb998dce] .spacer+.VPDocAsideSponsors,.VPDocAside[data-v-cb998dce] .spacer+.VPDocAsideCarbonAds{margin-top:24px}.VPDocAside[data-v-cb998dce] .VPDocAsideSponsors+.VPDocAsideCarbonAds{margin-top:16px}.VPLastUpdated[data-v-19a7ae4e]{line-height:24px;font-size:14px;font-weight:500;color:var(--vp-c-text-2)}@media (min-width: 640px){.VPLastUpdated[data-v-19a7ae4e]{line-height:32px;font-size:14px;font-weight:500}}.VPDocFooter[data-v-a2d931e4]{margin-top:64px}.edit-info[data-v-a2d931e4]{padding-bottom:18px}@media (min-width: 640px){.edit-info[data-v-a2d931e4]{display:flex;justify-content:space-between;align-items:center;padding-bottom:14px}}.edit-link-button[data-v-a2d931e4]{display:flex;align-items:center;border:0;line-height:32px;font-size:14px;font-weight:500;color:var(--vp-c-brand-1);transition:color .25s}.edit-link-button[data-v-a2d931e4]:hover{color:var(--vp-c-brand-2)}.edit-link-icon[data-v-a2d931e4]{margin-right:8px;width:14px;height:14px;fill:currentColor}.prev-next[data-v-a2d931e4]{border-top:1px solid var(--vp-c-divider);padding-top:24px;display:grid;grid-row-gap:8px}@media (min-width: 640px){.prev-next[data-v-a2d931e4]{grid-template-columns:repeat(2,1fr);grid-column-gap:16px}}.pager-link[data-v-a2d931e4]{display:block;border:1px solid var(--vp-c-divider);border-radius:8px;padding:11px 16px 13px;width:100%;height:100%;transition:border-color .25s}.pager-link[data-v-a2d931e4]:hover{border-color:var(--vp-c-brand-1)}.pager-link.next[data-v-a2d931e4]{margin-left:auto;text-align:right}.desc[data-v-a2d931e4]{display:block;line-height:20px;font-size:12px;font-weight:500;color:var(--vp-c-text-2)}.title[data-v-a2d931e4]{display:block;line-height:20px;font-size:14px;font-weight:500;color:var(--vp-c-brand-1);transition:color .25s}.VPDocOutlineDropdown[data-v-95bb0785]{margin-bottom:48px}.VPDocOutlineDropdown button[data-v-95bb0785]{display:block;font-size:14px;font-weight:500;line-height:24px;border:1px solid var(--vp-c-border);padding:4px 12px;color:var(--vp-c-text-2);background-color:var(--vp-c-default-soft);border-radius:8px;transition:color .5s}.VPDocOutlineDropdown button[data-v-95bb0785]:hover{color:var(--vp-c-text-1);transition:color .25s}.VPDocOutlineDropdown button.open[data-v-95bb0785]{color:var(--vp-c-text-1)}.icon[data-v-95bb0785]{display:inline-block;vertical-align:middle;width:16px;height:16px;fill:currentColor}[data-v-95bb0785] .outline-link{font-size:14px;font-weight:400}.open>.icon[data-v-95bb0785]{transform:rotate(90deg)}.items[data-v-95bb0785]{margin-top:12px;border-left:1px solid var(--vp-c-divider)}.VPDoc[data-v-a3c25e27]{padding:32px 24px 96px;width:100%}.VPDoc .VPDocOutlineDropdown[data-v-a3c25e27]{display:none}@media (min-width: 960px) and (max-width: 1279px){.VPDoc .VPDocOutlineDropdown[data-v-a3c25e27]{display:block}}@media (min-width: 768px){.VPDoc[data-v-a3c25e27]{padding:48px 32px 128px}}@media (min-width: 960px){.VPDoc[data-v-a3c25e27]{padding:32px 32px 0}.VPDoc:not(.has-sidebar) .container[data-v-a3c25e27]{display:flex;justify-content:center;max-width:992px}.VPDoc:not(.has-sidebar) .content[data-v-a3c25e27]{max-width:752px}}@media (min-width: 1280px){.VPDoc .container[data-v-a3c25e27]{display:flex;justify-content:center}.VPDoc .aside[data-v-a3c25e27]{display:block}}@media (min-width: 1440px){.VPDoc:not(.has-sidebar) .content[data-v-a3c25e27]{max-width:784px}.VPDoc:not(.has-sidebar) .container[data-v-a3c25e27]{max-width:1104px}}.container[data-v-a3c25e27]{margin:0 auto;width:100%}.aside[data-v-a3c25e27]{position:relative;display:none;order:2;flex-grow:1;padding-left:32px;width:100%;max-width:256px}.left-aside[data-v-a3c25e27]{order:1;padding-left:unset;padding-right:32px}.aside-container[data-v-a3c25e27]{position:fixed;top:0;padding-top:calc(var(--vp-nav-height) + var(--vp-layout-top-height, 0px) + var(--vp-doc-top-height, 0px) + 32px);width:224px;height:100vh;overflow-x:hidden;overflow-y:auto;scrollbar-width:none}.aside-container[data-v-a3c25e27]::-webkit-scrollbar{display:none}.aside-curtain[data-v-a3c25e27]{position:fixed;bottom:0;z-index:10;width:224px;height:32px;background:linear-gradient(transparent,var(--vp-c-bg) 70%)}.aside-content[data-v-a3c25e27]{display:flex;flex-direction:column;min-height:calc(100vh - (var(--vp-nav-height) + var(--vp-layout-top-height, 0px) + 32px));padding-bottom:32px}.content[data-v-a3c25e27]{position:relative;margin:0 auto;width:100%}@media (min-width: 960px){.content[data-v-a3c25e27]{padding:0 32px 128px}}@media (min-width: 1280px){.content[data-v-a3c25e27]{order:1;margin:0;min-width:640px}}.content-container[data-v-a3c25e27]{margin:0 auto}.VPDoc.has-aside .content-container[data-v-a3c25e27]{max-width:688px}.external-link-icon-enabled[data-v-a3c25e27] :is(.vp-doc a[href*="://"],.vp-doc a[target=_blank]):after{content:"";color:currentColor}.VPButton[data-v-1e76fe75]{display:inline-block;border:1px solid transparent;text-align:center;font-weight:600;white-space:nowrap;transition:color .25s,border-color .25s,background-color .25s}.VPButton[data-v-1e76fe75]:active{transition:color .1s,border-color .1s,background-color .1s}.VPButton.medium[data-v-1e76fe75]{border-radius:20px;padding:0 20px;line-height:38px;font-size:14px}.VPButton.big[data-v-1e76fe75]{border-radius:24px;padding:0 24px;line-height:46px;font-size:16px}.VPButton.brand[data-v-1e76fe75]{border-color:var(--vp-button-brand-border);color:var(--vp-button-brand-text);background-color:var(--vp-button-brand-bg)}.VPButton.brand[data-v-1e76fe75]:hover{border-color:var(--vp-button-brand-hover-border);color:var(--vp-button-brand-hover-text);background-color:var(--vp-button-brand-hover-bg)}.VPButton.brand[data-v-1e76fe75]:active{border-color:var(--vp-button-brand-active-border);color:var(--vp-button-brand-active-text);background-color:var(--vp-button-brand-active-bg)}.VPButton.alt[data-v-1e76fe75]{border-color:var(--vp-button-alt-border);color:var(--vp-button-alt-text);background-color:var(--vp-button-alt-bg)}.VPButton.alt[data-v-1e76fe75]:hover{border-color:var(--vp-button-alt-hover-border);color:var(--vp-button-alt-hover-text);background-color:var(--vp-button-alt-hover-bg)}.VPButton.alt[data-v-1e76fe75]:active{border-color:var(--vp-button-alt-active-border);color:var(--vp-button-alt-active-text);background-color:var(--vp-button-alt-active-bg)}.VPButton.sponsor[data-v-1e76fe75]{border-color:var(--vp-button-sponsor-border);color:var(--vp-button-sponsor-text);background-color:var(--vp-button-sponsor-bg)}.VPButton.sponsor[data-v-1e76fe75]:hover{border-color:var(--vp-button-sponsor-hover-border);color:var(--vp-button-sponsor-hover-text);background-color:var(--vp-button-sponsor-hover-bg)}.VPButton.sponsor[data-v-1e76fe75]:active{border-color:var(--vp-button-sponsor-active-border);color:var(--vp-button-sponsor-active-text);background-color:var(--vp-button-sponsor-active-bg)}html:not(.dark) .VPImage.dark[data-v-ab19afbb]{display:none}.dark .VPImage.light[data-v-ab19afbb]{display:none}.VPHero[data-v-5a3e9999]{margin-top:calc((var(--vp-nav-height) + var(--vp-layout-top-height, 0px)) * -1);padding:calc(var(--vp-nav-height) + var(--vp-layout-top-height, 0px) + 48px) 24px 48px}@media (min-width: 640px){.VPHero[data-v-5a3e9999]{padding:calc(var(--vp-nav-height) + var(--vp-layout-top-height, 0px) + 80px) 48px 64px}}@media (min-width: 960px){.VPHero[data-v-5a3e9999]{padding:calc(var(--vp-nav-height) + var(--vp-layout-top-height, 0px) + 80px) 64px 64px}}.container[data-v-5a3e9999]{display:flex;flex-direction:column;margin:0 auto;max-width:1152px}@media (min-width: 960px){.container[data-v-5a3e9999]{flex-direction:row}}.main[data-v-5a3e9999]{position:relative;z-index:10;order:2;flex-grow:1;flex-shrink:0}.VPHero.has-image .container[data-v-5a3e9999]{text-align:center}@media (min-width: 960px){.VPHero.has-image .container[data-v-5a3e9999]{text-align:left}}@media (min-width: 960px){.main[data-v-5a3e9999]{order:1;width:calc((100% / 3) * 2)}.VPHero.has-image .main[data-v-5a3e9999]{max-width:592px}}.name[data-v-5a3e9999],.text[data-v-5a3e9999]{max-width:392px;letter-spacing:-.4px;line-height:40px;font-size:32px;font-weight:700;white-space:pre-wrap}.VPHero.has-image .name[data-v-5a3e9999],.VPHero.has-image .text[data-v-5a3e9999]{margin:0 auto}.name[data-v-5a3e9999]{color:var(--vp-home-hero-name-color)}.clip[data-v-5a3e9999]{background:var(--vp-home-hero-name-background);-webkit-background-clip:text;background-clip:text;-webkit-text-fill-color:var(--vp-home-hero-name-color)}@media (min-width: 640px){.name[data-v-5a3e9999],.text[data-v-5a3e9999]{max-width:576px;line-height:56px;font-size:48px}}@media (min-width: 960px){.name[data-v-5a3e9999],.text[data-v-5a3e9999]{line-height:64px;font-size:56px}.VPHero.has-image .name[data-v-5a3e9999],.VPHero.has-image .text[data-v-5a3e9999]{margin:0}}.tagline[data-v-5a3e9999]{padding-top:8px;max-width:392px;line-height:28px;font-size:18px;font-weight:500;white-space:pre-wrap;color:var(--vp-c-text-2)}.VPHero.has-image .tagline[data-v-5a3e9999]{margin:0 auto}@media (min-width: 640px){.tagline[data-v-5a3e9999]{padding-top:12px;max-width:576px;line-height:32px;font-size:20px}}@media (min-width: 960px){.tagline[data-v-5a3e9999]{line-height:36px;font-size:24px}.VPHero.has-image .tagline[data-v-5a3e9999]{margin:0}}.actions[data-v-5a3e9999]{display:flex;flex-wrap:wrap;margin:-6px;padding-top:24px}.VPHero.has-image .actions[data-v-5a3e9999]{justify-content:center}@media (min-width: 640px){.actions[data-v-5a3e9999]{padding-top:32px}}@media (min-width: 960px){.VPHero.has-image .actions[data-v-5a3e9999]{justify-content:flex-start}}.action[data-v-5a3e9999]{flex-shrink:0;padding:6px}.image[data-v-5a3e9999]{order:1;margin:-76px -24px -48px}@media (min-width: 640px){.image[data-v-5a3e9999]{margin:-108px -24px -48px}}@media (min-width: 960px){.image[data-v-5a3e9999]{flex-grow:1;order:2;margin:0;min-height:100%}}.image-container[data-v-5a3e9999]{position:relative;margin:0 auto;width:320px;height:320px}@media (min-width: 640px){.image-container[data-v-5a3e9999]{width:392px;height:392px}}@media (min-width: 960px){.image-container[data-v-5a3e9999]{display:flex;justify-content:center;align-items:center;width:100%;height:100%;transform:translate(-32px,-32px)}}.image-bg[data-v-5a3e9999]{position:absolute;top:50%;left:50%;border-radius:50%;width:192px;height:192px;background-image:var(--vp-home-hero-image-background-image);filter:var(--vp-home-hero-image-filter);transform:translate(-50%,-50%)}@media (min-width: 640px){.image-bg[data-v-5a3e9999]{width:256px;height:256px}}@media (min-width: 960px){.image-bg[data-v-5a3e9999]{width:320px;height:320px}}[data-v-5a3e9999] .image-src{position:absolute;top:50%;left:50%;max-width:192px;max-height:192px;transform:translate(-50%,-50%)}@media (min-width: 640px){[data-v-5a3e9999] .image-src{max-width:256px;max-height:256px}}@media (min-width: 960px){[data-v-5a3e9999] .image-src{max-width:320px;max-height:320px}}.VPFeature[data-v-ee984185]{display:block;border:1px solid var(--vp-c-bg-soft);border-radius:12px;height:100%;background-color:var(--vp-c-bg-soft);transition:border-color .25s,background-color .25s}.VPFeature.link[data-v-ee984185]:hover{border-color:var(--vp-c-brand-1)}.box[data-v-ee984185]{display:flex;flex-direction:column;padding:24px;height:100%}.box[data-v-ee984185]>.VPImage{margin-bottom:20px}.icon[data-v-ee984185]{display:flex;justify-content:center;align-items:center;margin-bottom:20px;border-radius:6px;background-color:var(--vp-c-default-soft);width:48px;height:48px;font-size:24px;transition:background-color .25s}.title[data-v-ee984185]{line-height:24px;font-size:16px;font-weight:600}.details[data-v-ee984185]{flex-grow:1;padding-top:8px;line-height:24px;font-size:14px;font-weight:500;color:var(--vp-c-text-2)}.link-text[data-v-ee984185]{padding-top:8px}.link-text-value[data-v-ee984185]{display:flex;align-items:center;font-size:14px;font-weight:500;color:var(--vp-c-brand-1)}.link-text-icon[data-v-ee984185]{display:inline-block;margin-left:6px;width:14px;height:14px;fill:currentColor}.VPFeatures[data-v-b1eea84a]{position:relative;padding:0 24px}@media (min-width: 640px){.VPFeatures[data-v-b1eea84a]{padding:0 48px}}@media (min-width: 960px){.VPFeatures[data-v-b1eea84a]{padding:0 64px}}.container[data-v-b1eea84a]{margin:0 auto;max-width:1152px}.items[data-v-b1eea84a]{display:flex;flex-wrap:wrap;margin:-8px}.item[data-v-b1eea84a]{padding:8px;width:100%}@media (min-width: 640px){.item.grid-2[data-v-b1eea84a],.item.grid-4[data-v-b1eea84a],.item.grid-6[data-v-b1eea84a]{width:50%}}@media (min-width: 768px){.item.grid-2[data-v-b1eea84a],.item.grid-4[data-v-b1eea84a]{width:50%}.item.grid-3[data-v-b1eea84a],.item.grid-6[data-v-b1eea84a]{width:calc(100% / 3)}}@media (min-width: 960px){.item.grid-4[data-v-b1eea84a]{width:25%}}.VPHome[data-v-20eabd3a]{padding-bottom:96px}.VPHome[data-v-20eabd3a] .VPHomeSponsors{margin-top:112px;margin-bottom:-128px}@media (min-width: 768px){.VPHome[data-v-20eabd3a]{padding-bottom:128px}}.VPContent[data-v-3cf691b6]{flex-grow:1;flex-shrink:0;margin:var(--vp-layout-top-height, 0px) auto 0;width:100%}.VPContent.is-home[data-v-3cf691b6]{width:100%;max-width:100%}.VPContent.has-sidebar[data-v-3cf691b6]{margin:0}@media (min-width: 960px){.VPContent[data-v-3cf691b6]{padding-top:var(--vp-nav-height)}.VPContent.has-sidebar[data-v-3cf691b6]{margin:var(--vp-layout-top-height, 0px) 0 0;padding-left:var(--vp-sidebar-width)}}@media (min-width: 1440px){.VPContent.has-sidebar[data-v-3cf691b6]{padding-right:calc((100vw - var(--vp-layout-max-width)) / 2);padding-left:calc((100vw - var(--vp-layout-max-width)) / 2 + var(--vp-sidebar-width))}}.VPFooter[data-v-e4279f1c]{position:relative;z-index:var(--vp-z-index-footer);border-top:1px solid var(--vp-c-gutter);padding:32px 24px;background-color:var(--vp-c-bg)}.VPFooter.has-sidebar[data-v-e4279f1c]{display:none}@media (min-width: 768px){.VPFooter[data-v-e4279f1c]{padding:32px}}.container[data-v-e4279f1c]{margin:0 auto;max-width:var(--vp-layout-max-width);text-align:center}.message[data-v-e4279f1c],.copyright[data-v-e4279f1c]{line-height:24px;font-size:14px;font-weight:500;color:var(--vp-c-text-2)}.VPLocalNavOutlineDropdown[data-v-24251f6f]{padding:12px 20px 11px}.VPLocalNavOutlineDropdown button[data-v-24251f6f]{display:block;font-size:12px;font-weight:500;line-height:24px;color:var(--vp-c-text-2);transition:color .5s;position:relative}.VPLocalNavOutlineDropdown button[data-v-24251f6f]:hover{color:var(--vp-c-text-1);transition:color .25s}.VPLocalNavOutlineDropdown button.open[data-v-24251f6f]{color:var(--vp-c-text-1)}.icon[data-v-24251f6f]{display:inline-block;vertical-align:middle;margin-left:2px;width:14px;height:14px;fill:currentColor}[data-v-24251f6f] .outline-link{font-size:14px;padding:2px 0}.open>.icon[data-v-24251f6f]{transform:rotate(90deg)}.items[data-v-24251f6f]{position:absolute;top:64px;right:16px;left:16px;display:grid;gap:1px;border:1px solid var(--vp-c-border);border-radius:8px;background-color:var(--vp-c-gutter);max-height:calc(var(--vp-vh, 100vh) - 86px);overflow:hidden auto;box-shadow:var(--vp-shadow-3)}.header[data-v-24251f6f]{background-color:var(--vp-c-bg-soft)}.top-link[data-v-24251f6f]{display:block;padding:0 16px;line-height:48px;font-size:14px;font-weight:500;color:var(--vp-c-brand-1)}.outline[data-v-24251f6f]{padding:8px 0;background-color:var(--vp-c-bg-soft)}.flyout-enter-active[data-v-24251f6f]{transition:all .2s ease-out}.flyout-leave-active[data-v-24251f6f]{transition:all .15s ease-in}.flyout-enter-from[data-v-24251f6f],.flyout-leave-to[data-v-24251f6f]{opacity:0;transform:translateY(-16px)}.VPLocalNav[data-v-9e669cc1]{position:sticky;top:0;left:0;z-index:var(--vp-z-index-local-nav);display:flex;justify-content:space-between;align-items:center;border-top:1px solid var(--vp-c-gutter);border-bottom:1px solid var(--vp-c-gutter);padding-top:var(--vp-layout-top-height, 0px);width:100%;background-color:var(--vp-local-nav-bg-color)}.VPLocalNav.fixed[data-v-9e669cc1]{position:fixed}.VPLocalNav.reached-top[data-v-9e669cc1]{border-top-color:transparent}@media (min-width: 960px){.VPLocalNav[data-v-9e669cc1]{display:none}}.menu[data-v-9e669cc1]{display:flex;align-items:center;padding:12px 24px 11px;line-height:24px;font-size:12px;font-weight:500;color:var(--vp-c-text-2);transition:color .5s}.menu[data-v-9e669cc1]:hover{color:var(--vp-c-text-1);transition:color .25s}@media (min-width: 768px){.menu[data-v-9e669cc1]{padding:0 32px}}.menu-icon[data-v-9e669cc1]{margin-right:8px;width:16px;height:16px;fill:currentColor}.VPOutlineDropdown[data-v-9e669cc1]{padding:12px 24px 11px}@media (min-width: 768px){.VPOutlineDropdown[data-v-9e669cc1]{padding:12px 32px 11px}}.VPSwitch[data-v-1c29e291]{position:relative;border-radius:11px;display:block;width:40px;height:22px;flex-shrink:0;border:1px solid var(--vp-input-border-color);background-color:var(--vp-input-switch-bg-color);transition:border-color .25s!important}.VPSwitch[data-v-1c29e291]:hover{border-color:var(--vp-c-brand-1)}.check[data-v-1c29e291]{position:absolute;top:1px;left:1px;width:18px;height:18px;border-radius:50%;background-color:var(--vp-c-neutral-inverse);box-shadow:var(--vp-shadow-1);transition:transform .25s!important}.icon[data-v-1c29e291]{position:relative;display:block;width:18px;height:18px;border-radius:50%;overflow:hidden}.icon[data-v-1c29e291] svg{position:absolute;top:3px;left:3px;width:12px;height:12px;fill:var(--vp-c-text-2)}.dark .icon[data-v-1c29e291] svg{fill:var(--vp-c-text-1);transition:opacity .25s!important}.sun[data-v-3329432d]{opacity:1}.moon[data-v-3329432d],.dark .sun[data-v-3329432d]{opacity:0}.dark .moon[data-v-3329432d]{opacity:1}.dark .VPSwitchAppearance[data-v-3329432d] .check{transform:translate(18px)}.VPNavBarAppearance[data-v-283b26e9]{display:none}@media (min-width: 1280px){.VPNavBarAppearance[data-v-283b26e9]{display:flex;align-items:center}}.VPMenuGroup+.VPMenuLink[data-v-f51f088d]{margin:12px -12px 0;border-top:1px solid var(--vp-c-divider);padding:12px 12px 0}.link[data-v-f51f088d]{display:block;border-radius:6px;padding:0 12px;line-height:32px;font-size:14px;font-weight:500;color:var(--vp-c-text-1);white-space:nowrap;transition:background-color .25s,color .25s}.link[data-v-f51f088d]:hover{color:var(--vp-c-brand-1);background-color:var(--vp-c-default-soft)}.link.active[data-v-f51f088d]{color:var(--vp-c-brand-1)}.VPMenuGroup[data-v-a6b0397c]{margin:12px -12px 0;border-top:1px solid var(--vp-c-divider);padding:12px 12px 0}.VPMenuGroup[data-v-a6b0397c]:first-child{margin-top:0;border-top:0;padding-top:0}.VPMenuGroup+.VPMenuGroup[data-v-a6b0397c]{margin-top:12px;border-top:1px solid var(--vp-c-divider)}.title[data-v-a6b0397c]{padding:0 12px;line-height:32px;font-size:14px;font-weight:600;color:var(--vp-c-text-2);white-space:nowrap;transition:color .25s}.VPMenu[data-v-e42ed9b3]{border-radius:12px;padding:12px;min-width:128px;border:1px solid var(--vp-c-divider);background-color:var(--vp-c-bg-elv);box-shadow:var(--vp-shadow-3);transition:background-color .5s;max-height:calc(100vh - var(--vp-nav-height));overflow-y:auto}.VPMenu[data-v-e42ed9b3] .group{margin:0 -12px;padding:0 12px 12px}.VPMenu[data-v-e42ed9b3] .group+.group{border-top:1px solid var(--vp-c-divider);padding:11px 12px 12px}.VPMenu[data-v-e42ed9b3] .group:last-child{padding-bottom:0}.VPMenu[data-v-e42ed9b3] .group+.item{border-top:1px solid var(--vp-c-divider);padding:11px 16px 0}.VPMenu[data-v-e42ed9b3] .item{padding:0 16px;white-space:nowrap}.VPMenu[data-v-e42ed9b3] .label{flex-grow:1;line-height:28px;font-size:12px;font-weight:500;color:var(--vp-c-text-2);transition:color .5s}.VPMenu[data-v-e42ed9b3] .action{padding-left:24px}.VPFlyout[data-v-aa8de344]{position:relative}.VPFlyout[data-v-aa8de344]:hover{color:var(--vp-c-brand-1);transition:color .25s}.VPFlyout:hover .text[data-v-aa8de344]{color:var(--vp-c-text-2)}.VPFlyout:hover .icon[data-v-aa8de344]{fill:var(--vp-c-text-2)}.VPFlyout.active .text[data-v-aa8de344]{color:var(--vp-c-brand-1)}.VPFlyout.active:hover .text[data-v-aa8de344]{color:var(--vp-c-brand-2)}.VPFlyout:hover .menu[data-v-aa8de344],.button[aria-expanded=true]+.menu[data-v-aa8de344]{opacity:1;visibility:visible;transform:translateY(0)}.button[aria-expanded=false]+.menu[data-v-aa8de344]{opacity:0;visibility:hidden;transform:translateY(0)}.button[data-v-aa8de344]{display:flex;align-items:center;padding:0 12px;height:var(--vp-nav-height);color:var(--vp-c-text-1);transition:color .5s}.text[data-v-aa8de344]{display:flex;align-items:center;line-height:var(--vp-nav-height);font-size:14px;font-weight:500;color:var(--vp-c-text-1);transition:color .25s}.option-icon[data-v-aa8de344]{margin-right:0;width:16px;height:16px;fill:currentColor}.text-icon[data-v-aa8de344]{margin-left:4px;width:14px;height:14px;fill:currentColor}.icon[data-v-aa8de344]{width:20px;height:20px;fill:currentColor;transition:fill .25s}.menu[data-v-aa8de344]{position:absolute;top:calc(var(--vp-nav-height) / 2 + 20px);right:0;opacity:0;visibility:hidden;transition:opacity .25s,visibility .25s,transform .25s}.VPSocialLink[data-v-16cf740a]{display:flex;justify-content:center;align-items:center;width:36px;height:36px;color:var(--vp-c-text-2);transition:color .5s}.VPSocialLink[data-v-16cf740a]:hover{color:var(--vp-c-text-1);transition:color .25s}.VPSocialLink[data-v-16cf740a]>svg{width:20px;height:20px;fill:currentColor}.VPSocialLinks[data-v-e71e869c]{display:flex;justify-content:center}.VPNavBarExtra[data-v-c8c2ae4b]{display:none;margin-right:-12px}@media (min-width: 768px){.VPNavBarExtra[data-v-c8c2ae4b]{display:block}}@media (min-width: 1280px){.VPNavBarExtra[data-v-c8c2ae4b]{display:none}}.trans-title[data-v-c8c2ae4b]{padding:0 24px 0 12px;line-height:32px;font-size:14px;font-weight:700;color:var(--vp-c-text-1)}.item.appearance[data-v-c8c2ae4b],.item.social-links[data-v-c8c2ae4b]{display:flex;align-items:center;padding:0 12px}.item.appearance[data-v-c8c2ae4b]{min-width:176px}.appearance-action[data-v-c8c2ae4b]{margin-right:-2px}.social-links-list[data-v-c8c2ae4b]{margin:-4px -8px}.VPNavBarHamburger[data-v-6bee1efd]{display:flex;justify-content:center;align-items:center;width:48px;height:var(--vp-nav-height)}@media (min-width: 768px){.VPNavBarHamburger[data-v-6bee1efd]{display:none}}.container[data-v-6bee1efd]{position:relative;width:16px;height:14px;overflow:hidden}.VPNavBarHamburger:hover .top[data-v-6bee1efd]{top:0;left:0;transform:translate(4px)}.VPNavBarHamburger:hover .middle[data-v-6bee1efd]{top:6px;left:0;transform:translate(0)}.VPNavBarHamburger:hover .bottom[data-v-6bee1efd]{top:12px;left:0;transform:translate(8px)}.VPNavBarHamburger.active .top[data-v-6bee1efd]{top:6px;transform:translate(0) rotate(225deg)}.VPNavBarHamburger.active .middle[data-v-6bee1efd]{top:6px;transform:translate(16px)}.VPNavBarHamburger.active .bottom[data-v-6bee1efd]{top:6px;transform:translate(0) rotate(135deg)}.VPNavBarHamburger.active:hover .top[data-v-6bee1efd],.VPNavBarHamburger.active:hover .middle[data-v-6bee1efd],.VPNavBarHamburger.active:hover .bottom[data-v-6bee1efd]{background-color:var(--vp-c-text-2);transition:top .25s,background-color .25s,transform .25s}.top[data-v-6bee1efd],.middle[data-v-6bee1efd],.bottom[data-v-6bee1efd]{position:absolute;width:16px;height:2px;background-color:var(--vp-c-text-1);transition:top .25s,background-color .5s,transform .25s}.top[data-v-6bee1efd]{top:0;left:0;transform:translate(0)}.middle[data-v-6bee1efd]{top:6px;left:0;transform:translate(8px)}.bottom[data-v-6bee1efd]{top:12px;left:0;transform:translate(4px)}.VPNavBarMenuLink[data-v-cb318fec]{display:flex;align-items:center;padding:0 12px;line-height:var(--vp-nav-height);font-size:14px;font-weight:500;color:var(--vp-c-text-1);transition:color .25s}.VPNavBarMenuLink.active[data-v-cb318fec],.VPNavBarMenuLink[data-v-cb318fec]:hover{color:var(--vp-c-brand-1)}.VPNavBarMenu[data-v-f732b5d0]{display:none}@media (min-width: 768px){.VPNavBarMenu[data-v-f732b5d0]{display:flex}}/*! @docsearch/css 3.5.2 | MIT License | © Algolia, Inc. and contributors | https://docsearch.algolia.com */:root{--docsearch-primary-color:#5468ff;--docsearch-text-color:#1c1e21;--docsearch-spacing:12px;--docsearch-icon-stroke-width:1.4;--docsearch-highlight-color:var(--docsearch-primary-color);--docsearch-muted-color:#969faf;--docsearch-container-background:rgba(101,108,133,.8);--docsearch-logo-color:#5468ff;--docsearch-modal-width:560px;--docsearch-modal-height:600px;--docsearch-modal-background:#f5f6f7;--docsearch-modal-shadow:inset 1px 1px 0 0 hsla(0,0%,100%,.5),0 3px 8px 0 #555a64;--docsearch-searchbox-height:56px;--docsearch-searchbox-background:#ebedf0;--docsearch-searchbox-focus-background:#fff;--docsearch-searchbox-shadow:inset 0 0 0 2px var(--docsearch-primary-color);--docsearch-hit-height:56px;--docsearch-hit-color:#444950;--docsearch-hit-active-color:#fff;--docsearch-hit-background:#fff;--docsearch-hit-shadow:0 1px 3px 0 #d4d9e1;--docsearch-key-gradient:linear-gradient(-225deg,#d5dbe4,#f8f8f8);--docsearch-key-shadow:inset 0 -2px 0 0 #cdcde6,inset 0 0 1px 1px #fff,0 1px 2px 1px rgba(30,35,90,.4);--docsearch-footer-height:44px;--docsearch-footer-background:#fff;--docsearch-footer-shadow:0 -1px 0 0 #e0e3e8,0 -3px 6px 0 rgba(69,98,155,.12)}html[data-theme=dark]{--docsearch-text-color:#f5f6f7;--docsearch-container-background:rgba(9,10,17,.8);--docsearch-modal-background:#15172a;--docsearch-modal-shadow:inset 1px 1px 0 0 #2c2e40,0 3px 8px 0 #000309;--docsearch-searchbox-background:#090a11;--docsearch-searchbox-focus-background:#000;--docsearch-hit-color:#bec3c9;--docsearch-hit-shadow:none;--docsearch-hit-background:#090a11;--docsearch-key-gradient:linear-gradient(-26.5deg,#565872,#31355b);--docsearch-key-shadow:inset 0 -2px 0 0 #282d55,inset 0 0 1px 1px #51577d,0 2px 2px 0 rgba(3,4,9,.3);--docsearch-footer-background:#1e2136;--docsearch-footer-shadow:inset 0 1px 0 0 rgba(73,76,106,.5),0 -4px 8px 0 rgba(0,0,0,.2);--docsearch-logo-color:#fff;--docsearch-muted-color:#7f8497}.DocSearch-Button{align-items:center;background:var(--docsearch-searchbox-background);border:0;border-radius:40px;color:var(--docsearch-muted-color);cursor:pointer;display:flex;font-weight:500;height:36px;justify-content:space-between;margin:0 0 0 16px;padding:0 8px;-webkit-user-select:none;user-select:none}.DocSearch-Button:active,.DocSearch-Button:focus,.DocSearch-Button:hover{background:var(--docsearch-searchbox-focus-background);box-shadow:var(--docsearch-searchbox-shadow);color:var(--docsearch-text-color);outline:none}.DocSearch-Button-Container{align-items:center;display:flex}.DocSearch-Search-Icon{stroke-width:1.6}.DocSearch-Button .DocSearch-Search-Icon{color:var(--docsearch-text-color)}.DocSearch-Button-Placeholder{font-size:1rem;padding:0 12px 0 6px}.DocSearch-Button-Keys{display:flex;min-width:calc(40px + .8em)}.DocSearch-Button-Key{align-items:center;background:var(--docsearch-key-gradient);border-radius:3px;box-shadow:var(--docsearch-key-shadow);color:var(--docsearch-muted-color);display:flex;height:18px;justify-content:center;margin-right:.4em;position:relative;padding:0 0 2px;border:0;top:-1px;width:20px}@media (max-width:768px){.DocSearch-Button-Keys,.DocSearch-Button-Placeholder{display:none}}.DocSearch--active{overflow:hidden!important}.DocSearch-Container,.DocSearch-Container *{box-sizing:border-box}.DocSearch-Container{background-color:var(--docsearch-container-background);height:100vh;left:0;position:fixed;top:0;width:100vw;z-index:200}.DocSearch-Container a{text-decoration:none}.DocSearch-Link{-webkit-appearance:none;-moz-appearance:none;appearance:none;background:none;border:0;color:var(--docsearch-highlight-color);cursor:pointer;font:inherit;margin:0;padding:0}.DocSearch-Modal{background:var(--docsearch-modal-background);border-radius:6px;box-shadow:var(--docsearch-modal-shadow);flex-direction:column;margin:60px auto auto;max-width:var(--docsearch-modal-width);position:relative}.DocSearch-SearchBar{display:flex;padding:var(--docsearch-spacing) var(--docsearch-spacing) 0}.DocSearch-Form{align-items:center;background:var(--docsearch-searchbox-focus-background);border-radius:4px;box-shadow:var(--docsearch-searchbox-shadow);display:flex;height:var(--docsearch-searchbox-height);margin:0;padding:0 var(--docsearch-spacing);position:relative;width:100%}.DocSearch-Input{-webkit-appearance:none;-moz-appearance:none;appearance:none;background:transparent;border:0;color:var(--docsearch-text-color);flex:1;font:inherit;font-size:1.2em;height:100%;outline:none;padding:0 0 0 8px;width:80%}.DocSearch-Input::placeholder{color:var(--docsearch-muted-color);opacity:1}.DocSearch-Input::-webkit-search-cancel-button,.DocSearch-Input::-webkit-search-decoration,.DocSearch-Input::-webkit-search-results-button,.DocSearch-Input::-webkit-search-results-decoration{display:none}.DocSearch-LoadingIndicator,.DocSearch-MagnifierLabel,.DocSearch-Reset{margin:0;padding:0}.DocSearch-MagnifierLabel,.DocSearch-Reset{align-items:center;color:var(--docsearch-highlight-color);display:flex;justify-content:center}.DocSearch-Container--Stalled .DocSearch-MagnifierLabel,.DocSearch-LoadingIndicator{display:none}.DocSearch-Container--Stalled .DocSearch-LoadingIndicator{align-items:center;color:var(--docsearch-highlight-color);display:flex;justify-content:center}@media screen and (prefers-reduced-motion:reduce){.DocSearch-Reset{animation:none;-webkit-appearance:none;-moz-appearance:none;appearance:none;background:none;border:0;border-radius:50%;color:var(--docsearch-icon-color);cursor:pointer;right:0;stroke-width:var(--docsearch-icon-stroke-width)}}.DocSearch-Reset{animation:fade-in .1s ease-in forwards;-webkit-appearance:none;-moz-appearance:none;appearance:none;background:none;border:0;border-radius:50%;color:var(--docsearch-icon-color);cursor:pointer;padding:2px;right:0;stroke-width:var(--docsearch-icon-stroke-width)}.DocSearch-Reset[hidden]{display:none}.DocSearch-Reset:hover{color:var(--docsearch-highlight-color)}.DocSearch-LoadingIndicator svg,.DocSearch-MagnifierLabel svg{height:24px;width:24px}.DocSearch-Cancel{display:none}.DocSearch-Dropdown{max-height:calc(var(--docsearch-modal-height) - var(--docsearch-searchbox-height) - var(--docsearch-spacing) - var(--docsearch-footer-height));min-height:var(--docsearch-spacing);overflow-y:auto;overflow-y:overlay;padding:0 var(--docsearch-spacing);scrollbar-color:var(--docsearch-muted-color) var(--docsearch-modal-background);scrollbar-width:thin}.DocSearch-Dropdown::-webkit-scrollbar{width:12px}.DocSearch-Dropdown::-webkit-scrollbar-track{background:transparent}.DocSearch-Dropdown::-webkit-scrollbar-thumb{background-color:var(--docsearch-muted-color);border:3px solid var(--docsearch-modal-background);border-radius:20px}.DocSearch-Dropdown ul{list-style:none;margin:0;padding:0}.DocSearch-Label{font-size:.75em;line-height:1.6em}.DocSearch-Help,.DocSearch-Label{color:var(--docsearch-muted-color)}.DocSearch-Help{font-size:.9em;margin:0;-webkit-user-select:none;user-select:none}.DocSearch-Title{font-size:1.2em}.DocSearch-Logo a{display:flex}.DocSearch-Logo svg{color:var(--docsearch-logo-color);margin-left:8px}.DocSearch-Hits:last-of-type{margin-bottom:24px}.DocSearch-Hits mark{background:none;color:var(--docsearch-highlight-color)}.DocSearch-HitsFooter{color:var(--docsearch-muted-color);display:flex;font-size:.85em;justify-content:center;margin-bottom:var(--docsearch-spacing);padding:var(--docsearch-spacing)}.DocSearch-HitsFooter a{border-bottom:1px solid;color:inherit}.DocSearch-Hit{border-radius:4px;display:flex;padding-bottom:4px;position:relative}@media screen and (prefers-reduced-motion:reduce){.DocSearch-Hit--deleting{transition:none}}.DocSearch-Hit--deleting{opacity:0;transition:all .25s linear}@media screen and (prefers-reduced-motion:reduce){.DocSearch-Hit--favoriting{transition:none}}.DocSearch-Hit--favoriting{transform:scale(0);transform-origin:top center;transition:all .25s linear;transition-delay:.25s}.DocSearch-Hit a{background:var(--docsearch-hit-background);border-radius:4px;box-shadow:var(--docsearch-hit-shadow);display:block;padding-left:var(--docsearch-spacing);width:100%}.DocSearch-Hit-source{background:var(--docsearch-modal-background);color:var(--docsearch-highlight-color);font-size:.85em;font-weight:600;line-height:32px;margin:0 -4px;padding:8px 4px 0;position:sticky;top:0;z-index:10}.DocSearch-Hit-Tree{color:var(--docsearch-muted-color);height:var(--docsearch-hit-height);opacity:.5;stroke-width:var(--docsearch-icon-stroke-width);width:24px}.DocSearch-Hit[aria-selected=true] a{background-color:var(--docsearch-highlight-color)}.DocSearch-Hit[aria-selected=true] mark{text-decoration:underline}.DocSearch-Hit-Container{align-items:center;color:var(--docsearch-hit-color);display:flex;flex-direction:row;height:var(--docsearch-hit-height);padding:0 var(--docsearch-spacing) 0 0}.DocSearch-Hit-icon{height:20px;width:20px}.DocSearch-Hit-action,.DocSearch-Hit-icon{color:var(--docsearch-muted-color);stroke-width:var(--docsearch-icon-stroke-width)}.DocSearch-Hit-action{align-items:center;display:flex;height:22px;width:22px}.DocSearch-Hit-action svg{display:block;height:18px;width:18px}.DocSearch-Hit-action+.DocSearch-Hit-action{margin-left:6px}.DocSearch-Hit-action-button{-webkit-appearance:none;-moz-appearance:none;appearance:none;background:none;border:0;border-radius:50%;color:inherit;cursor:pointer;padding:2px}svg.DocSearch-Hit-Select-Icon{display:none}.DocSearch-Hit[aria-selected=true] .DocSearch-Hit-Select-Icon{display:block}.DocSearch-Hit-action-button:focus,.DocSearch-Hit-action-button:hover{background:rgba(0,0,0,.2);transition:background-color .1s ease-in}@media screen and (prefers-reduced-motion:reduce){.DocSearch-Hit-action-button:focus,.DocSearch-Hit-action-button:hover{transition:none}}.DocSearch-Hit-action-button:focus path,.DocSearch-Hit-action-button:hover path{fill:#fff}.DocSearch-Hit-content-wrapper{display:flex;flex:1 1 auto;flex-direction:column;font-weight:500;justify-content:center;line-height:1.2em;margin:0 8px;overflow-x:hidden;position:relative;text-overflow:ellipsis;white-space:nowrap;width:80%}.DocSearch-Hit-title{font-size:.9em}.DocSearch-Hit-path{color:var(--docsearch-muted-color);font-size:.75em}.DocSearch-Hit[aria-selected=true] .DocSearch-Hit-action,.DocSearch-Hit[aria-selected=true] .DocSearch-Hit-icon,.DocSearch-Hit[aria-selected=true] .DocSearch-Hit-path,.DocSearch-Hit[aria-selected=true] .DocSearch-Hit-text,.DocSearch-Hit[aria-selected=true] .DocSearch-Hit-title,.DocSearch-Hit[aria-selected=true] .DocSearch-Hit-Tree,.DocSearch-Hit[aria-selected=true] mark{color:var(--docsearch-hit-active-color)!important}@media screen and (prefers-reduced-motion:reduce){.DocSearch-Hit-action-button:focus,.DocSearch-Hit-action-button:hover{background:rgba(0,0,0,.2);transition:none}}.DocSearch-ErrorScreen,.DocSearch-NoResults,.DocSearch-StartScreen{font-size:.9em;margin:0 auto;padding:36px 0;text-align:center;width:80%}.DocSearch-Screen-Icon{color:var(--docsearch-muted-color);padding-bottom:12px}.DocSearch-NoResults-Prefill-List{display:inline-block;padding-bottom:24px;text-align:left}.DocSearch-NoResults-Prefill-List ul{display:inline-block;padding:8px 0 0}.DocSearch-NoResults-Prefill-List li{list-style-position:inside;list-style-type:"» "}.DocSearch-Prefill{-webkit-appearance:none;-moz-appearance:none;appearance:none;background:none;border:0;border-radius:1em;color:var(--docsearch-highlight-color);cursor:pointer;display:inline-block;font-size:1em;font-weight:700;padding:0}.DocSearch-Prefill:focus,.DocSearch-Prefill:hover{outline:none;text-decoration:underline}.DocSearch-Footer{align-items:center;background:var(--docsearch-footer-background);border-radius:0 0 8px 8px;box-shadow:var(--docsearch-footer-shadow);display:flex;flex-direction:row-reverse;flex-shrink:0;height:var(--docsearch-footer-height);justify-content:space-between;padding:0 var(--docsearch-spacing);position:relative;-webkit-user-select:none;user-select:none;width:100%;z-index:300}.DocSearch-Commands{color:var(--docsearch-muted-color);display:flex;list-style:none;margin:0;padding:0}.DocSearch-Commands li{align-items:center;display:flex}.DocSearch-Commands li:not(:last-of-type){margin-right:.8em}.DocSearch-Commands-Key{align-items:center;background:var(--docsearch-key-gradient);border-radius:2px;box-shadow:var(--docsearch-key-shadow);display:flex;height:18px;justify-content:center;margin-right:.4em;padding:0 0 1px;color:var(--docsearch-muted-color);border:0;width:20px}@media (max-width:768px){:root{--docsearch-spacing:10px;--docsearch-footer-height:40px}.DocSearch-Dropdown{height:100%}.DocSearch-Container{height:100vh;height:-webkit-fill-available;height:calc(var(--docsearch-vh, 1vh)*100);position:absolute}.DocSearch-Footer{border-radius:0;bottom:0;position:absolute}.DocSearch-Hit-content-wrapper{display:flex;position:relative;width:80%}.DocSearch-Modal{border-radius:0;box-shadow:none;height:100vh;height:-webkit-fill-available;height:calc(var(--docsearch-vh, 1vh)*100);margin:0;max-width:100%;width:100%}.DocSearch-Dropdown{max-height:calc(var(--docsearch-vh, 1vh)*100 - var(--docsearch-searchbox-height) - var(--docsearch-spacing) - var(--docsearch-footer-height))}.DocSearch-Cancel{-webkit-appearance:none;-moz-appearance:none;appearance:none;background:none;border:0;color:var(--docsearch-highlight-color);cursor:pointer;display:inline-block;flex:none;font:inherit;font-size:1em;font-weight:500;margin-left:var(--docsearch-spacing);outline:none;overflow:hidden;padding:0;-webkit-user-select:none;user-select:none;white-space:nowrap}.DocSearch-Commands,.DocSearch-Hit-Tree{display:none}}@keyframes fade-in{0%{opacity:0}to{opacity:1}}[class*=DocSearch]{--docsearch-primary-color: var(--vp-c-brand-1);--docsearch-highlight-color: var(--docsearch-primary-color);--docsearch-text-color: var(--vp-c-text-1);--docsearch-muted-color: var(--vp-c-text-2);--docsearch-searchbox-shadow: none;--docsearch-searchbox-background: transparent;--docsearch-searchbox-focus-background: transparent;--docsearch-key-gradient: transparent;--docsearch-key-shadow: none;--docsearch-modal-background: var(--vp-c-bg-soft);--docsearch-footer-background: var(--vp-c-bg)}.dark [class*=DocSearch]{--docsearch-modal-shadow: none;--docsearch-footer-shadow: none;--docsearch-logo-color: var(--vp-c-text-2);--docsearch-hit-background: var(--vp-c-default-soft);--docsearch-hit-color: var(--vp-c-text-2);--docsearch-hit-shadow: none}.DocSearch-Button{display:flex;justify-content:center;align-items:center;margin:0;padding:0;width:48px;height:55px;background:transparent;transition:border-color .25s}.DocSearch-Button:hover{background:transparent}.DocSearch-Button:focus{outline:1px dotted;outline:5px auto -webkit-focus-ring-color}.DocSearch-Button:focus:not(:focus-visible){outline:none!important}@media (min-width: 768px){.DocSearch-Button{justify-content:flex-start;border:1px solid transparent;border-radius:8px;padding:0 10px 0 12px;width:100%;height:40px;background-color:var(--vp-c-bg-alt)}.DocSearch-Button:hover{border-color:var(--vp-c-brand-1);background:var(--vp-c-bg-alt)}}.DocSearch-Button .DocSearch-Button-Container{display:flex;align-items:center}.DocSearch-Button .DocSearch-Search-Icon{position:relative;width:16px;height:16px;color:var(--vp-c-text-1);fill:currentColor;transition:color .5s}.DocSearch-Button:hover .DocSearch-Search-Icon{color:var(--vp-c-text-1)}@media (min-width: 768px){.DocSearch-Button .DocSearch-Search-Icon{top:1px;margin-right:8px;width:14px;height:14px;color:var(--vp-c-text-2)}}.DocSearch-Button .DocSearch-Button-Placeholder{display:none;margin-top:2px;padding:0 16px 0 0;font-size:13px;font-weight:500;color:var(--vp-c-text-2);transition:color .5s}.DocSearch-Button:hover .DocSearch-Button-Placeholder{color:var(--vp-c-text-1)}@media (min-width: 768px){.DocSearch-Button .DocSearch-Button-Placeholder{display:inline-block}}.DocSearch-Button .DocSearch-Button-Keys{direction:ltr;display:none;min-width:auto}@media (min-width: 768px){.DocSearch-Button .DocSearch-Button-Keys{display:flex;align-items:center}}.DocSearch-Button .DocSearch-Button-Key{display:block;margin:2px 0 0;border:1px solid var(--vp-c-divider);border-right:none;border-radius:4px 0 0 4px;padding-left:6px;min-width:0;width:auto;height:22px;line-height:22px;font-family:var(--vp-font-family-base);font-size:12px;font-weight:500;transition:color .5s,border-color .5s}.DocSearch-Button .DocSearch-Button-Key+.DocSearch-Button-Key{border-right:1px solid var(--vp-c-divider);border-left:none;border-radius:0 4px 4px 0;padding-left:2px;padding-right:6px}.DocSearch-Button .DocSearch-Button-Key:first-child{font-size:0!important}.DocSearch-Button .DocSearch-Button-Key:first-child:after{content:"Ctrl";font-size:12px;letter-spacing:normal;color:var(--docsearch-muted-color)}.mac .DocSearch-Button .DocSearch-Button-Key:first-child:after{content:"⌘"}.DocSearch-Button .DocSearch-Button-Key:first-child>*{display:none}.VPNavBarSearch{display:flex;align-items:center}@media (min-width: 768px){.VPNavBarSearch{flex-grow:1;padding-left:24px}}@media (min-width: 960px){.VPNavBarSearch{padding-left:32px}}.dark .DocSearch-Footer{border-top:1px solid var(--vp-c-divider)}.DocSearch-Form{border:1px solid var(--vp-c-brand-1);background-color:var(--vp-c-white)}.dark .DocSearch-Form{background-color:var(--vp-c-default-soft)}.DocSearch-Screen-Icon>svg{margin:auto}.VPNavBarSocialLinks[data-v-ef6192dc]{display:none}@media (min-width: 1280px){.VPNavBarSocialLinks[data-v-ef6192dc]{display:flex;align-items:center}}.title[data-v-2973dbb4]{display:flex;align-items:center;border-bottom:1px solid transparent;width:100%;height:var(--vp-nav-height);font-size:16px;font-weight:600;color:var(--vp-c-text-1);transition:opacity .25s}@media (min-width: 960px){.title[data-v-2973dbb4]{flex-shrink:0}.VPNavBarTitle.has-sidebar .title[data-v-2973dbb4]{border-bottom-color:var(--vp-c-divider)}}[data-v-2973dbb4] .logo{margin-right:8px;height:var(--vp-nav-logo-height)}.VPNavBarTranslations[data-v-ff4524ae]{display:none}@media (min-width: 1280px){.VPNavBarTranslations[data-v-ff4524ae]{display:flex;align-items:center}}.title[data-v-ff4524ae]{padding:0 24px 0 12px;line-height:32px;font-size:14px;font-weight:700;color:var(--vp-c-text-1)}.VPNavBar[data-v-f1abbc6e]{position:relative;border-bottom:1px solid transparent;padding:0 8px 0 24px;height:var(--vp-nav-height);pointer-events:none;white-space:nowrap}@media (min-width: 768px){.VPNavBar[data-v-f1abbc6e]{padding:0 32px}}@media (min-width: 960px){.VPNavBar.has-sidebar[data-v-f1abbc6e]{padding:0}.VPNavBar[data-v-f1abbc6e]:not(.has-sidebar):not(.top){border-bottom-color:var(--vp-c-gutter);background-color:var(--vp-nav-bg-color)}}.container[data-v-f1abbc6e]{display:flex;justify-content:space-between;margin:0 auto;max-width:calc(var(--vp-layout-max-width) - 64px);height:var(--vp-nav-height);pointer-events:none}.container>.title[data-v-f1abbc6e],.container>.content[data-v-f1abbc6e]{pointer-events:none}.container[data-v-f1abbc6e] *{pointer-events:auto}@media (min-width: 960px){.VPNavBar.has-sidebar .container[data-v-f1abbc6e]{max-width:100%}}.title[data-v-f1abbc6e]{flex-shrink:0;height:calc(var(--vp-nav-height) - 1px);transition:background-color .5s}@media (min-width: 960px){.VPNavBar.has-sidebar .title[data-v-f1abbc6e]{position:absolute;top:0;left:0;z-index:2;padding:0 32px;width:var(--vp-sidebar-width);height:var(--vp-nav-height);background-color:transparent}}@media (min-width: 1440px){.VPNavBar.has-sidebar .title[data-v-f1abbc6e]{padding-left:max(32px,calc((100% - (var(--vp-layout-max-width) - 64px)) / 2));width:calc((100% - (var(--vp-layout-max-width) - 64px)) / 2 + var(--vp-sidebar-width) - 32px)}}.content[data-v-f1abbc6e]{flex-grow:1}@media (min-width: 960px){.VPNavBar.has-sidebar .content[data-v-f1abbc6e]{position:relative;z-index:1;padding-right:32px;padding-left:var(--vp-sidebar-width)}}@media (min-width: 1440px){.VPNavBar.has-sidebar .content[data-v-f1abbc6e]{padding-right:calc((100vw - var(--vp-layout-max-width)) / 2 + 32px);padding-left:calc((100vw - var(--vp-layout-max-width)) / 2 + var(--vp-sidebar-width))}}.content-body[data-v-f1abbc6e]{display:flex;justify-content:flex-end;align-items:center;height:calc(var(--vp-nav-height) - 1px);transition:background-color .5s}@media (min-width: 960px){.VPNavBar:not(.top) .content-body[data-v-f1abbc6e]{position:relative;background-color:var(--vp-nav-bg-color)}}@media (max-width: 767px){.content-body[data-v-f1abbc6e]{column-gap:.5rem}}.menu+.translations[data-v-f1abbc6e]:before,.menu+.appearance[data-v-f1abbc6e]:before,.menu+.social-links[data-v-f1abbc6e]:before,.translations+.appearance[data-v-f1abbc6e]:before,.appearance+.social-links[data-v-f1abbc6e]:before{margin-right:8px;margin-left:8px;width:1px;height:24px;background-color:var(--vp-c-divider);content:""}.menu+.appearance[data-v-f1abbc6e]:before,.translations+.appearance[data-v-f1abbc6e]:before{margin-right:16px}.appearance+.social-links[data-v-f1abbc6e]:before{margin-left:16px}.social-links[data-v-f1abbc6e]{margin-right:-8px}@media (min-width: 960px){.VPNavBar.has-sidebar .curtain[data-v-f1abbc6e]{position:absolute;right:0;bottom:-31px;width:calc(100% - var(--vp-sidebar-width));height:32px}.VPNavBar.has-sidebar .curtain[data-v-f1abbc6e]:before{display:block;width:100%;height:32px;background:linear-gradient(var(--vp-c-bg),transparent 70%);content:""}}@media (min-width: 1440px){.VPNavBar.has-sidebar .curtain[data-v-f1abbc6e]{width:calc(100% - ((100vw - var(--vp-layout-max-width)) / 2 + var(--vp-sidebar-width)))}}.VPNavScreenAppearance[data-v-0dc5cf49]{display:flex;justify-content:space-between;align-items:center;border-radius:8px;padding:12px 14px 12px 16px;background-color:var(--vp-c-bg-soft)}.text[data-v-0dc5cf49]{line-height:24px;font-size:12px;font-weight:500;color:var(--vp-c-text-2)}.VPNavScreenMenuLink[data-v-fe523e3d]{display:block;border-bottom:1px solid var(--vp-c-divider);padding:12px 0 11px;line-height:24px;font-size:14px;font-weight:500;color:var(--vp-c-text-1);transition:border-color .25s,color .25s}.VPNavScreenMenuLink[data-v-fe523e3d]:hover{color:var(--vp-c-brand-1)}.VPNavScreenMenuGroupLink[data-v-aea78dd1]{display:block;margin-left:12px;line-height:32px;font-size:14px;font-weight:400;color:var(--vp-c-text-1);transition:color .25s}.VPNavScreenMenuGroupLink[data-v-aea78dd1]:hover{color:var(--vp-c-brand-1)}.VPNavScreenMenuGroupSection[data-v-f60dbfa7]{display:block}.title[data-v-f60dbfa7]{line-height:32px;font-size:13px;font-weight:700;color:var(--vp-c-text-2);transition:color .25s}.VPNavScreenMenuGroup[data-v-c2c554ed]{border-bottom:1px solid var(--vp-c-divider);height:48px;overflow:hidden;transition:border-color .5s}.VPNavScreenMenuGroup .items[data-v-c2c554ed]{visibility:hidden}.VPNavScreenMenuGroup.open .items[data-v-c2c554ed]{visibility:visible}.VPNavScreenMenuGroup.open[data-v-c2c554ed]{padding-bottom:10px;height:auto}.VPNavScreenMenuGroup.open .button[data-v-c2c554ed]{padding-bottom:6px;color:var(--vp-c-brand-1)}.VPNavScreenMenuGroup.open .button-icon[data-v-c2c554ed]{transform:rotate(45deg)}.button[data-v-c2c554ed]{display:flex;justify-content:space-between;align-items:center;padding:12px 4px 11px 0;width:100%;line-height:24px;font-size:14px;font-weight:500;color:var(--vp-c-text-1);transition:color .25s}.button[data-v-c2c554ed]:hover{color:var(--vp-c-brand-1)}.button-icon[data-v-c2c554ed]{width:14px;height:14px;fill:var(--vp-c-text-2);transition:fill .5s,transform .25s}.group[data-v-c2c554ed]:first-child{padding-top:0}.group+.group[data-v-c2c554ed],.group+.item[data-v-c2c554ed]{padding-top:4px}.VPNavScreenTranslations[data-v-41505286]{height:24px;overflow:hidden}.VPNavScreenTranslations.open[data-v-41505286]{height:auto}.title[data-v-41505286]{display:flex;align-items:center;font-size:14px;font-weight:500;color:var(--vp-c-text-1)}.icon[data-v-41505286]{width:16px;height:16px;fill:currentColor}.icon.lang[data-v-41505286]{margin-right:8px}.icon.chevron[data-v-41505286]{margin-left:4px}.list[data-v-41505286]{padding:4px 0 0 24px}.link[data-v-41505286]{line-height:32px;font-size:13px;color:var(--vp-c-text-1)}.VPNavScreen[data-v-57cce842]{position:fixed;top:calc(var(--vp-nav-height) + var(--vp-layout-top-height, 0px) + 1px);right:0;bottom:0;left:0;padding:0 32px;width:100%;background-color:var(--vp-nav-screen-bg-color);overflow-y:auto;transition:background-color .5s;pointer-events:auto}.VPNavScreen.fade-enter-active[data-v-57cce842],.VPNavScreen.fade-leave-active[data-v-57cce842]{transition:opacity .25s}.VPNavScreen.fade-enter-active .container[data-v-57cce842],.VPNavScreen.fade-leave-active .container[data-v-57cce842]{transition:transform .25s ease}.VPNavScreen.fade-enter-from[data-v-57cce842],.VPNavScreen.fade-leave-to[data-v-57cce842]{opacity:0}.VPNavScreen.fade-enter-from .container[data-v-57cce842],.VPNavScreen.fade-leave-to .container[data-v-57cce842]{transform:translateY(-8px)}@media (min-width: 768px){.VPNavScreen[data-v-57cce842]{display:none}}.container[data-v-57cce842]{margin:0 auto;padding:24px 0 96px;max-width:288px}.menu+.translations[data-v-57cce842],.menu+.appearance[data-v-57cce842],.translations+.appearance[data-v-57cce842]{margin-top:24px}.menu+.social-links[data-v-57cce842]{margin-top:16px}.appearance+.social-links[data-v-57cce842]{margin-top:16px}.VPNav[data-v-7ad780c2]{position:relative;top:var(--vp-layout-top-height, 0px);left:0;z-index:var(--vp-z-index-nav);width:100%;pointer-events:none;transition:background-color .5s}@media (min-width: 960px){.VPNav[data-v-7ad780c2]{position:fixed}}.VPSidebarItem.level-0[data-v-bd01e0d5]{padding-bottom:24px}.VPSidebarItem.collapsed.level-0[data-v-bd01e0d5]{padding-bottom:10px}.item[data-v-bd01e0d5]{position:relative;display:flex;width:100%}.VPSidebarItem.collapsible>.item[data-v-bd01e0d5]{cursor:pointer}.indicator[data-v-bd01e0d5]{position:absolute;top:6px;bottom:6px;left:-17px;width:2px;border-radius:2px;transition:background-color .25s}.VPSidebarItem.level-2.is-active>.item>.indicator[data-v-bd01e0d5],.VPSidebarItem.level-3.is-active>.item>.indicator[data-v-bd01e0d5],.VPSidebarItem.level-4.is-active>.item>.indicator[data-v-bd01e0d5],.VPSidebarItem.level-5.is-active>.item>.indicator[data-v-bd01e0d5]{background-color:var(--vp-c-brand-1)}.link[data-v-bd01e0d5]{display:flex;align-items:center;flex-grow:1}.text[data-v-bd01e0d5]{flex-grow:1;padding:4px 0;line-height:24px;font-size:14px;transition:color .25s}.VPSidebarItem.level-0 .text[data-v-bd01e0d5]{font-weight:700;color:var(--vp-c-text-1)}.VPSidebarItem.level-1 .text[data-v-bd01e0d5],.VPSidebarItem.level-2 .text[data-v-bd01e0d5],.VPSidebarItem.level-3 .text[data-v-bd01e0d5],.VPSidebarItem.level-4 .text[data-v-bd01e0d5],.VPSidebarItem.level-5 .text[data-v-bd01e0d5]{font-weight:500;color:var(--vp-c-text-2)}.VPSidebarItem.level-0.is-link>.item>.link:hover .text[data-v-bd01e0d5],.VPSidebarItem.level-1.is-link>.item>.link:hover .text[data-v-bd01e0d5],.VPSidebarItem.level-2.is-link>.item>.link:hover .text[data-v-bd01e0d5],.VPSidebarItem.level-3.is-link>.item>.link:hover .text[data-v-bd01e0d5],.VPSidebarItem.level-4.is-link>.item>.link:hover .text[data-v-bd01e0d5],.VPSidebarItem.level-5.is-link>.item>.link:hover .text[data-v-bd01e0d5]{color:var(--vp-c-brand-1)}.VPSidebarItem.level-0.has-active>.item>.text[data-v-bd01e0d5],.VPSidebarItem.level-1.has-active>.item>.text[data-v-bd01e0d5],.VPSidebarItem.level-2.has-active>.item>.text[data-v-bd01e0d5],.VPSidebarItem.level-3.has-active>.item>.text[data-v-bd01e0d5],.VPSidebarItem.level-4.has-active>.item>.text[data-v-bd01e0d5],.VPSidebarItem.level-5.has-active>.item>.text[data-v-bd01e0d5],.VPSidebarItem.level-0.has-active>.item>.link>.text[data-v-bd01e0d5],.VPSidebarItem.level-1.has-active>.item>.link>.text[data-v-bd01e0d5],.VPSidebarItem.level-2.has-active>.item>.link>.text[data-v-bd01e0d5],.VPSidebarItem.level-3.has-active>.item>.link>.text[data-v-bd01e0d5],.VPSidebarItem.level-4.has-active>.item>.link>.text[data-v-bd01e0d5],.VPSidebarItem.level-5.has-active>.item>.link>.text[data-v-bd01e0d5]{color:var(--vp-c-text-1)}.VPSidebarItem.level-0.is-active>.item .link>.text[data-v-bd01e0d5],.VPSidebarItem.level-1.is-active>.item .link>.text[data-v-bd01e0d5],.VPSidebarItem.level-2.is-active>.item .link>.text[data-v-bd01e0d5],.VPSidebarItem.level-3.is-active>.item .link>.text[data-v-bd01e0d5],.VPSidebarItem.level-4.is-active>.item .link>.text[data-v-bd01e0d5],.VPSidebarItem.level-5.is-active>.item .link>.text[data-v-bd01e0d5]{color:var(--vp-c-brand-1)}.caret[data-v-bd01e0d5]{display:flex;justify-content:center;align-items:center;margin-right:-7px;width:32px;height:32px;color:var(--vp-c-text-3);cursor:pointer;transition:color .25s;flex-shrink:0}.item:hover .caret[data-v-bd01e0d5]{color:var(--vp-c-text-2)}.item:hover .caret[data-v-bd01e0d5]:hover{color:var(--vp-c-text-1)}.caret-icon[data-v-bd01e0d5]{width:18px;height:18px;fill:currentColor;transform:rotate(90deg);transition:transform .25s}.VPSidebarItem.collapsed .caret-icon[data-v-bd01e0d5]{transform:rotate(0)}.VPSidebarItem.level-1 .items[data-v-bd01e0d5],.VPSidebarItem.level-2 .items[data-v-bd01e0d5],.VPSidebarItem.level-3 .items[data-v-bd01e0d5],.VPSidebarItem.level-4 .items[data-v-bd01e0d5],.VPSidebarItem.level-5 .items[data-v-bd01e0d5]{border-left:1px solid var(--vp-c-divider);padding-left:16px}.VPSidebarItem.collapsed .items[data-v-bd01e0d5]{display:none}.VPSidebar[data-v-ee2efba5]{position:fixed;top:var(--vp-layout-top-height, 0px);bottom:0;left:0;z-index:var(--vp-z-index-sidebar);padding:32px 32px 96px;width:calc(100vw - 64px);max-width:320px;background-color:var(--vp-sidebar-bg-color);opacity:0;box-shadow:var(--vp-c-shadow-3);overflow-x:hidden;overflow-y:auto;transform:translate(-100%);transition:opacity .5s,transform .25s ease;overscroll-behavior:contain}.VPSidebar.open[data-v-ee2efba5]{opacity:1;visibility:visible;transform:translate(0);transition:opacity .25s,transform .5s cubic-bezier(.19,1,.22,1)}.dark .VPSidebar[data-v-ee2efba5]{box-shadow:var(--vp-shadow-1)}@media (min-width: 960px){.VPSidebar[data-v-ee2efba5]{z-index:1;padding-top:var(--vp-nav-height);padding-bottom:128px;width:var(--vp-sidebar-width);max-width:100%;background-color:var(--vp-sidebar-bg-color);opacity:1;visibility:visible;box-shadow:none;transform:translate(0)}}@media (min-width: 1440px){.VPSidebar[data-v-ee2efba5]{padding-left:max(32px,calc((100% - (var(--vp-layout-max-width) - 64px)) / 2));width:calc((100% - (var(--vp-layout-max-width) - 64px)) / 2 + var(--vp-sidebar-width) - 32px)}}@media (min-width: 960px){.curtain[data-v-ee2efba5]{position:sticky;top:-64px;left:0;z-index:1;margin-top:calc(var(--vp-nav-height) * -1);margin-right:-32px;margin-left:-32px;height:var(--vp-nav-height);background-color:var(--vp-sidebar-bg-color)}}.nav[data-v-ee2efba5]{outline:0}.group+.group[data-v-ee2efba5]{border-top:1px solid var(--vp-c-divider);padding-top:10px}@media (min-width: 960px){.group[data-v-ee2efba5]{padding-top:10px;width:calc(var(--vp-sidebar-width) - 64px)}}.VPSkipLink[data-v-c8291ffa]{top:8px;left:8px;padding:8px 16px;z-index:999;border-radius:8px;font-size:12px;font-weight:700;text-decoration:none;color:var(--vp-c-brand-1);box-shadow:var(--vp-shadow-3);background-color:var(--vp-c-bg)}.VPSkipLink[data-v-c8291ffa]:focus{height:auto;width:auto;clip:auto;clip-path:none}@media (min-width: 1280px){.VPSkipLink[data-v-c8291ffa]{top:14px;left:16px}}.Layout[data-v-9d8abc1e]{display:flex;flex-direction:column;min-height:100vh}.VPHomeSponsors[data-v-843cc1b2]{border-top:1px solid var(--vp-c-gutter);padding:88px 24px 96px;background-color:var(--vp-c-bg)}.container[data-v-843cc1b2]{margin:0 auto;max-width:1152px}.love[data-v-843cc1b2]{margin:0 auto;width:28px;height:28px;color:var(--vp-c-text-3)}.icon[data-v-843cc1b2]{width:28px;height:28px;fill:currentColor}.message[data-v-843cc1b2]{margin:0 auto;padding-top:10px;max-width:320px;text-align:center;line-height:24px;font-size:16px;font-weight:500;color:var(--vp-c-text-2)}.sponsors[data-v-843cc1b2]{padding-top:32px}.action[data-v-843cc1b2]{padding-top:40px;text-align:center}.VPTeamPage[data-v-b1cfd8dc]{padding-bottom:96px}@media (min-width: 768px){.VPTeamPage[data-v-b1cfd8dc]{padding-bottom:128px}}.VPTeamPageSection+.VPTeamPageSection[data-v-b1cfd8dc-s],.VPTeamMembers+.VPTeamPageSection[data-v-b1cfd8dc-s]{margin-top:64px}.VPTeamMembers+.VPTeamMembers[data-v-b1cfd8dc-s]{margin-top:24px}@media (min-width: 768px){.VPTeamPageTitle+.VPTeamPageSection[data-v-b1cfd8dc-s]{margin-top:16px}.VPTeamPageSection+.VPTeamPageSection[data-v-b1cfd8dc-s],.VPTeamMembers+.VPTeamPageSection[data-v-b1cfd8dc-s]{margin-top:96px}}.VPTeamMembers[data-v-b1cfd8dc-s]{padding:0 24px}@media (min-width: 768px){.VPTeamMembers[data-v-b1cfd8dc-s]{padding:0 48px}}@media (min-width: 960px){.VPTeamMembers[data-v-b1cfd8dc-s]{padding:0 64px}}.VPTeamPageTitle[data-v-46c5e327]{padding:48px 32px;text-align:center}@media (min-width: 768px){.VPTeamPageTitle[data-v-46c5e327]{padding:64px 48px 48px}}@media (min-width: 960px){.VPTeamPageTitle[data-v-46c5e327]{padding:80px 64px 48px}}.title[data-v-46c5e327]{letter-spacing:0;line-height:44px;font-size:36px;font-weight:500}@media (min-width: 768px){.title[data-v-46c5e327]{letter-spacing:-.5px;line-height:56px;font-size:48px}}.lead[data-v-46c5e327]{margin:0 auto;max-width:512px;padding-top:12px;line-height:24px;font-size:16px;font-weight:500;color:var(--vp-c-text-2)}@media (min-width: 768px){.lead[data-v-46c5e327]{max-width:592px;letter-spacing:.15px;line-height:28px;font-size:20px}}.VPTeamPageSection[data-v-3bf2e850]{padding:0 32px}@media (min-width: 768px){.VPTeamPageSection[data-v-3bf2e850]{padding:0 48px}}@media (min-width: 960px){.VPTeamPageSection[data-v-3bf2e850]{padding:0 64px}}.title[data-v-3bf2e850]{position:relative;margin:0 auto;max-width:1152px;text-align:center;color:var(--vp-c-text-2)}.title-line[data-v-3bf2e850]{position:absolute;top:16px;left:0;width:100%;height:1px;background-color:var(--vp-c-divider)}.title-text[data-v-3bf2e850]{position:relative;display:inline-block;padding:0 24px;letter-spacing:0;line-height:32px;font-size:20px;font-weight:500;background-color:var(--vp-c-bg)}.lead[data-v-3bf2e850]{margin:0 auto;max-width:480px;padding-top:12px;text-align:center;line-height:24px;font-size:16px;font-weight:500;color:var(--vp-c-text-2)}.members[data-v-3bf2e850]{padding-top:40px}.VPTeamMembersItem[data-v-3a0078bd]{display:flex;flex-direction:column;gap:2px;border-radius:12px;width:100%;height:100%;overflow:hidden}.VPTeamMembersItem.small .profile[data-v-3a0078bd]{padding:32px}.VPTeamMembersItem.small .data[data-v-3a0078bd]{padding-top:20px}.VPTeamMembersItem.small .avatar[data-v-3a0078bd]{width:64px;height:64px}.VPTeamMembersItem.small .name[data-v-3a0078bd]{line-height:24px;font-size:16px}.VPTeamMembersItem.small .affiliation[data-v-3a0078bd]{padding-top:4px;line-height:20px;font-size:14px}.VPTeamMembersItem.small .desc[data-v-3a0078bd]{padding-top:12px;line-height:20px;font-size:14px}.VPTeamMembersItem.small .links[data-v-3a0078bd]{margin:0 -16px -20px;padding:10px 0 0}.VPTeamMembersItem.medium .profile[data-v-3a0078bd]{padding:48px 32px}.VPTeamMembersItem.medium .data[data-v-3a0078bd]{padding-top:24px;text-align:center}.VPTeamMembersItem.medium .avatar[data-v-3a0078bd]{width:96px;height:96px}.VPTeamMembersItem.medium .name[data-v-3a0078bd]{letter-spacing:.15px;line-height:28px;font-size:20px}.VPTeamMembersItem.medium .affiliation[data-v-3a0078bd]{padding-top:4px;font-size:16px}.VPTeamMembersItem.medium .desc[data-v-3a0078bd]{padding-top:16px;max-width:288px;font-size:16px}.VPTeamMembersItem.medium .links[data-v-3a0078bd]{margin:0 -16px -12px;padding:16px 12px 0}.profile[data-v-3a0078bd]{flex-grow:1;background-color:var(--vp-c-bg-soft)}.data[data-v-3a0078bd]{text-align:center}.avatar[data-v-3a0078bd]{position:relative;flex-shrink:0;margin:0 auto;border-radius:50%;box-shadow:var(--vp-shadow-3)}.avatar-img[data-v-3a0078bd]{position:absolute;top:0;right:0;bottom:0;left:0;border-radius:50%;object-fit:cover}.name[data-v-3a0078bd]{margin:0;font-weight:600}.affiliation[data-v-3a0078bd]{margin:0;font-weight:500;color:var(--vp-c-text-2)}.org.link[data-v-3a0078bd]{color:var(--vp-c-text-2);transition:color .25s}.org.link[data-v-3a0078bd]:hover{color:var(--vp-c-brand-1)}.desc[data-v-3a0078bd]{margin:0 auto}.desc[data-v-3a0078bd] a{font-weight:500;color:var(--vp-c-brand-1);text-decoration-style:dotted;transition:color .25s}.links[data-v-3a0078bd]{display:flex;justify-content:center;height:56px}.sp-link[data-v-3a0078bd]{display:flex;justify-content:center;align-items:center;text-align:center;padding:16px;font-size:14px;font-weight:500;color:var(--vp-c-sponsor);background-color:var(--vp-c-bg-soft);transition:color .25s,background-color .25s}.sp .sp-link.link[data-v-3a0078bd]:hover,.sp .sp-link.link[data-v-3a0078bd]:focus{outline:none;color:var(--vp-c-white);background-color:var(--vp-c-sponsor)}.sp-icon[data-v-3a0078bd]{margin-right:8px;width:16px;height:16px;fill:currentColor}.VPTeamMembers.small .container[data-v-bf782009]{grid-template-columns:repeat(auto-fit,minmax(224px,1fr))}.VPTeamMembers.small.count-1 .container[data-v-bf782009]{max-width:276px}.VPTeamMembers.small.count-2 .container[data-v-bf782009]{max-width:576px}.VPTeamMembers.small.count-3 .container[data-v-bf782009]{max-width:876px}.VPTeamMembers.medium .container[data-v-bf782009]{grid-template-columns:repeat(auto-fit,minmax(256px,1fr))}@media (min-width: 375px){.VPTeamMembers.medium .container[data-v-bf782009]{grid-template-columns:repeat(auto-fit,minmax(288px,1fr))}}.VPTeamMembers.medium.count-1 .container[data-v-bf782009]{max-width:368px}.VPTeamMembers.medium.count-2 .container[data-v-bf782009]{max-width:760px}.container[data-v-bf782009]{display:grid;gap:24px;margin:0 auto;max-width:1152px}.url-image-button[data-v-21baf7e3]{display:flex;flex-direction:column;align-items:center;justify-content:center;gap:10px;padding:32px;font-size:24px;cursor:pointer;border:none;border-radius:8px;box-shadow:0 4px 12px #0000001a;transition:all .2s ease;background-color:var(--vp-c-brand-dimm);color:var(--text-color);width:200px;height:225px}.url-image-button[data-v-21baf7e3]:hover{box-shadow:0 6px 14px #0000001a;transform:translateY(-2px)}.button-icon[data-v-21baf7e3]{width:72px;height:72px}.button-text[data-v-21baf7e3]{font-weight:600;text-align:center}.notes-text[data-v-21baf7e3]{display:block;margin-top:10px;font-size:14px;text-align:center;color:var(--text-color-secondary)}.button-link[data-v-21baf7e3]{text-decoration:none}:root{--vp-c-brand: #7b2bf9;--vp-c-brand-light: #9350fa;--vp-c-brand-lighter: #a970fb;--vp-c-brand-lightest: #bf90fc;--vp-c-brand-dark: #6902e0;--vp-c-brand-darker: #5801c9;--vp-c-brand-dimm: rgba(123, 43, 249, .08)}:root{--vp-button-brand-border: var(--vp-c-brand-light);--vp-button-brand-text: var(--vp-c-white);--vp-button-brand-bg: var(--vp-c-brand);--vp-button-brand-hover-border: var(--vp-c-brand-light);--vp-button-brand-hover-text: var(--vp-c-white);--vp-button-brand-hover-bg: var(--vp-c-brand-light);--vp-button-brand-active-border: var(--vp-c-brand-light);--vp-button-brand-active-text: var(--vp-c-white);--vp-button-brand-active-bg: var(--vp-button-brand-bg)}:root{--vp-home-hero-name-color: transparent;--vp-home-hero-name-background: -webkit-linear-gradient( 120deg, #7b2bf9 30%, #fd63d9 );--vp-home-hero-image-background-image: linear-gradient( -45deg, #7b2bf9 50%, #fd63d9 50% );--vp-home-hero-image-filter: blur(40px)}@media (min-width: 640px){:root{--vp-home-hero-image-filter: blur(56px)}}@media (min-width: 960px){:root{--vp-home-hero-image-filter: blur(72px)}}:root{--vp-custom-block-tip-border: var(--vp-c-brand);--vp-custom-block-tip-text: var(--vp-c-brand-darker);--vp-custom-block-tip-bg: var(--vp-c-brand-dimm);--tab-text-color: #000}.dark{--vp-custom-block-tip-border: var(--vp-c-brand);--vp-custom-block-tip-text: var(--vp-c-brand-lightest);--vp-custom-block-tip-bg: var(--vp-c-brand-dimm);--tab-text-color: #fff}.DocSearch{--docsearch-primary-color: var(--vp-c-brand) !important}@font-face{font-family:RuberoidBold;src:url(/fonts/Ruberoid/Ruberoid-Bold.otf) format("opentype")}:root{--font-primary: "RuberoidBold", sans-serif}h1,h2,h3,h4,h5,h6{font-family:var(--font-primary)}.youtube-wrapper{position:relative;overflow:hidden;width:100%;padding-top:56.25%}.youtube-video{position:absolute;top:0;left:0;bottom:0;right:0;width:100%;height:100%;border:none}.VPLocalSearchBox[data-v-2813d7e3]{position:fixed;z-index:100;top:0;right:0;bottom:0;left:0;display:flex}.backdrop[data-v-2813d7e3]{position:absolute;top:0;right:0;bottom:0;left:0;background:var(--vp-backdrop-bg-color);transition:opacity .5s}.shell[data-v-2813d7e3]{position:relative;padding:12px;margin:64px auto;display:flex;flex-direction:column;gap:16px;background:var(--vp-local-search-bg);width:min(100vw - 60px,900px);height:min-content;max-height:min(100vh - 128px,900px);border-radius:6px}@media (max-width: 767px){.shell[data-v-2813d7e3]{margin:0;width:100vw;height:100vh;max-height:none;border-radius:0}}.search-bar[data-v-2813d7e3]{border:1px solid var(--vp-c-divider);border-radius:4px;display:flex;align-items:center;padding:0 12px;cursor:text}@media (max-width: 767px){.search-bar[data-v-2813d7e3]{padding:0 8px}}.search-bar[data-v-2813d7e3]:focus-within{border-color:var(--vp-c-brand-1)}.search-icon[data-v-2813d7e3]{margin:8px}@media (max-width: 767px){.search-icon[data-v-2813d7e3]{display:none}}.search-input[data-v-2813d7e3]{padding:6px 12px;font-size:inherit;width:100%}@media (max-width: 767px){.search-input[data-v-2813d7e3]{padding:6px 4px}}.search-actions[data-v-2813d7e3]{display:flex;gap:4px}@media (any-pointer: coarse){.search-actions[data-v-2813d7e3]{gap:8px}}@media (min-width: 769px){.search-actions.before[data-v-2813d7e3]{display:none}}.search-actions button[data-v-2813d7e3]{padding:8px}.search-actions button[data-v-2813d7e3]:not([disabled]):hover,.toggle-layout-button.detailed-list[data-v-2813d7e3]{color:var(--vp-c-brand-1)}.search-actions button.clear-button[data-v-2813d7e3]:disabled{opacity:.37}.search-keyboard-shortcuts[data-v-2813d7e3]{font-size:.8rem;opacity:75%;display:flex;flex-wrap:wrap;gap:16px;line-height:14px}.search-keyboard-shortcuts span[data-v-2813d7e3]{display:flex;align-items:center;gap:4px}@media (max-width: 767px){.search-keyboard-shortcuts[data-v-2813d7e3]{display:none}}.search-keyboard-shortcuts kbd[data-v-2813d7e3]{background:rgba(128,128,128,.1);border-radius:4px;padding:3px 6px;min-width:24px;display:inline-block;text-align:center;vertical-align:middle;border:1px solid rgba(128,128,128,.15);box-shadow:0 2px 2px #0000001a}.results[data-v-2813d7e3]{display:flex;flex-direction:column;gap:6px;overflow-x:hidden;overflow-y:auto;overscroll-behavior:contain}.result[data-v-2813d7e3]{display:flex;align-items:center;gap:8px;border-radius:4px;transition:none;line-height:1rem;border:solid 2px var(--vp-local-search-result-border);outline:none}.result>div[data-v-2813d7e3]{margin:12px;width:100%;overflow:hidden}@media (max-width: 767px){.result>div[data-v-2813d7e3]{margin:8px}}.titles[data-v-2813d7e3]{display:flex;flex-wrap:wrap;gap:4px;position:relative;z-index:1001;padding:2px 0}.title[data-v-2813d7e3]{display:flex;align-items:center;gap:4px}.title.main[data-v-2813d7e3]{font-weight:500}.title-icon[data-v-2813d7e3]{opacity:.5;font-weight:500;color:var(--vp-c-brand-1)}.title svg[data-v-2813d7e3]{opacity:.5}.result.selected[data-v-2813d7e3]{--vp-local-search-result-bg: var(--vp-local-search-result-selected-bg);border-color:var(--vp-local-search-result-selected-border)}.excerpt-wrapper[data-v-2813d7e3]{position:relative}.excerpt[data-v-2813d7e3]{opacity:75%;pointer-events:none;max-height:140px;overflow:hidden;position:relative;opacity:.5;margin-top:4px}.result.selected .excerpt[data-v-2813d7e3]{opacity:1}.excerpt[data-v-2813d7e3] *{font-size:.8rem!important;line-height:130%!important}.titles[data-v-2813d7e3] mark,.excerpt[data-v-2813d7e3] mark{background-color:var(--vp-local-search-highlight-bg);color:var(--vp-local-search-highlight-text);border-radius:2px;padding:0 2px}.excerpt[data-v-2813d7e3] .vp-code-group .tabs{display:none}.excerpt[data-v-2813d7e3] .vp-code-group div[class*=language-]{border-radius:8px!important}.excerpt-gradient-bottom[data-v-2813d7e3]{position:absolute;bottom:-1px;left:0;width:100%;height:8px;background:linear-gradient(transparent,var(--vp-local-search-result-bg));z-index:1000}.excerpt-gradient-top[data-v-2813d7e3]{position:absolute;top:-1px;left:0;width:100%;height:8px;background:linear-gradient(var(--vp-local-search-result-bg),transparent);z-index:1000}.result.selected .titles[data-v-2813d7e3],.result.selected .title-icon[data-v-2813d7e3]{color:var(--vp-c-brand-1)!important}.no-results[data-v-2813d7e3]{font-size:.9rem;text-align:center;padding:12px}svg[data-v-2813d7e3]{flex:none} diff --git a/audits/Blobstream_X-Informal_Systems_Audit.pdf b/audits/Blobstream_X-Informal_Systems_Audit.pdf new file mode 100644 index 00000000000..a0a2f1ad1b2 Binary files /dev/null and b/audits/Blobstream_X-Informal_Systems_Audit.pdf differ diff --git a/audits/Blobstream_X-OtterSec_Audit.pdf b/audits/Blobstream_X-OtterSec_Audit.pdf new file mode 100644 index 00000000000..5970bb049fb Binary files /dev/null and b/audits/Blobstream_X-OtterSec_Audit.pdf differ diff --git a/audits/Blobstream_X-Veridise_Audit.pdf b/audits/Blobstream_X-Veridise_Audit.pdf new file mode 100644 index 00000000000..aaa8ea62e78 Binary files /dev/null and b/audits/Blobstream_X-Veridise_Audit.pdf differ diff --git a/audits/Blobstream_X-Zellic_Audit.pdf b/audits/Blobstream_X-Zellic_Audit.pdf new file mode 100644 index 00000000000..badcfb2e1e1 Binary files /dev/null and b/audits/Blobstream_X-Zellic_Audit.pdf differ diff --git a/audits/Celestia_OP_Stack_Audit.pdf b/audits/Celestia_OP_Stack_Audit.pdf new file mode 100644 index 00000000000..a273ec66fe0 Binary files /dev/null and b/audits/Celestia_OP_Stack_Audit.pdf differ diff --git a/build/altlayer.webp b/build/altlayer.webp new file mode 100644 index 00000000000..9875b7faac2 Binary files /dev/null and b/build/altlayer.webp differ diff --git a/build/arbitrum.webp b/build/arbitrum.webp new file mode 100644 index 00000000000..ddf00d04acc Binary files /dev/null and b/build/arbitrum.webp differ diff --git a/build/astria.webp b/build/astria.webp new file mode 100644 index 00000000000..20014b1bfd9 Binary files /dev/null and b/build/astria.webp differ diff --git a/build/caldera.webp b/build/caldera.webp new file mode 100644 index 00000000000..fc945f50180 Binary files /dev/null and b/build/caldera.webp differ diff --git a/build/conduit.webp b/build/conduit.webp new file mode 100644 index 00000000000..d9af8a85f89 Binary files /dev/null and b/build/conduit.webp differ diff --git a/build/dymension.webp b/build/dymension.webp new file mode 100644 index 00000000000..3b1d74dd229 Binary files /dev/null and b/build/dymension.webp differ diff --git a/build/gateway.webp b/build/gateway.webp new file mode 100644 index 00000000000..6730c2d9a2f Binary files /dev/null and b/build/gateway.webp differ diff --git a/build/gelato.webp b/build/gelato.webp new file mode 100644 index 00000000000..3ea1a51803f Binary files /dev/null and b/build/gelato.webp differ diff --git a/build/karnot.webp b/build/karnot.webp new file mode 100644 index 00000000000..ea64cc49ac4 Binary files /dev/null and b/build/karnot.webp differ diff --git a/build/lumoz.webp b/build/lumoz.webp new file mode 100644 index 00000000000..6ef36f5ca76 Binary files /dev/null and b/build/lumoz.webp differ diff --git a/build/opstack.webp b/build/opstack.webp new file mode 100644 index 00000000000..e56db6736f4 Binary files /dev/null and b/build/opstack.webp differ diff --git a/build/polygon.webp b/build/polygon.webp new file mode 100644 index 00000000000..bde994133d4 Binary files /dev/null and b/build/polygon.webp differ diff --git a/build/rollkit.webp b/build/rollkit.webp new file mode 100644 index 00000000000..9c014367ae8 Binary files /dev/null and b/build/rollkit.webp differ diff --git a/build/snapchain.webp b/build/snapchain.webp new file mode 100644 index 00000000000..c24aaad496e Binary files /dev/null and b/build/snapchain.webp differ diff --git a/build/sovereign.webp b/build/sovereign.webp new file mode 100644 index 00000000000..c384bc495c7 Binary files /dev/null and b/build/sovereign.webp differ diff --git a/build/vistara.webp b/build/vistara.webp new file mode 100644 index 00000000000..1bf4cf25ed8 Binary files /dev/null and b/build/vistara.webp differ diff --git a/build/zeeve.webp b/build/zeeve.webp new file mode 100644 index 00000000000..af38ae27be3 Binary files /dev/null and b/build/zeeve.webp differ diff --git a/celestia-app.sh b/celestia-app.sh new file mode 100755 index 00000000000..1e669c7ee9c --- /dev/null +++ b/celestia-app.sh @@ -0,0 +1,160 @@ +#!/bin/bash + +# ASCII art +echo " __ __ _ " +echo " _______ / /__ ___ / /_(_)__ ________ ____ ___ " +echo "/ __/ -_) / -_|_-> "$LOGFILE" 2>&1; then + echo "Download failed. Exiting." | tee -a "$LOGFILE" + exit 1 +fi + +# Detect if running on macOS and use appropriate command for checksum +if [ "$OS" = "Darwin" ]; then + CALCULATED_CHECKSUM=$(shasum -a 256 "celestia-app_$PLATFORM.tar.gz" | awk '{print $1}') +else + CALCULATED_CHECKSUM=$(sha256sum "celestia-app_$PLATFORM.tar.gz" | awk '{print $1}') +fi + +# Download checksums.txt +if ! wget "https://github.com/celestiaorg/celestia-app/releases/download/$VERSION/checksums.txt" >> "$LOGFILE" 2>&1; then + echo "Failed to download checksums. Exiting." | tee -a "$LOGFILE" + exit 1 +fi + +# Find the expected checksum in checksums.txt +EXPECTED_CHECKSUM=$(grep "celestia-app_$PLATFORM.tar.gz" checksums.txt | awk '{print $1}') + +# Verify the checksum +if [ "$CALCULATED_CHECKSUM" != "$EXPECTED_CHECKSUM" ]; then + echo "Checksum verification failed. Expected: $EXPECTED_CHECKSUM, but got: $CALCULATED_CHECKSUM. Exiting." | tee -a "$LOGFILE" + exit 1 +else + echo "Checksum verification successful." | tee -a "$LOGFILE" +fi + +# Extract the tarball to the temporary directory +if ! tar -xzf "celestia-app_$PLATFORM.tar.gz" >> "$LOGFILE" 2>&1; then + echo "Extraction failed. Exiting." | tee -a "$LOGFILE" + exit 1 +fi + +# Log and print a message +echo "Binary extracted to: $TEMP_DIR" | tee -a "$LOGFILE" + +# Remove the tarball to clean up +rm "celestia-app_$PLATFORM.tar.gz" + +# Log and print a message +echo "Temporary files cleaned up." | tee -a "$LOGFILE" + +# Ask the user if they want to move the binary to /usr/local/bin +read -p "Do you want to move the binary to /usr/local/bin? This will require sudo access. (y/n) " -n 1 -r +echo # move to a new line +if [[ $REPLY =~ ^[Yy]$ ]] +then + sudo mv "$TEMP_DIR/celestia-appd" /usr/local/bin/ + echo "Binary moved to /usr/local/bin" | tee -a "$LOGFILE" + # Create a symbolic link in the temporary directory + ln -s /usr/local/bin/celestia-appd "$TEMP_DIR/celestia-appd" + echo "Symbolic link created in $TEMP_DIR" | tee -a "$LOGFILE" + echo "" + echo "You can now run celestia-appd from anywhere." | tee -a "$LOGFILE" + echo "" + echo "To check its version and see the menu, execute the following command:" | tee -a "$LOGFILE" + echo "" + echo "celestia-appd version && celestia-appd --help" | tee -a "$LOGFILE" +else + echo "" + echo "You can navigate to $TEMP_DIR to find and run celestia-appd." | tee -a "$LOGFILE" + echo "" + echo "To check its version and see the menu, execute the following commands:" | tee -a "$LOGFILE" + echo "" + echo "cd $TEMP_DIR" | tee -a "$LOGFILE" + echo "chmod +x celestia-appd" | tee -a "$LOGFILE" + echo "./celestia-appd version && ./celestia-appd --help" | tee -a "$LOGFILE" +fi \ No newline at end of file diff --git a/celestia-node.sh b/celestia-node.sh new file mode 100644 index 00000000000..336424cfdfe --- /dev/null +++ b/celestia-node.sh @@ -0,0 +1,159 @@ +#!/bin/bash + +# ASCII art +echo " _ _ _ _ " +echo " __ ___| |___ __| |_(_)__ _ ___ _ _ ___ __| |___ " +echo "/ _/ -_) / -_|_-< _| / _\` |___| ' \\/ _ \\/ _\` / -_)" +echo "\\__\\___|_\\___/__/\__|_\\__,_| |_||_\\___/\\__,_\\___|" +echo " " + +# Declare a log file and a temp directory +LOGFILE="$HOME/celestia-node-temp/logfile.log" +TEMP_DIR="$HOME/celestia-node-temp" + +# Check if the directory exists +if [ -d "$TEMP_DIR" ]; then + read -p "Directory $TEMP_DIR exists. Do you want to clear it out? (y/n) " -n 1 -r + echo # move to a new line + if [[ $REPLY =~ ^[Yy]$ ]] + then + rm -rf "$TEMP_DIR" + echo "Directory $TEMP_DIR has been removed." + fi +fi + +# Create a temporary directory to work from +mkdir -p "$TEMP_DIR" +touch "$LOGFILE" + +# Log and print the log file location +echo "Log file is located at: $LOGFILE" | tee -a "$LOGFILE" + +# Change to $TEMP_DIR and print a message +cd "$TEMP_DIR" || exit 1 +echo "Working from temporary directory: $TEMP_DIR" | tee -a "$LOGFILE" + +# Fetch the latest release tag from GitHub +VERSION=$(curl -s "https://api.github.com/repos/celestiaorg/celestia-node/releases/latest" | grep '"tag_name":' | sed -E 's/.*"([^"]+)".*/\1/') + +# Check if VERSION is empty +if [ -z "$VERSION" ]; then + echo "Failed to fetch the latest version. Exiting." | tee -a "$LOGFILE" + exit 1 +fi + +# Log and print a message +echo "Latest version detected: $VERSION" | tee -a "$LOGFILE" + +# Detect the operating system and architecture +OS=$(uname -s) +ARCH=$(uname -m) + +# Translate architecture to expected format +case $ARCH in + x86_64) + ARCH="x86_64" + ;; + aarch64|arm64) + ARCH="arm64" + ;; + *) + echo "Unsupported architecture: $ARCH. Exiting." | tee -a "$LOGFILE" + exit 1 + ;; +esac + +# Translate OS to expected format +case $OS in + Linux|Darwin) + ;; + *) + echo "Unsupported operating system: $OS. Exiting." | tee -a "$LOGFILE" + exit 1 + ;; +esac + +# Construct the download URL +PLATFORM="${OS}_${ARCH}" +URL="https://github.com/celestiaorg/celestia-node/releases/download/$VERSION/celestia-node_$PLATFORM.tar.gz" + +# Check if URL is valid +if [[ ! $URL =~ ^https://github.com/celestiaorg/celestia-node/releases/download/[^/]+/celestia-node_[^/]+.tar.gz$ ]]; then + echo "Invalid URL: $URL. Exiting." | tee -a "$LOGFILE" + exit 1 +fi + +# Log and print a message +echo "Downloading from: $URL" | tee -a "$LOGFILE" + +# Download the tarball +if ! wget "$URL" >> "$LOGFILE" 2>&1; then + echo "Download failed. Exiting." | tee -a "$LOGFILE" + exit 1 +fi + +# Detect if running on macOS and use appropriate command for checksum +if [ "$OS" = "Darwin" ]; then + CALCULATED_CHECKSUM=$(shasum -a 256 "celestia-node_$PLATFORM.tar.gz" | awk '{print $1}') +else + CALCULATED_CHECKSUM=$(sha256sum "celestia-node_$PLATFORM.tar.gz" | awk '{print $1}') +fi + +# Download checksums.txt +if ! wget "https://github.com/celestiaorg/celestia-node/releases/download/$VERSION/checksums.txt" >> "$LOGFILE" 2>&1; then + echo "Failed to download checksums. Exiting." | tee -a "$LOGFILE" + exit 1 +fi + +# Find the expected checksum in checksums.txt +EXPECTED_CHECKSUM=$(grep "celestia-node_$PLATFORM.tar.gz" checksums.txt | awk '{print $1}') + +# Verify the checksum +if [ "$CALCULATED_CHECKSUM" != "$EXPECTED_CHECKSUM" ]; then + echo "Checksum verification failed. Expected: $EXPECTED_CHECKSUM, but got: $CALCULATED_CHECKSUM. Exiting." | tee -a "$LOGFILE" + exit 1 +else + echo "Checksum verification successful." | tee -a "$LOGFILE" +fi + +# Extract the tarball to the temporary directory +if ! tar -xzf "celestia-node_$PLATFORM.tar.gz" >> "$LOGFILE" 2>&1; then + echo "Extraction failed. Exiting." | tee -a "$LOGFILE" + exit 1 +fi + +# Log and print a message +echo "Binary extracted to: $TEMP_DIR" | tee -a "$LOGFILE" + +# Remove the tarball to clean up +rm "celestia-node_$PLATFORM.tar.gz" + +# Log and print a message +echo "Temporary files cleaned up." | tee -a "$LOGFILE" + +# Ask the user if they want to move the binary to /usr/local/bin +read -p "Do you want to move the binary to /usr/local/bin? This will require sudo access. (y/n) " -n 1 -r +echo # move to a new line +if [[ $REPLY =~ ^[Yy]$ ]] +then + sudo mv "$TEMP_DIR/celestia" /usr/local/bin/ + echo "Binary moved to /usr/local/bin" | tee -a "$LOGFILE" + # Create a symbolic link in the temporary directory + ln -s /usr/local/bin/celestia "$TEMP_DIR/celestia" + echo "Symbolic link created in $TEMP_DIR" | tee -a "$LOGFILE" + echo "" + echo "You can now run celestia from anywhere." | tee -a "$LOGFILE" + echo "" + echo "To check its version and see the menu, execute the following command:" | tee -a "$LOGFILE" + echo "" + echo "celestia version && celestia --help" | tee -a "$LOGFILE" +else + echo "" + echo "You can navigate to $TEMP_DIR to find and run celestia." | tee -a "$LOGFILE" + echo "" + echo "To check its version and see the menu, execute the following commands:" | tee -a "$LOGFILE" + echo "" + echo "cd $TEMP_DIR" | tee -a "$LOGFILE" + echo "chmod +x celestia" | tee -a "$LOGFILE" + echo "./celestia version && ./celestia --help" | tee -a "$LOGFILE" +fi diff --git a/community/calendar.html b/community/calendar.html new file mode 100644 index 00000000000..94804f95459 --- /dev/null +++ b/community/calendar.html @@ -0,0 +1,45 @@ + + + + + + Community calendar | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Community calendar

    The Celestia community calendar is available for finding all the different community call events happening in Celestia's community.

    Add the community calendar to your personal calendar to stay updated with all the events.

    Explore past community call agendas, notes, and recordings for more insights.

    + + + + \ No newline at end of file diff --git a/community/coc.html b/community/coc.html new file mode 100644 index 00000000000..269d3020b10 --- /dev/null +++ b/community/coc.html @@ -0,0 +1,45 @@ + + + + + + Celestia.org Code of Conduct | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Celestia.org Code of Conduct

    Our Pledge

    We as Celestia.org members, contributors, and leaders pledge to make participation in our community a harassment-free experience for everyone, regardless of age, body size, visible or invisible disability, ethnicity, sex characteristics, gender identity and expression, level of experience, education, socio-economic status, nationality, personal appearance, race, caste, color, religion, or sexual identity and orientation.

    We pledge to act and interact in ways that contribute to an open, welcoming, diverse, inclusive, and healthy community.

    Our Standards

    Examples of behavior that contributes to a positive environment for our community include:

    • Demonstrating empathy and kindness toward other people
    • Being respectful of differing opinions, viewpoints, and experiences
    • Giving and gracefully accepting constructive feedback
    • Accepting responsibility and apologizing to those affected by our mistakes, and learning from the experience
    • Focusing on what is best not just for us as individuals, but for the overall community
    • Contributing to conversations about Celestia’s technology and ecosystem

    Examples of unacceptable behavior include:

    • The use of sexualized language or imagery, and sexual attention or advances of any kind
    • Trolling, insulting or derogatory comments, and personal or political attacks
    • Public or private harassment
    • Publishing others' private information, such as a physical or email address, without their explicit permission
    • Focusing on the prices of digital assets or tokens, or where they can be purchased
    • Other conduct which could reasonably be considered inappropriate in a professional setting

    Enforcement Responsibilities

    Community leaders are responsible for clarifying and enforcing our standards of acceptable behavior and will take appropriate and fair corrective action in response to any behavior that they deem inappropriate, threatening, offensive, or harmful.

    Community leaders have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct, and will communicate reasons for moderation decisions when appropriate.

    Scope

    This Code of Conduct applies within all community spaces, and also applies when an individual is officially representing the community in public spaces. Examples of representing our community include using an official e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event.

    Enforcement

    Instances of abusive, harassing, or otherwise unacceptable behavior may be reported to the community leaders responsible for enforcement at Celestia.org Discord. All complaints will be reviewed and investigated promptly and fairly.

    All community leaders are obligated to respect the privacy and security of the reporter of any incident.

    Enforcement Guidelines

    Community leaders will follow these Community Impact Guidelines in determining the consequences for any action they deem in violation of this Code of Conduct:

    1. Correction

    Community Impact: Use of inappropriate language or other behavior deemed unprofessional or unwelcome in the community.

    Consequence: A private, written warning from community leaders, providing clarity around the nature of the violation and an explanation of why the behavior was inappropriate. A public apology may be requested.

    2. Warning

    Community Impact: A violation through a single incident or series of actions.

    Consequence: A warning with consequences for continued behavior. No interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, for a specified period of time. This includes avoiding interactions in community spaces as well as external channels like social media. Violating these terms may lead to a temporary or permanent ban.

    3. Temporary Ban

    Community Impact: A serious violation of community standards, including sustained inappropriate behavior.

    Consequence: A temporary ban from any sort of interaction or public communication with the community for a specified period of time. No public or private interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, is allowed during this period. Violating these terms may lead to a permanent ban.

    4. Permanent Ban

    Community Impact: Demonstrating a pattern of violation of community standards, including sustained inappropriate behavior, harassment of an individual, or aggression toward or disparagement of classes of individuals.

    Consequence: A permanent ban from any sort of public interaction within the community.

    Attribution

    This Code of Conduct is adapted from the Contributor Covenant, version 2.1, available at https://www.contributor-covenant.org/version/2/1/code_of_conduct.html.

    Community Impact Guidelines were inspired by Mozilla's code of conduct enforcement ladder.

    For answers to common questions about this code of conduct, see the FAQ at https://www.contributor-covenant.org/faq. Translations are available at https://www.contributor-covenant.org/translations.

    + + + + \ No newline at end of file diff --git a/community/foundation-delegation-program.html b/community/foundation-delegation-program.html new file mode 100644 index 00000000000..4b5ea18e679 --- /dev/null +++ b/community/foundation-delegation-program.html @@ -0,0 +1,45 @@ + + + + + + The Celestia Foundation Delegation Program | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    The Celestia Foundation Delegation Program

    Delegation program banner

    Objectives of the program

    The primary objectives of the Celestia Foundation Delegation Program are:

    • To provide a fair opportunity for Celestia’s users to join the validator set, while ensuring the validator set remains proficient, trustworthy, and dependable.
    • To maintain network stability by promoting a steady transition of validators and avoiding sudden and disruptive changes in participation.
    • To enable the Celestia Foundation to use its stake towards its mission of fostering a modular blockchain network that delivers exceptional performance.

    Foundation delegation process

    Program launch

    Prospective validators are welcome to apply to the program starting February 6, 2024. The application is designed to assess a validator’s uptime performance and contributions to the Celestia ecosystem. Of the 100 total slots in Celestia’s active validator set, up to 50 will receive delegations within the program.

    Application submissions will be reviewed by the Celestia Foundation. More details about the application and eligibility criteria are described below.

    Cohort process

    cohort timeline

    Every 4 months, the Celestia Foundation will distribute a portion of the Foundation’s total available stake to a cohort of validators who meet certain criteria, detailed below. Here is an overview of how the cohort process will work for Cohort 1 and what that means for future cohorts.

    Key Points

    • Initial Cohort (Cohort 1): 50 applicants will be accepted
      • Grading System: Applicants in Cohort 1 are divided into first, second, and third place based eligibility criteria outlined in this document.
      • Delegation Duration: This varies based on the applicant’s placement in Cohort 1. First place receives 12 months of delegation, second place receives 8 months, third place receives 4 months.
    TierPlacementDelegation DurationRenewal By Cohort
    First PlaceApplicants 1-2012 monthsCohort 4
    Second PlaceApplicants 21-358 monthsCohort 3
    Third PlaceApplicants 36-504 monthsCohort 2
    • Subsequent Cohorts (Cohorts 2-onwards):
      • After Cohort 1, open slots may be filled by Cohort 1 members up for renewals or new applicants. There will be no Tiers (e.g. First Place, Second Place, Third Place) in cohorts after Cohort 1. This structure allows for a steady flow of both existing applicants and new applicants to maintain a stable set of participants in the program.

    During this period, so long as the validator maintains high uptime and does not violate the rules of the program, the validator will receive the delegation for the duration of the cohort they are currently in.

    Eligibility criteria

    The minimum requirements for participation in the program are as follows:

    • Run an active mainnet validator or an active Mocha testnet validator for at least 1 month before application deadline
    • Run a bridge node (on mainnet if you are already an active mainnet validator or on Mocha testnet if not) that is connected and reporting to the Celestia Labs OTEL collector (for new applicants - on testnet, so that we can evaluate performance)
    • Not jailed or slashed in the 6 months before application deadline
    • Not associated with an exchange or custodian
    • Not in the top 10 validators by delegation power, unless it enters the top 10 as a result of the Foundation’s delegation under this program
    • Have 10% or less commission
    • Not based within the US, within any country subject to economic sanctions, or within any other prohibited jurisdiction, and successfully complete a compliance screen
    • Dedicated email address so that the Foundation can reach you in the event of emergency upgrades and fixes

    Not adhering to any of the criteria above will automatically disqualify your application, and violating any of the criteria after you have received delegation will result in withdrawal of the delegation. A participant who loses stake due to being jailed by the protocol may reapply to the program after 2 cohort periods.

    Applicants are also expected to have reviewed Celestia docs and recommended guides on devops and monitoring setups.

    Other optional but important criteria:

    • Develop and maintain developer tooling, services, applications, and dashboards
    • Work on projects aligned with Celestia's values
    • Contribute to documentation and new guides and tutorials
    • Quality of infrastructure
    • Operated within a location that improves geolocation of the validator set

    Undelegation criteria

    • Getting slashed/tombstoned (cannot apply for 1 year afterwards)
    • Getting jailed more than once during the cohort’s applicable delegation period
    • Violating the Celestia.org Community Code of Conduct or engaging in harmful activities towards the network
    • Failing to upgrade your node in a timely manner (24 hours or less)
    • If necessary to protect or secure Mainnet Beta or to comply with applicable law
    • For any other reason, in the Celestia Foundation’s sole discretion

    Application

    The program will be divided into cohorts with applications open for new applicants and renewal of existing applicants every 4 months. Validators will be delegated for up to a year. For each cohort, the deadline to apply/be evaluated (if you are reapplying) is exactly 1 month prior to the date of being delegated to.

    Application details

    Before applying, be ready to share the following:

    • General info
      • Security Email
      • Validator Entity Name
      • Discord ID
      • Mark if entity or individual
      • Website if any
      • Github page of your organization
      • Team experience and roster (including Twitter + Github links)
      • Which networks you validate on mainnet + links to your validators
      • A personal statement why you should receive delegation from the Foundation (max 1500 characters)
    • Infrastructure
      • Validator address and bridge node ID on MAINNET
      • If you don't run an active mainnet validator, please provide us with validator address, bridge node ID and blobstream address on Mocha-4
      • Have you been slashed or jailed in the last 6 months on Celestia or other chains you validated on.
      • Hosting provider and Data Center location (mainnet and testnet if applicable)
      • Setup of the 2 components (validator and bridge)
        • Hardware
        • Security setup (servers, private keys)
        • Monitoring and alerting
    • Contributions
      • Please list all technical contributions for Celestia and its ecosystem
      • Please list all community contributions for Celestia and its ecosystem

    Please note, the objective of the program is to contribute to Celestia’s resilience and uptime. If you contribute a lot to the Celestia ecosystem, but your validator uptime is low, this will negatively impact your chance at selection for the program. Furthermore, merely receiving delegation from the Foundation under the program does not guarantee your placement in the active validator set.

    Get Started with the Application Form

    Cohort information

    The Foundation will report each cohort’s composition and the duration of their respective delegations.

    Feedback process

    Validators in the program will receive a feedback form every quarter so the program can be continually improved.

    + + + + \ No newline at end of file diff --git a/community/itn-tos.html b/community/itn-tos.html new file mode 100644 index 00000000000..644fbc6d242 --- /dev/null +++ b/community/itn-tos.html @@ -0,0 +1,45 @@ + + + + + + SUPPLEMENTAL INCENTIVIZED TESTNET TERMS | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    SUPPLEMENTAL INCENTIVIZED TESTNET TERMS

    Last Revised on 1/16/2023

    Welcome to the Supplemental Terms (these "ITN Award Program Terms" or "Terms") for the Incentivized Testnet Award Program (the "ITN Award Program" or the "Program") as operated on behalf of Strange Loop Labs AG ("Company", "we" or "us"). The ITN Award Program provides eligible users of a Testnet designated by the Company the opportunity to earn rewards, which may include Celestia tokens. These Terms are supplemental to, and incorporate by reference, the broader Celestia Terms of Service ("Services Terms") available at Celestia Terms of Service.

    Defined terms used but not defined herein have the meaning set forth in the Services Terms. The Program and your participation in it is a Service as defined under the Services Terms.

    These Terms govern your ability to participate in the Program and any awards you receive from that participation, which may include Celestia tokens ("ITN Rewards").

    Please read these Terms carefully, as they include important information about your legal rights. By participating in the Program or claiming ITN Rewards, you are agreeing to these Terms. If you do not understand or agree to these Terms, please do not participate in the Program or claim ITN Rewards.

    In order to participate in the Program you must provide certain information about you. Our collection of such information, your rights with respect to such collection, and other relevant information is described in the Celestia Privacy Policy available at Celestia Privacy Policy, and is supplemented by Section 3 of these Terms.

    The Program is a discretionary Service provided by the Company, pursuant to which the Company may, in its sole discretion, provide you ITN Rewards for your successful completion of certain tasks on a Testnet designated by the Company. Please note that any such Testnet itself (as well as any other Testnets or any mainnet deployment of the Celestia Protocol) is not a Service and does not constitute an element of the Services. We do not control the Celestia Protocol and accept no liability for its operation or its deployment in any testnet or mainnet environment.

    1. General Terms

    1.1 You must be eighteen (18) years of age or older and capable of forming a binding contract with the Company in order to participate in the Program or receive ITN Rewards.

    1.2 You agree and acknowledge that you (a) may receive ITN Rewards for free (other than applicable taxes, if any) from your participation in the Program, (b) were not previously promised ITN Rewards, unless pursuant to a separate written agreement, and (c) took no action in anticipation of or in reliance on receiving any ITN Rewards, unless pursuant to a separate written agreement.

    1.3 Your eligibility to participate in or receive ITN Rewards from the Program is subject to our sole discretion. The complete list of actions you must complete to earn ITN Rewards may not have been described in the documentation released by us from time to time, you may not receive ITN Rewards even if you successfully complete such actions, and no documentation related to the Program entitles you to any ITN Rewards or to participate in the Program.

    1.4 You agree and acknowledge that (a) you are not a Prohibited Person, (b) you are not a U.S. Person as defined in Rule 902(k) of Regulation S under the U.S. Securities Act of 1933, as amended (the "1933 Act" or "Act"), (c) you will not use a VPN or other tool to circumvent any geoblock or other restrictions that we may have implemented for participants in the Program, and (d) you are not participating in, and have not become eligible to participate in, the Program by receiving credentials from any other person or entity. Any circumvention or violation of the above will permanently disqualify you from participation in the Program.

    1.5 You agree and acknowledge that if you are unable to claim ITN Rewards due to technical bugs, gas fees, loss of access to a Wallet or the keys thereto, or for any other reason, you will have no recourse or claim against us or any other Company Entity and that neither we nor any other Company Entity will bear any liability.

    1.6 You agree and acknowledge that claiming an ITN Reward may require reliance on or an integration with third party products (e.g., a Wallet or an unaffiliated network or blockchain) that we do not control. In the event that you are unable to access such products or integrations, or if they fail for any reason, and you are unable to participate in the Program or claim ITN Rewards, you will have no recourse or claim against us or any other Company Entity and neither we nor any other Company Entity will bear any liability.

    1.7 The Company may share identifying information and documentation with certain vendors or third-party providers who provide such identity verification and sanctions and watchlist screening services (the "Third-Party Services"). You agree that your access and use of such Third-Party Services is governed solely by the terms and conditions of such Third-Party Services, and the Company is not responsible or liable for, and make no representations as to any aspect of such Third-Party Services, including, without limitation, their content or the manner in which they handle, protect, manage or process data or any interaction between you and the provider of such Third-Party Services. You irrevocably waive any claim against the Company with respect to such Third-Party Services. We are not liable for any damage or loss caused or alleged to be caused by or in connection with your enablement, access or use of any such Third-Party Services, or your reliance on the privacy practices, data security processes or other policies of such Third-Party Services.

    2. Taxes

    2.1 You are responsible for the payment of all taxes associated with your participation in the Program and your receipt of ITN Rewards. You agree to provide the Company with any additional information and complete any required tax or other forms relating to your receipt of ITN Rewards. You may suffer adverse tax consequences as a result of your participation in the Program or your receipt of ITN Rewards. You hereby represent that (a) you have consulted with a tax adviser that you deem advisable in connection with your participation, or that you have had the opportunity to obtain tax advice but have chosen not to do so, (b) the Company has not provided you with any tax advice with respect to your participation, and (c) you are not relying on the Company for any tax advice.

    3. Supplemental Privacy Information

    We may collect information to help us determine the reliability or uptime of your activities within the Program, including through the use of telemetry or metrics endpoints to collect and analyse such information, and link this information to a unique identifier to represent your activities within the Program. We may display all of the foregoing information on a public dashboard.

    Additionally, we may collect certain information about you from Third-Party Services and may combine information we receive from you with information we obtain from Third-Party Services, including but not limited to:

    • Transaction information. Information related to transactions in your Wallet, your Wallet address, activities performed using your Wallet, tokens received by your Wallet, or transactions initiated or completed.
    • Identification information. We collect your government identification (e.g., driver’s license, passport, etc.), proof of address, biometric information, and entity formation information if applicable. By agreeing to these Terms, you consent to our use of your biometric information, and understand and agree that our use of the biometric information is necessary for the performance of these Terms and the implementation of the Services.

    We collect this information to confirm your eligibility to participate in the Program and receive ITN Rewards, comply with our legal obligations, detect and prevent fraud, and to provide you with the Program.

    Any information we receive from third-party sources will be treated in accordance with the Celestia Privacy Policy, available at Celestia Privacy Policy. We are not responsible or liable for the accuracy of the information provided to us by third parties and are not responsible for any third party’s policies or practices. See Section 9 of the Celestia Privacy Policy for more information.

    4. Certain Additional Representations

    4.1 Receipt of Rewards Entirely for Own Account. Your eligibility to receive ITN Rewards is made in reliance upon your representation to the Company, which by your agreement to these Terms you hereby confirm, that any ITN Rewards you receive will be for your own account, not as a nominee or agent, and not with a view to the resale or distribution of any part thereof, and that you have no present intention of selling, granting any participation in, or otherwise distributing the same. By agreeing to these Terms, you further represent that you do not presently have any contract, undertaking, agreement or arrangement with any person to sell, transfer or grant participations to such person or to any third person, with respect to any ITN Rewards. If you are agreeing to these terms on behalf of an entity, that entity has not been formed for the specific purpose of obtaining the ITN Rewards.

    4.2 Disclosure of Information. Your eligibility to receive ITN Rewards is made in reliance upon your representation to the Company, which by your agreement to these Terms you hereby confirm, that you have sufficient knowledge of and experience in business and financial matters to be able to evaluate the risks and merits of your participation in the Program and of any ITN Rewards and are able to bear the risks thereof. You hereby affirm that you have not relied on any representations or warranties made by the Company related to the Program, including, but not limited to, conversations of any kind, whether through oral or electronic communication, or any white paper.

    4.3 Compliance with United States Securities Laws. You understand that the ITN Rewards have not been, and will not be, registered under the 1933 Act or any applicable state securities laws. You acknowledge that the availability of an exemption from the registration provisions of the Securities Act and other applicable state securities laws depends upon, among other things, the bona fide nature of your intent as described in Section 4.1 above and with respect to the accuracy of your representations as expressed throughout these Terms. You understand that the ITN Rewards may be deemed "restricted securities" under applicable United States federal and state securities laws and that, pursuant to these laws, you may be restricted from transferring any ITN Rewards unless they are registered with the Securities and Exchange Commission and qualified by state authorities, or an exemption from such registration and qualification requirements is available. You acknowledge that the Company does not undertake any obligation to register or qualify the ITN Rewards for resale, and exemptions from registration and qualification may not be available or may not permit you to transfer all or any of the ITN Rewards in the amounts or at the times proposed by you. You further acknowledge that if an exemption from registration or qualification is available, such exemption may be conditioned on various requirements including, but not limited to, the time and manner of sale, the holding period for the ITN Rewards, and on other factors outside of your control, for which the Company makes no assurances and may not be able to satisfy.

    4.4 Compliance with Liechtenstein Security Law. You understand that nothing in these Terms will be deemed to constitute a prospectus of any sort in Liechtenstein or in any jurisdiction in the EU; nor does it in any way pertain to a public offering or a solicitation of an offer to buy any securities in Liechtenstein or in any jurisdiction in the EU.

    4.5 No Public Market. You understand that no public market now exists for the ITN Rewards, and that the Company has not made any assurances that a public market will ever exist for the ITN Rewards.

    4.6 No Solicitation. At no time were you presented with or solicited by any publicly issued or circulated newspaper, mail, radio, television or other form of general advertising or solicitation in connection with any invitation to participate in the Program or offer of the ITN Rewards.

    4.7 Other Applicable Laws. You hereby represent that you have satisfied yourself as to the full observance of the laws of your jurisdiction in connection with any invitation to participate in the Program, receipt of ITN Awards, and other use of these Terms, including (a) the legal requirements within your jurisdiction for participating in the Program and receiving ITN Rewards, (b) any foreign exchange restrictions applicable to such participation or receipt, (c) any governmental or other consents that may need to be obtained, and (d) the income tax and other tax consequences, if any, that may be relevant to the receipt, holding, sale, or transfer of the ITN Rewards. Your participation in the Program and continued beneficial ownership of ITN Rewards will not violate any applicable securities or other laws of your jurisdiction.

    4.8 Non-US Transaction. You are not a U.S. Person as defined in Rule 902(k) of Regulation S under the 1933 Act. The offer of the ITN Rewards to you was made in an offshore transaction (as defined in Rule 902(h) of Regulation S), no directed selling efforts (as defined in Rule 902(c) of Regulation S) were made in the United States, and you are not obtaining the ITN Rewards for the account or benefit of any U.S. Person.

    4.9 Transfer Restrictions. You will not, during the Restricted Period (as defined below) offer or sell any of the ITN Rewards (or create or maintain any derivative position equivalent thereto) in the United States, to or for the account or benefit of a U.S. Person or other than in accordance with Regulation S. The Company reserves the right to impose additional transfer restrictions with respect to the ITN Rewards in its sole discretion.

    4.10 Subsequent Sales. You will, after the expiration of the applicable Restricted Period, only offer, sell, pledge or otherwise transfer the ITN Rewards (or create or maintain any derivative position equivalent thereto) pursuant to registration under the 1933 Act or any available exemption therefrom and, in any case, in accordance with applicable state securities laws.

    4.11 Legends. You acknowledge and agree that the ITN Rewards will be deemed to bear the following legends: (a) any legend required by the securities laws of any state or country to the extent such laws are applicable to the ITN Rewards represented by the certificate so legended, and (b): the following legend (and even without such legend the following restrictions apply):

    THE ITN REWARDS HAVE NOT BEEN REGISTERED UNDER THE ACT WITH THE UNITED STATES SECURITIES AND EXCHANGE COMMISSION, AND THE COMPANY DOES NOT INTEND TO REGISTER THEM. THE ITN REWARDS HAVE BEEN OBTAINED TO HOLD FOR THE LONG TERM AND NOT WITH A VIEW TO, OR IN CONNECTION WITH, THE SALE OR DISTRIBUTION THEREFOR. PRIOR TO THE ONE YEAR ANNIVERSARY FROM THE TERMINATION OF THE ITN REWARD PROGRAM (THE "PROGRAM COMPLETION DATE" AND SUCH ONE YEAR PERIOD, THE "RESTRICTED PERIOD"), THE ITN REWARDS MAY NOT BE OFFERED OR SOLD (INCLUDING OPENING A SHORT POSITION IN SUCH ITN REWARDS) IN THE UNITED STATES OR TO U.S. PERSONS AS DEFINED BY RULE 902(k) ADOPTED UNDER THE ACT, OTHER THAN TO DISTRIBUTORS, UNLESS THE ITN REWARDS ARE REGISTERED UNDER THE ACT, OR AN EXEMPTION FROM THE REGISTRATION REQUIREMENTS OF THE ACT IS AVAILABLE. RECIPIENTS OF ITN REWARDS PRIOR TO THE ONE YEAR ANNIVERSARY OF THE PROGRAM COMPLETION DATE MAY SELL SUCH ITN REWARDS ONLY PURSUANT TO AN EXEMPTION FROM REGISTRATION UNDER THE ACT OR OTHERWISE IN ACCORDANCE WITH THE PROVISIONS OF REGULATION S OF THE ACT, OR IN TRANSACTIONS EFFECTED OUTSIDE OF THE UNITED STATES PROVIDED THEY DO NOT SOLICIT (AND NO ONE ACTING ON THEIR BEHALF SOLICITS) PURCHASERS IN THE UNITED STATES OR OTHERWISE ENGAGE(S) IN SELLING EFFORTS IN THE UNITED STATES AND PROVIDED THAT HEDGING TRANSACTIONS INVOLVING THESE ITN REWARDS MAY NOT BE CONDUCTED UNLESS IN COMPLIANCE WITH THE ACT. A HOLDER OF THE ITN REWARDS WHO IS A DISTRIBUTOR, DEALER, SUB-UNDERWRITER OR OTHER SECURITIES PROFESSIONAL, IN ADDITION, CANNOT PRIOR TO THE ONE YEAR ANNIVERSARY OF THE PROGRAM COMPLETION DATE SELL THE ITN REWARDS TO A U.S. PERSON AS DEFINED BY RULE 902(k) OF REGULATION S UNLESS THE ITN REWARDS ARE REGISTERED UNDER THE ACT OR AN EXEMPTION FROM REGISTRATION UNDER THE ACT IS AVAILABLE.

    + + + + \ No newline at end of file diff --git a/community/modular-meetup-guide.html b/community/modular-meetup-guide.html new file mode 100644 index 00000000000..19cd5a6b1ee --- /dev/null +++ b/community/modular-meetup-guide.html @@ -0,0 +1,45 @@ + + + + + + Modular Meetup guide | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Modular Meetup guide

    These are recommended steps you can follow that can help you organize a successful Modular Meetup. You aren’t required to follow the entire checklist but it is available to help aid you in your journey to organize a Modular Meetup.

    Before the Meetup

    1. Determine the meetup topic and objectives.
    2. Select a suitable date and time for the event.
    3. Secure a convenient and accessible venue that can accommodate the expected number of attendees.
    4. Finalize a schedule or agenda for the event, including speakers, presentations, and activities.
    5. Prepare and send out invitations to potential attendees using various channels (email, social media, community platforms, etc.).
    6. Create and share promotional materials (graphics, blog posts, etc.) to raise awareness about the meetup.
    7. Confirm speakers, including their availability, presentation topics, and technical requirements. If you are working from a Speaker List, please reach out to Nat for any support needed on coordination with speakers.
    8. Arrange any necessary equipment, such as microphones, projectors, and whiteboards.
    9. Plan and arrange refreshments, if applicable.
    10. Prepare and print any required materials, like agendas, name tags, and feedback forms.
    11. Coordinate with volunteers or team members to help manage the event.
    12. Set up a registration system or RSVP process to track attendance.
    13. Send out reminder messages to registered attendees prior to the event.
    14. Conduct a final review of the event logistics, including venue setup, equipment functionality, and volunteer roles.

    During the Meetup

    1. Set up the venue, including arranging seating, preparing equipment, and displaying any promotional materials.
    2. Designate a registration area and ensure that someone is available to greet and check-in attendees.
    3. Welcome the attendees and provide an overview of the event schedule.
    4. Introduce speakers and facilitate any presentations or discussions.
    5. Encourage networking and interaction among attendees during breaks or dedicated activities.
    6. Capture the event with photos or videos for future promotions and documentation.
    7. Collect feedback from attendees using forms, online surveys, or informal conversations if needed.

    After the Meetup

    1. Send follow-up messages to attendees, thanking them for their participation and soliciting additional feedback.
    2. Evaluate the success of the event by reviewing feedback and assessing key performance indicators (e.g., attendance, engagement, etc.).
    3. Analyze and document lessons learned, identifying areas for improvement in future meetups.
    4. Share event highlights, photos, and key takeaways with the community through social media, blog posts, or newsletters.
    5. Connect with speakers, attendees, and volunteers to maintain and strengthen relationships.
    6. Begin planning for the next meetup, applying insights gained from the previous event to enhance the experience for future attendees.

    Logistics and guidance

    These are provided guidelines for different components of kickstarting a Modular Meetup to help you get started on the logistics of organizing a meetup. A lot of those guidelines are thanks to the wonderful write ups in Ethereum Meetup Support Program and Elastic Community Organizer Guides.

    Venue

    Selecting the perfect venue for your Modular Meetup is essential in creating a welcoming atmosphere for the Celestia community.

    1. Startup Incubators
      • Often they would have spaces for meetups.
    2. Libraries
      • Libraries can normally have spaces for meetups at little to no costs
    3. Co-working spaces:
      • Co-working spaces might be able to offer necessary equipment like microphones, projectors, and whiteboards.
    4. Restaurants:
      • Restaurants might have private rooms for larger groups of people that you can book, depending on the restaurant.
    5. Universities and Blockchain Clubs:
      • Partnering with universities and their local university clubs can offer you a lot of spaces for meetups, as well as necessary equipments if needed for giving a talk.
    6. Other options:
      • If you're having difficulty finding a venue, don't hesitate to ask for help within the Celestia community. Your fellow organizers and attendees may have valuable suggestions or connections.

    An ideal venue should offer:

    • Sufficient seating for the audience
    • A projector and screen
    • A microphone (especially for meetups with 10+ attendees)
    • Optional: A whiteboard for speakers or workshops (not always necessary, but a nice addition)

    Consider recording the event, even with a smartphone, to share with the community later. If the speaker uses a microphone, their voice will be more audible in the video. We might be able to post those recordings of the meetup talks on a meetup youtube channel after.

    Remember to negotiate on costs and seek discounts, emphasizing that your meetup benefits the community.

    Collaborate with your venue provider to explore options like borrowing recording equipment, tripods, or even having them record the event for you. Some providers may offer these services for free or at a reduced cost.

    Catering and refreshments

    Providing refreshments or catering for your Modular Meetup enhances the overall experience and encourages networking among attendees.

    1. Determine your budget
      • Identify the amount you can allocate for food and drinks at your event which will help you determine how much you can bring in refreshments. You can also contact the Celestia Devrel team for ideas and support. Costs for refreshments and drinks can vary depending on your location, so be mindful to be flexible on your plans according to your specific location and budget.
    2. Offer a variety of refreshments
      • If you're ordering in, pizza and finger foods work well, but you can also have more budget-friendly options for food.
      • Offering drinks like beer, soda, or lemonade are great, but water also works. Keep in mind that not everyone drinks alcohol so it’s not a requirement. But having at least water and plastic cups works well.
    3. Plan ahead
      • Order refreshments 1 or 2 days in advance and schedule delivery to avoid last-minute stress during the event.

    By following these recommendations, you'll be able to provide enjoyable refreshments for your Celestia Modular Meetup attendees while fostering a friendly and engaging atmosphere.

    Audience

    Understanding your audience and estimating attendance are crucial for organizing successful Modular Meetups for Celestia.

    1. Research the local tech scene:
      • Investigate the types of meetups and events popular in your area. Attend other technology-focused events to get a sense of the audience size and interests. This information will help you tailor your meetup to attract a larger audience.
    2. Assess the availability of speakers:
      • Before organizing a meetup, ensure that you have access to a pool of knowledgeable speakers. If you anticipate difficulty in securing speakers, consider joining forces with an existing meetup group or speaking at other events before launching a new group. This approach will help spread the word about your planned Modular Meetup and gauge interest. Speakers are covered in the following section with a reference to Speaker List offered by the Modular Meetup program.
    3. Establish connections with other user groups:
      • Forge relationships with other tech-focused meetup groups to mutually promote each other's events, potentially increasing attendance.
    4. Set a regular routine for your meetup:
      • Communicate how often you plan to hold meetups, whether it's monthly or quarterly, to help attendees manage their expectations and maintain their interest.
    5. Organize casual meetups:
      • If there's a gap between more formal events, arrange casual meetups at pubs or cafes to keep people engaged and connect with potential speakers for future events.
    6. Estimate the number of attendees:
      • Consider factors such as the size of your city, the popularity of the topic, and the appeal of the event description when estimating attendance.

    Speakers

    Securing engaging and knowledgeable speakers is key to hosting an exceptional Celestia Modular Meetup.

    1. Define your event topic
      • Determine the theme of your event, focusing on areas such as DeFi, Gaming, NFTs, coding workshops, protocol changes, rollups, data availability, or other topics relevant to the Modular ecosystem. This will help you find speakers with expertise in the chosen subject.
    2. Utilize the Speaker List provided by the Modular Meetup program
      • As a meetup organizer, you have access to a curated list of talented speakers from Celestia Labs and the broader Modular ecosystem. This valuable resource can connect you with experts who can share their knowledge with your meetup attendees.
    3. Aim for multiple speakers
      • Ideally, invite 2-3 speakers to your event, allotting 20-40 minutes per talk. Schedule short breaks between presentations to maintain audience engagement.
    4. Organize the speaker lineup
      • Discuss the topics and slides with your speakers before the event to ensure a smooth flow. Arrange the talks in a logical order, saving the most impactful presentation for last.
    5. Invite local speakers
      • For your first meetup, consider presenting an introduction to the Celestia ecosystem or a specific area of interest. Encourage local experts or enthusiasts to speak at future meetups. This approach fosters community involvement and helps build a network of potential speakers.
    6. Seek speaker referrals
      • Ask your current speakers, attendees, or other meetup organizers for referrals. Personal connections often lead to discovering new speakers with valuable insights.
    7. Leverage your meetup discussion board
      • Post a call for speakers on your meetup discussion board to reach out to potential presenters within your community. Be clear about the event theme and requirements to attract relevant speakers.
    8. Offer incentives and appreciation
      • Reward speakers with tokens of gratitude, such as T-shirts, gifts, or public recognition, to show your appreciation for their contribution to the meetup. The Celestia Labs Devrel team can help with swag logistics if needed.

    Sponsors

    Finding sponsors for your meetup can be challenging, but securing financial support is crucial for covering costs related to venue and refreshments.

    1. Leverage Celestia Labs' support
      • Celestia Labs may offer co-sponsorship for your meetup. However, they also encourage organizers to find local co-sponsors to help cover costs and create a more sustainable event.
    2. Offer value to your sponsors:
      • Show potential sponsors how partnering with your event will benefit them. Include their logo on event banners, mention them in the event description, and give them a shoutout at the beginning of the event. If they desire, allow them to place a rollup banner at the venue.
    3. Reach out to your network
      • Ask friends and acquaintances if they know of coworking spaces, schools, universities, or companies interested in sponsorship. A personal connection can significantly increase the chances of securing support.
    4. Approach speakers for sponsorship
      • Request speakers or their affiliated projects to contribute towards the event's expenses, such as catering costs. This can be an effective way to obtain additional funding.
    5. Create a sponsorship deck
      • Develop a compelling sponsorship deck to pitch your event to tech companies in your city or potential online sponsors. This presentation should highlight the benefits of supporting your event and showcase past successful meetups.
    6. Research local companies
      • Investigate businesses in your area that may be interested in sponsoring your event. Tailor your pitch to align with their industry and demonstrate how the meetup can benefit their company.
    7. Engage sponsors during the event
      • Allow sponsors to briefly address the audience, participate in Q&A sessions, or mention their hiring needs. Remember to avoid sales pitches, as they can negatively impact the meetup experience.
    8. Thank your sponsors
      • Express gratitude to your sponsors at the beginning and end of the meetup. Acknowledging their support encourages continued collaboration and enhances the credibility of your event.

    Communications and marketing

    Effectively marketing and announcing your Modular Meetup is essential for attracting attendees and ensuring a successful event.

    1. Plan your announcement
      • Announce your meetup at least two weeks prior. This gives your audience ample time to prepare and increases the likelihood of their attendance.
    2. Utilize Celestia Labs' resources
      • Celestia Labs can help co-promote your meetup on social media and Discord. They can also add your event to their online calendar, email local contacts, and share the event via their Developer Relations Team on Twitter. Don't hesitate to reach out to them for assistance.
    3. Share on social media
      • Promote your event on Twitter and any other popular social media platforms in your area. Tag speakers, sponsors, and use relevant hashtags to increase visibility.
    4. Leverage local community groups
      • Post your event in local Telegram, Discord, or other community groups relevant to your city. These groups often have many members who may be interested in attending your event.
    5. Engage speakers and the venue
      • Maintain communication with speakers and the venue to ensure any changes can be announced in advance. Trust is critical, so avoid canceling planned meetups whenever possible.
    6. Send reminders via meetup.com (optional)
      • After announcing your event, send a warm invite to your meetup group members through meetup.com. Additionally, send a reminder email one day before the event to encourage attendance.
    7. Share with friends and family
      • Invite your friends, family, and acquaintances to the meetup, as they may help spread the word or know someone interested in the event.
    8. Utilize conference groups
      • If you've attended conferences, share your meetup in the associated Telegram or social media groups, as there may be members nearby who would be interested in attending.

    Recording

    Recording and live-streaming your Modular Meetup can greatly benefit those who cannot attend in person and expand the reach of your event.

    1. Plan for recording
      • Consider recording your meetup to create additional learning resources and share the knowledge with a broader audience.
    2. Consider live-streaming
      • Live-streaming your event on platforms like Twitch, YouTube, or Twitter allows remote participants to watch and engage in the meetup. This can also boost your event's reach and create a sense of inclusion for those who couldn't attend in person.
    3. Coordinate with Celestia Labs
      • Celestia Labs can potentially help cover recording costs and promote recorded meetups. Reach out to them at meetups@celestia.org to discuss available options and support. They can also assist with finding local contacts for recording if you don't have one already.
    4. Utilize available resources
      • If you have a small budget, consider allocating some funds for recording and live-streaming your event. Look for local professionals or affordable equipment rentals to ensure high-quality recordings.
    5. Share recordings on Celestia Labs' channels
      • Celestia Labs can help promote recorded meetups by sharing videos in their meetups playlist on YouTube and hosting them on their website. Ensure you coordinate with Celestia Labs to provide them with the recorded video.
    6. Promote recorded content
      • Share the recordings on your social media channels and meetup group after the event. This helps attendees revisit the content and allows those who couldn't attend to learn from the talks.

    Utilizing Meetup.com platform

    Celestia Labs is committed to supporting your meetup efforts by helping you with meetup.com, from setting up the group to covering organizer dues. Below is a detailed overview of how Celestia Labs can assist you.

    1. Meetup.com organizer dues
      • Celestia Labs is happy to cover organizer dues for meetup.com. To get started, send an email to meetups@celestia.org and let them help you with the process.
    2. Co-organizing existing groups
      • If you already have a meetup.com group but need Celestia Labs' assistance with organizing or covering dues, email meetups@celestia.org to add one of their employees as the Organizer.
    3. Setting up groups on Meetup.com
      • Celestia Labs will help you set up your meetup group, including logos, naming, custom URLs, group description, and other essential details. They provide a unique logo for Celestia Modular Meetups, which you can use for your group.
    4. User group naming
      • Celestia Labs recommends naming your group "Celestia Modular Meetup" to encompass various aspects of the Celestia community and create a consistent brand.
    5. Custom URL for the group
      • Meetup.com allows you to create a custom URL for your user group. Celestia Labs encourages consistent URLs across regions, making it easy for people to find your meetup group.
    6. Group description
      • Celestia Labs has a standard group description to ensure consistency across all meetups. However, if you'd like to customize it, let them know.
    7. New member intake questions
      • To improve the quality of your meetups, Celestia Labs suggests a set of intake questions for new members. The answers can help you better understand their needs and interests, allowing you to plan engaging meetups.
    8. Welcome message
      • Celestia Labs provides a welcome message for new members joining your group. If you'd like to customize this message, let them know.

    Onboarding questions for community members joining a Modular Meetup

    1. Q1: How did you hear about this Modular Meetup?
      • Why: This information helps us understand the most effective channels for recruiting new participants to the meetup.
    2. Q2: What do you hope to gain by participating in this meetup? (e.g., networking, learning about Celestia, learning about Modularity, finding a job, etc.)
      • Why: This information helps us cater to the needs of the user group members. If most attendees are looking for networking opportunities, we can schedule casual meetups alongside informative talks.
    3. Q3: Are you currently using or planning to use Celestia, Rollkit, Celestia’s Node API or any of the rollups deployed on Celestia? Tell us all about it. We will use your response to help us better understand what talks would be most beneficial to the group.
      • Why: Responses to this question help us determine the most relevant talks for the group members. If we discover that most participants are interested in a specific area, we can tailor the talks accordingly.
    4. Q4: We love Celestia and the modular ecosystem, but we also appreciate other Web3 technologies. What other topics would you like to hear about? (e.g., Infrastructure, Data Storage, DID, MEV)
      • Why: Knowing our group participants' interests in other technical areas helps us understand what related topics would be useful and valuable to the group, which in turn helps when recruiting speakers.
    5. Q5: Would you be interested in speaking at a future meetup? We welcome 2-minute lightning talks to 1-hour deep dives. Would you be interested in hosting a meetup? If you answer yes to this question, the group organizers will contact you to follow up.
      • Why: One of the challenges in hosting regular meetups is finding speakers. We hope this question will identify people eager to share their stories and expertise with the group, making the lives of organizers easier.
    + + + + \ No newline at end of file diff --git a/community/modular-meetup-intro.html b/community/modular-meetup-intro.html new file mode 100644 index 00000000000..0497fe76bb6 --- /dev/null +++ b/community/modular-meetup-intro.html @@ -0,0 +1,45 @@ + + + + + + Celestia Modular Meetup program | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Celestia Modular Meetup program

    Modular Meetup Banner

    Welcome to the ultimate guide for Modular Meetup organizers! This collection of resources is designed for those enthusiastic about fostering grassroots Modular Meetups with support from Celestia around the world.

    Program description

    The Celestia Modular Meetup Program aims to empower meetup organizers, providing education and support, and encouraging collaboration within the Web3 ecosystem. This rapidly growing community has already achieved incredible success with the first Modular Meetup in Lisbon, and will grow from there.

    Join fellow enthusiasts, engage in enlightening discussions, and make the most of the insightful resources provided. These resources are designed to serve as a go-to playbook for meetup organizers, especially when starting your journey.

    Important info

    Celestia.org Community Code of Conduct

    The purpose of our Community Code of Conduct is to foster an inclusive, welcoming, and supportive environment for everyone participating in Celestia community events. We're all here to learn from each other, expand our skillsets, and enjoy a positive experience together.

    All meetup attendees, speakers, sponsors, and volunteers, including the event organizing team, are kindly asked to adhere to the following Code of Conduct. Organizers will respectfully enforce this code throughout the event. We genuinely appreciate the cooperation of all participants in maintaining a safe and empowering space for everyone.

    Signup form

    To become part of the program, please complete the registration form.

    Following the review and approval of your submission, you will receive an email confirmation and an invitation to participate in the upcoming Modular Meetup call. Furthermore, you will be granted access to the exclusive Discord channel labeled "#modular-meetup" on our Discord server. Please take note that joining our Discord is a prerequisite for channel access. It's essential to recognize that this program is tailored for dedicated organizers with a genuine interest in nurturing their local modular ecosystem community.

    Emails

    As a participant in the Celestia Modular Meetup Program, you can expect to receive the following emails:

    1. Welcome email with links to calendar events and Discord channel
    2. Monthly Catch-up call invites
    3. Recap emails with notes from calls

    Discord

    Your active participation is key to unlocking the full potential of this vibrant community. Our primary communication tool is Discord, providing an engaging platform to connect with fellow organizers:

    Materials

    As a meetup organizer, you'll gain access to the Celestia Modular Meetup Program's list of resources. This collection should become your trusted companion in organizing events. Drawing upon the wisdom of seasoned event organizers, this resource is available for you and your co-organizers to explore and learn.

    + + + + \ No newline at end of file diff --git a/community/modular-meetup-toolkit.html b/community/modular-meetup-toolkit.html new file mode 100644 index 00000000000..735a7ab743c --- /dev/null +++ b/community/modular-meetup-toolkit.html @@ -0,0 +1,45 @@ + + + + + + Modular Meetup Toolkit | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Modular Meetup Toolkit

    Welcome to the Modular Meetups Toolkit! This kit is designed to help you run successful meetups for the Celestia community. As the first modular blockchain, Celestia offers a lot to discuss and explore. This kit includes useful resources and materials to help you plan and execute your meetups effectively.

    Celestia branding guidelines

    • Brand kit
      1. Includes logo files, color schemes, typography, icons and illustrations

    Sample “Introduction to Modularity” workshop presentation

    • Sample presentation - introduction to modularity
    • Summary: This is an overview presentation on Modular blockchains and dives deep into Celestia core technologies.
    • The sample presentation covers:
      1. What are modular blockchains?
      2. The benefits of modular over monolithic blockchains
      3. Introduction to Celestia: The first modular blockchain
      4. The concept of Data Availability Sampling
      5. Sovereign Rollups
      6. Q&A session

    Sample “Run a Celestia light node” workshop presentation

    • Sample presentation - run a light node
    • Summary: This is an overview presentation goes over running a Celestia light node. You can find existing video presentations for this here:
    • The sample presentation covers:
      1. What is a Celestia light node?
      2. The role of light nodes in the Celestia ecosystem
      3. Setting up a light node: hardware and software requirements
      4. Step-by-step guide on how to run a Celestia light node
      5. Troubleshooting common issues
      6. Best practices for maintaining a light node
      7. Q&A session

    Sample “Deploy a Sovereign Rollup” workshop presentation

    • Sample presentation - deploy a sovereign rollup
    • Summary: This is an overview presentation on deploying a sovereign rollup with Rollkit on Celestia. You can find existing video presentations for this here:
    • The sample presentation covers:
      1. What is a sovereign rollup?
      2. The role of sovereign rollups in the Celestia ecosystem
      3. Introduction to Rollkit
      4. Setting up a sovereign rollup: hardware and software requirements
      5. Q&A session

    Sample “Modular Meetup Introduction” workshop presentation

    Swag logistics

    With this Modular Meetups Organizer Kit, you’ll have everything you need to plan and execute engaging, informative, and successful meetups for the Celestia community. Happy organizing!

    + + + + \ No newline at end of file diff --git a/community/overview.html b/community/overview.html new file mode 100644 index 00000000000..13a546302fc --- /dev/null +++ b/community/overview.html @@ -0,0 +1,45 @@ + + + + + + Community overview | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Community overview

    This section will highlight all the different resources and activities for the Celestia community.

    Here you will find links to our community calendar, Code of Conduct and other community-related resources.

    + + + + \ No newline at end of file diff --git a/community/speaker-list.html b/community/speaker-list.html new file mode 100644 index 00000000000..250505887ce --- /dev/null +++ b/community/speaker-list.html @@ -0,0 +1,45 @@ + + + + + + Speaker list | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Speaker list

    As a Modular Meetup organizer, we understand the importance of delivering engaging and informative content to your attendees. That's why we've created an exclusive Speaker List specifically tailored for organizers participating in the Modular Meetup Program. This resource gives you access to a curated selection of top-tier speakers who are passionate about Celestia and the modular ecosystem. Due to privacy, the list is not shared publicly but is accessible to participants of the Modular Meetup program when they create a meetup.

    The Speaker List features experts from Celestia Labs, as well as prominent figures from the broader Celestia and modular communities. Each individual is well-versed in various aspects of the modular ecosystem, ensuring that your meetup attendees gain valuable insights and deepen their understanding of modular blockchains.

    By joining the Modular Meetup Program, you can enjoy the benefits of our Speaker List and bring a touch of expertise to your events. The speakers can participate either in person or virtually, depending on location and timing.

    You can expect benefits from the Speaker List including high-quality presentations, interactive Q&A sessions, and knowledge-sharing opportunities facilitated by the best and brightest in the Celestia ecosystem. With our Speaker List, you'll be able to create memorable and impactful Modular Meetups that foster genuine connections and promote growth within the community.

    + + + + \ No newline at end of file diff --git a/developers/arbitrum-bridge.html b/developers/arbitrum-bridge.html new file mode 100644 index 00000000000..a53e1f9d121 --- /dev/null +++ b/developers/arbitrum-bridge.html @@ -0,0 +1,45 @@ + + + + + + Bridging in and out of your Orbit rollup | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Bridging in and out of your Orbit rollup

    This guide covers how to bridge in (deposit) and bridge out (withdrawal) from your Arbitrum Orbit L3 rollup.

    This guide will cover bridging in and out of your Orbit rollup.

    Below are two example transactions, one of a deposit and one of a withdrawal:

    bridge-overview-deposit-and-withdrawal-l3

    Bridge in (deposit) to your rollup

    Step 1: Add your custom chain config

    (1a) In the Arbitrum Bridge UI, click the menu dropdown in the top right. Select Settings.

    bridge-settings

    (1b) Under Developer Mode, select Turn on testnet mode. Add your custom chain config from outputInfo.json in the root of your orbit-setup-script directory.

    add-testnet-orbit-chain

    (1c) You'll then see the chain under Live Orbit Chains:

    live-orbit-chains

    Step 2: Deposit to your Orbit rollup

    (2a) Choose an amount of Arbitrum Sepolia ETH to bridge into your rollup. Click Move funds to <YOUR_ROLLUP_NAME>, in this case Move funds to Arbitrum L3 Rollup. Approve the transaction in your wallet.

    bridge-in-start

    (2b) You'll then see it load in the Pending transactions tab:

    bridge-in-pending

    (2c) Shortly after, in the Settled transactions tab you can see the transaction status. Click Success.

    bridge-in-settled-txs

    (2d) View the transaction on your local instance of Blockscout for your rollup:

    bridge-in-explorer-rollup-tx

    (2e) Optionally, click See Details for an overview of your deposit:

    bridge-in-success

    (2f) From the details page, you can also see the transaction for your deposit on Arbitrum Sepolia:

    bridge-in-sepolia-tx-explorer

    Bridge out (withdrawal) from your rollup

    Step 1: Choose an amount to withdraw from your rollup

    (1a) In the Arbitrum Bridge UI, choose your origin chain to your Arbitrum L3 Rollup and the destination chain as Arbitrum Sepolia.

    bridge-out-small-screenshot

    (1b) Click Move funds to Arbitrum Sepolia and read the disclaimer, check the boxes, and click Continue.

    bridge-out-begin

    (1c) Optionally, set a reminder on your calendar so you don't forget.

    (1d) After approving the transaction in your wallet, you'll be able to see the transaction in the Pending transactions tab:

    bridge-out-pending

    After approximately two hours, you will be able to proceed to Step 2: Claim your withdrawal.

    (1e) Click See details to see an overview of your withdrawal:

    bridge-out-begin-overview

    (1f) Optionally, view the transaction on your local explorer.

    bridge-out-tx-details

    (1g) To learn more about what is going on, click the Logs tab:

    bridge-out-logs-details-1

    bridge-out-logs-explorer-2

    Step 2: Claim your withdrawal

    After approximately 2 hours, you will be able to claim your withdrawal.

    (2a) Head back to the bridge UI and you will have a notification to claim your withdrawal. Click Claim in the details of the transaction:

    bridge-out-claim-withdrawal

    (2b) Approve the transaction in your wallet.

    (2c) After your transaction goes through, you can see the details in the bridge UI under Settled transactions:

    bridge-out-claim-success-withdrawal

    + + + + \ No newline at end of file diff --git a/developers/arbitrum-dapp-deploy.html b/developers/arbitrum-dapp-deploy.html new file mode 100644 index 00000000000..524feed2722 --- /dev/null +++ b/developers/arbitrum-dapp-deploy.html @@ -0,0 +1,65 @@ + + + + + + Deploy a dapp on your Arbitrum rollup devnet | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Deploy a dapp on your Arbitrum rollup devnet

    First, review the Arbitrum integration, Quickstart: Deploy an Arbitrum Orbit rollup, and Deploy a smart contract to your Arbitrum rollup pages.

    Dependencies

    Setup and contract deployment

    1. Clone the gm-portal from GitHub and start the frontend:

      bash
      cd $HOME
      +git clone https://github.com/jcstein/gm-portal.git
      +cd gm-portal && git checkout arbitrum
      +cd frontend && yarn && yarn dev
      cd $HOME
      +git clone https://github.com/jcstein/gm-portal.git
      +cd gm-portal && git checkout arbitrum
      +cd frontend && yarn && yarn dev
    2. In a new terminal instance, set your private key for the faucet as a variable and the RPC URL you're using:

      bash
      export PRIVATE_KEY=0xb6b15c8cb491557369f3c7d2c287b053eb229daa9c22138887752191c9520659
      +export ARB_RPC_URL=http://localhost:8547
      export PRIVATE_KEY=0xb6b15c8cb491557369f3c7d2c287b053eb229daa9c22138887752191c9520659
      +export ARB_RPC_URL=http://localhost:8547
    3. Change into the gm-portal/contracts directory in the same terminal and deploy the contract using Foundry:

      bash
      cd $HOME/gm-portal/contracts
      +forge script script/GmPortal.s.sol:GmPortalScript --rpc-url $ARB_RPC_URL --private-key $PRIVATE_KEY --broadcast
      cd $HOME/gm-portal/contracts
      +forge script script/GmPortal.s.sol:GmPortalScript --rpc-url $ARB_RPC_URL --private-key $PRIVATE_KEY --broadcast
    4. In the output of the deployment, find the contract address and set it as a variable:

      bash
      export CONTRACT_ADDRESS=<your-contract-address-from-the-output-above>
      export CONTRACT_ADDRESS=<your-contract-address-from-the-output-above>

    Interact with the contract

    Next, you're ready to interact with the contract from your terminal!

    1. Send a "gm" to the contract:

      bash
      cast send $CONTRACT_ADDRESS \
      +"gm(string)" "gm" \
      +--private-key $PRIVATE_KEY \
      +--rpc-url $ARB_RPC_URL
      cast send $CONTRACT_ADDRESS \
      +"gm(string)" "gm" \
      +--private-key $PRIVATE_KEY \
      +--rpc-url $ARB_RPC_URL
    2. Now that you've posted to the contract, you can read all "gms" (GMs) from the contract with this command:

      bash
      cast call $CONTRACT_ADDRESS "getAllGms()" --rpc-url $ARB_RPC_URL
      cast call $CONTRACT_ADDRESS "getAllGms()" --rpc-url $ARB_RPC_URL
    3. Next, query the total number of gms, which will be returned as a hex value:

      bash
      cast call $CONTRACT_ADDRESS "getTotalGms()" --rpc-url $ARB_RPC_URL
      cast call $CONTRACT_ADDRESS "getTotalGms()" --rpc-url $ARB_RPC_URL
    4. (Optional) In order to interact with the contract on the frontend, you'll need to fund an account that you have in your Ethereum wallet. Transfer to an external account with this command:

      bash
      export RECEIVER=<receiver ETH address>
      +cast send --private-key $PRIVATE_KEY $RECEIVER --value 1ether --rpc-url $ARB_RPC_URL
      export RECEIVER=<receiver ETH address>
      +cast send --private-key $PRIVATE_KEY $RECEIVER --value 1ether --rpc-url $ARB_RPC_URL

      TIP

      If you are in a different terminal than the one you set the private key in, you may need to set it again.

    Update the frontend

    Next, you will need to update a few things before you can interact with the contract on the frontend:

    1. Change the contract address on gm-portal/frontend/src/App.tsx to your contract address
    2. Match the chain info on gm-portal/frontend/src/main.tsx with the chain config of your L2
    3. If you changed the contract, update the ABI in gm-portal/frontend/GmPortal.json from gm-portal/contracts/out/GmPortal.sol/GmPortal.json. This can be done with:
    bash
    cd $HOME
    +cp dev/gm-portal/contracts/out/GmPortal.sol/GmPortal.json dev/gm-portal/frontend
    cd $HOME
    +cp dev/gm-portal/contracts/out/GmPortal.sol/GmPortal.json dev/gm-portal/frontend

    Interact with the frontend

    Now, login with your wallet that you funded, and post a GM on your GM portal!

    gm-arb

    + + + + \ No newline at end of file diff --git a/developers/arbitrum-deploy.html b/developers/arbitrum-deploy.html new file mode 100644 index 00000000000..a8002e56d14 --- /dev/null +++ b/developers/arbitrum-deploy.html @@ -0,0 +1,155 @@ + + + + + + Quickstart: Deploy an Arbitrum Orbit rollup | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Quickstart: Deploy an Arbitrum Orbit rollup

    This guide covers deploying a rollup using the Celestia Orbit chain deployment portal.

    After completing this tutorial, you will have a local development network rollup capable of hosting EVM-compatible smart contracts. This rollup will process transactions locally, settle on the public Arbitrum Sepolia testnet, and post data to Celestia's Mocha testnet.

    If you're looking to learn more about the integration of Celestia and Orbit, read the Arbitrum Orbit integration overview. If you're looking to learn more about Orbit, read A gentle introduction: Orbit chains.

    Thank you, Offchain Labs!

    This guide was made possible with the support and information provided by the Offchain Labs team, the creators of Arbitrum. For more detailed information and support, visit Arbitrum documentation and the original deployment guide.

    Prerequisites

    Setup

    This section was adapted from Arbitrum's Orbit quickstart.

    Step 1: Acquire Arbitrum Sepolia ETH

    You'll need at least 1 testnet ETH for a regular Orbit rollup or 0.6 ETH plus 0.4 of your desired native token for Orbit rollups with a custom gas token. The funds will cover the cost of deploying the base contracts to the base chain, in this case, Arbitrum Sepolia.

    The simplest way to do this is to:

    1. Use an L1 testnet ETH faucet like sepoliafaucet.com to acquire some testnet ETH on Ethereum Sepolia testnet.
    2. Bridge your L1 testnet ETH to L2 Arbitrum Sepolia using the Arbitrum bridge.

    Step 2: Pick your deployment type

    Visit the Celestia Orbit chain deployment portal. This portal offers the following options:

    1. Celestia Rollup: Transaction data is posted to Celestia
    2. Rollup: Transaction data is posted to Ethereum
    3. AnyTrust: Transaction data is posted by a Data Availability Committee

    Connect your wallet to the deployment portal. You may be prompted to add the Arbitrum Sepolia network to your wallet and/or switch your wallet to this network; approve this.

    In this guide, we will select Celestia ✨ and deploy a rollup which posts data to Celestia (1 above).

    Choose Celestia for DA

    Click Next. In the next step, we will configure the deployment.

    Step 3: Configure your Orbit chain's deployment

    The deployment portal will then display a form that looks like this:

    configuration

    Parameter descriptions can be found in the table below (more in-depth descriptions can be found in the deployment UI). We recommend sticking to the defaults; to learn more about customizing your Orbit chain's deployment configuration, visit How to customize your Orbit chain's deployment configuration:

    ParameterDescription
    Chain IDThis is a unique integer identifier for your chain's network, primarily used on chain indexes like Chainlist.org. It's not crucial for development networks, but in production, you'll need to choose a unique ID.
    Chain NameThe name you assign to your Orbit chain, which helps users and developers distinguish it from other chains. It should be memorable and recognizable.
    Challenge Period BlocksDetermines the time frame within which validators can dispute the state of the chain posted to the base chain. It's measured in blocks on the underlying L1 chain. A longer period allows more time for disputes but also delays withdrawals.
    Stake TokenSpecifies the token that validators must stake to participate in the validation process, using the token's contract address on the base chain. This can be ETH or another token, defined by its address.
    Base StakeThe minimum amount of stake token required for validators to post state assertions. A lower base stake lowers the barrier to entry but increases vulnerability to attacks, whereas a higher stake encourages honest participation but raises the entry barrier.
    OwnerThe account address that has the authority to deploy, own, and update the base contracts of your Orbit chain on its base chain. In production, this is usually a high-stakes address controlled by a DAO or a multisig setup. For development chains, it's a lower-stakes administrative account.
    Gas TokenThe token used for gas payments on the network, which must be natively deployed on the parent chain. There are specific requirements for custom gas tokens, such as having 18 decimals and not being a rebasing or fee-on-transfer token. This feature is primarily for Orbit AnyTrust chains.
    ValidatorsThis is the number of validators for your chain, including their addresses. The first validator is auto-generated and immutable. Validators are crucial for maintaining the integrity of the chain and posting state assertions to the base chain.
    Batch PosterResponsible for posting transaction batches from your Orbit chain to the base chain. An address for this role is automatically generated, with the private key stored in a configuration file.

    In the Configure Validators section, specify the number of validators and their addresses for your chain. The initial validator's address is pre-generated and immutable, with its key stored in a JSON file. Validators ensure transaction integrity and state assertions on the base chain. They're added to an allow-list for validation and staking. Base contracts refer to your Orbit chain's L2 contracts, and base chain to the L2 chain they're deployed on.

    In the Configure Batch Poster section, a batch poster address is auto-generated for posting transaction batches to the base contracts on the base chain. The address and its private key are also stored in a JSON configuration file. After configuring, proceed to review and deploy your Orbit chain.

    After configuring your batch poster, proceed to the next step.

    Step 3: Review & Deploy your Orbit chain

    Now, deploy your chain's base contracts to Arbitrum Sepolia!

    Click the Deploy button on the Review & Deploy page. Your wallet should prompt you to submit a transaction to the Arbitrum testnet. You'll have to pay a little gas; your wallet may denominate this in ETH; as long as you see your chosen Arbitrum testnet in the transaction details, this gas fee will be paid in testnet ETH.

    Before proceeding, let's briefly review what just happened:

    1. You submitted a deployment transaction to an Orbit "factory" smart contract on the Arbitrum testnet, the public L2 chain that your local Orbit chain will settle transactions to.
    2. This Orbit smart contract then initialized your Orbit chain's base contracts with the values that you specified in the previous step, and deployed these base contracts to the Arbitrum testnet.

    Your Orbit chain's base contracts are responsible for facilitating the exchange of information between your chain's node(s) and its base chain's nodes. This includes the batch posting of transactions from your Orbit chain to its base chain, the staking of tokens by your Orbit chain's validators the challenge mechanism, bridging mechanisms, and more.

    Once your transaction is complete, continue to Step 4 to download your chain's configuration files and launch your chain.

    Step 4: Download your chain's configuration files and launch your chain

    After configuring your chain, you will need to download the necessary configuration files to launch your chain. Click the Download zip files button to download both the Rollup Config and L3 Config in a single ZIP file.

    • Rollup Config: This is the nodeConfig.json file, encapsulating your chain's node configuration. It is crucial as it contains the private keys for your validator and batch poster, essential for signing transactions for RBlocks and batch postings to your chain's base contracts on the L2 chain.

    • L3 Config: This is the orbitSetupScriptConfig.json file, which holds your chain's configuration, including configurations needed for your Token Bridge contracts.

    Ensure to securely store these downloaded files as they contain sensitive information crucial for your chain's operation.

    download config

    Step 5: Clone the setup script repository and add your configuration files

    1. Clone the orbit-setup-script repository:

      bash
      git clone https://github.com/celestiaorg/orbit-setup-script.git
      git clone https://github.com/celestiaorg/orbit-setup-script.git
    2. Move the nodeConfig.json and orbitSetupScriptConfig.json files that you downloaded into the config directory in the root of your cloned orbit-setup-script repository.

    3. Install dependencies by running yarn install from the root of the orbit-setup-script repository.

    Step 6: Pick an L2 RPC URL for the Batch Poster

    In order for the Batch Poster, which is responsible for posting batches of data, to subscribe to Blobstream's smart contract events, the node most use a WebSocket connection, since an HTTP one will not support subscriptions. This RPC URL is different from the parent-chain.connection.url object used in the node config, and is not necessary when running a full node. WebSocket (WSS) URLs which are essential for real-time data fetching and interaction with the Arbitrum Sepolia network.

    To establish a WebSocket connection for your rollup to Arbitrum Sepolia, it's recommended to find an RPC provider with WSS connections from Arbitrum's docs.

    For this example, we will make an account on Alchemy. Follow these steps to set up your account and obtain a WSS URL using Alchemy:

    1. Visit Alchemy's website and sign up for an account.
    2. Once logged in, create a new app by selecting the Arbitrum network, specifically targeting the Arbitrum Sepolia testnet.
    3. After creating your app, navigate to the "API key" section to find your WebSocket (WSS) URL.
    4. In the next step, use this WSS URL in your nodeConfig.json under the celestia-cfg.eth-rpc object to ensure your node can establish a WebSocket connection to the Arbitrum Sepolia network and successfully subscribe to Blobstream events.

    Without a WSS connection, the Batch Poster won't be able to subscribe to Blobstream events, and thus will fall back to posting data to parent chain.

    Step 7: Run your light node for Mocha testnet

    First, be sure that your light node is running, using a command similar to:

    TIP

    If you are on Linux (or are not using Docker desktop), you may need to add the extra flags: --rpc.addr 0.0.0.0 and --rpc.port 26658 to your start command for your light node.

    Additionally, you will need to add host.docker.internal as a host in your docker-compose.yml:

    yaml
    extra_hosts:
    +      - "host.docker.internal:host-gateway"
    extra_hosts:
    +      - "host.docker.internal:host-gateway"
    bash
    celestia light start --p2p.network mocha --core.ip <RPC_URL>
    celestia light start --p2p.network mocha --core.ip <RPC_URL>

    To set your light node's auth token, you will use the auth token that returns when you run:

    bash
    celestia light auth admin --p2p.network mocha
    celestia light auth admin --p2p.network mocha

    Since the contracts deployed through the factories above are already configured to communicate with Blobstream, you now only have to configure your node accordingly. First understand the different variables that will be set in the config:

    • enable: set it to true if you are using Celestia DA 😁
    • rpc: RPC endpoint for celestia-node
    • tendermint-rpc: a celestia-core endpoint from a full node (NOTE: only needed for a batch poster node)
    • eth-rpc: Ethereum Client WSS RPC endpoint, only used when the node is a batch poster. The eth-rpc must be WSS. Otherwise, it won't be able to subscribe to events for Blobstream.
    • namespace-id: namespace being used to post data to Celestia
    • auth-token: auth token for your Celestia Node
    • is-poster: is the node with Celestia DA the batch poster, set to true if so.
    • gas-price: how much to pay for gas (in uTIA)
    • event-channel-size: size of the events channel used by the batch poster to wait for a range of headers that contains the header for the block in which it posted a blob, before posting the batch to the base layer for verification on Blobstream X.
    • blobstreamx-address: address of the Blobstream X contract on the base chain.
      • Note that the SequencerInbox contract for each chain has a constant address for the BlobstreamX contract, thus make sure that the Blobstream X address in the SequencerInbox being used for the templates in RollupCreator matches the one in your config.

    Now enable Celestia DA in your Arbitrum chain params in config/nodeConfig.json. If you'd like to use your own namespace, use a custom 10 byte value or random value using openssl rand -hex 10 for namespace-id:

    WARNING

    The Orbit contracts depend on the existing Blobstream X deployments. Before using these addresses, please verify the contract addresses on the official source below to avoid any issues due to incorrect addresses. This is crucial to protect against potential misuse by copy-paste errors.

    ts
    "celestia-cfg": {
    +  "enable": true,
    +  "rpc": "http://host.docker.internal:26658",
    +  "tendermint-rpc": "http://consensus-full-mocha-4.celestia-mocha.com:26657",
    +  "eth-rpc": "wss://<YOUR_ETH_RPC_WSS_URL>",
    +  "namespace-id": "<YOUR_10_BYTE_NAMESPACE>",
    +  "auth-token": "<YOUR_AUTH_TOKEN>",
    +  "is-poster": true,
    +  "gas-price": 0.3,
    +  "event-channel-size": 100,
    +  "blobstreamx-address": "0xc3e209eb245Fd59c8586777b499d6A665DF3ABD2",
    +}
    "celestia-cfg": {
    +  "enable": true,
    +  "rpc": "http://host.docker.internal:26658",
    +  "tendermint-rpc": "http://consensus-full-mocha-4.celestia-mocha.com:26657",
    +  "eth-rpc": "wss://<YOUR_ETH_RPC_WSS_URL>",
    +  "namespace-id": "<YOUR_10_BYTE_NAMESPACE>",
    +  "auth-token": "<YOUR_AUTH_TOKEN>",
    +  "is-poster": true,
    +  "gas-price": 0.3,
    +  "event-channel-size": 100,
    +  "blobstreamx-address": "0xc3e209eb245Fd59c8586777b499d6A665DF3ABD2",
    +}

    See the compatibility matrix in the appendix to verify you're using the right versions.

    Step 8: Run your chain's node and block explorer

    Start Docker, then run docker-compose up -d from the root of the orbit-setup-script repository.

    A Nitro node and BlockScout explorer instance will be started. Visit http://localhost/ to access your BlockScout explorer instance - this will allow you to view your chain's transactions and blocks, which can be useful for debugging.

    blockscout

    After you have some activity on your rollup, it will look more like this:

    explorer-view

    Step 9: Finish setting up your chain

    The Offchain Labs team has provided a Hardhat script that handles the following tasks:

    1. Fund the batch-poster and validator (staker) accounts on your underlying L2 chain.
    2. Deposit ETH into your account on the chain using your chain's newly deployed bridge.
    3. Deploy your Token Bridge contracts on both L2 and local Orbit chains.
    4. Configure parameters on the chain.

    To run this script, issue the following command from the root of the orbit-setup-script repository, replacing YourPrivateKey with the private key of the Owner account you used to deploy your chain's contracts, and replacing http://localhost:8449 with the RPC URL of your chain's node.

    First, export your private key as a variable:

    bash
    PRIVATE_KEY="YourPrivateKey" \
    +  L2_RPC_URL="https://sepolia-rollup.arbitrum.io/rpc" \
    +  L3_RPC_URL="http://localhost:8449" yarn run setup
    PRIVATE_KEY="YourPrivateKey" \
    +  L2_RPC_URL="https://sepolia-rollup.arbitrum.io/rpc" \
    +  L3_RPC_URL="http://localhost:8449" yarn run setup

    Successful logs will appear similar to:

    bash
    Funding batch-poster accounts on parent chain with 0.3 ETH
    +Transaction hash on parent chain: 0x6c7360a96165c570dcb7ce609d748d612c5fa5b76e229cd81ba5f5c93c00f805
    +Transaction was mined in block 28217647 on parent chain
    +Funding staker accounts on parent chain with 0.3 ETH
    +Transaction hash on parent chain: 0x59d2db6c5095b9e329c80211b7a761d20064379e3382d156b69e5cf3b5fe2fc7
    +Transaction was mined in block 28217653 on parent chain
    +Running Orbit Chain Native token deposit to Deposit ETH or native ERC20 token from parent chain to your account on Orbit chain ... 💰💰💰💰💰💰
    +Transaction hash on parent chain:  0x8dee6e88d3b62b258c1574cbb7005e1c3cf193b60a99b5c2fcfae00819b7ed82
    +0.4 ETHs are deposited to your account
    +Balance not changed yet. Waiting for another 30 seconds ⏰⏰⏰⏰⏰⏰
    +Balance of your account on Orbit chain increased by the native token you have just sent.
    +Running tokenBridgeDeployment or erc20TokenBridge script to deploy token bridge contracts on parent chain and your Orbit chain 🌉🌉🌉🌉🌉
    +Creating token bridge for rollup 0x7fbEB5BC73a11b438891022786feb2C624f275F0
    +Token bridge deployed in transaction 0x4888fdf44251d456bbfca92bfc6e180cfe0b096ffbea2f6da2a203a16902214f
    +Waiting for retryables...
    +Retryable #1: 0xc61382d5609ab0ece36b2776349c8bdceeafdd13dde9624cdf3d746fb4cf7d79
    +Retryable #2: 0xf31fd34f8a9d9057198d8b13e755e583766bd528459733d948d9ffbc980c9506
    +Done!
    +Weth gateway set in tx 0xf2ddc2dad90e7e2b20a772bf89f989224165659d50824b98d7340e12265abf01
    +Waiting for retryables...
    +Retryable #1: 0xf47dc66514fd78e4666e35abd12df7d1ae2c79f69f7dfedb8d98e4106142ab7c
    +Done!
    +network.json updated
    +Done!
    +Running l3Configuration script to configure your Orbit chain 📝📝📝📝📝
    +Setting the Minimum Base Fee for the Orbit chain
    +Minimum Base Fee is set on the block number 13 on the Orbit chain
    +Setting the  network fee receiver for the Orbit chain
    +network fee receiver is set on the block number 14 on the Orbit chain
    +Setting the infrastructure fee collector address for the Orbit chain
    +infrastructure fee collector address is set on the block number 15 on the Orbit chain
    +Getting L1 base fee estimate
    +L1 Base Fee estimate on L2 is 4989526079
    +Setting L1 base fee estimate on L3 to 5158076079
    +L1 base fee estimate is set on the block number 16 on the Orbit chain
    +All things done! Enjoy your Orbit chain. LFG 🚀🚀🚀🚀
    +Transferring ownership on L3, from rollup owner to upgrade executor 🔃🔃🔃
    +Adding Upgrade Executor contract to the chain owners
    +Executor has been added to chain owners on TX: 0x97b50f60b60d0e658fdbf185969db0a0327bd0ae9e57cd65af2a7f9be0eeb5b0
    +Executing removeChainOwner through the UpgradeExecutor contract
    +Transaction complete, rollup owner removed from chain owners on TX: 0x019850732270d8c436585c7921219252422228b5d0f559da0da219f0fa2b7216
    +  Done in 58.49s.
    Funding batch-poster accounts on parent chain with 0.3 ETH
    +Transaction hash on parent chain: 0x6c7360a96165c570dcb7ce609d748d612c5fa5b76e229cd81ba5f5c93c00f805
    +Transaction was mined in block 28217647 on parent chain
    +Funding staker accounts on parent chain with 0.3 ETH
    +Transaction hash on parent chain: 0x59d2db6c5095b9e329c80211b7a761d20064379e3382d156b69e5cf3b5fe2fc7
    +Transaction was mined in block 28217653 on parent chain
    +Running Orbit Chain Native token deposit to Deposit ETH or native ERC20 token from parent chain to your account on Orbit chain ... 💰💰💰💰💰💰
    +Transaction hash on parent chain:  0x8dee6e88d3b62b258c1574cbb7005e1c3cf193b60a99b5c2fcfae00819b7ed82
    +0.4 ETHs are deposited to your account
    +Balance not changed yet. Waiting for another 30 seconds ⏰⏰⏰⏰⏰⏰
    +Balance of your account on Orbit chain increased by the native token you have just sent.
    +Running tokenBridgeDeployment or erc20TokenBridge script to deploy token bridge contracts on parent chain and your Orbit chain 🌉🌉🌉🌉🌉
    +Creating token bridge for rollup 0x7fbEB5BC73a11b438891022786feb2C624f275F0
    +Token bridge deployed in transaction 0x4888fdf44251d456bbfca92bfc6e180cfe0b096ffbea2f6da2a203a16902214f
    +Waiting for retryables...
    +Retryable #1: 0xc61382d5609ab0ece36b2776349c8bdceeafdd13dde9624cdf3d746fb4cf7d79
    +Retryable #2: 0xf31fd34f8a9d9057198d8b13e755e583766bd528459733d948d9ffbc980c9506
    +Done!
    +Weth gateway set in tx 0xf2ddc2dad90e7e2b20a772bf89f989224165659d50824b98d7340e12265abf01
    +Waiting for retryables...
    +Retryable #1: 0xf47dc66514fd78e4666e35abd12df7d1ae2c79f69f7dfedb8d98e4106142ab7c
    +Done!
    +network.json updated
    +Done!
    +Running l3Configuration script to configure your Orbit chain 📝📝📝📝📝
    +Setting the Minimum Base Fee for the Orbit chain
    +Minimum Base Fee is set on the block number 13 on the Orbit chain
    +Setting the  network fee receiver for the Orbit chain
    +network fee receiver is set on the block number 14 on the Orbit chain
    +Setting the infrastructure fee collector address for the Orbit chain
    +infrastructure fee collector address is set on the block number 15 on the Orbit chain
    +Getting L1 base fee estimate
    +L1 Base Fee estimate on L2 is 4989526079
    +Setting L1 base fee estimate on L3 to 5158076079
    +L1 base fee estimate is set on the block number 16 on the Orbit chain
    +All things done! Enjoy your Orbit chain. LFG 🚀🚀🚀🚀
    +Transferring ownership on L3, from rollup owner to upgrade executor 🔃🔃🔃
    +Adding Upgrade Executor contract to the chain owners
    +Executor has been added to chain owners on TX: 0x97b50f60b60d0e658fdbf185969db0a0327bd0ae9e57cd65af2a7f9be0eeb5b0
    +Executing removeChainOwner through the UpgradeExecutor contract
    +Transaction complete, rollup owner removed from chain owners on TX: 0x019850732270d8c436585c7921219252422228b5d0f559da0da219f0fa2b7216
    +  Done in 58.49s.

    Find your PFB on Celenium by looking at the namespace or account you posted from.

    See an example blob that was posted while making this guide.

    Congratulations with Celestia underneath

    Your local Orbit rollup is now running. You'll see an outputInfo.json file in the main directory of your script folder - this contains more information about your chain, including the addresses of your chain's base contracts.

    In the next guides, learn how to run a full and validating full node or bridge in and out of your rollup.

    Appendix

    Extra resources in Arbitrum documentation:

    Compatibility matrix

    ComponentVersionDetails
    Nitrov2.3.1-rc.1Includes the replay binary for the WASM root 0x10c65b27d5031ce2351c719072e58f3153228887f027f9f6d65300d2b5b30152. Read the overview for overall changes.
    Contractsv1.2.1-celestiaIntegrates Blobstream X functionality into nitro-contracts v1.2.1
    Orbit SDKv0.8.2 Orbit SDK for Celestia DAThis is not compatible with Orbit SDK v0.8.2 or with the latest changes to nitro-contracts for the Atlas upgrade. The Orbit SDK itself is in Alpha.
    celestia-nodev0.13.1This integration has only been tested with celestia-node 0.13.1 and only works with said version, and with future versions after that. Under the hood, the Nitro node uses this commit of celestia-openrpc.

    Blobstream X contract deployments

    The Orbit contracts depend on the following Blobstream X deployments. The current deployments, which can be found at 0xc3e209eb245Fd59c8586777b499d6A665DF3ABD2 in both chains, relays headers from the Mocha-4 testnet to the chains below:

    • Arbitrum Sepolia
    • Base Sepolia

    Arbitrum Sepolia

    Base Sepolia

    • RollupCreator: 0x1Bb8ADd5e878b12Fa37756392642eB94C53A1Cf4
    • TokenBridgeCreator: 0xAa3b8B63cCCa3c98b948FD1d6eD875d378dE2C6c
    • TokenBridgeRetryableSender: 0x4270889AdcB82338C5FF5e64B45c0A3d31CFd08C
    • Find additional Base Sepolia deployments below

    Arbitrum Sepolia additional deployments

    ContractAddress
    Bridge0x95FEA00e689e8D1CBa909836E1Ef1b941D5f21b1
    SequencerInbox0x95CBDa89325db5529eAF1813E181f66B83A7d65a
    Inbox0x3681Cbb0E95AB50b63F2FC524FbBcC78adEfBd33
    RollupEventInbox0x61e154128b6a1400ea8090B4431B4aA1DBb80Cc4
    Outbox0x5187a92539bB4A2befe1fc078745c84AB6d37171
    ERC20Bridge0xD0a6699Fc7519966685181c80BF98D35aFa1fC95
    SequencerInbox0x2588867F19E2DE51f90F0aB852C7Ad11228e3d83
    ERC20Inbox0x6cB49605f10831749c6090AD09918bC61439bacE
    ERC20RollupEventInbox0x7fC4D9A24949680faD666FeEe7cD6a100E39C4F0
    ERC20Outbox0xA773e19DC9e822933A7e72Df9c87eD1578701D29
    BridgeCreator0x3Bc040EAca40b91FA06cf55Ea91842FaC88b1AF4
    OneStepProver00x5810F0916BAE1067Ca1efcc00AaaF30301af001c
    OneStepProverMemory0xaC3427E621C6F10dC2ABdAB00188D92690503914
    OneStepProverMath0xFB612fb83959b8ACD3E49540B29C93c5A67e05f1
    OneStepProverHostIo0x630093954CbF19Fe4532A2edD0bD3B10dEcA7A4D
    OneStepProofEntry0x53DEA3A90Fd6C82840a1f7224F799D622f142Df4
    ChallengeManager0x01B5905B154F21a393F5B5a0C6d15B53a493C05e
    RollupAdminLogic0xe371AFcb8437bF61bd831EF57Be7A2496D88488B
    RollupUserLogic0xE24a60b758b51b0a3dA5E8F4F6ddf1cd0aFF646C
    ValidatorUtils0x7973D0b475E898082dF25c1617CBce1917cFED17
    ValidatorWalletCreator0xe2662ff9b41f39e63A850E50E013Ea66e60A4F37
    RollupCreator0x79751B011BCc20F413a2c4E3AF019b6E2a9738B9
    DeployHelper0xd2D353916B34a877793628049c99858f04123eE1

    Base Sepolia additional deployments

    ContractAddress
    Bridge0xb6052122545AACD2BDda0Ca9FA56416bD968cDbc
    SequencerInbox0xcd9FCa5015b5ce2B06a2266e4a5dd54D9ca39F1a
    Inbox0x44B412b291fEf00398501B2cA353EA912AD0fe13
    RollupEventInbox0x51D196e07a27DBA0F4461Dd6CC26108424F196f7
    Outbox0x5A48aDf22f526eBD06e3e8856cFEa2490923CC55
    ERC20Bridge0x9abC41fEfAe7E7543a01FA837AeC909F96147280
    SequencerInbox0x8f97Cb7c643Acd7f79f3B13841b24a243dA51242
    ERC20Inbox0x40f8c63e0a20B399bCd9631A22E57BB988a9400e
    ERC20RollupEventInbox0x3B6e845fb9f0c8Ee4E9F6D44781f6547d9c6359a
    ERC20Outbox0xc99eEA0B8e67D5b2226AB6D37882DAAf6dd7593b
    BridgeCreator0xC7535F078CB3880a0FD5E54FA7A3B4EAf09b3924
    OneStepProver00xf889a3174Fddd9f78E6cd250Ebf4c16F1bDd1b6a
    OneStepProverMemory0x61254e43e5c1e9E801F9C56B47a9ac3EADF6d1E9
    OneStepProverMath0x55527d53fdA37Dbf1924482b40AcF8625E1cAA5B
    OneStepProverHostIo0x03B43F7B61Fa100611191F481Ef48aa1fc98F434
    OneStepProofEntry0x89b7c7970c13BB587893a70697AD6d2A335b6A15
    ChallengeManager0x04CAe899Fc0B7Ef45c529f8Bf075D54F6fB70eD9
    RollupAdminLogic0x99E9D2F04352B42C18F1DA5Dd93a970F82C08aFe
    RollupUserLogic0x1ae3A8DC1e7eFD37F418B2987D3DF74c5a917a8B
    ValidatorUtils0x1cc4551922C069A9aDE06756BF14bF0410eA44fF
    ValidatorWalletCreator0x78f8B2941ddE5a8A312814Ebd29c2E2A36f25E91
    RollupCreator0x1Bb8ADd5e878b12Fa37756392642eB94C53A1Cf4
    DeployHelper0x20d8153AaCC4E6D29558fa3916BfF422BEDE9B5E
    + + + + \ No newline at end of file diff --git a/developers/arbitrum-full-node.html b/developers/arbitrum-full-node.html new file mode 100644 index 00000000000..14d4eec9a10 --- /dev/null +++ b/developers/arbitrum-full-node.html @@ -0,0 +1,47 @@ + + + + + + Running a full node and/or validator | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Running a full node and/or validator

    Prerequisites

    Running a full node

    To run a full node, you can follow the steps outlined in the Arbitrum docs, with the difference being that you will use this image: dfcelestia/nitro-node-dev:latest instead of the one mentioned in the Arbitrum docs.

    Note that you can either use the flags in the nitro binary + the flags found in the celestia package, or you can just provide a node config.json file with the celestia-cfg for them to run it, which would look something like this:

    json
    docker run --rm -v "$HOME/Documents/configs/nodeConfig.json:/config.json:ro" \
    +  --network host celestia-nitro:v2.3.1-rc.1 --conf.file /config.json
    docker run --rm -v "$HOME/Documents/configs/nodeConfig.json:/config.json:ro" \
    +  --network host celestia-nitro:v2.3.1-rc.1 --conf.file /config.json

    Running a full node with validation

    The information above applies to the steps outlined to run a validating full node (validator).

    Finally, note that this will require connection to a DA node, and we recommend running a Bridge node if you will be instantiating multiple rollups.

    + + + + \ No newline at end of file diff --git a/developers/arbitrum-integration.html b/developers/arbitrum-integration.html new file mode 100644 index 00000000000..0c265acff53 --- /dev/null +++ b/developers/arbitrum-integration.html @@ -0,0 +1,45 @@ + + + + + + Introduction to Arbitrum rollups with Celestia as DA | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Introduction to Arbitrum rollups with Celestia as DA

    Celestia_Arbitrum

    Overview

    The integration of Celestia with Arbitrum Orbit and the Nitro tech stack marks the first external contribution to the Arbitrum Orbit protocol layer, offering developers an additional option for selecting a data availability layer alongside Arbitrum AnyTrust. The integration allows developers to deploy an Orbit Chain that uses Celestia for data availability and settles on Arbitrum One, Ethereum, or other EVM chains.

    Learn more about Orbit in Arbitrum's introduction.

    Key components

    The integration of Celestia with Arbitrum orbit is possible thanks to 3 key components:

    Additionally, the Ethereum fallback mechanism is a feature of the integration, which is native in Nitro.

    DA provider implementation

    The Arbitrum Nitro code has a DataAvailabilityProvider interface that is used across the codebase to store and retrieve data from a specific provider (eip4844 blobs, Anytrust, and now Celestia).

    This integration implements the DataAvailabilityProvider interface for Celestia DA

    Additionally, this integration comes with the necessary code for a Nitro chain node to post and retrieve data from Celestia.

    The core logic behind posting and retrieving data happens in celestia.go where data is stored on Celestia and serialized into a small batch of data that gets published once the necessary range of headers (data roots) has been relayed to the BlobstreamX contract. Then the Read logic takes care of taking the deserialized Blob Pointer struct and consuming it in order to fetch the data from Celestia and additionally inform the fetcher about the position of the data on Celestia (we'll get back to this in the next section).

    The following represents a non-exhaustive list of considerations when running a Batch Poster node for a chain with Celestia underneath:

    • You will need to use a consensus full node RPC endpoint, you can find a list of them for Mocha here
    • The Batch Poster will only post a Celestia batch to the underlying chain if the height for which it posted is in a recent range in BlobstreamX and if the verification succeeds, otherwise it will discard the batch. Since it will wait until a range is relayed, it can take several minutes for a batch to be posted, but one can always make an on-chain request for the BlobstreamX contract to relay a header promptly.

    The following represents a non-exhaustive list of considerations when running a Nitro node for a chain with Celestia underneath:

    • The TendermintRpc endpoint is only needed by the batch poster, every other node can operate without a connection to a full node.
    • The message header flag for Celestia batches is 0x0c.
    • You will need to know the namespace for the chain that you are trying to connect to, but don't worry if you don't find it, as the information in the BlobPointer can be used to identify where a batch of data is in the Celestia Data Square for a given height, and thus can be used to find out the namespace as well!

    Preimage Oracle Implementation

    In order to support fraud proofs, this integration has the necessary code for a Nitro validator to populate its preimage mapping with Celestia hashes that then get "unpealed" in order to reveal the full data for a Blob. You can read more about the "Hash Oracle Trick".

    The data structures and hashing functions for this can be found in the nitro/das/celestia/tree folder

    You can see where the preimage oracle gets used in the fraud proof replay binary here

    Something important to note is that the preimage oracle only keeps track of hashes for the rows in the Celestia data square in which a blob resides in, this way each Orbit chain with Celestia underneath does not need validators to recompute an entire Celestia Data Square, but instead, only have to compute the row roots for the rows in which it's data lives in, and the header data root, which is the binary merkle tree hash built using the row roots and column roots fetched from a Celestia node. Because only data roots that can be confirmed on Blobstream get accepted into the sequencer inbox, one can have a high degree of certainty that the canonical data root being unpealed as well as the row roots are in fact correct.

    Blobstream X implementation

    Finally, the integration only accepts batches with information that can be confirmed on BlobstreamX, which gives us a high certainty that data was made available on Celestia.

    You can see how BlobstreamX is integrated into the SequencerInbox.sol contract here, which allows us to discard batches with otherwise faulty data roots, thus giving us a high degree of confidence that the data root can be safely unpacked in case of a challenge.

    The Celestia and Arbitrum integration also includes Blobstream, which relays commitments to Celestia’s data root to an onchain light client on Ethereum. This allows L2 solutions that settle on Ethereum to benefit from the scalability Celestia’s data availability layer can provide.

    Ethereum fallback mechanism in Nitro

    By default in Arbitrum Nitro, the Ethereum fallback mechanism in the BatchPoster function is handling the process of storing data, with a fallback mechanism to store data onchain if the primary data availability storage fails.

    The @celestiaorg/nitro integration uses the same fallback mechanism.

    More information can be found on the Ethereum fallback mechanisms for Celestia, which enables Ethereum L2s (or L3s) to “fall back” to using Ethereum calldata for data availability in the event of downtime on Celestia Mainnet Beta.

    The fallback logic for Celestia DA is configurable, providing an alternative to the previous default fallback mechanism. Additionally, an ability has been added to the Arbitrum node software which allows the sequencer to call VerifyAttestation to check if a data root has been posted on Blobstream or not, before it sends the sequencer message (data pointer) to the underlying chain.

    Next steps

    In the next page, learn how to deploy an Arbitrum rollup devnet using Celestia as DA.

    + + + + \ No newline at end of file diff --git a/developers/arbitrum-smart-contract.html b/developers/arbitrum-smart-contract.html new file mode 100644 index 00000000000..591afeca0cb --- /dev/null +++ b/developers/arbitrum-smart-contract.html @@ -0,0 +1,177 @@ + + + + + + Deploy a smart contract to your Arbitrum rollup | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Deploy a smart contract to your Arbitrum rollup

    Overview

    Welcome to the guide on deploying a smart contract to your Arbitrum rollup. In this tutorial, you will learn how to deploy a smart contract using the L2 Nitro devnet and the provided public and private keys for testing purposes.

    Prerequisites

    Setup

    First, in your $HOME directory, set up a new project folder for this tutorial and init the project with npm:

    bash
    cd $HOME
    +mkdir counter-project && cd counter-project && npm init -y
    cd $HOME
    +mkdir counter-project && cd counter-project && npm init -y

    Next, initialize a Foundry project with the following command:

    bash
    forge init counter_contract
    forge init counter_contract

    Create your smart contract

    Take a look at the Counter.sol file in your counter-project/counter_contract/src directory:

    solidity
    // SPDX-License-Identifier: UNLICENSED
    +pragma solidity ^0.8.13;
    +
    +contract Counter {
    +    uint256 public number;
    +
    +    function setNumber(uint256 newNumber) public {
    +        number = newNumber;
    +    }
    +
    +    function increment() public {
    +        number++;
    +    }
    +}
    // SPDX-License-Identifier: UNLICENSED
    +pragma solidity ^0.8.13;
    +
    +contract Counter {
    +    uint256 public number;
    +
    +    function setNumber(uint256 newNumber) public {
    +        number = newNumber;
    +    }
    +
    +    function increment() public {
    +        number++;
    +    }
    +}

    The contract contains a public unsigned integer variable named "number". There are two public functions in this contract. The setNumber function allows anyone to set a new value for the "number" variable, while the increment function increases the value of "number" by one each time it's called.

    You can learn more about Solidity and smart contract programming.

    To compile the contract, run the following forge command from the $HOME/counter-project/counter_contract/ directory:

    bash
    forge build
    forge build

    Your output should look similar to the following:

    bash
    [⠢] Compiling...
    +[⠔] Compiling 21 files with 0.8.19
    +[⠑] Solc 0.8.19 finished in 1.24s
    +Compiler run successful
    [⠢] Compiling...
    +[⠔] Compiling 21 files with 0.8.19
    +[⠑] Solc 0.8.19 finished in 1.24s
    +Compiler run successful

    Test your smart contract

    Now, open the test/Counter.t.sol file:

    solidity
    // SPDX-License-Identifier: UNLICENSED
    +pragma solidity ^0.8.13;
    +
    +import "forge-std/Test.sol";
    +import "../src/Counter.sol";
    +
    +contract CounterTest is Test {
    +    Counter public counter;
    +
    +    function setUp() public {
    +        counter = new Counter();
    +        counter.setNumber(0);
    +    }
    +
    +    function testIncrement() public {
    +        counter.increment();
    +        assertEq(counter.number(), 1);
    +    }
    +
    +    function testSetNumber(uint256 x) public {
    +        counter.setNumber(x);
    +        assertEq(counter.number(), x);
    +    }
    +}
    // SPDX-License-Identifier: UNLICENSED
    +pragma solidity ^0.8.13;
    +
    +import "forge-std/Test.sol";
    +import "../src/Counter.sol";
    +
    +contract CounterTest is Test {
    +    Counter public counter;
    +
    +    function setUp() public {
    +        counter = new Counter();
    +        counter.setNumber(0);
    +    }
    +
    +    function testIncrement() public {
    +        counter.increment();
    +        assertEq(counter.number(), 1);
    +    }
    +
    +    function testSetNumber(uint256 x) public {
    +        counter.setNumber(x);
    +        assertEq(counter.number(), x);
    +    }
    +}

    This file performs unit testing on the contract we created in the previous section. Here's what the test is doing:

    • The contract includes a public "Counter" type variable called "counter". In the setUp function, it initializes a new instance of the "Counter" contract and sets the "number" variable to 0.

    • There are two test functions in the contract: testIncrement and testSetNumber.

    • The testIncrement function tests the "increment" function of the "Counter" contract by calling it and then asserting that the "number" in the "Counter" contract is 1. It verifies if the increment operation correctly increases the number by one.

    • The testSetNumber function is more generic. It takes an unsigned integer argument 'x' and tests the "setNumber" function of the "Counter" contract. After calling the "setNumber" function with 'x', it asserts that the "number" in the "Counter" contract is equal to 'x'. This verifies that the "setNumber" function correctly updates the "number" in the "Counter" contract.

    Now, to test your code, run the following:

    bash
    forge test
    forge test

    If the test is successful, your output should be similar to this:

    bash
    [⠆] Compiling...
    +No files changed, compilation skipped
    +
    +Running 2 tests for test/Counter.t.sol:CounterTest
    +[PASS] testIncrement() (gas: 28334)
    +[PASS] testSetNumber(uint256) (runs: 256, μ: 27709, ~: 28409)
    +Test result: ok. 2 passed; 0 failed; finished in 8.96ms
    [⠆] Compiling...
    +No files changed, compilation skipped
    +
    +Running 2 tests for test/Counter.t.sol:CounterTest
    +[PASS] testIncrement() (gas: 28334)
    +[PASS] testSetNumber(uint256) (runs: 256, μ: 27709, ~: 28409)
    +Test result: ok. 2 passed; 0 failed; finished in 8.96ms

    Deploying your smart contract

    Funded accounts

    Your L2 Nitro devnet will have a public and private key funded as a faucet to use for testing:

    • On both L1 and L2
      • Public key: 0x3f1Eae7D46d88F08fc2F8ed27FCb2AB183EB2d0E
      • Private key: 0xb6b15c8cb491557369f3c7d2c287b053eb229daa9c22138887752191c9520659

    Alternatively, you can fund other addresses by using the scripts send-l1 and send-l2.

    The L1 Geth devnet will be running at http://localhost:8545 and the L2 Nitro devnet will be on http://localhost:8547 and ws://localhost:8548.

    Using our Arbitrum devnet

    We will use the local RPC endpoint (http://localhost:8547) and accounts above to test with.

    Let's deploy the contract now. First, set a private key from anvil:

    bash
    export L2_PRIVATE_KEY=0xe887f7d17d07cc7b8004053fb8826f6657084e88904bb61590e498ca04704cf2
    +export ARB_RPC_URL=http://localhost:8547
    export L2_PRIVATE_KEY=0xe887f7d17d07cc7b8004053fb8826f6657084e88904bb61590e498ca04704cf2
    +export ARB_RPC_URL=http://localhost:8547

    Now, deploy the contract:

    bash
    forge create --rpc-url $ARB_RPC_URL \
    +  --private-key $L2_PRIVATE_KEY \
    +  src/Counter.sol:Counter
    forge create --rpc-url $ARB_RPC_URL \
    +  --private-key $L2_PRIVATE_KEY \
    +  src/Counter.sol:Counter

    A successful deployment will return output similar to below:

    bash
    [⠆] Compiling...
    +No files changed, compilation skipped
    +Deployer: 0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266
    +Deployed to: 0x5FbDB2315678afecb367f032d93F642f64180aa3
    +Transaction hash: 0xf1a793a793cd9fc588f5132d99008565ea361eb3535d66499575e9e1908200b2
    [⠆] Compiling...
    +No files changed, compilation skipped
    +Deployer: 0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266
    +Deployed to: 0x5FbDB2315678afecb367f032d93F642f64180aa3
    +Transaction hash: 0xf1a793a793cd9fc588f5132d99008565ea361eb3535d66499575e9e1908200b2

    Once you've deployed the contract, you're ready to interact with it!

    First, we'll set it as a variable:

    bash
    export CONTRACT_ADDRESS=0x5FbDB2315678afecb367f032d93F642f64180aa3
    export CONTRACT_ADDRESS=0x5FbDB2315678afecb367f032d93F642f64180aa3

    Interacting with your smart contract

    Foundry uses cast, a CLI for performing Ethereum RPC calls.

    To write to the contract, we'll use the cast send command:

    bash
    cast send $CONTRACT_ADDRESS "setNumber(uint256)" 10 \
    +  --rpc-url $ARB_RPC_URL --private-key $L2_PRIVATE_KEY
    cast send $CONTRACT_ADDRESS "setNumber(uint256)" 10 \
    +  --rpc-url $ARB_RPC_URL --private-key $L2_PRIVATE_KEY

    Your output will look similar:

    bash
    blockHash               0x131822bef6eb59656d7e1387c19b75be667e587006710365ec5cf58030786c42
    +blockNumber             3
    +contractAddress
    +cumulativeGasUsed       43494
    +effectiveGasPrice       3767182372
    +gasUsed                 43494
    +logs                    []
    +logsBloom               0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
    +root
    +status                  1
    +transactionHash         0x8f15d6004598f0662dd673a9898dceef77be8cc28408cecc284b28d7be32307d
    +transactionIndex        0
    +type                    2
    blockHash               0x131822bef6eb59656d7e1387c19b75be667e587006710365ec5cf58030786c42
    +blockNumber             3
    +contractAddress
    +cumulativeGasUsed       43494
    +effectiveGasPrice       3767182372
    +gasUsed                 43494
    +logs                    []
    +logsBloom               0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
    +root
    +status                  1
    +transactionHash         0x8f15d6004598f0662dd673a9898dceef77be8cc28408cecc284b28d7be32307d
    +transactionIndex        0
    +type                    2

    Now, we can make a read call to view the state of the number variable, using the cast call command:

    bash
    cast call $CONTRACT_ADDRESS "number()" --rpc-url $ARB_RPC_URL
    cast call $CONTRACT_ADDRESS "number()" --rpc-url $ARB_RPC_URL

    The result will look similar:

    bash
    0x000000000000000000000000000000000000000000000000000000000000000a
    0x000000000000000000000000000000000000000000000000000000000000000a

    Convert the result from hexadecimal to a base 10 value with:

    bash
    echo $((0x000000000000000000000000000000000000000000000000000000000000000a))
    echo $((0x000000000000000000000000000000000000000000000000000000000000000a))

    Next steps

    Congratulations! You've learned how to deploy a smart contract to your Arbitrum rollup devnet.

    What will you build next? In our next tutorial, we will be going over how to deploy a dapp to your Arbitrum rollup.

    + + + + \ No newline at end of file diff --git a/developers/blobstream-contracts.html b/developers/blobstream-contracts.html new file mode 100644 index 00000000000..074698153e0 --- /dev/null +++ b/developers/blobstream-contracts.html @@ -0,0 +1,133 @@ + + + + + + Integrate with Blobstream contracts | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Integrate with Blobstream contracts

    Getting started

    Prerequisites

    Make sure to have the following installed:

    Installing Blobstream X contracts

    We will be using the Blobstream X implementation of Blobstream, so we can install its repo as a dependency:

    Install the Blobstream X contracts repo as a dependency:

    sh
    forge install succinctlabs/blobstreamx --no-commit
    forge install succinctlabs/blobstreamx --no-commit

    Make sure that the directory you're running this command from is an initialized git repository. If not, just initialize the repo using:

    sh
    git init
    git init

    Note that the minimum Solidity compiler version for using the Blobstream contracts is 0.8.19.

    Example usage

    Example minimal Solidity contract for a stub ZK rollup that leverages the BlobstreamX.sol contract to check that data has been posted to Celestia:

    solidity
    // SPDX-License-Identifier: Apache-2.0
    +pragma solidity ^0.8.19;
    +
    +TBD
    +import "blobstream-contracts/IDAOracle.sol";
    +import "blobstream-contracts/DataRootTuple.sol";
    +import "blobstream-contracts/lib/tree/binary/BinaryMerkleProof.sol";
    +
    +contract MyRollup {
    +    IDAOracle immutable blobstreamX;
    +    bytes32[] public rollup_block_hashes;
    +
    +    constructor(IDAOracle _blobstreamX) {
    +        blobstreamX = _blobstreamX;
    +    }
    +
    +    function submitRollupBlock(
    +        bytes32 _rollup_block_hash,
    +        bytes calldata _zk_proof,
    +        uint256 _blobstreamX_nonce,
    +        DataRootTuple calldata _tuple,
    +        BinaryMerkleProof calldata _proof
    +    ) public {
    +        // Verify that the data root tuple (analog. block header) has been
    +        // attested to by the Blobstream contract.
    +        require(
    +            blobstreamX.verifyAttestation(_blobstreamX_nonce, _tuple, _proof)
    +        );
    +
    +        // Verify the ZKP (zero-knowledge proof).
    +        // _tuple.dataRoot is a public input, leaves (shares) are private inputs.
    +        require(verifyZKP(_rollup_block_hash, _zk_proof, _tuple.dataRoot));
    +
    +        // Everything checks out, append rollup block hash to list.
    +        rollup_block_hashes.push(_rollup_block_hash);
    +    }
    +
    +    function verifyZKP(
    +        bytes32 _rollup_block_hash,
    +        bytes calldata _zk_proof,
    +        bytes32 _data_root
    +    ) private pure returns (bool) {
    +        return true;
    +    }
    +}
    // SPDX-License-Identifier: Apache-2.0
    +pragma solidity ^0.8.19;
    +
    +TBD
    +import "blobstream-contracts/IDAOracle.sol";
    +import "blobstream-contracts/DataRootTuple.sol";
    +import "blobstream-contracts/lib/tree/binary/BinaryMerkleProof.sol";
    +
    +contract MyRollup {
    +    IDAOracle immutable blobstreamX;
    +    bytes32[] public rollup_block_hashes;
    +
    +    constructor(IDAOracle _blobstreamX) {
    +        blobstreamX = _blobstreamX;
    +    }
    +
    +    function submitRollupBlock(
    +        bytes32 _rollup_block_hash,
    +        bytes calldata _zk_proof,
    +        uint256 _blobstreamX_nonce,
    +        DataRootTuple calldata _tuple,
    +        BinaryMerkleProof calldata _proof
    +    ) public {
    +        // Verify that the data root tuple (analog. block header) has been
    +        // attested to by the Blobstream contract.
    +        require(
    +            blobstreamX.verifyAttestation(_blobstreamX_nonce, _tuple, _proof)
    +        );
    +
    +        // Verify the ZKP (zero-knowledge proof).
    +        // _tuple.dataRoot is a public input, leaves (shares) are private inputs.
    +        require(verifyZKP(_rollup_block_hash, _zk_proof, _tuple.dataRoot));
    +
    +        // Everything checks out, append rollup block hash to list.
    +        rollup_block_hashes.push(_rollup_block_hash);
    +    }
    +
    +    function verifyZKP(
    +        bytes32 _rollup_block_hash,
    +        bytes calldata _zk_proof,
    +        bytes32 _data_root
    +    ) private pure returns (bool) {
    +        return true;
    +    }
    +}

    Data structures

    Each DataRootTuple is a tuple of block height and data root. It is analogous to a Celestia block header. DataRootTuples are relayed in batches, committed to as a DataRootTuples root (i.e. a Merkle root of DataRootTuples).

    The BinaryMerkleProof is an RFC-6962-compliant Merkle proof. Since DataRootTuples are Merkleized in a binary Merkle tree, verifying the inclusion of a DataRootTuple against a DataRootTuples root requires verifying a Merkle inclusion proof.

    Interface

    The IDAOracle (Data Availability Oracle Interface) interface allows L2 contracts on Ethereum to query the BlobstreamX.sol contract for relayed DataRootTuples. The single interface method verifyAttestation verifies a Merkle inclusion proof that a DataRootTuple is included under a specific batch (indexed by batch nonce). In other words, analogously it verifies that a specific block header is included in the Celestia chain.

    Querying the proof

    To prove that the data was published to Celestia, check out the proof queries documentation to understand how to query the proofs from Celestia consensus nodes and make them usable in the Blobstream X verifier contract.

    Verifying data inclusion for fraud proofs

    A high-level overview of how a fraud-proof based L2 would interact with Blobstream can be found in the inclusion proofs documentation.

    The DAVerifier library is available at blobstream-contracts/lib/verifier/DAVerifier.sol, and provides functions to verify the inclusion of individual (or multiple) shares against a DataRootTuple. The library is stateless, and allows to pass an IDAOracle interface as a parameter to verify inclusion against it.

    In the DAVerifier library, we find functions that help with data inclusion verification and calculating the square size of a Celestia block. These functions work with the Blobstream X smart contract, using different proofs to check and confirm the data's availability. Let's take a closer look at these functions:

    • verifySharesToDataRootTupleRoot: This function verifies that the shares, which were posted to Celestia, were committed to by the Blobstream X smart contract. It checks that the data root was committed to by the Blobstream X smart contract and that the shares were committed to by the rows roots.
    • verifyRowRootToDataRootTupleRoot: This function verifies that a row/column root, from a Celestia block, was committed to by the Blobstream X smart contract. It checks that the data root was committed to by the Blobstream X smart contract and that the row root commits to the data root.
    • verifyMultiRowRootsToDataRootTupleRoot: This function verifies that a set of rows/columns, from a Celestia block, were committed to by the Blobstream X smart contract. It checks that the data root was committed to by the Blobstream X smart contract and that the rows roots commit to the data root.
    • computeSquareSizeFromRowProof: This function computes the Celestia block square size from a row/column root to data root binary Merkle proof. It is the user's responsibility to verify that the proof is valid and was successfully committed to using the verifyRowRootToDataRootTupleRoot() method.
    • computeSquareSizeFromShareProof: This function computes the Celestia block square size from a shares to row/column root proof. It is the user's responsibility to verify that the proof is valid and that the shares were successfully committed to using the verifySharesToDataRootTupleRoot() method.

    For an overview of a demo rollup implementation, head to the next section.

    + + + + \ No newline at end of file diff --git a/developers/blobstream-offchain.html b/developers/blobstream-offchain.html new file mode 100644 index 00000000000..f1ddc88a06f --- /dev/null +++ b/developers/blobstream-offchain.html @@ -0,0 +1,341 @@ + + + + + + Integrate with Blobstream client | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Integrate with Blobstream client

    Blobstream demo rollup

    Rollups can use Blobstream for DA by posting their data to Celestia and then proving that it was posted on Ethereum. This is done identically to how any rollup or user would post data to Celestia. Then, a zero-knowledge proof that Celestia validators have come to consensus on Celestia block headers is generated, and subsequently relayed to Ethereum to the Blobstream X smart contract.

    This demo rollup will outline (the outline is not an implementation! Please do not expect to copy and paste this code 🙂) a very simple Blobstream rollup to illustrate at a high level what this could look like.

    Defining a chain

    The first step to starting a new chain is to define the structure of the commitments that each block consists of.

    go
    type Block struct {
    +    // Data is the data of a block that is submitted to Celestia.
    +    Data `json:"Data"`
    +    // Header is the set of commitments over a block that is submitted to
    +    // Ethereum.
    +    Header `json:"Header"`
    +}
    +
    +// Data is the data of a block that is submitted to Celestia.
    +type Data struct {
    +    Txs []json.RawMessage `json:"txs"`
    +}
    +
    +// Header is the set of commitments over a block that is submitted to Ethereum.
    +type Header struct {
    +    Height             uint64    `json:"height"`
    +    Namespace          []byte    `json:"namespace"`
    +    PreviousHash       []byte    `json:"previous_hash"`
    +    Span               Span      `json:"span"`
    +    SequencerSignature Signature `json:"sequencer_signature,omitempty"`
    +}
    type Block struct {
    +    // Data is the data of a block that is submitted to Celestia.
    +    Data `json:"Data"`
    +    // Header is the set of commitments over a block that is submitted to
    +    // Ethereum.
    +    Header `json:"Header"`
    +}
    +
    +// Data is the data of a block that is submitted to Celestia.
    +type Data struct {
    +    Txs []json.RawMessage `json:"txs"`
    +}
    +
    +// Header is the set of commitments over a block that is submitted to Ethereum.
    +type Header struct {
    +    Height             uint64    `json:"height"`
    +    Namespace          []byte    `json:"namespace"`
    +    PreviousHash       []byte    `json:"previous_hash"`
    +    Span               Span      `json:"span"`
    +    SequencerSignature Signature `json:"sequencer_signature,omitempty"`
    +}

    Note the Celestia-specific structures in the header such as the Namespace and the Blobstream-specific structure called the Span. The goal of these structures is to locate the data in the Celestia block so that we can prove that data's inclusion via Blobstream if needed. Read more in the namespace specifications, and you can think of this like a chain ID. Learn more information about shares, which are small pieces of the encoded Celestia block. We use the same encoding here so that the commitments to the rollup block match those committed to by validators in the Celestia data root.

    The Span could take many forms, but in this demo, we will use the following:

    go
    // Span describes the location of the rollup block data that is posted to
    +// Celestia. This is important for other nodes to be able to prove that data in
    +// the Celestia block. This can be thought of as a pointer to some data in the
    +// Celestia block.
    +type Span struct {
    +    // CelestiaHeight is the height of the Celestia block that contains the
    +    // rollup block data.
    +    CelestiaHeight uint64 `json:"celestia_height"`
    +    // DataShareStart is the index of the first share of the rollup block data.
    +    DataShareStart uint64 `json:"share_start"`
    +    // DataShareLen is length in shares of the rollup block data. This is used
    +    // to identify all of the rollup block data in a Celestia block.
    +    DataShareLen uint64 `json:"share_end"`
    +}
    // Span describes the location of the rollup block data that is posted to
    +// Celestia. This is important for other nodes to be able to prove that data in
    +// the Celestia block. This can be thought of as a pointer to some data in the
    +// Celestia block.
    +type Span struct {
    +    // CelestiaHeight is the height of the Celestia block that contains the
    +    // rollup block data.
    +    CelestiaHeight uint64 `json:"celestia_height"`
    +    // DataShareStart is the index of the first share of the rollup block data.
    +    DataShareStart uint64 `json:"share_start"`
    +    // DataShareLen is length in shares of the rollup block data. This is used
    +    // to identify all of the rollup block data in a Celestia block.
    +    DataShareLen uint64 `json:"share_end"`
    +}

    We can then define the blockchain as a collection of blocks and some additional information about the chain such as the sequencer address.

    go
    type Blockchain struct {
    +    Blocks           []Block
    +    SequencerAddress []byte
    +    Namespace        []byte
    +}
    type Blockchain struct {
    +    Blocks           []Block
    +    SequencerAddress []byte
    +    Namespace        []byte
    +}

    Rollup sequencer

    The rollup sequencer is responsible for creating blocks and, in this demo, writing that data to Celestia and Ethereum. The rollup full node is responsible for reading that data from Celestia and Ethereum and verifying that it follows the protocol rules of that rollup.

    Therefore, we can start by first defining the reading and writing interactions rollup nodes will have with both the Celestia and Ethereum networks. The actual implementations of these interfaces are left as exercises to the reader (🤪). Assume that those implementations of these interfaces are verifying the respective chain. For the connection to Celestia, this would likely mean connecting to a Celestia light node, which can detect faults in consensus such as hidden data. For the connection to Ethereum, this would likely mean running and connecting to a full node. More information on the RPC that is exposed by a Celestia light node can be found in the RPC documentation. Additionally, if you need more information on how to run a light node, you can check out the documentation.

    go
    // CelestiaLightNodeClient summarizes the actions that a rollup that uses
    +// Blobstream for DA would need from a Celestia light node. Note that the actual
    +// connection to this light node is arbitrary, but would likely involve an RPC
    +// connection to a Celestia light node.
    +type CelestiaLightNodeClient interface {
    +    GetBlockData(Span) (Data, error)
    +    SubmitBlockData(Data) (Span, error)
    +}
    +
    +// EthereumClient summarizes the actions that a rollup that uses Blobstream for
    +// DA would need from an Ethereum client.
    +type EthereumClient interface {
    +    // GetLatestRollupHeight returns the height of the latest rollup block by
    +    // querying the appropriate contract on Ethereum.
    +    LatestRollupHeight() (uint64, error)
    +    // GetHeader returns the rollup header of a specific height.
    +    GetHeader(uint64) (Header, error)
    +    // SubmitHeader submits a header to the rollup bridge contract on Ethereum.
    +    SubmitHeader(Header) error
    +}
    // CelestiaLightNodeClient summarizes the actions that a rollup that uses
    +// Blobstream for DA would need from a Celestia light node. Note that the actual
    +// connection to this light node is arbitrary, but would likely involve an RPC
    +// connection to a Celestia light node.
    +type CelestiaLightNodeClient interface {
    +    GetBlockData(Span) (Data, error)
    +    SubmitBlockData(Data) (Span, error)
    +}
    +
    +// EthereumClient summarizes the actions that a rollup that uses Blobstream for
    +// DA would need from an Ethereum client.
    +type EthereumClient interface {
    +    // GetLatestRollupHeight returns the height of the latest rollup block by
    +    // querying the appropriate contract on Ethereum.
    +    LatestRollupHeight() (uint64, error)
    +    // GetHeader returns the rollup header of a specific height.
    +    GetHeader(uint64) (Header, error)
    +    // SubmitHeader submits a header to the rollup bridge contract on Ethereum.
    +    SubmitHeader(Header) error
    +}

    Note that here we are waiting for the head to be posted to Ethereum, however, it would likely be better to simply download that header from the p2p network or directly from the sequencer instead.

    For the purposes of this demo, we will be using a single centralized sequencer, which can be defined by simply wrapping the full node to isolate the logic to create blocks.

    A rollup full node will just consist of some representation of a blockchain along with clients to read from with Celestia and Ethereum.

    go
    type Fullnode struct {
    +    Blockchain
    +    CelestiaLightNodeClient
    +    EthereumClient
    +}
    +
    +// Sequencer wraps the demo Fullnode struct to add specific functionality for
    +// producing blocks.
    +type Sequencer struct {
    +    Fullnode
    +}
    type Fullnode struct {
    +    Blockchain
    +    CelestiaLightNodeClient
    +    EthereumClient
    +}
    +
    +// Sequencer wraps the demo Fullnode struct to add specific functionality for
    +// producing blocks.
    +type Sequencer struct {
    +    Fullnode
    +}

    Committing to data

    Typical blockchains commit to the transactions included in each block using a Merkle root. Rollups that use Blobstream for DA need to use the commitments that are relayed to the Blobstream contracts.

    For optimistic rollups, this could be as simple as referencing the data in the Celestia block, not unlike using a pointer in memory. This is what is done below via a Span in the creating blocks section. We keep track of where the data is located in the Celestia block and the sequencer signs over that location in the header. If the sequencer commits to non-existent data or an invalid state root, then the invalid transaction is first proved to be included in the Span before the rest of the fraud proof process is followed. Find more information in the inclusion proofs documentation.

    For zk rollups, this would involve creating an inclusion proof to the data root tuple root in the Blobstream contracts and then verifying that proof in the zk proof used to verify state. Find more information in the data root inclusion proof documentation.

    Also, see the documentation for the data square layout and the shares of the Celestia block to see how the data is encoded in Celestia.

    Creating blocks

    The first step in creating a block is to post the block data to Celestia. Upon confirmation of the data being included in a block, the actual location of the data in Celestia can be determined. This data is used to create a Span which is included in the header and signed over by the sequencer. This Span can be used by contracts on Ethereum that use the Blobstream contracts to prove some specific data was included.

    go
    func (s *Sequencer) ProduceBlock(txs []json.RawMessage) (Block, error) {
    +    data := Data{Txs: txs}
    +
    +    span, err := s.CelestiaLightNodeClient.SubmitBlockData(data)
    +    if err != nil {
    +        return Block{}, err
    +    }
    +
    +    var lastBlock Block
    +    if len(s.Blocks) > 0 {
    +        lastBlock = s.Blocks[len(s.Blocks)-1]
    +    }
    +
    +    header := Header{
    +        Height:       uint64(len(s.Blocks) + 1),
    +        PreviousHash: lastBlock.Header.Hash(),
    +        Namespce:     s.Namespace,
    +        Span: span,
    +    }
    +
    +    signature := s.key.Sign(header.SignBytes())
    +
    +    header.SequencerSignature = signature
    +
    +    block := Block{
    +        Data:   data,
    +        Header: header,
    +    }
    +
    +    s.AddBlock(block)
    +
    +    return block, nil
    +}
    func (s *Sequencer) ProduceBlock(txs []json.RawMessage) (Block, error) {
    +    data := Data{Txs: txs}
    +
    +    span, err := s.CelestiaLightNodeClient.SubmitBlockData(data)
    +    if err != nil {
    +        return Block{}, err
    +    }
    +
    +    var lastBlock Block
    +    if len(s.Blocks) > 0 {
    +        lastBlock = s.Blocks[len(s.Blocks)-1]
    +    }
    +
    +    header := Header{
    +        Height:       uint64(len(s.Blocks) + 1),
    +        PreviousHash: lastBlock.Header.Hash(),
    +        Namespce:     s.Namespace,
    +        Span: span,
    +    }
    +
    +    signature := s.key.Sign(header.SignBytes())
    +
    +    header.SequencerSignature = signature
    +
    +    block := Block{
    +        Data:   data,
    +        Header: header,
    +    }
    +
    +    s.AddBlock(block)
    +
    +    return block, nil
    +}

    Note that the sequencer here is not yet posting headers to Ethereum. This is because the sequencer is waiting for the commitments from the Celestia validator set (the data root tuple roots) to be relayed to the contracts. Once the contracts are updated, the sequencer can post the header to Ethereum.

    go
    func (s *Sequencer) UpdateHeaders() error {
    +    latestRollupHeight, err := s.EthereumClient.LatestRollupHeight()
    +    if err != nil {
    +        return err
    +    }
    +
    +    for i := latestRollupHeight; i <= uint64(len(s.Blocks)+1); i++ {
    +        err := s.EthereumClient.SubmitHeader(s.Blocks[i].Header)
    +        if err != nil {
    +            return err
    +        }
    +    }
    +
    +    return nil
    +}
    func (s *Sequencer) UpdateHeaders() error {
    +    latestRollupHeight, err := s.EthereumClient.LatestRollupHeight()
    +    if err != nil {
    +        return err
    +    }
    +
    +    for i := latestRollupHeight; i <= uint64(len(s.Blocks)+1); i++ {
    +        err := s.EthereumClient.SubmitHeader(s.Blocks[i].Header)
    +        if err != nil {
    +            return err
    +        }
    +    }
    +
    +    return nil
    +}

    Rollup full node

    Downloading the block

    There are a few different mechanisms that could be used to download blocks. The simplest solution and what is outlined above is for Fullnodes to wait until the blocks and the headers are posted to the respective chains, and then download each as they are posted. It would also be possible to gossip the headers ahead of time and download the rollup blocks from Celestia instead of waiting for the headers to be posted to Ethereum. It's also possible to download the headers and the block data like a normal blockchain via a gossiping network and only fall back to downloading the data and headers from Celestia and Ethereum if the gossiping network is unavailable or the sequencer is malicious.

    go
    func (f *Fullnode) AddBlock(b Block) error {
    +    // Perform validation of the block
    +    if b.Header.Height != uint64(len(f.Blocks)+1) {
    +        return fmt.Errorf("failure to add block: expected block height %d, got %d", len(f.Blocks)+1, b.Header.Height)
    +    }
    +    // Check the sequencer's signature
    +    if !b.Header.SequencerSignature.IsValid(f.SequencerAddress) {
    +        return fmt.Errorf("failure to add block: invalid sequencer signature")
    +    }
    +
    +    f.Blocks = append(f.Blocks, b)
    +    return nil
    +}
    +
    +func (f *Fullnode) GetLatestBlock() error {
    +    nextHeight := uint64(len(f.Blocks) + 1)
    +
    +    // Download the next header from ethereum before we download the block data
    +    // from Celestia. Note that we could alternatively download the header
    +    // directly from the sequencer instead of waiting.
    +    header, err := f.EthereumClient.GetHeader(nextHeight)
    +    if err != nil {
    +        return err
    +    }
    +
    +    data, err := f.CelestiaLightNodeClient.GetBlockData(header.Span)
    +    if err != nil {
    +        return err
    +    }
    +
    +    return f.AddBlock(
    +        Block{
    +            Data:   data,
    +            Header: header,
    +        },
    +    )
    +}
    func (f *Fullnode) AddBlock(b Block) error {
    +    // Perform validation of the block
    +    if b.Header.Height != uint64(len(f.Blocks)+1) {
    +        return fmt.Errorf("failure to add block: expected block height %d, got %d", len(f.Blocks)+1, b.Header.Height)
    +    }
    +    // Check the sequencer's signature
    +    if !b.Header.SequencerSignature.IsValid(f.SequencerAddress) {
    +        return fmt.Errorf("failure to add block: invalid sequencer signature")
    +    }
    +
    +    f.Blocks = append(f.Blocks, b)
    +    return nil
    +}
    +
    +func (f *Fullnode) GetLatestBlock() error {
    +    nextHeight := uint64(len(f.Blocks) + 1)
    +
    +    // Download the next header from ethereum before we download the block data
    +    // from Celestia. Note that we could alternatively download the header
    +    // directly from the sequencer instead of waiting.
    +    header, err := f.EthereumClient.GetHeader(nextHeight)
    +    if err != nil {
    +        return err
    +    }
    +
    +    data, err := f.CelestiaLightNodeClient.GetBlockData(header.Span)
    +    if err != nil {
    +        return err
    +    }
    +
    +    return f.AddBlock(
    +        Block{
    +            Data:   data,
    +            Header: header,
    +        },
    +    )
    +}

    This outline of a Blobstream rollup isn't doing execution or state transitions induced by the transactions, however, that step would occur here. If fraud is detected, the fraud proof process will begin. The only difference between the fraud proof process of a normal optimistic rollup and a rollup that uses Blobstream for DA is that the full node would first prove the fraudulent transaction was committed to by the Sequencer using the Span in the header and before proceeding with the normal process.

    More documentation

    Proving inclusion via Blobstream

    Blobstream inclusion proof docs and the verifier helper contracts.

    Submitting block data to Celestia via light node

    As linked above, use the Celestia light node RPC to submit the data to Celestia.

    Posting headers to Ethereum

    How headers are posted to Ethereum is entirely dependent upon how the rollup light client contracts work. For examples of interacting with the Ethereum blockchain programmatically, please see the go-ethereum book or one of the many other resources for submitting transactions or writing contracts.

    + + + + \ No newline at end of file diff --git a/developers/blobstream-proof-queries.html b/developers/blobstream-proof-queries.html new file mode 100644 index 00000000000..c9187341f0e --- /dev/null +++ b/developers/blobstream-proof-queries.html @@ -0,0 +1,1493 @@ + + + + + + Blobstream proofs queries | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Blobstream proofs queries

    Prerequisites

    • Access to a Celestia full consensus node RPC endpoint (or full node). The node doesn't need to be a validating node in order for the proofs to be queried. A full node is enough.

    Overview of the proof queries

    To prove the inclusion of PayForBlobs (PFB) transactions, blobs or shares, committed to in a Celestia block, we use the Celestia consensus node's RPC to query for proofs that can be verified in a rollup settlement contract via Blobstream. In fact, when a PFB transaction is included in a block, it gets separated into a PFB transaction (without the blob), and the actual data blob that it carries. These two are split into shares, which are the low level constructs of a Celestia block, and saved to the corresponding Celestia block. Learn more about shares in the shares specs.

    The two diagrams below summarize how a single share, which can contain a PFB transaction, or a part of the rollup data that was posted using a PFB, is committed to in Blobstream.

    The share is highlighted in green. R0, R1 etc, represent the respective row and column roots, the blue and pink gradients are erasure encoded data. More details on the square layout can be found in the data square layout and data structures portion of the specs.

    The Celestia square

    Square

    The commitment scheme

    Blobstream Commitment Diagram

    So to prove inclusion of a share to a Celestia block, we use Blobstream as a source of truth. Currently, we will be using the Blobstream X implementation of Blobstream, more information on Blobstream X can be found in the overview. In a nutshell, Blobstream X attests to the data posted to Celestia in the Blobstream X contract via verifying a zk-proof of the headers of a batch of Celestia blocks. Then, it keeps reference of that batch of blocks using the merkleized commitment of their (dataRoot, height) resulting in a data root tuple root. Check the above diagram which shows:

    • 0: those are the shares, that when unified, contain the PFB or the rollup data blob.
    • 1: the row and column roots are the namespace merkle tree roots over the shares. More information on the NMT in the NMT specs. These commit to the rows and columns containing the above shares.
    • 2: the data roots: which are the binary merkle tree commitment over the row and column roots. This means that if you can prove that a share is part of a row, using a namespace merkle proof. Then prove that this row is committed to by the data root. Then you can be sure that that share was published to the corresponding block.
    • 3: in order to batch multiple blocks into the same commitment, we create a commitment over the (dataRoot, height) tuple for a batch of blocks, which results in a data root tuple root. It's this commitment that gets stored in the Blobstream X smart contract.

    So, if we're able to prove that a share is part of a row, then that row is committed to by a data root. Then, prove that that data root along with its height is committed to by the data root tuple root, which gets saved to the Blobstream X contract, we can be sure that that share was committed to in the corresponding Celestia block.

    In this document, we will provide details on how to query the above proofs, and how to adapt them to be sent to a rollup contract for verification.

    Hands-on demonstration

    This part will provide the details of proof generation, and the way to make the results of the proofs queries ready to be consumed by the target rollup contract.

    NOTE

    For the go client snippets, make sure to have the following replaces in your go.mod:

    go
    // go.mod
    +    github.com/cosmos/cosmos-sdk => github.com/celestiaorg/cosmos-sdk v1.18.3-sdk-v0.46.14
    +    github.com/gogo/protobuf => github.com/regen-network/protobuf v1.3.3-alpha.regen.1
    +    github.com/syndtr/goleveldb => github.com/syndtr/goleveldb v1.0.1-0.20210819022825-2ae1ddf74ef7
    +    github.com/tendermint/tendermint => github.com/celestiaorg/celestia-core v1.32.0-tm-v0.34.29
    +
    +)
    // go.mod
    +    github.com/cosmos/cosmos-sdk => github.com/celestiaorg/cosmos-sdk v1.18.3-sdk-v0.46.14
    +    github.com/gogo/protobuf => github.com/regen-network/protobuf v1.3.3-alpha.regen.1
    +    github.com/syndtr/goleveldb => github.com/syndtr/goleveldb v1.0.1-0.20210819022825-2ae1ddf74ef7
    +    github.com/tendermint/tendermint => github.com/celestiaorg/celestia-core v1.32.0-tm-v0.34.29
    +
    +)

    Also, make sure to update the versions to match the latest github.com/celestiaorg/cosmos-sdk and github.com/celestiaorg/celestia-core versions.

    1. Data root inclusion proof

    To prove the data root is committed to by the Blobstream X smart contract, we will need to provide a Merkle proof of the data root tuple to a data root tuple root. This can be created using the data_root_inclusion_proof query.

    This endpoint allows querying a data root to data root tuple root proof. It takes a block height, a starting block, and an end block, then it generates the binary Merkle proof of the DataRootTuple, corresponding to that height, to the DataRootTupleRoot which is committed to in the Blobstream X contract.

    The endpoint can be queried using the golang client:

    go
    package main
    +
    +import (
    +	"context"
    +	"fmt"
    +	"github.com/tendermint/tendermint/rpc/client/http"
    +	"os"
    +)
    +
    +func main() {
    +	ctx := context.Background()
    +	trpc, err := http.New("tcp://localhost:26657", "/websocket")
    +	if err != nil {
    +		fmt.Println(err)
    +		os.Exit(1)
    +	}
    +	err = trpc.Start()
    +	if err != nil {
    +		fmt.Println(err)
    +		os.Exit(1)
    +	}
    +	dcProof, err := trpc.DataRootInclusionProof(ctx, 15, 10, 20)
    +	if err != nil {
    +		fmt.Println(err)
    +		os.Exit(1)
    +	}
    +	fmt.Println(dcProof.Proof.String())
    +}
    package main
    +
    +import (
    +	"context"
    +	"fmt"
    +	"github.com/tendermint/tendermint/rpc/client/http"
    +	"os"
    +)
    +
    +func main() {
    +	ctx := context.Background()
    +	trpc, err := http.New("tcp://localhost:26657", "/websocket")
    +	if err != nil {
    +		fmt.Println(err)
    +		os.Exit(1)
    +	}
    +	err = trpc.Start()
    +	if err != nil {
    +		fmt.Println(err)
    +		os.Exit(1)
    +	}
    +	dcProof, err := trpc.DataRootInclusionProof(ctx, 15, 10, 20)
    +	if err != nil {
    +		fmt.Println(err)
    +		os.Exit(1)
    +	}
    +	fmt.Println(dcProof.Proof.String())
    +}

    Full example of proving that a Celestia block was committed to by Blobstream X contract

    go
    package main
    +
    +import (
    +	"context"
    +	"fmt"
    +	"github.com/celestiaorg/celestia-app/pkg/square"
    +	"github.com/ethereum/go-ethereum/accounts/abi/bind"
    +	ethcmn "github.com/ethereum/go-ethereum/common"
    +	"github.com/ethereum/go-ethereum/ethclient"
    +	blobstreamxwrapper "github.com/succinctlabs/blobstreamx/bindings"
    +	"github.com/tendermint/tendermint/crypto/merkle"
    +	"github.com/tendermint/tendermint/rpc/client/http"
    +	"math/big"
    +	"os"
    +)
    +
    +func main() {
    +	err := verify()
    +	if err != nil {
    +		fmt.Println(err)
    +		os.Exit(1)
    +	}
    +}
    +
    +func verify() error {
    +	ctx := context.Background()
    +
    +	// start the tendermint RPC client
    +	trpc, err := http.New("tcp://localhost:26657", "/websocket")
    +	if err != nil {
    +		return err
    +	}
    +	err = trpc.Start()
    +	if err != nil {
    +		return err
    +	}
    +
    +	// get the PayForBlob transaction that contains the published blob
    +	tx, err := trpc.Tx(ctx, []byte("tx_hash"), true)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// get the block containing the PayForBlob transaction
    +	blockRes, err := trpc.Block(ctx, &tx.Height)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// get the nonce corresponding to the block height that contains
    +	// the PayForBlob transaction
    +	// since BlobstreamX emits events when new batches are submitted,
    +	// we will query the events
    +	// and look for the range committing to the blob
    +	// first, connect to an EVM RPC endpoint
    +	ethClient, err := ethclient.Dial("evm_rpc_endpoint")
    +	if err != nil {
    +		return err
    +	}
    +	defer ethClient.Close()
    +
    +	// use the BlobstreamX contract binding
    +	wrapper, err := blobstreamxwrapper.NewBlobstreamX(ethcmn.HexToAddress("contract_Address"), ethClient)
    +	if err != nil {
    +		return err
    +	}
    +
    +	LatestBlockNumber, err := ethClient.BlockNumber(context.Background())
    +	if err != nil {
    +		return err
    +	}
    +
    +	eventsIterator, err := wrapper.FilterDataCommitmentStored(
    +		&bind.FilterOpts{
    +			Context: ctx,
    +			Start: LatestBlockNumber - 90000,
    +			End: &LatestBlockNumber,
    +		},
    +		nil,
    +		nil,
    +		nil,
    +	)
    +	if err != nil {
    +		return err
    +	}
    +
    +	var event *blobstreamxwrapper.BlobstreamXDataCommitmentStored
    +	for eventsIterator.Next() {
    +		e := eventsIterator.Event
    +		if int64(e.StartBlock) <= tx.Height && tx.Height < int64(e.EndBlock) {
    +			event = &blobstreamxwrapper.BlobstreamXDataCommitmentStored{
    +				ProofNonce:     e.ProofNonce,
    +				StartBlock:     e.StartBlock,
    +				EndBlock:       e.EndBlock,
    +				DataCommitment: e.DataCommitment,
    +			}
    +			break
    +		}
    +	}
    +	if err := eventsIterator.Error(); err != nil {
    +		return err
    +	}
    +	err = eventsIterator.Close()
    +	if err != nil {
    +		return err
    +	}
    +	if event == nil {
    +		return fmt.Errorf("couldn't find range containing the transaction height")
    +	}
    +
    +	// get the block data root inclusion proof to the data root tuple root
    +	dcProof, err := trpc.DataRootInclusionProof(ctx, uint64(tx.Height), event.StartBlock, event.EndBlock)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// verify that the data root was committed to by the BlobstreamX contract
    +	committed, err := VerifyDataRootInclusion(ctx, wrapper, event.ProofNonce.Uint64(), uint64(tx.Height), blockRes.Block.DataHash, dcProof.Proof)
    +	if err != nil {
    +		return err
    +	}
    +	if committed {
    +		fmt.Println("data root was committed to by the BlobstreamX contract")
    +	} else {
    +		fmt.Println("data root was not committed to by the BlobstreamX contract")
    +		return nil
    +	}
    +    return nil
    +}
    +
    +func VerifyDataRootInclusion(
    +	_ context.Context,
    +	blobstreamXwrapper *blobstreamxwrapper.BlobstreamX,
    +	nonce uint64,
    +	height uint64,
    +	dataRoot []byte,
    +	proof merkle.Proof,
    +) (bool, error) {
    +	tuple := blobstreamxwrapper.DataRootTuple{
    +		Height:   big.NewInt(int64(height)),
    +		DataRoot: *(*[32]byte)(dataRoot),
    +	}
    +
    +	sideNodes := make([][32]byte, len(proof.Aunts))
    +	for i, aunt := range proof.Aunts {
    +		sideNodes[i] = *(*[32]byte)(aunt)
    +	}
    +	wrappedProof := blobstreamxwrapper.BinaryMerkleProof{
    +		SideNodes: sideNodes,
    +		Key:       big.NewInt(proof.Index),
    +		NumLeaves: big.NewInt(proof.Total),
    +	}
    +
    +	valid, err := blobstreamXwrapper.VerifyAttestation(
    +		&bind.CallOpts{},
    +		big.NewInt(int64(nonce)),
    +		tuple,
    +		wrappedProof,
    +	)
    +	if err != nil {
    +		return false, err
    +	}
    +	return valid, nil
    +}
    package main
    +
    +import (
    +	"context"
    +	"fmt"
    +	"github.com/celestiaorg/celestia-app/pkg/square"
    +	"github.com/ethereum/go-ethereum/accounts/abi/bind"
    +	ethcmn "github.com/ethereum/go-ethereum/common"
    +	"github.com/ethereum/go-ethereum/ethclient"
    +	blobstreamxwrapper "github.com/succinctlabs/blobstreamx/bindings"
    +	"github.com/tendermint/tendermint/crypto/merkle"
    +	"github.com/tendermint/tendermint/rpc/client/http"
    +	"math/big"
    +	"os"
    +)
    +
    +func main() {
    +	err := verify()
    +	if err != nil {
    +		fmt.Println(err)
    +		os.Exit(1)
    +	}
    +}
    +
    +func verify() error {
    +	ctx := context.Background()
    +
    +	// start the tendermint RPC client
    +	trpc, err := http.New("tcp://localhost:26657", "/websocket")
    +	if err != nil {
    +		return err
    +	}
    +	err = trpc.Start()
    +	if err != nil {
    +		return err
    +	}
    +
    +	// get the PayForBlob transaction that contains the published blob
    +	tx, err := trpc.Tx(ctx, []byte("tx_hash"), true)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// get the block containing the PayForBlob transaction
    +	blockRes, err := trpc.Block(ctx, &tx.Height)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// get the nonce corresponding to the block height that contains
    +	// the PayForBlob transaction
    +	// since BlobstreamX emits events when new batches are submitted,
    +	// we will query the events
    +	// and look for the range committing to the blob
    +	// first, connect to an EVM RPC endpoint
    +	ethClient, err := ethclient.Dial("evm_rpc_endpoint")
    +	if err != nil {
    +		return err
    +	}
    +	defer ethClient.Close()
    +
    +	// use the BlobstreamX contract binding
    +	wrapper, err := blobstreamxwrapper.NewBlobstreamX(ethcmn.HexToAddress("contract_Address"), ethClient)
    +	if err != nil {
    +		return err
    +	}
    +
    +	LatestBlockNumber, err := ethClient.BlockNumber(context.Background())
    +	if err != nil {
    +		return err
    +	}
    +
    +	eventsIterator, err := wrapper.FilterDataCommitmentStored(
    +		&bind.FilterOpts{
    +			Context: ctx,
    +			Start: LatestBlockNumber - 90000,
    +			End: &LatestBlockNumber,
    +		},
    +		nil,
    +		nil,
    +		nil,
    +	)
    +	if err != nil {
    +		return err
    +	}
    +
    +	var event *blobstreamxwrapper.BlobstreamXDataCommitmentStored
    +	for eventsIterator.Next() {
    +		e := eventsIterator.Event
    +		if int64(e.StartBlock) <= tx.Height && tx.Height < int64(e.EndBlock) {
    +			event = &blobstreamxwrapper.BlobstreamXDataCommitmentStored{
    +				ProofNonce:     e.ProofNonce,
    +				StartBlock:     e.StartBlock,
    +				EndBlock:       e.EndBlock,
    +				DataCommitment: e.DataCommitment,
    +			}
    +			break
    +		}
    +	}
    +	if err := eventsIterator.Error(); err != nil {
    +		return err
    +	}
    +	err = eventsIterator.Close()
    +	if err != nil {
    +		return err
    +	}
    +	if event == nil {
    +		return fmt.Errorf("couldn't find range containing the transaction height")
    +	}
    +
    +	// get the block data root inclusion proof to the data root tuple root
    +	dcProof, err := trpc.DataRootInclusionProof(ctx, uint64(tx.Height), event.StartBlock, event.EndBlock)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// verify that the data root was committed to by the BlobstreamX contract
    +	committed, err := VerifyDataRootInclusion(ctx, wrapper, event.ProofNonce.Uint64(), uint64(tx.Height), blockRes.Block.DataHash, dcProof.Proof)
    +	if err != nil {
    +		return err
    +	}
    +	if committed {
    +		fmt.Println("data root was committed to by the BlobstreamX contract")
    +	} else {
    +		fmt.Println("data root was not committed to by the BlobstreamX contract")
    +		return nil
    +	}
    +    return nil
    +}
    +
    +func VerifyDataRootInclusion(
    +	_ context.Context,
    +	blobstreamXwrapper *blobstreamxwrapper.BlobstreamX,
    +	nonce uint64,
    +	height uint64,
    +	dataRoot []byte,
    +	proof merkle.Proof,
    +) (bool, error) {
    +	tuple := blobstreamxwrapper.DataRootTuple{
    +		Height:   big.NewInt(int64(height)),
    +		DataRoot: *(*[32]byte)(dataRoot),
    +	}
    +
    +	sideNodes := make([][32]byte, len(proof.Aunts))
    +	for i, aunt := range proof.Aunts {
    +		sideNodes[i] = *(*[32]byte)(aunt)
    +	}
    +	wrappedProof := blobstreamxwrapper.BinaryMerkleProof{
    +		SideNodes: sideNodes,
    +		Key:       big.NewInt(proof.Index),
    +		NumLeaves: big.NewInt(proof.Total),
    +	}
    +
    +	valid, err := blobstreamXwrapper.VerifyAttestation(
    +		&bind.CallOpts{},
    +		big.NewInt(int64(nonce)),
    +		tuple,
    +		wrappedProof,
    +	)
    +	if err != nil {
    +		return false, err
    +	}
    +	return valid, nil
    +}

    2. Transaction inclusion proof

    To prove that a rollup transaction is part of the data root, we will need to provide two proofs: (1) a namespace Merkle proof of the transaction to a row root. This could be done via proving the shares that contain the transaction to the row root using a namespace Merkle proof. (2) And, a binary Merkle proof of the row root to the data root.

    These proofs can be generated using the ProveShares query.

    This endpoint allows querying a shares proof to row roots, then a row roots to data root proofs. It takes a block height, a starting share index and an end share index which define a share range. Then, two proofs are generated:

    • An NMT proof of the shares to the row roots
    • A binary Merkle proof of the row root to the data root

    NOTE

    If the share range spans multiple rows, then the proof can contain multiple NMT and binary proofs.

    The endpoint can be queried using the golang client:

    go
    	sharesProof, err := trpc.ProveShares(ctx, 15, 0, 1)
    +	if err != nil {
    +		...
    +	}
    	sharesProof, err := trpc.ProveShares(ctx, 15, 0, 1)
    +	if err != nil {
    +		...
    +	}

    Converting the proofs to be usable in the DAVerifier library

    Smart contracts that use the DAVerifier library takes the following proof format:

    solidity
    /// @notice Contains the necessary parameters to prove that some shares, which were posted to
    +/// the Celestia network, were committed to by the BlobstreamX smart contract.
    +struct SharesProof {
    +    // The shares that were committed to.
    +    bytes[] data;
    +    // The shares proof to the row roots. If the shares span multiple rows, we will have multiple nmt proofs.
    +    NamespaceMerkleMultiproof[] shareProofs;
    +    // The namespace of the shares.
    +    Namespace namespace;
    +    // The rows where the shares belong. If the shares span multiple rows, we will have multiple rows.
    +    NamespaceNode[] rowRoots;
    +    // The proofs of the rowRoots to the data root.
    +    BinaryMerkleProof[] rowProofs;
    +    // The proof of the data root tuple to the data root tuple root that was posted to the BlobstreamX contract.
    +    AttestationProof attestationProof;
    +}
    +
    +/// @notice Contains the necessary parameters needed to verify that a data root tuple
    +/// was committed to, by the BlobstreamX smart contract, at some specif nonce.
    +struct AttestationProof {
    +    // the attestation nonce that commits to the data root tuple.
    +    uint256 tupleRootNonce;
    +    // the data root tuple that was committed to.
    +    DataRootTuple tuple;
    +    // the binary Merkle proof of the tuple to the commitment.
    +    BinaryMerkleProof proof;
    +}
    /// @notice Contains the necessary parameters to prove that some shares, which were posted to
    +/// the Celestia network, were committed to by the BlobstreamX smart contract.
    +struct SharesProof {
    +    // The shares that were committed to.
    +    bytes[] data;
    +    // The shares proof to the row roots. If the shares span multiple rows, we will have multiple nmt proofs.
    +    NamespaceMerkleMultiproof[] shareProofs;
    +    // The namespace of the shares.
    +    Namespace namespace;
    +    // The rows where the shares belong. If the shares span multiple rows, we will have multiple rows.
    +    NamespaceNode[] rowRoots;
    +    // The proofs of the rowRoots to the data root.
    +    BinaryMerkleProof[] rowProofs;
    +    // The proof of the data root tuple to the data root tuple root that was posted to the BlobstreamX contract.
    +    AttestationProof attestationProof;
    +}
    +
    +/// @notice Contains the necessary parameters needed to verify that a data root tuple
    +/// was committed to, by the BlobstreamX smart contract, at some specif nonce.
    +struct AttestationProof {
    +    // the attestation nonce that commits to the data root tuple.
    +    uint256 tupleRootNonce;
    +    // the data root tuple that was committed to.
    +    DataRootTuple tuple;
    +    // the binary Merkle proof of the tuple to the commitment.
    +    BinaryMerkleProof proof;
    +}

    To construct the SharesProof, we will need the proof that we queried above, and it goes as follows:

    data

    This is the raw shares that were submitted to Celestia in the bytes format. If we take the example blob that was submitted in the RollupInclusionProofs.t.sol, we can convert it to bytes using the abi.encode(...) as done for this variable. This can be gotten from the above result of the transaction inclusion proof query in the field data.

    shareProofs

    This is the shares proof to the row roots. These can contain multiple proofs if the shares containing the blob span across multiple rows. To construct them, we will use the result of the transaction inclusion proof section.

    While the NamespaceMerkleMultiproof being:

    solidity
    /// @notice Namespace Merkle Tree Multiproof structure. Proves multiple leaves.
    +struct NamespaceMerkleMultiproof {
    +    // The beginning key of the leaves to verify.
    +    uint256 beginKey;
    +    // The ending key of the leaves to verify.
    +    uint256 endKey;
    +    // List of side nodes to verify and calculate tree.
    +    NamespaceNode[] sideNodes;
    +}
    /// @notice Namespace Merkle Tree Multiproof structure. Proves multiple leaves.
    +struct NamespaceMerkleMultiproof {
    +    // The beginning key of the leaves to verify.
    +    uint256 beginKey;
    +    // The ending key of the leaves to verify.
    +    uint256 endKey;
    +    // List of side nodes to verify and calculate tree.
    +    NamespaceNode[] sideNodes;
    +}

    So, we can construct the NamespaceMerkleMultiproof with the following mapping:

    • beginKey in the Solidity struct == start in the query response

    • endKey in the Solidity struct == end in the query response

    • sideNodes in the Solidity struct == nodes in the query response

    • The NamespaceNode, which is the type of the sideNodes, is defined as follows:

    solidity
    /// @notice Namespace Merkle Tree node.
    +struct NamespaceNode {
    +    // Minimum namespace.
    +    Namespace min;
    +    // Maximum namespace.
    +    Namespace max;
    +    // Node value.
    +    bytes32 digest;
    +}
    /// @notice Namespace Merkle Tree node.
    +struct NamespaceNode {
    +    // Minimum namespace.
    +    Namespace min;
    +    // Maximum namespace.
    +    Namespace max;
    +    // Node value.
    +    bytes32 digest;
    +}

    So, we construct a NamespaceNode via taking the values from the nodes field in the query response, we convert them from base64 to hex, then we use the following mapping:

    • min == the first 29 bytes in the decoded value
    • max == the second 29 bytes in the decoded value
    • digest == the remaining 32 bytes in the decoded value

    The min and max are Namespace type which is:

    solidity
    /// @notice A representation of the Celestia-app namespace ID and its version.
    +/// See: https://celestiaorg.github.io/celestia-app/specs/namespace.html
    +struct Namespace {
    +    // The namespace version.
    +    bytes1 version;
    +    // The namespace ID.
    +    bytes28 id;
    +}
    /// @notice A representation of the Celestia-app namespace ID and its version.
    +/// See: https://celestiaorg.github.io/celestia-app/specs/namespace.html
    +struct Namespace {
    +    // The namespace version.
    +    bytes1 version;
    +    // The namespace ID.
    +    bytes28 id;
    +}

    So, to construct them, we separate the 29 bytes in the decoded value to:

    • first byte: version
    • remaining 28 bytes: id

    An example of doing this can be found in the RollupInclusionProofs.t.sol test.

    A golang helper that can be used to make this conversion is as follows:

    go
    func toNamespaceMerkleMultiProofs(proofs []*tmproto.NMTProof) []client.NamespaceMerkleMultiproof {
    +	shareProofs := make([]client.NamespaceMerkleMultiproof, len(proofs))
    +	for i, proof := range proofs {
    +		sideNodes := make([]client.NamespaceNode, len(proof.Nodes))
    +		for j, node := range proof.Nodes {
    +			sideNodes[j] = *toNamespaceNode(node)
    +		}
    +		shareProofs[i] = client.NamespaceMerkleMultiproof{
    +			BeginKey:  big.NewInt(int64(proof.Start)),
    +			EndKey:    big.NewInt(int64(proof.End)),
    +			SideNodes: sideNodes,
    +		}
    +	}
    +	return shareProofs
    +}
    +
    +func minNamespace(innerNode []byte) *client.Namespace {
    +	version := innerNode[0]
    +	var id [28]byte
    +	copy(id[:], innerNode[1:29])
    +	return &client.Namespace{
    +		Version: [1]byte{version},
    +		Id:      id,
    +	}
    +}
    +
    +func maxNamespace(innerNode []byte) *client.Namespace {
    +	version := innerNode[29]
    +	var id [28]byte
    +	copy(id[:], innerNode[30:58])
    +	return &client.Namespace{
    +		Version: [1]byte{version},
    +		Id:      id,
    +	}
    +}
    +
    +func toNamespaceNode(node []byte) *client.NamespaceNode {
    +	minNs := minNamespace(node)
    +	maxNs := maxNamespace(node)
    +	var digest [32]byte
    +	copy(digest[:], node[58:])
    +	return &client.NamespaceNode{
    +		Min:    *minNs,
    +		Max:    *maxNs,
    +		Digest: digest,
    +	}
    +}
    func toNamespaceMerkleMultiProofs(proofs []*tmproto.NMTProof) []client.NamespaceMerkleMultiproof {
    +	shareProofs := make([]client.NamespaceMerkleMultiproof, len(proofs))
    +	for i, proof := range proofs {
    +		sideNodes := make([]client.NamespaceNode, len(proof.Nodes))
    +		for j, node := range proof.Nodes {
    +			sideNodes[j] = *toNamespaceNode(node)
    +		}
    +		shareProofs[i] = client.NamespaceMerkleMultiproof{
    +			BeginKey:  big.NewInt(int64(proof.Start)),
    +			EndKey:    big.NewInt(int64(proof.End)),
    +			SideNodes: sideNodes,
    +		}
    +	}
    +	return shareProofs
    +}
    +
    +func minNamespace(innerNode []byte) *client.Namespace {
    +	version := innerNode[0]
    +	var id [28]byte
    +	copy(id[:], innerNode[1:29])
    +	return &client.Namespace{
    +		Version: [1]byte{version},
    +		Id:      id,
    +	}
    +}
    +
    +func maxNamespace(innerNode []byte) *client.Namespace {
    +	version := innerNode[29]
    +	var id [28]byte
    +	copy(id[:], innerNode[30:58])
    +	return &client.Namespace{
    +		Version: [1]byte{version},
    +		Id:      id,
    +	}
    +}
    +
    +func toNamespaceNode(node []byte) *client.NamespaceNode {
    +	minNs := minNamespace(node)
    +	maxNs := maxNamespace(node)
    +	var digest [32]byte
    +	copy(digest[:], node[58:])
    +	return &client.NamespaceNode{
    +		Min:    *minNs,
    +		Max:    *maxNs,
    +		Digest: digest,
    +	}
    +}

    with proofs being sharesProof.ShareProofs.

    namespace

    Which is the namespace used by the rollup when submitting data to Celestia. As described above, it can be constructed as follows:

    solidity
    /// @notice A representation of the Celestia-app namespace ID and its version.
    +/// See: https://celestiaorg.github.io/celestia-app/specs/namespace.html
    +struct Namespace {
    +    // The namespace version.
    +    bytes1 version;
    +    // The namespace ID.
    +    bytes28 id;
    +}
    /// @notice A representation of the Celestia-app namespace ID and its version.
    +/// See: https://celestiaorg.github.io/celestia-app/specs/namespace.html
    +struct Namespace {
    +    // The namespace version.
    +    bytes1 version;
    +    // The namespace ID.
    +    bytes28 id;
    +}

    Via taking the namespace value from the prove_shares query response, decoding it from base64 to hex, then:

    • first byte: version
    • remaining 28 bytes: id

    An example can be found in the RollupInclusionProofs.t.sol test.

    A method to convert to namespace, provided that the namespace size is 29, is as follows:

    go
    func namespace(namespaceID []byte, version uint8) *client.Namespace {
    +	var id [28]byte
    +	copy(id[:], namespaceID)
    +	return &client.Namespace{
    +		Version: [1]byte{version},
    +		Id:      id,
    +	}
    +}
    func namespace(namespaceID []byte, version uint8) *client.Namespace {
    +	var id [28]byte
    +	copy(id[:], namespaceID)
    +	return &client.Namespace{
    +		Version: [1]byte{version},
    +		Id:      id,
    +	}
    +}

    with namespace being sharesProof.NamespaceID.

    rowRoots

    Which are the roots of the rows where the shares containing the Rollup data are localised.

    In golang, the proof can be converted as follows:

    go
    func toRowRoots(roots []bytes.HexBytes) []client.NamespaceNode {
    +	rowRoots := make([]client.NamespaceNode, len(roots))
    +	for i, root := range roots {
    +		rowRoots[i] = *toNamespaceNode(root.Bytes())
    +	}
    +	return rowRoots
    +}
    func toRowRoots(roots []bytes.HexBytes) []client.NamespaceNode {
    +	rowRoots := make([]client.NamespaceNode, len(roots))
    +	for i, root := range roots {
    +		rowRoots[i] = *toNamespaceNode(root.Bytes())
    +	}
    +	return rowRoots
    +}

    with roots being sharesProof.RowProof.RowRoots.

    rowProofs

    These are the proofs of the rows to the data root. They are of type BinaryMerkleProof:

    solidity
    /// @notice Merkle Tree Proof structure.
    +struct BinaryMerkleProof {
    +    // List of side nodes to verify and calculate tree.
    +    bytes32[] sideNodes;
    +    // The key of the leaf to verify.
    +    uint256 key;
    +    // The number of leaves in the tree
    +    uint256 numLeaves;
    +}
    /// @notice Merkle Tree Proof structure.
    +struct BinaryMerkleProof {
    +    // List of side nodes to verify and calculate tree.
    +    bytes32[] sideNodes;
    +    // The key of the leaf to verify.
    +    uint256 key;
    +    // The number of leaves in the tree
    +    uint256 numLeaves;
    +}

    To construct them, we take the response of the prove_shares query, and do the following mapping:

    • key in the Solidity struct == index in the query response
    • numLeaves in the Solidity struct == total in the query response
    • sideNodes in the Solidity struct == aunts in the query response

    The type of the sideNodes is a bytes32.

    An example can be found in the RollupInclusionProofs.t.sol test.

    A golang helper to convert the row proofs is as follows:

    go
    func toRowProofs(proofs []*merkle.Proof) []client.BinaryMerkleProof {
    +	rowProofs := make([]client.BinaryMerkleProof, len(proofs))
    +	for i, proof := range proofs {
    +		sideNodes := make( [][32]byte, len(proof.Aunts))
    +		for j, sideNode :=  range proof.Aunts {
    +			var bzSideNode [32]byte
    +			copy(bzSideNode[:], sideNode)
    +			sideNodes[j] = bzSideNode
    +		}
    + 		rowProofs[i] = client.BinaryMerkleProof{
    +			SideNodes: sideNodes,
    +			Key:       big.NewInt(proof.Index),
    +			NumLeaves: big.NewInt(proof.Total),
    +		}
    +	}
    +	return rowProofs
    +}
    func toRowProofs(proofs []*merkle.Proof) []client.BinaryMerkleProof {
    +	rowProofs := make([]client.BinaryMerkleProof, len(proofs))
    +	for i, proof := range proofs {
    +		sideNodes := make( [][32]byte, len(proof.Aunts))
    +		for j, sideNode :=  range proof.Aunts {
    +			var bzSideNode [32]byte
    +			copy(bzSideNode[:], sideNode)
    +			sideNodes[j] = bzSideNode
    +		}
    + 		rowProofs[i] = client.BinaryMerkleProof{
    +			SideNodes: sideNodes,
    +			Key:       big.NewInt(proof.Index),
    +			NumLeaves: big.NewInt(proof.Total),
    +		}
    +	}
    +	return rowProofs
    +}

    with proofs being sharesProof.RowProof.Proofs.

    attestationProof

    This is the proof of the data root to the data root tuple root, which is committed to in the Blobstream X contract:

    solidity
    /// @notice Contains the necessary parameters needed to verify that a data root tuple
    +/// was committed to, by the BlobstreamX smart contract, at some specif nonce.
    +struct AttestationProof {
    +    // the attestation nonce that commits to the data root tuple.
    +    uint256 tupleRootNonce;
    +    // the data root tuple that was committed to.
    +    DataRootTuple tuple;
    +    // the binary Merkle proof of the tuple to the commitment.
    +    BinaryMerkleProof proof;
    +}
    /// @notice Contains the necessary parameters needed to verify that a data root tuple
    +/// was committed to, by the BlobstreamX smart contract, at some specif nonce.
    +struct AttestationProof {
    +    // the attestation nonce that commits to the data root tuple.
    +    uint256 tupleRootNonce;
    +    // the data root tuple that was committed to.
    +    DataRootTuple tuple;
    +    // the binary Merkle proof of the tuple to the commitment.
    +    BinaryMerkleProof proof;
    +}
    • tupleRootNonce: the nonce at which Blobstream X committed to the batch containing the block containing the data.
    • tuple: the DataRootTuple of the block:
    solidity
    /// @notice A tuple of data root with metadata. Each data root is associated
    +///  with a Celestia block height.
    +/// @dev `availableDataRoot` in
    +///  https://github.com/celestiaorg/celestia-specs/blob/master/src/specs/data_structures.md#header
    +struct DataRootTuple {
    +    // Celestia block height the data root was included in.
    +    // Genesis block is height = 0.
    +    // First queryable block is height = 1.
    +    uint256 height;
    +    // Data root.
    +    bytes32 dataRoot;
    +}
    /// @notice A tuple of data root with metadata. Each data root is associated
    +///  with a Celestia block height.
    +/// @dev `availableDataRoot` in
    +///  https://github.com/celestiaorg/celestia-specs/blob/master/src/specs/data_structures.md#header
    +struct DataRootTuple {
    +    // Celestia block height the data root was included in.
    +    // Genesis block is height = 0.
    +    // First queryable block is height = 1.
    +    uint256 height;
    +    // Data root.
    +    bytes32 dataRoot;
    +}

    which comprises a dataRoot, i.e. the block containing the Rollup data data root, and the height which is the height of that block.

    • proof: the BinaryMerkleProof of the data root tuple to the data root tuple root. Constructing it is similar to constructing the row roots to data root proof in the rowProofs section.

    An example can be found in the RollupInclusionProofs.t.sol test.

    A golang helper to create an attestation proof:

    go
    func toAttestationProof(
    +	nonce uint64,
    +	height uint64,
    +	blockDataRoot [32]byte,
    +	dataRootInclusionProof merkle.Proof,
    +) client.AttestationProof {
    +	sideNodes := make( [][32]byte, len(dataRootInclusionProof.Aunts))
    +	for i, sideNode :=  range dataRootInclusionProof.Aunts {
    +		var bzSideNode [32]byte
    +		copy(bzSideNode[:], sideNode)
    +		sideNodes[i] = bzSideNode
    +	}
    +
    +	return client.AttestationProof{
    +		TupleRootNonce: big.NewInt(int64(nonce)),
    +		Tuple:          client.DataRootTuple{
    +			Height:   big.NewInt(int64(height)),
    +			DataRoot: blockDataRoot,
    +		},
    +		Proof:          client.BinaryMerkleProof{
    +			SideNodes: sideNodes,
    +			Key:       big.NewInt(dataRootInclusionProof.Index),
    +			NumLeaves: big.NewInt(dataRootInclusionProof.Total),
    +		},
    +	}
    +}
    func toAttestationProof(
    +	nonce uint64,
    +	height uint64,
    +	blockDataRoot [32]byte,
    +	dataRootInclusionProof merkle.Proof,
    +) client.AttestationProof {
    +	sideNodes := make( [][32]byte, len(dataRootInclusionProof.Aunts))
    +	for i, sideNode :=  range dataRootInclusionProof.Aunts {
    +		var bzSideNode [32]byte
    +		copy(bzSideNode[:], sideNode)
    +		sideNodes[i] = bzSideNode
    +	}
    +
    +	return client.AttestationProof{
    +		TupleRootNonce: big.NewInt(int64(nonce)),
    +		Tuple:          client.DataRootTuple{
    +			Height:   big.NewInt(int64(height)),
    +			DataRoot: blockDataRoot,
    +		},
    +		Proof:          client.BinaryMerkleProof{
    +			SideNodes: sideNodes,
    +			Key:       big.NewInt(dataRootInclusionProof.Index),
    +			NumLeaves: big.NewInt(dataRootInclusionProof.Total),
    +		},
    +	}
    +}

    with the nonce being the attestation nonce, which can be retrieved using BlobstreamX contract events. Check below for an example. And height being the Celestia Block height that contains the rollup data, along with the blockDataRoot being the data root of the block height. Finally, dataRootInclusionProof is the Celestia block data root inclusion proof to the data root tuple root that was queried at the beginning of this page.

    If the dataRoot or the tupleRootNonce is unknown during the verification:

    • dataRoot: can be queried using the /block?height=15 query (15 in this example endpoint), and taking the data_hash field from the response.
    • tupleRootNonce: can be retried via querying the BlobstreamXDataCommitmentStored events from the BlobstreamX contract and looking for the nonce attesting to the corresponding data. An example:
    go
    	// get the nonce corresponding to the block height that contains the PayForBlob transaction
    +	// since BlobstreamX emits events when new batches are submitted, we will query the events
    +	// and look for the range committing to the blob
    +	// first, connect to an EVM RPC endpoint
    +	ethClient, err := ethclient.Dial("evm_rpc_endpoint")
    +	if err != nil {
    +		return err
    +	}
    +	defer ethClient.Close()
    +
    +	// use the BlobstreamX contract binding
    +	wrapper, err := blobstreamxwrapper.NewBlobstreamX(ethcmn.HexToAddress("contract_Address"), ethClient)
    +	if err != nil {
    +		return err
    +	}
    +
    +	LatestBlockNumber, err := ethClient.BlockNumber(ctx)
    +	if err != nil {
    +		return err
    +	}
    +
    +	eventsIterator, err := wrapper.FilterDataCommitmentStored(
    +		&bind.FilterOpts{
    +			Context: ctx,
    +			Start: LatestBlockNumber - 90000, // 90000 can be replaced with the range of EVM blocks to look for the events in
    +			End: &LatestBlockNumber,
    +		},
    +		nil,
    +		nil,
    +		nil,
    +	)
    +	if err != nil {
    +		return err
    +	}
    +
    +	var event *blobstreamxwrapper.BlobstreamXDataCommitmentStored
    +	for eventsIterator.Next() {
    +		e := eventsIterator.Event
    +		if int64(e.StartBlock) <= tx.Height && tx.Height < int64(e.EndBlock) {
    +			event = &blobstreamxwrapper.BlobstreamXDataCommitmentStored{
    +				ProofNonce:     e.ProofNonce,
    +				StartBlock:     e.StartBlock,
    +				EndBlock:       e.EndBlock,
    +				DataCommitment: e.DataCommitment,
    +			}
    +			break
    +		}
    +	}
    +	if err := eventsIterator.Error(); err != nil {
    +		return err
    +	}
    +	err = eventsIterator.Close()
    +	if err != nil {
    +		return err
    +	}
    +	if event == nil {
    +		return fmt.Errorf("couldn't find range containing the block height")
    +	}
    	// get the nonce corresponding to the block height that contains the PayForBlob transaction
    +	// since BlobstreamX emits events when new batches are submitted, we will query the events
    +	// and look for the range committing to the blob
    +	// first, connect to an EVM RPC endpoint
    +	ethClient, err := ethclient.Dial("evm_rpc_endpoint")
    +	if err != nil {
    +		return err
    +	}
    +	defer ethClient.Close()
    +
    +	// use the BlobstreamX contract binding
    +	wrapper, err := blobstreamxwrapper.NewBlobstreamX(ethcmn.HexToAddress("contract_Address"), ethClient)
    +	if err != nil {
    +		return err
    +	}
    +
    +	LatestBlockNumber, err := ethClient.BlockNumber(ctx)
    +	if err != nil {
    +		return err
    +	}
    +
    +	eventsIterator, err := wrapper.FilterDataCommitmentStored(
    +		&bind.FilterOpts{
    +			Context: ctx,
    +			Start: LatestBlockNumber - 90000, // 90000 can be replaced with the range of EVM blocks to look for the events in
    +			End: &LatestBlockNumber,
    +		},
    +		nil,
    +		nil,
    +		nil,
    +	)
    +	if err != nil {
    +		return err
    +	}
    +
    +	var event *blobstreamxwrapper.BlobstreamXDataCommitmentStored
    +	for eventsIterator.Next() {
    +		e := eventsIterator.Event
    +		if int64(e.StartBlock) <= tx.Height && tx.Height < int64(e.EndBlock) {
    +			event = &blobstreamxwrapper.BlobstreamXDataCommitmentStored{
    +				ProofNonce:     e.ProofNonce,
    +				StartBlock:     e.StartBlock,
    +				EndBlock:       e.EndBlock,
    +				DataCommitment: e.DataCommitment,
    +			}
    +			break
    +		}
    +	}
    +	if err := eventsIterator.Error(); err != nil {
    +		return err
    +	}
    +	err = eventsIterator.Close()
    +	if err != nil {
    +		return err
    +	}
    +	if event == nil {
    +		return fmt.Errorf("couldn't find range containing the block height")
    +	}

    Listening for new data commitments

    For listening for new BlobstreamXDataCommitmentStored events, sequencers can use the WatchDataCommitmentStored as follows:

    go
        ethClient, err := ethclient.Dial("evm_rpc")
    +    if err != nil {
    +	    return err
    +    }
    +    defer ethClient.Close()
    +    blobstreamWrapper, err := blobstreamxwrapper.NewBlobstreamXFilterer(ethcmn.HexToAddress("contract_address"), ethClient)
    +    if err != nil {
    +	    return err
    +    }
    +
    +    eventsChan := make(chan *blobstreamxwrapper.BlobstreamXDataCommitmentStored, 100)
    +    subscription, err := blobstreamWrapper.WatchDataCommitmentStored(
    +	    &bind.WatchOpts{
    +			Context: ctx,
    +        },
    +	    eventsChan,
    +	    nil,
    +	    nil,
    +	    nil,
    +	)
    +    if err != nil {
    +	    return err
    +    }
    +    defer subscription.Unsubscribe()
    +
    +    for {
    +	    select {
    +	    case <-ctx.Done():
    +		    return ctx.Err()
    +		case err := <-subscription.Err():
    +			return err
    +		case event := <-eventsChan:
    +			// process the event
    +		    fmt.Println(event)
    +	    }
    +    }
        ethClient, err := ethclient.Dial("evm_rpc")
    +    if err != nil {
    +	    return err
    +    }
    +    defer ethClient.Close()
    +    blobstreamWrapper, err := blobstreamxwrapper.NewBlobstreamXFilterer(ethcmn.HexToAddress("contract_address"), ethClient)
    +    if err != nil {
    +	    return err
    +    }
    +
    +    eventsChan := make(chan *blobstreamxwrapper.BlobstreamXDataCommitmentStored, 100)
    +    subscription, err := blobstreamWrapper.WatchDataCommitmentStored(
    +	    &bind.WatchOpts{
    +			Context: ctx,
    +        },
    +	    eventsChan,
    +	    nil,
    +	    nil,
    +	    nil,
    +	)
    +    if err != nil {
    +	    return err
    +    }
    +    defer subscription.Unsubscribe()
    +
    +    for {
    +	    select {
    +	    case <-ctx.Done():
    +		    return ctx.Err()
    +		case err := <-subscription.Err():
    +			return err
    +		case event := <-eventsChan:
    +			// process the event
    +		    fmt.Println(event)
    +	    }
    +    }

    Then, new proofs can be created as documented above using the new data commitments contained in the received events.

    Example rollup that uses the DAVerifier

    An example rollup that uses the DAVerifier can be as simple as:

    solidity
    pragma solidity ^0.8.22;
    +
    +import {DAVerifier} from "@blobstream/lib/verifier/DAVerifier.sol";
    +import {IDAOracle} from "@blobstream/IDAOracle.sol";
    +
    +contract SimpleRollup {
    +    IDAOracle bridge;
    +    ...
    +    function submitFraudProof(SharesProof memory _sharesProof, bytes32 _root) public {
    +        // (1) verify that the data is committed to by BlobstreamX contract
    +        (bool committedTo, DAVerifier.ErrorCodes err) = DAVerifier.verifySharesToDataRootTupleRoot(bridge, _sharesProof, _root);
    +        if (!committedTo) {
    +            revert("the data was not committed to by Blobstream");
    +        }
    +        // (2) verify that the data is part of the rollup block
    +        // (3) parse the data
    +        // (4) verify invalid state transition
    +        // (5) effects
    +    }
    +}
    pragma solidity ^0.8.22;
    +
    +import {DAVerifier} from "@blobstream/lib/verifier/DAVerifier.sol";
    +import {IDAOracle} from "@blobstream/IDAOracle.sol";
    +
    +contract SimpleRollup {
    +    IDAOracle bridge;
    +    ...
    +    function submitFraudProof(SharesProof memory _sharesProof, bytes32 _root) public {
    +        // (1) verify that the data is committed to by BlobstreamX contract
    +        (bool committedTo, DAVerifier.ErrorCodes err) = DAVerifier.verifySharesToDataRootTupleRoot(bridge, _sharesProof, _root);
    +        if (!committedTo) {
    +            revert("the data was not committed to by Blobstream");
    +        }
    +        // (2) verify that the data is part of the rollup block
    +        // (3) parse the data
    +        // (4) verify invalid state transition
    +        // (5) effects
    +    }
    +}

    Then, you can submit the fraud proof using golang as follows:

    go
    package main
    +
    +import (
    +	"context"
    +	"fmt"
    +	"github.com/celestiaorg/celestia-app/pkg/square"
    +	"github.com/celestiaorg/celestia-app/x/qgb/client"
    +	"github.com/ethereum/go-ethereum/accounts/abi/bind"
    +	ethcmn "github.com/ethereum/go-ethereum/common"
    +	"github.com/ethereum/go-ethereum/ethclient"
    +	blobstreamxwrapper "github.com/succinctlabs/blobstreamx/bindings"
    +	"github.com/tendermint/tendermint/crypto/merkle"
    +	"github.com/tendermint/tendermint/libs/bytes"
    +	tmproto "github.com/tendermint/tendermint/proto/tendermint/types"
    +	"github.com/tendermint/tendermint/rpc/client/http"
    +	"github.com/tendermint/tendermint/types"
    +	"math/big"
    +	"os"
    +)
    +
    +func main() {
    +	err := verify()
    +	if err != nil {
    +		fmt.Println(err)
    +		os.Exit(1)
    +	}
    +}
    +
    +func verify() error {
    +	ctx := context.Background()
    +
    +
    +	// ...
    +	// check the first section for this part of the implementation
    +
    +	// get the nonce corresponding to the block height that contains the PayForBlob transaction
    +	// since Blobstream X emits events when new batches are submitted, we will query the events
    +	// and look for the range committing to the blob
    +	// first, connect to an EVM RPC endpoint
    +	ethClient, err := ethclient.Dial("evm_rpc_endpoint")
    +	if err != nil {
    +		return err
    +	}
    +	defer ethClient.Close()
    +
    +	// ...
    +	// check the first section for this part of the implementation
    +
    +	// now we will create the shares proof to be verified by the SimpleRollup
    +	// contract that uses the DAVerifier library
    +
    +	// get the proof of the shares containing the blob to the data root
    +	sharesProof, err := trpc.ProveShares(ctx, 16, uint64(blobShareRange.Start), uint64(blobShareRange.End))
    +	if err != nil {
    +		return err
    +	}
    +
    +	// use the SimpleRollup contract binding to submit to it a fraud proof
    +	simpleRollupWrapper, err := client.NewWrappers(ethcmn.HexToAddress("contract_Address"), ethClient)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// submit the fraud proof containing the share data that had the invalid state transition for example
    +	// along with its proof
    +	err = submitFraudProof(
    +		ctx,
    +		simpleRollupWrapper,
    +		sharesProof,
    +		event.ProofNonce.Uint64(),
    +		uint64(tx.Height),
    +		dcProof.Proof,
    +		blockRes.Block.DataHash,
    +	)
    +
    +	return nil
    +}
    +
    +func submitFraudProof(
    +	ctx context.Context,
    +	simpleRollup *client.Wrappers,
    +	sharesProof types.ShareProof,
    +	nonce uint64,
    +	height uint64,
    +	dataRootInclusionProof merkle.Proof,
    +	dataRoot []byte,
    +) error {
    +	var blockDataRoot [32]byte
    +	copy(blockDataRoot[:], dataRoot)
    +	tx, err := simpleRollup.SubmitFraudProof(
    +		&bind.TransactOpts{
    +			Context: ctx,
    +		},
    +		client.SharesProof{
    +			Data:             sharesProof.Data,
    +			ShareProofs:      toNamespaceMerkleMultiProofs(sharesProof.ShareProofs),
    +			Namespace:        *namespace(sharesProof.NamespaceID),
    +			RowRoots:         toRowRoots(sharesProof.RowProof.RowRoots),
    +			RowProofs:        toRowProofs(sharesProof.RowProof.Proofs),
    +			AttestationProof: toAttestationProof(nonce, height, blockDataRoot, dataRootInclusionProof),
    +		},
    +		blockDataRoot,
    +	)
    +	if err != nil {
    +		return err
    +	}
    +	// wait for transaction
    +}
    +
    +func toAttestationProof(
    +	nonce uint64,
    +	height uint64,
    +	blockDataRoot [32]byte,
    +	dataRootInclusionProof merkle.Proof,
    +) client.AttestationProof {
    +	sideNodes := make( [][32]byte, len(dataRootInclusionProof.Aunts))
    +	for i, sideNode :=  range dataRootInclusionProof.Aunts {
    +		var bzSideNode [32]byte
    +		for k, b := range sideNode {
    +			bzSideNode[k] = b
    +		}
    +		sideNodes[i] = bzSideNode
    +	}
    +
    +	return client.AttestationProof{
    +		TupleRootNonce: big.NewInt(int64(nonce)),
    +		Tuple:          client.DataRootTuple{
    +			Height:   big.NewInt(int64(height)),
    +			DataRoot: blockDataRoot,
    +		},
    +		Proof:          client.BinaryMerkleProof{
    +			SideNodes: sideNodes,
    +			Key:       big.NewInt(dataRootInclusionProof.Index),
    +			NumLeaves: big.NewInt(dataRootInclusionProof.Total),
    +		},
    +	}
    +}
    +
    +func toRowRoots(roots []bytes.HexBytes) []client.NamespaceNode {
    +	rowRoots := make([]client.NamespaceNode, len(roots))
    +	for i, root := range roots {
    +		rowRoots[i] = *toNamespaceNode(root.Bytes())
    +	}
    +	return rowRoots
    +}
    +
    +func toRowProofs(proofs []*merkle.Proof) []client.BinaryMerkleProof {
    +	rowProofs := make([]client.BinaryMerkleProof, len(proofs))
    +	for i, proof := range proofs {
    +		sideNodes := make( [][32]byte, len(proof.Aunts))
    +		for j, sideNode :=  range proof.Aunts {
    +			var bzSideNode [32]byte
    +			for k, b := range sideNode {
    +				bzSideNode[k] = b
    +			}
    +			sideNodes[j] = bzSideNode
    +		}
    + 		rowProofs[i] = client.BinaryMerkleProof{
    +			SideNodes: sideNodes,
    +			Key:       big.NewInt(proof.Index),
    +			NumLeaves: big.NewInt(proof.Total),
    +		}
    +	}
    +	return rowProofs
    +}
    +
    +func toNamespaceMerkleMultiProofs(proofs []*tmproto.NMTProof) []client.NamespaceMerkleMultiproof {
    +	shareProofs := make([]client.NamespaceMerkleMultiproof, len(proofs))
    +	for i, proof := range proofs {
    +		sideNodes := make([]client.NamespaceNode, len(proof.Nodes))
    +		for j, node := range proof.Nodes {
    +			sideNodes[j] = *toNamespaceNode(node)
    +		}
    +		shareProofs[i] = client.NamespaceMerkleMultiproof{
    +			BeginKey:  big.NewInt(int64(proof.Start)),
    +			EndKey:    big.NewInt(int64(proof.End)),
    +			SideNodes: sideNodes,
    +		}
    +	}
    +	return shareProofs
    +}
    +
    +func minNamespace(innerNode []byte) *client.Namespace {
    +	version := innerNode[0]
    +	var id [28]byte
    +	for i, b := range innerNode[1:28] {
    +		id[i] = b
    +	}
    +	return &client.Namespace{
    +		Version: [1]byte{version},
    +		Id:      id,
    +	}
    +}
    +
    +func maxNamespace(innerNode []byte) *client.Namespace {
    +	version := innerNode[29]
    +	var id [28]byte
    +	for i, b := range innerNode[30:57] {
    +		id[i] = b
    +	}
    +	return &client.Namespace{
    +		Version: [1]byte{version},
    +		Id:      id,
    +	}
    +}
    +
    +func toNamespaceNode(node []byte) *client.NamespaceNode {
    +	minNs := minNamespace(node)
    +	maxNs := maxNamespace(node)
    +	var digest [32]byte
    +	for i, b := range node[58:] {
    +		digest[i] = b
    +	}
    +	return &client.NamespaceNode{
    +		Min:    *minNs,
    +		Max:    *maxNs,
    +		Digest: digest,
    +	}
    +}
    +
    +func namespace(namespaceID []byte) *client.Namespace {
    +	version := namespaceID[0]
    +	var id [28]byte
    +	for i, b := range namespaceID[1:] {
    +		id[i] = b
    +	}
    +	return &client.Namespace{
    +		Version: [1]byte{version},
    +		Id:      id,
    +	}
    +}
    package main
    +
    +import (
    +	"context"
    +	"fmt"
    +	"github.com/celestiaorg/celestia-app/pkg/square"
    +	"github.com/celestiaorg/celestia-app/x/qgb/client"
    +	"github.com/ethereum/go-ethereum/accounts/abi/bind"
    +	ethcmn "github.com/ethereum/go-ethereum/common"
    +	"github.com/ethereum/go-ethereum/ethclient"
    +	blobstreamxwrapper "github.com/succinctlabs/blobstreamx/bindings"
    +	"github.com/tendermint/tendermint/crypto/merkle"
    +	"github.com/tendermint/tendermint/libs/bytes"
    +	tmproto "github.com/tendermint/tendermint/proto/tendermint/types"
    +	"github.com/tendermint/tendermint/rpc/client/http"
    +	"github.com/tendermint/tendermint/types"
    +	"math/big"
    +	"os"
    +)
    +
    +func main() {
    +	err := verify()
    +	if err != nil {
    +		fmt.Println(err)
    +		os.Exit(1)
    +	}
    +}
    +
    +func verify() error {
    +	ctx := context.Background()
    +
    +
    +	// ...
    +	// check the first section for this part of the implementation
    +
    +	// get the nonce corresponding to the block height that contains the PayForBlob transaction
    +	// since Blobstream X emits events when new batches are submitted, we will query the events
    +	// and look for the range committing to the blob
    +	// first, connect to an EVM RPC endpoint
    +	ethClient, err := ethclient.Dial("evm_rpc_endpoint")
    +	if err != nil {
    +		return err
    +	}
    +	defer ethClient.Close()
    +
    +	// ...
    +	// check the first section for this part of the implementation
    +
    +	// now we will create the shares proof to be verified by the SimpleRollup
    +	// contract that uses the DAVerifier library
    +
    +	// get the proof of the shares containing the blob to the data root
    +	sharesProof, err := trpc.ProveShares(ctx, 16, uint64(blobShareRange.Start), uint64(blobShareRange.End))
    +	if err != nil {
    +		return err
    +	}
    +
    +	// use the SimpleRollup contract binding to submit to it a fraud proof
    +	simpleRollupWrapper, err := client.NewWrappers(ethcmn.HexToAddress("contract_Address"), ethClient)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// submit the fraud proof containing the share data that had the invalid state transition for example
    +	// along with its proof
    +	err = submitFraudProof(
    +		ctx,
    +		simpleRollupWrapper,
    +		sharesProof,
    +		event.ProofNonce.Uint64(),
    +		uint64(tx.Height),
    +		dcProof.Proof,
    +		blockRes.Block.DataHash,
    +	)
    +
    +	return nil
    +}
    +
    +func submitFraudProof(
    +	ctx context.Context,
    +	simpleRollup *client.Wrappers,
    +	sharesProof types.ShareProof,
    +	nonce uint64,
    +	height uint64,
    +	dataRootInclusionProof merkle.Proof,
    +	dataRoot []byte,
    +) error {
    +	var blockDataRoot [32]byte
    +	copy(blockDataRoot[:], dataRoot)
    +	tx, err := simpleRollup.SubmitFraudProof(
    +		&bind.TransactOpts{
    +			Context: ctx,
    +		},
    +		client.SharesProof{
    +			Data:             sharesProof.Data,
    +			ShareProofs:      toNamespaceMerkleMultiProofs(sharesProof.ShareProofs),
    +			Namespace:        *namespace(sharesProof.NamespaceID),
    +			RowRoots:         toRowRoots(sharesProof.RowProof.RowRoots),
    +			RowProofs:        toRowProofs(sharesProof.RowProof.Proofs),
    +			AttestationProof: toAttestationProof(nonce, height, blockDataRoot, dataRootInclusionProof),
    +		},
    +		blockDataRoot,
    +	)
    +	if err != nil {
    +		return err
    +	}
    +	// wait for transaction
    +}
    +
    +func toAttestationProof(
    +	nonce uint64,
    +	height uint64,
    +	blockDataRoot [32]byte,
    +	dataRootInclusionProof merkle.Proof,
    +) client.AttestationProof {
    +	sideNodes := make( [][32]byte, len(dataRootInclusionProof.Aunts))
    +	for i, sideNode :=  range dataRootInclusionProof.Aunts {
    +		var bzSideNode [32]byte
    +		for k, b := range sideNode {
    +			bzSideNode[k] = b
    +		}
    +		sideNodes[i] = bzSideNode
    +	}
    +
    +	return client.AttestationProof{
    +		TupleRootNonce: big.NewInt(int64(nonce)),
    +		Tuple:          client.DataRootTuple{
    +			Height:   big.NewInt(int64(height)),
    +			DataRoot: blockDataRoot,
    +		},
    +		Proof:          client.BinaryMerkleProof{
    +			SideNodes: sideNodes,
    +			Key:       big.NewInt(dataRootInclusionProof.Index),
    +			NumLeaves: big.NewInt(dataRootInclusionProof.Total),
    +		},
    +	}
    +}
    +
    +func toRowRoots(roots []bytes.HexBytes) []client.NamespaceNode {
    +	rowRoots := make([]client.NamespaceNode, len(roots))
    +	for i, root := range roots {
    +		rowRoots[i] = *toNamespaceNode(root.Bytes())
    +	}
    +	return rowRoots
    +}
    +
    +func toRowProofs(proofs []*merkle.Proof) []client.BinaryMerkleProof {
    +	rowProofs := make([]client.BinaryMerkleProof, len(proofs))
    +	for i, proof := range proofs {
    +		sideNodes := make( [][32]byte, len(proof.Aunts))
    +		for j, sideNode :=  range proof.Aunts {
    +			var bzSideNode [32]byte
    +			for k, b := range sideNode {
    +				bzSideNode[k] = b
    +			}
    +			sideNodes[j] = bzSideNode
    +		}
    + 		rowProofs[i] = client.BinaryMerkleProof{
    +			SideNodes: sideNodes,
    +			Key:       big.NewInt(proof.Index),
    +			NumLeaves: big.NewInt(proof.Total),
    +		}
    +	}
    +	return rowProofs
    +}
    +
    +func toNamespaceMerkleMultiProofs(proofs []*tmproto.NMTProof) []client.NamespaceMerkleMultiproof {
    +	shareProofs := make([]client.NamespaceMerkleMultiproof, len(proofs))
    +	for i, proof := range proofs {
    +		sideNodes := make([]client.NamespaceNode, len(proof.Nodes))
    +		for j, node := range proof.Nodes {
    +			sideNodes[j] = *toNamespaceNode(node)
    +		}
    +		shareProofs[i] = client.NamespaceMerkleMultiproof{
    +			BeginKey:  big.NewInt(int64(proof.Start)),
    +			EndKey:    big.NewInt(int64(proof.End)),
    +			SideNodes: sideNodes,
    +		}
    +	}
    +	return shareProofs
    +}
    +
    +func minNamespace(innerNode []byte) *client.Namespace {
    +	version := innerNode[0]
    +	var id [28]byte
    +	for i, b := range innerNode[1:28] {
    +		id[i] = b
    +	}
    +	return &client.Namespace{
    +		Version: [1]byte{version},
    +		Id:      id,
    +	}
    +}
    +
    +func maxNamespace(innerNode []byte) *client.Namespace {
    +	version := innerNode[29]
    +	var id [28]byte
    +	for i, b := range innerNode[30:57] {
    +		id[i] = b
    +	}
    +	return &client.Namespace{
    +		Version: [1]byte{version},
    +		Id:      id,
    +	}
    +}
    +
    +func toNamespaceNode(node []byte) *client.NamespaceNode {
    +	minNs := minNamespace(node)
    +	maxNs := maxNamespace(node)
    +	var digest [32]byte
    +	for i, b := range node[58:] {
    +		digest[i] = b
    +	}
    +	return &client.NamespaceNode{
    +		Min:    *minNs,
    +		Max:    *maxNs,
    +		Digest: digest,
    +	}
    +}
    +
    +func namespace(namespaceID []byte) *client.Namespace {
    +	version := namespaceID[0]
    +	var id [28]byte
    +	for i, b := range namespaceID[1:] {
    +		id[i] = b
    +	}
    +	return &client.Namespace{
    +		Version: [1]byte{version},
    +		Id:      id,
    +	}
    +}

    For the step (2), check the rollup inclusion proofs documentation for more information.

    Conclusion

    After creating all the proofs, and verifying them:

    1. Verify inclusion proof of the transaction to Celestia data root
    2. Prove that the data root tuple is committed to by the Blobstream X smart contract

    We can be sure that the data was published to Celestia, and then rollups can proceed with their normal fraud proving mechanism.

    NOTE

    The above proof constructions are implemented in Solidity, and may require different approaches in other programming languages.

    + + + + \ No newline at end of file diff --git a/developers/blobstream-rollups.html b/developers/blobstream-rollups.html new file mode 100644 index 00000000000..409220e41c2 --- /dev/null +++ b/developers/blobstream-rollups.html @@ -0,0 +1,47 @@ + + + + + + Introduction to Blobstream rollups | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Introduction to Blobstream rollups

    Blobstream is the first data availability solution for EVM chains that securely scales with the number of users. It allows rollups to post their data on Celestia while proving their availability in the rollup settlement contract.

    This document will outline a few ways to build optimistic or zk-rollups that post their data to Celestia and use Blobstream to prove that data's availability.

    Concepts

    Share commitment

    The share commitment is a commitment over the data contained in the MsgPayForBlobs transaction. This commitment allows proving that the corresponding data exists on Celestia efficiently.

    Share commitment: Proof details

    To prove that the data corresponding to a share commitment was posted to Celestia using Blobstream, the following proofs need to be verified:

    1. share inclusion proof to the share commitment: meaning creating two merkle proofs:
      1. share merkle proof up to the subtree root corresponding to that share
      2. subtree root merkle proof to the share commitment
    2. share commitment inclusion proof to the data root tuple root: meaning four merkle proofs:
      1. subtree roots merkle proofs to the share commitment: to make sure the subtree roots are valid
      2. subtree roots merkle proofs up to the row roots: to prove that the subtree roots belong to a set of rows in the Celestia block
      3. row roots proofs to the data root: to prove that those rows belong to the Celestia Block
      4. data root tuple proof to the data root tuple: to prove that the Celestia block referenced by its height and data root, was committed to by Blobstream.

    More details on the share commitment inclusion proof can be found in the commitment scheme docs and also the data square layout.

    If all of these proofs are valid, then you successfully managed to prove that the data corresponding to that share commitment has been posted to Celestia.

    NOTE

    Generating/verifying share commitment proofs is still not supported. It still needs tooling to generate the proofs on the node side, and verifying them on the Solidity side which will be built in the upcoming months.

    Share commitment: Compact proofs

    There is a way to have compact proofs, when using share commitments, unlike the ones defined above; that allow less costly inclusion proofs. These require the ability to parse the protobuf encoded PFBs.

    In fact, if the rollup project has a way to parse the protobuf encoded PFB, either in a smart contract or a zk-circuit, they will be able to create compact proofs of the rollup data.

    These proofs will work as follows:

    • Parsing the PFB and taking out the share commitment
    • Comparing the PFB commitment to the one saved in the rollup contract
    • Proving inclusion of the PFB to the data root tuple root. This will be a compact proof since we will only be proving two shares regardless of the size of the rollup data.

    More details on compact proofs can be found in ADR-011.

    Share commitment: Pros

    The pros of referencing rollup data using a share commitment:

    • Using the same commitment that exists on the PFB, without having to find another way of referencing the rollup data.
    • If the team has access to protobuf parsing, it allows for compact proof, but the parsing costs need to be investigated.

    Share commitment: Cons

    • Large/expensive proofs in the case of having no way to parse the protobuf PFB encoding.
    • In the optimistic rollups construction, defined below, this requires waiting for the Celestia block to be committed to by Blobstream before saving updating the settlement contract. This might require waiting for a few hours, depending on the batches size on each chain, to finally submit the rollup update.

    Given these limitations, an alternative design will be discussed in the next section.

    Sequence of spans

    An alternative way of referencing rollup data in the rollup settlement contract is using a sequence of spans.

    A sequence of spans is a data pointer that allows pointing to the rollup data inside a Celestia square using its location inside the square. It can be defined using the following information:

    • height: The height of the Celestia block containing the rollup data.
    • startIndex: The index of the first share containing the rollup data.
    • dataLen: The number of shares containing the rollup data.

    The startIndex and the dataLen can be queried from Celestia after the corresponding transaction gets included in a block and committed to the chain. An example of how to query them can be found in the verify command. The TxShareRange returns the start and end share of the data referenced by a transaction hash.

    NOTE

    If the rollup data is submitted in multiple blocks, the above sequence of spans can be generalized to include multiple blocks. For simplicity, we will stick with the data only submitted to a single Celestia block.

    Sequence of spans: Proof details

    Using sequence of spans is different from using the share commitment because we're referencing a location in the square, and not actual data commitment. So, the proof types and their generation are different.

    Sequence of spans: Proving unavailable data

    By construction, if the sequence of spans refers to a certain location in the square, that location is the data. This location can be in the reserved namespaces, the parity bytes, etc. What matters is that it's part of the square. So to prove that the sequence of spans is invalid, i.e., refers to data that is not available on Celestia, it is necessary and sufficient to show that the sequence of spans doesn't belong to the Celestia block, i.e., the span is out of bounds.

    We could create this proof via generating a binary Merkle proof of any row/column to the Celestia data root. This proof will provide the total which is the number of rows/columns in the extended data square. This can be used to calculate the square size. The computeSquareSizeFromRowProof method in the DAVerifier library allows calculating the square size from a row proof or a share proof.

    Then, we will use that information to check if the provided share index, in the header, is out of the square size bounds. In order words, we will check if the startIndex and the startIndex + dataLen are included in the range [0, 4*square_size].

    NOTE

    The square size is the number of rows of the original square.

    For the data root, we will use a binary Merkle proof to prove its inclusion in a data root tuple root that was committed to by the Blobstream smart contract. More on this in the data root inclusion proofs section.

    Sequence of spans: Proving inclusion of some data

    The difference between using a share commitment and a sequence of spans is that when using a share commitment, an extra merkle proof is needed to prove inclusion of the share to the share commitment. However, in the case of a sequence of spans, only the usual inclusion proof of a share to the data root tuple root is needed. The inclusion of the share to the sequence of spans is gotten using the same proof.

    In fact, proving that a share is part of the sequence of spans, i.e., part of the rollup data is done as follows:

    1. Prove that the data root tuple is committed to by the Blobstream smart contract:

      To prove the data root is committed to by the Blobstream smart contract, we will need to provide a Merkle proof of the data root tuple to a data root tuple root. This can be created using the data_root_inclusion_proof query. More on this can be found in the data root inclusion proofs documentation.

    2. Verify inclusion proof of the data to Celestia data root:

      To prove that the data is part of the data root, we will need to provide two proofs: a namespace Merkle proof of the data to a row root. This could be done via proving the shares that contain the data to the row root using a namespace Merkle proof. And, a binary Merkle proof of the row root to the data root.

      These proofs can be generated using the ProveShares query.

      More details on these proofs can be found in the transaction inclusion proof documentation.

    3. Prove that the data is in the sequence spans:

      To prove that the data is part of the rollup sequence of spans, we take the authenticated share proofs in step (2) and use the shares begin/end key to define the shares' positions in the row.

      Then, we use the row proof to get the row index in the extended Celestia square and get the index of the share in row major order:

      solidity
      uint256 shareIndexInRow = shareProof.shareProofs[0].beginKey;
      +uint256 shareIndexInRowMajorOrder = shareIndexInRow + shareProof.rowProofs[0].numLeaves * shareProof.rowProofs[0].key;
      uint256 shareIndexInRow = shareProof.shareProofs[0].beginKey;
      +uint256 shareIndexInRowMajorOrder = shareIndexInRow + shareProof.rowProofs[0].numLeaves * shareProof.rowProofs[0].key;

    Finally, we can compare the computed index with the sequence of spans, and be sure that the data/shares is part of the rollup data.

    Sequence of spans: Pros

    • Using a sequence of spans instead of the share commitment allows for simpler proofs

    Sequence of spans: Cons

    None

    Optimistic rollups

    One type of rollups that can be built with Blobstream is optimistic rollups. An optimistic rollup is a rollup that commits optimistically to a set of blocks, and allows the other parties to verify that the blocks are valid, and if they're not, they can create fraud proofs to signal that.

    Celestia allows optimistic rollups to post their data on its DA layer, and to prove that the data is available using Blobstream.

    To build an optimistic rollup that uses Celestia as a DA layer, the following constructions can be inspired by.

    Optimistic rollups that use a sequence of spans

    Optimistic rollups can post their data in Celestia, then in the rollup settlement contract, they can reference optimistically that data using a sequence of spans. Then, rollup full nodes can verify if that data is valid. If not, they can trigger a fraud proof.

    When using a sequence of spans, triggering the data availability fraud proofs, which are different from the state transitions fraud proofs (left for the rollup to define), goes back to the following cases:

    Optimistic rollups that use a sequence of spans: Pros

    • Not needing to verify anything at the moment of submitting the commitments to the rollup settlement contracts
    • The fraud proofs are simple and can be reduced to a single share: if, for example, a single transaction in the rollup data that was posted to Celestia is faulty, only the shares containing that transaction, which can be as minimal as a single share, need to be proven on chain and verified.

    Optimistic rollups that use a sequence of spans: Cons

    None

    Optimistic rollups that use a sequence of spans: Example

    An example optimistic rollup that uses sequence of spans to reference its data can be found in the RollupInclusionProofs. It portrays the different possible data availability proofs, constructs them and shows how to verify them.

    Also, more details on querying these kinds of proofs can be found in the proof queries documentation.

    Optimistic rollups that use share commitments

    Another way to build a rollup is to replace the sequence of spans with a height and a share commitment. Then, users/rollup full nodes will be able to query that data and validate it. If the rollup data is not valid, they can create a fraud proof.

    The first difference between the sequence of spans construction and the share commitment construction is having to verify that the provided share commitment is part of the Celestia block, referenced by its height in the moment of submitting the rollup commitments to the settlement contract. This is necessary to make sure that the commitment is part of Celestia. Otherwise, rollup sequencers can commit to random share commitments and there won't be a way to prove they're invalid.

    The second difference is the proof types. In the case of a fraud proof, the proofs outlined in the proofs details of share commitment section would need to be verified to be sure that the share containing the invalid state transition is part of the rollup data. Alternatively, the rollup settlement contract would need to have a library to parse protobuf encoded PFBs, as explained in the compact proofs of share commitment section, to have less expensive proofs. The cost of parsing the protobuf is not included in this analysis and needs to be investigated separately.

    Optimistic rollups that use share commitments: Pros

    • Using the same share commitment as the one saved in Celestia which gives access to existing tooling

    Optimistic rollups that use share commitments: Cons

    • The proofs are expensive in the base case. And if the settlement contract is able to parse the PFBs, thorough investigations of the cost of that would need to be done.

    Zk-rollups

    Zk-rollups, aka validity rollups, can also use Celestia as a DA and Blobstream to verify that the data was posted. However, the submission process is different from the above constructions, since there are no fraud proofs, and everything should be verified when submitting the commitment to the settlement contract.

    Similar to the optimistic case, the rollup settlement contract can reference the rollup data using either the sequence of spans approach or the share commitments. We will discuss both in this section.

    Zk-rollups that use sequence of spans

    When submitting the commitments to the rollup settlement contract, this latter will need to verify the following:

    1. Zk-proof of the state transitions, which is left for the rollup to define.
    2. Verify that the sequence of spans is valid, i.e., is part of the Celestia block referenced by its height, as described in the proof details section.
    3. Zk-proof of the rollup data to the data root. The verification process of this should accept a commitment as input so that the settlement contract makes sure it's the correct value that's being saved. The commitment can be the data root and the sequence of spans. And, when the rollup data is proven inside the circuit to the data root, the used data root is asserted to be the input one. Similarly, the data's location is asserted to be the same as the input sequence of spans. These arguments are the ones used in the sequence of spans verification in (2).

    Once these are valid, the settlement contract can be sure that the rollup data was posted to Celestia, and the sequence of spans references it correctly.

    Zk-rollups that use sequence of spans: Pros

    • The inclusion proof inside the zk-circuit is a simple proof that uses traditional merkle tree. In the case of using share commitment, as will be explained below, additional libraries that can be expensive to prove are required.

    Zk-rollups that use sequence of spans: Cons

    None

    Zk-rollups that use share commitments

    To use share commitments to reference rollup data in the zk-rollup settlement contract, the zk-circuits need to be able to deserialize protobuf encoded messages. Alternatively, more involved merkle proofs will need to be verified.

    Protobuf deserialization inside a zk-circuit

    One way of using the share commitment to reference the rollup data in zk-rollups is via using a protobuf deserialization library inside the zk-circuit. And the verification would proceed as follows:

    1. Zk-proof of the state transitions, which is left to the rollup team to define.
    2. Verify that the share commitment is valid using the proofs laid out in the proof details of share commitment section.
    3. The zk-proof verifier would take as argument the data root and the share commitment. Then, inside the circuit, the protobuf encoded PFB transaction will be deserialized and then verify the following:
    • The deserialized share commitment is the same as the one provided as input
    • The circuit will prove the inclusion of the PFB to the data root, then assert that the data root is the same as the one provided as input.

    If the above conditions are valid, the rollup settlement contract can be sure that the rollup data was posted to Celestia and is correctly referenced.

    Zk-rollups that use share commitments: Pros

    None

    Zk-rollups that use share commitments: Cons

    • This approach requires having access to a protobuf decoder inside a zk-circuit which is not straightforward to have. Also, the relative costs will need to be investigated.

    Heavy merkle proofs usage

    Similar to Protobuf deserialization inside a zk-circuit, the zk-circuit will proceed to the verification of the availability of the data. The difference is that instead of parsing the encoded protobuf, the proofs defined under the share commitment proof details section will need to be verified inside the zk-circuit as follows:

    1. Zk-proof of the state transitions, which is left to the rollup team to define.
    2. Verify that the share commitment is valid using the proofs laid out in the share commitment proof details section.
    3. The zk-proof verifier would take as argument the data root and the share commitment. Then, inside the circuit:
    • It will verify that the input share commitment corresponds to the rollup data.
    • Verify that the input data root commits to that share commitment. Check the share commitment proof details for more details

    Once these proofs are valid, the rollup settlement contract can be sure that the rollup data was posted to Celestia and is correctly referenced.

    heavy merkle proofs usage: Pros

    None

    heavy merkle proofs usage: Cons

    • More heavy usage of merkle proofs inside and outside the zk-circuit.

    Conclusion

    Given the above details, using the sequence of spans is the better solution in the general case as explained in the optimistic rollups that uses a sequence of spans and zk-rollups that use sequence of spans sections. The proof sizes are small and allow for greater flexibility. However, if the rollup team has different requirements, then the other designs can be explored.

    + + + + \ No newline at end of file diff --git a/developers/blobstream-x-deploy.html b/developers/blobstream-x-deploy.html new file mode 100644 index 00000000000..d187bfbfcb8 --- /dev/null +++ b/developers/blobstream-x-deploy.html @@ -0,0 +1,45 @@ + + + + + + New Blobstream X deployments | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    New Blobstream X deployments

    If you want to deploy Blobstream X to a new chain, where a Gateway contract does not exist, you need to do the following.

    If Succinct Gateway already exists on this chain, you can skip this step. You can check the list of deployed SuccinctGateway contracts.

    Deploy a new SuccinctGateway contract

    Deploy a new SuccinctGateway contract to the new chain.

    Deploy a BlobstreamX contract to the new chain

    Follow the guide to use your deployed SuccinctGateway to deploy a new Blobstream X contract.

    Run a local prover

    Now that you have the contracts deployed, you can run a local prover.

    + + + + \ No newline at end of file diff --git a/developers/blobstream.html b/developers/blobstream.html new file mode 100644 index 00000000000..0b7d174cfe4 --- /dev/null +++ b/developers/blobstream.html @@ -0,0 +1,45 @@ + + + + + + Blobstream: Streaming modular DA to Ethereum | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Blobstream: Streaming modular DA to Ethereum

    Blobstream logo

    What is Blobstream?

    Blobstream is the first data availability solution for Ethereum that securely scales with the number of users. Formerly known as the Quantum Gravity Bridge (QGB), Blobstream relays commitments to Celestia's data root to an onchain light client on Ethereum, for integration by developers into L2 contracts. This enables Ethereum developers to build high-throughput L2s using Celestia's optimised DA layer, the first with Data Availability Sampling (DAS). Any ecosystem can deploy a Blobstream light client onchain to allow L2s and L3s to access DA from Celestia.

    An implementation of Blobstream, by Succinct, called Blobstream X, is out and will be used in the upcoming deployments. This implementation proves the validity of Celestia block headers on a target EVM chain using zero-knowledge (ZK) proofs, which allow inheriting all the security guarantees of Celestia.

    Please note: Blobstream remains early-stage, experimental software and users should use Blobstream at their own risk.

    Blobstream vs. data availability committees (DACs)

    Decentralization and security

    Blobstream is built on Celestia, which uses a CometBFT-based proof-of-stake system. Blobstream shares the same security assumptions as Celestia. In contrast, data availability committees (DACs), are typically centralized or semi-centralized, relying on a specific set of entities or individuals to vouch for data availability.

    Mechanism of verification

    Blobstream uses data availability attestations, which are Merkle roots of the batched L2 data, to confirm that the necessary data is present on Celestia. The L2 contract on Ethereum can check directly with Blobstream if the data is published on Celestia. Similarly, a DAC would rely on attestations or confirmations from its permissioned members.

    Flexibility and scalability

    Blobstream is designed to offer high-throughput data availability for Ethereum L2s, aiming to strike a balance between scalability and security. It operates independently of Ethereum's gas costs, as Celestia's resource pricing is more byte-focused rather than computation-centric. On the other hand, the scalability and flexibility of a DAC would depend on its specific design and implementation.

    In summary, both Blobstream and DACs aim to ensure offchain data availability, but Blobstream offers a more decentralized, secure, and scalable solution compared to the potential centralized nature of DACs.

    What is Blobstream X?

    Blobstream X is an implementation of Blobstream with a ZK light client that bridges Celestia’s modular DA layer to Ethereum to allow high-throughput rollups to use Celestia’s DA while settling on Ethereum.

    Optimistic or ZK rollups that settle on Ethereum, but wish to use Celestia for DA, require a mechanism for bridging Celestia’s data root to Ethereum as part of the settlement process. This data root is used during inclusion proofs to prove that particular rollup transactions were included and made available in the Celestia network.

    Bridging Celestia’s data root to Ethereum requires running a Celestia light client as a smart contract on Ethereum, to make the latest state of the Celestia chain known on Ethereum and available to rollups. Blobstream X utilizes the latest advances in ZK proofs to generate a succinct proof that enough Celestia validators have come to consensus (according to the CometBFT consensus protocol) on a block header, and verifies this proof in the Blobstream X Ethereum smart contract to update it with the latest Celestia header.

    The Blobstream X ZK proof not only verifies the consensus of Celestia validators, but it also merkelizes and hashes all the data roots in the block range from the previous update to the current update, making accessible all Celestia data roots (verifiable with a Merkle inclusion proof against the stored Merkle root) to rollups.

    Blobstream X is built and deployed with Succinct's protocol.

    blobstream x draft diagram

    Integrate with Blobstream X

    The following docs go over how developers can integrate Blobstream X.

    You can find the repository for Blobstream X along with code for:

    The first deployments of Blobstream X will be maintained on the following chains: Arbitrum One, Base and Ethereum Mainnet. Every 1 hour, the prover/relayer will post an update to the Blobstream X contract that will include a new data commitment range that covers a 1-hour block range from the latestBlock in the Blobstream X contract. On Ethereum Mainnet, the Blobstream X contract will be updated every 4 hours.

    NOTE

    Custom ranges can be requested using the BlobstreamX contract to create proofs for specific Celestia block batches. These ranges can be constructed as [latestBlock, customTargetBlock), with latestBlock as the latest block height that was committed to by the BlobstreamX contract, and latestBlock > customTargetBlock, and customTargetBlock - latestBlock <= DATA_COMMITMENT_MAX.

    Block ranges that are before the contract's latestBlock can't be proven a second time in different batches.

    More information can be found in the requestHeaderRange(...) method.

    How Blobstream X works

    As shown in the diagram below, the entrypoint for updates to the Blobstream X contract is through the SuccinctGateway smart contract, which is a simple entrypoint contract that verifies proofs (against a deployed onchain verifier for the Blobstream X circuit) and then calls the BlobstreamX.sol contract to update it. Find more information about the SuccinctGateway.

    blobstream x overview diagram draft

    NOTE

    If the Blobstream X contract is not deployed on a desired chain, it needs to be deployed before it can be used by your rollup. See the deployment documentation for more details.

    How to integrate with Blobstream X

    Integrating your L2 with Blobstream X requires two components: your onchain smart contract logic, and your offchain client logic for your rollup. The next three sections cover these topics:

    Deployed contracts

    You can interact with the Blobstream X contracts today on testnet. The Blobstream X Solidity smart contracts are currently deployed on the following Ethereum testnets:

    ContractEVM networkContract addressAttested data on Celestia
    Blobstream XEthereum MainnetNot yet deployedMainnet Beta
    Blobstream XArbitrum One0xA83ca7775Bc2889825BcDeDfFa5b758cf69e8794Mainnet Beta
    Blobstream XBase0xA83ca7775Bc2889825BcDeDfFa5b758cf69e8794Mainnet Beta
    Blobstream XSepolia0xf0c6429ebab2e7dc6e05dafb61128be21f13cb1eMocha testnet
    Blobstream XArbitrum Sepolia0xc3e209eb245Fd59c8586777b499d6A665DF3ABD2Mocha testnet
    Blobstream XBase Sepolia0xc3e209eb245Fd59c8586777b499d6A665DF3ABD2Mocha testnet

    Blobstream rollups

    More on the different ways to build a blobstream rollups can be found in the blobstream rollups documentation.

    + + + + \ No newline at end of file diff --git a/developers/bubs-testnet.html b/developers/bubs-testnet.html new file mode 100644 index 00000000000..52a0da371b7 --- /dev/null +++ b/developers/bubs-testnet.html @@ -0,0 +1,45 @@ + + + + + + Bubs testnet | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Bubs testnet

    Bubs testnet

    Bubs Testnet is a fresh offering from Caldera with support from Celestia Labs, built with OP Stack and Celestia, and is dedicated to providing developers with an EVM-compatible execution layer to deploy their EVM applications on.

    Built with the OP Stack and Celestia

    The Bubs Testnet is a testnet rollup, a modified version of optimism-bedrock that uses Celestia as a data availability (DA) layer. This integration can be found in the @celestiaorg/optimism repository. The testnet is hosted by Caldera, who makes it easy to launch rollups with no code required.

    In this setup, data handling is accomplished in two ways. Firstly, data is written to the DA layer, in this case, Celestia (on the Mocha testnet). Then, the data commitment is written to the op-batcher. When reading, the op-node retrieves the data back from the DA layer by first reading the data commitment from the op-batcher, then reading the data from the DA layer using the data commitment. Hence, while previously op-node was reading from calldata on Ethereum, it now reads data from Celestia. View the namespace for Bubs on Celestia's Mocha testnet.

    The tools involved in the data handling process include op-batcher, which batches up rollup blocks and posts them to Ethereum, op-geth that handles execution, and op-proposer responsible for state commitment submission.

    By using Celestia as a DA layer, existing L2s can switch from posting their data as calldata on Ethereum to posting to Celestia. The commitment to the block is posted on Celestia, which is purpose-built for data availability. This is more scalable than the traditional method of posting this data as calldata on monolithic chains.

    Building on Bubs

    Bubs Testnet provides a robust environment for developers to test their Ethereum Virtual Machine (EVM) applications. It offers an EVM-compatible execution layer, making it an ideal platform for developers looking to build and test applications in a setting that closely mirrors an OP Stack rollup on Celestia.

    Learn more at https://bubs-sepolia.hub.caldera.xyz/.

    RPC URLs

    Remote Procedure Call (RPC) URLs are endpoints that allow developers to interact with the blockchain. They are essential for sending transactions, querying blockchain data, and performing other interactions with the blockchain.

    For the Bubs Testnet, you can connect to the following RPC URLs:

    HTTPS

    • https://bubs-sepolia.rpc.caldera.xyz/http

    WSS

    • wss://bubs-sepolia.rpc.caldera.xyz/ws

    This URL serves as the entry point to the Bubs Testnet. You can use it in your applications to connect to the testnet and interact with the smart contracts you deploy there.

    Remember, Bubs Testnet is a testing environment!

    Bridge

    Bridging is a process that enables the transfer of assets between different blockchains.

    To bridge between Ethereum Sepolia and Bubs Testnet, visit the Bubs Testnet bridge.

    Faucet

    To visit the Bubs testnet faucet, go to https://bubs-sepolia.hub.caldera.xyz/ and click the "Faucet" tab.

    Explorer

    To visit the explorer, go to https://bubs-sepolia.explorer.caldera.xyz/.

    Status

    To see the status and uptime information for Bubs, visit the status page.

    Next steps

    Now that you have a better understanding of the Bubs Testnet and its integration of OP Stack and Celestia, you can start exploring its capabilities:

    + + + + \ No newline at end of file diff --git a/developers/build-whatever.html b/developers/build-whatever.html new file mode 100644 index 00000000000..4fad17517b9 --- /dev/null +++ b/developers/build-whatever.html @@ -0,0 +1,45 @@ + + + + + + Build whatever | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Build whatever

    If you're a developer and want to know what the benefits of modular blockchains are for you, you’ve come to the right place. This page will give you the rundown on modular blockchains and their benefits for developers like you.

    This section provides various guides and tutorials that cover different options for deploying rollups on Celestia.

    Quickstart - Building on Celestia

    Choose a framework

    So, you’re ready to start experimenting and building on Celestia? Here are a few options that are currently available for developers.

    Rollups-as-a-Service

    Deploy your rollup with a RaaS provider.

    Smart contracts

    Deploy your smart contracts on dedicated EVM-compatible rollups.

    What is a rollup?

    A rollup is a type of blockchain that offloads some work to a layer 1, like Celestia. Rollups host applications and process user transactions. Once those transactions get processed, they are then published to layer 1. It’s layer 1s job to order those transactions and check that they are available, at minimum.

    What is a modular blockchain?

    With blockchains there are more or less four core functions that they do.

    • Execution: transaction execution and state update.
    • Settlement: finality and dispute resolution.
    • Consensus: agreement on transaction ordering.
    • Data availability: prove data was published to the network.

    Modular blockchains specialize in one or two of these functions rather than doing all of them like a monolithic blockchain. You probably know about layer 1s and layer 2s. That’s the general idea.

    A typical example of a modular blockchain you might’ve heard of is a rollup. Rollups host smart contracts and execute transactions, much like any monolithic chain. But, the data of those transactions get sent to a layer 1 blockchain to carry out the remaining functions.

    If you want to brush up on your understanding of modular blockchains, head over to learn modular.

    Benefits of modular blockchains

    Ease of deploying a chain

    One of the goals of modular blockchains is to make it as easy to deploy a blockchain as a smart contract. There are a few unique ways that modular blockchains can significantly reduce the cost of deploying a new blockchain.

    1. No validator set is required. Rollups can deploy without sourcing their own set of validators or sequencers.
    2. Inherit security from the start. Rollups don’t need to build all their security from scratch.
    3. Any part of the stack can be delegated. Development time can be reduced by outsourcing functions of the rollup to external providers.

    All in all, builders will be able to outsource as much of the stack as they need. Deploying a new blockchain will be as simple as clicking a few options to initialize a production-ready rollup.

    Scaling

    Of course, a much higher scale is necessary if we want to support many more users. And modular blockchains use some new innovative technologies that can help us get there.

    • Data availability sampling enables modular blockchains like Celestia to scale data availability with the number of light nodes - that means more capacity for rollups.
    • Fraud and validity proofs make rollups vastly more efficient to verify. Nodes only need to verify a small proof of transaction validity (validity proof) or assume transactions are valid by default (fraud proof). This means rollups don’t require every node in the network to re-execute every transaction.

    image

    • Decoupling execution from consensus lets developers define the VM that best fits the scaling needs of their application.
    • Separating applications across multiple rollups isolates congestion. If an application congests the execution capacity of one rollup, all other rollups remain unaffected in their execution capacity.

    All these scaling properties combined make new types of applications and features possible, like onchain gaming, dynamic metadata, and ephemeral rollups, to name a few.

    Customizability

    By design, modular blockchains don’t lock in any feature set. They promote experimentation and customization.

    Remember how decoupling execution from consensus enables VM customizability? Well, rollups are the execution component. Applications can run on their own rollup and adjust the VM to maximize their application's performance. Developers have that flexibility because Celestia's execution logic doesn't restrict rollups.

    Basically, rollups can be customized to integrate any new or existing VM stack.

    With existing rollup frameworks, developers can run rollup testnets using the EVM or Cosmos SDK. In the future, one can imagine a variety of VMs that rollup frameworks support, providing developers with more out-of-the-box options for their applications.

    Some customizations that could be made to a rollup's VM include custom precompiles, changing transaction processing from sequential to parallel, or adding support for private smart contracts.

    All of this only scratches the surface.

    + + + + \ No newline at end of file diff --git a/developers/celestia-node-key.html b/developers/celestia-node-key.html new file mode 100644 index 00000000000..afcfc044db2 --- /dev/null +++ b/developers/celestia-node-key.html @@ -0,0 +1,108 @@ + + + + + + Create a wallet with celestia-node | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Create a wallet with celestia-node

    This tutorial will go over using the cel-key utility to generate a wallet on celestia-node.

    While this tutorial will go over installation process of cel-key, it is recommended that you complete the following prerequisites first:

    Once you completed the prerequisite, you can proceed with this tutorial.

    Using the cel-key utility

    Inside the celestia-node repository is a utility called cel-key that uses the key utility provided by Cosmos-SDK under the hood. The utility can be used to add, delete, and manage keys for any DA node type (bridge || full || light), or just keys in general.

    Installation

    You need to first pull down the celestia-node repository:

    sh
    git clone https://github.com/celestiaorg/celestia-node.git
    +cd celestia-node/
    git clone https://github.com/celestiaorg/celestia-node.git
    +cd celestia-node/

    It can be built using either of the following commands:

    sh
    # dumps binary in current working directory, accessible via `./cel-key`
    +make cel-key
    # dumps binary in current working directory, accessible via `./cel-key`
    +make cel-key

    or

    sh
    # installs binary in GOBIN path, accessible via `cel-key`
    +make install-key
    # installs binary in GOBIN path, accessible via `cel-key`
    +make install-key

    For the purpose of this guide, we will use the make cel-key command.

    Steps for generating node keys

    To generate a key for a Celestia node, select the tab for your node type:

    TIP

    You do not need to declare a network for Mainnet Beta. Refer to the chain ID section on the troubleshooting page for more information

    bash
    ./cel-key add <key-name> --keyring-backend test --node.type bridge \
    +  --p2p.network <network>
    ./cel-key add <key-name> --keyring-backend test --node.type bridge \
    +  --p2p.network <network>
    bash
    ./cel-key add <key-name> --keyring-backend test --node.type full \
    +  --p2p.network <network>
    ./cel-key add <key-name> --keyring-backend test --node.type full \
    +  --p2p.network <network>
    bash
    ./cel-key add <key-name> --keyring-backend test --node.type light \
    +  --p2p.network <network>
    ./cel-key add <key-name> --keyring-backend test --node.type light \
    +  --p2p.network <network>

    This will load the key <key-name> into the directory of the node.

    Further flags you can use to customize your key are the following:

    • --node.store: Specifies a different directory you can use to save your node data and configurations. Expects a path to a directory.
    • --p2p.network: Specifies which network you want the key for. Values are arabica and mocha. Please note the default network will be mocha.

    Keep in mind that your celestia-node will only pick up keys that are inside the node.store directory under /keys so you should make sure to point cel-key utility to the correct directory via the node.store or p2p.network flags if you have specified a custom directory or network other than Mocha.

    Also keep in mind that if you do not specify a network with --p2p.network, the default one will always be mocha.

    Steps for exporting node keys

    You can export a private key from the local keyring in encrypted and ASCII-armored format.

    bash
    ./cel-key export <key-name> --keyring-backend test --node.type bridge \
    +  --p2p.network <network>
    ./cel-key export <key-name> --keyring-backend test --node.type bridge \
    +  --p2p.network <network>
    bash
    ./cel-key export <key-name> --keyring-backend test --node.type full \
    +  --p2p.network <network>
    ./cel-key export <key-name> --keyring-backend test --node.type full \
    +  --p2p.network <network>
    bash
    ./cel-key export <key-name> --keyring-backend test --node.type light \
    +  --p2p.network <network>
    ./cel-key export <key-name> --keyring-backend test --node.type light \
    +  --p2p.network <network>

    Steps for importing node keys

    To import from a mnemonic, use the following command, then enter your bip39 mnemonic:

    bash
    ./cel-key add <key-name> --recover --keyring-backend test \
    +  --node.type bridge --p2p.network <network>
    ./cel-key add <key-name> --recover --keyring-backend test \
    +  --node.type bridge --p2p.network <network>
    bash
    ./cel-key add <key-name> --recover --keyring-backend test \
    +  --node.type full --p2p.network <network>
    ./cel-key add <key-name> --recover --keyring-backend test \
    +  --node.type full --p2p.network <network>
    bash
    ./cel-key add <key-name> --recover --keyring-backend test \
    +  --node.type light --p2p.network <network>
    ./cel-key add <key-name> --recover --keyring-backend test \
    +  --node.type light --p2p.network <network>

    View all options for cel-key

    sh
    ./cel-key --help
    ./cel-key --help

    Docker and cel-key

    Prerequisites

    Running your node

    Run the Docker image (in this example, we are using a light node on Mocha testnet):

    bash
    docker run --name celestia-node -e NODE_TYPE=light -e P2P_NETWORK=mocha -p 26659:26659 \
    +ghcr.io/celestiaorg/celestia-node:v0.13.3 celestia light start \
    +--core.ip rpc-mocha.pops.one --p2p.network mocha
    docker run --name celestia-node -e NODE_TYPE=light -e P2P_NETWORK=mocha -p 26659:26659 \
    +ghcr.io/celestiaorg/celestia-node:v0.13.3 celestia light start \
    +--core.ip rpc-mocha.pops.one --p2p.network mocha

    TIP

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    You do not need to declare a network for Mainnet Beta. Refer to the chain ID section on the troubleshooting page for more information

    List active containers in another window with:

    bash
    docker ps
    docker ps

    The response will look like:

    bash
    CONTAINER ID   IMAGE           COMMAND                  CREATED          STATUS          PORTS      NAMES
    +<container-id>   celestia-node   "/entrypoint.sh cele…"   22 seconds ago   Up 21 seconds   2121/tcp   docker-compose-test-celestia-1
    CONTAINER ID   IMAGE           COMMAND                  CREATED          STATUS          PORTS      NAMES
    +<container-id>   celestia-node   "/entrypoint.sh cele…"   22 seconds ago   Up 21 seconds   2121/tcp   docker-compose-test-celestia-1

    Interact with the container by replacing <container-id> for the container ID:

    bash
    docker exec -ti <container-id> /bin/bash
    docker exec -ti <container-id> /bin/bash

    Now, interact with cel-key to check for the key that was autogenerated when you started the node:

    bash
    ./cel-key list --keyring-backend test --node.type light
    ./cel-key list --keyring-backend test --node.type light

    You can also export your key from the container. In the next section, you'll learn how to mount existing keys to the container.

    Mounting existing keys to container

    In this example, we'll be mounting an existing key to the container. We're also using an existing image called celestia-node. This will mount the entire /.celestia-light-<p2p-network>/keys directory to your image, or on Mainnet Beta the /.celestia-light/keys directory.

    Write a docker-compose.yml to accomplish this:

    yaml
    version: "3.8"
    +services:
    +  celestia:
    +    image: celestia-node
    +    environment:
    +      - NODE_TYPE=light
    +    command: celestia light start --core.ip rpc-mocha.pops.one --p2p.network mocha --keyring.accname my_celes_key
    +    volumes:
    +      - ${PWD}/keys:/root/.celestia-light-mocha-4/keys
    +    ports:
    +      - 26659:26659
    version: "3.8"
    +services:
    +  celestia:
    +    image: celestia-node
    +    environment:
    +      - NODE_TYPE=light
    +    command: celestia light start --core.ip rpc-mocha.pops.one --p2p.network mocha --keyring.accname my_celes_key
    +    volumes:
    +      - ${PWD}/keys:/root/.celestia-light-mocha-4/keys
    +    ports:
    +      - 26659:26659

    Start the container by running the following command in the directory with your docker-compose.yml:

    bash
    docker-compose up
    docker-compose up

    List active containers in another window with:

    bash
    docker ps
    docker ps

    The response will look like:

    bash
    CONTAINER ID   IMAGE           COMMAND                  CREATED          STATUS          PORTS      NAMES
    +<container-id>   celestia-node   "/entrypoint.sh cele…"   22 seconds ago   Up 21 seconds   2121/tcp   docker-compose-test-celestia-1
    CONTAINER ID   IMAGE           COMMAND                  CREATED          STATUS          PORTS      NAMES
    +<container-id>   celestia-node   "/entrypoint.sh cele…"   22 seconds ago   Up 21 seconds   2121/tcp   docker-compose-test-celestia-1

    Interact with the container by replacing <container-id> for the container ID:

    bash
    docker exec -ti <container-id> /bin/bash
    docker exec -ti <container-id> /bin/bash

    Now, interact with cel-key to check your address matches the address you expect with the key you mounted:

    bash
    root@<container-id>:/# ./cel-key list --keyring-backend test --node.type light
    +using directory:  ~/.celestia-light-mocha-4/keys
    +- address: celestia1wkhyhr7ngf0ayqlpnsnxg4d72hfs5453dvunm9
    +  name: my_celes_key
    +  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"A1/NsoY0RGL7Hqt4VWLg441GQKJsZ2fBUnZXipgns8oV"}'
    +  type: local
    root@<container-id>:/# ./cel-key list --keyring-backend test --node.type light
    +using directory:  ~/.celestia-light-mocha-4/keys
    +- address: celestia1wkhyhr7ngf0ayqlpnsnxg4d72hfs5453dvunm9
    +  name: my_celes_key
    +  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"A1/NsoY0RGL7Hqt4VWLg441GQKJsZ2fBUnZXipgns8oV"}'
    +  type: local
    + + + + \ No newline at end of file diff --git a/developers/deploy-on-bubs.html b/developers/deploy-on-bubs.html new file mode 100644 index 00000000000..c7bb6a95081 --- /dev/null +++ b/developers/deploy-on-bubs.html @@ -0,0 +1,181 @@ + + + + + + Deploy a smart contract on Bubs testnet | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Deploy a smart contract on Bubs testnet

    In this tutorial, we will deploy a smart contract to the Bubs testnet.

    Dependencies

    Setup

    First, in your $HOME directory, set up a new project folder for this tutorial and init the project with npm:

    bash
    cd $HOME
    +mkdir counter-project && cd counter-project && npm init -y
    cd $HOME
    +mkdir counter-project && cd counter-project && npm init -y

    Next, initialize a Foundry project with the following command:

    bash
    forge init counter_contract
    forge init counter_contract

    Create your smart contract

    Take a look at the Counter.sol file in your counter-project/counter_contract/src directory:

    solidity
    // SPDX-License-Identifier: UNLICENSED
    +pragma solidity ^0.8.13;
    +
    +contract Counter {
    +    uint256 public number;
    +
    +    function setNumber(uint256 newNumber) public {
    +        number = newNumber;
    +    }
    +
    +    function increment() public {
    +        number++;
    +    }
    +}
    // SPDX-License-Identifier: UNLICENSED
    +pragma solidity ^0.8.13;
    +
    +contract Counter {
    +    uint256 public number;
    +
    +    function setNumber(uint256 newNumber) public {
    +        number = newNumber;
    +    }
    +
    +    function increment() public {
    +        number++;
    +    }
    +}

    The contract contains a public unsigned integer variable named "number". There are two public functions in this contract. The setNumber function allows anyone to set a new value for the "number" variable, while the increment function increases the value of "number" by one each time it's called.

    You can learn more about Solidity and smart contract programming.

    To compile the contract, run the following forge command from the $HOME/counter-project/counter_contract/ directory:

    bash
    forge build
    forge build

    Your output should look similar to the following:

    bash
    [⠢] Compiling...
    +[⠔] Compiling 21 files with 0.8.19
    +[⠑] Solc 0.8.19 finished in 1.24s
    +Compiler run successful
    [⠢] Compiling...
    +[⠔] Compiling 21 files with 0.8.19
    +[⠑] Solc 0.8.19 finished in 1.24s
    +Compiler run successful

    Test your smart contract

    Now, open the test/Counter.t.sol file:

    solidity
    // SPDX-License-Identifier: UNLICENSED
    +pragma solidity ^0.8.13;
    +
    +import "forge-std/Test.sol";
    +import "../src/Counter.sol";
    +
    +contract CounterTest is Test {
    +    Counter public counter;
    +
    +    function setUp() public {
    +        counter = new Counter();
    +        counter.setNumber(0);
    +    }
    +
    +    function testIncrement() public {
    +        counter.increment();
    +        assertEq(counter.number(), 1);
    +    }
    +
    +    function testSetNumber(uint256 x) public {
    +        counter.setNumber(x);
    +        assertEq(counter.number(), x);
    +    }
    +}
    // SPDX-License-Identifier: UNLICENSED
    +pragma solidity ^0.8.13;
    +
    +import "forge-std/Test.sol";
    +import "../src/Counter.sol";
    +
    +contract CounterTest is Test {
    +    Counter public counter;
    +
    +    function setUp() public {
    +        counter = new Counter();
    +        counter.setNumber(0);
    +    }
    +
    +    function testIncrement() public {
    +        counter.increment();
    +        assertEq(counter.number(), 1);
    +    }
    +
    +    function testSetNumber(uint256 x) public {
    +        counter.setNumber(x);
    +        assertEq(counter.number(), x);
    +    }
    +}

    This file performs unit testing on the contract we created in the previous section. Here's what the test is doing:

    • The contract includes a public "Counter" type variable called "counter". In the setUp function, it initializes a new instance of the "Counter" contract and sets the "number" variable to 0.

    • There are two test functions in the contract: testIncrement and testSetNumber.

    • The testIncrement function tests the "increment" function of the "Counter" contract by calling it and then asserting that the "number" in the "Counter" contract is 1. It verifies if the increment operation correctly increases the number by one.

    • The testSetNumber function is more generic. It takes an unsigned integer argument 'x' and tests the "setNumber" function of the "Counter" contract. After calling the "setNumber" function with 'x', it asserts that the "number" in the "Counter" contract is equal to 'x'. This verifies that the "setNumber" function correctly updates the "number" in the "Counter" contract.

    Now, to test your code, run the following:

    bash
    forge test
    forge test

    If the test is successful, your output should be similar to this:

    bash
    [⠆] Compiling...
    +No files changed, compilation skipped
    +
    +Running 2 tests for test/Counter.t.sol:CounterTest
    +[PASS] testIncrement() (gas: 28334)
    +[PASS] testSetNumber(uint256) (runs: 256, μ: 27709, ~: 28409)
    +Test result: ok. 2 passed; 0 failed; finished in 8.96ms
    [⠆] Compiling...
    +No files changed, compilation skipped
    +
    +Running 2 tests for test/Counter.t.sol:CounterTest
    +[PASS] testIncrement() (gas: 28334)
    +[PASS] testSetNumber(uint256) (runs: 256, μ: 27709, ~: 28409)
    +Test result: ok. 2 passed; 0 failed; finished in 8.96ms

    Deploying your smart contract

    Using Anvil

    First, we'll test out our contract on a local devnet called "anvil". To start the local server, run:

    bash
    anvil
    anvil

    You'll see a local RPC endpoint (127.0.0.1:8545) and accounts to test with.

    Let's deploy the contract now. First, set a private key from anvil:

    bash
    export PRIVATE_KEY=0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80
    +export ANVIL_RPC_URL=http://localhost:8545
    export PRIVATE_KEY=0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80
    +export ANVIL_RPC_URL=http://localhost:8545

    Now, deploy the contract:

    bash
    forge create --rpc-url $ANVIL_RPC_URL \
    +--private-key $PRIVATE_KEY \
    +src/Counter.sol:Counter
    forge create --rpc-url $ANVIL_RPC_URL \
    +--private-key $PRIVATE_KEY \
    +src/Counter.sol:Counter

    Using Bubs

    First, set a private key from your funded Ethereum wallet and set the BUBS_RPC_URL variable with an RPC of your choosing:

    bash
    export BUBS_PRIVATE_KEY=0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80
    +export BUBS_RPC_URL=https://bubs-sepolia.rpc.caldera.xyz/http
    export BUBS_PRIVATE_KEY=0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80
    +export BUBS_RPC_URL=https://bubs-sepolia.rpc.caldera.xyz/http

    Now that we're ready to deploy the smart contract onto Bubs, we will run the forge create command.

    bash
    forge create --rpc-url $BUBS_RPC_URL \
    +--private-key $BUBS_PRIVATE_KEY \
    +src/Counter.sol:Counter
    forge create --rpc-url $BUBS_RPC_URL \
    +--private-key $BUBS_PRIVATE_KEY \
    +src/Counter.sol:Counter

    A successful deployment will return output similar to below:

    bash
    [⠆] Compiling...
    +No files changed, compilation skipped
    +Deployer: 0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266
    +Deployed to: 0x5FbDB2315678afecb367f032d93F642f64180aa3
    +Transaction hash: 0xf1a793a793cd9fc588f5132d99008565ea361eb3535d66499575e9e1908200b2
    [⠆] Compiling...
    +No files changed, compilation skipped
    +Deployer: 0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266
    +Deployed to: 0x5FbDB2315678afecb367f032d93F642f64180aa3
    +Transaction hash: 0xf1a793a793cd9fc588f5132d99008565ea361eb3535d66499575e9e1908200b2

    Once you've deployed the contract, you're ready to interact with it!

    First, we'll set it as a variable:

    bash
    export CONTRACT_ADDRESS=0x5FbDB2315678afecb367f032d93F642f64180aa3
    export CONTRACT_ADDRESS=0x5FbDB2315678afecb367f032d93F642f64180aa3

    Interacting with your smart contract

    Foundry uses cast, a CLI for performing Ethereum RPC calls.

    To write to the contract, we'll use the cast send command:

    bash
    cast send $CONTRACT_ADDRESS "setNumber(uint256)" 10 --rpc-url $BUBS_RPC_URL --private-key $BUBS_PRIVATE_KEY
    cast send $CONTRACT_ADDRESS "setNumber(uint256)" 10 --rpc-url $BUBS_RPC_URL --private-key $BUBS_PRIVATE_KEY

    Your output will look similar:

    bash
    blockHash               0x131822bef6eb59656d7e1387c19b75be667e587006710365ec5cf58030786c42
    +blockNumber             3
    +contractAddress
    +cumulativeGasUsed       43494
    +effectiveGasPrice       3767182372
    +gasUsed                 43494
    +logs                    []
    +logsBloom               0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
    +root
    +status                  1
    +transactionHash         0x8f15d6004598f0662dd673a9898dceef77be8cc28408cecc284b28d7be32307d
    +transactionIndex        0
    +type                    2
    blockHash               0x131822bef6eb59656d7e1387c19b75be667e587006710365ec5cf58030786c42
    +blockNumber             3
    +contractAddress
    +cumulativeGasUsed       43494
    +effectiveGasPrice       3767182372
    +gasUsed                 43494
    +logs                    []
    +logsBloom               0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
    +root
    +status                  1
    +transactionHash         0x8f15d6004598f0662dd673a9898dceef77be8cc28408cecc284b28d7be32307d
    +transactionIndex        0
    +type                    2

    Now, we can make a read call to view the state of the number variable, using the cast call command:

    bash
    cast call $CONTRACT_ADDRESS "number()" --rpc-url $BUBS_RPC_URL
    cast call $CONTRACT_ADDRESS "number()" --rpc-url $BUBS_RPC_URL

    The result will look similar:

    bash
    0x000000000000000000000000000000000000000000000000000000000000000a
    0x000000000000000000000000000000000000000000000000000000000000000a

    Convert the result from hexadecimal to a base 10 value with:

    bash
    echo $((0x000000000000000000000000000000000000000000000000000000000000000a))
    echo $((0x000000000000000000000000000000000000000000000000000000000000000a))

    Next steps

    Congratulations! You've learned how to deploy a smart contract to Bubs testnet.

    What will you build next? Now, you're ready to check out the GM Portal tutorial.

    + + + + \ No newline at end of file diff --git a/developers/ethereum-fallback.html b/developers/ethereum-fallback.html new file mode 100644 index 00000000000..f1ebe15e2db --- /dev/null +++ b/developers/ethereum-fallback.html @@ -0,0 +1,45 @@ + + + + + + Ethereum fallback | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Ethereum fallback

    Ethereum fallback is a fallback mechanism that enables Ethereum L2s (or L3s) to “fall back” to using Ethereum calldata for data availability in the event of downtime on Celestia Mainnet Beta. This feature is currently supported by Celestia integrations with:

    In the case of Celestia downtime or temporary unavailability, L2s can fallback to posting transactions as calldata on Ethereum or another DA layer for data availability instead of posting to Celestia. This mechanism ensures users can continue to transact securely and seamlessly, preventing disruptions and helping to ensure user funds do not get stuck in the L2's bridge on Ethereum.

    Ethereum fallback is triggered whenever the sequencer has an error sending the PayForBlobs transaction on Celestia. Fallback can be triggered due to a congested mempool or nonce error and can be simulated with an error such as low balance or incorrect sequence. Fallback can also be triggered in the event Blobstream stops relaying attestations.

    Ethereum fallback

    + + + + \ No newline at end of file diff --git a/developers/full-stack-modular-development-guide.html b/developers/full-stack-modular-development-guide.html new file mode 100644 index 00000000000..c2026eb4806 --- /dev/null +++ b/developers/full-stack-modular-development-guide.html @@ -0,0 +1,937 @@ + + + + + + Full stack modular blockchain development guide | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Full stack modular blockchain development guide

    Note

    This tutorial needs to be updated

    This guide will introduce you to modular blockchains like Celestia, explain their benefits, and show you how to build a full stack modular dapp with React, Vite, RainbowKit, Celestia, and Foundry.

    Current blockchain architectures are not scalable and face challenges around accessibility. In order for blockchains and web3 to reach mass adoption, these challenges must be addressed.

    Blockchains have evolved over time from application-specific networks like Bitcoin to shared smart contract platforms like Ethereum. This guide will cover how to build dapps on these newer, shared platforms.

    If you're interested in learning more about modular blockchains, or are new to the Celestia ecosystem, we recommend you read the build whatever page first.

    Getting started

    Now that you’ve had an overview of what Celestia is, let’s start building!

    The execution environment that we’ll be leveraging today is Ethermint, an EVM-compatible testnet that you will run locally for this tutorial.

    Pre-requisites

    Project setup

    To get started, create a new Foundry project:

    bash
    forge init celestia-dapp
    +cd celestia-dapp
    forge init celestia-dapp
    +cd celestia-dapp

    Foundry has created an example smart contract located at src/Contract.sol.

    Updating the contract and tests

    Let's update the contracts to include a basic blog example. Create a new file in the src directory named Contract.sol with the following code:

    solidity
    // SPDX-License-Identifier: MIT
    +pragma solidity ^0.8.13;
    +
    +contract Blog {
    +    string public name;
    +    address public owner;
    +
    +    uint private _postId;
    +
    +    struct Post {
    +      uint id;
    +      string title;
    +      string content;
    +      bool published;
    +    }
    +    /* mappings can be seen as hash tables */
    +    /* here we create lookups for posts by id and posts by ipfs hash */
    +    mapping(uint => Post) private idToPost;
    +    mapping(string => Post) private hashToPost;
    +
    +    /* events facilitate communication between smart contracts and their user interfaces  */
    +    /* i.e. we can create listeners for events in the client and also use them in The Graph  */
    +    event PostCreated(uint id, string title, string hash);
    +    event PostUpdated(uint id, string title, string hash, bool published);
    +
    +    /* when the blog is deployed, give it a name */
    +    /* also set the creator as the owner of the contract */
    +    constructor(string memory _name) {
    +        name = _name;
    +        owner = msg.sender;
    +    }
    +
    +    /* updates the blog name */
    +    function updateName(string memory _name) public {
    +        name = _name;
    +    }
    +
    +    /* transfers ownership of the contract to another address */
    +    function transferOwnership(address newOwner) public onlyOwner {
    +        owner = newOwner;
    +    }
    +
    +    /* fetches an individual post by the content hash */
    +    function fetchPost(string memory hash) public view returns(Post memory){
    +      return hashToPost[hash];
    +    }
    +
    +    /* creates a new post */
    +    function createPost(string memory title, string memory hash) public onlyOwner {
    +        _postId = _postId + 1;
    +        Post storage post = idToPost[_postId];
    +        post.id = _postId;
    +        post.title = title;
    +        post.published = true;
    +        post.content = hash;
    +        hashToPost[hash] = post;
    +        emit PostCreated(_postId, title, hash);
    +    }
    +
    +    /* updates an existing post */
    +    function updatePost(uint postId, string memory title, string memory hash, bool published) public onlyOwner {
    +        Post storage post =  idToPost[postId];
    +        post.title = title;
    +        post.published = published;
    +        post.content = hash;
    +        idToPost[postId] = post;
    +        hashToPost[hash] = post;
    +        emit PostUpdated(post.id, title, hash, published);
    +    }
    +
    +    /* fetches all posts */
    +    function fetchPosts() public view returns (Post[] memory) {
    +        uint itemCount = _postId;
    +
    +        Post[] memory posts = new Post[](itemCount);
    +        for (uint i = 0; i < itemCount; i++) {
    +            uint currentId = i + 1;
    +            Post storage currentItem = idToPost[currentId];
    +            posts[i] = currentItem;
    +        }
    +        return posts;
    +    }
    +
    +    /* this modifier means only the contract owner can */
    +    /* invoke the function */
    +    modifier onlyOwner() {
    +      require(msg.sender == owner);
    +    _;
    +  }
    +}
    // SPDX-License-Identifier: MIT
    +pragma solidity ^0.8.13;
    +
    +contract Blog {
    +    string public name;
    +    address public owner;
    +
    +    uint private _postId;
    +
    +    struct Post {
    +      uint id;
    +      string title;
    +      string content;
    +      bool published;
    +    }
    +    /* mappings can be seen as hash tables */
    +    /* here we create lookups for posts by id and posts by ipfs hash */
    +    mapping(uint => Post) private idToPost;
    +    mapping(string => Post) private hashToPost;
    +
    +    /* events facilitate communication between smart contracts and their user interfaces  */
    +    /* i.e. we can create listeners for events in the client and also use them in The Graph  */
    +    event PostCreated(uint id, string title, string hash);
    +    event PostUpdated(uint id, string title, string hash, bool published);
    +
    +    /* when the blog is deployed, give it a name */
    +    /* also set the creator as the owner of the contract */
    +    constructor(string memory _name) {
    +        name = _name;
    +        owner = msg.sender;
    +    }
    +
    +    /* updates the blog name */
    +    function updateName(string memory _name) public {
    +        name = _name;
    +    }
    +
    +    /* transfers ownership of the contract to another address */
    +    function transferOwnership(address newOwner) public onlyOwner {
    +        owner = newOwner;
    +    }
    +
    +    /* fetches an individual post by the content hash */
    +    function fetchPost(string memory hash) public view returns(Post memory){
    +      return hashToPost[hash];
    +    }
    +
    +    /* creates a new post */
    +    function createPost(string memory title, string memory hash) public onlyOwner {
    +        _postId = _postId + 1;
    +        Post storage post = idToPost[_postId];
    +        post.id = _postId;
    +        post.title = title;
    +        post.published = true;
    +        post.content = hash;
    +        hashToPost[hash] = post;
    +        emit PostCreated(_postId, title, hash);
    +    }
    +
    +    /* updates an existing post */
    +    function updatePost(uint postId, string memory title, string memory hash, bool published) public onlyOwner {
    +        Post storage post =  idToPost[postId];
    +        post.title = title;
    +        post.published = published;
    +        post.content = hash;
    +        idToPost[postId] = post;
    +        hashToPost[hash] = post;
    +        emit PostUpdated(post.id, title, hash, published);
    +    }
    +
    +    /* fetches all posts */
    +    function fetchPosts() public view returns (Post[] memory) {
    +        uint itemCount = _postId;
    +
    +        Post[] memory posts = new Post[](itemCount);
    +        for (uint i = 0; i < itemCount; i++) {
    +            uint currentId = i + 1;
    +            Post storage currentItem = idToPost[currentId];
    +            posts[i] = currentItem;
    +        }
    +        return posts;
    +    }
    +
    +    /* this modifier means only the contract owner can */
    +    /* invoke the function */
    +    modifier onlyOwner() {
    +      require(msg.sender == owner);
    +    _;
    +  }
    +}

    Next, let's create a test for this contract.

    Open test/Contract.t.sol and update the code with the following:

    solidity
    // SPDX-License-Identifier: MIT
    +pragma solidity ^0.8.13;
    +
    +import "forge-std/Test.sol";
    +import "src/Contract.sol";
    +
    +contract ContractTest is Test {
    +    Blog blog;
    +
    +    function setUp() public {
    +        blog = new Blog("Celestia Blog");
    +    }
    +
    +    function testCreatePost() public {
    +        blog.createPost("My first post", "12345");
    +        Blog.Post[] memory posts = blog.fetchPosts();
    +        assertEq(posts.length, 1);
    +    }
    +
    +    function testUpdatePost() public {
    +        blog.createPost("My first post", "12345");
    +        blog.updatePost(1, "My second post", "12345", true);
    +        Blog.Post memory updatedPost = blog.fetchPost("12345");
    +        assertEq(updatedPost.title, "My second post");
    +    }
    +
    +    function testFetchPosts() public {
    +        Blog.Post[] memory posts = blog.fetchPosts();
    +        assertEq(posts.length, 0);
    +        blog.createPost("My first post", "12345");
    +        posts = blog.fetchPosts();
    +        assertEq(posts.length, 1);
    +    }
    +
    +    function testOnlyOwner() public {
    +        blog.createPost("My first post", "12345");
    +        address bob = address(0x1);
    +        vm.startPrank(bob);
    +        vm.expectRevert();
    +        blog.updatePost(1, "My second post", "12345", true);
    +    }
    +}
    // SPDX-License-Identifier: MIT
    +pragma solidity ^0.8.13;
    +
    +import "forge-std/Test.sol";
    +import "src/Contract.sol";
    +
    +contract ContractTest is Test {
    +    Blog blog;
    +
    +    function setUp() public {
    +        blog = new Blog("Celestia Blog");
    +    }
    +
    +    function testCreatePost() public {
    +        blog.createPost("My first post", "12345");
    +        Blog.Post[] memory posts = blog.fetchPosts();
    +        assertEq(posts.length, 1);
    +    }
    +
    +    function testUpdatePost() public {
    +        blog.createPost("My first post", "12345");
    +        blog.updatePost(1, "My second post", "12345", true);
    +        Blog.Post memory updatedPost = blog.fetchPost("12345");
    +        assertEq(updatedPost.title, "My second post");
    +    }
    +
    +    function testFetchPosts() public {
    +        Blog.Post[] memory posts = blog.fetchPosts();
    +        assertEq(posts.length, 0);
    +        blog.createPost("My first post", "12345");
    +        posts = blog.fetchPosts();
    +        assertEq(posts.length, 1);
    +    }
    +
    +    function testOnlyOwner() public {
    +        blog.createPost("My first post", "12345");
    +        address bob = address(0x1);
    +        vm.startPrank(bob);
    +        vm.expectRevert();
    +        blog.updatePost(1, "My second post", "12345", true);
    +    }
    +}

    Foundry uses Dappsys Test to provide basic logging and assertion functionality. It's included in the Forge Standard Library.

    Here, we are using assertEq to assert equality. You can view all of the assertion functions available.

    Running the test

    We can now run our tests to make sure our contract is working properly:

    bash
    forge test -vv
    forge test -vv

    Updating the deployment script

    Now that we've tested the contract, let's try deploying it locally using Solidity Scripting.

    To do so, update the deployment script at script/Contract.s.sol with the following code:

    solidity
    // SPDX-License-Identifier: MIT
    +pragma solidity ^0.8.13;
    +
    +import "forge-std/Script.sol";
    +
    +import {Blog} from "src/Contract.sol";
    +
    +contract ContractScript is Script {
    +    function setUp() public {}
    +
    +    function run() public {
    +        vm.startBroadcast();
    +        new Blog("Celestia Blog");
    +        vm.stopBroadcast();
    +    }
    +}
    // SPDX-License-Identifier: MIT
    +pragma solidity ^0.8.13;
    +
    +import "forge-std/Script.sol";
    +
    +import {Blog} from "src/Contract.sol";
    +
    +contract ContractScript is Script {
    +    function setUp() public {}
    +
    +    function run() public {
    +        vm.startBroadcast();
    +        new Blog("Celestia Blog");
    +        vm.stopBroadcast();
    +    }
    +}

    Now we can use this script to deploy our smart contract to either a live or test network.

    Deploying locally

    Next start Anvil, the local testnet:

    bash
    anvil --port 9545
    anvil --port 9545

    caution

    We need to use port 9545, because Ethermint will use 8545.

    Once started, Anvil will give you a local RPC endpoint as well as a handful of Private Keys and Accounts that you can use.

    We can now use the local RPC along with one of the private keys to deploy locally:

    bash
    forge script script/Contract.s.sol:ContractScript --fork-url \
    +http://localhost:9545 --private-key $PRIVATE_KEY --broadcast
    forge script script/Contract.s.sol:ContractScript --fork-url \
    +http://localhost:9545 --private-key $PRIVATE_KEY --broadcast

    Once the contract has been deployed locally, Anvil will log out the contract address.

    Take a note of this local contract address as we’ll be using it later in the frontend application.

    Next, set the contract address as an environment variable:

    bash
    export CONTRACT_ADDRESS=<contract-address>
    export CONTRACT_ADDRESS=<contract-address>

    We can then test sending transactions to it with cast send.

    bash
    cast send $CONTRACT_ADDRESS \
    +"createPost(string,string)" "my first post" "12345" \
    +--private-key $PRIVATE_KEY
    cast send $CONTRACT_ADDRESS \
    +"createPost(string,string)" "my first post" "12345" \
    +--private-key $PRIVATE_KEY

    We can then perform read operations with cast call:

    bash
    cast call $CONTRACT_ADDRESS "fetchPosts()"
    cast call $CONTRACT_ADDRESS "fetchPosts()"

    Once the contract is deployed successfully, take a note of the contract address as we’ll also be needing it in just a moment when we test the live contract.

    Deploying to the Ethermint Sovereign Rollup

    First, we will need to follow the setup from the EVM tutorial.

    Pre-requisites

    It is required that you complete dependency setup, Rollkit installation, and Instantiating and EVM rollup from the EVM tutorial to complete the remainder of the tutorial.

    Now that we've deployed and tested locally, we can deploy to our Ethermint chain.

    First, we will need to export the private key generated by the ethermint init.sh script:

    bash
    PRIVATE_KEY=$(ethermintd keys unsafe-export-eth-key mykey --keyring-backend test)
    PRIVATE_KEY=$(ethermintd keys unsafe-export-eth-key mykey --keyring-backend test)

    NOTE

    Here, the key name from init.sh is mykey but you can modify the init.sh to change the name of your key.

    Now, we can start deploying the smart contract to our Ethermint chain.

    To do so, run the following script in the celestia-dapp directory:

    bash
    forge script script/Contract.s.sol:ContractScript \
    +--rpc-url http://localhost:8545 --private-key $PRIVATE_KEY --broadcast
    forge script script/Contract.s.sol:ContractScript \
    +--rpc-url http://localhost:8545 --private-key $PRIVATE_KEY --broadcast

    Set the contract address in the output as the CONTRACT_ADDRESS variable:

    bash
    export CONTRACT_ADDRESS=<new-contract-address>
    export CONTRACT_ADDRESS=<new-contract-address>

    Once the contract has been deployed to the Ethermint rollup, we can use cast send to test sending transactions to it:

    bash
    cast send $CONTRACT_ADDRESS \
    +"createPost(string,string)" "my first post" "12345" \
    +--rpc-url http://localhost:8545 --private-key $PRIVATE_KEY
    cast send $CONTRACT_ADDRESS \
    +"createPost(string,string)" "my first post" "12345" \
    +--rpc-url http://localhost:8545 --private-key $PRIVATE_KEY

    We can then perform read operations with cast call:

    bash
    cast call $CONTRACT_ADDRESS "fetchPosts()" --rpc-url http://localhost:8545
    cast call $CONTRACT_ADDRESS "fetchPosts()" --rpc-url http://localhost:8545

    NOTE

    You will want to redeploy the contract for your frontend, because the post is not uploaded to IPFS in the CLI.

    Building the frontend

    For the frontend project, we’ll be using the following libraries and frameworks:

    React - JavaScript library for building user interfaces

    Vite - Project generator / rapid development tool for modern web projects

    Rainbowkit - Easy and beautiful library to connect a wallet

    WAGMI - 20+ hooks for working with wallets, ENS, contracts, transactions, signing, etc

    In the root of the Foundry project, create a new React.js application using Vite:

    jsx
    yarn create vite
    +
    +? Project name: › frontend
    +? Select a framework › React
    +? Select a variant > JavaScript
    yarn create vite
    +
    +? Project name: › frontend
    +? Select a framework › React
    +? Select a variant > JavaScript

    Next, copy the ABI that was created by Foundry into the frontend directory so that we can have it later (or manually copy it into a file named Blog.json in the frontend directory):

    bash
    cp out/Contract.sol/Blog.json frontend/
    cp out/Contract.sol/Blog.json frontend/

    Now, change into the frontend directory and install the node_modules:

    bash
    cd frontend
    +yarn
    cd frontend
    +yarn

    Configuring environment variables

    Next we need to configure the environment variables for the Infura project ID and secret.

    First, create an Infura account and new project for IPFS.

    Create a file named .env.local in the frontend/ directory and add the following configuration with your own credentials:

    txt
    VITE_INFURA_ID=your-project-api-key
    +VITE_INFURA_SECRET=your-project-api-key-secret
    VITE_INFURA_ID=your-project-api-key
    +VITE_INFURA_SECRET=your-project-api-key-secret

    Now that the project is created, let’s install the additional dependencies using either NPM, Yarn, or PNPM:

    jsx
    npm install @rainbow-me/rainbowkit@0.8.0 wagmi@0.8.10 ethers ipfs-http-client react-markdown
    npm install @rainbow-me/rainbowkit@0.8.0 wagmi@0.8.10 ethers ipfs-http-client react-markdown

    Configuring the entrypoint

    Next we’ll update the entrypoint at src/main.jsx.

    The main things we’re doing here have to do with the configuration of Rainbowkit so that we can have a nice way for the user to connect their wallet.

    Rainbowkit also allows a customizable array of network providers, so we’re creating a new network configuration for Ethermint.

    jsx
    import "./polyfills";
    +import ReactDOM from "react-dom/client";
    +import App from "./App";
    +import "./index.css";
    +import "@rainbow-me/rainbowkit/styles.css";
    +import { RainbowKitProvider } from "@rainbow-me/rainbowkit";
    +import { chain, configureChains, createClient, WagmiConfig } from "wagmi";
    +import { publicProvider } from "wagmi/providers/public";
    +import { injectedWallet, metaMaskWallet } from "@rainbow-me/rainbowkit/wallets";
    +import { connectorsForWallets } from "@rainbow-me/rainbowkit";
    +
    +/* create configuration for Ethermint testnet */
    +const ethermint = {
    +  id: 9000,
    +  name: "Ethermint",
    +  network: "ethermint",
    +  nativeCurrency: {
    +    decimals: 18,
    +    name: "Ethermint",
    +    symbol: "CTE",
    +  },
    +  rpcUrls: {
    +    default: {
    +      http: ["http://localhost:8545/"],
    +    },
    +  },
    +  testnet: true,
    +};
    +
    +// remove chain.localhost or ethermint depending on which you want to connect to
    +const { chains, provider } = configureChains(
    +  [chain.localhost, ethermint],
    +  [publicProvider()],
    +);
    +
    +const connectors = connectorsForWallets([
    +  {
    +    groupName: "Recommended",
    +    wallets: [metaMaskWallet({ chains }), injectedWallet({ chains })],
    +  },
    +]);
    +
    +const wagmiClient = createClient({
    +  autoConnect: true,
    +  connectors,
    +  provider,
    +});
    +
    +const containerStyle = {
    +  width: "900px",
    +  margin: "0 auto",
    +};
    +
    +ReactDOM.createRoot(document.getElementById("root")).render(
    +  <WagmiConfig client={wagmiClient}>
    +    <RainbowKitProvider chains={chains}>
    +      <div style={containerStyle}>
    +        <App />
    +      </div>
    +    </RainbowKitProvider>
    +  </WagmiConfig>,
    +);
    import "./polyfills";
    +import ReactDOM from "react-dom/client";
    +import App from "./App";
    +import "./index.css";
    +import "@rainbow-me/rainbowkit/styles.css";
    +import { RainbowKitProvider } from "@rainbow-me/rainbowkit";
    +import { chain, configureChains, createClient, WagmiConfig } from "wagmi";
    +import { publicProvider } from "wagmi/providers/public";
    +import { injectedWallet, metaMaskWallet } from "@rainbow-me/rainbowkit/wallets";
    +import { connectorsForWallets } from "@rainbow-me/rainbowkit";
    +
    +/* create configuration for Ethermint testnet */
    +const ethermint = {
    +  id: 9000,
    +  name: "Ethermint",
    +  network: "ethermint",
    +  nativeCurrency: {
    +    decimals: 18,
    +    name: "Ethermint",
    +    symbol: "CTE",
    +  },
    +  rpcUrls: {
    +    default: {
    +      http: ["http://localhost:8545/"],
    +    },
    +  },
    +  testnet: true,
    +};
    +
    +// remove chain.localhost or ethermint depending on which you want to connect to
    +const { chains, provider } = configureChains(
    +  [chain.localhost, ethermint],
    +  [publicProvider()],
    +);
    +
    +const connectors = connectorsForWallets([
    +  {
    +    groupName: "Recommended",
    +    wallets: [metaMaskWallet({ chains }), injectedWallet({ chains })],
    +  },
    +]);
    +
    +const wagmiClient = createClient({
    +  autoConnect: true,
    +  connectors,
    +  provider,
    +});
    +
    +const containerStyle = {
    +  width: "900px",
    +  margin: "0 auto",
    +};
    +
    +ReactDOM.createRoot(document.getElementById("root")).render(
    +  <WagmiConfig client={wagmiClient}>
    +    <RainbowKitProvider chains={chains}>
    +      <div style={containerStyle}>
    +        <App />
    +      </div>
    +    </RainbowKitProvider>
    +  </WagmiConfig>,
    +);

    Creating and reading posts

    Now that the base configuration is set up we’ll create a view that allows users to create and view posts.

    We’ll be using IPFS to upload the content of the post, then anchoring the hash of the post on chain. When we retrieve the post, we can then read the value from IPFS to view the post.

    Update App.jsx with the following code:

    jsx
    import { useState, useEffect } from "react";
    +import { ConnectButton } from "@rainbow-me/rainbowkit";
    +import { ethers } from "ethers";
    +import { create } from "ipfs-http-client";
    +import { Buffer } from "buffer";
    +import Blog from "../Blog.json";
    +import { useAccount } from "wagmi";
    +
    +/* configure authorization for Infura and IPFS */
    +const auth =
    +  "Basic " +
    +  Buffer.from(
    +    import.meta.env.VITE_INFURA_ID + ":" + import.meta.env.VITE_INFURA_SECRET,
    +  ).toString("base64");
    +
    +/* create an IPFS client */
    +const client = create({
    +  host: "ipfs.infura.io",
    +  port: 5001,
    +  protocol: "https",
    +  headers: {
    +    authorization: auth,
    +  },
    +});
    +
    +const contractAddress = "your-ethermint-contract-address";
    +
    +function App() {
    +  useEffect(() => {
    +    fetchPosts();
    +  }, []);
    +  const [viewState, setViewState] = useState("view-posts");
    +  const [posts, setPosts] = useState([]);
    +  const [title, setTitle] = useState("");
    +  const [content, setContent] = useState("");
    +  const { address } = useAccount();
    +
    +  /* when the component loads, useEffect will call this function */
    +  async function fetchPosts() {
    +    const provider = new ethers.providers.Web3Provider(window.ethereum);
    +    const contract = new ethers.Contract(contractAddress, Blog.abi, provider);
    +    let data = await contract.fetchPosts();
    +    /* once the data is returned from the network we map over it and */
    +    /* transform the data into a more readable format  */
    +    data = data.map((d) => ({
    +      content: d["content"],
    +      title: d["title"],
    +      published: d["published"],
    +      id: d["id"].toString(),
    +    }));
    +
    +    /* we then fetch the post content from IPFS and add it to the post objects */
    +    data = await Promise.all(
    +      data.map(async (d) => {
    +        const endpoint = `https://infura-ipfs.io/ipfs/${d.content}`;
    +        const options = {
    +          mode: "no-cors",
    +        };
    +        const response = await fetch(endpoint, options);
    +        const value = await response.text();
    +        d.postContent = value;
    +        return d;
    +      }),
    +    );
    +
    +    setPosts(data);
    +  }
    +
    +  async function createPost() {
    +    const added = await client.add(content);
    +    const provider = new ethers.providers.Web3Provider(window.ethereum);
    +    const signer = provider.getSigner();
    +
    +    const contract = new ethers.Contract(contractAddress, Blog.abi, signer);
    +    const tx = await contract.createPost(title, added.path);
    +    await tx.wait();
    +    setViewState("view-posts");
    +  }
    +
    +  function toggleView(value) {
    +    setViewState(value);
    +    if (value === "view-posts") {
    +      fetchPosts();
    +    }
    +  }
    +
    +  return (
    +    <div style={outerContainerStyle}>
    +      <div style={innerContainerStyle}>
    +        <h1>Modular Rollup Blog</h1>
    +        <p>
    +          This allows users to securely create and share blog posts on the
    +          blockchain without the need for a centralized server or authority.
    +        </p>
    +        {!address ? (
    +          <div>
    +            <h3>Getting Started</h3>
    +            <p>
    +              First, you will need to connect your Ethereum wallet to Ethermint
    +              to display the posts from the smart contract and make posts.
    +            </p>
    +          </div>
    +        ) : null}
    +        <br />
    +        <h3 style={{ justifyContent: "right", textAlign: "right" }}>
    +          Connect your Ethereum wallet to begin ✨
    +        </h3>
    +        <div style={buttonContainerStyle}>
    +          <ConnectButton />
    +        </div>
    +        {address ? (
    +          <div style={buttonContainerStyle}>
    +            <button
    +              onClick={() => toggleView("view-posts")}
    +              style={buttonStyle}
    +            >
    +              View Posts
    +            </button>
    +            <button
    +              onClick={() => toggleView("create-post")}
    +              style={buttonStyle}
    +            >
    +              Create Post
    +            </button>
    +          </div>
    +        ) : null}
    +        {viewState === "view-posts" && address && (
    +          <div>
    +            <div style={postContainerStyle}>
    +              <h1>Posts</h1>
    +              {posts.map((post, index) => (
    +                <div key={index}>
    +                  <h2>{post.title}</h2>
    +                  <button
    +                    style={{ fontSize: "16px" }}
    +                    onClick={() =>
    +                      window.open(`https://infura-ipfs.io/ipfs/${post.content}`)
    +                    }
    +                  >
    +                    Read on IPFS
    +                  </button>
    +                  {/* <ReactMarkdown>
    +                    {post.postContent}
    +                  </ReactMarkdown> */}
    +                  <p style={mbidStyle}>GMID: {post.id}</p>
    +                </div>
    +              ))}
    +            </div>
    +          </div>
    +        )}
    +        {viewState === "create-post" && (
    +          <div style={formContainerStyle}>
    +            <h2>Create Post</h2>
    +            <input
    +              placeholder="Title"
    +              onChange={(e) => setTitle(e.target.value)}
    +              style={inputStyle}
    +            />
    +            <textarea
    +              placeholder="Content"
    +              onChange={(e) => setContent(e.target.value)}
    +              style={inputStyle}
    +            />
    +            <button onClick={createPost}>Create Post</button>
    +          </div>
    +        )}
    +      </div>
    +    </div>
    +  );
    +}
    +
    +const outerContainerStyle = {
    +  width: "90vw",
    +  height: "100vh",
    +  padding: "50px 0px",
    +};
    +
    +const innerContainerStyle = {
    +  width: "100%",
    +  maxWidth: "800px",
    +  margin: "0 auto",
    +};
    +
    +const formContainerStyle = {
    +  display: "flex",
    +  flexDirection: "column",
    +  alignItems: "center",
    +};
    +
    +const inputStyle = {
    +  width: "400px",
    +  marginBottom: "10px",
    +  padding: "10px",
    +  height: "40px",
    +};
    +
    +const postContainerStyle = {
    +  margin: "0 auto",
    +  padding: "1em",
    +  width: "90%",
    +  maxWidth: "800px",
    +  display: "flex",
    +  flexDirection: "column",
    +  alignItems: "start",
    +  justifyContent: "center",
    +};
    +
    +const mbidStyle = {
    +  fontSize: "10px",
    +  textAlign: "start",
    +};
    +
    +const buttonStyle = {
    +  marginTop: 15,
    +  marginRight: 5,
    +  border: "1px solid rgba(255, 255, 255, .2)",
    +};
    +
    +const buttonContainerStyle = {
    +  marginTop: 15,
    +  marginRight: 5,
    +  display: "flex",
    +  justifyContent: "right",
    +};
    +
    +export default App;
    import { useState, useEffect } from "react";
    +import { ConnectButton } from "@rainbow-me/rainbowkit";
    +import { ethers } from "ethers";
    +import { create } from "ipfs-http-client";
    +import { Buffer } from "buffer";
    +import Blog from "../Blog.json";
    +import { useAccount } from "wagmi";
    +
    +/* configure authorization for Infura and IPFS */
    +const auth =
    +  "Basic " +
    +  Buffer.from(
    +    import.meta.env.VITE_INFURA_ID + ":" + import.meta.env.VITE_INFURA_SECRET,
    +  ).toString("base64");
    +
    +/* create an IPFS client */
    +const client = create({
    +  host: "ipfs.infura.io",
    +  port: 5001,
    +  protocol: "https",
    +  headers: {
    +    authorization: auth,
    +  },
    +});
    +
    +const contractAddress = "your-ethermint-contract-address";
    +
    +function App() {
    +  useEffect(() => {
    +    fetchPosts();
    +  }, []);
    +  const [viewState, setViewState] = useState("view-posts");
    +  const [posts, setPosts] = useState([]);
    +  const [title, setTitle] = useState("");
    +  const [content, setContent] = useState("");
    +  const { address } = useAccount();
    +
    +  /* when the component loads, useEffect will call this function */
    +  async function fetchPosts() {
    +    const provider = new ethers.providers.Web3Provider(window.ethereum);
    +    const contract = new ethers.Contract(contractAddress, Blog.abi, provider);
    +    let data = await contract.fetchPosts();
    +    /* once the data is returned from the network we map over it and */
    +    /* transform the data into a more readable format  */
    +    data = data.map((d) => ({
    +      content: d["content"],
    +      title: d["title"],
    +      published: d["published"],
    +      id: d["id"].toString(),
    +    }));
    +
    +    /* we then fetch the post content from IPFS and add it to the post objects */
    +    data = await Promise.all(
    +      data.map(async (d) => {
    +        const endpoint = `https://infura-ipfs.io/ipfs/${d.content}`;
    +        const options = {
    +          mode: "no-cors",
    +        };
    +        const response = await fetch(endpoint, options);
    +        const value = await response.text();
    +        d.postContent = value;
    +        return d;
    +      }),
    +    );
    +
    +    setPosts(data);
    +  }
    +
    +  async function createPost() {
    +    const added = await client.add(content);
    +    const provider = new ethers.providers.Web3Provider(window.ethereum);
    +    const signer = provider.getSigner();
    +
    +    const contract = new ethers.Contract(contractAddress, Blog.abi, signer);
    +    const tx = await contract.createPost(title, added.path);
    +    await tx.wait();
    +    setViewState("view-posts");
    +  }
    +
    +  function toggleView(value) {
    +    setViewState(value);
    +    if (value === "view-posts") {
    +      fetchPosts();
    +    }
    +  }
    +
    +  return (
    +    <div style={outerContainerStyle}>
    +      <div style={innerContainerStyle}>
    +        <h1>Modular Rollup Blog</h1>
    +        <p>
    +          This allows users to securely create and share blog posts on the
    +          blockchain without the need for a centralized server or authority.
    +        </p>
    +        {!address ? (
    +          <div>
    +            <h3>Getting Started</h3>
    +            <p>
    +              First, you will need to connect your Ethereum wallet to Ethermint
    +              to display the posts from the smart contract and make posts.
    +            </p>
    +          </div>
    +        ) : null}
    +        <br />
    +        <h3 style={{ justifyContent: "right", textAlign: "right" }}>
    +          Connect your Ethereum wallet to begin ✨
    +        </h3>
    +        <div style={buttonContainerStyle}>
    +          <ConnectButton />
    +        </div>
    +        {address ? (
    +          <div style={buttonContainerStyle}>
    +            <button
    +              onClick={() => toggleView("view-posts")}
    +              style={buttonStyle}
    +            >
    +              View Posts
    +            </button>
    +            <button
    +              onClick={() => toggleView("create-post")}
    +              style={buttonStyle}
    +            >
    +              Create Post
    +            </button>
    +          </div>
    +        ) : null}
    +        {viewState === "view-posts" && address && (
    +          <div>
    +            <div style={postContainerStyle}>
    +              <h1>Posts</h1>
    +              {posts.map((post, index) => (
    +                <div key={index}>
    +                  <h2>{post.title}</h2>
    +                  <button
    +                    style={{ fontSize: "16px" }}
    +                    onClick={() =>
    +                      window.open(`https://infura-ipfs.io/ipfs/${post.content}`)
    +                    }
    +                  >
    +                    Read on IPFS
    +                  </button>
    +                  {/* <ReactMarkdown>
    +                    {post.postContent}
    +                  </ReactMarkdown> */}
    +                  <p style={mbidStyle}>GMID: {post.id}</p>
    +                </div>
    +              ))}
    +            </div>
    +          </div>
    +        )}
    +        {viewState === "create-post" && (
    +          <div style={formContainerStyle}>
    +            <h2>Create Post</h2>
    +            <input
    +              placeholder="Title"
    +              onChange={(e) => setTitle(e.target.value)}
    +              style={inputStyle}
    +            />
    +            <textarea
    +              placeholder="Content"
    +              onChange={(e) => setContent(e.target.value)}
    +              style={inputStyle}
    +            />
    +            <button onClick={createPost}>Create Post</button>
    +          </div>
    +        )}
    +      </div>
    +    </div>
    +  );
    +}
    +
    +const outerContainerStyle = {
    +  width: "90vw",
    +  height: "100vh",
    +  padding: "50px 0px",
    +};
    +
    +const innerContainerStyle = {
    +  width: "100%",
    +  maxWidth: "800px",
    +  margin: "0 auto",
    +};
    +
    +const formContainerStyle = {
    +  display: "flex",
    +  flexDirection: "column",
    +  alignItems: "center",
    +};
    +
    +const inputStyle = {
    +  width: "400px",
    +  marginBottom: "10px",
    +  padding: "10px",
    +  height: "40px",
    +};
    +
    +const postContainerStyle = {
    +  margin: "0 auto",
    +  padding: "1em",
    +  width: "90%",
    +  maxWidth: "800px",
    +  display: "flex",
    +  flexDirection: "column",
    +  alignItems: "start",
    +  justifyContent: "center",
    +};
    +
    +const mbidStyle = {
    +  fontSize: "10px",
    +  textAlign: "start",
    +};
    +
    +const buttonStyle = {
    +  marginTop: 15,
    +  marginRight: 5,
    +  border: "1px solid rgba(255, 255, 255, .2)",
    +};
    +
    +const buttonContainerStyle = {
    +  marginTop: 15,
    +  marginRight: 5,
    +  display: "flex",
    +  justifyContent: "right",
    +};
    +
    +export default App;

    Adding Ethermint Chain to MetaMask

    Before we can test out our dapp, we'll need to configure the chains on MetaMask if we're deploying our rollup any

    1. Open your MetaMask wallet and click "Ethereum Mainnet" to open the dropdown.
    2. Select "Add network"
    3. Then "Add network manually"
    4. Enter the following details:
    • Network Name: Ethermint
    • New RPC URL: http://localhost:8545 or https://your.custom.ip.address:port
    • Chain ID: 9000
    • Currency symbol: CTE

    Testing it out on Ethermint

    Now we’re ready to run the app.

    Right now, the app is configured to be using localhost:8545 using the Ethermint rollup we're running with Rollkit.

    First, you'll need to install MetaMask.

    To use the test account, you will need to import the private key from Ethermint to MetaMask. First, run the following command:

    bash
    PRIVATE_KEY=$(ethermintd keys unsafe-export-eth-key mykey --keyring-backend test)
    +&& echo $PRIVATE_KEY | pbcopy
    PRIVATE_KEY=$(ethermintd keys unsafe-export-eth-key mykey --keyring-backend test)
    +&& echo $PRIVATE_KEY | pbcopy

    Now, import the private key to MetaMask and switch to that account.

    Next, let’s run it on your Ethermint rollup.

    To do so, first update the contractAddress variable with the contract address deployed to Ethermint:

    jsx
    /* src/App.jsx */
    +const contractAddress = "your-ethermint-contract-address";
    /* src/App.jsx */
    +const contractAddress = "your-ethermint-contract-address";

    Next, run the React application:

    bash
    npm run dev
    npm run dev

    When you run the app, you should now be connected to and using the Ethermint rollup.

    If you imported the address that started the chain, you'll see quite a large balance.

    Now give it a spin 🌀

    Now that you have your dapp running, go ahead and test out a new post on your Ethermint sovereign rollup. If you enjoyed this tutorial, be sure to share your example in our Discord!

    + + + + \ No newline at end of file diff --git a/developers/gm-portal-bubs.html b/developers/gm-portal-bubs.html new file mode 100644 index 00000000000..5ec6a2eb9a3 --- /dev/null +++ b/developers/gm-portal-bubs.html @@ -0,0 +1,65 @@ + + + + + + Deploying a dapp on Bubs testnet | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Deploying a dapp on Bubs testnet

    First, review the Bubs testnet page and the Deploy a smart contract to Bubs testnet tutorial.

    You will need a funded account to deploy your smart contract.

    Next, clone the gm-portal from Github and start the frontend:

    bash
    cd $HOME
    +git clone https://github.com/jcstein/gm-portal.git
    +cd gm-portal/frontend
    +yarn && yarn dev
    cd $HOME
    +git clone https://github.com/jcstein/gm-portal.git
    +cd gm-portal/frontend
    +yarn && yarn dev

    In a new terminal instance, set your private key for the faucet as a variable and the RPC URL you're using:

    bash
    export PRIVATE_KEY=ac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80
    +export BUBS_RPC_URL=https://bubs-sepolia.rpc.caldera.xyz/http
    export PRIVATE_KEY=ac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80
    +export BUBS_RPC_URL=https://bubs-sepolia.rpc.caldera.xyz/http

    Now, change into the gm-portal/contracts directory in the same terminal and deploy the contract using Foundry:

    bash
    cd $HOME/gm-portal/contracts
    +forge script script/GmPortal.s.sol:GmPortalScript --rpc-url $BUBS_RPC_URL --private-key $PRIVATE_KEY --broadcast
    cd $HOME/gm-portal/contracts
    +forge script script/GmPortal.s.sol:GmPortalScript --rpc-url $BUBS_RPC_URL --private-key $PRIVATE_KEY --broadcast

    gm-contract

    In the output of the deployment, find the contract address and set it as a variable:

    bash
    export CONTRACT_ADDRESS=<your-contract-address-from-the-output-above>
    export CONTRACT_ADDRESS=<your-contract-address-from-the-output-above>

    Next, you're ready to interact with the contract from your terminal!

    First, send a "gm" to the contract:

    bash
    cast send $CONTRACT_ADDRESS \
    +"gm(string)" "gm" \
    +--private-key $PRIVATE_KEY \
    +--rpc-url $BUBS_RPC_URL
    cast send $CONTRACT_ADDRESS \
    +"gm(string)" "gm" \
    +--private-key $PRIVATE_KEY \
    +--rpc-url $BUBS_RPC_URL

    Now that you've posted to the contract, you can read all "gms" (GMs) from the contract with this command:

    bash
    cast call $CONTRACT_ADDRESS "getAllGms()" --rpc-url $BUBS_RPC_URL
    cast call $CONTRACT_ADDRESS "getAllGms()" --rpc-url $BUBS_RPC_URL

    Next, query the total number of gms, which will be returned as a hex value:

    bash
    cast call $CONTRACT_ADDRESS "getTotalGms()" --rpc-url $BUBS_RPC_URL
    cast call $CONTRACT_ADDRESS "getTotalGms()" --rpc-url $BUBS_RPC_URL

    In order to interact with the contract on the frontend, you'll need to fund an account that you have in your Ethereum wallet. Transfer to an external account with this command:

    bash
    export RECEIVER=<receiver ETH address>
    +cast send --private-key $PRIVATE_KEY $RECEIVER --value 1ether --rpc-url $BUBS_RPC_URL
    export RECEIVER=<receiver ETH address>
    +cast send --private-key $PRIVATE_KEY $RECEIVER --value 1ether --rpc-url $BUBS_RPC_URL

    If you are in a different terminal than the one you set the private key in, you may need to set it again.

    Update the frontend

    Next, you will need to update a few things before you can interact with the contract on the frontend:

    1. Change the contract address on gm-portal/frontend/src/App.tsx to your contract address
    2. Match the chain info on gm-portal/frontend/src/main.tsx with the chain config of your L2
    3. If you changed the contract, update the ABI in gm-portal/frontend/GmPortal.json from gm-portal/contracts/out/GmPortal.sol/GmPortal.json. This can be done with:
    bash
    cd $HOME
    +cp dev/gm-portal/contracts/out/GmPortal.sol/GmPortal.json dev/gm-portal/frontend
    cd $HOME
    +cp dev/gm-portal/contracts/out/GmPortal.sol/GmPortal.json dev/gm-portal/frontend

    Interact with the frontend

    Now, login with your wallet that you funded, and post a GM on your GM portal!

    gm-bubs

    Next steps

    There are many possibilities of what could be built with this stack. These projects would be good to build on this stack:

    • onchain gaming
    • decentralized social media
    • an NFT ticketing rollup
    • Optimism on CelOPstia
    • OP Craft on Celestia
    + + + + \ No newline at end of file diff --git a/developers/integrate-celestia.html b/developers/integrate-celestia.html new file mode 100644 index 00000000000..610091b3a90 --- /dev/null +++ b/developers/integrate-celestia.html @@ -0,0 +1,45 @@ + + + + + + Integrate Celestia for service providers | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Integrate Celestia for service providers

    This document is for third-party service providers, such as custodians and explorers, integrating the Celestia network.

    Getting started

    When getting started Celestia, we recommend checking out these resources first:

    Celestia service provider notes

    Celestia is a fairly standard Cosmos-SDK based chain. We use the latest version of Tendermint and the Cosmos-SDK, with only minor modifications to each. This means that we are:

    • Using the default Cosmos-SDK modules: auth, bank, distribution, staking, slashing, mint, crisis, ibchost, genutil, evidence, ibctransfer, params, gov (limited in some TBD capacities), upgrade, vesting, feegrant, capability, and payment.
    • Use the standard digital keys schemes provided by the Cosmos-SDK and Tendermint, those being secp256k1 for user transactions, and tm-ed25519 for signing and verifying consensus messages.

    While exactly which modules used is subject to change, Celestia aims to be as minimal as possible.

    Custody and key management

    Celestia supports many already existing key management systems, as we rely on the Cosmos-SDK and Tendermint libraries for signing and verifying transactions. Learn more in the Cosmos-SDK documentation

    RPC and querying

    In celestia-app, only the standard RPC endpoints for Tendermint and the Cosmos-SDK are exposed. We do not currently add or subtract any core functionality, but this could change in the future. The same goes for querying data from the chain.

    In celestia-node, the Data Availability node client, there is a JSON-RPC API that allows you to interact directly with Celestia's Data Availability layer. Learn how to use the API in this tutorial.

    Compatibility

    Linux, particularly Ubuntu 20.04 LTS, is the most well tested. Potentially compatible with other OSs, but they are currently untested. Some of the cryptography libraries used for erasure data are not guaranteed to work on other platforms.

    Syncing

    Since we utilize Tendermint and the Cosmos-SDK, syncing the chain can be performed by any method that is supported by those libraries. This includes fast-sync, state sync, and quick sync.

    Notable exceptions relative to other blockchains

    Relative to other Tendermint based chains, Celestia will have significantly longer blocktimes of roughly 12* seconds. The reason behind this block time is to optimize the bandwidth used by light clients that are sampling the chain, and is not because we have modified Tendermint consensus in any meaningful way. Validators will likely download/upload relatively large blocks. It should be noted that while these blocks are large, very little typical blockchain state execution is actually occurring on Celestia. Meaning that the bandwidth requirements will likely be larger than that of a typical Cosmos-SDK based blockchain full node, the computing requirements should be similar in magnitude.

    *Subject to Change

    + + + + \ No newline at end of file diff --git a/developers/intro-to-op-stack.html b/developers/intro-to-op-stack.html new file mode 100644 index 00000000000..cb8ba4d554d --- /dev/null +++ b/developers/intro-to-op-stack.html @@ -0,0 +1,45 @@ + + + + + + Introduction to OP Stack integration | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Introduction to OP Stack integration

    Optimism is a low-cost and lightning-fast Ethereum L2 blockchain, built with the OP Stack.

    Celestia is a modular consensus and data network, built to enable anyone to easily deploy their own blockchain with minimal overhead.

    About the integration

    Discover how to integrate existing blockchain frameworks like the OP Stack with Celestia in this category.

    TIP

    Tested on a machine with 8GB RAM, 160 GB SSD, Ubuntu 22.10, and a 4 core AMD CPU.

    This is a beta integration and we are working on resolving open issues.

    Optimism uses Ethereum as a data availability (DA) layer. Currently, settlement and DA for Optimism are on Ethereum, both onchain. op-batcher batches up rollup blocks and posts to Ethereum.

    Table of contents of the category

    Celestia and OP Stack repository

    Find the repository for this integration at https://github.com/celestiaorg/optimism.

    What are Optimism and the OP Stack?

    Optimism, an Ethereum L2 blockchain, is powered by the OP Stack, which is also the foundation for the Optimism Collective committed to the impact=profit principle. This rewards individuals for their positive contributions to the collective.

    Optimism addresses crypto ecosystem coordination failures, like funding public goods and infrastructure. The OP Stack fosters collaboration and prevents redundancy by creating a shared, open-source system for developing new L2 blockchains within the proposed Superchain ecosystem.

    As Optimism evolves, the OP Stack will adapt to include components from blockchain infrastructure to governance systems. This software suite aims to simplify L2 blockchain creation and support the Optimism ecosystem's growth and development.

    Learn more about Optimism.

    What is Celestia?

    Celestia is a modular consensus and data network, built to enable anyone to easily deploy their own blockchain with minimal overhead.

    Celestia is a minimal blockchain that only orders and publishes transactions and does not execute them. By decoupling the consensus and application execution layers, Celestia modularizes the blockchain technology stack and unlocks new possibilities for decentralized application builders. Learn more at Celestia.org.

    OP Stack and Celestia

    This category will guide you through how to start your own devnet with a modified version of optimism-bedrock that uses Celestia as a DA layer.

    The handling of data is accomplished in two ways. First, data is written to the data availability (DA) layer i.e. in this case Celestia, then the data commitment is written to the op-batcher. When reading op-node simply reads the data back from the DA layer by reading the data commitment from the op-batcher first, then reading the data from the DA layer using the data commitment. Hence, while previously op-node was reading from calldata on Ethereum, but now it reads data from Celestia.

    There are a few tools involved in the data handling process. op-batcher batches up rollup blocks and posts them to Ethereum. op-geth handles execution, while op-proposer is responsible for state commitment submission.

    By using Celestia as a DA layer, existing L2s can switch from posting their data as calldata on Ethereum, to posting to Celestia. The commitment to the block is posted on Celestia, which is purpose-built for data availability. This is a more scalable than the traditional method of posting this data as calldata on monolithic chains.

    If you'd like to go modular, bedrock has made it easy to swap this out!

    Next steps

    Now that you understand the integration, you can start learning about the Bubs testnet, built with OP Stack and Celestia! This testnet is a great way to explore the possibilities of this integration and test your applications in a live environment.

    + + + + \ No newline at end of file diff --git a/developers/node-api.html b/developers/node-api.html new file mode 100644 index 00000000000..c92d1de2619 --- /dev/null +++ b/developers/node-api.html @@ -0,0 +1,45 @@ + + + + + + Node API | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Node API

    The celestia-node API is made for interacting with celestia-node. There are two ways in which a user and developer can interact with the API, the RPC API and the Gateway API. View the API's documentation.

    RPC API

    The RPC API primarily focuses on developers and projects building on top of Celestia, who are willing to run their own DA nodes. The RPC API provides a richer set of features and a superior user experience. Unlike the Gateway API, the RPC API allows access to the internal wallet and keyring of the DA node, as well as other sensitive and administrative capabilities.

    Library

    The node can be used as a Golang library and designed for programmatic API access.

    RPC

    The RPC API is also exposed to OpenRPC(JSON-RPC 2.0) for users wanting to run their DA node as a separate DA service. It provides the same set of features as the library with an additional authentication system with different permissions levels to protect the wallet and signing + providing RPC-level DOS protection.

    RPC API tutorial

    The node tutorial, which uses the RPC CLI, is the recommended way to get started interacting with your Celestia node.

    Gateway API

    WARNING

    The gateway endpoints have been deprecated and will be removed in the future. If you would like to use them anyway, you can find more details on GitHub.

    The gateway API is a REST API which is meant to be deployed by infra providers to enable the public read-only gateway to the DA network for external users who don't want or can't run light nodes (like browsers currently) over HTTP. It has no wallet or signing functionality.

    Gateway API tutorial

    Check out the Prompt scavenger gateway API tutorial for more details.

    + + + + \ No newline at end of file diff --git a/developers/node-tutorial.html b/developers/node-tutorial.html new file mode 100644 index 00000000000..6a3e1545a9d --- /dev/null +++ b/developers/node-tutorial.html @@ -0,0 +1,523 @@ + + + + + + Celestia-node RPC CLI tutorial | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Celestia-node RPC CLI tutorial

    In this tutorial, we will cover how to use the celestia-node RPC API to submit and retrieve data (blobs) from the data availability layer by their namespace.

    Introduction

    Blobs

    Data is posted to Celestia's DA layer by using MsgPayForBlobs transactions to the core network. Read more about MsgPayForBlobs.

    Namespaces

    Celestia partitions the block data into multiple namespaces, one for every application. This allows applications to only download their data, and not the data of other applications. Read more about Namespaced Merkle trees (NMTs).

    TIP

    If you already have a running and funded node, you can skip to the RPC CLI guide section.

    WARNING

    The gateway endpoints have been deprecated and will be removed in the future. If you would like to use them anyway, you can find more details on GitHub.

    Hardware requirements

    The following minimum hardware requirements are recommended for running a light node:

    • Memory: 500 MB RAM (minimum)
    • CPU: Single Core
    • Disk: 50 GB SSD Storage
    • Bandwidth: 56 Kbps for Download/56 Kbps for Upload

    Setting up dependencies

    This portion of the tutorial will go over setting up your development environment to run Celestia software. This environment can be used for development, building binaries, and running nodes.

    In your terminal, set up dependencies needed to install and build celestia-node.

    1. If you are on Ubuntu, first update and upgrade your OS:

      bash
      sudo apt update && sudo apt upgrade -y
      sudo apt update && sudo apt upgrade -y
      bash
      sudo yum update
      sudo yum update
    2. Install essential packages that are necessary to execute many tasks like downloading files, compiling, and monitoring the node:

      bash
      sudo apt install curl tar wget clang pkg-config libssl-dev jq \
      +build-essential git make ncdu -y
      sudo apt install curl tar wget clang pkg-config libssl-dev jq \
      +build-essential git make ncdu -y
      bash
      sudo yum install curl tar wget clang pkg-config libssl-dev jq \
      +build-essential git make ncdu -y
      sudo yum install curl tar wget clang pkg-config libssl-dev jq \
      +build-essential git make ncdu -y
      bash
      # these commands are for installing Homebrew, wget and jq
      +# follow the instructions from the output after running this command
      +/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
      +
      +# then install wget & jq
      +brew install wget && brew install jq
      # these commands are for installing Homebrew, wget and jq
      +# follow the instructions from the output after running this command
      +/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
      +
      +# then install wget & jq
      +brew install wget && brew install jq

    Install Golang

    celestia-node is written in Golang so we must install Golang to build and run our node.

    1. Set the version for your desired network:

      bash
      ver="1.21.1"
      ver="1.21.1"
      bash
      ver="1.22.0"
      ver="1.22.0"
      bash
      ver="1.22.0"
      ver="1.22.0"
    2. Download and install Golang:

      bash
      cd $HOME
      +wget "https://golang.org/dl/go$ver.linux-amd64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.linux-amd64.tar.gz"
      +rm "go$ver.linux-amd64.tar.gz"
      cd $HOME
      +wget "https://golang.org/dl/go$ver.linux-amd64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.linux-amd64.tar.gz"
      +rm "go$ver.linux-amd64.tar.gz"
      bash
      cd $HOME
      +wget "https://golang.org/dl/go$ver.linux-arm64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.linux-arm64.tar.gz"
      +rm "go$ver.linux-arm64.tar.gz"
      cd $HOME
      +wget "https://golang.org/dl/go$ver.linux-arm64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.linux-arm64.tar.gz"
      +rm "go$ver.linux-arm64.tar.gz"
      bash
      cd $HOME
      +wget "https://golang.org/dl/go$ver.darwin-arm64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.darwin-arm64.tar.gz"
      +rm "go$ver.darwin-arm64.tar.gz"
      cd $HOME
      +wget "https://golang.org/dl/go$ver.darwin-arm64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.darwin-arm64.tar.gz"
      +rm "go$ver.darwin-arm64.tar.gz"
      bash
      cd $HOME
      +wget "https://golang.org/dl/go$ver.darwin-amd64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.darwin-amd64.tar.gz"
      +rm "go$ver.darwin-amd64.tar.gz"
      cd $HOME
      +wget "https://golang.org/dl/go$ver.darwin-amd64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.darwin-amd64.tar.gz"
      +rm "go$ver.darwin-amd64.tar.gz"
    3. Add your /usr/local/go/bin directory to your $PATH if you have not already:

      bash
      echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.bash_profile
      +source $HOME/.bash_profile
      echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.bash_profile
      +source $HOME/.bash_profile
      bash
      echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.zshrc
      +source $HOME/.zshrc
      echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.zshrc
      +source $HOME/.zshrc
    4. To verify that the correct version of Go was installed correctly run:

      bash
      go version
      go version

    The output will show the version installed.

    Celestia-node

    Install celestia-node

    Installing celestia-node for Arabica devnet or Mocha testnet means installing a specific version to be compatible with the network.

    Install the celestia-node binary by running the following commands:

    1. Remove any existing copy of celestia-node, clone the repository, and change into the directory.

      bash
      cd $HOME
      +rm -rf celestia-node
      +git clone https://github.com/celestiaorg/celestia-node.git
      +cd celestia-node/
      cd $HOME
      +rm -rf celestia-node
      +git clone https://github.com/celestiaorg/celestia-node.git
      +cd celestia-node/
    2. Check out to the desired version, based on the network you will use:

      bash
      git checkout tags/v0.13.2
      git checkout tags/v0.13.2
      bash
      git checkout tags/v0.13.3
      git checkout tags/v0.13.3
      bash
      git checkout tags/v0.13.3
      git checkout tags/v0.13.3
    3. Build the celestia binary:

      a. Standard build

      bash
      make build
      make build

      b. Experimental build

      OPTIONAL

      If you're a node operator comfortable with experimental features and seeking optimal performance with minimal RAM usage, this option is recommended for you.

      bash
      make build-jemalloc
      make build-jemalloc

      This build option enables CGO, and downloads and installs jemalloc. Learn more about the build command.

    4. Install the binary:

      bash
      make install
      make install
    5. Build the cel-key utility:

      bash
      make cel-key
      make cel-key
    6. Verify that the binary is working and check the version:

      bash
      celestia version
      celestia version

    The output will show the semantic version of celestia-node, commit hash, build date, system version, and Golang version.

    Instantiate a Celestia light node

    Now, let's instantiate a Celestia Light node:

    TIP

    RPC endpoints are exposed in all celestia-node types such as light, bridge and full nodes.

    bash
    celestia light init
    celestia light init
    bash
    celestia light init --p2p.network mocha
    celestia light init --p2p.network mocha
    bash
    celestia light init --p2p.network arabica
    celestia light init --p2p.network arabica

    Instantiating (or initializing) the node means setting up a node store on your machine. This is where the data and your keys will be stored.

    Connect to a public core endpoint

    Let's now run the Celestia Light node with a gRPC connection to an example public core endpoint.

    Note: You are also encouraged to find a community-run API endpoint and there are several in the Discord. This one is used for demonstration purposes. Check out the Mocha testnet page, or Arabica devnet page.

    bash
    celestia light start --core.ip <URI>
    celestia light start --core.ip <URI>
    bash
    celestia light start --core.ip <URI> --p2p.network mocha
    celestia light start --core.ip <URI> --p2p.network mocha
    bash
    celestia light start --core.ip <URI> --p2p.network arabica
    celestia light start --core.ip <URI> --p2p.network arabica

    TIP

    The --core.ip gRPC port defaults to 9090, so if you do not specify it in the command line, it will default to that port. You can add the port after the IP address or use the --core.grpc.port flag to specify another port if you prefer.

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    For example, your command along with an RPC endpoint might look like this:

    bash
    celestia light start --core.ip consensus.lunaroasis.net
    celestia light start --core.ip consensus.lunaroasis.net
    bash
    celestia light start --core.ip rpc-mocha.pops.one --p2p.network mocha
    celestia light start --core.ip rpc-mocha.pops.one --p2p.network mocha
    bash
    celestia light start --core.ip validator-1.celestia-arabica-11.com \
    +  --p2p.network arabica
    celestia light start --core.ip validator-1.celestia-arabica-11.com \
    +  --p2p.network arabica

    Keys and wallets

    You can create your key for your node by running the following command from the celestia-node directory:

    TIP

    You do not need to declare a network for Mainnet Beta. Refer to the chain ID section on the troubleshooting page for more information

    bash
    ./cel-key add <key-name> --keyring-backend test --node.type light \
    +  --p2p.network <network>
    ./cel-key add <key-name> --keyring-backend test --node.type light \
    +  --p2p.network <network>

    You can start your light node with the key created by running the following command:

    bash
    celestia light start --core.ip <URI> --keyring.accname <key-name>
    celestia light start --core.ip <URI> --keyring.accname <key-name>
    bash
    celestia light start --core.ip <URI> --keyring.accname <key-name> \
    +  --p2p.network mocha
    celestia light start --core.ip <URI> --keyring.accname <key-name> \
    +  --p2p.network mocha
    bash
    celestia light start --core.ip <URI> --keyring.accname <key-name> \
    +  --p2p.network arabica
    celestia light start --core.ip <URI> --keyring.accname <key-name> \
    +  --p2p.network arabica

    Once you start the light node, a wallet key will be generated for you. You will need to fund that address with Mocha testnet or Arabica devnet tokens to pay for PayForBlobs transactions.

    You can find the address by running the following command in the celestia-node directory:

    bash
    ./cel-key list --node.type light --keyring-backend test --p2p.network <network>
    ./cel-key list --node.type light --keyring-backend test --p2p.network <network>

    If you would like to fund your wallet with testnet tokens, head over to either the #mocha-faucet or #arabica-faucet channels on the Celestia Discord.

    You can request funds to your wallet address using the following command in Discord:

    text
    $request <CELESTIA-ADDRESS>
    $request <CELESTIA-ADDRESS>

    Where <CELESTIA-ADDRESS> is the celestia1****** address generated when you created the wallet.

    With your wallet funded, you can move on to the next step.

    RPC CLI guide

    This section of the tutorial will teach you how to interact with a Celestia node's remote procedure call (RPC) API using the command line interface (CLI).

    You will need to setup dependencies, install, and run celestia-node if you have not already.

    Command formatting

    The format for interacting with the RPC CLI methods is as follows:

    bash
    celestia <module> <method> [args...] [flags...]
    celestia <module> <method> [args...] [flags...]

    Where:

    • celestia is the main command to interact with the node.
    • <module> is the specific module in the node you want to interact with, such as blob, state, p2p, etc.
    • <method> is the specific method within the module that performs the action you want, such as blob.Submit, state.AccountAddress, p2p.Info, etc.
    • [args...] represents any additional arguments that the method might require.
    • [flags...] are parameters that modify the behavior of the command. They start with -- (e.g., --node.store, --token, or --url).

    For example, to submit a blob to Celestia, you can use this command once your node store is set:

    bash
    celestia blob submit 0x42690c204d39600fddd3 'gm' --node.store $NODE_STORE
    celestia blob submit 0x42690c204d39600fddd3 'gm' --node.store $NODE_STORE

    Alternatively, you could use the --token flag to set your auth token:

    bash
    celestia blob submit 0x42690c204d39600fddd3 'gm' --token $AUTH_TOKEN
    celestia blob submit 0x42690c204d39600fddd3 'gm' --token $AUTH_TOKEN

    Before you try that out, let's go over the basic flags that you will need to use when interacting with the RPC CLI. We'll also cover how to set your auth token and how to use the node store to set it.

    Basic flags

    All RPC CLI commands have basic flags that can be used to interact with the API.

    These include:

    • --node.store string - the path to root/home directory of your celestia-node store
    • --token string - authorization token for making requests
    • --url string - the address of the RPC, default is http://localhost:26658

    When running RPC CLI commands, you will need to set either the authentication token or set the node store, so the auth token can be retrieved from the store.

    The RPC CLI handles these flags in the following order:

    1. If user passes auth token, auth token is used.
    2. If user doesn't pass auth token, check node store flag, create token from node store, and use auth token from node store.

    Auth token 🔐

    In order to interact with the API using RPC CLI, you will need to set the authentication token.

    The --token string flag sets the authentication token. If a token is not found, authentication will not be set. And if authentication is not set, the request will fail.

    To set your authentication token, you can use the following command. Be sure to replace <node-type> with the type of node and <network> with the network that you are running your node on:

    bash
    export AUTH_TOKEN=$(celestia <node-type> auth admin --p2p.network <network>)
    export AUTH_TOKEN=$(celestia <node-type> auth admin --p2p.network <network>)

    Here's an example of how to set your auth token on a light node on Arabica:

    bash
    export AUTH_TOKEN=$(celestia light auth admin --p2p.network arabica)
    export AUTH_TOKEN=$(celestia light auth admin --p2p.network arabica)

    Node store

    In order to interact with the API using RPC CLI, you can also use your node store to set your auth token. This will allow you to interact with the API without setting an authentication token directly.

    To set your node store for a light node on mocha-4, you can use the following command:

    bash
    export NODE_STORE=$HOME/.celestia-light-mocha-4
    export NODE_STORE=$HOME/.celestia-light-mocha-4

    Then, set the --node.store flag to the $NODE_STORE variable to set the auth token from your node store:

    bash
    celestia <module> <method> [args...] --node.store $NODE_STORE
    celestia <module> <method> [args...] --node.store $NODE_STORE
    Auth token on custom or private network

    This section is for users who are using a CELESTIA_CUSTOM or private network.

    TIP

    If you are using a private and custom network, you will need to set the location of the node store in your auth command.

    bash
    --node.store $HOME/.celestia-light-private)
    --node.store $HOME/.celestia-light-private)

    The above is an example from the following custom network set up with:

    bash
    CELESTIA_CUSTOM=private celestia light init
    CELESTIA_CUSTOM=private celestia light init

    or

    bash
    celestia light init --p2p.network private
    celestia light init --p2p.network private

    As an example, this is what a completely custom network would look like:

    bash
    # Initialize node store
    +CELESTIA_CUSTOM=robusta-22 celestia light init
    +
    +# Set auth token
    +export AUTH_TOKEN=$(celestia light auth admin --p2p.network private \
    +  --node.store $HOME/.celestia-light-robusta-22)
    # Initialize node store
    +CELESTIA_CUSTOM=robusta-22 celestia light init
    +
    +# Set auth token
    +export AUTH_TOKEN=$(celestia light auth admin --p2p.network private \
    +  --node.store $HOME/.celestia-light-robusta-22)

    Submitting data

    In this example, we will be submitting a blob to the network with a blob.Submit transaction with our light node.

    Some things to consider:

    • The endpoint takes in namespace and data values.
      • The commitment will be generated by the node.
      • Share version is set by the node.
    • Namespace should be 10 bytes, prefixed by 0x if hex; otherwise use base64
    • Data can be hex-encoded (0x...), base64-encoded ("..."), or a plaintext string which will be encoded to base64 ('Hello There!')
    • Optionally, user can provide a gas fee and gas limit.

    We use the following namespace of 0x42690c204d39600fddd3 and the data value of 0x676d.

    Here is an example of the format of the blob.Submit transaction:

    bash
    celestia blob submit <hex-encoded namespace> <hex-encoded data> \
    +  [optional: fee] [optional: gasLimit] [node store | auth token]
    celestia blob submit <hex-encoded namespace> <hex-encoded data> \
    +  [optional: fee] [optional: gasLimit] [node store | auth token]

    We run the following to submit a blob to the network in hexadecimal format:

    bash
    celestia blob submit 0x42690c204d39600fddd3 0x676d \
    +  --node.store $NODE_STORE
    celestia blob submit 0x42690c204d39600fddd3 0x676d \
    +  --node.store $NODE_STORE

    We get the following output:

    json
    {
    +  "result": {
    +    "height": 252607,
    +    "commitment": "0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY="
    +  }
    +}
    {
    +  "result": {
    +    "height": 252607,
    +    "commitment": "0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY="
    +  }
    +}

    We can also use a string of text as the data value, which will be converted to base64. Here is an example of the format:

    bash
    celestia blob submit <hex-encoded namespace> <'data'> \
    +  [optional: fee] [optional: gasLimit] [node store | auth token]
    celestia blob submit <hex-encoded namespace> <'data'> \
    +  [optional: fee] [optional: gasLimit] [node store | auth token]

    And an example to submit "gm" as the plain-text data:

    bash
    celestia blob submit 0x42690c204d39600fddd3 'gm' --node.store $HOME/.celestia-light-mocha-4/
    celestia blob submit 0x42690c204d39600fddd3 'gm' --node.store $HOME/.celestia-light-mocha-4/

    Output:

    json
    {
    +  "result": {
    +    "height": 252614,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}
    {
    +  "result": {
    +    "height": 252614,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}

    If you notice from the above output, it returns a result of 252614 which we will use for the next command. The result corresponds to the height of the block in which the transaction was included.

    Optional: Submit with curl

    Refer to the submitting a blob using curl section.

    Retrieving data

    After submitting your PFB transaction, upon success, the node will return the block height for which the PFB transaction was included. You can then use that block height and the namespace ID with which you submitted your PFB transaction to get your message shares (data) returned to you. In this example, the block height we got was 252614 which we will use for the following command. Read more about shares in the Celestia Specs.

    Here is what an example of the format of the get command looks like:

    bash
    celestia blob get <block height> <hex-encoded namespace> \
    +  <commitment from output above> <node store | auth>
    celestia blob get <block height> <hex-encoded namespace> \
    +  <commitment from output above> <node store | auth>

    Here is an example command to retrieve the data from above, on arabica-11:

    bash
    celestia blob get 252614 0x42690c204d39600fddd3 IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E= \
    +  --node.store $NODE_STORE
    celestia blob get 252614 0x42690c204d39600fddd3 IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E= \
    +  --node.store $NODE_STORE

    Will generate the following output:

    json
    {
    +  "result": {
    +    "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +    "data": "gm",
    +    "share_version": 0,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}
    {
    +  "result": {
    +    "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +    "data": "gm",
    +    "share_version": 0,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}

    The output here is base64 decoded to plain-text.

    To see the base64 response, use the --base64 flag set to TRUE (--base64=TRUE):

    bash
    celestia blob get 252614 0x42690c204d39600fddd3 IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E= \
    +  --base64=TRUE --node.store $NODE_STORE
    celestia blob get 252614 0x42690c204d39600fddd3 IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E= \
    +  --base64=TRUE --node.store $NODE_STORE

    The response will look similar to this:

    json
    {
    +  "result": {
    +    "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +    "data": "Z20=",
    +    "share_version": 0,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}
    {
    +  "result": {
    +    "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +    "data": "Z20=",
    +    "share_version": 0,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}

    To get all blobs in the namespace at the block height, use get-all instead of get:

    bash
    celestia blob get-all 252614 0x42690c204d39600fddd3 \
    +  --node.store $NODE_STORE
    celestia blob get-all 252614 0x42690c204d39600fddd3 \
    +  --node.store $NODE_STORE

    This will return the following:

    json
    {
    +  "result": [
    +    {
    +      "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +      "data": "gm",
    +      "share_version": 0,
    +      "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +    }
    +  ]
    +}
    {
    +  "result": [
    +    {
    +      "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +      "data": "gm",
    +      "share_version": 0,
    +      "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +    }
    +  ]
    +}

    To display the response in base64, use:

    bash
    celestia blob get-all 252614 0x42690c204d39600fddd3 \
    +  --base64=TRUE --node.store $NODE_STORE
    celestia blob get-all 252614 0x42690c204d39600fddd3 \
    +  --base64=TRUE --node.store $NODE_STORE

    Which will return:

    json
    {
    +  "result": [
    +    {
    +      "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +      "data": "gm",
    +      "share_version": 0,
    +      "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +    }
    +  ]
    +}
    {
    +  "result": [
    +    {
    +      "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +      "data": "gm",
    +      "share_version": 0,
    +      "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +    }
    +  ]
    +}

    Setting the gas price

    The --gas.price flag allows you to specify the gas price for the submission. If not specified, a default gas price will be used. The gas limit is automatically calculated based on the size of the blob being submitted.

    To set the gas price, you can use the --gas.price flag. The gas price will be set to default (0.002) if no value is passed.

    Learn more about gas fees and limits.

    To set a higher gas price of 0.004 utia, use the --gas.price 0.004 flag:

    bash
    celestia blob submit 0x42690c204d39600fddd3 'gm' --gas.price 0.004 \
    +  --node.store $NODE_STORE
    celestia blob submit 0x42690c204d39600fddd3 'gm' --gas.price 0.004 \
    +  --node.store $NODE_STORE

    You will receive the height and commitment of the block in which the transaction was included for these three examples:

    json
    {
    +  "result": {
    +    "height": 62562,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}
    {
    +  "result": {
    +    "height": 62562,
    +    "commitment": "IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E="
    +  }
    +}

    Examples

    Check your balance

    Let's query our node for the balance of its default account (which is the account associated with the CELESTIA_NODE_AUTH_TOKEN key we generated above):

    bash
    celestia state balance --node.store $NODE_STORE
    celestia state balance --node.store $NODE_STORE

    The response will look similar to:

    json
    {
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "denom": "utia",
    +    "amount": "172118057"
    +  },
    +  "id": 1
    +}
    {
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "denom": "utia",
    +    "amount": "172118057"
    +  },
    +  "id": 1
    +}

    Check the balance of another address

    Here is an example of the format of the balance-for-address command:

    bash
    celestia state balance-for-address <address> \
    +  --node.store $NODE_STORE
    celestia state balance-for-address <address> \
    +  --node.store $NODE_STORE

    Let's query our node for the balance of another address:

    bash
    celestia state balance-for-address celestia10rtd9lhel2cuh6c659l25yncl6atcyt37umard \
    +  --node.store $NODE_STORE
    celestia state balance-for-address celestia10rtd9lhel2cuh6c659l25yncl6atcyt37umard \
    +  --node.store $NODE_STORE

    The response will be the balance of the address you queried:

    json
    {
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "denom": "utia",
    +    "amount": "1000000"
    +  },
    +  "id": 1
    +}
    {
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "denom": "utia",
    +    "amount": "1000000"
    +  },
    +  "id": 1
    +}

    Get your node ID

    This is an RPC call in order to get your node's peerId information:

    bash
    celestia p2p info --node.store $NODE_STORE
    celestia p2p info --node.store $NODE_STORE

    The node ID is in the ID value from the response:

    json
    {
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "ID": "12D3KooWFFhCaAqY56oEqY3pLZUdLsv4RYAfVWKATZRepUPdosLp",
    +    "Addrs": [
    +      "/ip4/10.0.0.171/tcp/2121",
    +      "/ip4/10.0.0.171/udp/2121/quic-v1",
    +      "/ip4/71.200.65.106/tcp/25630",
    +      "/ip4/71.200.65.106/udp/25630/quic-v1",
    +      "/ip6/::1/tcp/2121",
    +      "/ip6/::1/udp/2121/quic-v1"
    +    ]
    +  },
    +  "id": 1
    +}
    {
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "ID": "12D3KooWFFhCaAqY56oEqY3pLZUdLsv4RYAfVWKATZRepUPdosLp",
    +    "Addrs": [
    +      "/ip4/10.0.0.171/tcp/2121",
    +      "/ip4/10.0.0.171/udp/2121/quic-v1",
    +      "/ip4/71.200.65.106/tcp/25630",
    +      "/ip4/71.200.65.106/udp/25630/quic-v1",
    +      "/ip6/::1/tcp/2121",
    +      "/ip6/::1/udp/2121/quic-v1"
    +    ]
    +  },
    +  "id": 1
    +}

    Get your account address

    This is an RPC call in order to get your node's account address:

    bash
    celestia state account-address --node.store $NODE_STORE
    celestia state account-address --node.store $NODE_STORE

    Response:

    json
    {
    +  "jsonrpc": "2.0",
    +  "result": "celestia1znk24rh52pgcd9z5x2x42jztjh6raaaphuvrt3",
    +  "id": 1
    +}
    {
    +  "jsonrpc": "2.0",
    +  "result": "celestia1znk24rh52pgcd9z5x2x42jztjh6raaaphuvrt3",
    +  "id": 1
    +}

    Get block header by height

    Here is an example of the format of the GetByHeight command:

    bash
    celestia header get-by-height <height> \
    +  --node.store $NODE_STORE
    celestia header get-by-height <height> \
    +  --node.store $NODE_STORE

    Now, let's get the block header information.

    Here we will get the header from Block 1:

    bash
    celestia header get-by-height 1 \
    +  --node.store $NODE_STORE```
    +
    +It will output something like this:
    +
    +<!-- markdownlint-disable MD013 -->
    +
    +```json
    +{
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "header": {
    +      "version": {
    +        "block": "11",
    +        "app": "1"
    +      },
    +      "chain_id": "arabica-11",
    +      "height": "1",
    +      "time": "2023-06-27T13:02:39.741743Z",
    +      "last_block_id": {
    +        "hash": "",
    +        "parts": {
    +          "total": 0,
    +          "hash": ""
    +        }
    +      },
    +      "last_commit_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "data_hash": "3D96B7D238E7E0456F6AF8E7CDF0A67BD6CF9C2089ECB559C659DCAA1F880353",
    +      "validators_hash": "6363C68770C200FD794445668F9B18F5B1DD1125180D6E8D5AB004F7DD7A0F48",
    +      "next_validators_hash": "6363C68770C200FD794445668F9B18F5B1DD1125180D6E8D5AB004F7DD7A0F48",
    +      "consensus_hash": "048091BC7DDC283F77BFBF91D73C44DA58C3DF8A9CBC867405D8B7F3DAADA22F",
    +      "app_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "last_results_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "evidence_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "proposer_address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86"
    +    },
    +    "validator_set": {
    +      "validators": [
    +        {
    +          "address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86",
    +          "pub_key": {
    +            "type": "tendermint/PubKeyEd25519",
    +            "value": "9aNBAxno1B4X5LR2qY5qWqwrMNOzejkctXwzq9BExsg="
    +          },
    +          "voting_power": "500000000",
    +          "proposer_priority": "0"
    +        }
    +      ],
    +      "proposer": {
    +        "address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86",
    +        "pub_key": {
    +          "type": "tendermint/PubKeyEd25519",
    +          "value": "9aNBAxno1B4X5LR2qY5qWqwrMNOzejkctXwzq9BExsg="
    +        },
    +        "voting_power": "500000000",
    +        "proposer_priority": "0"
    +      }
    +    },
    +    "commit": {
    +      "height": 1,
    +      "round": 0,
    +      "block_id": {
    +        "hash": "7A5FABB19713D732D967B1DA84FA0DF5E87A7B62302D783F78743E216C1A3550",
    +        "parts": {
    +          "total": 1,
    +          "hash": "D85C907CE660878A8203AC74BAA147CCC1F87114B45B568B72AD207B62AFE45E"
    +        }
    +      },
    +      "signatures": [
    +        {
    +          "block_id_flag": 2,
    +          "validator_address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86",
    +          "timestamp": "2023-06-30T08:40:19.299137127Z",
    +          "signature": "qmaEzrnbtgEXCRYc8pCvGRbS+uMuknIBoRAE4qyE7oSgWCRwBVYS/oPReXQLg9ER1oEY1De4MkWvMjlFnQOOCg=="
    +        }
    +      ]
    +    },
    +    "dah": {
    +      "row_roots": [
    +        "//////////////////////////////////////7//////////////////////////////////////huZWOTTDmD36N1F75A9BshxNlRasCnNpQiWqIhdVHcU",
    +        "/////////////////////////////////////////////////////////////////////////////5iieeroHBMfF+sER3JpvROIeEJZjbY+TRE0ntADQLL3"
    +      ],
    +      "column_roots": [
    +        "//////////////////////////////////////7//////////////////////////////////////huZWOTTDmD36N1F75A9BshxNlRasCnNpQiWqIhdVHcU",
    +        "/////////////////////////////////////////////////////////////////////////////5iieeroHBMfF+sER3JpvROIeEJZjbY+TRE0ntADQLL3"
    +      ]
    +    }
    +  },
    +  "id": 1
    +}
    celestia header get-by-height 1 \
    +  --node.store $NODE_STORE```
    +
    +It will output something like this:
    +
    +<!-- markdownlint-disable MD013 -->
    +
    +```json
    +{
    +  "jsonrpc": "2.0",
    +  "result": {
    +    "header": {
    +      "version": {
    +        "block": "11",
    +        "app": "1"
    +      },
    +      "chain_id": "arabica-11",
    +      "height": "1",
    +      "time": "2023-06-27T13:02:39.741743Z",
    +      "last_block_id": {
    +        "hash": "",
    +        "parts": {
    +          "total": 0,
    +          "hash": ""
    +        }
    +      },
    +      "last_commit_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "data_hash": "3D96B7D238E7E0456F6AF8E7CDF0A67BD6CF9C2089ECB559C659DCAA1F880353",
    +      "validators_hash": "6363C68770C200FD794445668F9B18F5B1DD1125180D6E8D5AB004F7DD7A0F48",
    +      "next_validators_hash": "6363C68770C200FD794445668F9B18F5B1DD1125180D6E8D5AB004F7DD7A0F48",
    +      "consensus_hash": "048091BC7DDC283F77BFBF91D73C44DA58C3DF8A9CBC867405D8B7F3DAADA22F",
    +      "app_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "last_results_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "evidence_hash": "E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855",
    +      "proposer_address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86"
    +    },
    +    "validator_set": {
    +      "validators": [
    +        {
    +          "address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86",
    +          "pub_key": {
    +            "type": "tendermint/PubKeyEd25519",
    +            "value": "9aNBAxno1B4X5LR2qY5qWqwrMNOzejkctXwzq9BExsg="
    +          },
    +          "voting_power": "500000000",
    +          "proposer_priority": "0"
    +        }
    +      ],
    +      "proposer": {
    +        "address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86",
    +        "pub_key": {
    +          "type": "tendermint/PubKeyEd25519",
    +          "value": "9aNBAxno1B4X5LR2qY5qWqwrMNOzejkctXwzq9BExsg="
    +        },
    +        "voting_power": "500000000",
    +        "proposer_priority": "0"
    +      }
    +    },
    +    "commit": {
    +      "height": 1,
    +      "round": 0,
    +      "block_id": {
    +        "hash": "7A5FABB19713D732D967B1DA84FA0DF5E87A7B62302D783F78743E216C1A3550",
    +        "parts": {
    +          "total": 1,
    +          "hash": "D85C907CE660878A8203AC74BAA147CCC1F87114B45B568B72AD207B62AFE45E"
    +        }
    +      },
    +      "signatures": [
    +        {
    +          "block_id_flag": 2,
    +          "validator_address": "91E04695CF9CF531BC0891E7B1D602B3E8022C86",
    +          "timestamp": "2023-06-30T08:40:19.299137127Z",
    +          "signature": "qmaEzrnbtgEXCRYc8pCvGRbS+uMuknIBoRAE4qyE7oSgWCRwBVYS/oPReXQLg9ER1oEY1De4MkWvMjlFnQOOCg=="
    +        }
    +      ]
    +    },
    +    "dah": {
    +      "row_roots": [
    +        "//////////////////////////////////////7//////////////////////////////////////huZWOTTDmD36N1F75A9BshxNlRasCnNpQiWqIhdVHcU",
    +        "/////////////////////////////////////////////////////////////////////////////5iieeroHBMfF+sER3JpvROIeEJZjbY+TRE0ntADQLL3"
    +      ],
    +      "column_roots": [
    +        "//////////////////////////////////////7//////////////////////////////////////huZWOTTDmD36N1F75A9BshxNlRasCnNpQiWqIhdVHcU",
    +        "/////////////////////////////////////////////////////////////////////////////5iieeroHBMfF+sER3JpvROIeEJZjbY+TRE0ntADQLL3"
    +      ]
    +    }
    +  },
    +  "id": 1
    +}

    Combined commands

    bash
    celestia share get-by-namespace "$(celestia header get-by-height 147105 \
    +  --node.store  $NODE_STORE | jq '.result.dah' -r)" 0x42690c204d39600fddd3 \
    +  --node.store $NODE_STORE
    celestia share get-by-namespace "$(celestia header get-by-height 147105 \
    +  --node.store  $NODE_STORE | jq '.result.dah' -r)" 0x42690c204d39600fddd3 \
    +  --node.store $NODE_STORE

    Get data availability sampler stats

    bash
    celestia das sampling-stats --node.store $NODE_STORE
    celestia das sampling-stats --node.store $NODE_STORE

    Transfer balance of utia to another account

    First, set your address as a variable:

    bash
    export ADDRESS=celestia1c425ckmve2489atttx022qpc02gxspa29wmh0d
    export ADDRESS=celestia1c425ckmve2489atttx022qpc02gxspa29wmh0d

    Then, transfer the amount of tokens that you would like, while setting the recipient's address, gas fee, and gasLimit. This is what the format will look like:

    bash
    celestia state transfer $ADDRESS <amount in utia> <gas fee in utia> <gas fee in utia> \
    +  --node.store $NODE_STORE
    celestia state transfer $ADDRESS <amount in utia> <gas fee in utia> <gas fee in utia> \
    +  --node.store $NODE_STORE

    Here is an example, sending 0.1 TIA, with a gas fee of 0.008 TIA, and a gas limit of 0.08:

    bash
    celestia state transfer $ADDRESS 100000 8000 80000 \
    +  --node.store $NODE_STORE
    celestia state transfer $ADDRESS 100000 8000 80000 \
    +  --node.store $NODE_STORE

    If you'd just like to return the transaction hash, you can use jq:

    bash
    celestia state transfer $ADDRESS 100000 8000 80000 --node.store $NODE_STORE | jq .result.txhash
    celestia state transfer $ADDRESS 100000 8000 80000 --node.store $NODE_STORE | jq .result.txhash

    API version

    To query your node's API version, you can use the following command:

    bash
    celestia node info --node.store $NODE_STORE
    celestia node info --node.store $NODE_STORE

    Help

    To get help and view the CLI menu, use the following command:

    bash
    celestia --help
    celestia --help

    To view help menu for a specific method, use the following command:

    bash
    celestia <module> <method> --node.store $NODE_STORE --help
    celestia <module> <method> --node.store $NODE_STORE --help

    Advanced example

    This example shows us using the jq command to parse the output of the celestia header get-by-height method to get the extended header used in celestia share get-by-namespace:

    bash
    celestia share get-by-namespace \
    +  "$(celestia header get-by-height 252614 --node.store $NODE_STORE | jq '.result.dah' -r)" \
    +  0x42690c204d39600fddd3 --node.store $NODE_STORE
    celestia share get-by-namespace \
    +  "$(celestia header get-by-height 252614 --node.store $NODE_STORE | jq '.result.dah' -r)" \
    +  0x42690c204d39600fddd3 --node.store $NODE_STORE

    Additional resources

    Submitting a blob using curl

    In order to post a blob using curl, you will need a light node running with the --core.ip string flag, providing access to a consensus endpoint. The flag indicates node to connect to the given core consensus node. Examples: 127.0.0.1 or subdomain.domain.tld. Using either IP or DNS assumes RPC port 26657 and gRPC port 9090 as default unless otherwise specified.

    1. In your terminal, set the auth token for the desired network. In this example, we will use Mainnet Beta.
    bash
    export CELESTIA_NODE_AUTH_TOKEN=$(celestia light auth admin --p2p.network celestia)
    export CELESTIA_NODE_AUTH_TOKEN=$(celestia light auth admin --p2p.network celestia)
    1. Post your blob with:
    bash
    curl -H "Content-Type: application/json" -H "Authorization: Bearer $CELESTIA_NODE_AUTH_TOKEN" -X POST --data '{"id": 1,
    +  "jsonrpc": "2.0",
    +  "method": "blob.Submit",
    +  "params": [
    +    [
    +      {
    +        "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAAECAwQFBgcICRA=",
    +        "data": "VGhpcyBpcyBhbiBleGFtcGxlIG9mIHNvbWUgYmxvYiBkYXRh",
    +        "share_version": 0,
    +        "commitment": "AD5EzbG0/EMvpw0p8NIjMVnoCP4Bv6K+V6gjmwdXUKU="
    +      }
    +    ],
    +    0.002
    +  ]
    +}' 127.0.0.1:26658
    curl -H "Content-Type: application/json" -H "Authorization: Bearer $CELESTIA_NODE_AUTH_TOKEN" -X POST --data '{"id": 1,
    +  "jsonrpc": "2.0",
    +  "method": "blob.Submit",
    +  "params": [
    +    [
    +      {
    +        "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAAECAwQFBgcICRA=",
    +        "data": "VGhpcyBpcyBhbiBleGFtcGxlIG9mIHNvbWUgYmxvYiBkYXRh",
    +        "share_version": 0,
    +        "commitment": "AD5EzbG0/EMvpw0p8NIjMVnoCP4Bv6K+V6gjmwdXUKU="
    +      }
    +    ],
    +    0.002
    +  ]
    +}' 127.0.0.1:26658
    1. Upon successful blob submission, the result will show the block height:
    bash
    {"jsonrpc":"2.0","result":362101,"id":1}
    {"jsonrpc":"2.0","result":362101,"id":1}

    The example transaction can be found on Celenium.

    Post an SVG as a PFB

    If you'd like to create your own SVG, post it to Celestia, and retrieve it, you can check out the Base64 SVG Tutorial.

    Golang guide

    If you're interested in interacting with the node's API in Go (client.go), you can use the da-rpc-client-tutorial repo.

    Troubleshooting

    If you encounter an error like:

    sh
    "rpc error: code = NotFound desc = account celestia1krkle0n547u0znz3unnln8paft2dq4z3rznv86 not found"
    "rpc error: code = NotFound desc = account celestia1krkle0n547u0znz3unnln8paft2dq4z3rznv86 not found"

    It is possible that the account you are trying to submit a PayForBlobs from doesn't have testnet tokens yet. Ensure the testnet faucet has funded your account with tokens and then try again.

    + + + + \ No newline at end of file diff --git a/developers/optimism-devnet.html b/developers/optimism-devnet.html new file mode 100644 index 00000000000..b7d75878a71 --- /dev/null +++ b/developers/optimism-devnet.html @@ -0,0 +1,175 @@ + + + + + + Deploy an OP Stack devnet | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Deploy an OP Stack devnet

    This guide will show you how to run your own OP Stack devnet.

    Dependency setup

    Environment setup and Golang installation

    First, install dependencies for Celestia software and for OP Stack.

    Clone repository

    Next, clone the repo:

    bash
    cd $HOME
    +git clone https://github.com/celestiaorg/optimism
    +cd optimism
    cd $HOME
    +git clone https://github.com/celestiaorg/optimism
    +cd optimism

    Check out to the version for either the stable version or upstream version:

    bash
    git checkout tags/v1.2.0-OP_v1.7.0-CN_v0.12.4
    +git submodule update --init --recursive
    git checkout tags/v1.2.0-OP_v1.7.0-CN_v0.12.4
    +git submodule update --init --recursive
    bash
    git checkout celestia-develop
    +git submodule update --init --recursive
    git checkout celestia-develop
    +git submodule update --init --recursive

    Build devnet

    Build TypeScript definitions for TS dependencies:

    bash
    cd $HOME
    +cd optimism
    +make
    cd $HOME
    +cd optimism
    +make

    Set environment variables to start network:

    bash
    export SEQUENCER_BATCH_INBOX_ADDRESS=0xff00000000000000000000000000000000000000
    +export L2OO_ADDRESS=0x70997970C51812dc3A010C7d01b50e0d17dc79C8
    export SEQUENCER_BATCH_INBOX_ADDRESS=0xff00000000000000000000000000000000000000
    +export L2OO_ADDRESS=0x70997970C51812dc3A010C7d01b50e0d17dc79C8

    Start devnet

    Start the network by running:

    bash
    make devnet-up
    make devnet-up

    This starts up the layer 1 (ETH), layer 2 (op-geth), data availability layer (Celestia), the sequencer (op-node), batch submitter (op-batcher), state commitment service (op-proposer).

    View the logs of the devnet

    If you'd like to view the logs of the devnet, run the following command from the root of the Optimism directory:

    bash
    make devnet-logs
    make devnet-logs
    Optional: Docker tips

    Stop devnet

    If you'd like to start the network over, use the following command to safely shut down all of the containers:

    bash
    make devnet-down
    make devnet-down

    Then clean out the old config:

    bash
    make devnet-clean
    make devnet-clean

    Viewing containers

    To view the containers running, send:

    bash
    docker ps
    docker ps

    Find the container ID of the ops-bedrock_op-batcher_1 and run the following to follow the logs:

    bash
    docker logs -f <container-id>
    docker logs -f <container-id>

    In a new terminal, find the container ID of the ghcr.io/celestiaorg/local-celestia-devnet:main and run the following to follow the logs:

    bash
    docker logs -f <container-id>
    docker logs -f <container-id>

    You can do the same for other containers as you explore the stack.

    Find a transaction

    Now, we'll check for a recent transaction on the L1 with:

    bash
    cast block latest --rpc-url localhost:8545
    cast block latest --rpc-url localhost:8545

    Output of a block that contains a transaction will look like this:

    console
    baseFeePerGas        7
    +difficulty           2
    +extraData            0xd883010d04846765746888676f312e32312e33856c696e7578000000000000006b3afa42dce1f87f1f07a1ef569c4d43e41738ef93c865098bfa1458645f384e2e4498bcfe4ad9353ff1913a2e16162f496fafe5b0939a6c78fb5b503248d6da01
    +gasLimit             30000000
    +gasUsed              21568
    +hash                 0x1cb54d2369752ef73511c202ff9cdfd0eadf3a77b7aef0092bea63f2b5d57659
    +logsBloom            0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
    +miner                0x0000000000000000000000000000000000000000
    +mixHash              0x0000000000000000000000000000000000000000000000000000000000000000
    +nonce                0x0000000000000000
    +number               1141
    +parentHash           0x664bf4bb4a57dd5768a0a98991d77c58fb7a4e164c2581c79fb33ce9c3d4c250
    +receiptsRoot         0xaf8ff6af1180c8be9e4e8f3a5f882b3b227233f4abbefa479836d3721682a389
    +sealFields           []
    +sha3Uncles           0x1dcc4de8dec75d7aab85b567b6ccd41ad312451b948a7413f0a142fd40d49347
    +size                 767
    +stateRoot            0xd4b998a35d20d98ed3488221f0c161a0a9572d3de66399482553c8e3d2fae751
    +timestamp            1699638350
    +withdrawalsRoot
    +totalDifficulty      2283
    +transactions:        [
    +  0x79a0a7a1b4936aafe7a37dbfb07a6a9e55c145a4ed6fd54f962649b4b7db8de7
    +]
    baseFeePerGas        7
    +difficulty           2
    +extraData            0xd883010d04846765746888676f312e32312e33856c696e7578000000000000006b3afa42dce1f87f1f07a1ef569c4d43e41738ef93c865098bfa1458645f384e2e4498bcfe4ad9353ff1913a2e16162f496fafe5b0939a6c78fb5b503248d6da01
    +gasLimit             30000000
    +gasUsed              21568
    +hash                 0x1cb54d2369752ef73511c202ff9cdfd0eadf3a77b7aef0092bea63f2b5d57659
    +logsBloom            0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
    +miner                0x0000000000000000000000000000000000000000
    +mixHash              0x0000000000000000000000000000000000000000000000000000000000000000
    +nonce                0x0000000000000000
    +number               1141
    +parentHash           0x664bf4bb4a57dd5768a0a98991d77c58fb7a4e164c2581c79fb33ce9c3d4c250
    +receiptsRoot         0xaf8ff6af1180c8be9e4e8f3a5f882b3b227233f4abbefa479836d3721682a389
    +sealFields           []
    +sha3Uncles           0x1dcc4de8dec75d7aab85b567b6ccd41ad312451b948a7413f0a142fd40d49347
    +size                 767
    +stateRoot            0xd4b998a35d20d98ed3488221f0c161a0a9572d3de66399482553c8e3d2fae751
    +timestamp            1699638350
    +withdrawalsRoot
    +totalDifficulty      2283
    +transactions:        [
    +  0x79a0a7a1b4936aafe7a37dbfb07a6a9e55c145a4ed6fd54f962649b4b7db8de7
    +]

    Copy the transaction hash from transactions: <transaction-hash> and set it as a variable:

    bash
    export TX_HASH=0x79a0a7a1b4936aafe7a37dbfb07a6a9e55c145a4ed6fd54f962649b4b7db8de7
    export TX_HASH=0x79a0a7a1b4936aafe7a37dbfb07a6a9e55c145a4ed6fd54f962649b4b7db8de7

    Read the transaction call data

    Now read the transaction call data on the L1:

    bash
    cast tx $TX_HASH --rpc-url localhost:8545
    cast tx $TX_HASH --rpc-url localhost:8545

    The output will look similar to below:

    console
    blockHash            0x9f4dfae061b5ddd86f95a81be5daa0d7fe32e7f7f770f86dc375e0007d249bd2
    +blockNumber          24
    +from                 0x3C44CdDdB6a900fa2b585dd299e03d12FA4293BC
    +gas                  21572
    +gasPrice             1040676758
    +hash                 0xadd3a5dc0b8c605aeac891098e87cbaff43bb642896ebbf74f964c0690e46df2
    +input                0xce3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f
    +nonce                4
    +r                    0xaf5c1505c7dfcebca94d9a6a8c0caf99b6c87a8ed6d6c0b3161c9026f270a84f
    +s                    0x383ed2debf9f9055920cd7340418dda7e2bca6b989eb6992d83d123d4e322f2a
    +to                   0xFf00000000000000000000000000000000000901
    +transactionIndex     0
    +v                    0
    +value                0
    +yParity              0
    blockHash            0x9f4dfae061b5ddd86f95a81be5daa0d7fe32e7f7f770f86dc375e0007d249bd2
    +blockNumber          24
    +from                 0x3C44CdDdB6a900fa2b585dd299e03d12FA4293BC
    +gas                  21572
    +gasPrice             1040676758
    +hash                 0xadd3a5dc0b8c605aeac891098e87cbaff43bb642896ebbf74f964c0690e46df2
    +input                0xce3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f
    +nonce                4
    +r                    0xaf5c1505c7dfcebca94d9a6a8c0caf99b6c87a8ed6d6c0b3161c9026f270a84f
    +s                    0x383ed2debf9f9055920cd7340418dda7e2bca6b989eb6992d83d123d4e322f2a
    +to                   0xFf00000000000000000000000000000000000901
    +transactionIndex     0
    +v                    0
    +value                0
    +yParity              0

    TIP

    You are looking for a batcher transaction to the address 0xFf00000000000000000000000000000000000901.

    First, remove the prefix 0xce. Now, set the input as the INPUT variable and encode it as base64:

    bash
    export INPUT=ce3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f
    +export ENCODED_INPUT=$(echo "$INPUT" | xxd -r -p | base64)
    export INPUT=ce3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f
    +export ENCODED_INPUT=$(echo "$INPUT" | xxd -r -p | base64)

    TIP

    Remember to remove the 0xce prefix!

    Find the data on Celestia

    Clone the go-da repository:

    bash
    cd $HOME
    +git clone https://github.com/rollkit/go-da.git
    +cd go-da/proto/da
    cd $HOME
    +git clone https://github.com/rollkit/go-da.git
    +cd go-da/proto/da

    Now, from go-da/proto/da run:

    bash
    grpcurl -proto da.proto -plaintext -d "{\"ids\": [{\"value\": \"$ENCODED_INPUT\"}]}" 127.0.0.1:26650 da.DAService.Get
    grpcurl -proto da.proto -plaintext -d "{\"ids\": [{\"value\": \"$ENCODED_INPUT\"}]}" 127.0.0.1:26650 da.DAService.Get

    Your result will look similar to the below!

    console
    {
    +  "blobs": [
    +    {
    +      "value": "AKUumhJ8FnuyVrBs38FDKEIAAAAAAZB42trhw/DDc4GFAlv4klkv5Zh4E16mmO5fpNOS1f5wzpds8YK3S0Rvs4ULLJj13euw+Ovdv6Q23zuV1ShROEvk5aptIT7bGmZunvc1OiKwJTXVbN0BiGm6k2zNWq78cNsT2ez3+nzQq84Ds28or/aKz/o1w4NpV7w4caZtgJomX71w96m63+xzYnarXLu7WWvRrwbeb6cW8R93YHXt1r4+TXCBGVe76obzf5JLTNu22gksD2cL+83D8DGjX0FKcwZD0VofkGmboKY1uTddu8704s2MwgNNe09s1bzw+n9Fq6fKFw7pvwJL200eCS0oFJ3HfPAEywnlgyyGQc89dh+98GD5TrdU4aNql9afmW+sDzJtC9S0fzLWYROOS0bvK3W7EvNpmWXe5qrdzKlBmv1LZi4ofrrxLHGmbYOaJhHsEn+B81lGUh33HDet8K9nVKKSF2+W3Xul6uPSxydPBwsv2GHskR+yfUlDbvyl1ROTvtS1zXlpEPz0M1e/RIIt57fVj0Gm7TgACAAA//+Qdel2AQ=="
    +    }
    +  ]
    +}
    {
    +  "blobs": [
    +    {
    +      "value": "AKUumhJ8FnuyVrBs38FDKEIAAAAAAZB42trhw/DDc4GFAlv4klkv5Zh4E16mmO5fpNOS1f5wzpds8YK3S0Rvs4ULLJj13euw+Ovdv6Q23zuV1ShROEvk5aptIT7bGmZunvc1OiKwJTXVbN0BiGm6k2zNWq78cNsT2ez3+nzQq84Ds28or/aKz/o1w4NpV7w4caZtgJomX71w96m63+xzYnarXLu7WWvRrwbeb6cW8R93YHXt1r4+TXCBGVe76obzf5JLTNu22gksD2cL+83D8DGjX0FKcwZD0VofkGmboKY1uTddu8704s2MwgNNe09s1bzw+n9Fq6fKFw7pvwJL200eCS0oFJ3HfPAEywnlgyyGQc89dh+98GD5TrdU4aNql9afmW+sDzJtC9S0fzLWYROOS0bvK3W7EvNpmWXe5qrdzKlBmv1LZi4ofrrxLHGmbYOaJhHsEn+B81lGUh33HDet8K9nVKKSF2+W3Xul6uPSxydPBwsv2GHskR+yfUlDbvyl1ROTvtS1zXlpEPz0M1e/RIIt57fVj0Gm7TgACAAA//+Qdel2AQ=="
    +    }
    +  ]
    +}

    Ethereum fallback mechanism in OP Stack

    The Ethereum fallback mechanism allows rollups to "fall back" to Ethereum or another EVM chain in the case of downtime or errors submitting data to Celestia.

    Implementation of fallback

    The Ethereum fallback mechanism is implemented in the celestiaorg/optimism v1.1.0 release.

    The op-batcher/batcher/driver.go and op-node/rollup/derive/calldata_source.go files are part of the Ethereum fallback mechanism in the op-batcher and op-node respectively.

    In driver.go, the calldataTxCandidate function is responsible for the write path of the Ethereum fallback. This function creates and submits a transaction to the batch inbox address with the given data. It uses the underlying txmgr to handle transaction sending and gas price management.

    If the transaction data can be published as a blob to Celestia, it replaces the calldata with a blob identifier and sends the transaction with this data. If it cannot be published to Celestia, it falls back to Ethereum without any change to the transaction.

    The blob identifier starts with the special prefix 0xce, which was chosen as a mnemonic for Celestia, and indicates that the remaining data has to interpreted as a little-endian encoded Block Height (8 bytes) and Blob Commitment (32 bytes). The combination of these can later be used to retrieve the original calldata from Celestia.

    Prefix8 bytes32 bytes
    0xceBlock HeightBlob Commitment
    go
    func (l *BatchSubmitter) sendTransaction(
    +    txdata txData,
    +    queue *txmgr.Queue[txData],
    +    receiptsCh chan txmgr.TxReceipt[txData],
    +) {
    +    // ...
    +}
    func (l *BatchSubmitter) sendTransaction(
    +    txdata txData,
    +    queue *txmgr.Queue[txData],
    +    receiptsCh chan txmgr.TxReceipt[txData],
    +) {
    +    // ...
    +}

    In calldata_source.go, the DataFromEVMTransactions function defines the read path of the Ethereum fallback. This function filters all of the transactions and returns the calldata from transactions that are sent to the batch inbox address from the batch sender address.

    If the calldata matches the version prefix 0xce, it is decoded as a blob identifier, the original calldata is retrieved from Celestia and returned for derivation. If the calldata does not match the prefix, the entire calldata is returned for derivation.

    go
    func DataFromEVMTransactions(
    +    config *rollup.Config,
    +    batcherAddr common.Address,
    +    txs types.Transactions,
    +    log log.Logger
    +) ([]eth.Data, error) {
    +    // ...
    +}
    func DataFromEVMTransactions(
    +    config *rollup.Config,
    +    batcherAddr common.Address,
    +    txs types.Transactions,
    +    log log.Logger
    +) ([]eth.Data, error) {
    +    // ...
    +}

    These two functions work together to ensure that the Ethereum fallback mechanism operates correctly, allowing the rollup to continue functioning even during periods of downtime on Celestia.

    Testing the fallback

    Testing out the Ethereum fallback mechanism can be done with the go-da tool. Triggering a simultaneous blob transaction will cause the op-batcher blob transaction to fail, with an incorrect account sequence error, which triggers a fallback to Ethereum.

    To trigger the transaction, send this command from the same go/proto/da directory:

    bash
    grpcurl -proto da.proto -plaintext -d '{"blobs": [{"value": "SGVsbG8gd28ybGQh"}]}' 127.0.0.1:26650 da.DAService.Submit
    grpcurl -proto da.proto -plaintext -d '{"blobs": [{"value": "SGVsbG8gd28ybGQh"}]}' 127.0.0.1:26650 da.DAService.Submit

    Alternatively, you can shut off the local-celestia-devnet and see that the OP Stack devnet logs show that the rollup has fallen back to the L1, in this case Ethereum, for posting data.

    Span batches

    Span batches can be enabled by setting OP_BATCHER_BATCH_TYPE: 1 in your docker-compose.yml file.

    Note that this requires the Delta activation time to be configured. For your devnet, you should set "l2GenesisDeltaTimeOffset": "0x0", in devnetL1-template.json. This will enable span batches and can be tested by grepping docker compose logs -f | grep batch_type which should include batch_type=SpanBatch and batch_type=1.

    + + + + \ No newline at end of file diff --git a/developers/optimism.html b/developers/optimism.html new file mode 100644 index 00000000000..5c6b7c49cda --- /dev/null +++ b/developers/optimism.html @@ -0,0 +1,100 @@ + + + + + + Deploy an OP Stack devnet to Celestia | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Deploy an OP Stack devnet to Celestia

    In order to deploy a devnet to Celestia, you will need to have a modified version of optimism-bedrock. Refer to the steps to install dependencies and the modified version of OP Stack for your environment setup.

    Pick your deployment type

    Using Celestia and OP stack, you have the option to either run a light node of your own or a local-celestia-devnet, both of which will give you a local devnet to test things out with.

    Using a local devnet

    If you'd like to use the local-celestia-devnet, you're in luck! This is the default for the OP Stack + Celestia repository. Head to the previous page to get started.

    Using a light node

    This is a beta integration and we are working on resolving open issues.

    In order to allow your light node to post and retrieve data without errors, you will need to change UseShareExchange to false in:

    bash
    $HOME/.celestia-light/config.toml
    $HOME/.celestia-light/config.toml
    bash
    $HOME/.celestia-light-mocha-4/config.toml
    $HOME/.celestia-light-mocha-4/config.toml
    bash
    $HOME/.celestia-light-arabica-11/config.toml
    $HOME/.celestia-light-arabica-11/config.toml

    If you choose to use your own node store, the light node must be fully synced and funded for you to be able to submit and retrieve PayForBlobs to a Celestia network.

    If it is not synced, you will run into errors similar to this.

    Visit the Arabica or Mocha pages to visit their faucets.

    In order to mount existing data, you must have a node store that is in this directory:

    bash
    $HOME/.celestia-light
    $HOME/.celestia-light
    bash
    $HOME/.celestia-light-mocha-4
    $HOME/.celestia-light-mocha-4
    bash
    $HOME/.celestia-light-arabica-11
    $HOME/.celestia-light-arabica-11

    This is the default location of the node store when you initialize and run a new Celestia node.

    By default, the node will run with the account named my_celes_key.

    If you have your own setup you'd like to try, you can always edit optimism/ops-bedrock/docker-compose.yml to work with your setup.

    Using a RaaS provider

    If you'd like to use a Rollups as a Service (RaaS) provider, you can do so by going to the RaaS category in the menu.

    Build the devnet

    Build TypeScript definitions for TS dependencies:

    bash
    cd $HOME
    +cd optimism
    +make
    cd $HOME
    +cd optimism
    +make

    Set environment variables to start network:

    bash
    export SEQUENCER_BATCH_INBOX_ADDRESS=0xff00000000000000000000000000000000000000
    +export L2OO_ADDRESS=0x70997970C51812dc3A010C7d01b50e0d17dc79C8
    export SEQUENCER_BATCH_INBOX_ADDRESS=0xff00000000000000000000000000000000000000
    +export L2OO_ADDRESS=0x70997970C51812dc3A010C7d01b50e0d17dc79C8

    Start the devnet

    First, make sure your light node is synced and funded. It must not be running for this example to work.

    This example is for Mainnet Beta. You can modify the da: section of your $HOME/optimism/ops-bedrock/docker-compose.yml for your specific use, similarly to the example below:

    This setup will use celestia-da, which is celestia-node with a DA server on port 26650.

    For the P2P_NETWORK variable, you'll need to supply the network of choice, either celestia, mocha, or arabica. Using celestia, the volume path will be just .celestia-light instead of .celestia-light-<network>. You will also need to provide a core.ip RPC URL for the network you are using.

    yaml
    da:
    +  image: ghcr.io/rollkit/local-celestia-devnet:v0.12.1 
    +  image: ghcr.io/rollkit/celestia-da:v0.12.9 
    +  command: > 
    +    celestia-da light start 
    +    --p2p.network=<network> 
    +    --da.grpc.namespace=000008e5f679bf7116cb 
    +    --da.grpc.listen=0.0.0.0:26650 
    +    --core.ip <rpc-url> 
    +    --gateway 
    +  environment: 
    +      - NODE_TYPE=light 
    +      - P2P_NETWORK=<network> 
    +  ports:
    +    - "26650:26650"
    +    - "26658:26658"
    +    - "26659:26659"
    +  volumes: 
    +    - $HOME/.celestia-light-<network>/:/home/celestia/.celestia-light-<network>/ 
    +  healthcheck:
    +    test: ["CMD", "curl", "-f", "http://localhost:26659/header/1"]
    +    interval: 10s
    +    timeout: 5s
    +    retries: 5
    +    start_period: 30s
    da:
    +  image: ghcr.io/rollkit/local-celestia-devnet:v0.12.1 
    +  image: ghcr.io/rollkit/celestia-da:v0.12.9 
    +  command: > 
    +    celestia-da light start 
    +    --p2p.network=<network> 
    +    --da.grpc.namespace=000008e5f679bf7116cb 
    +    --da.grpc.listen=0.0.0.0:26650 
    +    --core.ip <rpc-url> 
    +    --gateway 
    +  environment: 
    +      - NODE_TYPE=light 
    +      - P2P_NETWORK=<network> 
    +  ports:
    +    - "26650:26650"
    +    - "26658:26658"
    +    - "26659:26659"
    +  volumes: 
    +    - $HOME/.celestia-light-<network>/:/home/celestia/.celestia-light-<network>/ 
    +  healthcheck:
    +    test: ["CMD", "curl", "-f", "http://localhost:26659/header/1"]
    +    interval: 10s
    +    timeout: 5s
    +    retries: 5
    +    start_period: 30s

    Now start the devnet:

    bash
    make devnet-up
    make devnet-up

    View the logs of the devnet

    If you'd like to view the logs of the devnet, run the following command from the root of the Optimism directory:

    bash
    make devnet-logs
    make devnet-logs

    Stop the devnet

    To safely stop the devnet, run the following command:

    bash
    make devnet-down
    make devnet-down

    Clean the devnet

    To remove all data from the devnet, run the following command:

    bash
    make devnet-clean
    make devnet-clean

    Deploying to an L1 (or L2)

    If you'd like to deploy to an EVM L1 or L2, reference the OP stack deployment guide.

    + + + + \ No newline at end of file diff --git a/developers/prompt-scavenger.html b/developers/prompt-scavenger.html new file mode 100644 index 00000000000..d1c8eeb37c4 --- /dev/null +++ b/developers/prompt-scavenger.html @@ -0,0 +1,319 @@ + + + + + + Prompt scavenger | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Prompt scavenger

    Note

    This tutorial needs to be updated.

    Welcome to the world of Prompt Scavenger, a game where you’ll be using Celestia’s Node API and OpenAI’s GPT-3.5 to decode hidden messages scattered throughout Celestia’s blockchain. In this tutorial, we’ll be using Golang to write the code for the game.

    Through this tutorial, you’ll gain experience using Celestia’s Node API to fetch data from the blockchain, process it, and submit new transactions with that data. You’ll also learn how to integrate OpenAI’s GPT-3.5 API to generate fun responses based on the data you’ve found.

    So if you’re ready to embark on an adventure that combines blockchain technology with the power of AI, and learn some Golang along the way, let’s get started!

    Dependencies

    The following dependencies are needed to be installed or obtained:

    Install Celestia Node and run a light node

    First, install the celestia-node binary.

    Let’s run the following commands to get our env vars setup (NOTE: For CORE_IP you can select from the list of available RPC endpoints on the Blockspace Race page:

    sh
    KEYNAME="scavenger_key"
    +NODETYPE="light"
    +NETWORK="blockspacerace"
    +AUTHTYPE="admin"
    +CORE_IP=""
    KEYNAME="scavenger_key"
    +NODETYPE="light"
    +NETWORK="blockspacerace"
    +AUTHTYPE="admin"
    +CORE_IP=""

    Next, let’s generate a wallet key for our light node:

    sh
    ./cel-key add $KEYNAME --keyring-backend test --node.type $NODETYPE --p2p.network $NETWORK
    ./cel-key add $KEYNAME --keyring-backend test --node.type $NODETYPE --p2p.network $NETWORK

    Be sure to save the mnemonics and your celestia1 public address in a safe place.

    Then head over to our Discord Server and request tokens from the #faucet channel under Blockspace Race.

    You can track receiving the funds on the explorer here: Interchain Explorer by Cosmostation

    Just make sure to paste your celestia1**** address in order to look it up.

    We will be running this version of Celestia Node with Blockspace Race test network. First, let’s initialize our node:

    sh
    celestia light init --p2p.network blockspacerace
    celestia light init --p2p.network blockspacerace

    Next, we will start our node:

    sh
    celestia light start --core.ip $CORE_IP --p2p.network $NETWORK --gateway.deprecated-endpoints --gateway --gateway.addr 127.0.0.1 --gateway.port 26659 --keyring.accname $KEYNAME
    celestia light start --core.ip $CORE_IP --p2p.network $NETWORK --gateway.deprecated-endpoints --gateway --gateway.addr 127.0.0.1 --gateway.port 26659 --keyring.accname $KEYNAME

    TIP

    The --core.ip gRPC port defaults to 9090, so if you do not specify it in the command line, it will default to that port. You can add the port after the IP address or use the --core.grpc.port flag to specify another port if you prefer.

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    You should now have a running light node on your machine. The rest of the tutorial will assume you will be building the script and running it where the light node is in your localhost.

    Node API authentication key

    On the same machine where your Celestia light node is running, run the following:

    sh
    export AUTH_TOKEN=$(celestia $NODETYPE auth $AUTHTYPE)
    export AUTH_TOKEN=$(celestia $NODETYPE auth $AUTHTYPE)

    Now run the following to get the auth token for your node:

    sh
    echo $AUTH_TOKEN
    echo $AUTH_TOKEN

    This will be used for the env var file we setup later.

    OpenAI key

    Make sure to go over to OpenAI in order to sign up for an account and generate an OpenAI API key. The key will be needed to communicate with OpenAI.

    Building the Prompt Scavenger

    We will first need to setup a .env file with the following pasted in:

    txt
    NODE_RPC_IP="http://localhost:26658"
    +NODE_JWT_TOKEN=""
    +OPENAI_KEY=""
    +NAMESPACE_ID="00000000ce1e571a"
    NODE_RPC_IP="http://localhost:26658"
    +NODE_JWT_TOKEN=""
    +OPENAI_KEY=""
    +NAMESPACE_ID="00000000ce1e571a"

    The OPENAI_KEY is the API key you got from Open AI. For NODE_RPC_IP, it is assumed to be the local host but it can point to a light node that is remote. NODE_JWT_TOKEN is the AUTH_TOKEN you generated earlier. For Namespace ID, we made a sample one for you to use but you can generate your own.

    Now, let’s build!

    Copy over Go files

    Run the following:

    sh
    mkdir test_scavenger
    +git clone https://github.com/celestiaorg/PromptScavenger.git
    +cp PromptScavenger/go.mod test_scavenger/
    +cp PromptScavenger/go.sum test_scavenger/
    +cd test_scavenger
    mkdir test_scavenger
    +git clone https://github.com/celestiaorg/PromptScavenger.git
    +cp PromptScavenger/go.mod test_scavenger/
    +cp PromptScavenger/go.sum test_scavenger/
    +cd test_scavenger

    This will copy over the required go.sum and go.mod files to a new directory we will use to import the Node API golang library.

    Build your import statements

    Inside the directory, create a main.gofile and setup the import statements:

    go
    package main
    +
    +import (
    +        "context"
    +        "fmt"
    +        "log"
    +        "os"
    +
    +        "github.com/celestiaorg/celestia-node/api/rpc/client"
    +        nodeheader "github.com/celestiaorg/celestia-node/header"
    +        "github.com/celestiaorg/nmt/namespace"
    +        "github.com/joho/godotenv"
    +        cosmosmath "cosmossdk.io/math"
    +        openai "github.com/sashabaranov/go-openai"
    +        "encoding/base64"
    +        "encoding/hex"
    +)
    +
    +func main() {
    +  // TODO
    +}
    package main
    +
    +import (
    +        "context"
    +        "fmt"
    +        "log"
    +        "os"
    +
    +        "github.com/celestiaorg/celestia-node/api/rpc/client"
    +        nodeheader "github.com/celestiaorg/celestia-node/header"
    +        "github.com/celestiaorg/nmt/namespace"
    +        "github.com/joho/godotenv"
    +        cosmosmath "cosmossdk.io/math"
    +        openai "github.com/sashabaranov/go-openai"
    +        "encoding/base64"
    +        "encoding/hex"
    +)
    +
    +func main() {
    +  // TODO
    +}

    Here we setup all required libraries we need to use plus the main function that we will use for our game.

    Helpful functions

    First we will need to create some helpful functions that you will need later.

    go
    // loadEnv loads environment variables from the .env file.
    +func loadEnv() {
    +        err := godotenv.Load(".env")
    +        if err != nil {
    +                log.Fatal("Error loading .env file")
    +        }
    +}
    // loadEnv loads environment variables from the .env file.
    +func loadEnv() {
    +        err := godotenv.Load(".env")
    +        if err != nil {
    +                log.Fatal("Error loading .env file")
    +        }
    +}

    The loadEnv function allows us to load up our .env file which has all the necessary env vars needed.

    Next, let’s create a helper function that allows us to load an instance of the Celestia Node client given the correct env vars passed to it:

    go
    // createClient initializes a new Celestia node client.
    +func createClient(ctx context.Context) *client.Client {
    +        nodeRPCIP := os.Getenv("NODE_RPC_IP")
    +        jwtToken := os.Getenv("NODE_JWT_TOKEN")
    +
    +        rpc, err := client.NewClient(ctx, nodeRPCIP, jwtToken)
    +        if err != nil {
    +                log.Fatalf("Error creating client: %v", err)
    +        }
    +
    +        return rpc
    +}
    // createClient initializes a new Celestia node client.
    +func createClient(ctx context.Context) *client.Client {
    +        nodeRPCIP := os.Getenv("NODE_RPC_IP")
    +        jwtToken := os.Getenv("NODE_JWT_TOKEN")
    +
    +        rpc, err := client.NewClient(ctx, nodeRPCIP, jwtToken)
    +        if err != nil {
    +                log.Fatalf("Error creating client: %v", err)
    +        }
    +
    +        return rpc
    +}

    As you can see, here the Celestia Node client takes in the Node RPC IP and the JWT Token we setup before.

    Now, if we go back to our main function, we can do the following to setup and load our env and client:

    go
    
    +func main() {
    +  ctx, cancel := context.WithCancel(context.Background())
    +  defer cancel()
    +  loadEnv()
    +
    +  // Close the client when you are finished
    +  client.Close()
    +}
    
    +func main() {
    +  ctx, cancel := context.WithCancel(context.Background())
    +  defer cancel()
    +  loadEnv()
    +
    +  // Close the client when you are finished
    +  client.Close()
    +}

    Here, we setup a workflow that allows us to load our env vars, instantiate the client with it, then close the client.

    Now, let’s build some more helpful functions:

    go
    func createNamespaceID() []byte {
    +        nIDString := os.Getenv("NAMESPACE_ID")
    +        data, err := hex.DecodeString(nIDString)
    +        if err != nil {
    +                log.Fatalf("Error decoding hex string:", err)
    +        }
    +        // Encode the byte array in Base64
    +        base64Str := base64.StdEncoding.EncodeToString(data)
    +        namespaceID, err := base64.StdEncoding.DecodeString(base64Str)
    +        if err != nil {
    +                log.Fatalf("Error decoding Base64 string:", err)
    +        }
    +        return namespaceID
    +}
    func createNamespaceID() []byte {
    +        nIDString := os.Getenv("NAMESPACE_ID")
    +        data, err := hex.DecodeString(nIDString)
    +        if err != nil {
    +                log.Fatalf("Error decoding hex string:", err)
    +        }
    +        // Encode the byte array in Base64
    +        base64Str := base64.StdEncoding.EncodeToString(data)
    +        namespaceID, err := base64.StdEncoding.DecodeString(base64Str)
    +        if err != nil {
    +                log.Fatalf("Error decoding Base64 string:", err)
    +        }
    +        return namespaceID
    +}

    Here, we are creating a helpful function called createNameSpaceID that given a string for a namespace ID, it can decode the hex string, encode it after to a byte array, then decode it as a base64 string which is needed by Node API.

    We will need to create just a few more functions before we wrap up things.

    go
    // postDataAndGetHeight submits a new transaction with the
    +// provided data to the Celestia node.
    +func postDataAndGetHeight(client *client.Client, namespaceID namespace.ID, payLoad [
    +]byte, fee cosmosmath.Int, gasLimit uint64) uint64 {
    +        response, err := client.State.SubmitPayForBlob(context.Background(), namespa
    +ceID, payLoad, fee, gasLimit)
    +        if err != nil {
    +                log.Fatalf("Error submitting pay for blob: %v", err)
    +        }
    +        fmt.Printf("Got output: %v", response)
    +        height := uint64(response.Height)
    +        fmt.Printf("Height that data was submitted at: %v", height)
    +        return height
    +}
    // postDataAndGetHeight submits a new transaction with the
    +// provided data to the Celestia node.
    +func postDataAndGetHeight(client *client.Client, namespaceID namespace.ID, payLoad [
    +]byte, fee cosmosmath.Int, gasLimit uint64) uint64 {
    +        response, err := client.State.SubmitPayForBlob(context.Background(), namespa
    +ceID, payLoad, fee, gasLimit)
    +        if err != nil {
    +                log.Fatalf("Error submitting pay for blob: %v", err)
    +        }
    +        fmt.Printf("Got output: %v", response)
    +        height := uint64(response.Height)
    +        fmt.Printf("Height that data was submitted at: %v", height)
    +        return height
    +}

    In the function postDataAndGetHeight, we show you how to submit a message to a specific namespace ID provided on Celestia. After a successful submission, the function returns back to you the block height it was submitted at.

    Next, implement the following function:

    go
    func getDataAsPrompt(client *client.Client, height uint64, namespaceID namespace.ID)
    + string {
    +        headerParam := getHeader(client, height)
    +        response, err := client.Share.GetSharesByNamespace(context.Background(), hea
    +derParam.DAH, namespaceID)
    +        if err != nil {
    +                log.Fatalf("Error getting shares by namespace data for block height:
    + %v. Error is %v", height, err)
    +        }
    +        var dataString string
    +        for _, shares := range response {
    +                for _, share := range shares.Shares {
    +                        dataString = string(share[8:])
    +                }
    +        }
    +        return dataString
    +}
    func getDataAsPrompt(client *client.Client, height uint64, namespaceID namespace.ID)
    + string {
    +        headerParam := getHeader(client, height)
    +        response, err := client.Share.GetSharesByNamespace(context.Background(), hea
    +derParam.DAH, namespaceID)
    +        if err != nil {
    +                log.Fatalf("Error getting shares by namespace data for block height:
    + %v. Error is %v", height, err)
    +        }
    +        var dataString string
    +        for _, shares := range response {
    +                for _, share := range shares.Shares {
    +                        dataString = string(share[8:])
    +                }
    +        }
    +        return dataString
    +}

    In the function getDataAsPrompt, we show a helpful function that given a particular block height and a namespace ID, it can return back the block data (called shares here) which we then convert to a string and return it back.

    Finally, we implement a GPT-3.5 helper function that given a prompt, it returns back a statement:

    go
    // gpt3 processes a given message using GPT-3 and prints the response.
    +func gpt3(msg string) {
    +        // Set the authentication header
    +        openAIKey := os.Getenv("OPENAI_KEY")
    +        client := openai.NewClient(openAIKey)
    +        resp, err := client.CreateChatCompletion(
    +                context.Background(),
    +                openai.ChatCompletionRequest{
    +                        Model: openai.GPT3Dot5Turbo,
    +                        Messages: []openai.ChatCompletionMessage{
    +                                {
    +                                        Role:    openai.ChatMessageRoleUser,
    +                                        Content: msg,
    +                                },
    +                        },
    +                },
    +        )
    +
    +        if err != nil {
    +                fmt.Printf("ChatCompletion error: %v\n", err)
    +                return
    +        }
    +        fmt.Println(resp.Choices[0].Message.Content)
    +}
    // gpt3 processes a given message using GPT-3 and prints the response.
    +func gpt3(msg string) {
    +        // Set the authentication header
    +        openAIKey := os.Getenv("OPENAI_KEY")
    +        client := openai.NewClient(openAIKey)
    +        resp, err := client.CreateChatCompletion(
    +                context.Background(),
    +                openai.ChatCompletionRequest{
    +                        Model: openai.GPT3Dot5Turbo,
    +                        Messages: []openai.ChatCompletionMessage{
    +                                {
    +                                        Role:    openai.ChatMessageRoleUser,
    +                                        Content: msg,
    +                                },
    +                        },
    +                },
    +        )
    +
    +        if err != nil {
    +                fmt.Printf("ChatCompletion error: %v\n", err)
    +                return
    +        }
    +        fmt.Println(resp.Choices[0].Message.Content)
    +}

    Wrapping up the functions

    Now, we will update our mainfunction to include the logic from the functions we built where we show you how to do the following:

    • Instantiate namespace ID, fee, gas limit and GPT prompt
    • Submit the GPT prompt as a PayForBlob transaction and then get back the Block Height
    • Get Back the Prompt from that Block Height as a Data Share and convert it to a string and return it
    • Submit that string to the GPT function to get a prompt output
    go
    func main() {
    +        ctx, cancel := context.WithCancel(context.Background())
    +        defer cancel()
    +        loadEnv()
    +        var namespaceID namespace.ID = createNamespaceID()
    +        client := createClient(ctx)
    +        var gasLimit uint64 = 6000000
    +        fee := cosmosmath.NewInt(10000)
    +        var gptPrompt string = "What are modular blockchains?"
    +        prompt := []byte{0x00, 0x01, 0x02}
    +        prompt = append(prompt, []byte(gptPrompt)...)
    +        height := postDataAndGetHeight(client, namespaceID, prompt, fee, gasLimit)
    +        promptString := getDataAsPrompt(client, height, namespaceID)
    +        gpt3(promptString)
    +        // Close the client when you are finished
    +        client.Close()
    +}
    func main() {
    +        ctx, cancel := context.WithCancel(context.Background())
    +        defer cancel()
    +        loadEnv()
    +        var namespaceID namespace.ID = createNamespaceID()
    +        client := createClient(ctx)
    +        var gasLimit uint64 = 6000000
    +        fee := cosmosmath.NewInt(10000)
    +        var gptPrompt string = "What are modular blockchains?"
    +        prompt := []byte{0x00, 0x01, 0x02}
    +        prompt = append(prompt, []byte(gptPrompt)...)
    +        height := postDataAndGetHeight(client, namespaceID, prompt, fee, gasLimit)
    +        promptString := getDataAsPrompt(client, height, namespaceID)
    +        gpt3(promptString)
    +        // Close the client when you are finished
    +        client.Close()
    +}

    And now you have the final version of the game!

    Run the golang script with the following command:

    sh
    go run main.go
    go run main.go

    After some time, it’ll post the output of the prompt you submitted to OpenAI that you pulled from Celestia’s blockchain.

    Next steps

    With this tutorial, you were able to construct a PFB transaction, submit it to Celestia, get it back from Celestia and decode its contents, then for added bonus, submit the message to GPT-3.5.

    For the next steps, we will be releasing quests to this tutorial where users will need to complete challenges that help familiarize them with the Celestia Data Availability layer.

    + + + + \ No newline at end of file diff --git a/developers/requesting-data-commitment-ranges.html b/developers/requesting-data-commitment-ranges.html new file mode 100644 index 00000000000..df719eb0883 --- /dev/null +++ b/developers/requesting-data-commitment-ranges.html @@ -0,0 +1,45 @@ + + + + + + Requesting data commitment ranges | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Requesting data commitment ranges

    By default, the Blobstream X deployments on Ethereum will be updating every 4 hours, and on Arbitrum One and Base, updating every 1 hour. If you wish for the Blobstream X contract to be updated at a different cadence, then you have several different options for how to update the smart contract.

    To request proofs to be submitted to the Blobstream X contract at a different cadence, you can do one of the following:

    Run the Blobstream X operator with hosted proving on the Succinct platform, by running an operator script that pings the platform with proof requests at a specified cadence.

    Follow these instructions to run the operator script.

    Here are example values for the .env file:

    1. TENDERMINT_RPC_URL from the public Celestia list.
    2. SUCCINCT_RPC_URL = https://alpha.succinct.xyz/api
    3. Request for SUCCINCT_API_KEY from the Succinct team.
    4. CHAIN_ID is the chain ID of the deployed Blobstream X contract.
    5. CONTRACT_ADDRESS: Blobstream X proxy contract address.
    6. NEXT_HEADER_FUNCTION_ID & HEADER_RANGE_FUNCTION_ID: Get the functionId's from the Blobstream X contract by using the nextHeaderFunctionId and headerRangeFunctionId respectively, which are public storage variables.

    Local proving

    Run the Blobstream X operator with local proving.

    TIP

    Note: Requires a large cloud machine to run in a reasonable amount of time. EC2 r6a.16xlarge takes ~30 minutes to generate a header range proof.

    Request proof onchain

    Directly request a proof via the Blobstream X contract interface. Unlike the Blobstream X operator which handles requests off-chain, requesting on-chain requires gas, but the proof will be generated and relayed by the Succinct platform.

    1. Call requestHeaderRange(uint64 _targetBlock) with the end of the range you want a commitment for.

    2. A DataCommitmentStored(uint256, uint64, uint64, bytes32) will be emitted for the requested range when it is stored in the contract. Listen to this event to know that the proof has been generated successfully.

    + + + + \ No newline at end of file diff --git a/developers/rollkit.html b/developers/rollkit.html new file mode 100644 index 00000000000..2bf3bb9d9e4 --- /dev/null +++ b/developers/rollkit.html @@ -0,0 +1,45 @@ + + + + + + Rollkit | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Rollkit

    Validator nodes allow you to participate in consensus in the Celestia network.

    rollkit

    Rollkit is an ABCI (Application Blockchain Interface) implementation for sovereign rollups to deploy on top of Celestia.

    It is built by replacing Tendermint, the Cosmos-SDK consensus layer, with a drop-in replacement that communicates directly with Celestia's Data Availability layer.

    It spins up a sovereign rollup, which collects transactions into blocks and posts them onto Celestia for consensus and data availability.

    The goal of Rollkit is to enable anyone to design and deploy a sovereign rollup on Celestia in minutes.

    Furthermore, while Rollkit allows you to build sovereign rollups on Celestia, it currently does not support fraud proofs yet and is therefore running in "pessimistic" mode, where nodes would need to re-execute the transactions to check the validity of the chain (i.e. a full node). Furthermore, Rollkit currently only supports a single sequencer.

    Tutorials

    The following tutorials will help you get started building Cosmos-SDK applications that connect to Celestia's Data Availability Layer via Rollkit. We call those chains Sovereign Rollups.

    You can get started with the following tutorials:

    + + + + \ No newline at end of file diff --git a/developers/submit-data.html b/developers/submit-data.html new file mode 100644 index 00000000000..489891ef08a --- /dev/null +++ b/developers/submit-data.html @@ -0,0 +1,253 @@ + + + + + + Submitting data blobs to Celestia | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Submitting data blobs to Celestia

    To submit data to Celestia, users submit blob transactions (BlobTx). Blob transactions contain two components, a standard Cosmos-SDK transaction called MsgPayForBlobs and one or more Blobs of data.

    Fee market and mempool

    Celestia makes use of a standard gas-priced prioritized mempool. By default, transactions with gas prices higher than that of other transactions in the mempool will be prioritized by validators.

    Fees and gas limits

    As of version v1.0.0 of the application (celestia-app), there is no protocol enforced minimum fee (similar to EIP-1559 in Ethereum). Instead, each consensus node running a mempool uses a locally configured gas price threshold that must be met in order for that node to accept a transaction, either directly from a user or gossiped from another node, into its mempool.

    As of version v1.0.0 of the application (celestia-app), gas is not refunded. Instead, transaction fees are deducted by a flat fee, originally specified by the user in their tx (where fees = gasLimit * gasPrice). This means that users should use an accurate gas limit value if they do not wish to overpay.

    Under the hood, fees are currently handled by specifying and deducting a flat fee. However gas price is often specified by users instead of calculating the flat fee from the gas used and the gas price. Since the state machine does not refund users for unused gas, gas price is calculated by dividing the total fee by the gas limit.

    Estimating PFB gas

    Generally, the gas used by a PFB transaction involves a static fixed cost and a dynamic cost based on the size of each blob in the transaction.

    NOTE

    For a general use case of a normal account submitting a PFB, the static costs can be treated as such. However, due to the description above of how gas works in the Cosmos-SDK this is not always the case. Notably, if a vesting account or the feegrant modules are used, then these static costs change.

    The fixed cost is an approximation of the gas consumed by operations outside the function GasToConsume (for example, signature verification, tx size, read access to accounts), which has a default value of 65,000 gas.

    NOTE

    The first transaction sent by an account (sequence number == 0) has an additional one time gas cost of 10,000 gas. If this is the case, this should be accounted for.

    Each blob in the PFB contributes to the total gas cost based on its size. The function GasToConsume calculates the total gas consumed by all the blobs involved in a PFB, where each blob's gas cost is computed by first determining how many shares are needed to store the blob size. Then, it computes the product of the number of shares, the number of bytes per share, and the gasPerByte parameter. Finally, it adds a static amount per blob.

    The blob.GasPerBlobByte and auth.TxSizeCostPerByte are parameters that could potentially be adjusted through the system's governance mechanisms. Hence, actual costs may vary depending on the current state of these parameters.

    Gas fee calculation

    The total fee for a transaction is calculated as the product of the gas limit for the transaction and the gas price set by the user:

    Total Fee=Gas Limit×Gas Price

    The gas limit for a transaction is the maximum amount of gas that a user is willing to spend on a transaction. It is determined by both a static fixed cost (FC) and a variable dynamic cost based on the size of each blob involved in the transaction:

    Gas Limit=FC+i=1nSSN(Bi)×SS×GCPBB

    Where:

    • FC = Fixed Cost, is a static value (65,000 gas)
    • i=1nSSN(Bi) = SparseSharesNeeded for the ith Blob, is the number of shares needed for the ith blob in the transaction
    • SS = Share Size, is the size of each share
    • GCPBB = Gas Cost Per Blob Byte, is a parameter that could potentially be adjusted through the system's governance mechanisms.

    The gas fee is set by the user when they submit a transaction. The fee is often specified by users directly. The total cost for the transaction is then calculated as the product of the estimated gas limit and the gas price. Since the state machine does not refund users for unused gas, it's important for users to estimate the gas limit accurately to avoid overpaying.

    For more details on how gas is calculated per blob, refer to the PayForBlobs function that consumes gas based on the blob sizes. This function uses the GasToConsume function to calculate the extra gas charged to pay for a set of blobs in a MsgPayForBlobs transaction. This function calculates the total shares used by all blobs and multiplies it by the ShareSize and gasPerByte to get the total gas to consume.

    For estimating the total gas required for a set of blobs, refer to the EstimateGas function. This function estimates the gas based on a linear model that is dependent on the governance parameters: gasPerByte and txSizeCost. It assumes other variables are constant, including the assumption that the MsgPayForBlobs is the only message in the transaction. The DefaultEstimateGas function runs EstimateGas with the system defaults.

    Estimating gas programmatically

    Users can estimate an efficient gas limit by using this function:

    go
    import (
    +    blobtypes "github.com/celestiaorg/celestia-app/x/blob/types"
    +)
    +gasLimit := blobtypes.DefaultEstimateGas([]uint32{uint32(sizeOfDataInBytes)})
    import (
    +    blobtypes "github.com/celestiaorg/celestia-app/x/blob/types"
    +)
    +gasLimit := blobtypes.DefaultEstimateGas([]uint32{uint32(sizeOfDataInBytes)})

    If using a celestia-node light client, then this function is automatically called for you when submitting a blob. This function works by breaking down the components of calculating gas for a blob transaction. These components consist of a flat costs for all PFBs, the size of each blob and how many shares each uses and the parameter for gas used per byte. More information about how gas is used can be found in the gas specs and the exact formula can be found in the blob module.

    Submitting multiple transactions in one block from the same account

    The mempool Celestia uses works by maintaining a fork of the canonical state each block. This means that each time we submit a transaction to it, it will update the sequence number (aka nonce) for the account that submitted the transaction. If users wish to submit a second transaction, they can, but must specify the nonce manually. If this is not done, the new transactions will not be able to be submitted until the first transaction is reaped from the mempool (i.e. included in a block), or dropped due to timing out.

    By default, nodes will drop a transaction if it does not get included in 5 blocks (roughly 1 minute). At this point, the user must resubmit their transaction if they want it to eventually be included.

    As of v1.0.0 of the application (celestia-app), users are unable to replace an existing transaction with a different one with higher fees. They must instead wait 5 blocks from the original submitted time and then resubmit the transaction. Again, community members have already suggested solutions and a willingness to accept changes to fix this issue.

    API

    Users can currently create and submit BlobTxs in four ways.

    The celestia-app consensus node CLI

    bash
    celestia-appd tx blob PayForBlobs <hex-encoded namespace> <hex-encoded data> [flags]
    celestia-appd tx blob PayForBlobs <hex-encoded namespace> <hex-encoded data> [flags]

    The celestia-node light node CLI

    Using blob.Submit:

    bash
    celestia blob submit <hex-encoded namespace> <hex-encoded data>
    celestia blob submit <hex-encoded namespace> <hex-encoded data>

    Learn more in the node tutorial.

    GRPC to a consensus node via the user package

    go
    import (
    +    "context"
    +    "fmt"
    +
    +    "github.com/celestiaorg/celestia-app/app"
    +    "github.com/celestiaorg/celestia-app/app/encoding"
    +    "github.com/celestiaorg/celestia-app/pkg/appconsts"
    +    "github.com/celestiaorg/celestia-app/pkg/namespace"
    +    "github.com/celestiaorg/celestia-app/pkg/user"
    +    blobtypes "github.com/celestiaorg/celestia-app/x/blob/types"
    +    "github.com/cosmos/cosmos-sdk/crypto/keyring"
    +    tmproto "github.com/tendermint/tendermint/proto/tendermint/types"
    +    "google.golang.org/grpc"
    +    "google.golang.org/grpc/credentials/insecure"
    +)
    +
    +// SubmitData is a demo function that shows how to use the signer to submit data
    +// to the blockchain directly via a celestia node. We can manage this keyring
    +// using the `celestia-appd keys` or `celestia keys` sub commands and load this
    +// keyring from a file and use it to programmatically sign transactions.
    +func DemoSubmitData(grpcAddr string, kr keyring.Keyring) error {
    +    // create an encoding config that can decode and encode all celestia-app
    +    // data structures.
    +    ecfg := encoding.MakeConfig(app.ModuleEncodingRegisters...)
    +
    +    // create a connection to the grpc server on the consensus node.
    +    conn, err := grpc.Dial(grpcAddr, grpc.WithTransportCredentials(insecure.NewCredentials()))
    +    if err != nil {
    +        return err
    +    }
    +    defer conn.Close()
    +
    +    // get the address of the account we want to use to sign transactions.
    +    rec, err := kr.Key("accountName")
    +    if err != nil {
    +        return err
    +    }
    +
    +    addr, err := rec.GetAddress()
    +    if err != nil {
    +        return err
    +    }
    +
    +    // Setup the signer. This function will automatically query the relevant
    +    // account information such as sequence (nonce) and account number.
    +    signer, err := user.SetupSigner(context.TODO(), kr, conn, addr, ecfg)
    +    if err != nil {
    +        return err
    +    }
    +
    +    ns := namespace.MustNewV0([]byte("1234567890"))
    +
    +    fmt.Println("namespace", len(ns.Bytes()))
    +
    +    blob, err := blobtypes.NewBlob(ns, []byte("some data"), appconsts.ShareVersionZero)
    +    if err != nil {
    +        return err
    +    }
    +
    +    gasLimit := blobtypes.DefaultEstimateGas([]uint32{uint32(len(blob.Data))})
    +
    +    options := []user.TxOption{
    +        // here we're setting estimating the gas limit from the above estimated
    +        // function, and then setting the gas price to 0.1utia per unit of gas.
    +        user.SetGasLimitAndFee(gasLimit, 0.1),
    +    }
    +
    +    // this function will submit the transaction and block until a timeout is
    +    // reached or the transaction is committed.
    +    resp, err := signer.SubmitPayForBlob(context.TODO(), []*tmproto.Blob{blob}, options...)
    +    if err != nil {
    +        return err
    +    }
    +
    +    // check the response code to see if the transaction was successful.
    +    if resp.Code != 0 {
    +        // handle code
    +        fmt.Println(resp.Code, resp.Codespace, resp.RawLog)
    +    }
    +
    +    // if we don't want to wait for the transaction to be confirmed, we can
    +    // manually sign and submit the transaction using the same package.
    +    blobTx, err := signer.CreatePayForBlob([]*tmproto.Blob{blob}, options...)
    +    if err != nil {
    +        return err
    +    }
    +
    +    resp, err = signer.BroadcastTx(context.TODO(), blobTx)
    +    if err != nil {
    +        return err
    +    }
    +
    +    // check the response code to see if the transaction was successful. Note
    +    // that this time we're not waiting for the transaction to be committed.
    +    // Therefore the code here is only from the consensus node's mempool.
    +    if resp.Code != 0 {
    +        // handle code
    +        fmt.Println(resp.Code, resp.Codespace, resp.RawLog)
    +    }
    +
    +    return err
    +}
    import (
    +    "context"
    +    "fmt"
    +
    +    "github.com/celestiaorg/celestia-app/app"
    +    "github.com/celestiaorg/celestia-app/app/encoding"
    +    "github.com/celestiaorg/celestia-app/pkg/appconsts"
    +    "github.com/celestiaorg/celestia-app/pkg/namespace"
    +    "github.com/celestiaorg/celestia-app/pkg/user"
    +    blobtypes "github.com/celestiaorg/celestia-app/x/blob/types"
    +    "github.com/cosmos/cosmos-sdk/crypto/keyring"
    +    tmproto "github.com/tendermint/tendermint/proto/tendermint/types"
    +    "google.golang.org/grpc"
    +    "google.golang.org/grpc/credentials/insecure"
    +)
    +
    +// SubmitData is a demo function that shows how to use the signer to submit data
    +// to the blockchain directly via a celestia node. We can manage this keyring
    +// using the `celestia-appd keys` or `celestia keys` sub commands and load this
    +// keyring from a file and use it to programmatically sign transactions.
    +func DemoSubmitData(grpcAddr string, kr keyring.Keyring) error {
    +    // create an encoding config that can decode and encode all celestia-app
    +    // data structures.
    +    ecfg := encoding.MakeConfig(app.ModuleEncodingRegisters...)
    +
    +    // create a connection to the grpc server on the consensus node.
    +    conn, err := grpc.Dial(grpcAddr, grpc.WithTransportCredentials(insecure.NewCredentials()))
    +    if err != nil {
    +        return err
    +    }
    +    defer conn.Close()
    +
    +    // get the address of the account we want to use to sign transactions.
    +    rec, err := kr.Key("accountName")
    +    if err != nil {
    +        return err
    +    }
    +
    +    addr, err := rec.GetAddress()
    +    if err != nil {
    +        return err
    +    }
    +
    +    // Setup the signer. This function will automatically query the relevant
    +    // account information such as sequence (nonce) and account number.
    +    signer, err := user.SetupSigner(context.TODO(), kr, conn, addr, ecfg)
    +    if err != nil {
    +        return err
    +    }
    +
    +    ns := namespace.MustNewV0([]byte("1234567890"))
    +
    +    fmt.Println("namespace", len(ns.Bytes()))
    +
    +    blob, err := blobtypes.NewBlob(ns, []byte("some data"), appconsts.ShareVersionZero)
    +    if err != nil {
    +        return err
    +    }
    +
    +    gasLimit := blobtypes.DefaultEstimateGas([]uint32{uint32(len(blob.Data))})
    +
    +    options := []user.TxOption{
    +        // here we're setting estimating the gas limit from the above estimated
    +        // function, and then setting the gas price to 0.1utia per unit of gas.
    +        user.SetGasLimitAndFee(gasLimit, 0.1),
    +    }
    +
    +    // this function will submit the transaction and block until a timeout is
    +    // reached or the transaction is committed.
    +    resp, err := signer.SubmitPayForBlob(context.TODO(), []*tmproto.Blob{blob}, options...)
    +    if err != nil {
    +        return err
    +    }
    +
    +    // check the response code to see if the transaction was successful.
    +    if resp.Code != 0 {
    +        // handle code
    +        fmt.Println(resp.Code, resp.Codespace, resp.RawLog)
    +    }
    +
    +    // if we don't want to wait for the transaction to be confirmed, we can
    +    // manually sign and submit the transaction using the same package.
    +    blobTx, err := signer.CreatePayForBlob([]*tmproto.Blob{blob}, options...)
    +    if err != nil {
    +        return err
    +    }
    +
    +    resp, err = signer.BroadcastTx(context.TODO(), blobTx)
    +    if err != nil {
    +        return err
    +    }
    +
    +    // check the response code to see if the transaction was successful. Note
    +    // that this time we're not waiting for the transaction to be committed.
    +    // Therefore the code here is only from the consensus node's mempool.
    +    if resp.Code != 0 {
    +        // handle code
    +        fmt.Println(resp.Code, resp.Codespace, resp.RawLog)
    +    }
    +
    +    return err
    +}

    RPC to a celestia-node

    Using the JSON RPC API, submit data using the following methods:

    Learn more in the celestia-node API docs.

    + + + + \ No newline at end of file diff --git a/developers/transaction-resubmission.html b/developers/transaction-resubmission.html new file mode 100644 index 00000000000..b09f0e79009 --- /dev/null +++ b/developers/transaction-resubmission.html @@ -0,0 +1,45 @@ + + + + + + Transaction resubmission | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Transaction resubmission

    In cases where transactions are not included within a 75-second window, resubmission is necessary. This is especially important during network congestion, as transactions with relatively low fees may not be processed even after the network clears up.

    Regardless of whether they originate from celestia-app or celestia-node, transactions will not be re-gossiped, except in the presence of a new peer.

    Monitoring and resubmission

    Monitor the status of your transactions. If a transaction is not included within a 75-second window, it should be resubmitted. This can be done manually or through automated processes.

    Changes introduced in celestiaorg/celestia-core#1089 may affect transaction gossiping and inclusion speed.

    Notes

    • All transactions, regardless of their origin, are subject to being sorted and pruned based on fees.
    • It is the user or developer's responsibility to monitor and possibly resubmit transactions if they are not included in a 75-second window.
    + + + + \ No newline at end of file diff --git a/developers/wallets.html b/developers/wallets.html new file mode 100644 index 00000000000..0b8714e847a --- /dev/null +++ b/developers/wallets.html @@ -0,0 +1,248 @@ + + + + + + Wallet integrations with Celestia | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Wallet integrations with Celestia

    This page covers how developers can use Keplr and React to add Celestia network parameters to wallets, and how to add custom networks to Leap and Cosmostation.

    Add Celestia network parameters to Keplr with React

    Before we demonstrate how to export the specific parameters for Celestia's testnets, we need to create a ReactJS component that allows us to connect directly to Keplr and pass it the network parameters.

    In the following code, we show how you can export a component that detects whether Keplr is installed and sets the network params for it:

    jsx
    // @site/src/components/AddNetworkKeplr.js
    +import React from "react";
    +import styles from "./Keplr.module.css";
    +
    +export default function AddNetworkKeplr({ params }) {
    +  async function add() {
    +    if (!window.keplr) {
    +      alert("Please install keplr extension");
    +    } else {
    +      if (window.keplr.experimentalSuggestChain) {
    +        try {
    +          await window.keplr.experimentalSuggestChain({
    +            chainId: params.chainId,
    +            chainName: params.chainName,
    +            rpc: params.rpc,
    +            rest: params.rest,
    +            bip44: {
    +              coinType: 118,
    +            },
    +            bech32Config: {
    +              bech32PrefixAccAddr: "celestia",
    +              bech32PrefixAccPub: "celestia" + "pub",
    +              bech32PrefixValAddr: "celestia" + "valoper",
    +              bech32PrefixValPub: "celestia" + "valoperpub",
    +              bech32PrefixConsAddr: "celestia" + "valcons",
    +              bech32PrefixConsPub: "celestia" + "valconspub",
    +            },
    +            currencies: [
    +              {
    +                coinDenom: "TIA",
    +                coinMinimalDenom: "utia",
    +                coinDecimals: 6,
    +                coinGeckoId: "celestia",
    +              },
    +            ],
    +            feeCurrencies: [
    +              {
    +                coinDenom: "TIA",
    +                coinMinimalDenom: "utia",
    +                coinDecimals: 6,
    +                coinGeckoId: "celestia",
    +                gasPriceStep: {
    +                  low: 0.01,
    +                  average: 0.02,
    +                  high: 0.1,
    +                },
    +              },
    +            ],
    +            stakeCurrency: {
    +              coinDenom: "TIA",
    +              coinMinimalDenom: "utia",
    +              coinDecimals: 6,
    +              coinGeckoId: "celestia",
    +            },
    +          });
    +        } catch {
    +          alert("Failed to suggest the chain");
    +        }
    +      }
    +      const chainId = params.chainId;
    +      // Enabling before using the Keplr is recommended.
    +      // This method will ask the user whether to allow access if they haven't visited this website.
    +      // Also, it will request that the user unlock the wallet if the wallet is locked.
    +      await window.keplr.enable(chainId);
    +    }
    +  }
    +
    +  return (
    +    <div className={styles.center}>
    +      <button className={styles.keplrButton} onClick={add}>
    +        Add/switch To {params.chainName}
    +      </button>
    +    </div>
    +  );
    +}
    // @site/src/components/AddNetworkKeplr.js
    +import React from "react";
    +import styles from "./Keplr.module.css";
    +
    +export default function AddNetworkKeplr({ params }) {
    +  async function add() {
    +    if (!window.keplr) {
    +      alert("Please install keplr extension");
    +    } else {
    +      if (window.keplr.experimentalSuggestChain) {
    +        try {
    +          await window.keplr.experimentalSuggestChain({
    +            chainId: params.chainId,
    +            chainName: params.chainName,
    +            rpc: params.rpc,
    +            rest: params.rest,
    +            bip44: {
    +              coinType: 118,
    +            },
    +            bech32Config: {
    +              bech32PrefixAccAddr: "celestia",
    +              bech32PrefixAccPub: "celestia" + "pub",
    +              bech32PrefixValAddr: "celestia" + "valoper",
    +              bech32PrefixValPub: "celestia" + "valoperpub",
    +              bech32PrefixConsAddr: "celestia" + "valcons",
    +              bech32PrefixConsPub: "celestia" + "valconspub",
    +            },
    +            currencies: [
    +              {
    +                coinDenom: "TIA",
    +                coinMinimalDenom: "utia",
    +                coinDecimals: 6,
    +                coinGeckoId: "celestia",
    +              },
    +            ],
    +            feeCurrencies: [
    +              {
    +                coinDenom: "TIA",
    +                coinMinimalDenom: "utia",
    +                coinDecimals: 6,
    +                coinGeckoId: "celestia",
    +                gasPriceStep: {
    +                  low: 0.01,
    +                  average: 0.02,
    +                  high: 0.1,
    +                },
    +              },
    +            ],
    +            stakeCurrency: {
    +              coinDenom: "TIA",
    +              coinMinimalDenom: "utia",
    +              coinDecimals: 6,
    +              coinGeckoId: "celestia",
    +            },
    +          });
    +        } catch {
    +          alert("Failed to suggest the chain");
    +        }
    +      }
    +      const chainId = params.chainId;
    +      // Enabling before using the Keplr is recommended.
    +      // This method will ask the user whether to allow access if they haven't visited this website.
    +      // Also, it will request that the user unlock the wallet if the wallet is locked.
    +      await window.keplr.enable(chainId);
    +    }
    +  }
    +
    +  return (
    +    <div className={styles.center}>
    +      <button className={styles.keplrButton} onClick={add}>
    +        Add/switch To {params.chainName}
    +      </button>
    +    </div>
    +  );
    +}

    We still need to pass the Celestia network parameters to the AddNetworkKeplr function:

    js
    import '@site/src/components/AddNetworkKeplr'
    +
    +export const MAINNET_PARAMS = {`{
    +  chainId: 'celestia',
    +  chainName: 'Celestia',
    +  rpc: 'https://rpc.lunaroasis.net/',
    +  rest: 'https://api.lunaroasis.net/'
    +}`}
    +
    +{<AddNetworkKeplr params={MAINNET_PARAMS}/>}
    import '@site/src/components/AddNetworkKeplr'
    +
    +export const MAINNET_PARAMS = {`{
    +  chainId: 'celestia',
    +  chainName: 'Celestia',
    +  rpc: 'https://rpc.lunaroasis.net/',
    +  rest: 'https://api.lunaroasis.net/'
    +}`}
    +
    +{<AddNetworkKeplr params={MAINNET_PARAMS}/>}
    js
    import '@site/src/components/AddNetworkKeplr'
    +
    +export const MOCHA_PARAMS = {`{
    +  chainId: 'mocha-4',
    +  chainName: 'Mocha testnet',
    +  rpc: 'https://rpc-mocha.pops.one/',
    +  rest: 'https://api-mocha.pops.one/'
    +}`}
    +
    +{<AddNetworkKeplr params={MOCHA_PARAMS}/>}
    import '@site/src/components/AddNetworkKeplr'
    +
    +export const MOCHA_PARAMS = {`{
    +  chainId: 'mocha-4',
    +  chainName: 'Mocha testnet',
    +  rpc: 'https://rpc-mocha.pops.one/',
    +  rest: 'https://api-mocha.pops.one/'
    +}`}
    +
    +{<AddNetworkKeplr params={MOCHA_PARAMS}/>}
    js
    import '@site/src/components/AddNetworkKeplr'
    +
    +export const ARABICA_PARAMS = {`{
    +  chainId: 'arabica-11',
    +  chainName: 'Arabica devnet',
    +  rpc: 'https://rpc.celestia-arabica-11.com/',
    +  rest: 'https://api.celestia-arabica-11.com'
    +}`}
    +
    +{<AddNetworkKeplr params={ARABICA_PARAMS}/>}
    import '@site/src/components/AddNetworkKeplr'
    +
    +export const ARABICA_PARAMS = {`{
    +  chainId: 'arabica-11',
    +  chainName: 'Arabica devnet',
    +  rpc: 'https://rpc.celestia-arabica-11.com/',
    +  rest: 'https://api.celestia-arabica-11.com'
    +}`}
    +
    +{<AddNetworkKeplr params={ARABICA_PARAMS}/>}

    Now, we can connect to the network that you would like to use in Keplr wallet.

    Adding a custom chain to Leap

    If you want to add a custom chain to Leap, you can do so by:

    1. Clicking the Cosmos logo in the top corner of Leap wallet
    2. Scrolling down and clicking "Add new chain"

    You can then add the following parameters:

    • Chain Id: arabica-11
    • Chain Name: Arabica devnet
    • New RPC URL: https://rpc.celestia-arabica-11.com/
    • New REST URL: https://api.celestia-arabica-11.com
    • Address Prefix: celestia
    • Native Denom: utia
    • Coin Type: 118
    • Decimals: 6
    • Block explorer URL (optional): https://explorer.celestia-arabica-10.com

    Now, click Add chain and you will be able to view your Arabica account balance and transactions in Leap wallet.

    You'll see that you're connected to Arabica Devnet.

    Adding a custom chain to Cosmostation

    Click the hamburger menu icon in the top corner of Cosmostation wallet. Scroll down and click "Add Custom Chain"

    You can then add the following parameters:

    • Custom Chain name: Mocha testnet
    • Rest URL: https://api-mocha.pops.one
    • New RPC URL: https://rpc-mocha.pops.one
    • Currency symbol: TIA
    • Address prefix: celestia
    • Denom: utia
    • Symbol image URL (optional): https://raw.githubusercontent.com/cosmos/chain-registry/master/testnets/celestiatestnet/images/celestia.svg
    • Explorer URL (optional): https://testnet.mintscan.io/celestia-testnet
    • Coin Type: 118
    • Decimals: 6
    • Gas rate Tiny: 0.1
    • Gas rate Low: 0.25
    • Gas rate Average: 0.5

    Now, click Add a custom chain and you will be able to view your Celestia account balance and transactions in Cosmostation wallet.

    Switch chains to "Mocha testnet" and you'll see that you're connected to Celestia's Mocha testnet!

    + + + + \ No newline at end of file diff --git a/favicons/favicon-dark.ico b/favicons/favicon-dark.ico new file mode 100644 index 00000000000..298e787efa7 Binary files /dev/null and b/favicons/favicon-dark.ico differ diff --git a/favicons/favicon-dark.png b/favicons/favicon-dark.png new file mode 100644 index 00000000000..c2a1cee0b0d Binary files /dev/null and b/favicons/favicon-dark.png differ diff --git a/favicons/favicon-dark.svg b/favicons/favicon-dark.svg new file mode 100644 index 00000000000..e7ec82a775c --- /dev/null +++ b/favicons/favicon-dark.svg @@ -0,0 +1,11 @@ + + + + + + + \ No newline at end of file diff --git a/favicons/favicon.ico b/favicons/favicon.ico new file mode 100644 index 00000000000..32e9a6a2604 Binary files /dev/null and b/favicons/favicon.ico differ diff --git a/favicons/favicon.png b/favicons/favicon.png new file mode 100644 index 00000000000..b97a76500c0 Binary files /dev/null and b/favicons/favicon.png differ diff --git a/favicons/favicon.svg b/favicons/favicon.svg new file mode 100644 index 00000000000..0338485cf22 --- /dev/null +++ b/favicons/favicon.svg @@ -0,0 +1,11 @@ + + + + + + + \ No newline at end of file diff --git a/fonts/Ruberoid/Ruberoid-Bold.otf b/fonts/Ruberoid/Ruberoid-Bold.otf new file mode 100644 index 00000000000..d9fae2ea6ca Binary files /dev/null and b/fonts/Ruberoid/Ruberoid-Bold.otf differ diff --git a/grove/grove-sandbox.png b/grove/grove-sandbox.png new file mode 100644 index 00000000000..f9ea80093ea Binary files /dev/null and b/grove/grove-sandbox.png differ diff --git a/hashmap.json b/hashmap.json new file mode 100644 index 00000000000..5174fe7b741 --- /dev/null +++ b/hashmap.json @@ -0,0 +1 @@ +{"developers_arbitrum-bridge.md":"7321b996","developers_blobstream-x-deploy.md":"33965d1e","community_modular-meetup-guide.md":"ad87bef9","developers_blobstream-rollups.md":"b8b40693","community_modular-meetup-intro.md":"00a93576","developers_arbitrum-full-node.md":"1f75a4dc","developers_integrate-celestia.md":"02a50a7f","developers_arbitrum-dapp-deploy.md":"3f1872f8","community_itn-tos.md":"fda6f837","readme.md":"73690789","community_foundation-delegation-program.md":"0eb350d7","developers_arbitrum-deploy.md":"be261105","developers_blobstream-offchain.md":"162a7fc9","developers_gm-portal-bubs.md":"230e5c37","developers_optimism-devnet.md":"3b7c5762","developers_blobstream.md":"bbaa04fb","developers_bubs-testnet.md":"50110685","community_speaker-list.md":"fd89d412","community_calendar.md":"46d438d4","community_overview.md":"c9f170b3","developers_ethereum-fallback.md":"88698cb8","developers_blobstream-contracts.md":"8c20e3c9","nodes_mocha-testnet.md":"d00a2606","nodes_overview.md":"b302dbe6","nodes_participate.md":"4b9d8d8b","developers_celestia-node-key.md":"db3dee61","developers_node-tutorial.md":"8d60e9b7","nodes_systemd.md":"459598f1","community_modular-meetup-toolkit.md":"056ae043","developers_optimism.md":"8200414c","developers_arbitrum-smart-contract.md":"3069abf8","nodes_celestia-app-vesting.md":"0867750b","nodes_environment.md":"31fdd4c4","nodes_full-storage-node.md":"03ad71bf","nodes_light-node.md":"92409172","learn_how-celestia-works_transaction-lifecycle.md":"c6f24695","nodes_celestia-app-multisig.md":"e4b4f44c","developers_blobstream-proof-queries.md":"4443c323","nodes_quick-start.md":"05f1f426","learn_paying-for-blobspace.md":"50c0da43","learn_retrievability.md":"c6864d62","learn_staking-governance-supply.md":"b7f1bae9","learn_staking.md":"d69d2c40","learn_tia.md":"fc5441b2","nodes_arabica-devnet.md":"1651e52a","learn_how-celestia-works_monolithic-vs-modular.md":"db0fb9d5","developers_intro-to-op-stack.md":"0252701c","nodes_validator-node.md":"6e656228","developers_arbitrum-integration.md":"fa70b233","nodes_bridge-node.md":"c77c0ff2","developers_build-whatever.md":"1a071340","nodes_celestia-app-upgrade-monitor.md":"91d63439","developers_deploy-on-bubs.md":"c2d7ac48","community_coc.md":"e9cffa22","developers_node-api.md":"df79f906","developers_rollkit.md":"26b9c7c3","developers_requesting-data-commitment-ranges.md":"3fe57c2b","developers_wallets.md":"a10fbd85","index.md":"10ada84c","developers_submit-data.md":"64bfc8e2","developers_transaction-resubmission.md":"96bd7ea9","developers_full-stack-modular-development-guide.md":"9f203595","learn_how-celestia-works_data-availability-faq.md":"3d484283","learn_how-celestia-works_data-availability-layer.md":"c6171925","nodes_celestia-app-slashing.md":"4d50cd08","nodes_celestia-app-commands.md":"63e455f7","nodes_celestia-app-metrics.md":"74ef2ef7","developers_prompt-scavenger.md":"27b40ddb","nodes_celestia-app-wallet.md":"e787539c","nodes_decide-node.md":"03a657c5","nodes_celestia-node-custom-networks.md":"804b9f53","nodes_celestia-app.md":"1b04d1e9","nodes_celestia-node-troubleshooting.md":"b993a166","learn_how-celestia-works_overview.md":"3d86106d","nodes_config-toml.md":"2c893551","learn_how-to-stake-tia.md":"4ed3b67d","nodes_hardfork-process.md":"431e0126","nodes_celestia-node-metrics.md":"cfabfbdc","nodes_celestia-node.md":"791b437d","nodes_instantiate-testnet.md":"db07a97a","nodes_ibc-relayer.md":"b3114711","nodes_docker-images.md":"a7ae248f","nodes_full-consensus-node.md":"efda1850","nodes_mainnet.md":"dd077830"} diff --git a/img/Celestia-Arbitrum.png b/img/Celestia-Arbitrum.png new file mode 100644 index 00000000000..d66ff56797f Binary files /dev/null and b/img/Celestia-Arbitrum.png differ diff --git a/img/Celestia_Bubs_Testnet.jpg b/img/Celestia_Bubs_Testnet.jpg new file mode 100644 index 00000000000..67ce6d5b13d Binary files /dev/null and b/img/Celestia_Bubs_Testnet.jpg differ diff --git a/img/Celestia_Modular_meetup2.jpg b/img/Celestia_Modular_meetup2.jpg new file mode 100644 index 00000000000..97705e5b971 Binary files /dev/null and b/img/Celestia_Modular_meetup2.jpg differ diff --git a/img/Celestia_ethereum-fallback.jpg b/img/Celestia_ethereum-fallback.jpg new file mode 100644 index 00000000000..6a225429f9a Binary files /dev/null and b/img/Celestia_ethereum-fallback.jpg differ diff --git a/img/Mainnet-Beta.png b/img/Mainnet-Beta.png new file mode 100644 index 00000000000..b83c77c24be Binary files /dev/null and b/img/Mainnet-Beta.png differ diff --git a/img/arabica-devnet.png b/img/arabica-devnet.png new file mode 100644 index 00000000000..2a91f8f8e0e Binary files /dev/null and b/img/arabica-devnet.png differ diff --git a/img/blobstream/Blobstream.png b/img/blobstream/Blobstream.png new file mode 100644 index 00000000000..c527ddada0f Binary files /dev/null and b/img/blobstream/Blobstream.png differ diff --git a/img/blobstream/Celestia_Blobstream_X1b.png b/img/blobstream/Celestia_Blobstream_X1b.png new file mode 100644 index 00000000000..eb6f9e97dba Binary files /dev/null and b/img/blobstream/Celestia_Blobstream_X1b.png differ diff --git a/img/blobstream/Celestia_Blobstream_X2b.png b/img/blobstream/Celestia_Blobstream_X2b.png new file mode 100644 index 00000000000..651457bb2c8 Binary files /dev/null and b/img/blobstream/Celestia_Blobstream_X2b.png differ diff --git a/img/blobstream/Celestia_Blobstream_attestation_flow.jpg b/img/blobstream/Celestia_Blobstream_attestation_flow.jpg new file mode 100644 index 00000000000..7abb2d47972 Binary files /dev/null and b/img/blobstream/Celestia_Blobstream_attestation_flow.jpg differ diff --git a/img/blobstream/blobstream-commitment-diagram.png b/img/blobstream/blobstream-commitment-diagram.png new file mode 100644 index 00000000000..d64f4335301 Binary files /dev/null and b/img/blobstream/blobstream-commitment-diagram.png differ diff --git a/img/blobstream/blobstream-orchestrator.png b/img/blobstream/blobstream-orchestrator.png new file mode 100644 index 00000000000..2fc711ea077 Binary files /dev/null and b/img/blobstream/blobstream-orchestrator.png differ diff --git a/img/blobstream/blobstream-relayer.png b/img/blobstream/blobstream-relayer.png new file mode 100644 index 00000000000..ed0ffa9d4a3 Binary files /dev/null and b/img/blobstream/blobstream-relayer.png differ diff --git a/img/blobstream/blobstream-square.png b/img/blobstream/blobstream-square.png new file mode 100644 index 00000000000..66382ed42d2 Binary files /dev/null and b/img/blobstream/blobstream-square.png differ diff --git a/img/blobstream/blobstream_logo.png b/img/blobstream/blobstream_logo.png new file mode 100644 index 00000000000..583a74c6e4a Binary files /dev/null and b/img/blobstream/blobstream_logo.png differ diff --git a/img/cohort-timeline.jpg b/img/cohort-timeline.jpg new file mode 100644 index 00000000000..a59709e6a45 Binary files /dev/null and b/img/cohort-timeline.jpg differ diff --git a/img/da-and-validity.png b/img/da-and-validity.png new file mode 100644 index 00000000000..5f1589effb6 Binary files /dev/null and b/img/da-and-validity.png differ diff --git a/img/foundation-delegation-program.jpg b/img/foundation-delegation-program.jpg new file mode 100644 index 00000000000..bd7c0c1f618 Binary files /dev/null and b/img/foundation-delegation-program.jpg differ diff --git a/img/gm-arb.png b/img/gm-arb.png new file mode 100644 index 00000000000..b72abf72455 Binary files /dev/null and b/img/gm-arb.png differ diff --git a/img/gm_bubs.png b/img/gm_bubs.png new file mode 100644 index 00000000000..cef39c92e82 Binary files /dev/null and b/img/gm_bubs.png differ diff --git a/img/gm_contract.png b/img/gm_contract.png new file mode 100644 index 00000000000..a1f6cf8d44b Binary files /dev/null and b/img/gm_contract.png differ diff --git a/img/keplr.png b/img/keplr.png new file mode 100644 index 00000000000..3f558ef7024 Binary files /dev/null and b/img/keplr.png differ diff --git a/img/keplr/keplr1.gif b/img/keplr/keplr1.gif new file mode 100644 index 00000000000..2198944c436 Binary files /dev/null and b/img/keplr/keplr1.gif differ diff --git a/img/keplr/keplr2.gif b/img/keplr/keplr2.gif new file mode 100644 index 00000000000..51ba03fbfbf Binary files /dev/null and b/img/keplr/keplr2.gif differ diff --git a/img/keplr/keplr3.gif b/img/keplr/keplr3.gif new file mode 100644 index 00000000000..dd3f5652815 Binary files /dev/null and b/img/keplr/keplr3.gif differ diff --git a/img/keplr/keplr4.gif b/img/keplr/keplr4.gif new file mode 100644 index 00000000000..0698cebc5bb Binary files /dev/null and b/img/keplr/keplr4.gif differ diff --git a/img/leap.png b/img/leap.png new file mode 100644 index 00000000000..153bcdc999c Binary files /dev/null and b/img/leap.png differ diff --git a/img/leap/leap1.gif b/img/leap/leap1.gif new file mode 100644 index 00000000000..71cf3d600ee Binary files /dev/null and b/img/leap/leap1.gif differ diff --git a/img/leap/leap2.gif b/img/leap/leap2.gif new file mode 100644 index 00000000000..d7ac54e7da1 Binary files /dev/null and b/img/leap/leap2.gif differ diff --git a/img/leap/leap3.gif b/img/leap/leap3.gif new file mode 100644 index 00000000000..9691005b010 Binary files /dev/null and b/img/leap/leap3.gif differ diff --git a/img/learn/Celestia_TIA_Allocation_at_Genesis.png b/img/learn/Celestia_TIA_Allocation_at_Genesis.png new file mode 100644 index 00000000000..1dc9b99b9a0 Binary files /dev/null and b/img/learn/Celestia_TIA_Allocation_at_Genesis.png differ diff --git a/img/learn/Celestia_TIA_Available_Supply.png b/img/learn/Celestia_TIA_Available_Supply.png new file mode 100644 index 00000000000..c8c94521a4f Binary files /dev/null and b/img/learn/Celestia_TIA_Available_Supply.png differ diff --git a/img/learn/Celestia_TIA_Inflation.png b/img/learn/Celestia_TIA_Inflation.png new file mode 100644 index 00000000000..fd5249ac17e Binary files /dev/null and b/img/learn/Celestia_TIA_Inflation.png differ diff --git a/img/learn/celestia-app.png b/img/learn/celestia-app.png new file mode 100644 index 00000000000..725fc719d7a Binary files /dev/null and b/img/learn/celestia-app.png differ diff --git a/img/learn/consensus-da.png b/img/learn/consensus-da.png new file mode 100644 index 00000000000..ac20d9417fe Binary files /dev/null and b/img/learn/consensus-da.png differ diff --git a/img/learn/data-availability-faq/Data-availability.png b/img/learn/data-availability-faq/Data-availability.png new file mode 100644 index 00000000000..fcdd70e1fc0 Binary files /dev/null and b/img/learn/data-availability-faq/Data-availability.png differ diff --git a/img/learn/data-availability-faq/Data-storage.png b/img/learn/data-availability-faq/Data-storage.png new file mode 100644 index 00000000000..a29e7eb8b40 Binary files /dev/null and b/img/learn/data-availability-faq/Data-storage.png differ diff --git a/img/learn/monolithic-modular.png b/img/learn/monolithic-modular.png new file mode 100644 index 00000000000..86acb55b375 Binary files /dev/null and b/img/learn/monolithic-modular.png differ diff --git a/img/learn/nmt.png b/img/learn/nmt.png new file mode 100644 index 00000000000..709f1dc9c4a Binary files /dev/null and b/img/learn/nmt.png differ diff --git a/img/learn/reed-solomon-encoding.png b/img/learn/reed-solomon-encoding.png new file mode 100644 index 00000000000..b520ef68ebd Binary files /dev/null and b/img/learn/reed-solomon-encoding.png differ diff --git a/img/learn/tx-lifecycle.png b/img/learn/tx-lifecycle.png new file mode 100644 index 00000000000..2df4ca8954a Binary files /dev/null and b/img/learn/tx-lifecycle.png differ diff --git a/img/mocha.jpg b/img/mocha.jpg new file mode 100644 index 00000000000..62fd00630d8 Binary files /dev/null and b/img/mocha.jpg differ diff --git a/img/modular_fellows.jpg b/img/modular_fellows.jpg new file mode 100644 index 00000000000..2927f8b7e4d Binary files /dev/null and b/img/modular_fellows.jpg differ diff --git a/img/nitrogen-testnet.jpg b/img/nitrogen-testnet.jpg new file mode 100644 index 00000000000..922e38bca10 Binary files /dev/null and b/img/nitrogen-testnet.jpg differ diff --git a/img/node-requirements.jpg b/img/node-requirements.jpg new file mode 100644 index 00000000000..b8d7db35a07 Binary files /dev/null and b/img/node-requirements.jpg differ diff --git a/img/nodes/BridgeNodes.png b/img/nodes/BridgeNodes.png new file mode 100644 index 00000000000..6777e8bc20b Binary files /dev/null and b/img/nodes/BridgeNodes.png differ diff --git a/img/nodes/LightNodes.png b/img/nodes/LightNodes.png new file mode 100644 index 00000000000..342e96971f6 Binary files /dev/null and b/img/nodes/LightNodes.png differ diff --git a/img/nodes/full-consensus-node.png b/img/nodes/full-consensus-node.png new file mode 100644 index 00000000000..6badd498e25 Binary files /dev/null and b/img/nodes/full-consensus-node.png differ diff --git a/img/nodes/full-storage-node.png b/img/nodes/full-storage-node.png new file mode 100644 index 00000000000..c28f2c41884 Binary files /dev/null and b/img/nodes/full-storage-node.png differ diff --git a/img/nodes/validator.png b/img/nodes/validator.png new file mode 100644 index 00000000000..0766fa467f3 Binary files /dev/null and b/img/nodes/validator.png differ diff --git a/img/rollkit.png b/img/rollkit.png new file mode 100644 index 00000000000..e9ecd3471c4 Binary files /dev/null and b/img/rollkit.png differ diff --git a/index.html b/index.html new file mode 100644 index 00000000000..8df2b68cdd9 --- /dev/null +++ b/index.html @@ -0,0 +1,45 @@ + + + + + + Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/learn/how-celestia-works/data-availability-faq.html b/learn/how-celestia-works/data-availability-faq.html new file mode 100644 index 00000000000..914f0fcceea --- /dev/null +++ b/learn/how-celestia-works/data-availability-faq.html @@ -0,0 +1,45 @@ + + + + + + Data availability FAQ | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Data availability FAQ

    What is data availability?

    Data availability answers the question, has this data been published? Specifically, a node will verify data availability when it receives a new block that is getting added to the chain. The node will attempt to download all the transaction data for the new block to verify availability. If the node can download all the transaction data, then it successfully verified data availability, proving that the block data was actually published to the network.

    Modular VS Monolithic

    As you’ll see, modular blockchains like Celestia employ other primitives that allow nodes to verify data availability more efficiently. Data availability is critical to the security of any blockchain because it ensures that anyone can inspect the ledger of transactions and verify it. Data availability becomes particularly problematic when scaling blockchains. As the blocks get bigger, it becomes impractical for normal users to download all the data, and therefore users can no longer verify the chain.

    What is the data availability problem?

    The problem with data availability occurs when the transaction data for a newly proposed block cannot be downloaded and verified. This type of attack by a block producer is called a data withholding attack, which sees the block producer withhold transaction data of a new block.

    Since transaction data is withheld, nodes cannot update to the latest state. Such an attack can have numerous consequences, from halting a chain to gaining the ability to steal funds. The severity of the consequences will depend on the type of blockchain (L1 or L2) and whether data availability is kept onchain or offchain. The data availability problem commonly arises around L2 scaling solutions like rollups and validiums.

    How do nodes verify data availability in Celestia?

    In most blockchains, nodes that verify data availability do so by downloading all transaction data for a block. If they are able to download all the data, they have verified its availability. In Celestia, light nodes have access to a new mechanism to verify data availability without needing to download all the data for a block. This new primitive for verifying data availability is called data availability sampling.

    What is data availability sampling?

    Data availability sampling is a mechanism for light nodes to verify data availability without having to download all data for a block. Data availability sampling (DAS) works by having light nodes conduct multiple rounds of random sampling for small portions of block data. As a light node completes more rounds of sampling for block data, it increases its confidence that data is available. Once the light node successfully reaches a predetermined confidence level (e.g. 99%) it will consider the block data as available.

    Want a simpler explanation? Check out this thread on how data availability sampling is like flipping a coin.

    What are some of the security assumptions that Celestia makes for data availability sampling?

    Celestia assumes that there is a minimum number of light nodes that are conducting data availability sampling for a given block size. This assumption is necessary so that a full node can reconstruct an entire block from the portions of data light nodes sampled and stored. The amount of light nodes that are needed will depend on the block size - for bigger blocks more light nodes are assumed to be running.

    A second notable assumption that is made by light nodes is that they are connected to at least one honest full node. This ensures that they can receive fraud proofs for incorrectly erasure coded blocks. If a light node is not connected to an honest full node, such as during an eclipse attack, it can’t verify that the block is improperly constructed.

    Why is block reconstruction necessary for security?

    In Celestia, blocks need to be erasure coded so that there is redundant data to aid the data availability sampling process. However, nodes tasked with erasure coding the data could do so incorrectly. Since Celestia uses fraud proofs to verify that erasure coding is incorrect, the full block data is needed to generate a bad encoding fraud proof.

    There could be a situation where validators only provide data to light nodes and not full nodes. If the full nodes don’t have the ability to reconstruct the full block from the portions of data stored by light nodes, they wouldn’t be able to generate a bad encoding fraud proof.

    What is data storage?

    Data storage is concerned with the ability to store and access past transaction data.

    Modular VS Monolithic

    Data storage and retrieval is needed for multiple purposes, such as:

    • Reading the information of a previous transaction
    • Syncing a node
    • Indexing and serving transaction data
    • Retrieving NFT information

    What is the problem around data storage?

    The issue with data storage is whether past transaction data can be stored and successfully retrieved at a later time. The inability to retrieve historical transaction data can cause problems, such as users being unable to access information about their past transactions or nodes that cannot sync from genesis. Luckily, the assumptions around storing and accessing past data are weak. Only a single copy of a blockchain’s history needs to be accessible for users to gain access to historical transaction data. In other words, data storage security is a 1 of N honesty assumption.

    What is the difference between data availability and data storage?

    Data availability is about verifying that transaction data for a new block is public and available. In contrast, data storage involves storing and accessing past transaction data from old blocks.

    Where does blockchain state fit into this?

    Up until now it’s been all about transaction data, but blockchain state is a related topic. The state is different from transaction data. Specifically, the state is like a current snapshot of the network, which includes information about account balances, smart contract balances, and validator set info. Problems that arise from the size of the state are different in nature than those around data availability and retrievability.

    Why doesn’t Celestia incentivize storage of historical data?

    Most blockchains don’t incentivize storage of data because it shouldn’t be the responsibility of a blockchain to guarantee past data will be retrievable forever. In addition, the data storage problem only requires a single party to store and provide the data for users, which is not a strong problem. As such, Celestia’s purpose is to provide a secure and scalable way to verify the availability of data. Once data has been verified as available, the job of storing and retrieving historical data is left up to other entities that require the data. Luckily, there are natural incentives for outside parties to store and serve historical data to users.

    Who may store historical data if there is no reward?

    There are multiple types of actors that may be likely to store historical data. Some of those include:

    • Block explorers that provide access to past transaction data.
    • Indexers that provide API queries for past data.
    • Applications or rollups that require historical data for certain processes.
    • Users that want to guarantee that they will have access to their transaction history.

    What are some things blockchains can do to provide stronger assurances of data retrievability?

    • Reward nodes based on the amount of transaction data they store and requests for data they serve (this is the case with some data storage blockchains, like Filecoin).
    • Publish transaction data onto a data storage blockchain that incentivizes storing and serving requests for historical data.
    + + + + \ No newline at end of file diff --git a/learn/how-celestia-works/data-availability-layer.html b/learn/how-celestia-works/data-availability-layer.html new file mode 100644 index 00000000000..aad4f2c8450 --- /dev/null +++ b/learn/how-celestia-works/data-availability-layer.html @@ -0,0 +1,45 @@ + + + + + + Celestia's data availability layer | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Celestia's data availability layer

    Celestia is a data availability (DA) layer that provides a scalable solution to the data availability problem. Due to the permissionless nature of the blockchain networks, a DA layer must provide a mechanism for the execution and settlement layers to check in a trust-minimized way whether transaction data is indeed available.

    Two key features of Celestia's DA layer are data availability sampling (DAS) and Namespaced Merkle trees (NMTs). Both features are novel blockchain scaling solutions: DAS enables light nodes to verify data availability without needing to download an entire block; NMTs enable execution and settlement layers on Celestia to download transactions that are only relevant to them.

    Data availability sampling (DAS)

    In general, light nodes download only block headers that contain commitments (i.e., Merkle roots) of the block data (i.e., the list of transactions).

    To make DAS possible, Celestia uses a 2-dimensional Reed-Solomon encoding scheme to encode the block data: every block data is split into k×k shares, arranged in a k×k matrix, and extended with parity data into a 2k×2k extended matrix by applying multiple times Reed-Solomon encoding.

    Then, 4k separate Merkle roots are computed for the rows and columns of the extended matrix; the Merkle root of these Merkle roots is used as the block data commitment in the block header.

    2D Reed-Soloman (RS) Encoding

    To verify that the data is available, Celestia light nodes are sampling the 2k×2k data shares.

    Every light node randomly chooses a set of unique coordinates in the extended matrix and queries full nodes for the data shares and the corresponding Merkle proofs at those coordinates. If light nodes receive a valid response for each sampling query, then there is a high probability guarantee that the whole block's data is available.

    Additionally, every received data share with a correct Merkle proof is gossiped to the network. As a result, as long as the Celestia light nodes are sampling together enough data shares (i.e., at least k×k unique shares), the full block can be recovered by honest full nodes.

    For more details on DAS, take a look at the original paper.

    Scalability

    DAS enables Celestia to scale the DA layer. DAS can be performed by resource-limited light nodes since each light node only samples a small portion of the block data. The more light nodes there are in the network, the more data they can collectively download and store.

    This means that increasing the number of light nodes performing DAS allows for larger blocks (i.e., with more transactions), while still keeping DAS feasible for resource-limited light nodes. However, in order to validate block headers, Celestia light nodes need to download the 4k intermediate Merkle roots.

    For a block data size of n2 bytes, this means that every light node must download O(n) bytes. Therefore, any improvement in the bandwidth capacity of Celestia light nodes has a quadratic effect on the throughput of Celestia's DA layer.

    Fraud proofs of incorrectly extended data

    The requirement of downloading the 4k intermediate Merkle roots is a consequence of using a 2-dimensional Reed-Solomon encoding scheme. Alternatively, DAS could be designed with a standard (i.e., 1-dimensional) Reed-Solomon encoding, where the original data is split into k shares and extended with k additional shares of parity data. Since the block data commitment is the Merkle root of the 2k resulting data shares, light nodes no longer need to download O(n) bytes to validate block headers.

    The downside of the standard Reed-Solomon encoding is dealing with malicious block producers that generate the extended data incorrectly.

    This is possible as Celestia does not require a majority of the consensus (i.e., block producers) to be honest to guarantee data availability. Thus, if the extended data is invalid, the original data might not be recoverable, even if the light nodes are sampling sufficient unique shares (i.e., at least k for a standard encoding and k×k for a 2-dimensional encoding).

    As a solution, Fraud Proofs of Incorrectly Generated Extended Data enable light nodes to reject blocks with invalid extended data. Such proofs require reconstructing the encoding and verifying the mismatch. With standard Reed-Solomon encoding, this entails downloading the original data, i.e., n2 bytes. Contrastingly, with 2-dimensional Reed-Solomon encoding, only O(n) bytes are required as it is sufficient to verify only one row or one column of the extended matrix.

    Namespaced Merkle trees (NMTs)

    Celestia partitions the block data into multiple namespaces, one for every application (e.g., rollup) using the DA layer. As a result, every application needs to download only its own data and can ignore the data of other applications.

    For this to work, the DA layer must be able to prove that the provided data is complete, i.e., all the data for a given namespace is returned. To this end, Celestia is using Namespaced Merkle trees (NMTs).

    An NMT is a Merkle tree with the leafs ordered by the namespace identifiers and the hash function modified so that every node in the tree includes the range of namespaces of all its descendants. The following figure shows an example of an NMT with height three (i.e., eight data shares). The data is partitioned into three namespaces.

    Namespaced Merkle Tree

    When an application requests the data for namespace 2, the DA layer must provide the data shares D3, D4, D5, and D6 and the nodes N2, N8 and N7 as proof (note that the application already has the root N14 from the block header).

    As a result, the application is able to check that the provided data is part of the block data. Furthermore, the application can verify that all the data for namespace 2 was provided. If the DA layer provides for example only the data shares D4 and D5, it must also provide nodes N12 and N11 as proofs. However, the application can identify that the data is incomplete by checking the namespace range of the two nodes, i.e., both N12 and N11 have descendants part of namespace 2.

    For more details on NMTs, refer to the original paper.

    Building a PoS blockchain for DA

    Providing data availability

    The Celestia DA layer consists of a PoS blockchain. Celestia is dubbing this blockchain as the celestia-app, an application that provides transactions to facilitate the DA layer and is built using Cosmos SDK. The following figure shows the main components of celestia-app.

    Main components of celestia-app

    celestia-app is built on top of celestia-core, a modified version of the Tendermint consensus algorithm. Among the more important changes to vanilla Tendermint, celestia-core:

    • Enables the erasure coding of block data (using the 2-dimensional Reed-Solomon encoding scheme).
    • Replaces the regular Merkle tree used by Tendermint to store block data with a Namespaced Merkle tree that enables the above layers (i.e., execution and settlement) to only download the needed data (for more details, see the section below describing use cases).

    For more details on the changes to Tendermint, take a look at the ADRs. Notice that celestia-core nodes are still using the Tendermint p2p network.

    Similarly to Tendermint, celestia-core is connected to the application layer (i.e., the state machine) by ABCI++, a major evolution of ABCI (Application Blockchain Interface).

    The celestia-app state machine is necessary to execute the PoS logic and to enable the governance of the DA layer.

    However, the celestia-app is data-agnostic -- the state machine neither validates nor stores the data that is made available by the celestia-app.

    + + + + \ No newline at end of file diff --git a/learn/how-celestia-works/monolithic-vs-modular.html b/learn/how-celestia-works/monolithic-vs-modular.html new file mode 100644 index 00000000000..937d0496da5 --- /dev/null +++ b/learn/how-celestia-works/monolithic-vs-modular.html @@ -0,0 +1,45 @@ + + + + + + Monolithic vs. modular blockchains | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Monolithic vs. modular blockchains

    Blockchains instantiate replicated state machines: the nodes in a permissionless distributed network apply an ordered sequence of deterministic transactions to an initial state resulting in a common final state.

    In other words, this means that nodes in a network all follow the same set of rules (i.e., an ordered sequence of transactions) to go from a starting point (i.e., an initial state) to an ending point (i.e., a common final state). This process ensures that all nodes in the network agree on the final state of the blockchain, even though they operate independently.

    This means blockchains require the following four functions:

    • Execution entails executing transactions that update the state correctly. Thus, execution must ensure that only valid transactions are executed, i.e., transactions that result in valid state machine transitions.
    • Settlement entails an environment for execution layers to verify proofs, resolve fraud disputes, and bridge between other execution layers.
    • Consensus entails agreeing on the order of the transactions.
    • Data Availability (DA) entails making the transaction data available. Note that execution, settlement, and consensus require DA.

    Traditional blockchains, i.e. monolithic blockchains, implement all four functions together in a single base consensus layer. The problem with monolithic blockchains is that the consensus layer must perform numerous different tasks, and it cannot be optimized for only one of these functions. As a result, the monolithic paradigm limits the throughput of the system.

    Modular VS Monolithic

    As a solution, modular blockchains decouple these functions among multiple specialized layers as part of a modular stack. Due to the flexibility that specialization provides, there are many possibilities in which that stack can be arranged. For example, one such arrangement is the separation of the four functions into three specialized layers.

    The base layer consists of DA and consensus and thus, is referred to as the Consensus and DA layer (or for brevity, the DA layer), while both settlement and execution are moved on top in their own layers. As a result, every layer can be specialized to optimally perform only its function, and thus, increase the throughput of the system. Furthermore, this modular paradigm enables multiple execution layers, i.e., rollups, to use the same settlement and DA layers.

    + + + + \ No newline at end of file diff --git a/learn/how-celestia-works/overview.html b/learn/how-celestia-works/overview.html new file mode 100644 index 00000000000..6c070ec4777 --- /dev/null +++ b/learn/how-celestia-works/overview.html @@ -0,0 +1,45 @@ + + + + + + Introduction | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Introduction

    Celestia is a modular data availability network that securely scales with the number of users, making it easy for anyone to launch their own blockchain.

    Celestia enables the next generation of scalable blockchain architectures - modular blockchains. Celestia scales by decoupling execution from consensus and introducing a new primitive, data availability sampling.

    The former entails that Celestia is only responsible for ordering transactions and guaranteeing their data availability; this is similar to reducing consensus to atomic broadcast.

    The latter provides an efficient solution to the data availability problem by only requiring resource-limited light nodes to sample a small number of random shares from each block to verify data availability.

    Interestingly, more light nodes that participate in sampling increases the amount of data that the network can safely handle, enabling the block size to increase without equally increasing the cost to verify the chain.

    + + + + \ No newline at end of file diff --git a/learn/how-celestia-works/transaction-lifecycle.html b/learn/how-celestia-works/transaction-lifecycle.html new file mode 100644 index 00000000000..28840ebaf13 --- /dev/null +++ b/learn/how-celestia-works/transaction-lifecycle.html @@ -0,0 +1,46 @@ + + + + + + The lifecycle of a celestia-app transaction | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    The lifecycle of a celestia-app transaction

    Users request the celestia-app to make data available by sending PayForBlobs transactions. Every such transaction consists of the identity of the sender, the data to be made available, also referred to as the message, the data size, the namespace, and a signature. Every block producer batches multiple PayForBlobs transactions into a block.

    Before proposing the block though, the producer passes it to the state machine via ABCI++, where each PayForBlobs transaction is split into a namespaced message (denoted by Msg in the figure below), i.e., the data together with the namespace ID, and an executable transaction (denoted by e-Tx in the figure below) that does not contain the data, but only a commitment that can be used at a later time to prove that the data was indeed made available.

    Thus, the block data consists of data partitioned into namespaces and executable transactions. Note that only these transactions are executed by the Celestia state machine once the block is committed.

    Lifecycle of a celestia-app Transaction

    Next, the block producer adds to the block header a commitment of the block data. As described in the "Celestia's data availability layer" page, the commitment is the Merkle root of the 4k intermediate Merkle roots (i.e., one for each row and column of the extended matrix). To compute this commitment, the block producer performs the following operations:

    • It splits the executable transactions and the namespaced data into shares. Every share consists of some bytes prefixed by a namespace. To this end, the executable transactions are associated with a reserved namespace.
    • It arranges these shares into a square matrix (row-wise). Note that the shares are padded to the next power of two. The outcome square of size k×k is referred to as the original data.
    • It extends the original data to a 2k×2k square matrix using the 2-dimensional Reed-Solomon encoding scheme described above. The extended shares (i.e., containing erasure data) are associated with another reserved namespace.
    • It computes a commitment for every row and column of the extended matrix using the NMTs described above.

    Thus, the commitment of the block data is the root of a Merkle tree with the leaves the roots of a forest of Namespaced Merkle subtrees, one for every row and column of the extended matrix.

    Checking data availability

    DA network

    To enhance connectivity, the celestia-node augments the celestia-app with a separate libp2p network, i.e., the so-called DA network, that serves DAS requests.

    Light nodes connect to a celestia-node in the DA network, listen to extended block headers (i.e., the block headers together with the relevant DA metadata, such as the 4k intermediate Merkle roots), and perform DAS on the received headers (i.e., ask for random data shares).

    Note that although it is recommended, performing DAS is optional -- light nodes could just trust that the data corresponding to the commitments in the block headers was indeed made available by the Celestia DA layer. In addition, light nodes can also submit transactions to the celestia-app, i.e., PayForBlobs transactions.

    While performing DAS for a block header, every light node queries Celestia Nodes for a number of random data shares from the extended matrix and the corresponding Merkle proofs. If all the queries are successful, then the light node accepts the block header as valid (from a DA perspective).

    If at least one of the queries fails (i.e., either the data share is not received or the Merkle proof is invalid), then the light node rejects the block header and tries again later. The retrial is necessary to deal with false negatives, i.e., block headers being rejected although the block data is available. This may happen due to network congestion for example.

    Alternatively, light nodes may accept a block header although the data is not available, i.e., a false positive. This is possible since the soundness property (i.e., if an honest light node accepts a block as available, then at least one honest full node will eventually have the entire block data) is probabilistically guaranteed (for more details, take a look at the original paper).

    By fine tuning Celestia's parameters (e.g., the number of data shares sampled by each light node) the likelihood of false positives can be sufficiently reduced such that block producers have no incentive to withhold the block data.

    + + + + \ No newline at end of file diff --git a/learn/how-to-stake-tia.html b/learn/how-to-stake-tia.html new file mode 100644 index 00000000000..bc6d3a812e0 --- /dev/null +++ b/learn/how-to-stake-tia.html @@ -0,0 +1,45 @@ + + + + + + How to stake TIA | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    How to stake TIA

    Celestia is a proof-of-stake blockchain based on the Cosmos SDK.

    Staking TIA as a delegator allows you to secure the Celestia network. This means that you can stake the native token TIA and vote on governance proposals.

    In this tutorial, you will learn how to stake TIA tokens via Keplr and Leap wallets.

    Select your preferred wallet

    Keplr
    Leap

    Stake TIA with Keplr wallet

    1️⃣ Open your Keplr browser extension

    Navigate to Staked and select Stake with Keplr Dashboard.

    This will open the Keplr dashboard in a new browser page.

    Keplr1

    2️⃣ Select Celestia network and search for a validator

    In the Keplr dashboard, select the Celestia network and pick a validator of your choice.

    Keplr1

    3️⃣ Stake your TIA tokens

    On the following screen enter amount of TIA tokens and select Stake.

    A Keplr popup will appear, requesting your approval for the transaction. Select Approve.

    Keplr1

    4️⃣ Confirm and manage your TIA

    After the transaction is confirmed, you will see the following overview dashboard where you can claim rewards, unstake, redelegate, or stake additional tokens.

    Keplr1

    Stake TIA with Leap wallet

    1️⃣ Open your Leap browser extension

    In top right select Celestia network and navigate to Stake.

    Similarly to previous step, select the +Stake button.

    Keplr1

    2️⃣ Select a validator and stake TIA

    On the following screen choose a validator of your choice, enter the desired amount, and click Review.

    Following that, review the transaction details and select Stake, then wait for the transaction to finalize.

    Keplr1

    3️⃣ Confirm and manage your TIA

    After the transaction is confirmed, you will see the following overview dashboard where you can claim rewards, unstake, redelegate, or stake additional tokens.

    Keplr1

    + + + + \ No newline at end of file diff --git a/learn/paying-for-blobspace.html b/learn/paying-for-blobspace.html new file mode 100644 index 00000000000..4d21e264c59 --- /dev/null +++ b/learn/paying-for-blobspace.html @@ -0,0 +1,46 @@ + + + + + + Paying for blobspace | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Paying for blobspace

    PayForBlobs transactions

    To publish data on Celestia, developers can submit PayForBlobs transactions. A PayForBlobs transaction consists of the identity of the sender, the data to be made available, the data size, the namespace, and a signature.

    Each PayForBlobs transaction is split into two parts: the blob or blobs which include the data to be made available along with the namespace, and the executable payment transaction which includes a commitment to the data.

    Both the blobs and executable payment transactions are put into the block within the appropriate namespace. The block data is extended using erasure coding and then Merkelized into a data root commitment included in the block header.

    Lifecycle of a celestia-app Transaction

    See the detailed life cycle of a Celestia transaction.

    Learn how to submit data to Celestia’s data availability layer.

    Fee market overview

    Celestia uses a standard gas-price prioritised mempool. This means that transactions with higher fees will be prioritised by validators. Fees are comprised of a flat fee per transaction and then a variable fee based on the size of each blob in the transaction.

    Understand how fees are calculated on Celestia in the overview on submitting PFB transactions.

    + + + + \ No newline at end of file diff --git a/learn/retrievability.html b/learn/retrievability.html new file mode 100644 index 00000000000..180264f8663 --- /dev/null +++ b/learn/retrievability.html @@ -0,0 +1,45 @@ + + + + + + Data retrievability and pruning | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Data retrievability and pruning

    The purpose of data availability layers such as Celestia is to ensure that block data is provably published, so that applications and rollups can know what the state of their chain is, and store that data. Once the data is published, data availability layers do not inherently guarantee that historical data will be permanently stored and remain retrievable.

    In this document, we discuss the state of data retrievability and pruning in Celestia, as well as some tips for rollup developers in order to ensure that syncing new rollup nodes is possible.

    Data retrievability and pruning in celestia-node

    As of version v0.13.0, celestia-node has implemented a light node sampling window of 30 days, as specified in CIP-4. This means that once pruning is implemented, light nodes will now only sample blocks within a 30-day window instead of sampling all blocks from genesis. This change introduces the concept of pruning to celestia-node, where data outside of the 30-day window may not be stored by light nodes, marking a significant update in how data retrievability and storage are managed within the network (v0.13.0 release notes).

    Data blobs older than the recency window will be pruned by default on light nodes, after pruning is fully implemented, but will continue to be stored by archival nodes that do not prune data. Light nodes will be able to query historic blob data in namespaces from archival nodes, as long as archival nodes exist on the public network.

    Once pruning is fully implemented, light nodes will only perform data availability sampling for blocks within the data recency window of 30 days.

    Suggested practices for rollups

    Rollups may need to access historic data in order to allow new rollup nodes to reconstruct the latest state by replaying historic blocks. Once data has been published on Celestia and guaranteed to have been made available, rollups and applications are responsible for storing their historical data.

    While it is possible to continue to do this by using the GetAll API method in celestia-node on historic blocks as long as archival nodes exist on the public Celestia network, rollup developers should not rely on this as the only method to access historical data, as archival nodes serving requests for historical data for free is not guaranteed. Below are some other suggested methods to access historical data.

    • Use professional archival node or data providers. It is expected that professional infrastructure providers will provide paid access to archival nodes, where historical data can be retrieved, for example using the GetAll API method. This provides better guarantees than solely relying on free archival nodes on the public Celestia network.
    • Share snapshots of rollup nodes. Rollups could share snapshots of their data directories which can be downloaded manually by users bootstrapping new nodes. These snapshots could contain the latest state of the rollup, and/or all the historical blocks.
    • Add peer-to-peer support for historical block sync. A less manual version of sharing snapshots, where rollup nodes could implement built-in support for block sync, where rollup nodes download historical block data from each other over a peer-to-peer network.
      • Namespace pinning. In the future, celestia-node is expected to allow nodes to choose to "pin" data from selected namespaces that they wish to store and make available for other nodes. This will allow rollup nodes to be responsible for storing their data, without needing to implement their own peer-to-peer historical block sync mechanism.
    + + + + \ No newline at end of file diff --git a/learn/staking-governance-supply.html b/learn/staking-governance-supply.html new file mode 100644 index 00000000000..25a6f69e433 --- /dev/null +++ b/learn/staking-governance-supply.html @@ -0,0 +1,45 @@ + + + + + + Staking, governance, & supply | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Staking, governance, & supply

    Proof-of-stake on Celestia

    Celestia is a proof-of-stake blockchain based on CometBFT and the Cosmos SDK. Celestia supports in-protocol delegation and will start with an initial validator set of 100.

    Staking TIA as a validator or delegator enables you to earn staking rewards from the network. Validators charge a fee to delegators which gives them a percentage of staking rewards.

    Learn how proof of stake works on Cosmos SDK chains like Celestia.

    Consensus mechanismProof-of-stake
    Blockchain frameworkCosmos SDK
    Validator set size100
    Delegation supportYes

    Learn how to stake on your own at the community dashboards.

    Inflation

    TIA inflation starts at 8% annually and decreases by 10% every year until it reaches the long term issuance rate of 1.5%. Exact annual inflation rates can be found in the diagram below.

    inflation diagram

    The annual provisions for inflation are calculated based on the total supply of TIA at the beginning of each year. To calculate how many TIA to issue per block, Celestia uses the block timestamp rather than the block height since the time between blocks can vary and cause actual issuance to be higher than the target.

    For an in-depth understanding, refer to ADR019.

    Decentralised governance

    Network parameters

    TIA holders (not just stakers) can propose and vote on governance proposals to change a subset of network parameters. To learn more, see a complete list of both the changeable and non-changeable parameters and their values. Additionally, learn how to submit and vote on governance proposals.

    Community pool

    Starting at genesis, Celestia’s community pool receives 2% of all Celestia block rewards. TIA stakers may vote to fund ecosystem initiatives as in many other Cosmos SDK chains.

    Learn how to submit a governance proposal to spend community pool funds.

    TIA allocation at genesis

    Celestia will have a total supply of 1,000,000,000 TIA at genesis, split across five categories described in the chart and table below.

    allocation diagram

    CategoryDescription%
    Public AllocationGenesis Drop and Incentivized Testnet: 7.4%
    Future initiatives: 12.6%
    20%
    R&D & EcosystemTokens allocated to the Celestia Foundation and core devs for research, development, and ecosystem initiatives including:
    - Protocol maintenance and development
    - Programs for rollup developers, infrastructure, and node operators
    26.8%
    Early Backers: Series A&BEarly supporters of Celestia19.7%
    Early Backers: SeedEarly supporters of Celestia15.9%
    Initial Core ContributorsMembers of Celestia Labs, the first core contributor to Celestia17.6%

    Unlocks

    Celestia’s 1 billion TIA supply at genesis will be subject to several different unlock schedules. All tokens, locked or unlocked, may be staked, but staking rewards are unlocked upon receipt and will add to the circulating supply.

    Circulating supply is defined as the amount of TIA tokens in general circulation without onchain transfer restrictions.

    Available supply is defined as the amount of TIA tokens that are either part of the circulating supply or are unlocked but subject to some form of governance to determine when the tokens are allocated. This includes the unlocked portion of the R&D & Ecosystem tokens and the tokens set aside for future initiatives.

    The definitions for circulating and available supply were adapted from Optimism’s definitions.

    supply diagram

    Unlock schedule by category is described in the table below.

    CategoryUnlock Schedule
    Public AllocationFully unlocked at launch.
    R&D & Ecosystem25% unlocked at launch.
    Remaining 75% unlocks continuously from year 1 to year 4.
    Initial Core Contributors33% unlocked at year 1.
    Remaining 67% unlocks continuously from year 1 to year 3.
    Early Backers: Seed33% unlocked at year 1.
    Remaining 67% unlocks continuously from year 1 to year 2.
    Early Backers: Series A&B33% unlocked at year 1.
    Remaining 67% unlocks continuously from year 1 to year 2.
    + + + + \ No newline at end of file diff --git a/learn/staking.html b/learn/staking.html new file mode 100644 index 00000000000..af1e3e363ad --- /dev/null +++ b/learn/staking.html @@ -0,0 +1,45 @@ + + + + + + Staking on Celestia | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Staking on Celestia

    Engage with the Celestia network at a deeper level through staking. An essential mechanism to a proof-of-stake network, users can secure the network by delegating to a validator and receive a share of its staking rewards.

    Mainnet Beta

    Currently, the following staking interfaces exist for the Mainnet Beta.

    Just connect your wallet to get started!

    Mocha testnet

    Currently, the following staking interfaces exist for the Mocha testnet.

    Just connect your wallet to get started!

    + + + + \ No newline at end of file diff --git a/learn/tia.html b/learn/tia.html new file mode 100644 index 00000000000..915a1f6a3eb --- /dev/null +++ b/learn/tia.html @@ -0,0 +1,45 @@ + + + + + + Overview of TIA | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Overview of TIA

    TIA at a glance

    PropertyDetails
    AbbreviationTIA
    Total supply at genesis1,000,000,000 TIA
    Inflation schedule8% in the first year, decreasing 10% per year until reaching an inflation floor of 1.5% annually
    Decimals6
    Conversion1 uTIA=TIA×106

    Role of TIA

    Paying for blobspace

    Celestia’s native asset, TIA, is an essential part of how developers build on the first modular blockchain network. To use Celestia for data availability, rollup developers submit PayForBlobs transactions on the network for a fee, denominated in TIA.

    Bootstrapping new rollups

    A core part of the Celestia vision is that deploying a blockchain should be as easy as deploying a smart contract. In the modular era, developers no longer need to issue a token to launch their own blockchain.

    Similarly to ETH on Ethereum-based rollups, developers may opt to bootstrap their chain quickly by using TIA as a gas token and currency, in addition to paying for data availability. In this mode, developers can focus on creating their application or execution layer, instead of issuing a token right away.

    Proof-of-stake

    As a permissionless network built with Cosmos SDK, Celestia uses proof-of-stake to secure its own consensus. Like in other Cosmos networks, any user can help secure the network by delegating their TIA to a Celestia validator for a portion of their validator’s staking rewards.

    Learn how proof-of-stake works in Cosmos.

    Decentralised governance

    TIA staking also allows the community to play a critical role in decentralised governance over key parts of Celestia, such as voting on network parameters through governance proposals, and governing the community pool, which receives 2% of block rewards.

    Learn more about Celestia’s decentralised governance model.

    Denominations

    TIA: display token

    TIA is the DisplayDenom that you will typically see in wallets and user interfaces.

    utia: staking denomination

    utia is the BondDenom and stands for "micro TIA", with 1 TIA = 1,000,000 utia. This is the native staking denomination.

    In staking operations or transactions, if no denomination is specified, utia is assumed.

    microtia: staking denomination alias

    microtia is the BondDenomAlias, an alias for utia.

    + + + + \ No newline at end of file diff --git a/logo-dark.svg b/logo-dark.svg new file mode 100644 index 00000000000..1f3978ecb38 --- /dev/null +++ b/logo-dark.svg @@ -0,0 +1,95 @@ + + + + + + + + + + + + + diff --git a/logo-light.svg b/logo-light.svg new file mode 100644 index 00000000000..0076b1bd6f1 --- /dev/null +++ b/logo-light.svg @@ -0,0 +1,95 @@ + + + + + + + + + + + + + diff --git a/modular.svg b/modular.svg new file mode 100644 index 00000000000..521e284ea17 --- /dev/null +++ b/modular.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/nodes/arabica-devnet.html b/nodes/arabica-devnet.html new file mode 100644 index 00000000000..8b0154b1521 --- /dev/null +++ b/nodes/arabica-devnet.html @@ -0,0 +1,52 @@ + + + + + + Arabica devnet | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Arabica devnet

    arabica-devnet

    Arabica devnet is a testnet from Celestia Labs that is focused exclusively on providing developers with enhanced performance and the latest upgrades for testing their rollups and applications.

    Arabica does not focus on validator or consensus-level testing, rather, that is what Mocha testnet is used for. If you are a validator, we recommend testing your validator operations on the Mocha testnet.

    Network stability and upgrades

    Arabica has the latest updates from all Celestia's products deployed on it, it can be subject to many changes. Therefore, as a fair warning, Arabica can break unexpectedly, but given it will be continuously updated, it is a useful way to keep testing the latest changes in the software.

    Developers can still deploy on Mocha testnet their sovereign rollups if they chose to do so, it just will always lag behind Arabica devnet until Mocha undergoes Hardfork Upgrades in coordination with Validators.

    Network details

    DetailValue
    Chain IDarabica-11
    Genesis hash27122593765E07329BC348E8D16E92DCB4C75B34CCCB35C640FD7A4484D4C711
    Genesis file https://github.com/celestiaorg/networks/blob/master/arabica-11/genesis.json
    Peers file https://github.com/celestiaorg/networks/blob/master/arabica-11/peers.txt
    Validators 4

    Software version numbers

    SoftwareVersion
    celestia-nodev0.13.3
    celestia-appv1.7.0

    Integrations

    This guide contains the relevant sections for how to connect to Arabica devnet, depending on the type of node you are running. Your best approach to participating is to first determine which node you would like to run. Each node’s guide will link to the relevant network in order to show you how to connect to them. Learn about the different endpoint types in the Cosmos SDK documentation.

    RPC endpoints

    RPC endpoints and types of nodes you can run in order to participate in Arabica devnet:

    Node typeEndpoint typeEndpoint
    Consensus nodes (full)Consensus RPChttps://rpc.celestia-arabica-11.com
    APIhttps://api.celestia-arabica-11.com
    gRPCgrpc.celestia-arabica-11.com:443
    Direct endpoints with open portsOpen ports: 26656 (p2p), 26657 (RPC), 1317 (API), 9090 (GRPC)
    validator-1.celestia-arabica-11.com
    validator-2.celestia-arabica-11.com
    validator-3.celestia-arabica-11.com
    validator-4.celestia-arabica-11.com
    Data availability nodesDA Bridge Node Endpoints/dns4/da-bridge-1.celestia-arabica-11.com/tcp/2121/p2p/12D3KooWGqwzdEqM54Dce6LXzfFr97Bnhvm6rN7KM7MFwdomfm4S
    (light, bridge, full)/dns4/da-bridge-2.celestia-arabica-11.com/tcp/2121/p2p/12D3KooWCMGM5eZWVfCN9ZLAViGfLUWAfXP5pCm78NFKb9jpBtua
    /dns4/da-bridge-3.celestia-arabica-11.com/tcp/2121/p2p/12D3KooWEWuqrjULANpukDFGVoHW3RoeUU53Ec9t9v5cwW3MkVdQ
    /dns4/da-bridge-4.celestia-arabica-11.com/tcp/2121/p2p/12D3KooWLT1ysSrD7XWSBjh7tU1HQanF5M64dHV6AuM6cYEJxMPk
    --core.ip string endpointsRefer to "Direct endpoints with open ports" above

    You can find the status of these endpoints.

    Using consensus endpoints with DA nodes

    Consensus RPC endpoints are used to provide DA nodes with state access for querying the chain’s state and broadcasting transactions (balances, blobs, etc.) to the Celestia network.

    Developers will need to provide a –core.ip string from a consensus node’s URI or an IP that populates 2 ports for 2 types (RPC and gRPC, at ports 26657 and 9090, respectively) to their respective DA node.

    EXAMPLE

    bash
    celestia <da_type> start –core.ip <url> –core.rpc.port <port> \
    +    –core.grpc.port <port> \
    celestia <da_type> start –core.ip <url> –core.rpc.port <port> \
    +    –core.grpc.port <port> \

    RPCs for DA nodes to initialise or start your celestia-node to Arabica devnet with can be found in the table in the "Direct endpoints with open ports" section above.

    As an example, this command will work to start a light node with state access, using default ports:

    bash
    celestia light start --p2p.network arabica \
    +  --core.ip validator-1.celestia-arabica-11.com
    celestia light start --p2p.network arabica \
    +  --core.ip validator-1.celestia-arabica-11.com

    Bridge node runners

    Not all of the RPC endpoints do not guarantee the full block history. Find an archive endpoint on the community dashboard or run your own full consensus node with no pruning for your bridge node.

    Arabica devnet faucet

    WARNING

    USING THIS FAUCET DOES NOT ENTITLE YOU TO ANY AIRDROP OR OTHER DISTRIBUTION OF MAINNET CELESTIA TOKENS. THERE ARE NO PUBLIC SALES OF ANY MAINNET CELESTIA TOKENS.

    Discord

    You can request from Arabica devnet Faucet on the #arabica-faucet channel on Celestia's Discord server with the following command:

    text
    $request <CELESTIA-ADDRESS>
    $request <CELESTIA-ADDRESS>

    Where <CELESTIA-ADDRESS> is a celestia1****** generated address.

    NOTE

    Faucet has a limit of 10 tokens per week per address/Discord ID.

    Web

    The web faucet is available at https://faucet.celestia-arabica-11.com/.

    Explorers

    There are multiple explorers you can use for Arabica:

    Network upgrades

    Join our Telegram announcement channel for network upgrades.

    + + + + \ No newline at end of file diff --git a/nodes/bridge-node.html b/nodes/bridge-node.html new file mode 100644 index 00000000000..0d61c039222 --- /dev/null +++ b/nodes/bridge-node.html @@ -0,0 +1,51 @@ + + + + + + Setting up a Celestia bridge node | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Setting up a Celestia bridge node

    This tutorial will go over the steps to setting up your Celestia bridge node.

    Bridge nodes connect the data availability layer and the consensus layer.

    Overview of bridge nodes

    A Celestia bridge node has the following properties:

    1. Import and process “raw” headers & blocks from a trusted core process (meaning a trusted RPC connection to a celestia-core node) in the Consensus network. Bridge nodes can run this core process internally (embedded) or simply connect to a remote endpoint. Bridge nodes also have the option of being an active validator in the consensus network.
    2. Validate and erasure code the “raw” blocks
    3. Supply block shares with data availability headers to light nodes in the DA network.

    bridge-node-diagram

    From an implementation perspective, Bridge nodes run two separate processes:

    1. celestia-app with celestia-core (see repo)

      • celestia-app is the state machine where the application and the proof-of-stake logic is run. celestia-app is built on Cosmos SDK and also encompasses celestia-core.
      • celestia-core is the state interaction, consensus and block production layer. celestia-core is built on Tendermint Core, modified to store data roots of erasure coded blocks among other changes (see ADRs).
    2. celestia-node (see repo)

      • celestia-node augments the above with a separate libp2p network that serves data availability sampling requests. The team sometimes refers to this as the “halo” network.

    Hardware requirements

    The following hardware minimum requirements are recommended for running the bridge node:

    • Memory: 4 GB RAM (minimum)
    • CPU: 6 cores
    • Disk: 10 TB SSD Storage
    • Bandwidth: 1 Gbps for Download/1 Gbps for Upload

    Setting up your bridge node

    The following tutorial is done on an Ubuntu Linux 20.04 (LTS) x64 instance machine.

    Setup the dependencies

    Follow the tutorial for installing the dependencies.

    Deploy the Celestia bridge node

    Install Celestia Node

    Install the celestia-node binary, which will be used to run the bridge node.

    Follow the tutorial for installing celestia-node.

    Initialize the bridge node

    Run the following:

    sh
    celestia bridge init --core.ip <URI>
    celestia bridge init --core.ip <URI>

    The --core.ip gRPC port defaults to 9090, so if you do not specify it in the command line, it will default to that port. You can add the port after the IP address or use the --core.grpc.port flag to specify another port if you prefer.

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    Using an RPC of your own, or one from the list on the Mocha testnet page or list on the Arabica devnet page, start your node.

    Here is an example of initializing the bridge node:

    sh
    celestia bridge init --core.ip <URI>
    celestia bridge init --core.ip <URI>
    sh
    celestia bridge init --core.ip <URI> --p2p.network mocha
    celestia bridge init --core.ip <URI> --p2p.network mocha
    sh
    celestia bridge init --core.ip <URI> --p2p.network arabica
    celestia bridge init --core.ip <URI> --p2p.network arabica

    Run the bridge node

    Start the bridge node with a connection to a validator node's gRPC endpoint (which is usually exposed on port 9090):

    sh
    celestia bridge start --core.ip <URI>
    celestia bridge start --core.ip <URI>

    Here is an example of starting the bridge node on Mocha:

    sh
    celestia bridge start --core.ip rpc-mocha.pops.one:26657 --p2p.network mocha
    celestia bridge start --core.ip rpc-mocha.pops.one:26657 --p2p.network mocha

    And on Arabica:

    sh
    celestia bridge start --core.ip validator-1.celestia-arabica-11.com \
    +  --p2p.network arabica
    celestia bridge start --core.ip validator-1.celestia-arabica-11.com \
    +  --p2p.network arabica

    You can create your key for your node by following the cel-key instructions.

    Once you start the bridge node, a wallet key will be generated for you. You will need to fund that address with Testnet tokens to pay for PayForBlob transactions. You can find the address by running the following command:

    sh
    ./cel-key list --node.type bridge --keyring-backend test --p2p.network <network>
    ./cel-key list --node.type bridge --keyring-backend test --p2p.network <network>

    TIP

    You do not need to declare a network for Mainnet Beta. Refer to the chain ID section on the troubleshooting page for more information

    You can get testnet tokens from:

    NOTE

    If you are running a bridge node for your validator it is highly recommended to request Mocha testnet tokens as this is the testnet used to test out validator operations.

    Optional: run the bridge node with a custom key

    In order to run a bridge node using a custom key:

    1. The custom key must exist inside the celestia bridge node directory at the correct path (default: ~/.celestia-bridge/keys/keyring-test)
    2. The name of the custom key must be passed upon start, like so:
    sh
    celestia bridge start --core.ip <URI> --keyring.accname <name-of-custom-key>
    celestia bridge start --core.ip <URI> --keyring.accname <name-of-custom-key>
    sh
    celestia bridge start --core.ip <URI> --keyring.accname <name-of-custom-key> \
    +  --p2p.network mocha
    celestia bridge start --core.ip <URI> --keyring.accname <name-of-custom-key> \
    +  --p2p.network mocha
    sh
    celestia bridge start --core.ip <URI> --keyring.accname <name-of-custom-key> \
    +  --p2p.network arabica
    celestia bridge start --core.ip <URI> --keyring.accname <name-of-custom-key> \
    +  --p2p.network arabica

    Optional: start the bridge node with SystemD

    Follow the tutorial on setting up the bridge node as a background process with SystemD.

    You have successfully set up a bridge node that is syncing with the network.

    + + + + \ No newline at end of file diff --git a/nodes/celestia-app-commands.html b/nodes/celestia-app-commands.html new file mode 100644 index 00000000000..404b9dac545 --- /dev/null +++ b/nodes/celestia-app-commands.html @@ -0,0 +1,257 @@ + + + + + + Helpful CLI commands | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Helpful CLI commands

    View all options:

    console
    $ celestia-appd --help
    +Start celestia-app
    +
    +Usage:
    +  celestia-appd [command]
    +
    +Available Commands:
    +  add-genesis-account Add a genesis account to genesis.json
    +  collect-gentxs      Collect genesis txs and output a genesis.json file
    +  config              Create or query an application CLI configuration file
    +  debug               Tool for helping with debugging your application
    +  export              Export state to JSON
    +  gentx               Generate a genesis tx carrying a self delegation
    +  help                Help about any command
    +  init                Initialize private validator, p2p, genesis,
    +  and application configuration files
    +  keys                Manage your application's keys
    +  migrate             Migrate genesis to a specified target version
    +  query               Querying subcommands
    +  rollback            rollback tendermint state by one height
    +  rollback            rollback cosmos-sdk and tendermint state by one height
    +  start               Run the full node
    +  status              Query remote node for status
    +  tendermint          Tendermint subcommands
    +  tx                  Transactions subcommands
    +  validate-genesis    validates the genesis file at the default
    +  location or at the location passed as an arg
    +  version             Print the application binary version information
    $ celestia-appd --help
    +Start celestia-app
    +
    +Usage:
    +  celestia-appd [command]
    +
    +Available Commands:
    +  add-genesis-account Add a genesis account to genesis.json
    +  collect-gentxs      Collect genesis txs and output a genesis.json file
    +  config              Create or query an application CLI configuration file
    +  debug               Tool for helping with debugging your application
    +  export              Export state to JSON
    +  gentx               Generate a genesis tx carrying a self delegation
    +  help                Help about any command
    +  init                Initialize private validator, p2p, genesis,
    +  and application configuration files
    +  keys                Manage your application's keys
    +  migrate             Migrate genesis to a specified target version
    +  query               Querying subcommands
    +  rollback            rollback tendermint state by one height
    +  rollback            rollback cosmos-sdk and tendermint state by one height
    +  start               Run the full node
    +  status              Query remote node for status
    +  tendermint          Tendermint subcommands
    +  tx                  Transactions subcommands
    +  validate-genesis    validates the genesis file at the default
    +  location or at the location passed as an arg
    +  version             Print the application binary version information

    Creating a wallet

    sh
    celestia-appd config keyring-backend test
    celestia-appd config keyring-backend test

    keyring-backend configures the keyring's backend, where the keys are stored.

    Options are: os|file|kwallet|pass|test|memory.

    You can learn more on the Cosmos documentation or Go Package documentation.

    Key management

    sh
    # listing keys
    +celestia-appd keys list
    +
    +# adding keys
    +celestia-appd keys add <KEY_NAME>
    +
    +# deleting keys
    +celestia-appd keys delete <KEY_NAME>
    +
    +# renaming keys
    +celestia-appd keys rename <CURRENT_KEY_NAME> <NEW_KEY_NAME>
    # listing keys
    +celestia-appd keys list
    +
    +# adding keys
    +celestia-appd keys add <KEY_NAME>
    +
    +# deleting keys
    +celestia-appd keys delete <KEY_NAME>
    +
    +# renaming keys
    +celestia-appd keys rename <CURRENT_KEY_NAME> <NEW_KEY_NAME>

    Importing and exporting keys

    Import an encrypted and ASCII-armored private key into the local keybase.

    sh
    celestia-appd keys import <KEY_NAME> <KEY_FILE>
    celestia-appd keys import <KEY_NAME> <KEY_FILE>

    Example usage:

    sh
    celestia-appd keys import amanda ./keyfile.txt
    celestia-appd keys import amanda ./keyfile.txt

    Export a private key from the local keyring in encrypted and ASCII-armored format:

    sh
    celestia-appd keys export <KEY_NAME>
    +
    +# you will then be prompted to set a password for the encrypted private key:
    +Enter passphrase to encrypt the exported key:
    celestia-appd keys export <KEY_NAME>
    +
    +# you will then be prompted to set a password for the encrypted private key:
    +Enter passphrase to encrypt the exported key:

    After you set a password, your encrypted key will be displayed.

    Querying subcommands

    Usage:

    sh
    celestia-appd query <FLAGS> | <COMMAND>
    +
    +# alias q
    +celestia-appd q <FLAGS> | <COMMAND>
    celestia-appd query <FLAGS> | <COMMAND>
    +
    +# alias q
    +celestia-appd q <FLAGS> | <COMMAND>

    To see all options:

    sh
    celestia-appd q --help
    celestia-appd q --help

    Token management

    Get token balances:

    sh
    celestia-appd q bank balances <ADDRESS> --node <NODE_URI>
    celestia-appd q bank balances <ADDRESS> --node <NODE_URI>

    Example usage:

    sh
    celestia-appd q bank balances celestia1czpgn3hdh9sodm06d5qk23xzgpq2uyc8ggdqgw \
    +    --node https://rpc-mocha.pops.one
    celestia-appd q bank balances celestia1czpgn3hdh9sodm06d5qk23xzgpq2uyc8ggdqgw \
    +    --node https://rpc-mocha.pops.one

    Transfer tokens from one wallet to another:

    sh
    celestia-appd tx bank send <FROM_ADDRESS> <TO_ADDRESS> \
    +    <amount> --node <NODE_URI> --chain-id <CHAIN_ID>
    celestia-appd tx bank send <FROM_ADDRESS> <TO_ADDRESS> \
    +    <amount> --node <NODE_URI> --chain-id <CHAIN_ID>

    Example usage:

    sh
    celestia-appd tx bank send <FROM_ADDRESS> <TO_ADDRESS> \
    +    19000000utia --node https://rpc-mocha.pops.one/ --chain-id mocha
    celestia-appd tx bank send <FROM_ADDRESS> <TO_ADDRESS> \
    +    19000000utia --node https://rpc-mocha.pops.one/ --chain-id mocha

    To see options:

    sh
    celestia-appd tx bank send --help
    celestia-appd tx bank send --help

    Governance

    Governance proposals on Celestia are limited as there are no text proposals, upgrades occur via social consensus, and some params are not modifiable. However, one can submit governance proposals to change certain parameters and spend community funds. More detailed information on this topic can be found in the cosmos-sdk documentation for submitting proposals, the list of parameter defaults in the specs, and the x/paramfilter module specs.

    Viewing the available proposals can be done with the query command:

    sh
    celestia-appd q gov proposals
    celestia-appd q gov proposals

    There are four options when voting "yes", "no", "no_with_veto" and "abstain". The "no_with_veto" vote is different from the "no" vote in that the submitter of the proposer's deposit will get burned, and a minority of stake (1/3) can stop a proposal that might otherwise pass quorum. You can use those options to vote on a governance proposal with the following command:

    sh
    celestia-appd tx gov vote <proposal id> <option> \
    +    --from <wallet> --chain-id <chain-id>
    celestia-appd tx gov vote <proposal id> <option> \
    +    --from <wallet> --chain-id <chain-id>

    To submit a proposal, there are two commands that can be used. The first is the legacy command, which is the recommended way to submit a proposal.

    To change the max validators to 105, one would first save this JSON file:

    json
    {
    +  "title": "Staking Param Change",
    +  "description": "Update max validators",
    +  "changes": [
    +    {
    +      "subspace": "staking",
    +      "key": "MaxValidators",
    +      "value": 105
    +    }
    +  ],
    +  "deposit": "1000000000utia"
    +}
    {
    +  "title": "Staking Param Change",
    +  "description": "Update max validators",
    +  "changes": [
    +    {
    +      "subspace": "staking",
    +      "key": "MaxValidators",
    +      "value": 105
    +    }
    +  ],
    +  "deposit": "1000000000utia"
    +}

    Then you can submit the proposal with:

    sh
    celestia-appd tx gov submit-legacy-proposal \
    +    parameter-change <path to json file> \
    +    --from <wallet> --chain-id <chain-id>
    celestia-appd tx gov submit-legacy-proposal \
    +    parameter-change <path to json file> \
    +    --from <wallet> --chain-id <chain-id>

    If we want to use the newer api, we can submit a proposal by first saving the sdk.Msg proposal in the json encoded format to a json.

    json
    {
    +  "messages": [
    +    {
    +      "@type": "/cosmos.gov.v1beta1.MsgSubmitProposal",
    +      "content": {
    +        "@type": "/cosmos.params.v1beta1.ParameterChangeProposal",
    +        "title": "title",
    +        "description": "description",
    +        "changes": [
    +          { "subspace": "staking", "key": "MaxValidators", "value": "103" }
    +        ]
    +      },
    +      "initial_deposit": [{ "denom": "utia", "amount": "1000000000" }],
    +      "proposer": "celestia10d07y265gmmuvt4z0w9aw880jnsr700jtgz4v7"
    +    }
    +  ]
    +}
    {
    +  "messages": [
    +    {
    +      "@type": "/cosmos.gov.v1beta1.MsgSubmitProposal",
    +      "content": {
    +        "@type": "/cosmos.params.v1beta1.ParameterChangeProposal",
    +        "title": "title",
    +        "description": "description",
    +        "changes": [
    +          { "subspace": "staking", "key": "MaxValidators", "value": "103" }
    +        ]
    +      },
    +      "initial_deposit": [{ "denom": "utia", "amount": "1000000000" }],
    +      "proposer": "celestia10d07y265gmmuvt4z0w9aw880jnsr700jtgz4v7"
    +    }
    +  ]
    +}

    Note that the proposer here must be the gov module account. That account can be found by using this command:

    sh
    celestia-appd q auth module-account gov
    celestia-appd q auth module-account gov

    Then one can submit the proposal with:

    sh
    celestia-appd tx gov submit-proposal <path to json file> \
    +    --from <wallet> --chain-id <chain-id>
    celestia-appd tx gov submit-proposal <path to json file> \
    +    --from <wallet> --chain-id <chain-id>

    Community Pool

    A percentage of the block rewards are allocated to the community pool. Community members can submit governance proposals to spend the community pool funds, and token holders can vote on these proposals. The proposals to spend are arbitrary in nature in that they can only contain text and some address to send funds to. To view the community pool balance, use the following command:

    sh
    celestia-appd q distribution community-pool
    celestia-appd q distribution community-pool

    To submit a proposal to spend the community pool funds, first create a JSON file that contains the proposal.

    json
    {
    +  "title": "Community Pool Spend",
    +  "description": "Fund an open source project.",
    +  "recipient": "celestia17adsjkuecgjheugrdrwdqv9uh3qkrfmj9xzawx",
    +  "amount": "100000000000utia",
    +  "deposit": "1000000000utia"
    +}
    {
    +  "title": "Community Pool Spend",
    +  "description": "Fund an open source project.",
    +  "recipient": "celestia17adsjkuecgjheugrdrwdqv9uh3qkrfmj9xzawx",
    +  "amount": "100000000000utia",
    +  "deposit": "1000000000utia"
    +}

    The json file can be submitted using a similar proposal submission command as above:

    sh
    celestia-appd tx gov submit-legacy-proposal \
    +    community-pool-spend <path to json file> \
    +    --from <wallet>
    celestia-appd tx gov submit-legacy-proposal \
    +    community-pool-spend <path to json file> \
    +    --from <wallet>

    Claim validator rewards

    You can claim your validator rewards with the following command:

    sh
    celestia-appd tx distribution withdraw-rewards <validator valoper> \
    +    --commission --from=<validator wallet> --chain-id <chain-id> \
    +    --gas auto -y
    celestia-appd tx distribution withdraw-rewards <validator valoper> \
    +    --commission --from=<validator wallet> --chain-id <chain-id> \
    +    --gas auto -y

    Delegate & undelegate tokens

    You can delegate your tokens to a validator with the following command:

    sh
    celestia-appd tx staking delegate <validator valoper> <amount>\
    +    --from <wallet> --chain-id <chain-id>
    celestia-appd tx staking delegate <validator valoper> <amount>\
    +    --from <wallet> --chain-id <chain-id>

    You can undelegate tokens to a validator with the unbond command:

    sh
    celestia-appd tx staking unbond <validator valoper> <amount>\
    +    --from <wallet> --chain-id <chain-id>
    celestia-appd tx staking unbond <validator valoper> <amount>\
    +    --from <wallet> --chain-id <chain-id>

    Unjailing the validator

    You can unjail your validator with the following command:

    sh
    celestia-appd tx slashing unjail --from <validator wallet>\
    +    --chain-id <chain-id> --gas auto -y
    celestia-appd tx slashing unjail --from <validator wallet>\
    +    --chain-id <chain-id> --gas auto -y

    How to export logs with SystemD

    You can export your logs if you are running a SystemD service with the following command:

    sh
    sudo journalctl -u <your systemd service> -S yesterday > node_logs.txt
    +sudo journalctl -u <your systemd service> -S today > node_logs.txt
    +# This command outputs the last 1 million lines!
    +sudo journalctl -u <your systemd service> -n 1000000 > node_logs.txt
    sudo journalctl -u <your systemd service> -S yesterday > node_logs.txt
    +sudo journalctl -u <your systemd service> -S today > node_logs.txt
    +# This command outputs the last 1 million lines!
    +sudo journalctl -u <your systemd service> -n 1000000 > node_logs.txt

    Signing genesis for a new network

    You can first run the following commands:

    sh
    VALIDATOR_NAME=validator1
    +CHAIN_ID=testnet
    +celestia-appd init $VALIDATOR_NAME --chain-id $CHAIN_ID
    +MONIKER=validator_name
    VALIDATOR_NAME=validator1
    +CHAIN_ID=testnet
    +celestia-appd init $VALIDATOR_NAME --chain-id $CHAIN_ID
    +MONIKER=validator_name

    Next create a wallet:

    sh
    KEY_NAME=validator
    +celestia-appd keys add $KEY_NAME
    KEY_NAME=validator
    +celestia-appd keys add $KEY_NAME

    Create or assign an EVM address:

    sh
    EVM_ADDRESS=<EVM_ADDRESS>
    EVM_ADDRESS=<EVM_ADDRESS>

    Then add genesis account:

    sh
    CELES_AMOUNT="5000100000000utia"
    +celestia-appd add-genesis-account $KEY_NAME $CELES_AMOUNT
    CELES_AMOUNT="5000100000000utia"
    +celestia-appd add-genesis-account $KEY_NAME $CELES_AMOUNT

    Then generate your gentx:

    sh
    STAKING_AMOUNT=5000000000000utia
    +celestia-appd gentx $KEY_NAME $STAKING_AMOUNT --chain-id $CHAIN_ID \
    +    --pubkey=$(celestia-appd tendermint show-validator) \
    +    --moniker=$MONIKER \
    +    --commission-rate=0.1 \
    +    --commission-max-rate=0.2 \
    +    --commission-max-change-rate=0.01 \
    +    --min-self-delegation=1 \
    +    --evm-address=$EVM_ADDRESS \
    STAKING_AMOUNT=5000000000000utia
    +celestia-appd gentx $KEY_NAME $STAKING_AMOUNT --chain-id $CHAIN_ID \
    +    --pubkey=$(celestia-appd tendermint show-validator) \
    +    --moniker=$MONIKER \
    +    --commission-rate=0.1 \
    +    --commission-max-rate=0.2 \
    +    --commission-max-change-rate=0.01 \
    +    --min-self-delegation=1 \
    +    --evm-address=$EVM_ADDRESS \

    You can then share your gentx JSON file on the networks repo in the respective network directory you are participating in.

    + + + + \ No newline at end of file diff --git a/nodes/celestia-app-metrics.html b/nodes/celestia-app-metrics.html new file mode 100644 index 00000000000..88d0bbae0d4 --- /dev/null +++ b/nodes/celestia-app-metrics.html @@ -0,0 +1,127 @@ + + + + + + Metrics | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Metrics

    Metrics are a powerful tool for monitoring the health and performance of a system. Celestia provides support for metrics to make sure, as an operator, your system continues to remain up and running. Metrics can also provide critical insight into how Celestia is used and how it can be improved.

    Setup

    Celestia uses Prometheus to publish metrics. It can be enabled through the config.toml file.

    bash
    #######################################################
    +###       Instrumentation Configuration Options     ###
    +#######################################################
    +[instrumentation]
    +
    +# When true, Prometheus metrics are served under /metrics on
    +# PrometheusListenAddr.
    +# Check out the documentation for the list of available metrics.
    +prometheus = true
    +
    +# Address to listen for Prometheus collector(s) connections
    +prometheus_listen_addr = ":26660"
    +
    +# Maximum number of simultaneous connections.
    +# If you want to accept a larger number than the default, make sure
    +# you increase your OS limits.
    +# 0 - unlimited.
    +max_open_connections = 3
    +
    +# Instrumentation namespace
    +namespace = "celestia"
    #######################################################
    +###       Instrumentation Configuration Options     ###
    +#######################################################
    +[instrumentation]
    +
    +# When true, Prometheus metrics are served under /metrics on
    +# PrometheusListenAddr.
    +# Check out the documentation for the list of available metrics.
    +prometheus = true
    +
    +# Address to listen for Prometheus collector(s) connections
    +prometheus_listen_addr = ":26660"
    +
    +# Maximum number of simultaneous connections.
    +# If you want to accept a larger number than the default, make sure
    +# you increase your OS limits.
    +# 0 - unlimited.
    +max_open_connections = 3
    +
    +# Instrumentation namespace
    +namespace = "celestia"

    If you restart your node, you can check to see it's working by running:

    bash
    curl localhost:26660/metrics
    curl localhost:26660/metrics

    Visualization

    Now your nodes are publishing metrics, we need something to scrape it and a visualizer to create a dashboard. Commonly, Prometheus is paired with Grafana.

    First, you will need to install Prometheus either from their downloads page or through a package manager like brew.

    Next, create a config file $HOME/.celestia-app/config/prometheus.yml and fill out some basic settings as follows:

    yml
    global:
    +  scrape_interval: 15s # By default, scrape targets every 15 seconds.
    +
    +  # Attach these labels to any time series or alerts when communicating
    +  # with external systems (federation, remote storage, Alertmanager).
    +  external_labels:
    +    monitor: "codelab-monitor"
    +
    +# A scrape configuration containing exactly one endpoint to scrape:
    +# Here it's Prometheus itself.
    +scrape_configs:
    +  # The job name is added as a label `job=<job_name>` to any timeseries
    +  # scraped from this config.
    +  - job_name: "prometheus"
    +
    +    # Override the global default and scrape targets from this job every
    +    # 5 seconds.
    +    scrape_interval: 5s
    +
    +    static_configs:
    +      # Point to the same endpoint that Celestia is publishing on
    +      - targets: ["localhost:26660"]
    global:
    +  scrape_interval: 15s # By default, scrape targets every 15 seconds.
    +
    +  # Attach these labels to any time series or alerts when communicating
    +  # with external systems (federation, remote storage, Alertmanager).
    +  external_labels:
    +    monitor: "codelab-monitor"
    +
    +# A scrape configuration containing exactly one endpoint to scrape:
    +# Here it's Prometheus itself.
    +scrape_configs:
    +  # The job name is added as a label `job=<job_name>` to any timeseries
    +  # scraped from this config.
    +  - job_name: "prometheus"
    +
    +    # Override the global default and scrape targets from this job every
    +    # 5 seconds.
    +    scrape_interval: 5s
    +
    +    static_configs:
    +      # Point to the same endpoint that Celestia is publishing on
    +      - targets: ["localhost:26660"]

    Note, that Prometheus by default runs its server on :9090. If you are running this on the same machine as your consensus node, it will collide with gRPC which runs on the same port. To avoid this, either switch off gRPC (if it's not needed), change the gRPC port in app.toml, or run Prometheus on a different port e.g. --web.listen-address="0.0.0.0:8000"

    To run the prometheus server:

    bash
    prometheus --config.file="$HOME/.celestia-app/config/prometheus.yml"
    prometheus --config.file="$HOME/.celestia-app/config/prometheus.yml"

    A prometheus server can scrape metrics from multiple nodes at once; a good way of bringing together information from many machines to one place.

    To visualize the information, you can use Grafana: either with their cloud option or run the open source code yourself.

    Once setup, run:

    bash
    grafana server
    grafana server

    which will begin a server on localhost:3000. If you open the url on your browser you will see the Grafana login page. Use admin for both the user and password to log in.

    You will need to link the prometheus server as a data source. To do that go to "Configuration" in the sidebar and then "Data Sources". Add a new data source specifying the URL of the Prometheus instance (default at localhost:9090). Click "Save & test" to confirm.

    Lastly, you will need to setup a dashboard. You can choose to do this yourself, handpicking the metrics that are important or you can simply export an existing design. Fortunately the cosmos ecosystem has conjured a "Cosmos Dashboard". On the sidebar, click "Dashboards" and then "import". Enter the following dashboard ID: 11036 and then link it to the "Prometheus" data source you just set up. Finally click the "Import" button and the "Cosmos Dashboard" should appear.

    Node exporter

    Celestia's metrics include a plethora of application specific trackers but it's also important to keep an eye on system level metrics such as memory usage and disk space. This can be best achieved by running Node Exporter. Follow the guide in the link to get set up, adding the port number to the prometheus.yml file.

    Alerts

    The final cherry on the cake is to integrate your monitoring system with a mechanism for producing alerts to warn you if your node has crashed or is no longer able to stay at the head of the chain.

    Since we're already using Grafana, we can install the Grafana OnCall plugin. OnCall allows you to setup integrations. It could be a webhook or a direct integration into Telegram or Slack. You can find more information on Grafana's Docs Page.

    + + + + \ No newline at end of file diff --git a/nodes/celestia-app-multisig.html b/nodes/celestia-app-multisig.html new file mode 100644 index 00000000000..a7d5b0ac3f2 --- /dev/null +++ b/nodes/celestia-app-multisig.html @@ -0,0 +1,147 @@ + + + + + + Multisig | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Multisig

    Celestia inherits support for multisig accounts from the Cosmos SDK. Multisig accounts behave similarly to regular accounts with the added requirement that a threshold of signatures is needed to authorize a transaction.

    Multisig accounts can be created from the command line or using a graphical interface such as Keplr.

    Command line

    bash
    #!/bin/sh
    +
    +# Prerequisite: prior to running this script, start a single node devnet with ./scripts/single-node.sh
    +CHAIN_ID="private"
    +KEY_NAME="validator"
    +KEYRING_BACKEND="test"
    +BROADCAST_MODE="block"
    +
    +# Create 3 test keys
    +celestia-appd keys add test1
    +celestia-appd keys add test2
    +celestia-appd keys add test3
    +# Create the multisig account
    +celestia-appd keys add multisig \
    +    --multisig test1,test2,test3 \
    +    --multisig-threshold 2
    +
    +# Send some funds from the validator account to the multisig account
    +celestia-appd tx bank send $VALIDATOR $MULTISIG 100000utia \
    +    --from $VALIDATOR \
    +    --fees 1000utia \
    +    --chain-id $CHAIN_ID \
    +    --keyring-backend $KEYRING_BACKEND \
    +    --broadcast-mode $BROADCAST_MODE \
    +    --yes
    +
    +# Send some funds from the multisig account to the validator account.
    +# Note this transaction will need to be signed by at least 2 of the 3 test accounts.
    +celestia-appd tx bank send $MULTISIG $VALIDATOR 1utia \
    +    --from $MULTISIG \
    +    --fees 1000utia \
    +    --chain-id $CHAIN_ID \
    +    --keyring-backend $KEYRING_BACKEND \
    +    --generate-only > unsignedTx.json
    +
    +# Sign from test1 and test2
    +celestia-appd tx sign unsignedTx.json \
    +    --multisig $MULTISIG \
    +    --from test1 \
    +    --output-document test1sig.json \
    +    --chain-id $CHAIN_ID
    +celestia-appd tx sign unsignedTx.json \
    +    --multisig $MULTISIG \
    +    --from test2 \
    +    --output-document test2sig.json \
    +    --chain-id $CHAIN_ID
    +
    +# Generate the final signed transaction
    +celestia-appd tx multisign unsignedTx.json multisig \
    +    test1sig.json test2sig.json \
    +    --output-document signedTx.json \
    +    --chain-id $CHAIN_ID
    #!/bin/sh
    +
    +# Prerequisite: prior to running this script, start a single node devnet with ./scripts/single-node.sh
    +CHAIN_ID="private"
    +KEY_NAME="validator"
    +KEYRING_BACKEND="test"
    +BROADCAST_MODE="block"
    +
    +# Create 3 test keys
    +celestia-appd keys add test1
    +celestia-appd keys add test2
    +celestia-appd keys add test3
    +# Create the multisig account
    +celestia-appd keys add multisig \
    +    --multisig test1,test2,test3 \
    +    --multisig-threshold 2
    +
    +# Send some funds from the validator account to the multisig account
    +celestia-appd tx bank send $VALIDATOR $MULTISIG 100000utia \
    +    --from $VALIDATOR \
    +    --fees 1000utia \
    +    --chain-id $CHAIN_ID \
    +    --keyring-backend $KEYRING_BACKEND \
    +    --broadcast-mode $BROADCAST_MODE \
    +    --yes
    +
    +# Send some funds from the multisig account to the validator account.
    +# Note this transaction will need to be signed by at least 2 of the 3 test accounts.
    +celestia-appd tx bank send $MULTISIG $VALIDATOR 1utia \
    +    --from $MULTISIG \
    +    --fees 1000utia \
    +    --chain-id $CHAIN_ID \
    +    --keyring-backend $KEYRING_BACKEND \
    +    --generate-only > unsignedTx.json
    +
    +# Sign from test1 and test2
    +celestia-appd tx sign unsignedTx.json \
    +    --multisig $MULTISIG \
    +    --from test1 \
    +    --output-document test1sig.json \
    +    --chain-id $CHAIN_ID
    +celestia-appd tx sign unsignedTx.json \
    +    --multisig $MULTISIG \
    +    --from test2 \
    +    --output-document test2sig.json \
    +    --chain-id $CHAIN_ID
    +
    +# Generate the final signed transaction
    +celestia-appd tx multisign unsignedTx.json multisig \
    +    test1sig.json test2sig.json \
    +    --output-document signedTx.json \
    +    --chain-id $CHAIN_ID

    Resources

    + + + + \ No newline at end of file diff --git a/nodes/celestia-app-slashing.html b/nodes/celestia-app-slashing.html new file mode 100644 index 00000000000..a17675a76bf --- /dev/null +++ b/nodes/celestia-app-slashing.html @@ -0,0 +1,45 @@ + + + + + + Slashing on Celestia | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Slashing on Celestia

    Slashing is a mechanism employed in proof of stake blockchains that is used to deter and punish malicious behavior. It functions by removing a percentage of a validator's stake each time they act harmfully towards the network.

    Celestia is built with the Cosmos SDK and uses the x/slashing module.

    If a validator gets slashed, delegators bonded to that validator will also have the same percentage of their delegated funds slashed.

    The following are the conditions for a validator to get slashed:

    1. Downtime: If a validator is offline for more than 25% of a rolling window of the last 5,000 blocks, they will be jailed for 1 minute. During this period, the validator is removed from the validator set temporarily, and will be unable to propose new blocks or earn rewards. After the jail period, the validator can send an unjail request to rejoin the validator set.

    2. Double signing: This is a more severe offense and results in a larger percentage loss. If a validator engages in double signing, the validator will lose 2% of their stake and the remainder of their stake will be returned to them. The validator will be permanently removed from the validator set and will not have the ability to unjail. Delegators bonded to that validator will automatically enter the unbonding period for 21 days, and can delegate to another validator after they have been unbonded.

    For more details on the slashing parameters, refer to the celestia-app specifications page.

    + + + + \ No newline at end of file diff --git a/nodes/celestia-app-upgrade-monitor.html b/nodes/celestia-app-upgrade-monitor.html new file mode 100644 index 00000000000..8e8820c1225 --- /dev/null +++ b/nodes/celestia-app-upgrade-monitor.html @@ -0,0 +1,45 @@ + + + + + + Upgrade Monitor | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Upgrade Monitor

    Upgrade monitor is a binary that monitors that status of upgrades on a Celestia network. See the README for instructions on how to install and use the binary.

    + + + + \ No newline at end of file diff --git a/nodes/celestia-app-vesting.html b/nodes/celestia-app-vesting.html new file mode 100644 index 00000000000..4249c11d967 --- /dev/null +++ b/nodes/celestia-app-vesting.html @@ -0,0 +1,259 @@ + + + + + + How to create a vesting account with celestia-app | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    How to create a vesting account with celestia-app

    In this guide, we will learn how to create a vesting account using celestia-app for both a local devnet and on Mocha testnet.

    note

    The instructions for this tutorial are for a continuous vesting account, if you'd like to make a delayed vesting account, just add the --delayed flag to your vesting transaction.

    Local devnet

    First, download and install celestia-app, selecting the network and corresponding version that you would like to use.

    Setting up the local devnet

    Run the devnet

    Next, change into the $HOME/celestia-app directory and run the single-node-devnet script.

    bash
    cd $HOME/celestia-app
    +./scripts/build-run-single-node.sh
    cd $HOME/celestia-app
    +./scripts/build-run-single-node.sh

    Save the home directory path

    At the top of the output, you will see a path to the "Home directory", find yours from the output (it will be unique every time):

    bash
    ./scripts/build-run-single-node.sh
    +Home directory: /var/folders/_8/ljj6hspn0kn09qf9fy8kdyh40000gn/T/celestia_app_XXXXXXXXXXXXX.XV92a3qx
    +--> Updating go.mod
    ./scripts/build-run-single-node.sh
    +Home directory: /var/folders/_8/ljj6hspn0kn09qf9fy8kdyh40000gn/T/celestia_app_XXXXXXXXXXXXX.XV92a3qx
    +--> Updating go.mod

    And set the location as the CElESTIA_APP_HOME variable. We will use this for the remainder of the devnet section.

    bash
    export CElESTIA_APP_HOME=/var/folders/_8/ljj6hspn0kn09qf9fy8kdyh40000gn/T/celestia_app_XXXXXXXXXXXXX.XV92a3qx
    export CElESTIA_APP_HOME=/var/folders/_8/ljj6hspn0kn09qf9fy8kdyh40000gn/T/celestia_app_XXXXXXXXXXXXX.XV92a3qx

    note

    This does not replace the celestia-appd binary that was installed with celestia-appd, but builds and runs one in the $HOME/celestia-app/build directory.

    Check the version of the devnet

    If you'd like to check the version of your local devnet, you can use:

    bash
    cd $HOME/celestia-app/build
    +./celestia-appd version
    cd $HOME/celestia-app/build
    +./celestia-appd version

    Next steps

    Congratulations! You now have a private devnet running locally on your machine. The devnet is made up of one validator that is creating new blocks. This is the Celestia consensus network on your machine! The key that was created to run the validator also lives in a temporary directory for the devnet.

    Now you are ready to test creating a vesting account on our devnet before going to Mocha, a live testnet.

    Setting up vesting account on devnet

    You already have one key setup, but you will need one more to create a vesting account.

    Create a new key

    First, create a vesting key:

    bash
    cd $HOME/celestia-app/build
    +./celestia-appd keys add vesting-key --home $CElESTIA_APP_HOME
    cd $HOME/celestia-app/build
    +./celestia-appd keys add vesting-key --home $CElESTIA_APP_HOME

    You will see the address, mnemonic, and more details about your key in the output:

    bash
    - address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +  name: vesting-key
    +  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"A5JF/we+s5gFt6g944XbKVVYgQB9OY+U/l5dhZjLDczO"}'
    +  type: local
    +
    +
    +**Important** write this mnemonic phrase in a safe place.
    +It is the only way to recover your account if you ever forget your password.
    +
    +index enter egg broken ostrich duty bitter blind all car hollow coral youth early verify point void anger daring sausage decline net shove oil
    - address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +  name: vesting-key
    +  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"A5JF/we+s5gFt6g944XbKVVYgQB9OY+U/l5dhZjLDczO"}'
    +  type: local
    +
    +
    +**Important** write this mnemonic phrase in a safe place.
    +It is the only way to recover your account if you ever forget your password.
    +
    +index enter egg broken ostrich duty bitter blind all car hollow coral youth early verify point void anger daring sausage decline net shove oil

    List your keys

    bash
    ./celestia-appd keys list --home $CElESTIA_APP_HOME
    ./celestia-appd keys list --home $CElESTIA_APP_HOME

    Output:

    bash
    - address: celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +  name: validator
    +  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"Ahzu6yr9XMPIxLquhgBhj9xL3wIaOz6PE3CvML/oPQym"}'
    +  type: local
    +- address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +  name: vesting-key
    +  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"A5JF/we+s5gFt6g944XbKVVYgQB9OY+U/l5dhZjLDczO"}'
    +  type: local
    - address: celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +  name: validator
    +  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"Ahzu6yr9XMPIxLquhgBhj9xL3wIaOz6PE3CvML/oPQym"}'
    +  type: local
    +- address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +  name: vesting-key
    +  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"A5JF/we+s5gFt6g944XbKVVYgQB9OY+U/l5dhZjLDczO"}'
    +  type: local

    Set variables

    Set the keys as variables, using the validator address as the FROM_ADDRESS and the vesting-key as the TO_ADDRESS.

    bash
    export FROM_ADDRESS=celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +export TO_ADDRESS=celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    export FROM_ADDRESS=celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +export TO_ADDRESS=celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5

    Create your devnet vesting account

    Create the vesting account with the following command:

    note

    The remainder of the instructions are for a continuous vesting account, if you'd like to make a delayed vesting account, use the --delayed flag.

    For example, the command to create a delayed vesting account would look like:

    bash
    ./celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas auto --fees 100000utia --chain-id private --home $CElESTIA_APP_HOME --delayed
    ./celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas auto --fees 100000utia --chain-id private --home $CElESTIA_APP_HOME --delayed
    bash
    ./celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas auto --fees 100000utia --chain-id private --home $CElESTIA_APP_HOME
    ./celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas auto --fees 100000utia --chain-id private --home $CElESTIA_APP_HOME

    Select "Y" to choose "yes".

    Optional

    If you'd like to run the command with the -y flag, it will execute the transaction without needing to provide the "y" answer as above.

    bash
    ./celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas auto --fees 100000utia --chain-id private --home $CElESTIA_APP_HOME -y
    ./celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas auto --fees 100000utia --chain-id private --home $CElESTIA_APP_HOME -y

    Output:

    bash
    gas estimate: 96112
    +auth_info:
    +  fee:
    +    amount:
    +    - amount: "100000"
    +      denom: utia
    +    gas_limit: "96112"
    +    granter: ""
    +    payer: ""
    +  signer_infos: []
    +  tip: null
    +body:
    +  extension_options: []
    +  memo: ""
    +  messages:
    +  - '@type': /cosmos.vesting.v1beta1.MsgCreateVestingAccount
    +    amount:
    +    - amount: "100000"
    +      denom: utia
    +    delayed: false
    +    end_time: "1686748051"
    +    from_address: celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +    to_address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +  non_critical_extension_options: []
    +  timeout_height: "0"
    +signatures: []
    +confirm transaction before signing and broadcasting [y/N]: y
    +code: 0
    +codespace: ""
    +data: ""
    +events: []
    +gas_used: "0"
    +gas_wanted: "0"
    +height: "0"
    +info: ""
    +logs: []
    +raw_log: '[]'
    +timestamp: ""
    +tx: null
    +txhash: 6093DF76DBA90F04FF63D197FC1569F04ED3DBE64081A0BBA9BAD4E69AA570D2
    gas estimate: 96112
    +auth_info:
    +  fee:
    +    amount:
    +    - amount: "100000"
    +      denom: utia
    +    gas_limit: "96112"
    +    granter: ""
    +    payer: ""
    +  signer_infos: []
    +  tip: null
    +body:
    +  extension_options: []
    +  memo: ""
    +  messages:
    +  - '@type': /cosmos.vesting.v1beta1.MsgCreateVestingAccount
    +    amount:
    +    - amount: "100000"
    +      denom: utia
    +    delayed: false
    +    end_time: "1686748051"
    +    from_address: celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +    to_address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +  non_critical_extension_options: []
    +  timeout_height: "0"
    +signatures: []
    +confirm transaction before signing and broadcasting [y/N]: y
    +code: 0
    +codespace: ""
    +data: ""
    +events: []
    +gas_used: "0"
    +gas_wanted: "0"
    +height: "0"
    +info: ""
    +logs: []
    +raw_log: '[]'
    +timestamp: ""
    +tx: null
    +txhash: 6093DF76DBA90F04FF63D197FC1569F04ED3DBE64081A0BBA9BAD4E69AA570D2

    The timestamp for the previous command is in the past, so once you create the vesting account, the tokens will vest. You can check your account balances to verify this.

    Query the devnet vesting account details

    Check that the account has been created and works as expected by querying the TO_ADDRESS account details:

    bash
    ./celestia-appd query account $TO_ADDRESS --home $CElESTIA_APP_HOME
    ./celestia-appd query account $TO_ADDRESS --home $CElESTIA_APP_HOME

    In the output, you will notice that the account type is a ContinuousVestingAccount:

    bash
    '@type': /cosmos.vesting.v1beta1.ContinuousVestingAccount
    +base_vesting_account:
    +  base_account:
    +    account_number: "7"
    +    address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +    pub_key: null
    +    sequence: "0"
    +  delegated_free: []
    +  delegated_vesting: []
    +  end_time: "1686748051"
    +  original_vesting:
    +  - amount: "100000"
    +    denom: utia
    +start_time: "1687908352"
    '@type': /cosmos.vesting.v1beta1.ContinuousVestingAccount
    +base_vesting_account:
    +  base_account:
    +    account_number: "7"
    +    address: celestia127fpaygehlsgjdknwvlr2mux7h5uvhkxktgkc5
    +    pub_key: null
    +    sequence: "0"
    +  delegated_free: []
    +  delegated_vesting: []
    +  end_time: "1686748051"
    +  original_vesting:
    +  - amount: "100000"
    +    denom: utia
    +start_time: "1687908352"

    Query the devnet base account details

    Check the FROM_ADDRESS account details:

    bash
    ./celestia-appd query account $FROM_ADDRESS --home $CElESTIA_APP_HOME
    ./celestia-appd query account $FROM_ADDRESS --home $CElESTIA_APP_HOME

    In the output, you will notice the account type is BaseAccount:

    bash
    '@type': /cosmos.auth.v1beta1.BaseAccount
    +account_number: "0"
    +address: celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +pub_key:
    +  '@type': /cosmos.crypto.secp256k1.PubKey
    +  key: Ahzu6yr9XMPIxLquhgBhj9xL3wIaOz6PE3CvML/oPQym
    +sequence: "2"
    '@type': /cosmos.auth.v1beta1.BaseAccount
    +account_number: "0"
    +address: celestia1adgkqcmzuxvg7x5avx8a8rjwpmxgzex3ztef6j
    +pub_key:
    +  '@type': /cosmos.crypto.secp256k1.PubKey
    +  key: Ahzu6yr9XMPIxLquhgBhj9xL3wIaOz6PE3CvML/oPQym
    +sequence: "2"

    Query the balances of the devnet accounts

    Next, we can check the balance of the accounts:

    bash
    ./celestia-appd query bank balances $TO_ADDRESS --home $CElESTIA_APP_HOME
    ./celestia-appd query bank balances $TO_ADDRESS --home $CElESTIA_APP_HOME

    Output will show you the balance of the vesting account:

    bash
    balances:
    +- amount: "100000"
    +  denom: utia
    +pagination:
    +  next_key: null
    +  total: "0"
    balances:
    +- amount: "100000"
    +  denom: utia
    +pagination:
    +  next_key: null
    +  total: "0"
    bash
    ./celestia-appd query bank balances $FROM_ADDRESS --home $CElESTIA_APP_HOME
    ./celestia-appd query bank balances $FROM_ADDRESS --home $CElESTIA_APP_HOME

    The output will show the remaining balance of the validator:

    bash
    balances:
    +- amount: "999994999800000"
    +  denom: utia
    +pagination:
    +  next_key: null
    +  total: "0"
    balances:
    +- amount: "999994999800000"
    +  denom: utia
    +pagination:
    +  next_key: null
    +  total: "0"

    Congratulations! You've now made your own vesting account on a local devnet. Next, you can learn how to create a vesting account on Mocha testnet.

    Mocha

    In the previous section of this tutorial, we learned how to create a vesting account on a local devnet. In this portion of the tutorial, we'll cover how to set up a full consensus node and set up a vesting account on Mocha testnet.

    First, be sure that you have installed celestia-app for the latest version for Mocha testnet.

    Create a wallet

    Set the keyring backend, so you don't need to use the flag for every command:

    bash
    celestia-appd config keyring-backend test
    celestia-appd config keyring-backend test

    Add a new key for a full node and one for a vesting account:

    bash
    celestia-appd keys add origin && celestia-appd keys add vesting
    celestia-appd keys add origin && celestia-appd keys add vesting

    List the keys:

    bash
    celestia-appd keys list
    celestia-appd keys list

    Set your keys as variables:

    bash
    export FROM_ADDRESS=address_of_origin_account
    +export TO_ADDRESS=address_of_vesting_account
    export FROM_ADDRESS=address_of_origin_account
    +export TO_ADDRESS=address_of_vesting_account

    Fund your account

    Head to the faucet, and fund your origin address.

    Create a vesting account on Mocha

    To create a vesting account on Mocha, you will need an RPC URL to send the transaction to. You can find the RPC endpoints on the Mocha testnet page.

    Set your RPC URL:

    bash
    export RPC_URL=https://rpc-mocha.pops.one:443
    export RPC_URL=https://rpc-mocha.pops.one:443

    We will use a few flags in our vesting command that are different than the devnet version. Since we aren't using our own validator or full node, we will use an RPC URL.

    We also need to declare the chain ID as mocha.

    View the help menu for vesting to understand these flags more:

    bash
    celestia-appd tx vesting --help
    celestia-appd tx vesting --help

    Here's an example command to set up the vesting account:

    bash
    celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas 100000 --fees 100000utia --node $RPC_URL --chain-id mocha --delayed
    celestia-appd tx vesting create-vesting-account $TO_ADDRESS 100000utia 1686748051 --from $FROM_ADDRESS --gas 100000 --fees 100000utia --node $RPC_URL --chain-id mocha --delayed

    Optional: Set up a full consensus node or validator

    Running a full consensus node or validator will prevent you from needing to use an RPC.

    You can set up a validator or full consensus node for the previous portion of the tutorial.

    Note: this may take some time depending on how you choose to sync the state of the chain.

    Optional: Change your client.toml

    If you edit your client configuration in client.toml, you can set both the chain ID and the node RPC URL. This will prevent you from needing to run each flag for every command line that you use.

    toml
    # This is a TOML config file.
    +# For more information, see https://github.com/toml-lang/toml
    +
    +###############################################################################
    +###                           Client Configuration                            ###
    +###############################################################################
    +
    +# The network chain ID
    +chain-id = "mocha"
    +# The keyring's backend, where the keys are stored (os|file|kwallet|pass|test|memory)
    +keyring-backend = "test"
    +# CLI output format (text|json)
    +output = "text"
    +# <host>:<port> to Tendermint RPC interface for this chain
    +node = "tcp://rpc-mocha.pops.one:443"
    +# Transaction broadcasting mode (sync|async|block)
    +broadcast-mode = "sync"
    # This is a TOML config file.
    +# For more information, see https://github.com/toml-lang/toml
    +
    +###############################################################################
    +###                           Client Configuration                            ###
    +###############################################################################
    +
    +# The network chain ID
    +chain-id = "mocha"
    +# The keyring's backend, where the keys are stored (os|file|kwallet|pass|test|memory)
    +keyring-backend = "test"
    +# CLI output format (text|json)
    +output = "text"
    +# <host>:<port> to Tendermint RPC interface for this chain
    +node = "tcp://rpc-mocha.pops.one:443"
    +# Transaction broadcasting mode (sync|async|block)
    +broadcast-mode = "sync"

    Notes

    Not all vesting accounts can be created with a message, some need to be set at genesis. You can learn more in the Cosmos Network documentation.

    Conclusion

    Congratulations! You've learned how to create a local devnet, create a vesting account on it, and how to make a vesting account on the Mocha testnet!

    + + + + \ No newline at end of file diff --git a/nodes/celestia-app-wallet.html b/nodes/celestia-app-wallet.html new file mode 100644 index 00000000000..ba0c4897d16 --- /dev/null +++ b/nodes/celestia-app-wallet.html @@ -0,0 +1,47 @@ + + + + + + Create a wallet with celestia-app | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Create a wallet with celestia-app

    For this guide, we will go over how you can generate a Celestia wallet using celestia-app.

    Prerequisites

    Note, you do not need to install celestia-node for this tutorial.

    Create a wallet

    First, create an application CLI configuration file:

    sh
    celestia-appd config keyring-backend test
    celestia-appd config keyring-backend test

    You can pick whatever wallet name you want. For our example we used "validator" as the wallet name:

    sh
    celestia-appd keys add validator --interactive
    celestia-appd keys add validator --interactive

    Save the mnemonic output as this is the only way to recover your validator wallet in case you lose it!

    To check all your wallets you can run:

    sh
    celestia-appd keys list
    celestia-appd keys list

    Fund a wallet

    For the public celestia address, you can fund the previously created wallet via Discord by sending this message to either the #mocha-faucet or #arabica-faucet channel:

    text
    $request celestia1xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
    $request celestia1xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx

    Wait to see if you get a confirmation that the tokens have been successfully sent. To check if tokens have arrived successfully to the destination wallet run the command below replacing the public address with your own:

    sh
    celestia-appd start
    +celestia-appd query bank balances celestia1xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
    celestia-appd start
    +celestia-appd query bank balances celestia1xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx

    TIP

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    + + + + \ No newline at end of file diff --git a/nodes/celestia-app.html b/nodes/celestia-app.html new file mode 100644 index 00000000000..53784c12103 --- /dev/null +++ b/nodes/celestia-app.html @@ -0,0 +1,54 @@ + + + + + + Install celestia-app | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Install celestia-app

    This tutorial will guide you through installing celestia-app, both from source and with a pre-built binary

    Building binary from source

    This section of the tutorial presumes you completed the steps in setting up your own environment.

    The steps below will create a binary file named celestia-appd inside $HOME/go/bin folder which will be used later to run the node. Be sure to select the correct network to install the binary for.

    1. Remove any existing copy of celestia-app, clone the repository, and change into the directory:

      bash
      cd $HOME
      +rm -rf celestia-app
      +git clone https://github.com/celestiaorg/celestia-app.git
      +cd celestia-app
      cd $HOME
      +rm -rf celestia-app
      +git clone https://github.com/celestiaorg/celestia-app.git
      +cd celestia-app
    2. Check out to the desired version, based on the network you will use:

      bash
      git checkout tags/v1.7.0 -b v1.7.0
      git checkout tags/v1.7.0 -b v1.7.0
      bash
      git checkout tags/v1.7.0 -b v1.7.0
      git checkout tags/v1.7.0 -b v1.7.0
      bash
      git checkout tags/v1.7.0 -b v1.7.0
      git checkout tags/v1.7.0 -b v1.7.0
    3. Build and install the celestia-appd binary:

      bash
      make install
      make install
    4. To check if the binary was successfully installed you can run the binary using the --help flag:

      sh
      celestia-appd --help
      celestia-appd --help

    You will see an output with the menu for celestia-appd. Learn more on the helpful CLI commands page

    Installing a pre-built binary

    Installing a pre-built binary is the fastest way to get started with your Celestia consensus node. Releases after celestia-app v1.3.0 should have these binaries available.

    The steps below will download a binary file named celestia-appd. Depending on the setup that you choose during installation, the celestia-appd binary will be available at either:

    • $HOME/celestia-app-temp/celestia-appd
    • /usr/local/bin/celestia-appd

    Pre-built binaries are available for:

    • Operating systems: Darwin (Apple), Linux
    • Architectures: x86_64 (amd64), arm64

    To install the latest pre-built binary you can run this command in your terminal:

    bash
    bash -c "$(curl -sL https://docs.celestia.org/celestia-app.sh)"
    bash -c "$(curl -sL https://docs.celestia.org/celestia-app.sh)"

    Follow the instructions in the terminal output to choose your installation preferences.

    You will see an output with the menu for celestia-appd. Learn more on the helpful CLI commands page

    View the script to learn more about what it is doing.

    Ports

    When interacting with a consensus node, you may need to open ports on your machine to allow communication between nodes, such as bridge nodes. It is essential that specific ports are accessible. Make sure that your firewall allows connections to the correct ports.

    If you run a node on a cloud server, make sure that the ports are open on the server's firewall. If you run a node at home, make sure that your router allows connections to the correct ports.

    For example, validator ports 9090 and 26657 need to be accessible by the bridge, and port 2121 is required for P2P connections for all node types.

    The following ports are used by Celestia app nodes:

    PortProtocolAddressDescriptionEnabled by default on nodeFlag
    2121TCP/UDPlocalhostP2PtrueN/A
    9090HTTP0.0.0.0gRPCtrue--grpc.address string
    26657TCPlocalhostRPCfalse (only open to localhost)--rpc.laddr string
    + + + + \ No newline at end of file diff --git a/nodes/celestia-node-custom-networks.html b/nodes/celestia-node-custom-networks.html new file mode 100644 index 00000000000..2c6d617a1f8 --- /dev/null +++ b/nodes/celestia-node-custom-networks.html @@ -0,0 +1,57 @@ + + + + + + Custom networks and values | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Custom networks and values

    This section will cover importing bootstrapper IDs, chain ID, and network ID. This will allow you to import custom values for a chain that is not in the default configuration.

    If you have a custom network you can export CELESTIA_CUSTOM, which will look something like:

    bash
    export BRIDGE="/ip4/<ip-address>/tcp/2121/p2p/<node-ID>"
    +export GENESIS_HASH=<genesis-hash>
    +export NETWORK=<network-name>
    +export CELESTIA_CUSTOM="${NETWORK}:${GENESIS_HASH}:${BRIDGE}"
    export BRIDGE="/ip4/<ip-address>/tcp/2121/p2p/<node-ID>"
    +export GENESIS_HASH=<genesis-hash>
    +export NETWORK=<network-name>
    +export CELESTIA_CUSTOM="${NETWORK}:${GENESIS_HASH}:${BRIDGE}"

    Query your node ID using the RPC CLI. These values with examples would look like:

    bash
    export BRIDGE="/ip4/151.115.14.33/tcp/2121/p2p/12D3KooWKEeRtzVMPUdxYsZo2edqps6mS67n6LT5mPdULSkPSxBQ"
    +export GENESIS_HASH=580B3DFF8A7C716968161D91116A1E171F486298D582874E93714E489C9E6E88
    +export NETWORK=custom
    +export CELESTIA_CUSTOM="${NETWORK}:${GENESIS_HASH}:${BRIDGE}"
    export BRIDGE="/ip4/151.115.14.33/tcp/2121/p2p/12D3KooWKEeRtzVMPUdxYsZo2edqps6mS67n6LT5mPdULSkPSxBQ"
    +export GENESIS_HASH=580B3DFF8A7C716968161D91116A1E171F486298D582874E93714E489C9E6E88
    +export NETWORK=custom
    +export CELESTIA_CUSTOM="${NETWORK}:${GENESIS_HASH}:${BRIDGE}"

    Then, start your node with:

    bash
    celestia <node-type> start [flags...]
    celestia <node-type> start [flags...]
    + + + + \ No newline at end of file diff --git a/nodes/celestia-node-metrics.html b/nodes/celestia-node-metrics.html new file mode 100644 index 00000000000..2606eb6fb5c --- /dev/null +++ b/nodes/celestia-node-metrics.html @@ -0,0 +1,73 @@ + + + + + + celestia-node metrics | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    celestia-node metrics

    This tutorial is for running metrics for your celestia-node data availability instance. This tutorial will focus on running metrics for a light node.

    This tutorial assumes you have already setup your light node by following the tutorial in the celestia-node API tutorial.

    Running metrics flags

    You can enable the celestia-node metric flags with the following command:

    sh
    celestia <node-type> start --metrics.tls=<boolean> \
    +    --metrics --metrics.endpoint <URI> \
    +    --p2p.network <network> --core.ip <URI>
    celestia <node-type> start --metrics.tls=<boolean> \
    +    --metrics --metrics.endpoint <URI> \
    +    --p2p.network <network> --core.ip <URI>

    Add metrics flags to your node start command and restart your node to apply it. The metrics endpoint will gather your node's data to track your uptime.

    Note that the --metrics flag enables metrics and expects an input into --metrics.endpoint.

    We will go over what the endpoint will need to be in the metrics endpoint design considerations section.

    Mainnet Beta

    Here is an example for Mainnet Beta:

    sh
    celestia <node-type> start --metrics.tls=true \
    +    --metrics --metrics.endpoint otel.celestia.observer \
    +    --core.ip <URI>
    celestia <node-type> start --metrics.tls=true \
    +    --metrics --metrics.endpoint otel.celestia.observer \
    +    --core.ip <URI>

    Mocha testnet

    Here is an example for Mocha testnet:

    sh
    celestia <node-type> start --metrics.tls=true \
    +    --metrics --metrics.endpoint otel.celestia-mocha.com \
    +    --core.ip <URI> --p2p.network mocha
    celestia <node-type> start --metrics.tls=true \
    +    --metrics --metrics.endpoint otel.celestia-mocha.com \
    +    --core.ip <URI> --p2p.network mocha

    TLS connections

    The --metrics.tls flag enables or disables a TLS connection to the OpenTelemetry Protocol metrics backend. You need to choose a boolean value (true or false) for this flag.

    It's also common to set this flag to false when spinning up a local collector to check the metrics locally.

    However, if the collector is hosted in the cloud as a separate entity (like in a DevOps environment), enabling TLS is a necessity for secure communication.

    Here are examples of how to use it:

    bash
    # To enable TLS connection
    +celestia <node-type> start --metrics.tls=true --metrics \
    +    --metrics.endpoint <URI> \
    +    --p2p.network <network> --core.ip <URI>
    +
    +# To disable TLS connection
    +celestia <node-type> start --metrics.tls=false --metrics \
    +    --metrics.endpoint <URI> \
    +    --p2p.network <network> --core.ip <URI>
    # To enable TLS connection
    +celestia <node-type> start --metrics.tls=true --metrics \
    +    --metrics.endpoint <URI> \
    +    --p2p.network <network> --core.ip <URI>
    +
    +# To disable TLS connection
    +celestia <node-type> start --metrics.tls=false --metrics \
    +    --metrics.endpoint <URI> \
    +    --p2p.network <network> --core.ip <URI>

    Metrics endpoint design considerations

    At the moment, the architecture of celestia-node metrics works as specified in the following ADR #010.

    Essentially, the design considerations here will necessitate running an OpenTelemetry (OTEL) collector that connects to Celestia light node.

    For an overview of OTEL, check out the guide.

    The ADR and the OTEL docs will help you run your collector on the metrics endpoint. This will then allow you to process the data in the collector on a Prometheus server which can then be viewed on a Grafana dashboard.

    In the future, we do want to open-source some developer toolings around this infrastructure to allow for node operators to be able to monitor their data availability nodes.

    + + + + \ No newline at end of file diff --git a/nodes/celestia-node-troubleshooting.html b/nodes/celestia-node-troubleshooting.html new file mode 100644 index 00000000000..3530b1c9d4e --- /dev/null +++ b/nodes/celestia-node-troubleshooting.html @@ -0,0 +1,76 @@ + + + + + + Troubleshooting | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Troubleshooting

    Network selection

    Note: If you do not select a network, the default network will be Mainnet Beta.

    sh
    celestia <node-type> init --p2p.network <network>
    +celestia <node-type> start --p2p.network <network> --core.ip <URI>
    celestia <node-type> init --p2p.network <network>
    +celestia <node-type> start --p2p.network <network> --core.ip <URI>

    TIP

    Refer to the ports section of this page for information on which ports are required to be open on your machine.

    NOTE

    It is advised before switching networks to reinitialize your node via init command. This is due to an old config being present. Re-initialisation will reset the config.

    Chain ID

    When interacting with celestia-node, it is important to take into account the different chain IDs for different networks. For Mainnet Beta, there is no need to declare a chain ID, as the default is celestia, i.e. no --p2p.network string flag is required for Mainnet Beta.

    NetworkChain ID--p2p.network string
    Mainnet Betacelestianot required (--p2p.network celestia)
    Mochamocha-4--p2p.network mocha
    Arabicaarabica-11--p2p.network arabica

    Ports

    When interacting with a Celestia node, you may need to open ports on your machine to allow communication between nodes, such as bridge nodes. It is essential that specific ports are accessible. Make sure that your firewall allows connections to the correct ports.

    If you run a node on a cloud server, make sure that the ports are open on the server's firewall. If you run a node at home, make sure that your router allows connections to the correct ports.

    For example, validator ports 9090 and 26657 need to be accessible by the bridge, and port 2121 is required for P2P connections for all node types.

    The following ports are used by Celestia nodes:

    PortProtocolAddressDescriptionEnabled by default on nodeFlag
    2121TCP/UDPlocalhostP2PtrueN/A
    26658HTTPlocalhostRPCtrue--rpc.port string
    26659HTTPlocalhostREST Gatewayfalse--gateway.port string

    WARNING

    The gateway endpoints have been deprecated and will be removed in the future. If you would like to use them anyway, you can find more details on GitHub.

    Changing the location of your node store

    In this section, we'll guide you through starting your node using a node store in a different location than you originally started with.

    First, stop your node safely using control + C.

    Then, init your node again with a new node store:

    bash
    celestia <node-type> init --node.store /home/user/celestia-<node-type>-location/ \
    +    --p2p.network mocha
    celestia <node-type> init --node.store /home/user/celestia-<node-type>-location/ \
    +    --p2p.network mocha

    Next, start your node:

    bash
    celestia full start --core.ip rpc-mocha.pops.one --p2p.network mocha \
    +    --node.store /home/user/celestia-<node-type>-location/
    celestia full start --core.ip rpc-mocha.pops.one --p2p.network mocha \
    +    --node.store /home/user/celestia-<node-type>-location/

    If you choose to change the location of your node store, you will need to execute each command on your node with the following flag:

    bash
    --node.store /home/user/celestia-<node-type>-location/
    --node.store /home/user/celestia-<node-type>-location/

    When using cel-key, the process is different. To show the keys you should add --keyring-dir like this example:

    bash
    ./cel-key list --p2p.network mocha --node.type full \
    +    --keyring-dir /home/user/celestia-<node-type>-location/keys/
    ./cel-key list --p2p.network mocha --node.type full \
    +    --keyring-dir /home/user/celestia-<node-type>-location/keys/

    Resetting your config

    If you an encounter an error, it is likely that an old config file is present:

    sh
    Error: nodebuilder/share: interval must be positive; nodebuilder/core: invalid IP addr given:
    +
    +# or
    +
    +Error: nodebuilder/share: interval must be positive
    Error: nodebuilder/share: interval must be positive; nodebuilder/core: invalid IP addr given:
    +
    +# or
    +
    +Error: nodebuilder/share: interval must be positive

    You can re-initialize your node's config with the following commands:

    TIP

    Save your config so custom values are not lost.

    Run the following command to update your config:

    bash
    celestia <node-type> config-update --p2p.network <network>
    celestia <node-type> config-update --p2p.network <network>

    This will pull in any new values from new configuration and merge them into the existing configuration.

    TIP

    After using the config-update command, it is encouraged to double-check that your custom values are preserved.

    Then, to start your node again:

    bash
    celestia <node-type> start --p2p.network <network>
    celestia <node-type> start --p2p.network <network>

    Clearing the data store

    For bridge, full, and light nodes, remove the data store with this command:

    bash
    celestia <node-type> unsafe-reset-store --p2p.network <network>
    celestia <node-type> unsafe-reset-store --p2p.network <network>
    bash
    celestia light unsafe-reset-store --p2p.network mocha
    celestia light unsafe-reset-store --p2p.network mocha

    FATAL headers given to the heightSub are in the wrong order

    If you observe a FATAL log line like:

    bash
    FATAL   header/store   store/heightsub.go:87    PLEASE FILE A BUG REPORT: headers given to the heightSub are in the wrong order"
    FATAL   header/store   store/heightsub.go:87    PLEASE FILE A BUG REPORT: headers given to the heightSub are in the wrong order"

    then it is possible the celestia-node data/ directory contains headers from a previous instance of the network that you are currently trying to run against. One resolution strategy is to delete the existing celestia-node config for the target network and re-initialize it:

    sh
    # rm -rf ~/.celestia-<node-type>-<network>
    +rm -rf ~/.celestia-bridge-private
    +
    +# celestia <node-type> init --p2p.network <network>
    +celestia bridge init --p2p.network private
    # rm -rf ~/.celestia-<node-type>-<network>
    +rm -rf ~/.celestia-bridge-private
    +
    +# celestia <node-type> init --p2p.network <network>
    +celestia bridge init --p2p.network private

    Error: "too many open files"

    When running a Celestia bridge node, you may encounter an error in the logs similar to this:

    bash
    Error while creating log file in valueLog.open error: while opening file: /opt/celestia/.celestia-bridge/data/003442.vlog error: open /opt/celestia/.celestia-bridge/data/003442.vlog: too many open files
    Error while creating log file in valueLog.open error: while opening file: /opt/celestia/.celestia-bridge/data/003442.vlog error: open /opt/celestia/.celestia-bridge/data/003442.vlog: too many open files

    This error indicates that the Celestia application is trying to open more files than the operating system's limit allows. To fix this, you will need to edit the Celestia bridge service file to increase the number of file descriptors that the service can open.

    1. Open the service file for editing:
    bash
    nano /etc/systemd/system/celestia-bridge.service
    nano /etc/systemd/system/celestia-bridge.service
    1. Modify the LimitNOFILE parameter:

    In the service file, find the LimitNOFILE parameter under the [Service] section and set its value to 1400000. It should look like this:

    ini
    [Service]
    +...
    +LimitNOFILE=1400000
    +...
    [Service]
    +...
    +LimitNOFILE=1400000
    +...

    NOTE

    Be cautious when increasing file descriptor limits. Setting this value too high might affect system performance. Ensure the value is appropriate for your system's capabilities.

    1. Reload daemon and restart bridge service:
    bash
    sudo systemctl daemon-reload
    sudo systemctl daemon-reload
    bash
    sudo systemctl restart celestia-bridge
    sudo systemctl restart celestia-bridge
    + + + + \ No newline at end of file diff --git a/nodes/celestia-node.html b/nodes/celestia-node.html new file mode 100644 index 00000000000..db5890ece46 --- /dev/null +++ b/nodes/celestia-node.html @@ -0,0 +1,54 @@ + + + + + + Install celestia-node | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Install celestia-node

    Installing from source

    This section goes over building and installing celestia-node. This tutorial assumes you completed the steps in setting up your development environment.

    Install the celestia-node binary by running the following commands:

    1. Remove any existing copy of celestia-node, clone the repository, and change into the directory:

      bash
      cd $HOME
      +rm -rf celestia-node
      +git clone https://github.com/celestiaorg/celestia-node.git
      +cd celestia-node/
      cd $HOME
      +rm -rf celestia-node
      +git clone https://github.com/celestiaorg/celestia-node.git
      +cd celestia-node/
    2. Check out to the desired version, based on the network you will use:

      bash
      git checkout tags/v0.13.2
      git checkout tags/v0.13.2
      bash
      git checkout tags/v0.13.3
      git checkout tags/v0.13.3
      bash
      git checkout tags/v0.13.3
      git checkout tags/v0.13.3
    3. Build the celestia binary:

      a. Standard build

      bash
      make build
      make build

      b. Experimental build

      OPTIONAL

      If you're a node operator comfortable with experimental features and seeking optimal performance with minimal RAM usage, this option is recommended for you.

      bash
      make build-jemalloc
      make build-jemalloc

      This build option enables CGO, and downloads and installs jemalloc. Learn more about the build command.

    4. Install the binary:

      bash
      make install
      make install
    5. Build the cel-key utility:

      bash
      make cel-key
      make cel-key
    6. Verify that the binary is working and check the version:

      bash
      celestia version
      celestia version

    The output will show the semantic version of celestia-node, commit hash, build date, system version, and Golang version.

    Installing a pre-built binary

    Installing a pre-built binary is the fastest way to get started with your Celestia data availability node. Releases after celestia-node v0.13.3 should have these binaries available.

    The steps below will download a binary file named celestia. Depending on the setup that you choose during installation, the celestia binary will be available at either:

    • $HOME/celestia-node-temp/celestia
    • /usr/local/bin/celestia

    Pre-built binaries are available for:

    • Operating systems: Darwin (Apple), Linux
    • Architectures: x86_64 (amd64), arm64

    To install the latest pre-built binary you can run this command in your terminal:

    bash
    bash -c "$(curl -sL https://docs.celestia.org/celestia-node.sh)"
    bash -c "$(curl -sL https://docs.celestia.org/celestia-node.sh)"

    Follow the instructions in the terminal output to choose your installation preferences.

    You will see an output with the menu for celestia.

    View the script to learn more about what it is doing.

    Next steps

    First, we recommend reading the overview of our node types, if you haven't yet.

    Now that you've installed Celestia Node, it's time to pick your node type and run your node!

    If you're planning to run a light node, we recommend the node RPC CLI tutorial.

    Upgrading your binary

    To upgrade your binary, you can install the latest version from the instructions above and restart your node. If you run into any issues, Refer to the troubleshooting section.

    + + + + \ No newline at end of file diff --git a/nodes/config-toml.html b/nodes/config-toml.html new file mode 100644 index 00000000000..9c51266cf24 --- /dev/null +++ b/nodes/config-toml.html @@ -0,0 +1,45 @@ + + + + + + config.toml guide | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    config.toml guide

    Pre-requisites

    Please, make sure that you have installed and initialized celestia-node

    Understanding config.toml

    After initialization, for any type of node, you will find a config.toml in the following path (default location):

    • $HOME/.celestia-bridge/config.toml for bridge node
    • $HOME/.celestia-light/config.toml for light node
    • $HOME/.celestia-full/config.toml for a full DA node

    Let's break down some of the most used sections.

    Core

    This section is needed for the Celestia bridge node. By default, Remote = false. Still for devnet, we are going to use the remote core option and this can also be set by the command line flag --core.remote.

    P2P

    Bootstrap

    Bootstrappers help new nodes to find peers faster in the network. By default, the Bootstrapper = false and the BootstrapPeers is empty. If you want your node to be a bootstrapper, then activate Bootstrapper = true. BootstrapPeers are already provided by default during initialisation. If you want to add your own manually, you need to provide the multiaddresses of the peers.

    Mutual peers

    The purpose of this config is to set up a bidirectional communication. This is usually the case for Celestia bridge nodes. In addition, you need to change the field PeerExchange from false to true.

    Services

    TrustedHash and TrustedPeer

    TrustedHash is needed to properly initialize a Celestia bridge node with an already-running Remote celestia-core node. Celestia light node will take a genesis hash as the trusted one, if no hash is manually provided during initialization phase.

    TrustedPeers is the array of bridge nodes' peers that Celestia light node trusts. By default, bootstrap peers becomes trusted peers for Celestia light nodes if a user is not setting the trusted peer params in config file.

    Any Celestia bridge node can be a trusted peer for the light one. However, the light node by design can not be a trusted peer for another light node.

    + + + + \ No newline at end of file diff --git a/nodes/decide-node.html b/nodes/decide-node.html new file mode 100644 index 00000000000..94c583bb78d --- /dev/null +++ b/nodes/decide-node.html @@ -0,0 +1,45 @@ + + + + + + Deciding which node to run | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Deciding which node to run

    Now that you have installed the basic dependencies, you can start exploring which nodes to run!

    Beginner

    It is highly recommended if you are a beginner to get started with running a Data-Availability light node.

    In order to get started, you can proceed to the light node section.

    You can also play around with the Data Availability API in this tutorial for posting and retrieving data with a light node.

    Advanced

    If you are looking to run a consensus node, please follow the tutorial for running a consensus node.

    Note that running a validator means you must also run a bridge node, which is covered in this section.

    + + + + \ No newline at end of file diff --git a/nodes/docker-images.html b/nodes/docker-images.html new file mode 100644 index 00000000000..2c91315c320 --- /dev/null +++ b/nodes/docker-images.html @@ -0,0 +1,106 @@ + + + + + + 🐳 Docker setup | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    🐳 Docker setup

    This page has instructions to run celestia-node using Docker. If you are looking for instructions to run celestia-node using a binary, please refer to the celestia-node page.

    Using Docker is the easiest way to run celestia-node for most users. Docker is a containerization platform that allows you to run celestia-node in an isolated environment.

    This means that you can run celestia-node on your machine without having to worry about installing and configuring all of the dependencies required to run the node.

    If you would like to learn more about key management in Docker, visit the Docker and cel-key section.

    The easiest way to install Docker is to use the Docker Desktop installer or Ubuntu. You can follow the instructions for your operating system.

    Prerequisites

    Quick start

    1. Set the network you would like to run your node on:

      bash
      export NETWORK=celestia
      export NETWORK=celestia
      bash
      export NETWORK=mocha
      export NETWORK=mocha
      bash
      export NETWORK=arabica
      export NETWORK=arabica
    2. Set the node type

      bash
      export NODE_TYPE=light
      export NODE_TYPE=light
      bash
      export NODE_TYPE=bridge
      export NODE_TYPE=bridge
      bash
      export NODE_TYPE=full
      export NODE_TYPE=full
    3. Set an RPC endpoint for either Mainnet Beta, Mocha, or Arabica using the bare URL (without http or https):

      bash
      export RPC_URL=this-is-an-rpc-url.com
      export RPC_URL=this-is-an-rpc-url.com
    4. Run the image from the command line:

      bash
      docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
      +    ghcr.io/celestiaorg/celestia-node:v0.13.2 \
      +    celestia $NODE_TYPE start --core.ip $RPC_URL --p2p.network $NETWORK
      docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
      +    ghcr.io/celestiaorg/celestia-node:v0.13.2 \
      +    celestia $NODE_TYPE start --core.ip $RPC_URL --p2p.network $NETWORK
      bash
      docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
      +    ghcr.io/celestiaorg/celestia-node:v0.13.3 \
      +    celestia $NODE_TYPE start --core.ip $RPC_URL --p2p.network $NETWORK
      docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
      +    ghcr.io/celestiaorg/celestia-node:v0.13.3 \
      +    celestia $NODE_TYPE start --core.ip $RPC_URL --p2p.network $NETWORK
      bash
      docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
      +    ghcr.io/celestiaorg/celestia-node:v0.13.3 \
      +    celestia $NODE_TYPE start --core.ip $RPC_URL --p2p.network $NETWORK
      docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
      +    ghcr.io/celestiaorg/celestia-node:v0.13.3 \
      +    celestia $NODE_TYPE start --core.ip $RPC_URL --p2p.network $NETWORK

    Congratulations! You now have a celestia-node running!

    If you would like to run the node with custom flags, you can refer to the celestia-node tutorial page. Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    Light node setup with persistent storage

    If you delete a container that you started above, all data will be lost. To avoid this, you can mount a volume to the container. This will allow you to persist data even after the container is deleted.

    First, you will need to create a directory on your host machine. This directory will be used to store the data for the container. Create a directory on your host machine and give it a name. For example, you can name it my-node-store:

    bash
    cd $HOME
    +mkdir my-node-store
    cd $HOME
    +mkdir my-node-store

    Now, you can mount this directory to the container. Before mounting a volume, you may need to set permissions for the user on the host machine by running:

    bash
    sudo chown 10001:10001 $HOME/my-node-store
    sudo chown 10001:10001 $HOME/my-node-store
    bash
    # you're good to go 😎
    # you're good to go 😎

    Initialize the node store and key

    In order to mount a volume to the container, you need to specify the path to the volume. When you run your container, you can specify the path to the volume using the --volume (or -v for short) flag. In this command, we'll create our key and initialize the node store, using the variables we set in the quick start section:

    bash
    # --volume == -v [local path]:[container path]
    +docker run [args...] -v $HOME/my-node-store:/home/celestia \
    +    celestia $NODE_TYPE init [args...]
    # --volume == -v [local path]:[container path]
    +docker run [args...] -v $HOME/my-node-store:/home/celestia \
    +    celestia $NODE_TYPE init [args...]

    An example init command will look similar to below:

    bash
    docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
    +    -v $HOME/my-node-store:/home/celestia \
    +    ghcr.io/celestiaorg/celestia-node:v0.13.2 \
    +    celestia light init --p2p.network $NETWORK
    docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
    +    -v $HOME/my-node-store:/home/celestia \
    +    ghcr.io/celestiaorg/celestia-node:v0.13.2 \
    +    celestia light init --p2p.network $NETWORK
    bash
    docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
    +    -v $HOME/my-node-store:/home/celestia \
    +    ghcr.io/celestiaorg/celestia-node:v0.13.3 \
    +    celestia light init --p2p.network $NETWORK
    docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
    +    -v $HOME/my-node-store:/home/celestia \
    +    ghcr.io/celestiaorg/celestia-node:v0.13.3 \
    +    celestia light init --p2p.network $NETWORK
    bash
    docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
    +    -v $HOME/my-node-store:/home/celestia \
    +    ghcr.io/celestiaorg/celestia-node:v0.13.3 \
    +    celestia light init --p2p.network $NETWORK
    docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
    +    -v $HOME/my-node-store:/home/celestia \
    +    ghcr.io/celestiaorg/celestia-node:v0.13.3 \
    +    celestia light init --p2p.network $NETWORK

    Start the node

    Run the following command to start the node:

    bash
    # --volume == -v [local path]:[container path]
    +docker run [...args] -v $HOME/my-node-store:/home/celestia \
    +    celestia <node-type> start [...args]
    # --volume == -v [local path]:[container path]
    +docker run [...args] -v $HOME/my-node-store:/home/celestia \
    +    celestia <node-type> start [...args]

    A full start command will look similar to below.

    bash
    docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
    +    -v $HOME/my-node-store:/home/celestia \
    +    ghcr.io/celestiaorg/celestia-node:v0.13.2 \
    +    celestia light start --core.ip $RPC_URL
    docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
    +    -v $HOME/my-node-store:/home/celestia \
    +    ghcr.io/celestiaorg/celestia-node:v0.13.2 \
    +    celestia light start --core.ip $RPC_URL
    bash
    docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
    +    -v $HOME/my-node-store:/home/celestia \
    +    ghcr.io/celestiaorg/celestia-node:v0.13.3 \
    +    celestia light start --core.ip $RPC_URL
    docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
    +    -v $HOME/my-node-store:/home/celestia \
    +    ghcr.io/celestiaorg/celestia-node:v0.13.3 \
    +    celestia light start --core.ip $RPC_URL
    bash
    docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
    +    -v $HOME/my-node-store:/home/celestia \
    +    ghcr.io/celestiaorg/celestia-node:v0.13.3 \
    +    celestia light start --core.ip $RPC_URL
    docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \
    +    -v $HOME/my-node-store:/home/celestia \
    +    ghcr.io/celestiaorg/celestia-node:v0.13.3 \
    +    celestia light start --core.ip $RPC_URL

    Congratulations! You now have a node running with persistent storage.

    Video walkthrough

    2.5 minute version

    Troubleshooting

    For security purposes Celestia expects to interact with the your node's keys in a read-only manner. This is enforced using linux style permissions on the filesystem. Windows NTFS does not support these types of permissions. As a result the recommended path for Windows users to mount a persisted volume is to do so within WSL. You can find instructions for installing WSL.

    + + + + \ No newline at end of file diff --git a/nodes/environment.html b/nodes/environment.html new file mode 100644 index 00000000000..196e9117a59 --- /dev/null +++ b/nodes/environment.html @@ -0,0 +1,96 @@ + + + + + + Development environment | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Development environment

    This page will go over setting up your development environment to run Celestia software. This environment can be used for development, building binaries, and running nodes.

    Install dependencies

    1. If you are on Ubuntu, first update and upgrade your OS:

      bash
      sudo apt update && sudo apt upgrade -y
      sudo apt update && sudo apt upgrade -y
      bash
      sudo yum update
      sudo yum update
    2. Install essential packages that are necessary to execute many tasks like downloading files, compiling, and monitoring the node:

      bash
      sudo apt install curl tar wget aria2 clang pkg-config libssl-dev jq build-essential \
      +git make ncdu -y
      sudo apt install curl tar wget aria2 clang pkg-config libssl-dev jq build-essential \
      +git make ncdu -y
      bash
      sudo yum install curl tar wget aria2 clang pkg-config libssl-dev jq build-essential \
      +git make ncdu -y
      sudo yum install curl tar wget aria2 clang pkg-config libssl-dev jq build-essential \
      +git make ncdu -y
      bash
      # these commands are for installing Homebrew, wget and jq
      +# follow the instructions from the output after running this command
      +/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
      +
      +# then install wget & jq
      +brew install wget && brew install jq
      # these commands are for installing Homebrew, wget and jq
      +# follow the instructions from the output after running this command
      +/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
      +
      +# then install wget & jq
      +brew install wget && brew install jq

    Install Golang

    celestia-node is written in Golang so we must install Golang to build and run our node.

    1. Set the version for your desired network:

      bash
      ver="1.21.1"
      ver="1.21.1"
      bash
      ver="1.22.0"
      ver="1.22.0"
      bash
      ver="1.22.0"
      ver="1.22.0"
    2. Download and install Golang:

      bash
      cd $HOME
      +wget "https://golang.org/dl/go$ver.linux-amd64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.linux-amd64.tar.gz"
      +rm "go$ver.linux-amd64.tar.gz"
      cd $HOME
      +wget "https://golang.org/dl/go$ver.linux-amd64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.linux-amd64.tar.gz"
      +rm "go$ver.linux-amd64.tar.gz"
      bash
      cd $HOME
      +wget "https://golang.org/dl/go$ver.linux-arm64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.linux-arm64.tar.gz"
      +rm "go$ver.linux-arm64.tar.gz"
      cd $HOME
      +wget "https://golang.org/dl/go$ver.linux-arm64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.linux-arm64.tar.gz"
      +rm "go$ver.linux-arm64.tar.gz"
      bash
      cd $HOME
      +wget "https://golang.org/dl/go$ver.darwin-arm64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.darwin-arm64.tar.gz"
      +rm "go$ver.darwin-arm64.tar.gz"
      cd $HOME
      +wget "https://golang.org/dl/go$ver.darwin-arm64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.darwin-arm64.tar.gz"
      +rm "go$ver.darwin-arm64.tar.gz"
      bash
      cd $HOME
      +wget "https://golang.org/dl/go$ver.darwin-amd64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.darwin-amd64.tar.gz"
      +rm "go$ver.darwin-amd64.tar.gz"
      cd $HOME
      +wget "https://golang.org/dl/go$ver.darwin-amd64.tar.gz"
      +sudo rm -rf /usr/local/go
      +sudo tar -C /usr/local -xzf "go$ver.darwin-amd64.tar.gz"
      +rm "go$ver.darwin-amd64.tar.gz"
    3. Add your /usr/local/go/bin directory to your $PATH if you have not already:

      bash
      echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.bash_profile
      +source $HOME/.bash_profile
      echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.bash_profile
      +source $HOME/.bash_profile
      bash
      echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.zshrc
      +source $HOME/.zshrc
      echo "export PATH=$PATH:/usr/local/go/bin:$HOME/go/bin" >> $HOME/.zshrc
      +source $HOME/.zshrc
    4. To verify that the correct version of Go was installed correctly run:

      bash
      go version
      go version

    The output will show the version installed.

    + + + + \ No newline at end of file diff --git a/nodes/full-consensus-node.html b/nodes/full-consensus-node.html new file mode 100644 index 00000000000..e9023461d83 --- /dev/null +++ b/nodes/full-consensus-node.html @@ -0,0 +1,122 @@ + + + + + + Setting up a Celestia full consensus node | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Setting up a Celestia full consensus node

    This guide covers how to set up a full consensus node on Celestia. Full consensus nodes allow you to sync blockchain history in the Celestia consensus layer.

    full consensus node

    Hardware requirements

    The following hardware minimum requirements are recommended for running a full consensus node:

    • Memory: 8 GB RAM
    • CPU: Quad-Core
    • Disk: 250 GB SSD Storage
    • Bandwidth: 1 Gbps for Download/1 Gbps for Upload

    Running a full consensus node requires significant storage capacity to store the entire blockchain history. As of the latest recommendation, it is advisable to have at least 250 GB of SSD storage for a Celestia full consensus node if you are using pruning. If you are not using pruning, you are running an archive node, and it is recommended to have 500 GB of SSD storage. Please ensure that your storage meets this requirement to ensure smooth syncing and operation of the node.

    Setting up a full consensus node

    The following tutorial is done on an Ubuntu Linux 20.04 (LTS) x64 instance machine.

    Set up the dependencies

    Follow the instructions on installing dependencies.

    Install celestia-app

    Follow the tutorial on installing celestia-app.

    Set up the P2P networks

    Now we will set up the P2P Networks by cloning the networks repository:

    sh
    cd $HOME
    +rm -rf networks
    +git clone https://github.com/celestiaorg/networks.git
    cd $HOME
    +rm -rf networks
    +git clone https://github.com/celestiaorg/networks.git

    To initialize the network, pick a "node-name" that describes your node. Keep in mind that this might change if a new testnet is deployed.

    bash
    celestia-appd init "node-name" --chain-id celestia
    celestia-appd init "node-name" --chain-id celestia
    bash
    celestia-appd init "node-name" --chain-id mocha-4
    celestia-appd init "node-name" --chain-id mocha-4
    bash
    celestia-appd init "node-name" --chain-id arabica-11
    celestia-appd init "node-name" --chain-id arabica-11

    Download the genesis.json file:

    bash
    celestia-appd download-genesis celestia
    celestia-appd download-genesis celestia
    bash
    celestia-appd download-genesis mocha-4
    celestia-appd download-genesis mocha-4
    bash
    celestia-appd download-genesis arabica-11
    celestia-appd download-genesis arabica-11

    Set seeds in the $HOME/.celestia-app/config/config.toml file:

    bash
    SEEDS=$(curl -sL https://raw.githubusercontent.com/celestiaorg/networks/master/celestia/seeds.txt | tr '\n' ',')
    +echo $SEEDS
    +sed -i.bak -e "s/^seeds *=.*/seeds = \"$SEEDS\"/" $HOME/.celestia-app/config/config.toml
    SEEDS=$(curl -sL https://raw.githubusercontent.com/celestiaorg/networks/master/celestia/seeds.txt | tr '\n' ',')
    +echo $SEEDS
    +sed -i.bak -e "s/^seeds *=.*/seeds = \"$SEEDS\"/" $HOME/.celestia-app/config/config.toml
    bash
    SEEDS=$(curl -sL https://raw.githubusercontent.com/celestiaorg/networks/master/mocha-4/seeds.txt | tr '\n' ',')
    +echo $SEEDS
    +sed -i.bak -e "s/^seeds *=.*/seeds = \"$SEEDS\"/" $HOME/.celestia-app/config/config.toml
    SEEDS=$(curl -sL https://raw.githubusercontent.com/celestiaorg/networks/master/mocha-4/seeds.txt | tr '\n' ',')
    +echo $SEEDS
    +sed -i.bak -e "s/^seeds *=.*/seeds = \"$SEEDS\"/" $HOME/.celestia-app/config/config.toml
    bash
    # For Arabica, you can set seeds manually in the
    +# `$HOME/.celestia-app/config/config.toml` file:
    +# Comma separated list of seed nodes to connect to
    +seeds = ""
    # For Arabica, you can set seeds manually in the
    +# `$HOME/.celestia-app/config/config.toml` file:
    +# Comma separated list of seed nodes to connect to
    +seeds = ""

    Optionally, you can set persistent peers in your config.toml file. You can get the persistent peers from the networks repository with the following commands:

    Setting persistent peers is advised only if you are running a sentry node.

    bash
    PERSISTENT_PEERS=$(curl -sL https://raw.githubusercontent.com/celestiaorg/networks/master/celestia/peers.txt | tr '\n' ',')
    +echo $PERSISTENT_PEERS
    +sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \"$PERSISTENT_PEERS\"/" $HOME/.celestia-app/config/config.toml
    PERSISTENT_PEERS=$(curl -sL https://raw.githubusercontent.com/celestiaorg/networks/master/celestia/peers.txt | tr '\n' ',')
    +echo $PERSISTENT_PEERS
    +sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \"$PERSISTENT_PEERS\"/" $HOME/.celestia-app/config/config.toml
    bash
    PERSISTENT_PEERS=$(curl -sL https://raw.githubusercontent.com/celestiaorg/networks/master/mocha-4/peers.txt | tr '\n' ',')
    +echo $PERSISTENT_PEERS
    +sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \"$PERSISTENT_PEERS\"/" $HOME/.celestia-app/config/config.toml
    PERSISTENT_PEERS=$(curl -sL https://raw.githubusercontent.com/celestiaorg/networks/master/mocha-4/peers.txt | tr '\n' ',')
    +echo $PERSISTENT_PEERS
    +sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \"$PERSISTENT_PEERS\"/" $HOME/.celestia-app/config/config.toml
    bash
    PERSISTENT_PEERS=$(curl -sL https://raw.githubusercontent.com/celestiaorg/networks/master/arabica-11/peers.txt | tr '\n' ',')
    +echo $PERSISTENT_PEERS
    +sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \"$PERSISTENT_PEERS\"/" $HOME/.celestia-app/config/config.toml
    PERSISTENT_PEERS=$(curl -sL https://raw.githubusercontent.com/celestiaorg/networks/master/arabica-11/peers.txt | tr '\n' ',')
    +echo $PERSISTENT_PEERS
    +sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \"$PERSISTENT_PEERS\"/" $HOME/.celestia-app/config/config.toml

    Storage and pruning configurations

    Connecting a consensus node to a bridge node

    If your consensus node is being connected to a celestia-node bridge node, you will need to enable transaction indexing and retain all block data. This can be achieved with the following settings in your config.toml.

    Enable transaction indexing

    toml
    indexer = "kv"
    indexer = "kv"

    Retain all block data

    And in your app.toml, min-retain-blocks should remain as the default setting of 0:

    toml
    min-retain-blocks = 0 # retain all block data, this is default setting
    min-retain-blocks = 0 # retain all block data, this is default setting

    Querying transactions by hash

    To query transactions using their hash, transaction indexing must be turned on. Set the indexer to "kv" in your config.toml:

    toml
    indexer = "kv"
    indexer = "kv"

    Accessing historical state

    If you want to query the historical state — for example, you might want to know the balance of a Celestia wallet at a given height in the past — you should run an archive node with pruning = "nothing" in your app.toml. Note that this configuration is resource-intensive and will require significant storage:

    toml
    pruning = "nothing"
    pruning = "nothing"

    Saving on storage requirements

    If you want to save on storage requirements, consider using pruning = "everything" in your app.toml to prune everything. If you select "everything" or "default", but still want to keep the block data, you can do so by not changing the default value of min-retain-blocks = 0 in your app.toml. A value of 0 for min-retain-blocks will keep all block data. This will prune snapshots of the state, but it will keep block data:

    toml
    pruning = "everything"
    +min-retain-blocks = 0 # this is the default setting
    pruning = "everything"
    +min-retain-blocks = 0 # this is the default setting

    Syncing

    By default, a consensus node will sync using block sync; that is request, validate and execute every block up to the head of the blockchain. This is the most secure mechanism yet the slowest (taking up to days depending on the height of the blockchain).

    There are two alternatives for quicker syncing.

    State sync

    State sync uses light client verification to verify state snapshots from peers and then apply them. State sync relies on weak subjectivity; a trusted header (specifically the hash and height) must be provided. This can be found by querying a trusted RPC endpoint (/block). RPC endpoints are also required for retrieving light blocks. These can be found in the docs here under the respective networks or from the chain-registry.

    In $HOME/.celestia-app/config/config.toml, set

    toml
    rpc_servers = ""
    +trust_height = 0
    +trust_hash = ""
    rpc_servers = ""
    +trust_height = 0
    +trust_hash = ""

    To their respective fields. At least two different rpc endpoints should be provided. The more, the greater the chance of detecting any fraudulent behavior.

    Once setup, you should be ready to start the node as normal. In the logs, you should see: Discovering snapshots. This may take a few minutes before snapshots are found depending on the network topology.

    Quick sync

    Quick sync effectively downloads the entire data directory from a third-party provider meaning the node has all the application and blockchain state as the node it was copied from.

    Run the following command to quick-sync from a snapshot:

    bash
    cd $HOME
    +rm -rf ~/.celestia-app/data
    +mkdir -p ~/.celestia-app/data
    +SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \
    +    egrep -o ">celestia.*tar" | tr -d ">")
    +aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"
    +tar xf celestia-snap.tar -C ~/.celestia-app/data/
    cd $HOME
    +rm -rf ~/.celestia-app/data
    +mkdir -p ~/.celestia-app/data
    +SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \
    +    egrep -o ">celestia.*tar" | tr -d ">")
    +aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"
    +tar xf celestia-snap.tar -C ~/.celestia-app/data/
    bash
    cd $HOME
    +rm -rf ~/.celestia-app/data
    +mkdir -p ~/.celestia-app/data
    +SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \
    +    egrep -o ">mocha-4.*tar" | tr -d ">")
    +aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"
    +tar xf celestia-snap.tar -C ~/.celestia-app/data/
    cd $HOME
    +rm -rf ~/.celestia-app/data
    +mkdir -p ~/.celestia-app/data
    +SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \
    +    egrep -o ">mocha-4.*tar" | tr -d ">")
    +aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"
    +tar xf celestia-snap.tar -C ~/.celestia-app/data/
    bash
    cd $HOME
    +rm -rf ~/.celestia-app/data
    +mkdir -p ~/.celestia-app/data
    +SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \
    +    egrep -o ">arabica-11.*tar" | tr -d ">")
    +aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"
    +tar xf celestia-snap.tar -C ~/.celestia-app/data/
    cd $HOME
    +rm -rf ~/.celestia-app/data
    +mkdir -p ~/.celestia-app/data
    +SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \
    +    egrep -o ">arabica-11.*tar" | tr -d ">")
    +aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"
    +tar xf celestia-snap.tar -C ~/.celestia-app/data/

    Start the consensus node

    In order to start your full consensus node, run the following:

    sh
    celestia-appd start
    celestia-appd start

    Optional: If you would like celestia-app to run as a background process, you can follow the SystemD tutorial.

    TIP

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    Extra resources for consensus nodes

    Optional: Reset network

    This will delete all data folders so we can start fresh:

    sh
    celestia-appd tendermint unsafe-reset-all --home $HOME/.celestia-app
    celestia-appd tendermint unsafe-reset-all --home $HOME/.celestia-app

    Optional: Configuring an RPC endpoint

    You can configure your full consensus node to be a public RPC endpoint. This allows it to accept connections from data availability nodes and serve requests for the data availability API.

    Expose RPC

    By default, the RPC service listens on localhost which means it can't be accessed from other machines. To make the RPC service available publicly, you need to bind it to a public IP or 0.0.0.0 (which means listening on all available network interfaces).

    You can do this by editing the config.toml file:

    sh
    sed -i 's#"tcp://127.0.0.1:26657"#"tcp://0.0.0.0:26657"#g' ~/.celestia-app/config/config.toml
    sed -i 's#"tcp://127.0.0.1:26657"#"tcp://0.0.0.0:26657"#g' ~/.celestia-app/config/config.toml

    This command replaces the localhost IP address with 0.0.0.0, making the RPC service listen on all available network interfaces.

    Note on external-address

    The external-address field in the configuration is used when your node is behind a NAT and you need to advertise a different address for peers to dial. Populating this field is not necessary for making the RPC endpoint public.

    sh
    EXTERNAL-ADDRESS=$(wget -qO- eth0.me)
    +sed -i.bak -e "s/^external-address = ""/external-address = "$EXTERNAL-ADDRESS:26656"/" \
    +    $HOME/.celestia-app/config/config.toml
    EXTERNAL-ADDRESS=$(wget -qO- eth0.me)
    +sed -i.bak -e "s/^external-address = ""/external-address = "$EXTERNAL-ADDRESS:26656"/" \
    +    $HOME/.celestia-app/config/config.toml

    Restart the node

    After making these changes, restart celestia-appd to load the new configurations.

    Optional: Transaction indexer configuration options

    This section guides you on how to configure your config.toml file in celestia-app to select which transactions to index. Depending on the application's configuration, a node operator may decide which transactions to index.

    The available options are:

    1. null: This option disables indexing. If you don't need to query transactions, you can choose this option to save space.
    2. kv (default): This is the simplest indexer, backed by key-value storage (defaults to levelDB; see DBBackend). When kv is chosen, tx.height and tx.hash will always be indexed. This option is suitable for basic queries on transactions.
    3. psql: This indexer is backed by PostgreSQL. When psql is chosen, tx.height and tx.hash will always be indexed. This option is suitable for complex queries on transactions.

    An example to set the value to kv in config.toml is:

    toml
    indexer = "kv"
    indexer = "kv"

    Remember to restart celestia-appd after making changes to the configuration to load the new settings.

    Optional: Discard ABCI responses configuration

    This section will guide you on how to configure your config.toml file in celestia-app to manage the storage of ABCI responses. ABCI responses are the results of executing transactions and are used for /block_results RPC queries and to reindex events in the command-line tool.

    The discard_abci_responses option allows you to control whether these responses are persisted in the state store:

    • false (default): ABCI responses are stored in the state store. This ensures that ABCI responses are available for /block_results RPC queries and for reindexing events. However, it can consume a significant amount of disk space.
    • true: ABCI responses are not stored in the state store. This can save a considerable amount of disk space, but /block_results RPC queries and event reindexing will not be available.

    An example to set the value to false in config.toml is:

    toml
    discard_abci_responses = false
    discard_abci_responses = false

    Remember to restart celestia-appd after making changes to the configuration to load the new settings.

    + + + + \ No newline at end of file diff --git a/nodes/full-storage-node.html b/nodes/full-storage-node.html new file mode 100644 index 00000000000..6bd2c26f508 --- /dev/null +++ b/nodes/full-storage-node.html @@ -0,0 +1,55 @@ + + + + + + Setting up a Celestia full storage Node | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Setting up a Celestia full storage Node

    This tutorial will guide you through setting up a Celestia full storage node, which is a celestia-node that doesn't connect to celestia-app (hence not a full consensus node), but stores all the data.

    Overview of full storage nodes

    Full storage nodes are Celestia nodes that store all the data. Full storage nodes send block shares, headers, and fraud proofs to light nodes. The light nodes gossip headers, fraud proofs, and sometimes block shares, between one another.

    Full storage node

    Hardware requirements

    The following hardware minimum requirements are recommended for running the full storage node:

    • Memory: 4 GB RAM (minimum)
    • CPU: 6 cores
    • Disk: 10 TB SSD Storage
    • Bandwidth: 1 Gbps for Download/1 Gbps for Upload

    Setting up your full storage node

    The following tutorial is done on an Ubuntu Linux 20.04 (LTS) x64 instance machine.

    Setup the dependencies

    You can follow the tutorial for setting up your dependencies

    Install celestia-node

    You can follow the tutorial for installing celestia-node

    Run the full storage node

    Initialize the full storage node

    Run the following command:

    sh
    celestia full init
    celestia full init
    sh
    celestia full init --p2p.network mocha
    celestia full init --p2p.network mocha
    sh
    celestia full init --p2p.network arabica
    celestia full init --p2p.network arabica

    Start the full storage node

    Start the full storage node with a connection to a validator node's gRPC endpoint (which is usually exposed on port 9090):

    In order for access to the ability to get/submit state-related information, such as the ability to submit PayForBlob transactions, or query for the node's account balance, a gRPC endpoint of a validator (core) node must be passed as directed below.

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    sh
    celestia full start --core.ip <URI>
    celestia full start --core.ip <URI>

    Using an RPC of your own, or one from the list on the Mocha testnet page or list on the Arabica devnet page, start your node.

    You can create your key for your node by following the cel-key instructions

    Once you start the full storage node, a wallet key will be generated for you. You will need to fund that address with testnet tokens to pay for PayForBlob transactions. You can find the address by running the following command:

    sh
    ./cel-key list --node.type full --keyring-backend test --p2p.network <network>
    ./cel-key list --node.type full --keyring-backend test --p2p.network <network>

    TIP

    You do not need to declare a network for Mainnet Beta. Refer to the chain ID section on the troubleshooting page for more information

    You can get testnet tokens from:

    NOTE

    If you are running a full-storage node for your sovereign rollup, it is highly recommended to request Arabica devnet tokens as Arabica has the latest changes that can be used to test for developing your sovereign rollup. You can still use Mocha testnet as well, it is just mostly used for validator operations.

    Optional: run the full storage node with a custom key

    In order to run a full storage node using a custom key:

    1. The custom key must exist inside the celestia full storage node directory at the correct path (default: ~/.celestia-full/keys/keyring-test)
    2. The name of the custom key must be passed upon start, like so:
    sh
    celestia full start --core.ip <URI> \
    +  --keyring.accname <name-of-custom-key> \
    celestia full start --core.ip <URI> \
    +  --keyring.accname <name-of-custom-key> \
    sh
    celestia full start --core.ip <URI> \
    +  --keyring.accname <name-of-custom-key> \
    +  --p2p.network mocha
    celestia full start --core.ip <URI> \
    +  --keyring.accname <name-of-custom-key> \
    +  --p2p.network mocha
    sh
    celestia full start --core.ip <URI> \
    +  --keyring.accname <name-of-custom-key> \
    +  --p2p.network arabica
    celestia full start --core.ip <URI> \
    +  --keyring.accname <name-of-custom-key> \
    +  --p2p.network arabica

    Optional: start the full storage node with SystemD

    If you would like to run the full storage node as a background process, follow the SystemD tutorial.

    With that, you are now running a Celestia full storage node.

    Stop the full storage node

    In order to gracefully stop the full storage node, use Control + C in the terminal window where the node is running. Be sure to only do this once as the shutdown will not be instantaneous.

    + + + + \ No newline at end of file diff --git a/nodes/hardfork-process.html b/nodes/hardfork-process.html new file mode 100644 index 00000000000..08fecbbf89a --- /dev/null +++ b/nodes/hardfork-process.html @@ -0,0 +1,45 @@ + + + + + + Celestia hardfork process | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Celestia hardfork process

    Blockchain networks often times need to upgrade with new features which require coordination work among the validators prior to activating the upgrades.

    This process is called a hardfork or a network upgrade. In those events, the Celestia Labs team will be coordinating with the validators on what they need to do in order to be ready for an upcoming hardfork.

    Hardforks are not backward-compatible with older versions of the network software which is why it is important that validators upgrade their software to continue validating on the network after the network upgrades.

    General process

    The general process can be broken down into several components:

    • Hardfork specifications and features (defined by description of features and code implementation of those features).
    • Binary used to add those features (a new binary release with those features will be provided by Celestia team in order for validators to upgrade their nodes to the new binary).
    • A block number for when the network upgrades (even if validators upgrade their binary to be hardfork ready, the network upgrade does not happen right away, but some short time in the future at a specific block number).
    • Testing of the features (happens on testnets first prior to activating on mainnet in order to ensure the network can upgrade securely).

    The two testnets were hardforks are deployed on are:

    Mocha hardfork

    Celestia is planning the Mocha Hardfork upgrade on the Mamaki Testnet. This hardfork is unique as it will reset the Mamaki network to block 0 while maintaining the existing state and also will rename Mamaki to Mocha.

    The new chain-id will be mocha.

    You can find the release logs for consensus nodes on the celestia-app releases page.

    The most exciting feature included is setting the stage for Blobstream on Mocha.

    Validators will need to generate 2 new keys in order to be Blobstream-ready. Note that for the Mocha Hardfork, Blobstream will not launch yet so you can swap those keys after for new ones if needed. The keys needed are:

    • 1 EVM key
    • 1 Celestia key

    So, in order for this to happen, validators will need to maintain two new keys in order to have a successful upgrade.

    Those two keys will need to be added to 2 new flags on celestia-app:

    • --evm-address: This flag should contain a 0x EVM address.
    + + + + \ No newline at end of file diff --git a/nodes/ibc-relayer.html b/nodes/ibc-relayer.html new file mode 100644 index 00000000000..e065666c252 --- /dev/null +++ b/nodes/ibc-relayer.html @@ -0,0 +1,499 @@ + + + + + + IBC relaying guide | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    IBC relaying guide

    Celestia uses IBC (Inter-Blockchain Communication protocol) to enable cross-chain transfer of tokens. To support this capability it relies on relayers, processes that can be run by anyone which constantly scan for outbound packets on one chain and submits these packets alongside corresponding proofs on the destination chain. This section describes how one can setup a relayer and create new connections between chains. There are two standard implementations:

    The following guide explains how to establish IBC connections and relay packets between Mocha testnet and Cosmos hub testnet networks by using the Hermes relayer.

    Check the latest celestia-app release's go.mod for the version of ibc-go that is currently used.

    Hermes

    Hermes is an open-source Rust implementation of an IBC relayer released as part of the ibc-relayer-cli crate. It includes a CLI for relaying packets between Cosmos SDK chains, as well as Prometheus metrics and a REST API.

    Please follow the steps at Hermes Quick Start to install Hermes. Before proceeding, verify that Hermes is installed correctly by running hermes version.

    TIP

    Hermes currently doesn't support configuring the Tendermint CompatMode in chain config (see hermes#3623). Until that issue is resolved, please use Hermes v1.7.0+ because it falls back to Tendermint CompatMode v0.34 (see hermes#3663) which is compatible with Celestia.

    Configuration

    After you have successfully installed Hermes and created the necessary folders, you now have to edit config.toml and add the appropriate configurations for the chains you want to relay between.

    For this tutorial, we will be using the following chains:

    • Celestia's mocha-4 testnet
    • Cosmos Hub's theta-testnet-001 testnet

    Edit the Hermes configuration.

    bash
    vim $HOME/.hermes/config.toml
    vim $HOME/.hermes/config.toml
    toml
    [global]
    +log_level = "info"
    +
    +[mode.clients]
    +enabled = true
    +refresh = true
    +misbehaviour = true
    +
    +[mode.connections]
    +enabled = false
    +
    +[mode.channels]
    +enabled = false
    +
    +[mode.packets]
    +enabled = true
    +clear_interval = 100
    +clear_on_start = true
    +tx_confirmation = false
    +auto_register_counterparty_payee = false
    +
    +[rest]
    +enabled = false
    +host = "127.0.0.1"
    +port = 3000
    +
    +[telemetry]
    +enabled = false
    +host = "127.0.0.1"
    +port = 3001
    +
    +[telemetry.buckets.latency_submitted]
    +start = 500
    +end = 20000
    +buckets = 10
    +
    +[telemetry.buckets.latency_confirmed]
    +start = 1000
    +end = 30000
    +buckets = 10
    +
    +[[chains]]
    +id = "theta-testnet-001"
    +type = "CosmosSdk"
    +rpc_addr = "https://rpc.sentry-01.theta-testnet.polypore.xyz"
    +grpc_addr = "https://grpc.sentry-01.theta-testnet.polypore.xyz"
    +rpc_timeout = "10s"
    +trusted_node = false
    +account_prefix = "cosmos"
    +key_name = "key-cosmos"
    +key_store_type = "Test"
    +store_prefix = "ibc"
    +default_gas = 100000
    +max_gas = 400000
    +gas_multiplier = 1.5
    +max_msg_num = 30
    +max_tx_size = 180000
    +max_grpc_decoding_size = 33554432
    +clock_drift = "5s"
    +max_block_time = "30s"
    +ccv_consumer_chain = false
    +memo_prefix = ""
    +sequential_batch_tx = false
    +
    +[chains.event_source]
    +mode = "push"
    +url = "ws://rpc.sentry-01.theta-testnet.polypore.xyz:26657/websocket"
    +batch_delay = "500ms"
    +
    +[chains.trust_threshold]
    +numerator = "1"
    +denominator = "3"
    +
    +[chains.gas_price]
    +price = 0.025
    +denom = "uatom"
    +
    +[chains.packet_filter]
    +policy = "allow"
    +list = [["transfer", "channel-3108"]]
    +
    +[chains.packet_filter.min_fees]
    +
    +[chains.address_type]
    +derivation = "cosmos"
    +
    +[[chains]]
    +id = "mocha-4"
    +type = "CosmosSdk"
    +rpc_addr = "https://rpc-celestia-mocha.architectnodes.com"
    +grpc_addr = "https://grpc.celestia-mocha.com:443"
    +rpc_timeout = "10s"
    +trusted_node = false
    +account_prefix = "celestia"
    +key_name = "celestia-key"
    +key_store_type = "Test"
    +store_prefix = "ibc"
    +default_gas = 100000
    +max_gas = 400000
    +gas_multiplier = 1.5
    +max_msg_num = 30
    +max_tx_size = 180000
    +max_grpc_decoding_size = 33554432
    +clock_drift = "5s"
    +max_block_time = "30s"
    +ccv_consumer_chain = false
    +memo_prefix = ""
    +sequential_batch_tx = false
    +
    +[chains.event_source]
    +mode = "push"
    +url = "ws://rpc-mocha.pops.one:26657/websocket"
    +batch_delay = "500ms"
    +
    +[chains.trust_threshold]
    +numerator = "1"
    +denominator = "3"
    +
    +[chains.gas_price]
    +price = 0.1
    +denom = "utia"
    +
    +[chains.packet_filter]
    +policy = "allow"
    +list = [["transfer", "channel-0"]]
    +
    +[chains.packet_filter.min_fees]
    +
    +[chains.address_type]
    +derivation = "cosmos"
    [global]
    +log_level = "info"
    +
    +[mode.clients]
    +enabled = true
    +refresh = true
    +misbehaviour = true
    +
    +[mode.connections]
    +enabled = false
    +
    +[mode.channels]
    +enabled = false
    +
    +[mode.packets]
    +enabled = true
    +clear_interval = 100
    +clear_on_start = true
    +tx_confirmation = false
    +auto_register_counterparty_payee = false
    +
    +[rest]
    +enabled = false
    +host = "127.0.0.1"
    +port = 3000
    +
    +[telemetry]
    +enabled = false
    +host = "127.0.0.1"
    +port = 3001
    +
    +[telemetry.buckets.latency_submitted]
    +start = 500
    +end = 20000
    +buckets = 10
    +
    +[telemetry.buckets.latency_confirmed]
    +start = 1000
    +end = 30000
    +buckets = 10
    +
    +[[chains]]
    +id = "theta-testnet-001"
    +type = "CosmosSdk"
    +rpc_addr = "https://rpc.sentry-01.theta-testnet.polypore.xyz"
    +grpc_addr = "https://grpc.sentry-01.theta-testnet.polypore.xyz"
    +rpc_timeout = "10s"
    +trusted_node = false
    +account_prefix = "cosmos"
    +key_name = "key-cosmos"
    +key_store_type = "Test"
    +store_prefix = "ibc"
    +default_gas = 100000
    +max_gas = 400000
    +gas_multiplier = 1.5
    +max_msg_num = 30
    +max_tx_size = 180000
    +max_grpc_decoding_size = 33554432
    +clock_drift = "5s"
    +max_block_time = "30s"
    +ccv_consumer_chain = false
    +memo_prefix = ""
    +sequential_batch_tx = false
    +
    +[chains.event_source]
    +mode = "push"
    +url = "ws://rpc.sentry-01.theta-testnet.polypore.xyz:26657/websocket"
    +batch_delay = "500ms"
    +
    +[chains.trust_threshold]
    +numerator = "1"
    +denominator = "3"
    +
    +[chains.gas_price]
    +price = 0.025
    +denom = "uatom"
    +
    +[chains.packet_filter]
    +policy = "allow"
    +list = [["transfer", "channel-3108"]]
    +
    +[chains.packet_filter.min_fees]
    +
    +[chains.address_type]
    +derivation = "cosmos"
    +
    +[[chains]]
    +id = "mocha-4"
    +type = "CosmosSdk"
    +rpc_addr = "https://rpc-celestia-mocha.architectnodes.com"
    +grpc_addr = "https://grpc.celestia-mocha.com:443"
    +rpc_timeout = "10s"
    +trusted_node = false
    +account_prefix = "celestia"
    +key_name = "celestia-key"
    +key_store_type = "Test"
    +store_prefix = "ibc"
    +default_gas = 100000
    +max_gas = 400000
    +gas_multiplier = 1.5
    +max_msg_num = 30
    +max_tx_size = 180000
    +max_grpc_decoding_size = 33554432
    +clock_drift = "5s"
    +max_block_time = "30s"
    +ccv_consumer_chain = false
    +memo_prefix = ""
    +sequential_batch_tx = false
    +
    +[chains.event_source]
    +mode = "push"
    +url = "ws://rpc-mocha.pops.one:26657/websocket"
    +batch_delay = "500ms"
    +
    +[chains.trust_threshold]
    +numerator = "1"
    +denominator = "3"
    +
    +[chains.gas_price]
    +price = 0.1
    +denom = "utia"
    +
    +[chains.packet_filter]
    +policy = "allow"
    +list = [["transfer", "channel-0"]]
    +
    +[chains.packet_filter.min_fees]
    +
    +[chains.address_type]
    +derivation = "cosmos"

    Add relayer wallets

    Now that we have successfully configured our relaying chains, we need to import the wallets that will be used for relaying. Please note that both wallets need to be funded with the native tokens of each chain.

    You can get testnet tokens from faucets for bot testnets via Discord:

    Add your seed phrase to a file and upload it to the server. Do not use wallets for anything else but relaying to avoid running into account sequence errors.

    Follow the steps at adding-keys-to-hermes to add keys for each chain

    bash
    hermes keys add --chain mocha-4 --mnemonic-file <seed-file>
    +hermes keys add --chain theta-testnet-001 --mnemonic-file <seed-file>
    hermes keys add --chain mocha-4 --mnemonic-file <seed-file>
    +hermes keys add --chain theta-testnet-001 --mnemonic-file <seed-file>

    Verify configuration files

    After editing config.toml and adding wallet keys, it’s time to test the configurations and ensure the system is healthy. Run the following:

    bash
    hermes health-check
    +hermes config validate
    hermes health-check
    +hermes config validate

    If everything was set up correctly, you should see output like:

    bash
    SUCCESS performed health check for all chains in the config
    +SUCCESS "configuration is valid"
    SUCCESS performed health check for all chains in the config
    +SUCCESS "configuration is valid"

    Create a connection between 2 chains

    If you’re attempting to create new connections, verify that the chains in question don’t already have connections and clients in place and use the existing ones if they do. In that case you can skip this step and go to Configure channels in Hermes section.

    In this example, we are creating new clients and a new connection between mocha-4 and theta-testnet-001 networks.

    Create clients

    To learn if a client already exists, you can use the following command:

    bash
    hermes query clients --host-chain mocha-4 --reference-chain theta-testnet-001
    hermes query clients --host-chain mocha-4 --reference-chain theta-testnet-001

    To create a new client, use the create-client command:

    bash
    hermes create client --host-chain mocha-4 --reference-chain theta-testnet-001
    hermes create client --host-chain mocha-4 --reference-chain theta-testnet-001

    Create a second client:

    bash
    hermes create client --host-chain theta-testnet-001 --reference-chain mocha-4
    hermes create client --host-chain theta-testnet-001 --reference-chain mocha-4

    Open connection over new clients

    To create a new connection over clients, use the following command:

    bash
    hermes create connection --a-chain mocha-4 --b-chain theta-testnet-001
    hermes create connection --a-chain mocha-4 --b-chain theta-testnet-001

    You should be seeing a similar output to this:

    bash
    SUCCESS Connection {
    +    delay_period: 0ns,
    +    a_side: ConnectionSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "theta-testnet-001",
    +                version: 0,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-2382",
    +        ),
    +        connection_id: Some(
    +            ConnectionId(
    +                "connection-2727",
    +            ),
    +        ),
    +    },
    +    b_side: ConnectionSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "mocha-4",
    +                version: 4,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-0",
    +        ),
    +        connection_id: Some(
    +            ConnectionId(
    +                "connection-0",
    +            ),
    +        ),
    +    },
    +}
    SUCCESS Connection {
    +    delay_period: 0ns,
    +    a_side: ConnectionSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "theta-testnet-001",
    +                version: 0,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-2382",
    +        ),
    +        connection_id: Some(
    +            ConnectionId(
    +                "connection-2727",
    +            ),
    +        ),
    +    },
    +    b_side: ConnectionSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "mocha-4",
    +                version: 4,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-0",
    +        ),
    +        connection_id: Some(
    +            ConnectionId(
    +                "connection-0",
    +            ),
    +        ),
    +    },
    +}

    Now that the connection has been established over the clients, we need to create a new channel, by leveraging an existing connection:

    bash
    hermes create channel --a-chain theta-testnet-001 --a-connection connection-2727 --a-port transfer --b-port transfer
    hermes create channel --a-chain theta-testnet-001 --a-connection connection-2727 --a-port transfer --b-port transfer

    You should be seeing a similar output to this:

    bash
    SUCCESS Channel {
    +    ordering: Unordered,
    +    a_side: ChannelSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "theta-testnet-001",
    +                version: 0,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-2382",
    +        ),
    +        connection_id: ConnectionId(
    +            "connection-2727",
    +        ),
    +        port_id: PortId(
    +            "transfer",
    +        ),
    +        channel_id: Some(
    +            ChannelId(
    +                "channel-3152",
    +            ),
    +        ),
    +        version: None,
    +    },
    +    b_side: ChannelSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "mocha-4",
    +                version: 4,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-0",
    +        ),
    +        connection_id: ConnectionId(
    +            "connection-0",
    +        ),
    +        port_id: PortId(
    +            "transfer",
    +        ),
    +        channel_id: Some(
    +            ChannelId(
    +                "channel-0",
    +            ),
    +        ),
    +        version: None,
    +    },
    +    connection_delay: 0ns,
    +}
    SUCCESS Channel {
    +    ordering: Unordered,
    +    a_side: ChannelSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "theta-testnet-001",
    +                version: 0,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-2382",
    +        ),
    +        connection_id: ConnectionId(
    +            "connection-2727",
    +        ),
    +        port_id: PortId(
    +            "transfer",
    +        ),
    +        channel_id: Some(
    +            ChannelId(
    +                "channel-3152",
    +            ),
    +        ),
    +        version: None,
    +    },
    +    b_side: ChannelSide {
    +        chain: BaseChainHandle {
    +            chain_id: ChainId {
    +                id: "mocha-4",
    +                version: 4,
    +            },
    +            runtime_sender: Sender { .. },
    +        },
    +        client_id: ClientId(
    +            "07-tendermint-0",
    +        ),
    +        connection_id: ConnectionId(
    +            "connection-0",
    +        ),
    +        port_id: PortId(
    +            "transfer",
    +        ),
    +        channel_id: Some(
    +            ChannelId(
    +                "channel-0",
    +            ),
    +        ),
    +        version: None,
    +    },
    +    connection_delay: 0ns,
    +}

    Congratulations!

    You have successfully created a new IBC connection between two networks.

    Configure channels in Hermes

    Now that we have created new connections and opened channels, we need to edit config.toml again and add the newly created channels, or use the already existing ones.

    For mocha-4 add:

    bash
    [chains.packet_filter]
    +policy = 'allow'
    +list = [
    +  ['transfer', 'channel-0'], # theta-testnet-001
    +]
    [chains.packet_filter]
    +policy = 'allow'
    +list = [
    +  ['transfer', 'channel-0'], # theta-testnet-001
    +]

    For theta-testnet-001 add:

    bash
    [chains.packet_filter]
    +policy = 'allow'
    +list = [
    +  ['transfer', 'channel-3108'], # mocha-4
    +]
    [chains.packet_filter]
    +policy = 'allow'
    +list = [
    +  ['transfer', 'channel-3108'], # mocha-4
    +]

    Start the relayer

    Start the relayer via hermes start

    Transfer

    The Celestia state machine is built with the IBC transfer module, allowing for the native Celestia token to be transferred to any other IBC enabled chain. Transfer can be initialized through the celestia-appd CLI. Information can be found via the help label as follows:

    bash
    celestia-appd tx ibc-transfer transfer --help
    celestia-appd tx ibc-transfer transfer --help

    Token filter

    The transfer module uses a token filter middleware which serves to prevent non-native Celestia tokens from being on Celestia. If a user is to try to send a token from another chain across, it will be simply rejected and the token returned back to the user.

    + + + + \ No newline at end of file diff --git a/nodes/instantiate-testnet.html b/nodes/instantiate-testnet.html new file mode 100644 index 00000000000..4cce28fddef --- /dev/null +++ b/nodes/instantiate-testnet.html @@ -0,0 +1,65 @@ + + + + + + Celestia App network instantiation guide | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Celestia App network instantiation guide

    This guide is for helping instantiate a new testnetwork and following the correct steps to do so with Celestia App. You should only follow this guide if you want to experiment with your own Celestia test network (testnet) or if you want to test out new features to build as a core developer.

    Hardware requirements

    You will need to follow hardware requirements.

    Setup dependencies

    You will need to setup dependencies by following the guide.

    celestia-app installation

    You will need to install celestia-app by following the guide.

    Spin up a Celestia testnet

    If you want to spin up a quick testnet with your friends, you can follow these steps. Unless otherwise noted, every step must be done by everyone who wants to participate in this testnet.

    Optional: Reset working directory

    If you have already initialized a working directory for celestia-appd in the past, you must clean up before reinitializing a new directory. You can do so by running the following command:

    sh
    celestia-appd tendermint unsafe-reset-all --home $HOME/.celestia-app
    celestia-appd tendermint unsafe-reset-all --home $HOME/.celestia-app

    Initialize a working directory

    Run the following command:

    sh
    VALIDATOR_NAME=validator1
    +CHAIN_ID=testnet
    +celestia-appd init $VALIDATOR_NAME --chain-id $CHAIN_ID
    VALIDATOR_NAME=validator1
    +CHAIN_ID=testnet
    +celestia-appd init $VALIDATOR_NAME --chain-id $CHAIN_ID
    • The value we will use for $VALIDATOR_NAME is validator1 but you should choose your own node name.
    • The value we will use for $CHAIN_ID is testnet. The $CHAIN_ID must remain the same for everyone participating in this network.

    Create a new key

    Next, run the following command:

    sh
    KEY_NAME=validator
    +celestia-appd keys add $KEY_NAME --keyring-backend test
    KEY_NAME=validator
    +celestia-appd keys add $KEY_NAME --keyring-backend test

    This will create a new key, with a name of your choosing. Save the output of this command somewhere; you'll need the address generated here later. Here, we set the value of our key $KEY_NAME to validator for demonstration.

    Add genesis account KeyName

    Run the following command:

    sh
    TIA_AMOUNT="10000000utia"
    +celestia-appd add-genesis-account $KEY_NAME $TIA_AMOUNT --keyring-backend test
    TIA_AMOUNT="10000000utia"
    +celestia-appd add-genesis-account $KEY_NAME $TIA_AMOUNT --keyring-backend test

    Here $KEY_NAME is the same key name as before.

    Optional: Adding other validators

    If other participants in your testnet also want to be validators, repeat the command above with the specific amount for their public keys.

    Once all the validators are added, the genesis.json file is created. You need to share it with all other validators in your testnet in order for everyone to proceed with the following step.

    You can find the genesis.json at $HOME/.celestia-app/config/genesis.json

    Create the genesis transaction for new chain

    Run the following command:

    sh
    STAKING_AMOUNT=9000000utia
    +celestia-appd gentx $KEY_NAME $STAKING_AMOUNT --chain-id $CHAIN_ID \
    +  --keyring-backend test
    STAKING_AMOUNT=9000000utia
    +celestia-appd gentx $KEY_NAME $STAKING_AMOUNT --chain-id $CHAIN_ID \
    +  --keyring-backend test

    This will create the genesis transaction for your new chain. Here $STAKING_AMOUNT should be at least 9000000utia. If you provide too much or too little, you will encounter an error when starting your node.

    You will find the generated gentx JSON file inside $HOME/.celestia-app/config/gentx/gentx-$KEY_NAME.json

    NOTE

    If you have other validators in your network, they need to also run the above command with the genesis.json file you shared with them in the previous step.

    Creating the genesis JSON file

    Once all participants have submitted their gentx JSON files to you, you will pull all those gentx files inside the following directory: $HOME/.celestia-appd/config/gentx and use them to create the final genesis.json file.

    Once you added the gentx files of all the participants, run the following command:

    sh
    celestia-appd collect-gentxs
    celestia-appd collect-gentxs

    This command will look for the gentx files in this repo which should be moved to the following directory $HOME/.celestia-app/config/gentx.

    It will update the genesis.json file after in this location $HOME/.celestia-app/config/genesis.json which now includes the gentx of other participants.

    You should then share this final genesis.json file with all the other participants who must add it to their $HOME/.celestia-app/config directory.

    Everyone must ensure that they replace their existing genesis.json file with this new one created.

    Modify your config file

    Open the following file $HOME/.celestia-app/config/config.toml to modify it.

    Inside the file, add the other participants by modifying the following line to include other participants as persistent peers:

    text
    # Comma separated list of nodes to keep persistent connections to
    +persistent_peers = "[validator_address]@[ip_address]:[port],[validator_address]@[ip_address]:[port]"
    # Comma separated list of nodes to keep persistent connections to
    +persistent_peers = "[validator_address]@[ip_address]:[port],[validator_address]@[ip_address]:[port]"

    Add your node as a persistent peer

    The following allows you to share your node as a persistent peer that you can share in the networks repo or with others so other participants can peer with you.

    Run the following command:

    sh
    IP_ADDRESS=$(curl ifconfig.me)
    +NODE_ID=$(celestia-appd tendermint show-node-id)
    +PORT_NUMBER=26656
    IP_ADDRESS=$(curl ifconfig.me)
    +NODE_ID=$(celestia-appd tendermint show-node-id)
    +PORT_NUMBER=26656

    Note that the default port is 26656

    Now you can run the following command to output your validator node address:

    sh
    PEER="$NODE_ID@$IP_ADDRESS:$PORT_NUMBER"
    +echo $PEER
    PEER="$NODE_ID@$IP_ADDRESS:$PORT_NUMBER"
    +echo $PEER

    The output is your validator node address which you can share with other validators so they can peer with you.

    Instantiate the network

    You can start your node by running the following command:

    sh
    celestia-appd start
    celestia-appd start

    TIP

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    Now you have a new Celestia testnet to play around with!

    + + + + \ No newline at end of file diff --git a/nodes/light-node.html b/nodes/light-node.html new file mode 100644 index 00000000000..ba30e54baa6 --- /dev/null +++ b/nodes/light-node.html @@ -0,0 +1,69 @@ + + + + + + Setting up a Celestia light node | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Setting up a Celestia light node

    This tutorial will guide you through setting up a Celestia light node, which will allow you to perform data availability sampling (DAS) on Celestia's data availability (DA) network.

    Overview of light nodes

    Light nodes ensure data availability. This is the most common way to interact with Celestia networks.

    light-node

    Light nodes have the following behavior:

    1. They listen for ExtendedHeaders, i.e. wrapped “raw” headers, that notify Celestia nodes of new block headers and relevant DA metadata.
    2. They perform DAS on the received headers

    Hardware requirements

    The following minimum hardware requirements are recommended for running a light node:

    • Memory: 500 MB RAM (minimum)
    • CPU: Single Core
    • Disk: 50 GB SSD Storage
    • Bandwidth: 56 Kbps for Download/56 Kbps for Upload

    Setting up your light node

    This tutorial was performed on an Ubuntu Linux 20.04 (LTS) x64 instance machine.

    Set up dependencies on the setting up environment page.

    Install celestia-node

    Install the celestia binary by building and installing celestia-node.

    Initialize the light node

    Run the following command:

    sh
    celestia light init
    celestia light init
    sh
    celestia light init --p2p.network mocha
    celestia light init --p2p.network mocha
    sh
    celestia light init --p2p.network arabica
    celestia light init --p2p.network arabica

    The output in your terminal will show the location of your node store and config. It will also show confirmation that the node store has been initialized.

    Start the light node

    Start the light node with a connection to a validator node's gRPC endpoint (which is usually exposed on port 9090):

    In order for access to the ability to get and submit state-related information, such as the ability to submit PayForBlobs transactions, or query for the node's account balance, a gRPC endpoint of a validator (core) node must be passed as directed below.

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    To start the light node with a connection to a validator node's gRPC endpoint (which is usually exposed on port 9090):

    sh
    celestia light start --core.ip <URI> --p2p.network <network>
    celestia light start --core.ip <URI> --p2p.network <network>

    TIP

    You do not need to declare a network for Mainnet Beta. Refer to the chain ID section on the troubleshooting page for more information

    Using an RPC of your own, or one from the list on the Mocha testnet page or list on the Arabica devnet page, start your node.

    For example, your command might look something like this for Mocha:

    sh
    celestia light start --core.ip rpc-mocha.pops.one --p2p.network mocha
    celestia light start --core.ip rpc-mocha.pops.one --p2p.network mocha

    Or for Arabica:

    sh
    celestia light start --core.ip validator-1.celestia-arabica-11.com \
    +    --p2p.network arabica
    celestia light start --core.ip validator-1.celestia-arabica-11.com \
    +    --p2p.network arabica

    Keys and wallets

    You can create your key for your node by running the following command with the cel-key utility in the celestia-node directory:

    sh
    ./cel-key add <key-name> --keyring-backend test \
    +    --node.type light --p2p.network <network>
    ./cel-key add <key-name> --keyring-backend test \
    +    --node.type light --p2p.network <network>

    You can start your light node with the key created above by running the following command:

    sh
    celestia light start --keyring.accname my_celes_key \
    +    --core.ip consensus.lunaroasis.net
    celestia light start --keyring.accname my_celes_key \
    +    --core.ip consensus.lunaroasis.net
    sh
    celestia light start --keyring.accname my_celes_key \
    +    --core.ip rpc-mocha.pops.one --p2p.network mocha
    celestia light start --keyring.accname my_celes_key \
    +    --core.ip rpc-mocha.pops.one --p2p.network mocha
    sh
    celestia light start --keyring.accname my_celes_key \
    +    --core.ip validator-1.celestia-arabica-11.com \
    +    --p2p.network arabica
    celestia light start --keyring.accname my_celes_key \
    +    --core.ip validator-1.celestia-arabica-11.com \
    +    --p2p.network arabica

    Once you start the light node, a wallet key will be generated for you. You will need to fund that address with testnet tokens to pay for PayForBlob transactions.

    You can find the address using the RPC CLI or by running the following command in the celestia-node directory:

    sh
    ./cel-key list --node.type light --keyring-backend test \
    +    --p2p.network <network>
    ./cel-key list --node.type light --keyring-backend test \
    +    --p2p.network <network>

    Testnet tokens

    You have two networks to get testnet tokens from:

    You can request funds to your wallet address using the following command in Discord:

    console
    $request <CELESTIA-ADDRESS>
    $request <CELESTIA-ADDRESS>

    Where <CELESTIA-ADDRESS> is the celestia1****** address generated when you created the wallet.

    Optional: run the light node with a custom key

    In order to run a light node using a custom key:

    1. The custom key must exist inside the celestia light node directory at the correct path (default: ~/.celestia-light/keys/keyring-test)
    2. The name of the custom key must be passed upon start, like so:
    sh
    celestia light start --core.ip <URI> \
    +    --keyring.accname <name-of-custom-key> \
    celestia light start --core.ip <URI> \
    +    --keyring.accname <name-of-custom-key> \
    sh
    celestia light start --core.ip <URI> \
    +    --keyring.accname <name-of-custom-key> \
    +    --p2p.network arabica
    celestia light start --core.ip <URI> \
    +    --keyring.accname <name-of-custom-key> \
    +    --p2p.network arabica
    sh
    celestia light start --core.ip <URI> \
    +    --keyring.accname <name-of-custom-key> \
    +    --p2p.network mocha
    celestia light start --core.ip <URI> \
    +    --keyring.accname <name-of-custom-key> \
    +    --p2p.network mocha

    Optional: start light node with SystemD

    Follow the tutorial on setting up the light node as a background process with SystemD.

    Data availability sampling

    With your light node running, you can check out this tutorial on submitting PayForBlob transactions.

    + + + + \ No newline at end of file diff --git a/nodes/mainnet.html b/nodes/mainnet.html new file mode 100644 index 00000000000..88a01a16ff7 --- /dev/null +++ b/nodes/mainnet.html @@ -0,0 +1,50 @@ + + + + + + Mainnet Beta | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Mainnet Beta

    Mainnet Beta

    Welcome to the guide for Celestia’s Mainnet Beta, the production network that marks the pinnacle of Celestia’s evolution since its inception in 2019. This network is where all components of the Celestia ecosystem come to life in a real-world environment.

    Mainnet Beta is the culmination of rigorous community testing, upgrades, and feedback. It serves as the platform for deploying mainnet rollups and applications.

    Network stability and upgrades

    Mainnet Beta is a stable network, but will still receive updates and improvements. Any changes or upgrades will be coordinated with node operators and the broader Celestia community to ensure seamless integration and minimal service interruptions.

    As we step into unexplored territories with groundbreaking technologies like data availability sampling, it's crucial to remember that Mainnet Beta remains experimental at this stage. While the network is live and functional, users may encounter occasional instability or reduced performance.

    Network details

    DetailValue
    Chain IDcelestia
    Genesis hash6BE39EFD10BA412A9DB5288488303F5DD32CF386707A5BEF33617F4C43301872
    Genesis file https://github.com/celestiaorg/networks/blob/master/celestia/genesis.json
    Peers file https://github.com/celestiaorg/networks/blob/master/celestia/peers.txt
    Validators 100

    Software version numbers

    SoftwareVersion
    celestia-nodev0.13.2
    celestia-appv1.7.0

    Network parameters

    Full network parameters, such as max bytes, can be found in the celestia-app specifications.

    CIP-13 has been drafted to create a living document for these parameters as a part of the CIP process.

    Maximum bytes

    There is a hard limit on the total blob size in a transaction, which is determined by the effective maximum square size. Given that the current governance maximum square size is 64, the total blob size in a transaction must be slightly less than ~2 MiB, or 1,973,786 bytes to be exact.

    The following provides an approximation of the maximum block size:

    • The maximum square size is 64x64, which gives us 4096 shares.
    • One share is reserved for the PFB transaction, leaving us with 4095 shares.
    • The first sparse share has 478 bytes available, and the remaining sparse shares have 482 bytes each.

    This can be calculated as follows:

    Total Bytes=(1×478bytes)+(4094×482bytes)=1,973,786bytes

    Please note that there isn't a precise upper bound on the maximum total blob size. It depends on several factors:

    • The maximum square size, which is determined by a governance parameter and a versioned constant.
    • The maximum bytes in a block, which is determined by a governance parameter and a hard-coded constant in CometBFT.
    • The number of shares occupied by the PFB transaction share.

    These factors can cause the maximum total blob size that can be included in one block to vary.

    Integrations

    This guide contains the relevant sections for how to connect to Mainnet Beta, depending on the type of node you are running. Your best approach to participating is to first determine which node you would like to run. Each node’s guide will link to the relevant network in order to show you how to connect to them. Learn about the different endpoint types in the Cosmos SDK documentation.

    Here is a list of options of the types of nodes you can run in order to participate in Mainnet Beta:

    Consensus nodes

    Consensus RPC endpoints

    • public-celestia-rpc.numia.xyz
    • celestia-rpc.mesa.newmetric.xyz
    • rpc.celestia.pops.one
    • rpc.lunaroasis.net
    • rpc.celestia.nodestake.top
    • celestia-rpc.brightlystake.com
    • celestia-rpc.spidey.services
    • rpc-celestia.contributiondao.com
    • celestia.rpc.stakin-nodes.com
    • celestia.cumulo.org.es
    • rpc-celestia.mzonder.com
    • rpc-celestia-01.stakeflow.io
    • rpc-celestia.alphab.ai
    • rpc-celestia-full.avril14th.org
    • celestia-rpc.easy2stake.com
    • celestia.rpc.kjnodes.com
    • celestia-rpc.0xcryptovestor.com
    • rpc-celestia-mainnet.trusted-point.com
    • celestia.rpc.archives.validao.xyz
    • rpc-archive.celestia.bitszn.com
    • celestia-rpc.f5nodes.com
    • celestia-rpc.chainode.tech:33373
    • https://rpc-celestia.staker.space/
    • https://celestia-rpc.noders.services/

    API endpoints

    • public-celestia-lcd.numia.xyz
    • celestia-rest.mesa.newmetric.xyz
    • api.celestia.pops.one
    • api.lunaroasis.net
    • api.celestia.nodestake.top
    • celestia-rpc.brightlystake.com/api
    • celestia-api.spidey.services
    • api-celestia.contributiondao.com
    • celestia.rest.stakin-nodes.com
    • celestia.api.cumulo.org.es
    • api-celestia.mzonder.com
    • api-celestia-01.stakeflow.io
    • api-celestia.alphab.ai
    • api-celestia-full.avril14th.org
    • celestia-lcd.easy2stake.com
    • celestia.api.kjnodes.com
    • api-celestia-mainnet.trusted-point.com
    • celestia.rest.archives.validao.xyz
    • api-archive.celestia.bitszn.com
    • celestia-api.f5nodes.com
    • https://celestia-api.chainode.tech
    • https://api-celestia.staker.space/
    • https://celestia-api.noders.services/

    gRPC endpoints

    • public-celestia-grpc.numia.xyz
    • celestia-grpc.mesa.newmetric.xyz
    • grpc.celestia.pops.one
    • grpc.lunaroasis.net:443
    • grpc.celestia.nodestake.top
    • celestia-rpc.brightlystake.com:9090
    • celestia-grpc.spidey.services
    • grpc-celestia.contributiondao.com
    • celestia.grpc.stakin-nodes.com:443
    • celestia.grpc.cumulo.org.es:443
    • grpc-celestia.mzonder.com:443
    • grpc-celestia-01.stakeflow.io:15002
    • rpc-celestia.alphab.ai:9090
    • grpc-celestia-full.avril14th.org
    • celestia.grpc.kjnodes.com:443
    • grpc-celestia-mainnet.trusted-point.com:9095
    • celestia.grpc.archives.validao.xyz:9090
    • gprc-archive.celestia.bitszn.com
    • celestia-grpc.f5nodes.com:9390
    • celestia-grpc.chainode.tech:443
    • https://grpc-celestia.staker.space/
    • celestia-grpc.noders.services:11090

    WebSocket endpoints

    • wss://celestia-ws.chainode.tech:33373/websocket

    Data availability nodes

    Data availability (DA) RPC endpoints

    These RPC endpoints for DA nodes are to provide state access for querying the chain’s state and broadcasting transactions (balances, blobs, etc.) to the Celestia network. For users, they will need to provide a –core.ip string from a consensus node’s URL or IP that populates 2 ports for 2 types (RPC and gRPC, at ports 26657 and 9090, respectively) to their respective DA node.

    TIP

    bash
    celestia <da_type> start --core.ip <url> –core.rpc.port <port> \
    +    –core.grpc.port <port>
    celestia <da_type> start --core.ip <url> –core.rpc.port <port> \
    +    –core.grpc.port <port>

    Bridge nodes

    Not all RPC endpoints guarantee the full block history. Find an archive endpoint on the community dashboard or run your own full consensus node with no pruning for your bridge node.

    RPCs for DA nodes to initialise or start your celestia-node to Mainnet Beta with:

    • public-celestia-consensus.numia.xyz
      • gRPC: port 9090
      • RPC: port 26657
    • celestia-consensus.mesa.newmetric.xyz
      • gRPC: port 9090
      • RPC: port 26657
    • rpc.celestia.pops.one
      • gRPC: port 9090
      • RPC: port 26657
    • consensus.lunaroasis.net
      • gRPC: port 9090
      • RPC: port 26657
    • rpc-celestia.alphab.ai
      • gRPC: port 9090
      • RPC: port 26657

    DA full and light nodes might have troubles connecting to the networks, so you can check out this Grafana dashboard to see health/uptime status of DA bootstrappers (now celestia network only).

    You can find the status of these endpoints.

    Archival DA RPC endpoints

    By default, light nodes prune recent data to save on storage space. Archival data availability (DA) nodes store the entire history of the chain without pruning any data so all data available data is retrievable. You can read more about light vs archival nodes.

    Grove archival endpoints

    You can provision your own Celestia Archival endpoint on Grove. Learn more about Celestia on Grove, or find the fully supported spec.

    There is a sandbox you can leverage for testing straight in your browser:

    grove-sandbox

    Explorers

    There are multiple explorers you can use for Mainnet Beta:

    Analytics

    The following websites provide analytics for Celestia:

    Network upgrades

    Join the Community Telegram announcement channel for network upgrades.

    + + + + \ No newline at end of file diff --git a/nodes/mocha-testnet.html b/nodes/mocha-testnet.html new file mode 100644 index 00000000000..cb8abd38e4b --- /dev/null +++ b/nodes/mocha-testnet.html @@ -0,0 +1,48 @@ + + + + + + Mocha testnet | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Mocha testnet

    mocha-testnet

    This guide contains the relevant sections for how to connect to Mocha, depending on the type of node you are running. Mocha testnet is designed to help validators test out their infrastructure and node software. Developers are encouraged to deploy their sovereign rollups on Mocha, but we also recommend Arabica devnet for that as it is designed for development purposes.

    Mocha is a milestone in Celestia, allowing everyone to test out core functionalities on the network. Read the announcement. Your best approach to participating is to first determine which node you would like to run. Each node's guide will link to the relevant networks, to show you how to connect to them.

    You have a list of options on the types of nodes you can run to participate in Mocha:

    Consensus:

    Data Availability:

    Select the type of node you would like to run and follow the instructions on each respective page. Whenever you are asked to select the type of network you want to connect to in those guides, select Mocha to refer to the correct instructions on this page on how to connect to Mocha.

    Network details

    DetailValue
    Chain IDmocha-4
    Genesis hashB93BBE20A0FBFDF955811B6420F8433904664D45DB4BF51022BE4200C1A1680D
    Genesis file https://github.com/celestiaorg/networks/blob/master/mocha-4/genesis.json
    Peers file https://github.com/celestiaorg/networks/blob/master/mocha-4/peers.txt
    Validators 100

    Software version numbers

    SoftwareVersion
    celestia-nodev0.13.3
    celestia-appv1.7.0

    RPC for DA bridge, full, and light nodes

    These RPC endpoints for DA nodes are to provide state access for querying the chain’s state and broadcasting transactions (balances, blobs, etc.) to the Celestia network. For users, they will need to provide a –core.ip string from a consensus node’s URL or IP that populates 2 ports for 2 types (RPC and gRPC, at ports 26657 and 9090, respectively) to their respective DA node.

    Bridge nodes

    Mentioned below RPC endpoints do not guarantee you the download of full blocks from them. We advise that if you are running a bridge node, that you also run a local full consensus node in order to download full blocks from it.

    • public-celestia-mocha4-consensus.numia.xyz
    • mocha-4-consensus.mesa.newmetric.xyz
    • full.consensus.mocha-4.celestia-mocha.com
    • consensus-full-mocha-4.celestia-mocha.com
    • rpc-mocha.pops.one

    RPC endpoints

    The RPC endpoint is to allow users to interact with Celestia's nodes by querying the node's state and broadcasting transactions on the Celestia network. The default port is 26657.

    • public-celestia-mocha4-consensus.numia.xyz:26657
    • mocha-4-consensus.mesa.newmetric.xyz:26657
    • rpc.celestia-mocha.com
    • rpc-2.celestia-mocha.com
    • celestia-testnet-rpc.f5nodes.com
    • celestia-testnet.brightlystake.com
    • rpc-celestia-mocha.architectnodes.com
    • rpc-celestia-mocha.trusted-point.com
    • rpc-celestia-testnet-01.stakeflow.io
    • mocha.celestia.rpc.cumulo.me
    • rpc-mocha-4.spidey.services
    • rpc-mocha-full.avril14th.org
    • rpc.mocha.bitszn.com
    • celestia-t-rpc.noders.services/

    API endpoints

    The API endpoint is to allow users to interact with the REST API in Cosmos SDK which is implemented using gRPC-gateway, which exposes gRPC endpoints as REST endpoints. This allows for communication with the node using REST calls, which can be useful if the client does not support gRPC or HTTP2. The default port is 1317.

    gRPC endpoints

    The gRPC endpoint is to allow users to interact with a Celestia Node using gRPC, a modern open-source and high-performance RPC framework. The default port is 9090. In the Cosmos SDK, gRPC is used to define state queries and broadcast transactions.

    • public-celestia-mocha4-consensus.numia.xyz:9090
    • mocha-4-consensus.mesa.newmetric.xyz:9090
    • grpc-mocha.pops.one
    • grpc.celestia-mocha.com:443
    • grpc-2.celestia-mocha.com:443
    • full.consensus.mocha-4.celestia-mocha.com:9090
    • consensus-full-mocha-4.celestia-mocha.com:9090
    • celestia-testnet-grpc.f5nodes.com
    • celestia-testnet.brightlystake.com:9390
    • grpc-celestia-mocha.architectnodes.com:1443
    • grpc-celestia-mocha.trusted-point.com:9099
    • grpc-celestia-testnet-01.stakeflow.io:16002
    • mocha.grpc.cumulo.me:443
    • grpc-mocha-4.spidey.services
    • grpc-mocha-full.avril14th.org
    • grpc.mocha.bitszn.com
    • celestia-grpc.noders.services:21090

    Bridge and full node endpoints

    The endpoints below are for bridge and full nodes only. They can be used to find bootstrapper peers in the p2p network.

    Bridge node 1:

    • da-bridge-mocha-4.celestia-mocha.com
    • bridge-mocha-4.da.celestia-mocha.com

    Bridge node 2:

    • da-bridge-mocha-4-2.celestia-mocha.com
    • bridge-mocha-4-2.da.celestia-mocha.com

    Full node 1:

    • da-full-1-mocha-4.celestia-mocha.com
    • full-1-mocha-4.da.celestia-mocha.com

    Full node 2:

    • da-full-2-mocha-4.celestia-mocha.com
    • full-2-mocha-4.da.celestia-mocha.com

    Mocha testnet faucet

    WARNING

    USING THIS FAUCET DOES NOT ENTITLE YOU TO ANY AIRDROP OR OTHER DISTRIBUTION OF MAINNET CELESTIA TOKENS. THERE ARE NO PUBLIC SALES OF ANY MAINNET CELESTIA TOKENS.

    You can request from Mocha testnet Faucet on the #mocha-faucet channel on Celestia's Discord server with the following command:

    text
    $request <CELESTIA-ADDRESS>
    $request <CELESTIA-ADDRESS>

    Where <CELESTIA-ADDRESS> is a celestia1****** generated address.

    NOTE

    Faucet has a limit of 10 tokens per week per address/Discord ID.

    Explorers

    There are several explorers you can use for Mocha:

    Network upgrades

    Join our Telegram announcement channel for network upgrades.

    + + + + \ No newline at end of file diff --git a/nodes/overview.html b/nodes/overview.html new file mode 100644 index 00000000000..af0ea94a797 --- /dev/null +++ b/nodes/overview.html @@ -0,0 +1,45 @@ + + + + + + Overview to running nodes on Celestia | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Overview to running nodes on Celestia

    There are many ways you can participate in the Celestia networks.

    Celestia node operators can run several options on the network.

    Consensus:

    • Validator node: This type of node participates in consensus by producing and voting on blocks.
    • Full consensus node: A celestia-app Full node to sync blockchain history.

    Data Availability:

    • Bridge node: This node bridges blocks between the Data-Availability network and the Consensus network.
    • Full storage node: This node stores all the data but does not connect to Consensus.
    • Light node: Light clients conduct data availability sampling on the Data Availability network.

    You can learn more about how to set up each different node by going through each tutorial guide.

    Banner

    Please provide any feedback on the tutorials and guides. If you notice a bug or issue, feel free to make a pull request or write up a Github issue!

    + + + + \ No newline at end of file diff --git a/nodes/participate.html b/nodes/participate.html new file mode 100644 index 00000000000..a147486cc29 --- /dev/null +++ b/nodes/participate.html @@ -0,0 +1,52 @@ + + + + + + Participate in the Celestia networks | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Participate in the Celestia networks

    Mainnet Beta

    Celestia’s Mainnet Beta is the production network for deploying mainnet rollups and applications. This marks the culmination of years of development and community testing. While the network is stable and continues to receive updates, it remains experimental and users may experience occasional instability or reduced performance.

    Compatible software versions for Mainnet Beta

    SoftwareVersion
    celestia-nodev0.13.2
    celestia-appv1.7.0

    Testnets

    Celestia currently has two existing testnets that you can participate in:

    Arabica Devnet

    Arabica devnet is a devnet focused on developers who want to deploy sovereign rollups on the latest changes from Celestia's codebase. Arabica will be updated frequently and might be unstable at times given new updates. Validators won't be able to validate on Arabica as it is not designed for validators to participate.

    Compatible software versions for Arabica devnet

    SoftwareVersion
    celestia-nodev0.13.3
    celestia-appv1.7.0

    Mocha testnet

    Mocha testnet is a testnet focused on enabling validators to test out their infrastructure by running nodes connected to the network. Developers can also deploy sovereign rollups on Mocha, it just will always be behind Arabica as Mocha upgrades are slower given they need to be done via hardforks in coordination with the validator community on Mocha.

    Compatible software versions for Mocha testnet

    SoftwareVersion
    celestia-nodev0.13.3
    celestia-appv1.7.0

    Network upgrades

    Join our Telegram announcement channel for network upgrades.

    + + + + \ No newline at end of file diff --git a/nodes/quick-start.html b/nodes/quick-start.html new file mode 100644 index 00000000000..560089c65d2 --- /dev/null +++ b/nodes/quick-start.html @@ -0,0 +1,45 @@ + + + + + + Quick start guide | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Quick start guide

    In this section, we show you how to get started with installing the needed libraries and packages in Celestia to help you run a node on Celestia.

    Celestia Node

    Install celestia-node allows you to get started with running a light node and do data availability sampling.

    Light nodes are the best nodes to test out initially if you are new to participating in Celestia.

    celestia-node client also allows you to run other types of data availability (DA) nodes like bridge and full DA storage nodes, which will be covered in later sections.

    Celestia App

    Install celestia-app allows you to get started running a consensus node.

    celestia-app is the software that allows you to run validator nodes and also provide RPC endpoints.

    celestia-app covers the consensus layer, while celestia-node covers the DA layer.

    Getting started

    As covered in the previous section, Celestia offers two different test networks, Arabica devnet and Mocha testnet.

    If you are planning to run a light node, it is recommended to use Arabica, which you will find options to connecting to in the later sections.

    If you plan on running a validator, your only option is to run your node on Mocha.

    In this quick start guide, we will go over installing both of the software clients: celestia-node and celestia-app.

    NOTE

    If you just want to run a light node, you don't need to install celestia-app and can skip that part.

    Proceed to the next section in order to get started.

    + + + + \ No newline at end of file diff --git a/nodes/systemd.html b/nodes/systemd.html new file mode 100644 index 00000000000..f56ccbb0697 --- /dev/null +++ b/nodes/systemd.html @@ -0,0 +1,169 @@ + + + + + + Setting up your node as a background process with SystemD | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Setting up your node as a background process with SystemD

    SystemD is a daemon service useful for running applications as background processes.

    Consensus nodes

    If you are running a validator or full consensus node, here are the steps to setting up celestia-appd as a background process.

    Start the celestia-app with SystemD

    SystemD is a daemon service useful for running applications as background processes.

    Create Celestia-App systemd file:

    sh
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-appd.service
    +[Unit]
    +Description=celestia-appd Cosmos daemon
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia-appd) start
    +Restart=on-failure
    +RestartSec=3
    +LimitNOFILE=65535
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-appd.service
    +[Unit]
    +Description=celestia-appd Cosmos daemon
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia-appd) start
    +Restart=on-failure
    +RestartSec=3
    +LimitNOFILE=65535
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF

    If the file was created successfully you will be able to see its content:

    sh
    cat /etc/systemd/system/celestia-appd.service
    cat /etc/systemd/system/celestia-appd.service

    Enable and start celestia-appd daemon:

    sh
    sudo systemctl enable celestia-appd
    +sudo systemctl start celestia-appd
    sudo systemctl enable celestia-appd
    +sudo systemctl start celestia-appd

    Check if daemon has been started correctly:

    sh
    sudo systemctl status celestia-appd
    sudo systemctl status celestia-appd

    Check daemon logs in real time:

    sh
    sudo journalctl -u celestia-appd.service -f
    sudo journalctl -u celestia-appd.service -f

    To check if your node is in sync before going forward:

    sh
    curl -s localhost:26657/status | jq .result | jq .sync_info
    curl -s localhost:26657/status | jq .result | jq .sync_info

    Make sure that you have "catching_up": false, otherwise leave it running until it is in sync.

    Data availability nodes

    Celestia full storage node

    Create Celestia full storage node systemd file:

    sh
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-full.service
    +[Unit]
    +Description=celestia-full Cosmos daemon
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia) full start
    +Restart=on-failure
    +RestartSec=3
    +LimitNOFILE=1400000
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-full.service
    +[Unit]
    +Description=celestia-full Cosmos daemon
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia) full start
    +Restart=on-failure
    +RestartSec=3
    +LimitNOFILE=1400000
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF

    If the file was created successfully you will be able to see its content:

    sh
    cat /etc/systemd/system/celestia-full.service
    cat /etc/systemd/system/celestia-full.service

    Enable and start celestia-full daemon:

    sh
    sudo systemctl enable celestia-full
    +sudo systemctl start celestia-full && sudo journalctl -u \
    +celestia-full.service -f
    sudo systemctl enable celestia-full
    +sudo systemctl start celestia-full && sudo journalctl -u \
    +celestia-full.service -f

    You should be seeing logs coming through of the full storage node syncing.

    Celestia bridge node

    Create Celestia Bridge systemd file:

    sh
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-bridge.service
    +[Unit]
    +Description=celestia-bridge Cosmos daemon
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia) bridge start
    +Restart=on-failure
    +RestartSec=3
    +LimitNOFILE=1400000
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-bridge.service
    +[Unit]
    +Description=celestia-bridge Cosmos daemon
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia) bridge start
    +Restart=on-failure
    +RestartSec=3
    +LimitNOFILE=1400000
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF

    If the file was created successfully you will be able to see its content:

    sh
    cat /etc/systemd/system/celestia-bridge.service
    cat /etc/systemd/system/celestia-bridge.service

    Enable and start celestia-bridge daemon:

    sh
    sudo systemctl enable celestia-bridge
    +sudo systemctl start celestia-bridge && sudo journalctl -u \
    +celestia-bridge.service -f
    sudo systemctl enable celestia-bridge
    +sudo systemctl start celestia-bridge && sudo journalctl -u \
    +celestia-bridge.service -f

    Now, the Celestia bridge node will start syncing headers and storing blocks from celestia-app.

    NOTE

    At startup, we can see the multiaddress from Celestia bridge node. This is needed for future light node connections and communication between Celestia Bridge Nodes

    Example:

    sh
    NODE_IP=<URI>]
    +/ip4/$NODE_IP/tcp/2121/p2p/12D3KooWD5wCBJXKQuDjhXFjTFMrZoysGVLtVht5hMoVbSLCbV22
    NODE_IP=<URI>]
    +/ip4/$NODE_IP/tcp/2121/p2p/12D3KooWD5wCBJXKQuDjhXFjTFMrZoysGVLtVht5hMoVbSLCbV22

    You should be seeing logs coming through of the bridge node syncing.

    Celestia light node

    Start the light node as daemon process in the background

    sh
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-lightd.service
    +[Unit]
    +Description=celestia-lightd light node
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia) light start --core.ip <URI>
    +Restart=on-failure
    +RestartSec=3
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF
    sudo tee <<EOF >/dev/null /etc/systemd/system/celestia-lightd.service
    +[Unit]
    +Description=celestia-lightd light node
    +After=network-online.target
    +
    +[Service]
    +User=$USER
    +ExecStart=$(which celestia) light start --core.ip <URI>
    +Restart=on-failure
    +RestartSec=3
    +
    +[Install]
    +WantedBy=multi-user.target
    +EOF

    If the file was created successfully you will be able to see its content:

    sh
    cat /etc/systemd/system/celestia-lightd.service
    cat /etc/systemd/system/celestia-lightd.service

    Enable and start celestia-lightd daemon:

    sh
    sudo systemctl enable celestia-lightd
    +sudo systemctl start celestia-lightd
    sudo systemctl enable celestia-lightd
    +sudo systemctl start celestia-lightd

    Check if daemon has been started correctly:

    sh
    sudo systemctl status celestia-lightd
    sudo systemctl status celestia-lightd

    Check daemon logs in real time:

    sh
    sudo journalctl -u celestia-lightd.service -f
    sudo journalctl -u celestia-lightd.service -f

    Now, the Celestia light node will start syncing headers. After sync is finished, light node will do Data Availability Sampling (DAS) from the bridge node.

    + + + + \ No newline at end of file diff --git a/nodes/validator-node.html b/nodes/validator-node.html new file mode 100644 index 00000000000..62eafffea7b --- /dev/null +++ b/nodes/validator-node.html @@ -0,0 +1,128 @@ + + + + + + Setting up a Celestia validator node | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Setting up a Celestia validator node

    This tutorial will guide you through setting up a validator node on Celestia. Validator nodes allow you to participate in consensus in the Celestia network.

    validator node

    Hardware requirements

    The following hardware minimum requirements are recommended for running a validator node:

    • Memory: 8 GB RAM
    • CPU: 6 cores
    • Disk: 500 GB SSD Storage
    • Bandwidth: 1 Gbps for Download/1 Gbps for Upload

    Setting up a validator node

    The following tutorial is done on an Ubuntu Linux 20.04 (LTS) x64 instance machine.

    First, follow the instructions on setting up a full consensus node.

    Wallet

    Follow the tutorial on creating a wallet.

    Delegate stake to a validator

    Create an environment variable for the address:

    bash
    VALIDATOR_WALLET=<validator-wallet-name>
    VALIDATOR_WALLET=<validator-wallet-name>

    If you want to delegate more stake to any validator, including your own you will need the celesvaloper address of the validator in question. You can run the command below to get the celesvaloper of your local validator wallet in case you want to delegate more to it:

    bash
    celestia-appd keys show $VALIDATOR_WALLET --bech val -a
    celestia-appd keys show $VALIDATOR_WALLET --bech val -a

    After entering the wallet passphrase you should see a similar output:

    bash
    Enter keyring passphrase:
    +celesvaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u43cv6hd
    Enter keyring passphrase:
    +celesvaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u43cv6hd

    To delegate tokens to the celestiavaloper validator, as an example you can run:

    bash
    celestia-appd tx staking delegate \
    +celestiavaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u4q4gx4p 1000000utia \
    +--from=$VALIDATOR_WALLET --chain-id=mocha-4 \
    +--fees=21000utia
    celestia-appd tx staking delegate \
    +celestiavaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u4q4gx4p 1000000utia \
    +--from=$VALIDATOR_WALLET --chain-id=mocha-4 \
    +--fees=21000utia

    If successful, you should see a similar output as:

    console
    code: 0
    +codespace: ""
    +data: ""
    +gas_used: "0"
    +gas_wanted: "0"
    +height: "0"
    +info: ""
    +logs: []
    +raw_log: '[]'
    +timestamp: ""
    +tx: null
    +txhash: <tx-hash>
    code: 0
    +codespace: ""
    +data: ""
    +gas_used: "0"
    +gas_wanted: "0"
    +height: "0"
    +info: ""
    +logs: []
    +raw_log: '[]'
    +timestamp: ""
    +tx: null
    +txhash: <tx-hash>

    You can check if the TX hash went through using the block explorer by inputting the txhash ID that was returned.

    Optional: Deploy the celestia-node

    Running a bridge node is critical to the Celestia network as it enables the data availability and consensus nodes to communicate with one another. It is recommended to support the data availability network, but is not required for celestia-app.

    If you are not running a bridge node, you can skip to run a validator node.

    This section describes part 2 of Celestia validator node setup: running a Celestia bridge node daemon.

    Install celestia-node

    You can follow the tutorial for installing celestia-node

    Initialize the bridge node

    Run the following:

    bash
    celestia bridge init --core.ip <URI>
    celestia bridge init --core.ip <URI>

    TIP

    Refer to the ports section of the celestia-node troubleshooting page for information on which ports are required to be open on your machine.

    If you need a list of RPC endpoints to connect to, you can find the list on the Mocha testnet page or list on the Arabica devnet page.

    Run the bridge node

    Run the following:

    bash
    celestia bridge start
    celestia bridge start

    Optional: start the bridge node with SystemD

    Follow the tutorial on setting up the bridge node as a background process with SystemD.

    You have successfully set up a bridge node that is syncing with the network.

    Run the validator node

    In order to start your validator node, run the following:

    bash
    celestia-appd start
    celestia-appd start

    After completing all the necessary steps, you are now ready to run a validator! In order to create your validator onchain, follow the instructions below. Keep in mind that these steps are necessary ONLY if you want to participate in the consensus.

    Pick a moniker name of your choice! This is the validator name that will show up on public dashboards and explorers. VALIDATOR_WALLET must be the same you defined previously. Parameter --min-self-delegation=1000000 defines the amount of tokens that are self delegated from your validator wallet.

    Now, connect to the network of your choice.

    You have the following option of connecting to list of networks shown below:

    Continuing the validator tutorial, here are the steps to connect your validator to Mocha:

    bash
    MONIKER="your_moniker"
    +VALIDATOR_WALLET="validator"
    +
    +celestia-appd tx staking create-validator \
    +    --amount=1000000utia \
    +    --pubkey=$(celestia-appd tendermint show-validator) \
    +    --moniker=$MONIKER \
    +    --chain-id=mocha-4 \
    +    --commission-rate=0.1 \
    +    --commission-max-rate=0.2 \
    +    --commission-max-change-rate=0.01 \
    +    --min-self-delegation=1000000 \
    +    --from=$VALIDATOR_WALLET \
    +    --keyring-backend=test \
    +    --fees=21000utia \
    +    --gas=220000
    MONIKER="your_moniker"
    +VALIDATOR_WALLET="validator"
    +
    +celestia-appd tx staking create-validator \
    +    --amount=1000000utia \
    +    --pubkey=$(celestia-appd tendermint show-validator) \
    +    --moniker=$MONIKER \
    +    --chain-id=mocha-4 \
    +    --commission-rate=0.1 \
    +    --commission-max-rate=0.2 \
    +    --commission-max-change-rate=0.01 \
    +    --min-self-delegation=1000000 \
    +    --from=$VALIDATOR_WALLET \
    +    --keyring-backend=test \
    +    --fees=21000utia \
    +    --gas=220000

    You will be prompted to confirm the transaction:

    console
    confirm transaction before signing and broadcasting [y/N]: y
    confirm transaction before signing and broadcasting [y/N]: y

    Inputting y should provide an output similar to:

    console
    code: 0
    +codespace: ""
    +data: ""
    +gas_used: "0"
    +gas_wanted: "0"
    +height: "0"
    +info: ""
    +logs: []
    +raw_log: '[]'
    +timestamp: ""
    +tx: null
    +txhash: <tx-hash>
    code: 0
    +codespace: ""
    +data: ""
    +gas_used: "0"
    +gas_wanted: "0"
    +height: "0"
    +info: ""
    +logs: []
    +raw_log: '[]'
    +timestamp: ""
    +tx: null
    +txhash: <tx-hash>

    You should now be able to see your validator from a block explorer

    Submit your validator information

    After starting your node, please submit your node as a seed and peer to the networks repository.

    Optional: Transaction indexer configuration options

    Follow the instructions under transaction indexer configuration options to configure your config.toml file to select which transactions to index.

    Additional resources

    For additional resources, refer to the extra resources for consensus nodessection of the full consensus node page .

    + + + + \ No newline at end of file diff --git a/sitemap.xml b/sitemap.xml new file mode 100644 index 00000000000..b86c16e0560 --- /dev/null +++ b/sitemap.xml @@ -0,0 +1 @@ +https://docs.celestia.org/README2023-10-30T16:12:34.000Zhttps://docs.celestia.org/community/calendar2023-10-23T20:05:00.000Zhttps://docs.celestia.org/community/coc2023-10-26T17:10:53.000Zhttps://docs.celestia.org/community/foundation-delegation-program2024-02-12T16:33:14.000Zhttps://docs.celestia.org/community/itn-tos2023-11-03T14:02:07.000Zhttps://docs.celestia.org/community/modular-meetup-guide2023-10-23T20:05:00.000Zhttps://docs.celestia.org/community/modular-meetup-intro2024-03-06T17:05:43.000Zhttps://docs.celestia.org/community/modular-meetup-toolkit2023-10-23T20:05:00.000Zhttps://docs.celestia.org/community/overview2023-10-23T20:05:00.000Zhttps://docs.celestia.org/community/speaker-list2023-10-23T20:05:00.000Zhttps://docs.celestia.org/developers/arbitrum-bridge2024-04-15T14:35:09.000Zhttps://docs.celestia.org/developers/arbitrum-dapp-deploy2024-04-02T19:44:10.000Zhttps://docs.celestia.org/developers/arbitrum-deploy2024-04-15T14:35:09.000Zhttps://docs.celestia.org/developers/arbitrum-full-node2024-04-02T19:44:10.000Zhttps://docs.celestia.org/developers/arbitrum-integration2024-04-15T14:35:09.000Zhttps://docs.celestia.org/developers/arbitrum-smart-contract2023-12-22T20:48:10.000Zhttps://docs.celestia.org/developers/blobstream-contracts2024-03-08T15:21:05.000Zhttps://docs.celestia.org/developers/blobstream-offchain2024-04-15T14:35:09.000Zhttps://docs.celestia.org/developers/blobstream-proof-queries2024-04-15T14:35:09.000Zhttps://docs.celestia.org/developers/blobstream-rollups2024-04-18T14:32:58.000Zhttps://docs.celestia.org/developers/blobstream-x-deploy2024-03-08T15:21:05.000Zhttps://docs.celestia.org/developers/blobstream2024-04-18T14:32:58.000Zhttps://docs.celestia.org/developers/bubs-testnet2024-03-25T18:50:21.000Zhttps://docs.celestia.org/developers/build-whatever2024-04-11T21:24:32.000Zhttps://docs.celestia.org/developers/celestia-node-key2024-02-06T15:04:43.000Zhttps://docs.celestia.org/developers/deploy-on-bubs2024-03-25T17:40:43.000Zhttps://docs.celestia.org/developers/ethereum-fallback2023-12-13T18:56:35.000Zhttps://docs.celestia.org/developers/full-stack-modular-development-guide2024-04-11T21:24:32.000Zhttps://docs.celestia.org/developers/gm-portal-bubs2024-03-25T17:40:43.000Zhttps://docs.celestia.org/developers/integrate-celestia2024-04-11T21:24:32.000Zhttps://docs.celestia.org/developers/intro-to-op-stack2023-11-15T13:49:48.000Zhttps://docs.celestia.org/developers/node-api2023-11-06T16:30:46.000Zhttps://docs.celestia.org/developers/node-tutorial2024-04-15T14:35:09.000Zhttps://docs.celestia.org/developers/optimism-devnet2024-04-15T14:35:09.000Zhttps://docs.celestia.org/developers/optimism2024-04-15T14:35:09.000Zhttps://docs.celestia.org/developers/prompt-scavenger2024-04-15T14:35:09.000Zhttps://docs.celestia.org/developers/requesting-data-commitment-ranges2024-03-08T15:21:05.000Zhttps://docs.celestia.org/developers/rollkit2024-02-06T15:04:43.000Zhttps://docs.celestia.org/developers/submit-data2024-04-22T13:37:46.000Zhttps://docs.celestia.org/developers/transaction-resubmission2024-02-19T10:50:16.000Zhttps://docs.celestia.org/developers/wallets2024-04-15T14:35:09.000Zhttps://docs.celestia.org/2024-04-11T21:24:32.000Zhttps://docs.celestia.org/learn/how-celestia-works/data-availability-faq2023-11-17T15:21:22.000Zhttps://docs.celestia.org/learn/how-celestia-works/data-availability-layer2024-03-05T21:22:56.000Zhttps://docs.celestia.org/learn/how-celestia-works/monolithic-vs-modular2024-04-01T14:11:11.000Zhttps://docs.celestia.org/learn/how-celestia-works/overview2024-03-05T21:22:56.000Zhttps://docs.celestia.org/learn/how-celestia-works/transaction-lifecycle2024-03-05T21:22:56.000Zhttps://docs.celestia.org/learn/how-to-stake-tia2024-02-06T15:04:43.000Zhttps://docs.celestia.org/learn/paying-for-blobspace2023-10-23T20:05:00.000Zhttps://docs.celestia.org/learn/retrievability2024-02-22T12:10:50.000Zhttps://docs.celestia.org/learn/staking-governance-supply2024-04-15T14:35:09.000Zhttps://docs.celestia.org/learn/staking2024-02-06T15:04:43.000Zhttps://docs.celestia.org/learn/tia2023-12-11T18:21:14.000Zhttps://docs.celestia.org/nodes/arabica-devnet2024-04-09T23:35:56.000Zhttps://docs.celestia.org/nodes/bridge-node2024-02-28T14:48:31.000Zhttps://docs.celestia.org/nodes/celestia-app-commands2023-12-11T18:21:14.000Zhttps://docs.celestia.org/nodes/celestia-app-metrics2024-02-06T15:04:43.000Zhttps://docs.celestia.org/nodes/celestia-app-multisig2024-01-09T15:17:50.000Zhttps://docs.celestia.org/nodes/celestia-app-slashing2024-01-31T19:41:09.000Zhttps://docs.celestia.org/nodes/celestia-app-upgrade-monitor2023-12-14T20:10:16.000Zhttps://docs.celestia.org/nodes/celestia-app-vesting2024-03-29T20:40:58.000Zhttps://docs.celestia.org/nodes/celestia-app-wallet2024-02-06T15:04:43.000Zhttps://docs.celestia.org/nodes/celestia-app2024-04-12T19:24:40.000Zhttps://docs.celestia.org/nodes/celestia-node-custom-networks2024-04-22T13:37:46.000Zhttps://docs.celestia.org/nodes/celestia-node-metrics2024-03-04T15:39:24.000Zhttps://docs.celestia.org/nodes/celestia-node-troubleshooting2024-02-29T14:40:18.000Zhttps://docs.celestia.org/nodes/celestia-node2024-04-12T19:26:37.000Zhttps://docs.celestia.org/nodes/config-toml2023-10-23T20:05:00.000Zhttps://docs.celestia.org/nodes/decide-node2024-03-29T20:40:58.000Zhttps://docs.celestia.org/nodes/docker-images2023-11-15T13:49:48.000Zhttps://docs.celestia.org/nodes/environment2024-02-08T20:48:59.000Zhttps://docs.celestia.org/nodes/full-consensus-node2024-03-29T20:40:58.000Zhttps://docs.celestia.org/nodes/full-storage-node2023-12-14T17:29:41.000Zhttps://docs.celestia.org/nodes/hardfork-process2023-12-14T19:54:10.000Zhttps://docs.celestia.org/nodes/ibc-relayer2024-02-06T15:04:43.000Zhttps://docs.celestia.org/nodes/instantiate-testnet2024-03-29T20:40:58.000Zhttps://docs.celestia.org/nodes/light-node2024-02-10T00:09:03.000Zhttps://docs.celestia.org/nodes/mainnet2024-04-15T14:35:31.000Zhttps://docs.celestia.org/nodes/mocha-testnet2024-04-15T14:36:09.000Zhttps://docs.celestia.org/nodes/overview2024-03-29T20:40:58.000Zhttps://docs.celestia.org/nodes/participate2024-03-11T16:56:42.000Zhttps://docs.celestia.org/nodes/quick-start2023-10-27T14:33:58.000Zhttps://docs.celestia.org/nodes/systemd2024-02-13T14:12:47.000Zhttps://docs.celestia.org/nodes/validator-node2024-03-29T20:40:58.000Z \ No newline at end of file