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..9cb5325c887 --- /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..36bf96bbb93 --- /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.03785d82.js b/assets/README.md.03785d82.js new file mode 100644 index 00000000000..3bd17a6c183 --- /dev/null +++ b/assets/README.md.03785d82.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.1a91c06a.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":1726496322000}'),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.03785d82.lean.js b/assets/README.md.03785d82.lean.js new file mode 100644 index 00000000000..20908748a92 --- /dev/null +++ b/assets/README.md.03785d82.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.1a91c06a.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":1726496322000}'),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.e2086950.js b/assets/app.e2086950.js new file mode 100644 index 00000000000..af47e70ce66 --- /dev/null +++ b/assets/app.e2086950.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.1a91c06a.js";import{t as p}from"./chunks/theme.9c6b4fd0.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.b53f000d.js b/assets/chunks/@localSearchIndexroot.b53f000d.js new file mode 100644 index 00000000000..ca4807e3b0e --- /dev/null +++ b/assets/chunks/@localSearchIndexroot.b53f000d.js @@ -0,0 +1 @@ +const e='{"documentCount":757,"nextId":757,"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/coc#celestia-org-code-of-conduct","6":"/community/coc#our-pledge","7":"/community/coc#our-standards","8":"/community/coc#enforcement-responsibilities","9":"/community/coc#scope","10":"/community/coc#enforcement","11":"/community/coc#enforcement-guidelines","12":"/community/coc#_1-correction","13":"/community/coc#_2-warning","14":"/community/coc#_3-temporary-ban","15":"/community/coc#_4-permanent-ban","16":"/community/coc#attribution","17":"/community/foundation-delegation-program#the-celestia-foundation-delegation-program","18":"/community/foundation-delegation-program#objectives-of-the-program","19":"/community/foundation-delegation-program#foundation-delegation-process","20":"/community/foundation-delegation-program#program-launch","21":"/community/foundation-delegation-program#cohort-process","22":"/community/foundation-delegation-program#key-points","23":"/community/foundation-delegation-program#eligibility-criteria","24":"/community/foundation-delegation-program#undelegation-criteria","25":"/community/foundation-delegation-program#application","26":"/community/foundation-delegation-program#application-details","27":"/community/foundation-delegation-program#cohort-information","28":"/community/foundation-delegation-program#feedback-process","29":"/community/modular-meetup-guide#modular-meetup-guide","30":"/community/modular-meetup-guide#before-the-meetup","31":"/community/modular-meetup-guide#during-the-meetup","32":"/community/modular-meetup-guide#after-the-meetup","33":"/community/modular-meetup-guide#logistics-and-guidance","34":"/community/modular-meetup-guide#venue","35":"/community/modular-meetup-guide#catering-and-refreshments","36":"/community/modular-meetup-guide#audience","37":"/community/modular-meetup-guide#speakers","38":"/community/modular-meetup-guide#sponsors","39":"/community/modular-meetup-guide#communications-and-marketing","40":"/community/modular-meetup-guide#recording","41":"/community/modular-meetup-guide#utilizing-meetup-com-platform","42":"/community/modular-meetup-guide#onboarding-questions-for-community-members-joining-a-modular-meetup","43":"/community/modular-meetup-intro#celestia-modular-meetup-program","44":"/community/modular-meetup-intro#program-description","45":"/community/modular-meetup-intro#important-info","46":"/community/modular-meetup-intro#celestia-org-community-code-of-conduct","47":"/community/modular-meetup-intro#signup-form","48":"/community/modular-meetup-intro#emails","49":"/community/modular-meetup-intro#discord","50":"/community/modular-meetup-intro#materials","51":"/community/modular-meetup-toolkit#modular-meetup-toolkit","52":"/community/modular-meetup-toolkit#celestia-branding-guidelines","53":"/community/modular-meetup-toolkit#sample-introduction-to-modularity-workshop-presentation","54":"/community/modular-meetup-toolkit#sample-run-a-celestia-light-node-workshop-presentation","55":"/community/modular-meetup-toolkit#sample-deploy-a-sovereign-rollup-workshop-presentation","56":"/community/modular-meetup-toolkit#sample-modular-meetup-introduction-workshop-presentation","57":"/community/modular-meetup-toolkit#swag-logistics","58":"/community/speaker-list#speaker-list","59":"/developers/arbitrum-bridge#bridging-in-and-out-of-your-orbit-rollup","60":"/developers/arbitrum-bridge#bridge-in-deposit-to-your-rollup","61":"/developers/arbitrum-bridge#step-1-add-your-custom-chain-config","62":"/developers/arbitrum-bridge#step-2-deposit-to-your-orbit-rollup","63":"/developers/arbitrum-bridge#bridge-out-withdrawal-from-your-rollup","64":"/developers/arbitrum-bridge#step-1-choose-an-amount-to-withdraw-from-your-rollup","65":"/developers/arbitrum-bridge#step-2-claim-your-withdrawal","66":"/developers/arbitrum-deploy#quickstart-deploy-an-arbitrum-orbit-rollup","67":"/developers/arbitrum-deploy#prerequisites","68":"/developers/arbitrum-deploy#setup","69":"/developers/arbitrum-deploy#step-1-acquire-arbitrum-sepolia-eth","70":"/developers/arbitrum-deploy#step-2-pick-your-deployment-type","71":"/developers/arbitrum-deploy#step-3-configure-your-orbit-chain-s-deployment","72":"/developers/arbitrum-deploy#step-3-review-deploy-your-orbit-chain","73":"/developers/arbitrum-deploy#step-4-download-your-chain-s-configuration-files-and-launch-your-chain","74":"/developers/arbitrum-deploy#step-5-clone-the-setup-script-repository-and-add-your-configuration-files","75":"/developers/arbitrum-deploy#step-6-pick-an-l2-rpc-url-for-the-batch-poster","76":"/developers/arbitrum-deploy#step-7-run-your-light-node-for-mocha-testnet","77":"/developers/arbitrum-deploy#step-8-run-your-chain-s-node-and-block-explorer","78":"/developers/arbitrum-deploy#step-9-finish-setting-up-your-chain","79":"/developers/arbitrum-deploy#congratulations-with-celestia-underneath","80":"/developers/arbitrum-deploy#appendix","81":"/developers/arbitrum-deploy#compatibility-matrix","82":"/developers/arbitrum-deploy#blobstream-x-contract-deployments","83":"/developers/arbitrum-deploy#arbitrum-sepolia","84":"/developers/arbitrum-deploy#base-sepolia","85":"/developers/arbitrum-deploy#arbitrum-sepolia-additional-deployments","86":"/developers/arbitrum-deploy#base-sepolia-additional-deployments","87":"/developers/arbitrum-full-node#running-a-full-node-and-or-validator","88":"/developers/arbitrum-full-node#prerequisites","89":"/developers/arbitrum-full-node#running-a-full-node","90":"/developers/arbitrum-full-node#running-a-full-node-with-validation","91":"/developers/arbitrum-integration#introduction-to-arbitrum-rollups-with-celestia-as-da","92":"/developers/arbitrum-integration#overview","93":"/developers/arbitrum-integration#key-components","94":"/developers/arbitrum-integration#da-provider-implementation","95":"/developers/arbitrum-integration#preimage-oracle-implementation","96":"/developers/arbitrum-integration#blobstream-x-implementation","97":"/developers/arbitrum-integration#ethereum-fallback-mechanism-in-nitro","98":"/developers/arbitrum-integration#next-steps","99":"/developers/blobstream-contracts#integrate-with-blobstream-contracts","100":"/developers/blobstream-contracts#getting-started","101":"/developers/blobstream-contracts#prerequisites","102":"/developers/blobstream-contracts#installing-blobstream-contracts","103":"/developers/blobstream-contracts#example-usage","104":"/developers/blobstream-contracts#data-structures","105":"/developers/blobstream-contracts#interface","106":"/developers/blobstream-contracts#querying-the-proof","107":"/developers/blobstream-contracts#verifying-data-inclusion-for-fraud-proofs","108":"/developers/blobstream-offchain#integrate-with-blobstream-client","109":"/developers/blobstream-offchain#blobstream-demo-rollup","110":"/developers/blobstream-offchain#defining-a-chain","111":"/developers/blobstream-offchain#rollup-sequencer","112":"/developers/blobstream-offchain#committing-to-data","113":"/developers/blobstream-offchain#creating-blocks","114":"/developers/blobstream-offchain#rollup-full-node","115":"/developers/blobstream-offchain#downloading-the-block","116":"/developers/blobstream-offchain#more-documentation","117":"/developers/blobstream-offchain#proving-inclusion-via-blobstream","118":"/developers/blobstream-offchain#submitting-block-data-to-celestia-via-light-node","119":"/developers/blobstream-offchain#posting-headers-to-ethereum","120":"/developers/blobstream-proof-queries#blobstream-proofs-queries","121":"/developers/blobstream-proof-queries#prerequisites","122":"/developers/blobstream-proof-queries#overview-of-the-proof-queries","123":"/developers/blobstream-proof-queries#the-celestia-square","124":"/developers/blobstream-proof-queries#the-commitment-scheme","125":"/developers/blobstream-proof-queries#hands-on-demonstration","126":"/developers/blobstream-proof-queries#_1-data-root-inclusion-proof","127":"/developers/blobstream-proof-queries#http-query","128":"/developers/blobstream-proof-queries#golang-client","129":"/developers/blobstream-proof-queries#full-example-of-proving-that-a-celestia-block-was-committed-to-by-blobstream-contract","130":"/developers/blobstream-proof-queries#_2-transaction-inclusion-proof","131":"/developers/blobstream-proof-queries#transaction-inclusion-proof-using-the-transaction-hash","132":"/developers/blobstream-proof-queries#http-request","133":"/developers/blobstream-proof-queries#golang-client-1","134":"/developers/blobstream-proof-queries#blob-inclusion-proof-using-the-corresponding-pfb-transaction-hash","135":"/developers/blobstream-proof-queries#golang-client-2","136":"/developers/blobstream-proof-queries#specific-share-range-inclusion-proof","137":"/developers/blobstream-proof-queries#http-request-1","138":"/developers/blobstream-proof-queries#golang-client-3","139":"/developers/blobstream-proof-queries#converting-the-proofs-to-be-usable-in-the-daverifier-library","140":"/developers/blobstream-proof-queries#data","141":"/developers/blobstream-proof-queries#shareproofs","142":"/developers/blobstream-proof-queries#namespace","143":"/developers/blobstream-proof-queries#rowroots","144":"/developers/blobstream-proof-queries#rowproofs","145":"/developers/blobstream-proof-queries#attestationproof","146":"/developers/blobstream-proof-queries#querying-the-proof-s-tuplerootnonce","147":"/developers/blobstream-proof-queries#listening-for-new-data-commitments","148":"/developers/blobstream-proof-queries#example-rollup-that-uses-the-daverifier","149":"/developers/blobstream-proof-queries#conclusion","150":"/developers/blobstream-rollups#introduction-to-blobstream-rollups","151":"/developers/blobstream-rollups#concepts","152":"/developers/blobstream-rollups#blob-share-commitment","153":"/developers/blobstream-rollups#blob-share-commitment-proof-details","154":"/developers/blobstream-rollups#blob-share-commitment-compact-proofs","155":"/developers/blobstream-rollups#blob-share-commitment-pros","156":"/developers/blobstream-rollups#blob-share-commitment-cons","157":"/developers/blobstream-rollups#sequence-of-spans","158":"/developers/blobstream-rollups#sequence-of-spans-proof-details","159":"/developers/blobstream-rollups#sequence-of-spans-proving-unavailable-data","160":"/developers/blobstream-rollups#sequence-of-spans-proving-inclusion-of-some-data","161":"/developers/blobstream-rollups#sequence-of-spans-pros","162":"/developers/blobstream-rollups#sequence-of-spans-cons","163":"/developers/blobstream-rollups#optimistic-rollups","164":"/developers/blobstream-rollups#optimistic-rollups-that-use-a-sequence-of-spans","165":"/developers/blobstream-rollups#optimistic-rollups-that-use-a-sequence-of-spans-pros","166":"/developers/blobstream-rollups#optimistic-rollups-that-use-a-sequence-of-spans-cons","167":"/developers/blobstream-rollups#optimistic-rollups-that-use-a-sequence-of-spans-example","168":"/developers/blobstream-rollups#optimistic-rollups-that-use-blob-share-commitments","169":"/developers/blobstream-rollups#optimistic-rollups-that-use-blob-share-commitments-pros","170":"/developers/blobstream-rollups#optimistic-rollups-that-use-blob-share-commitments-cons","171":"/developers/blobstream-rollups#zk-rollups","172":"/developers/blobstream-rollups#zk-rollups-that-use-sequence-of-spans","173":"/developers/blobstream-rollups#zk-rollups-that-use-sequence-of-spans-pros","174":"/developers/blobstream-rollups#zk-rollups-that-use-sequence-of-spans-cons","175":"/developers/blobstream-rollups#zk-rollups-that-use-blob-share-commitments","176":"/developers/blobstream-rollups#protobuf-deserialization-inside-a-zk-circuit","177":"/developers/blobstream-rollups#zk-rollups-that-use-blob-share-commitments-pros","178":"/developers/blobstream-rollups#zk-rollups-that-use-blob-share-commitments-cons","179":"/developers/blobstream-rollups#heavy-merkle-proofs-usage","180":"/developers/blobstream-rollups#heavy-merkle-proofs-usage-pros","181":"/developers/blobstream-rollups#heavy-merkle-proofs-usage-cons","182":"/developers/blobstream-rollups#conclusion","183":"/developers/blobstream-rollups#faq","184":"/developers/blobstream-rollups#should-i-use-the-celestia-transaction-hash-to-reference-the-rollup-data","185":"/developers/blobstream-x-deploy#new-blobstream-x-deployments","186":"/developers/blobstream-x-deploy#deploying-the-contracts","187":"/developers/blobstream-x-deploy#deploy-a-new-succinctgateway","188":"/developers/blobstream-x-deploy#deploy-the-function-verifiers","189":"/developers/blobstream-x-deploy#register-the-function-verifier-in-the-deployed-succinctgateway","190":"/developers/blobstream-x-deploy#enable-prover-whitelisting","191":"/developers/blobstream-x-deploy#set-whitelist-status","192":"/developers/blobstream-x-deploy#add-custom-prover","193":"/developers/blobstream-x-deploy#deploy-the-blobstreamx-contract","194":"/developers/blobstream-x-deploy#querying-the-trusted-hash","195":"/developers/blobstream-x-deploy#deployment-instructions","196":"/developers/blobstream-x-deploy#run-a-local-prover","197":"/developers/blobstream-x-deploy#run-a-proof-replayer-from-an-existing-deployment","198":"/developers/blobstream-x-deploy#optional-regenerating-the-downloaded-artifacts","199":"/developers/blobstream-x-deploy#regenerate-the-verifier-build","200":"/developers/blobstream-x-deploy#build-the-circuits-and-function-verifiers","201":"/developers/blobstream-x-requesting-data-commitment-ranges#requesting-data-commitment-ranges","202":"/developers/blobstream-x-requesting-data-commitment-ranges#local-proving","203":"/developers/blobstream-x-requesting-data-commitment-ranges#request-proofs-from-the-succinct-platform","204":"/developers/blobstream-x-requesting-data-commitment-ranges#request-proofs-onchain","205":"/developers/blobstream#blobstream-streaming-modular-da-to-ethereum","206":"/developers/blobstream#what-is-blobstream","207":"/developers/blobstream#implementations-of-blobstream","208":"/developers/blobstream#blobstream-vs-data-availability-committees-dacs","209":"/developers/blobstream#decentralization-and-security","210":"/developers/blobstream#mechanism-of-verification","211":"/developers/blobstream#flexibility-and-scalability","212":"/developers/blobstream#what-is-sp1-blobstream","213":"/developers/blobstream#integrate-with-sp1-blobstream","214":"/developers/blobstream#how-to-integrate-with-blobstream","215":"/developers/blobstream#blobstream-rollups","216":"/developers/blobstream#deployed-contracts","217":"/developers/blobstreamx#blobstream-x-the-previous-zk-implementation-of-blobstream","218":"/developers/blobstreamx#what-is-blobstream-x","219":"/developers/blobstreamx#how-blobstream-x-works","220":"/developers/blobstreamx#deploy-blobstream-x","221":"/developers/bubs-testnet#bubs-testnet","222":"/developers/bubs-testnet#built-with-the-op-stack-and-celestia","223":"/developers/bubs-testnet#building-on-bubs","224":"/developers/bubs-testnet#rpc-urls","225":"/developers/bubs-testnet#https","226":"/developers/bubs-testnet#wss","227":"/developers/bubs-testnet#bridge","228":"/developers/bubs-testnet#faucet","229":"/developers/bubs-testnet#explorer","230":"/developers/bubs-testnet#status","231":"/developers/bubs-testnet#next-steps","232":"/developers/build-whatever#build-whatever","233":"/developers/build-whatever#quickstart-building-on-celestia","234":"/developers/build-whatever#choose-a-framework","235":"/developers/build-whatever#rollups-as-a-service","236":"/developers/build-whatever#smart-contracts","237":"/developers/build-whatever#what-is-a-rollup","238":"/developers/build-whatever#what-is-a-modular-blockchain","239":"/developers/build-whatever#benefits-of-modular-blockchains","240":"/developers/build-whatever#ease-of-deploying-a-chain","241":"/developers/build-whatever#scaling","242":"/developers/build-whatever#customizability","243":"/developers/celestia-node-key#create-a-wallet-with-celestia-node","244":"/developers/celestia-node-key#using-the-cel-key-utility","245":"/developers/celestia-node-key#installation","246":"/developers/celestia-node-key#steps-for-generating-node-keys","247":"/developers/celestia-node-key#steps-for-exporting-node-keys","248":"/developers/celestia-node-key#steps-for-importing-node-keys","249":"/developers/celestia-node-key#view-all-options-for-cel-key","250":"/developers/celestia-node-key#docker-and-cel-key","251":"/developers/celestia-node-key#prerequisites","252":"/developers/celestia-node-key#running-your-node","253":"/developers/celestia-node-key#mounting-existing-keys-to-container","254":"/developers/ethereum-fallback#ethereum-fallback","255":"/developers/feegrant-for-blobs#feegrant-module-for-blobs-submission","256":"/developers/feegrant-for-blobs#overview","257":"/developers/feegrant-for-blobs#pre-requisites","258":"/developers/feegrant-for-blobs#introduction","259":"/developers/feegrant-for-blobs#granting-fee-allowances-using-celestia-node","260":"/developers/feegrant-for-blobs#feegrant-module-implementation-in-celestia-node","261":"/developers/feegrant-for-blobs#grant-permission-for-an-allowance-as-a-granter","262":"/developers/feegrant-for-blobs#using-a-feegrant-allowance-as-a-grantee-in-celestia-node","263":"/developers/feegrant-for-blobs#checking-account-balances-after-submission","264":"/developers/feegrant-for-blobs#optional-revoke-permission-for-a-feegrant-allowance-as-a-granter","265":"/developers/feegrant-for-blobs#optional-submitting-a-blob-from-file-input","266":"/developers/feegrant-for-blobs#optional-granting-fee-allowances-using-celestia-appd","267":"/developers/feegrant-for-blobs#optional-checking-the-granter-s-account","268":"/developers/golang-client-tutorial#golang-client-library","269":"/developers/golang-client-tutorial#project-setup","270":"/developers/golang-client-tutorial#submitting-and-retrieving-blobs","271":"/developers/golang-client-tutorial#subscribing-to-new-blobs","272":"/developers/golang-client-tutorial#subscribing-to-new-headers","273":"/developers/golang-client-tutorial#fetching-an-extended-data-square-eds","274":"/developers/golang-client-tutorial#api-documentation","275":"/developers/integrate-celestia#integrate-celestia-for-service-providers","276":"/developers/integrate-celestia#getting-started","277":"/developers/integrate-celestia#celestia-service-provider-notes","278":"/developers/integrate-celestia#custody-and-key-management","279":"/developers/integrate-celestia#rpc-and-querying","280":"/developers/integrate-celestia#compatibility","281":"/developers/integrate-celestia#syncing","282":"/developers/integrate-celestia#notable-exceptions-relative-to-other-blockchains","283":"/developers/intro-to-op-stack#introduction-to-op-stack-integration","284":"/developers/intro-to-op-stack#about-the-integration","285":"/developers/intro-to-op-stack#github-repository","286":"/developers/intro-to-op-stack#next-steps","287":"/developers/multiaccounts#multiaccounts-feature-for-blobs-submission","288":"/developers/multiaccounts#overview","289":"/developers/multiaccounts#running-a-node-with-a-different-default-key-name","290":"/developers/multiaccounts#submitting-blobs-with-a-different-signer-key-name","291":"/developers/multiaccounts#option-1-submit-passing-key-name","292":"/developers/multiaccounts#option-2-submit-passing-signer-address","293":"/developers/multiaccounts#key-management","294":"/developers/multiaccounts#creating-a-new-key","295":"/developers/multiaccounts#importing-an-existing-key","296":"/developers/multiaccounts#optional-flags-for-write-transactions","297":"/developers/node-api#node-api","298":"/developers/node-api#rpc-api","299":"/developers/node-api#library","300":"/developers/node-api#rpc","301":"/developers/node-api#rpc-api-tutorial","302":"/developers/node-api#gateway-api","303":"/developers/node-api#gateway-api-tutorial","304":"/developers/node-tutorial#celestia-node-rpc-cli-tutorial","305":"/developers/node-tutorial#introduction","306":"/developers/node-tutorial#blobs","307":"/developers/node-tutorial#namespaces","308":"/developers/node-tutorial#hardware-requirements","309":"/developers/node-tutorial#setting-up-dependencies","310":"/developers/node-tutorial#instantiate-a-celestia-light-node","311":"/developers/node-tutorial#connect-to-a-core-endpoint","312":"/developers/node-tutorial#keys-and-wallets","313":"/developers/node-tutorial#rpc-cli-guide","314":"/developers/node-tutorial#command-formatting","315":"/developers/node-tutorial#basic-flags","316":"/developers/node-tutorial#auth-token","317":"/developers/node-tutorial#node-store","318":"/developers/node-tutorial#auth-token-on-custom-or-private-network","319":"/developers/node-tutorial#submitting-data","320":"/developers/node-tutorial#optional-submit-with-curl","321":"/developers/node-tutorial#retrieving-data","322":"/developers/node-tutorial#setting-the-gas-price","323":"/developers/node-tutorial#examples","324":"/developers/node-tutorial#check-your-balance","325":"/developers/node-tutorial#check-the-balance-of-another-address","326":"/developers/node-tutorial#get-your-node-id","327":"/developers/node-tutorial#get-your-account-address","328":"/developers/node-tutorial#get-block-header-by-height","329":"/developers/node-tutorial#combined-commands","330":"/developers/node-tutorial#get-data-availability-sampler-stats","331":"/developers/node-tutorial#transfer-balance-of-utia-to-another-account","332":"/developers/node-tutorial#api-version","333":"/developers/node-tutorial#help","334":"/developers/node-tutorial#advanced-example","335":"/developers/node-tutorial#additional-resources","336":"/developers/node-tutorial#submitting-a-blob-using-curl","337":"/developers/node-tutorial#post-an-svg-as-a-pfb","338":"/developers/node-tutorial#troubleshooting","339":"/developers/optimism-devnet#optimism-devnet-deep-dive","340":"/developers/optimism-devnet#find-a-transaction","341":"/developers/optimism-devnet#read-the-transaction-call-data","342":"/developers/optimism-devnet#find-the-data-on-celestia","343":"/developers/optimism-devnet#span-batches","344":"/developers/optimism#run-an-op-stack-rollup-with-celestia-underneath","345":"/developers/optimism#dependency-setup","346":"/developers/optimism#setting-up-your-light-node","347":"/developers/optimism#deploying-a-devnet-to-mocha","348":"/developers/optimism#deploying-a-testnet-to-an-l1-or-l2-and-mocha","349":"/developers/optimism#congratulations","350":"/developers/prompt-scavenger#prompt-scavenger","351":"/developers/prompt-scavenger#dependencies","352":"/developers/prompt-scavenger#install-celestia-node-and-run-a-light-node","353":"/developers/prompt-scavenger#openai-key","354":"/developers/prompt-scavenger#building-the-prompt-scavenger","355":"/developers/prompt-scavenger#initialize-your-go-project","356":"/developers/prompt-scavenger#build-your-import-statements","357":"/developers/prompt-scavenger#main-function","358":"/developers/prompt-scavenger#utility-functions","359":"/developers/prompt-scavenger#prompting-chatgpt","360":"/developers/prompt-scavenger#wrapping-things-up","361":"/developers/prompt-scavenger#next-steps","362":"/developers/rust-client-tutorial#rust-client-library","363":"/developers/rust-client-tutorial#project-setup","364":"/developers/rust-client-tutorial#submitting-and-retrieving-blobs","365":"/developers/rust-client-tutorial#subscribing-to-new-headers","366":"/developers/rust-client-tutorial#fetching-an-extended-data-square-eds","367":"/developers/rust-client-tutorial#api-documentation","368":"/developers/sp1-blobstream-deploy#new-sp1-blobstream-deployments","369":"/developers/sp1-blobstream-deploy#deploying-the-contracts","370":"/developers/submit-data#submitting-data-blobs-to-celestia","371":"/developers/submit-data#maximum-blob-size","372":"/developers/submit-data#fee-market-and-mempool","373":"/developers/submit-data#fees-and-gas-limits","374":"/developers/submit-data#estimating-pfb-gas","375":"/developers/submit-data#gas-fee-calculation","376":"/developers/submit-data#estimating-gas-programmatically","377":"/developers/submit-data#submitting-multiple-transactions-in-one-block-from-the-same-account","378":"/developers/submit-data#api","379":"/developers/submit-data#the-celestia-app-consensus-node-cli","380":"/developers/submit-data#the-celestia-node-light-node-cli","381":"/developers/submit-data#the-celestia-node-api-golang-client","382":"/developers/submit-data#grpc-to-a-consensus-node-via-the-user-package","383":"/developers/submit-data#rpc-to-a-celestia-node","384":"/developers/submit-data#post-a-blob-directly-from-celenium","385":"/developers/transaction-resubmission#transaction-resubmission","386":"/developers/transaction-resubmission#monitoring-and-resubmission","387":"/developers/transaction-resubmission#notes","388":"/developers/wallets#wallet-integrations-with-celestia","389":"/developers/wallets#add-celestia-network-parameters-to-keplr-with-react","390":"/developers/wallets#adding-a-custom-chain-to-leap","391":"/developers/wallets#adding-a-custom-chain-to-cosmostation","392":"/learn/how-celestia-works/data-availability-faq#data-availability-faq","393":"/learn/how-celestia-works/data-availability-faq#what-is-data-availability","394":"/learn/how-celestia-works/data-availability-faq#what-is-the-data-availability-problem","395":"/learn/how-celestia-works/data-availability-faq#how-do-nodes-verify-data-availability-in-celestia","396":"/learn/how-celestia-works/data-availability-faq#what-is-data-availability-sampling","397":"/learn/how-celestia-works/data-availability-faq#what-are-some-of-the-security-assumptions-that-celestia-makes-for-data-availability-sampling","398":"/learn/how-celestia-works/data-availability-faq#why-is-block-reconstruction-necessary-for-security","399":"/learn/how-celestia-works/data-availability-faq#what-is-data-storage","400":"/learn/how-celestia-works/data-availability-faq#what-is-the-problem-around-data-storage","401":"/learn/how-celestia-works/data-availability-faq#what-is-the-difference-between-data-availability-and-data-storage","402":"/learn/how-celestia-works/data-availability-faq#where-does-blockchain-state-fit-into-this","403":"/learn/how-celestia-works/data-availability-faq#why-doesn-t-celestia-incentivize-storage-of-historical-data","404":"/learn/how-celestia-works/data-availability-faq#who-may-store-historical-data-if-there-is-no-reward","405":"/learn/how-celestia-works/data-availability-faq#what-are-some-things-blockchains-can-do-to-provide-stronger-assurances-of-data-retrievability","406":"/learn/how-celestia-works/data-availability-layer#celestia-s-data-availability-layer","407":"/learn/how-celestia-works/data-availability-layer#data-availability-sampling-das","408":"/learn/how-celestia-works/data-availability-layer#scalability","409":"/learn/how-celestia-works/data-availability-layer#fraud-proofs-of-incorrectly-extended-data","410":"/learn/how-celestia-works/data-availability-layer#namespaced-merkle-trees-nmts","411":"/learn/how-celestia-works/data-availability-layer#building-a-pos-blockchain-for-da","412":"/learn/how-celestia-works/data-availability-layer#providing-data-availability","413":"/learn/how-celestia-works/monolithic-vs-modular#monolithic-vs-modular-blockchains","414":"/learn/how-celestia-works/overview#introduction","415":"/learn/how-celestia-works/transaction-lifecycle#the-lifecycle-of-a-celestia-app-transaction","416":"/learn/how-celestia-works/transaction-lifecycle#checking-data-availability","417":"/learn/how-to-stake-tia#how-to-stake-tia","418":"/learn/how-to-stake-tia#select-your-preferred-wallet","419":"/learn/how-to-stake-tia#stake-tia-with-keplr-wallet","420":"/learn/how-to-stake-tia#open-your-keplr-browser-extension","421":"/learn/how-to-stake-tia#select-celestia-network-and-search-for-a-validator","422":"/learn/how-to-stake-tia#stake-your-tia-tokens","423":"/learn/how-to-stake-tia#confirm-and-manage-your-tia","424":"/learn/how-to-stake-tia#stake-tia-with-leap-wallet","425":"/learn/how-to-stake-tia#open-your-leap-browser-extension","426":"/learn/how-to-stake-tia#select-a-validator-and-stake-tia","427":"/learn/how-to-stake-tia#confirm-and-manage-your-tia-1","428":"/learn/how-to-stake-tia#stake-tia-with-gem-wallet","429":"/learn/how-to-stake-tia#open-your-gem-wallet-app","430":"/learn/how-to-stake-tia#choose-the-amount-of-celestia-and-search-for-a-validator","431":"/learn/how-to-stake-tia#stake-your-tia-tokens-1","432":"/learn/how-to-stake-tia#manage-your-tia","433":"/learn/paying-for-blobspace#paying-for-blobspace","434":"/learn/paying-for-blobspace#payforblobs-transactions","435":"/learn/paying-for-blobspace#fee-market-overview","436":"/learn/retrievability#data-retrievability-and-pruning","437":"/learn/retrievability#data-retrievability-and-pruning-in-celestia-node","438":"/learn/retrievability#suggested-practices-for-rollups","439":"/learn/staking-governance-supply#staking-governance-supply","440":"/learn/staking-governance-supply#proof-of-stake-on-celestia","441":"/learn/staking-governance-supply#inflation","442":"/learn/staking-governance-supply#decentralised-governance","443":"/learn/staking-governance-supply#network-parameters","444":"/learn/staking-governance-supply#community-pool","445":"/learn/staking-governance-supply#tia-allocation-at-genesis","446":"/learn/staking-governance-supply#unlocks","447":"/learn/staking#staking-on-celestia","448":"/learn/staking#mainnet-beta","449":"/learn/staking#mocha-testnet","450":"/learn/tia#overview-of-tia","451":"/learn/tia#tia-at-a-glance","452":"/learn/tia#role-of-tia","453":"/learn/tia#paying-for-blobspace","454":"/learn/tia#bootstrapping-new-rollups","455":"/learn/tia#proof-of-stake","456":"/learn/tia#decentralised-governance","457":"/learn/tia#denominations","458":"/learn/tia#tia-display-token","459":"/learn/tia#utia-staking-denomination","460":"/learn/tia#microtia-staking-denomination-alias","461":"/nodes/arabica-devnet#arabica-devnet","462":"/nodes/arabica-devnet#network-stability-and-upgrades","463":"/nodes/arabica-devnet#network-details","464":"/nodes/arabica-devnet#software-version-numbers","465":"/nodes/arabica-devnet#integrations","466":"/nodes/arabica-devnet#production-rpc-endpoints","467":"/nodes/arabica-devnet#community-rpc-endpoints","468":"/nodes/arabica-devnet#using-consensus-endpoints-with-da-nodes","469":"/nodes/arabica-devnet#data-availability-da-rpc-endpoints-for-bridge-node-sync","470":"/nodes/arabica-devnet#data-availability-da-grpc-endpoints-for-state-access","471":"/nodes/arabica-devnet#arabica-devnet-faucet","472":"/nodes/arabica-devnet#discord","473":"/nodes/arabica-devnet#web","474":"/nodes/arabica-devnet#explorers","475":"/nodes/arabica-devnet#network-upgrades","476":"/nodes/bridge-node#setting-up-a-celestia-bridge-node","477":"/nodes/bridge-node#overview-of-bridge-nodes","478":"/nodes/bridge-node#hardware-requirements","479":"/nodes/bridge-node#setting-up-your-bridge-node","480":"/nodes/bridge-node#setup-the-dependencies","481":"/nodes/bridge-node#deploy-the-celestia-bridge-node","482":"/nodes/bridge-node#install-celestia-node","483":"/nodes/bridge-node#initialize-the-bridge-node","484":"/nodes/bridge-node#run-the-bridge-node","485":"/nodes/bridge-node#optional-run-the-bridge-node-with-a-custom-key","486":"/nodes/bridge-node#optional-migrate-node-id-to-another-server","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#jailing-and-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-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#background","555":"/nodes/celestia-node-troubleshooting#demonstration","556":"/nodes/celestia-node-troubleshooting#examples","557":"/nodes/celestia-node-troubleshooting#mainnet-beta-full-and-mocha-light","558":"/nodes/celestia-node-troubleshooting#mocha-full-and-arabica-light","559":"/nodes/celestia-node-troubleshooting#using-a-custom-rpc-config-address","560":"/nodes/celestia-node-troubleshooting#resetting-your-config","561":"/nodes/celestia-node-troubleshooting#clearing-the-data-store","562":"/nodes/celestia-node-troubleshooting#fatal-headers-given-to-the-heightsub-are-in-the-wrong-order","563":"/nodes/celestia-node-troubleshooting#error-too-many-open-files","564":"/nodes/celestia-node-trusted-hash#syncing-a-light-node-from-a-trusted-hash","565":"/nodes/celestia-node-trusted-hash#for-service-operators","566":"/nodes/celestia-node#install-celestia-node","567":"/nodes/celestia-node#installing-from-source","568":"/nodes/celestia-node#installing-a-pre-built-binary","569":"/nodes/celestia-node#next-steps","570":"/nodes/celestia-node#upgrading-your-binary","571":"/nodes/config-toml#config-toml-guide","572":"/nodes/config-toml#pre-requisites","573":"/nodes/config-toml#understanding-config-toml","574":"/nodes/config-toml#core","575":"/nodes/config-toml#p2p","576":"/nodes/config-toml#bootstrap","577":"/nodes/config-toml#mutual-peers","578":"/nodes/config-toml#services","579":"/nodes/config-toml#trustedhash-and-trustedpeer","580":"/nodes/consensus-node#consensus-node","581":"/nodes/consensus-node#minimum-hardware-requirements","582":"/nodes/consensus-node#set-up-a-consensus-node","583":"/nodes/consensus-node#set-up-the-dependencies","584":"/nodes/consensus-node#install-celestia-app","585":"/nodes/consensus-node#set-up-the-p2p-networks","586":"/nodes/consensus-node#storage-and-pruning-configurations","587":"/nodes/consensus-node#optional-connect-a-consensus-node-to-a-bridge-node","588":"/nodes/consensus-node#enable-transaction-indexing","589":"/nodes/consensus-node#retain-all-block-data","590":"/nodes/consensus-node#query-transactions-by-hash","591":"/nodes/consensus-node#optional-access-historical-state","592":"/nodes/consensus-node#save-on-storage-requirements","593":"/nodes/consensus-node#sync-types","594":"/nodes/consensus-node#option-1-block-sync","595":"/nodes/consensus-node#option-2-state-sync","596":"/nodes/consensus-node#option-3-quick-sync","597":"/nodes/consensus-node#start-the-consensus-node","598":"/nodes/consensus-node#extra-resources-for-consensus-nodes","599":"/nodes/consensus-node#optional-reset-network","600":"/nodes/consensus-node#optional-configure-an-rpc-endpoint","601":"/nodes/consensus-node#expose-rpc","602":"/nodes/consensus-node#note-on-external-address","603":"/nodes/consensus-node#restart-the-node","604":"/nodes/consensus-node#optional-transaction-indexer-configuration-options","605":"/nodes/consensus-node#optional-discard-abci-responses-configuration","606":"/nodes/consensus-node#faq","607":"/nodes/consensus-node#_2-3-committed-an-invalid-block-wrong-block-header-version","608":"/nodes/decide-node#deciding-which-node-to-run","609":"/nodes/decide-node#light-node","610":"/nodes/decide-node#other-da-nodes","611":"/nodes/decide-node#consensus-node","612":"/nodes/docker-images#🐳-docker-setup","613":"/nodes/docker-images#prerequisites","614":"/nodes/docker-images#quick-start","615":"/nodes/docker-images#light-node-setup-with-persistent-storage","616":"/nodes/docker-images#initialize-the-node-store-and-key","617":"/nodes/docker-images#start-the-node","618":"/nodes/docker-images#video-walkthrough","619":"/nodes/docker-images#_2-5-minute-version","620":"/nodes/docker-images#troubleshooting","621":"/nodes/environment#development-environment","622":"/nodes/environment#install-dependencies","623":"/nodes/environment#install-golang","624":"/nodes/full-storage-node#setting-up-a-celestia-full-storage-node","625":"/nodes/full-storage-node#overview-of-full-storage-nodes","626":"/nodes/full-storage-node#hardware-requirements","627":"/nodes/full-storage-node#setting-up-your-full-storage-node","628":"/nodes/full-storage-node#setup-the-dependencies","629":"/nodes/full-storage-node#install-celestia-node","630":"/nodes/full-storage-node#run-the-full-storage-node","631":"/nodes/full-storage-node#initialize-the-full-storage-node","632":"/nodes/full-storage-node#start-the-full-storage-node","633":"/nodes/full-storage-node#optional-run-the-full-storage-node-with-a-custom-key","634":"/nodes/full-storage-node#optional-migrate-node-id-to-another-server","635":"/nodes/full-storage-node#optional-start-the-full-storage-node-with-systemd","636":"/nodes/full-storage-node#stop-the-full-storage-node","637":"/nodes/ibc-relayer#ibc-relaying-guide","638":"/nodes/ibc-relayer#hermes","639":"/nodes/ibc-relayer#configuration","640":"/nodes/ibc-relayer#add-relayer-wallets","641":"/nodes/ibc-relayer#verify-configuration-files","642":"/nodes/ibc-relayer#create-a-connection-between-2-chains","643":"/nodes/ibc-relayer#create-clients","644":"/nodes/ibc-relayer#open-connection-over-new-clients","645":"/nodes/ibc-relayer#configure-channels-in-hermes","646":"/nodes/ibc-relayer#start-the-relayer","647":"/nodes/ibc-relayer#transfer","648":"/nodes/ibc-relayer#token-filter","649":"/nodes/instantiate-testnet#celestia-app-network-instantiation-guide","650":"/nodes/instantiate-testnet#hardware-requirements","651":"/nodes/instantiate-testnet#setup-dependencies","652":"/nodes/instantiate-testnet#celestia-app-installation","653":"/nodes/instantiate-testnet#spin-up-a-celestia-testnet","654":"/nodes/instantiate-testnet#optional-reset-working-directory","655":"/nodes/instantiate-testnet#initialize-a-working-directory","656":"/nodes/instantiate-testnet#create-a-new-key","657":"/nodes/instantiate-testnet#add-genesis-account-keyname","658":"/nodes/instantiate-testnet#optional-adding-other-validators","659":"/nodes/instantiate-testnet#create-the-genesis-transaction-for-new-chain","660":"/nodes/instantiate-testnet#creating-the-genesis-json-file","661":"/nodes/instantiate-testnet#modify-your-config-file","662":"/nodes/instantiate-testnet#add-your-node-as-a-persistent-peer","663":"/nodes/instantiate-testnet#instantiate-the-network","664":"/nodes/light-node#setting-up-a-celestia-light-node","665":"/nodes/light-node#overview-of-light-nodes","666":"/nodes/light-node#hardware-requirements","667":"/nodes/light-node#quickstart-run-a-light-node-in-your-browser","668":"/nodes/light-node#setting-up-your-light-node","669":"/nodes/light-node#install-celestia-node","670":"/nodes/light-node#initialize-the-light-node","671":"/nodes/light-node#start-the-light-node","672":"/nodes/light-node#keys-and-wallets","673":"/nodes/light-node#testnet-tokens","674":"/nodes/light-node#optional-run-the-light-node-with-a-custom-key","675":"/nodes/light-node#optional-migrate-node-id-to-another-server","676":"/nodes/light-node#optional-start-light-node-with-systemd","677":"/nodes/light-node#data-availability-sampling","678":"/nodes/mainnet#mainnet-beta","679":"/nodes/mainnet#network-stability-and-upgrades","680":"/nodes/mainnet#network-details","681":"/nodes/mainnet#software-version-numbers","682":"/nodes/mainnet#network-parameters","683":"/nodes/mainnet#maximum-bytes","684":"/nodes/mainnet#integrations","685":"/nodes/mainnet#production-rpc-endpoints","686":"/nodes/mainnet#consensus-nodes","687":"/nodes/mainnet#community-consensus-rpc-endpoints","688":"/nodes/mainnet#community-api-endpoints","689":"/nodes/mainnet#community-grpc-endpoints","690":"/nodes/mainnet#community-websocket-endpoints","691":"/nodes/mainnet#data-availability-nodes","692":"/nodes/mainnet#community-data-availability-da-rpc-endpoints-for-bridge-node-sync","693":"/nodes/mainnet#community-data-availability-da-grpc-endpoints-for-state-access","694":"/nodes/mainnet#archival-da-rpc-endpoints","695":"/nodes/mainnet#grove-archival-endpoints","696":"/nodes/mainnet#explorers","697":"/nodes/mainnet#analytics","698":"/nodes/mainnet#network-upgrades","699":"/nodes/mocha-testnet#mocha-testnet","700":"/nodes/mocha-testnet#network-details","701":"/nodes/mocha-testnet#software-version-numbers","702":"/nodes/mocha-testnet#rpc-for-da-bridge-full-and-light-nodes","703":"/nodes/mocha-testnet#production-rpc-endpoints","704":"/nodes/mocha-testnet#community-data-availability-da-rpc-endpoints-for-bridge-node-sync","705":"/nodes/mocha-testnet#community-data-availability-da-grpc-endpoints-for-state-access","706":"/nodes/mocha-testnet#community-rpc-endpoints","707":"/nodes/mocha-testnet#community-api-endpoints","708":"/nodes/mocha-testnet#community-grpc-endpoints","709":"/nodes/mocha-testnet#community-bridge-and-full-node-endpoints","710":"/nodes/mocha-testnet#mocha-testnet-faucet","711":"/nodes/mocha-testnet#analytics","712":"/nodes/mocha-testnet#explorers","713":"/nodes/mocha-testnet#network-upgrades","714":"/nodes/network-upgrade-process#celestia-network-upgrade-process","715":"/nodes/network-upgrade-process#general-process","716":"/nodes/network-upgrade-process#lemongrass-network-upgrade","717":"/nodes/overview#overview-to-running-nodes-on-celestia","718":"/nodes/overview#recommended-celestia-node-requirements","719":"/nodes/overview#data-availability-nodes","720":"/nodes/overview#consensus-nodes","721":"/nodes/participate#participate-in-the-celestia-networks","722":"/nodes/participate#mainnet-beta","723":"/nodes/participate#compatible-software-versions-for-mainnet-beta","724":"/nodes/participate#testnets","725":"/nodes/participate#arabica-devnet","726":"/nodes/participate#compatible-software-versions-for-arabica-devnet","727":"/nodes/participate#mocha-testnet","728":"/nodes/participate#compatible-software-versions-for-mocha-testnet","729":"/nodes/participate#network-upgrades","730":"/nodes/quick-start#quick-start-guide","731":"/nodes/quick-start#celestia-node","732":"/nodes/quick-start#celestia-app","733":"/nodes/quick-start#getting-started","734":"/nodes/systemd#setting-up-your-node-as-a-background-process-with-systemd","735":"/nodes/systemd#consensus-nodes","736":"/nodes/systemd#start-the-celestia-app-with-systemd","737":"/nodes/systemd#data-availability-nodes","738":"/nodes/systemd#celestia-full-storage-node","739":"/nodes/systemd#celestia-bridge-node","740":"/nodes/systemd#celestia-light-node","741":"/nodes/validator-node#setting-up-a-celestia-validator-node","742":"/nodes/validator-node#hardware-requirements","743":"/nodes/validator-node#setting-up-a-validator-node","744":"/nodes/validator-node#wallet","745":"/nodes/validator-node#delegate-stake-to-a-validator","746":"/nodes/validator-node#optional-deploy-the-celestia-node","747":"/nodes/validator-node#install-celestia-node","748":"/nodes/validator-node#initialize-the-bridge-node","749":"/nodes/validator-node#run-the-bridge-node","750":"/nodes/validator-node#optional-start-the-bridge-node-with-systemd","751":"/nodes/validator-node#run-the-validator-node","752":"/nodes/validator-node#submit-your-validator-information","753":"/nodes/validator-node#optional-transaction-indexer-configuration-options","754":"/nodes/validator-node#additional-resources","755":"/nodes/validator-node#faq","756":"/nodes/validator-node#_2-3-committed-an-invalid-block-wrong-block-header-version"},"fieldIds":{"title":0,"titles":1,"text":2},"fieldLength":{"0":[3,1,24],"1":[3,3,22],"2":[2,3,68],"3":[2,3,34],"4":[3,3,25],"5":[5,1,1],"6":[2,5,62],"7":[2,5,114],"8":[2,5,57],"9":[1,5,43],"10":[1,5,39],"11":[2,5,22],"12":[2,7,38],"13":[2,7,57],"14":[3,7,50],"15":[3,7,34],"16":[1,5,47],"17":[5,1,1],"18":[4,5,57],"19":[3,5,1],"20":[2,7,53],"21":[2,7,41],"22":[2,7,110],"23":[2,7,173],"24":[2,7,62],"25":[1,5,45],"26":[2,6,147],"27":[2,5,53],"28":[2,5,17],"29":[3,1,31],"30":[3,3,129],"31":[3,3,69],"32":[3,3,72],"33":[3,3,39],"34":[1,6,164],"35":[3,6,124],"36":[1,6,150],"37":[1,6,190],"38":[1,6,170],"39":[3,6,159],"40":[1,6,132],"41":[4,6,145],"42":[9,6,167],"43":[4,1,28],"44":[2,4,64],"45":[2,4,1],"46":[6,6,71],"47":[2,6,70],"48":[1,6,35],"49":[1,6,28],"50":[1,6,48],"51":[3,1,37],"52":[3,3,12],"53":[6,3,44],"54":[8,3,58],"55":[7,3,48],"56":[6,3,23],"57":[2,3,37],"58":[2,1,143],"59":[8,1,29],"60":[6,8,1],"61":[7,11,47],"62":[7,11,68],"63":[6,8,1],"64":[10,11,81],"65":[5,11,40],"66":[6,1,77],"67":[1,6,52],"68":[1,6,10],"69":[6,7,63],"70":[6,7,58],"71":[8,7,238],"72":[8,7,115],"73":[10,7,74],"74":[12,7,39],"75":[11,7,129],"76":[9,7,238],"77":[10,7,56],"78":[7,7,231],"79":[4,7,49],"80":[1,6,17],"81":[2,7,75],"82":[4,7,32],"83":[2,11,13],"84":[2,11,13],"85":[4,7,48],"86":[4,7,48],"87":[7,1,1],"88":[1,7,20],"89":[4,7,71],"90":[6,7,34],"91":[8,1,1],"92":[1,8,51],"93":[2,8,33],"94":[3,10,174],"95":[3,10,128],"96":[3,10,77],"97":[5,10,86],"98":[2,8,17],"99":[4,1,1],"100":[2,4,1],"101":[1,6,9],"102":[3,6,57],"103":[2,6,93],"104":[2,4,43],"105":[1,4,50],"106":[3,4,31],"107":[6,4,124],"108":[4,1,1],"109":[3,4,74],"110":[3,4,149],"111":[2,4,168],"112":[3,6,101],"113":[2,6,115],"114":[3,4,1],"115":[3,7,164],"116":[2,7,1],"117":[4,9,10],"118":[8,9,13],"119":[4,9,39],"120":[3,1,1],"121":[1,3,118],"122":[5,3,101],"123":[3,7,1],"124":[3,7,130],"125":[3,3,72],"126":[5,6,50],"127":[2,11,78],"128":[2,11,56],"129":[14,6,237],"130":[4,6,44],"131":[7,6,13],"132":[2,13,163],"133":[2,13,31],"134":[9,6,35],"135":[2,15,89],"136":[5,6,60],"137":[2,11,108],"138":[2,11,50],"139":[9,3,86],"140":[1,11,54],"141":[1,11,198],"142":[1,11,94],"143":[1,11,47],"144":[1,11,113],"145":[1,11,185],"146":[5,11,136],"147":[5,11,94],"148":[6,11,379],"149":[1,3,58],"150":[4,1,47],"151":[1,4,53],"152":[3,5,23],"153":[5,8,92],"154":[5,8,83],"155":[4,8,41],"156":[4,8,63],"157":[3,5,85],"158":[5,8,31],"159":[6,8,119],"160":[7,8,119],"161":[4,8,15],"162":[4,8,2],"163":[2,4,57],"164":[8,5,70],"165":[9,11,50],"166":[9,11,2],"167":[9,11,42],"168":[7,5,107],"169":[8,10,18],"170":[8,10,27],"171":[2,4,66],"172":[7,5,88],"173":[8,10,36],"174":[8,10,2],"175":[7,5,30],"176":[6,10,82],"177":[8,10,2],"178":[8,10,26],"179":[4,5,79],"180":[5,9,2],"181":[5,9,13],"182":[1,4,48],"183":[1,4,1],"184":[12,5,104],"185":[4,1,14],"186":[3,4,34],"187":[4,7,232],"188":[4,7,104],"189":[7,7,113],"190":[3,13,59],"191":[3,13,61],"192":[3,13,22],"193":[4,7,116],"194":[4,10,40],"195":[2,10,180],"196":[4,7,186],"197":[8,7,36],"198":[5,4,34],"199":[4,9,175],"200":[6,9,176],"201":[4,1,89],"202":[2,4,75],"203":[6,4,92],"204":[3,4,63],"205":[6,1,1],"206":[4,6,119],"207":[3,9,7],"208":[7,6,1],"209":[3,12,40],"210":[3,12,42],"211":[3,12,69],"212":[5,6,126],"213":[4,6,67],"214":[5,9,30],"215":[2,9,17],"216":[2,9,43],"217":[7,1,1],"218":[5,7,86],"219":[4,7,58],"220":[3,7,57],"221":[2,1,32],"222":[7,2,56],"223":[3,2,49],"224":[2,4,34],"225":[1,6,8],"226":[1,6,40],"227":[1,4,22],"228":[1,4,20],"229":[1,4,12],"230":[1,4,12],"231":[2,2,22],"232":[2,1,43],"233":[4,2,1],"234":[3,6,23],"235":[4,6,9],"236":[2,6,10],"237":[5,2,41],"238":[6,2,100],"239":[4,2,1],"240":[5,6,81],"241":[1,6,123],"242":[1,6,107],"243":[6,1,42],"244":[5,6,42],"245":[1,11,56],"246":[5,11,106],"247":[5,11,32],"248":[5,11,33],"249":[6,11,5],"250":[4,6,1],"251":[1,10,16],"252":[3,10,144],"253":[5,10,147],"254":[2,1,95],"255":[5,1,1],"256":[1,5,37],"257":[2,5,31],"258":[1,5,60],"259":[6,5,52],"260":[6,11,36],"261":[8,11,42],"262":[9,5,69],"263":[5,5,37],"264":[9,5,13],"265":[7,12,11],"266":[7,5,87],"267":[6,12,48],"268":[4,1,27],"269":[2,4,66],"270":[4,4,132],"271":[4,4,86],"272":[4,4,101],"273":[7,4,59],"274":[2,4,11],"275":[5,1,18],"276":[2,5,28],"277":[4,5,84],"278":[4,7,27],"279":[3,7,62],"280":[1,7,36],"281":[1,7,28],"282":[6,7,81],"283":[5,1,43],"284":[3,5,109],"285":[2,7,26],"286":[2,5,35],"287":[5,1,1],"288":[1,5,27],"289":[8,5,39],"290":[8,5,1],"291":[6,12,26],"292":[6,12,36],"293":[2,5,19],"294":[4,7,15],"295":[4,7,16],"296":[5,5,56],"297":[2,1,28],"298":[2,2,46],"299":[1,3,16],"300":[1,3,48],"301":[3,3,19],"302":[2,2,66],"303":[3,3,12],"304":[5,1,26],"305":[1,5,1],"306":[1,6,19],"307":[1,6,66],"308":[2,5,30],"309":[3,5,11],"310":[5,8,55],"311":[5,8,123],"312":[3,8,107],"313":[3,5,39],"314":[2,6,124],"315":[2,6,77],"316":[3,8,67],"317":[2,8,65],"318":[7,10,70],"319":[2,6,118],"320":[4,8,10],"321":[2,6,113],"322":[4,6,74],"323":[1,6,1],"324":[3,7,46],"325":[6,7,43],"326":[4,7,49],"327":[4,7,28],"328":[5,7,107],"329":[2,7,20],"330":[5,7,6],"331":[7,7,72],"332":[2,7,17],"333":[1,7,21],"334":[2,6,35],"335":[2,5,1],"336":[5,7,124],"337":[6,7,21],"338":[1,7,46],"339":[4,1,21],"340":[3,4,83],"341":[5,4,84],"342":[5,4,51],"343":[2,4,57],"344":[8,1,47],"345":[2,8,3],"346":[5,10,73],"347":[5,8,48],"348":[10,8,115],"349":[1,8,22],"350":[2,1,80],"351":[1,2,28],"352":[8,3,145],"353":[2,3,42],"354":[4,2,1],"355":[4,5,38],"356":[4,5,100],"357":[2,5,110],"358":[2,5,232],"359":[2,5,84],"360":[3,5,160],"361":[2,2,60],"362":[4,1,29],"363":[2,4,63],"364":[4,4,107],"365":[4,4,87],"366":[7,4,57],"367":[2,4,11],"368":[4,1,22],"369":[3,4,106],"370":[5,1,27],"371":[3,5,94],"372":[4,5,26],"373":[4,9,103],"374":[3,12,141],"375":[3,12,152],"376":[3,12,78],"377":[10,9,118],"378":[1,5,11],"379":[6,6,14],"380":[5,6,19],"381":[6,6,100],"382":[9,6,199],"383":[5,6,21],"384":[6,6,83],"385":[2,1,73],"386":[3,2,42],"387":[1,2,37],"388":[4,1,21],"389":[8,4,181],"390":[6,4,82],"391":[6,4,100],"392":[3,1,1],"393":[5,3,82],"394":[7,3,73],"395":[9,3,41],"396":[6,3,62],"397":[15,3,80],"398":[8,3,65],"399":[5,3,33],"400":[8,3,70],"401":[10,3,26],"402":[8,3,51],"403":[10,3,73],"404":[11,3,42],"405":[15,3,34],"406":[5,1,69],"407":[5,5,122],"408":[1,9,86],"409":[6,9,122],"410":[5,5,130],"411":[6,5,1],"412":[3,11,120],"413":[4,1,172],"414":[1,1,90],"415":[7,1,158],"416":[3,7,163],"417":[4,1,44],"418":[4,4,1],"419":[5,4,1],"420":[6,8,20],"421":[9,8,15],"422":[5,8,25],"423":[6,8,22],"424":[5,4,1],"425":[6,8,17],"426":[7,8,29],"427":[6,8,22],"428":[5,4,1],"429":[6,8,7],"430":[12,8,13],"431":[5,8,12],"432":[4,8,24],"433":[3,1,1],"434":[2,3,70],"435":[3,3,44],"436":[4,1,60],"437":[7,4,93],"438":[4,4,143],"439":[4,1,1],"440":[5,4,68],"441":[1,4,73],"442":[2,4,1],"443":[2,5,35],"444":[2,5,37],"445":[4,4,85],"446":[1,8,111],"447":[3,1,30],"448":[2,3,34],"449":[2,3,31],"450":[3,1,1],"451":[4,3,33],"452":[3,3,1],"453":[3,4,33],"454":[3,4,68],"455":[3,4,41],"456":[2,4,41],"457":[1,4,1],"458":[3,5,15],"459":[3,5,27],"460":[4,5,9],"461":[2,1,45],"462":[4,2,64],"463":[2,2,1],"464":[3,4,1],"465":[1,2,49],"466":[3,3,37],"467":[3,3,95],"468":[6,3,1],"469":[9,9,39],"470":[8,9,111],"471":[3,2,24],"472":[1,3,41],"473":[1,3,12],"474":[1,2,19],"475":[2,2,21],"476":[6,1,23],"477":[4,6,108],"478":[2,6,30],"479":[5,6,16],"480":[3,7,7],"481":[5,6,1],"482":[3,8,17],"483":[4,8,102],"484":[4,8,112],"485":[9,9,49],"486":[7,9,30],"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,47],"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":[5,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,41],"527":[3,10,50],"528":[3,10,10],"529":[6,10,96],"530":[8,10,39],"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,110],"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,1],"554":[1,8,100],"555":[1,8,81],"556":[1,8,1],"557":[6,9,58],"558":[5,9,42],"559":[6,9,47],"560":[3,1,81],"561":[4,1,26],"562":[10,1,71],"563":[7,1,109],"564":[7,1,92],"565":[3,7,27],"566":[3,1,1],"567":[3,3,126],"568":[5,3,102],"569":[2,3,39],"570":[3,3,26],"571":[3,1,1],"572":[2,3,12],"573":[3,3,34],"574":[1,4,35],"575":[1,4,1],"576":[1,5,45],"577":[2,5,31],"578":[1,4,1],"579":[3,5,61],"580":[2,1,24],"581":[3,2,30],"582":[5,2,16],"583":[4,6,7],"584":[3,6,8],"585":[5,6,140],"586":[4,2,1],"587":[7,6,34],"588":[3,12,6],"589":[4,12,25],"590":[4,6,25],"591":[4,6,49],"592":[4,6,54],"593":[2,2,50],"594":[4,4,41],"595":[4,4,113],"596":[4,4,72],"597":[4,2,78],"598":[5,2,1],"599":[3,7,22],"600":[5,7,28],"601":[2,12,65],"602":[4,12,56],"603":[3,12,13],"604":[5,7,94],"605":[5,7,90],"606":[1,2,1],"607":[10,3,159],"608":[5,1,16],"609":[2,5,31],"610":[3,6,18],"611":[2,5,29],"612":[3,1,71],"613":[1,3,14],"614":[2,3,105],"615":[6,3,77],"616":[6,8,79],"617":[3,8,64],"618":[2,3,1],"619":[4,5,1],"620":[1,3,54],"621":[2,1,24],"622":[2,2,80],"623":[2,2,108],"624":[7,1,31],"625":[5,7,26],"626":[2,7,31],"627":[6,7,16],"628":[3,9,11],"629":[3,7,10],"630":[5,10,1],"631":[5,12,17],"632":[5,12,141],"633":[10,10,50],"634":[7,16,31],"635":[8,10,24],"636":[5,10,32],"637":[3,1,93],"638":[1,3,85],"639":[1,4,180],"640":[3,4,84],"641":[3,4,49],"642":[6,3,52],"643":[2,3,31],"644":[5,5,85],"645":[4,5,44],"646":[3,3,7],"647":[1,3,41],"648":[2,4,37],"649":[5,1,42],"650":[2,5,8],"651":[2,5,11],"652":[3,5,12],"653":[5,5,32],"654":[4,9,38],"655":[4,9,44],"656":[4,9,49],"657":[4,9,29],"658":[4,9,48],"659":[7,9,76],"660":[5,9,74],"661":[4,9,38],"662":[7,9,66],"663":[3,9,42],"664":[6,1,25],"665":[4,6,42],"666":[2,6,30],"667":[8,6,27],"668":[5,6,22],"669":[3,7,10],"670":[4,6,37],"671":[4,6,87],"672":[3,8,80],"673":[2,11,38],"674":[9,8,49],"675":[7,13,30],"676":[6,8,15],"677":[3,6,16],"678":[2,1,48],"679":[4,2,66],"680":[2,2,1],"681":[3,2,1],"682":[2,2,31],"683":[2,4,109],"684":[1,2,56],"685":[3,3,59],"686":[2,3,4],"687":[4,5,74],"688":[3,5,55],"689":[3,5,57],"690":[3,5,14],"691":[3,3,6],"692":[10,6,39],"693":[9,6,129],"694":[4,6,37],"695":[3,10,31],"696":[1,2,30],"697":[1,2,26],"698":[2,2,30],"699":[2,1,106],"700":[2,2,1],"701":[3,2,1],"702":[8,2,1],"703":[3,10,59],"704":[10,10,39],"705":[9,10,99],"706":[3,2,67],"707":[3,2,81],"708":[3,2,85],"709":[6,2,33],"710":[3,2,63],"711":[1,2,14],"712":[1,2,32],"713":[2,2,30],"714":[4,1,63],"715":[2,4,86],"716":[3,5,118],"717":[6,1,71],"718":[4,6,1],"719":[3,6,30],"720":[2,6,47],"721":[5,1,1],"722":[2,5,39],"723":[6,7,1],"724":[1,5,12],"725":[2,6,45],"726":[6,8,1],"727":[2,6,48],"728":[6,6,1],"729":[2,5,30],"730":[3,1,25],"731":[2,3,48],"732":[2,3,29],"733":[2,3,76],"734":[10,1,13],"735":[2,10,21],"736":[6,12,113],"737":[3,10,1],"738":[4,13,72],"739":[3,13,106],"740":[3,13,93],"741":[6,1,23],"742":[2,6,30],"743":[5,6,25],"744":[1,6,8],"745":[5,6,113],"746":[5,6,48],"747":[3,10,10],"748":[4,10,46],"749":[4,10,8],"750":[7,12,23],"751":[4,6,192],"752":[4,6,16],"753":[5,6,19],"754":[2,6,14],"755":[1,6,1],"756":[10,7,182]},"averageFieldLength":[3.922060766182301,5.635402906208716,55.32760898282695],"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":"Celestia.org Code of Conduct","titles":[]},"6":{"title":"Our Pledge","titles":["Celestia.org Code of Conduct"]},"7":{"title":"Our Standards","titles":["Celestia.org Code of Conduct"]},"8":{"title":"Enforcement Responsibilities","titles":["Celestia.org Code of Conduct"]},"9":{"title":"Scope","titles":["Celestia.org Code of Conduct"]},"10":{"title":"Enforcement","titles":["Celestia.org Code of Conduct"]},"11":{"title":"Enforcement Guidelines","titles":["Celestia.org Code of Conduct"]},"12":{"title":"1. Correction","titles":["Celestia.org Code of Conduct","Enforcement Guidelines"]},"13":{"title":"2. Warning","titles":["Celestia.org Code of Conduct","Enforcement Guidelines"]},"14":{"title":"3. Temporary Ban","titles":["Celestia.org Code of Conduct","Enforcement Guidelines"]},"15":{"title":"4. Permanent Ban","titles":["Celestia.org Code of Conduct","Enforcement Guidelines"]},"16":{"title":"Attribution","titles":["Celestia.org Code of Conduct"]},"17":{"title":"The Celestia Foundation Delegation Program","titles":[]},"18":{"title":"Objectives of the program","titles":["The Celestia Foundation Delegation Program"]},"19":{"title":"Foundation delegation process","titles":["The Celestia Foundation Delegation Program"]},"20":{"title":"Program launch","titles":["The Celestia Foundation Delegation Program","Foundation delegation process"]},"21":{"title":"Cohort process","titles":["The Celestia Foundation Delegation Program","Foundation delegation process"]},"22":{"title":"Key Points","titles":["The Celestia Foundation Delegation Program","Foundation delegation process"]},"23":{"title":"Eligibility criteria","titles":["The Celestia Foundation Delegation Program","Foundation delegation process"]},"24":{"title":"Undelegation criteria","titles":["The Celestia Foundation Delegation Program","Foundation delegation process"]},"25":{"title":"Application","titles":["The Celestia Foundation Delegation Program"]},"26":{"title":"Application details","titles":["The Celestia Foundation Delegation Program","Application"]},"27":{"title":"Cohort information","titles":["The Celestia Foundation Delegation Program"]},"28":{"title":"Feedback process","titles":["The Celestia Foundation Delegation Program"]},"29":{"title":"Modular Meetup guide","titles":[]},"30":{"title":"Before the Meetup","titles":["Modular Meetup guide"]},"31":{"title":"During the Meetup","titles":["Modular Meetup guide"]},"32":{"title":"After the Meetup","titles":["Modular Meetup guide"]},"33":{"title":"Logistics and guidance","titles":["Modular Meetup guide"]},"34":{"title":"Venue","titles":["Modular Meetup guide","Logistics and guidance"]},"35":{"title":"Catering and refreshments","titles":["Modular Meetup guide","Logistics and guidance"]},"36":{"title":"Audience","titles":["Modular Meetup guide","Logistics and guidance"]},"37":{"title":"Speakers","titles":["Modular Meetup guide","Logistics and guidance"]},"38":{"title":"Sponsors","titles":["Modular Meetup guide","Logistics and guidance"]},"39":{"title":"Communications and marketing","titles":["Modular Meetup guide","Logistics and guidance"]},"40":{"title":"Recording","titles":["Modular Meetup guide","Logistics and guidance"]},"41":{"title":"Utilizing Meetup.com platform","titles":["Modular Meetup guide","Logistics and guidance"]},"42":{"title":"Onboarding questions for community members joining a Modular Meetup","titles":["Modular Meetup guide","Logistics and guidance"]},"43":{"title":"Celestia Modular Meetup program","titles":[]},"44":{"title":"Program description","titles":["Celestia Modular Meetup program"]},"45":{"title":"Important info","titles":["Celestia Modular Meetup program"]},"46":{"title":"Celestia.org Community Code of Conduct","titles":["Celestia Modular Meetup program","Important info"]},"47":{"title":"Signup form","titles":["Celestia Modular Meetup program","Important info"]},"48":{"title":"Emails","titles":["Celestia Modular Meetup program","Important info"]},"49":{"title":"Discord","titles":["Celestia Modular Meetup program","Important info"]},"50":{"title":"Materials","titles":["Celestia Modular Meetup program","Important info"]},"51":{"title":"Modular Meetup Toolkit","titles":[]},"52":{"title":"Celestia branding guidelines","titles":["Modular Meetup Toolkit"]},"53":{"title":"Sample “Introduction to Modularity” workshop presentation","titles":["Modular Meetup Toolkit"]},"54":{"title":"Sample “Run a Celestia light node” workshop presentation","titles":["Modular Meetup Toolkit"]},"55":{"title":"Sample “Deploy a Sovereign Rollup” workshop presentation","titles":["Modular Meetup Toolkit"]},"56":{"title":"Sample “Modular Meetup Introduction” workshop presentation","titles":["Modular Meetup Toolkit"]},"57":{"title":"Swag logistics","titles":["Modular Meetup Toolkit"]},"58":{"title":"Speaker list","titles":[]},"59":{"title":"Bridging in and out of your Orbit rollup","titles":[]},"60":{"title":"Bridge in (deposit) to your rollup","titles":["Bridging in and out of your Orbit rollup"]},"61":{"title":"Step 1: Add your custom chain config","titles":["Bridging in and out of your Orbit rollup","Bridge in (deposit) to your rollup"]},"62":{"title":"Step 2: Deposit to your Orbit rollup","titles":["Bridging in and out of your Orbit rollup","Bridge in (deposit) to your rollup"]},"63":{"title":"Bridge out (withdrawal) from your rollup","titles":["Bridging in and out of your Orbit rollup"]},"64":{"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"]},"65":{"title":"Step 2: Claim your withdrawal","titles":["Bridging in and out of your Orbit rollup","Bridge out (withdrawal) from your rollup"]},"66":{"title":"Quickstart: Deploy an Arbitrum Orbit rollup","titles":[]},"67":{"title":"Prerequisites","titles":["Quickstart: Deploy an Arbitrum Orbit rollup"]},"68":{"title":"Setup","titles":["Quickstart: Deploy an Arbitrum Orbit rollup"]},"69":{"title":"Step 1: Acquire Arbitrum Sepolia ETH","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"70":{"title":"Step 2: Pick your deployment type","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"71":{"title":"Step 3: Configure your Orbit chain\'s deployment","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"72":{"title":"Step 3: Review & Deploy your Orbit chain","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"73":{"title":"Step 4: Download your chain\'s configuration files and launch your chain","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"74":{"title":"Step 5: Clone the setup script repository and add your configuration files","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"75":{"title":"Step 6: Pick an L2 RPC URL for the Batch Poster","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"76":{"title":"Step 7: Run your light node for Mocha testnet","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"77":{"title":"Step 8: Run your chain\'s node and block explorer","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"78":{"title":"Step 9: Finish setting up your chain","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"79":{"title":"Congratulations with Celestia underneath","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Setup"]},"80":{"title":"Appendix","titles":["Quickstart: Deploy an Arbitrum Orbit rollup"]},"81":{"title":"Compatibility matrix","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Appendix"]},"82":{"title":"Blobstream X contract deployments","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Appendix"]},"83":{"title":"Arbitrum Sepolia","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Appendix","Blobstream X contract deployments"]},"84":{"title":"Base Sepolia","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Appendix","Blobstream X contract deployments"]},"85":{"title":"Arbitrum Sepolia additional deployments","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Appendix"]},"86":{"title":"Base Sepolia additional deployments","titles":["Quickstart: Deploy an Arbitrum Orbit rollup","Appendix"]},"87":{"title":"Running a full node and/or validator","titles":[]},"88":{"title":"Prerequisites","titles":["Running a full node and/or validator"]},"89":{"title":"Running a full node","titles":["Running a full node and/or validator"]},"90":{"title":"Running a full node with validation","titles":["Running a full node and/or validator"]},"91":{"title":"Introduction to Arbitrum rollups with Celestia as DA","titles":[]},"92":{"title":"Overview","titles":["Introduction to Arbitrum rollups with Celestia as DA"]},"93":{"title":"Key components","titles":["Introduction to Arbitrum rollups with Celestia as DA"]},"94":{"title":"DA provider implementation","titles":["Introduction to Arbitrum rollups with Celestia as DA","Key components"]},"95":{"title":"Preimage Oracle Implementation","titles":["Introduction to Arbitrum rollups with Celestia as DA","Key components"]},"96":{"title":"Blobstream X implementation","titles":["Introduction to Arbitrum rollups with Celestia as DA","Key components"]},"97":{"title":"Ethereum fallback mechanism in Nitro","titles":["Introduction to Arbitrum rollups with Celestia as DA","Key components"]},"98":{"title":"Next steps","titles":["Introduction to Arbitrum rollups with Celestia as DA"]},"99":{"title":"Integrate with Blobstream contracts","titles":[]},"100":{"title":"Getting started","titles":["Integrate with Blobstream contracts"]},"101":{"title":"Prerequisites","titles":["Integrate with Blobstream contracts","Getting started"]},"102":{"title":"Installing Blobstream contracts","titles":["Integrate with Blobstream contracts","Getting started"]},"103":{"title":"Example usage","titles":["Integrate with Blobstream contracts","Getting started"]},"104":{"title":"Data structures","titles":["Integrate with Blobstream contracts"]},"105":{"title":"Interface","titles":["Integrate with Blobstream contracts"]},"106":{"title":"Querying the proof","titles":["Integrate with Blobstream contracts"]},"107":{"title":"Verifying data inclusion for fraud proofs","titles":["Integrate with Blobstream contracts"]},"108":{"title":"Integrate with Blobstream client","titles":[]},"109":{"title":"Blobstream demo rollup","titles":["Integrate with Blobstream client"]},"110":{"title":"Defining a chain","titles":["Integrate with Blobstream client"]},"111":{"title":"Rollup sequencer","titles":["Integrate with Blobstream client"]},"112":{"title":"Committing to data","titles":["Integrate with Blobstream client","Rollup sequencer"]},"113":{"title":"Creating blocks","titles":["Integrate with Blobstream client","Rollup sequencer"]},"114":{"title":"Rollup full node","titles":["Integrate with Blobstream client"]},"115":{"title":"Downloading the block","titles":["Integrate with Blobstream client","Rollup full node"]},"116":{"title":"More documentation","titles":["Integrate with Blobstream client","Rollup full node"]},"117":{"title":"Proving inclusion via Blobstream","titles":["Integrate with Blobstream client","Rollup full node","More documentation"]},"118":{"title":"Submitting block data to Celestia via light node","titles":["Integrate with Blobstream client","Rollup full node","More documentation"]},"119":{"title":"Posting headers to Ethereum","titles":["Integrate with Blobstream client","Rollup full node","More documentation"]},"120":{"title":"Blobstream proofs queries","titles":[]},"121":{"title":"Prerequisites","titles":["Blobstream proofs queries"]},"122":{"title":"Overview of the proof queries","titles":["Blobstream proofs queries"]},"123":{"title":"The Celestia square","titles":["Blobstream proofs queries","Overview of the proof queries"]},"124":{"title":"The commitment scheme","titles":["Blobstream proofs queries","Overview of the proof queries"]},"125":{"title":"Hands-on demonstration","titles":["Blobstream proofs queries"]},"126":{"title":"1. Data root inclusion proof","titles":["Blobstream proofs queries","Hands-on demonstration"]},"127":{"title":"HTTP query","titles":["Blobstream proofs queries","Hands-on demonstration","1. Data root inclusion proof"]},"128":{"title":"Golang client","titles":["Blobstream proofs queries","Hands-on demonstration","1. Data root inclusion proof"]},"129":{"title":"Full example of proving that a Celestia block was committed to by Blobstream contract","titles":["Blobstream proofs queries","Hands-on demonstration"]},"130":{"title":"2. Transaction inclusion proof","titles":["Blobstream proofs queries","Hands-on demonstration"]},"131":{"title":"Transaction inclusion proof using the transaction hash","titles":["Blobstream proofs queries","Hands-on demonstration"]},"132":{"title":"HTTP request","titles":["Blobstream proofs queries","Hands-on demonstration","Transaction inclusion proof using the transaction hash"]},"133":{"title":"Golang client","titles":["Blobstream proofs queries","Hands-on demonstration","Transaction inclusion proof using the transaction hash"]},"134":{"title":"Blob inclusion proof using the corresponding PFB transaction hash","titles":["Blobstream proofs queries","Hands-on demonstration"]},"135":{"title":"Golang client","titles":["Blobstream proofs queries","Hands-on demonstration","Blob inclusion proof using the corresponding PFB transaction hash"]},"136":{"title":"Specific share range inclusion proof","titles":["Blobstream proofs queries","Hands-on demonstration"]},"137":{"title":"HTTP request","titles":["Blobstream proofs queries","Hands-on demonstration","Specific share range inclusion proof"]},"138":{"title":"Golang client","titles":["Blobstream proofs queries","Hands-on demonstration","Specific share range inclusion proof"]},"139":{"title":"Converting the proofs to be usable in the DAVerifier library","titles":["Blobstream proofs queries"]},"140":{"title":"data","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"141":{"title":"shareProofs","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"142":{"title":"namespace","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"143":{"title":"rowRoots","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"144":{"title":"rowProofs","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"145":{"title":"attestationProof","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"146":{"title":"Querying the proof\'s tupleRootNonce","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"147":{"title":"Listening for new data commitments","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"148":{"title":"Example rollup that uses the DAVerifier","titles":["Blobstream proofs queries","Converting the proofs to be usable in the DAVerifier library"]},"149":{"title":"Conclusion","titles":["Blobstream proofs queries"]},"150":{"title":"Introduction to Blobstream rollups","titles":[]},"151":{"title":"Concepts","titles":["Introduction to Blobstream rollups"]},"152":{"title":"Blob share commitment","titles":["Introduction to Blobstream rollups","Concepts"]},"153":{"title":"Blob share commitment: Proof details","titles":["Introduction to Blobstream rollups","Concepts","Blob share commitment"]},"154":{"title":"Blob share commitment: Compact proofs","titles":["Introduction to Blobstream rollups","Concepts","Blob share commitment"]},"155":{"title":"Blob share commitment: Pros","titles":["Introduction to Blobstream rollups","Concepts","Blob share commitment"]},"156":{"title":"Blob share commitment: Cons","titles":["Introduction to Blobstream rollups","Concepts","Blob share commitment"]},"157":{"title":"Sequence of spans","titles":["Introduction to Blobstream rollups","Concepts"]},"158":{"title":"Sequence of spans: Proof details","titles":["Introduction to Blobstream rollups","Concepts","Sequence of spans"]},"159":{"title":"Sequence of spans: Proving unavailable data","titles":["Introduction to Blobstream rollups","Concepts","Sequence of spans"]},"160":{"title":"Sequence of spans: Proving inclusion of some data","titles":["Introduction to Blobstream rollups","Concepts","Sequence of spans"]},"161":{"title":"Sequence of spans: Pros","titles":["Introduction to Blobstream rollups","Concepts","Sequence of spans"]},"162":{"title":"Sequence of spans: Cons","titles":["Introduction to Blobstream rollups","Concepts","Sequence of spans"]},"163":{"title":"Optimistic rollups","titles":["Introduction to Blobstream rollups"]},"164":{"title":"Optimistic rollups that use a sequence of spans","titles":["Introduction to Blobstream rollups","Optimistic rollups"]},"165":{"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"]},"166":{"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"]},"167":{"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"]},"168":{"title":"Optimistic rollups that use blob share commitments","titles":["Introduction to Blobstream rollups","Optimistic rollups"]},"169":{"title":"Optimistic rollups that use blob share commitments: Pros","titles":["Introduction to Blobstream rollups","Optimistic rollups","Optimistic rollups that use blob share commitments"]},"170":{"title":"Optimistic rollups that use blob share commitments: Cons","titles":["Introduction to Blobstream rollups","Optimistic rollups","Optimistic rollups that use blob share commitments"]},"171":{"title":"Zk-rollups","titles":["Introduction to Blobstream rollups"]},"172":{"title":"Zk-rollups that use sequence of spans","titles":["Introduction to Blobstream rollups","Zk-rollups"]},"173":{"title":"Zk-rollups that use sequence of spans: Pros","titles":["Introduction to Blobstream rollups","Zk-rollups","Zk-rollups that use sequence of spans"]},"174":{"title":"Zk-rollups that use sequence of spans: Cons","titles":["Introduction to Blobstream rollups","Zk-rollups","Zk-rollups that use sequence of spans"]},"175":{"title":"Zk-rollups that use blob share commitments","titles":["Introduction to Blobstream rollups","Zk-rollups"]},"176":{"title":"Protobuf deserialization inside a zk-circuit","titles":["Introduction to Blobstream rollups","Zk-rollups","Zk-rollups that use blob share commitments"]},"177":{"title":"Zk-rollups that use blob share commitments: Pros","titles":["Introduction to Blobstream rollups","Zk-rollups","Zk-rollups that use blob share commitments"]},"178":{"title":"Zk-rollups that use blob share commitments: Cons","titles":["Introduction to Blobstream rollups","Zk-rollups","Zk-rollups that use blob share commitments"]},"179":{"title":"Heavy merkle proofs usage","titles":["Introduction to Blobstream rollups","Zk-rollups"]},"180":{"title":"heavy merkle proofs usage: Pros","titles":["Introduction to Blobstream rollups","Zk-rollups","Heavy merkle proofs usage"]},"181":{"title":"heavy merkle proofs usage: Cons","titles":["Introduction to Blobstream rollups","Zk-rollups","Heavy merkle proofs usage"]},"182":{"title":"Conclusion","titles":["Introduction to Blobstream rollups"]},"183":{"title":"FAQ","titles":["Introduction to Blobstream rollups"]},"184":{"title":"Should I use the Celestia transaction hash to reference the rollup data?","titles":["Introduction to Blobstream rollups","FAQ"]},"185":{"title":"New Blobstream X deployments","titles":[]},"186":{"title":"Deploying the contracts","titles":["New Blobstream X deployments"]},"187":{"title":"Deploy a new SuccinctGateway","titles":["New Blobstream X deployments","Deploying the contracts"]},"188":{"title":"Deploy the function verifiers","titles":["New Blobstream X deployments","Deploying the contracts"]},"189":{"title":"Register the function verifier in the deployed SuccinctGateway","titles":["New Blobstream X deployments","Deploying the contracts"]},"190":{"title":"Enable prover whitelisting","titles":["New Blobstream X deployments","Deploying the contracts","Register the function verifier in the deployed SuccinctGateway"]},"191":{"title":"Set Whitelist Status","titles":["New Blobstream X deployments","Deploying the contracts","Register the function verifier in the deployed SuccinctGateway"]},"192":{"title":"Add Custom Prover","titles":["New Blobstream X deployments","Deploying the contracts","Register the function verifier in the deployed SuccinctGateway"]},"193":{"title":"Deploy the BlobstreamX contract","titles":["New Blobstream X deployments","Deploying the contracts"]},"194":{"title":"Querying the trusted hash","titles":["New Blobstream X deployments","Deploying the contracts","Deploy the BlobstreamX contract"]},"195":{"title":"Deployment instructions","titles":["New Blobstream X deployments","Deploying the contracts","Deploy the BlobstreamX contract"]},"196":{"title":"Run a local prover","titles":["New Blobstream X deployments","Deploying the contracts"]},"197":{"title":"Run a proof replayer from an existing deployment","titles":["New Blobstream X deployments","Deploying the contracts"]},"198":{"title":"Optional: Regenerating the downloaded artifacts","titles":["New Blobstream X deployments"]},"199":{"title":"Regenerate the verifier-build","titles":["New Blobstream X deployments","Optional: Regenerating the downloaded artifacts"]},"200":{"title":"Build the circuits and function verifiers","titles":["New Blobstream X deployments","Optional: Regenerating the downloaded artifacts"]},"201":{"title":"Requesting data commitment ranges","titles":[]},"202":{"title":"Local proving","titles":["Requesting data commitment ranges"]},"203":{"title":"Request proofs from the Succinct platform","titles":["Requesting data commitment ranges"]},"204":{"title":"Request proofs onchain","titles":["Requesting data commitment ranges"]},"205":{"title":"Blobstream: Streaming modular DA to Ethereum","titles":[]},"206":{"title":"What is Blobstream?","titles":["Blobstream: Streaming modular DA to Ethereum"]},"207":{"title":"Implementations of Blobstream","titles":["Blobstream: Streaming modular DA to Ethereum","What is Blobstream?"]},"208":{"title":"Blobstream vs. data availability committees (DACs)","titles":["Blobstream: Streaming modular DA to Ethereum"]},"209":{"title":"Decentralization and security","titles":["Blobstream: Streaming modular DA to Ethereum","Blobstream vs. data availability committees (DACs)"]},"210":{"title":"Mechanism of verification","titles":["Blobstream: Streaming modular DA to Ethereum","Blobstream vs. data availability committees (DACs)"]},"211":{"title":"Flexibility and scalability","titles":["Blobstream: Streaming modular DA to Ethereum","Blobstream vs. data availability committees (DACs)"]},"212":{"title":"What is SP1 Blobstream?","titles":["Blobstream: Streaming modular DA to Ethereum"]},"213":{"title":"Integrate with SP1 Blobstream","titles":["Blobstream: Streaming modular DA to Ethereum"]},"214":{"title":"How to integrate with Blobstream","titles":["Blobstream: Streaming modular DA to Ethereum","Integrate with SP1 Blobstream"]},"215":{"title":"Blobstream rollups","titles":["Blobstream: Streaming modular DA to Ethereum","Integrate with SP1 Blobstream"]},"216":{"title":"Deployed contracts","titles":["Blobstream: Streaming modular DA to Ethereum","Integrate with SP1 Blobstream"]},"217":{"title":"Blobstream X: the previous zk implementation of Blobstream","titles":[]},"218":{"title":"What is Blobstream X?","titles":["Blobstream X: the previous zk implementation of Blobstream"]},"219":{"title":"How Blobstream X works","titles":["Blobstream X: the previous zk implementation of Blobstream"]},"220":{"title":"Deploy Blobstream X","titles":["Blobstream X: the previous zk implementation of Blobstream"]},"221":{"title":"Bubs testnet","titles":[]},"222":{"title":"Built with the OP Stack and Celestia","titles":["Bubs testnet"]},"223":{"title":"Building on Bubs","titles":["Bubs testnet"]},"224":{"title":"RPC URLs","titles":["Bubs testnet","Building on Bubs"]},"225":{"title":"HTTPS","titles":["Bubs testnet","Building on Bubs","RPC URLs"]},"226":{"title":"WSS","titles":["Bubs testnet","Building on Bubs","RPC URLs"]},"227":{"title":"Bridge","titles":["Bubs testnet","Building on Bubs"]},"228":{"title":"Faucet","titles":["Bubs testnet","Building on Bubs"]},"229":{"title":"Explorer","titles":["Bubs testnet","Building on Bubs"]},"230":{"title":"Status","titles":["Bubs testnet","Building on Bubs"]},"231":{"title":"Next steps","titles":["Bubs testnet"]},"232":{"title":"Build whatever","titles":[]},"233":{"title":"Quickstart - Building on Celestia","titles":["Build whatever"]},"234":{"title":"Choose a framework","titles":["Build whatever","Quickstart - Building on Celestia"]},"235":{"title":"Rollups as a Service","titles":["Build whatever","Quickstart - Building on Celestia"]},"236":{"title":"Smart contracts","titles":["Build whatever","Quickstart - Building on Celestia"]},"237":{"title":"What is a rollup?","titles":["Build whatever"]},"238":{"title":"What is a modular blockchain?","titles":["Build whatever"]},"239":{"title":"Benefits of modular blockchains","titles":["Build whatever"]},"240":{"title":"Ease of deploying a chain","titles":["Build whatever","Benefits of modular blockchains"]},"241":{"title":"Scaling","titles":["Build whatever","Benefits of modular blockchains"]},"242":{"title":"Customizability","titles":["Build whatever","Benefits of modular blockchains"]},"243":{"title":"Create a wallet with celestia-node","titles":[]},"244":{"title":"Using the cel-key utility","titles":["Create a wallet with celestia-node"]},"245":{"title":"Installation","titles":["Create a wallet with celestia-node","Using the cel-key utility"]},"246":{"title":"Steps for generating node keys","titles":["Create a wallet with celestia-node","Using the cel-key utility"]},"247":{"title":"Steps for exporting node keys","titles":["Create a wallet with celestia-node","Using the cel-key utility"]},"248":{"title":"Steps for importing node keys","titles":["Create a wallet with celestia-node","Using the cel-key utility"]},"249":{"title":"View all options for cel-key","titles":["Create a wallet with celestia-node","Using the cel-key utility"]},"250":{"title":"Docker and cel-key","titles":["Create a wallet with celestia-node"]},"251":{"title":"Prerequisites","titles":["Create a wallet with celestia-node","Docker and cel-key"]},"252":{"title":"Running your node","titles":["Create a wallet with celestia-node","Docker and cel-key"]},"253":{"title":"Mounting existing keys to container","titles":["Create a wallet with celestia-node","Docker and cel-key"]},"254":{"title":"Ethereum fallback","titles":[]},"255":{"title":"FeeGrant module for blobs submission","titles":[]},"256":{"title":"Overview","titles":["FeeGrant module for blobs submission"]},"257":{"title":"Pre-requisites","titles":["FeeGrant module for blobs submission"]},"258":{"title":"Introduction","titles":["FeeGrant module for blobs submission"]},"259":{"title":"Granting fee allowances using celestia-node","titles":["FeeGrant module for blobs submission"]},"260":{"title":"FeeGrant module implementation in celestia-node","titles":["FeeGrant module for blobs submission","Granting fee allowances using celestia-node"]},"261":{"title":"Grant permission for an allowance as a granter","titles":["FeeGrant module for blobs submission","Granting fee allowances using celestia-node"]},"262":{"title":"Using a FeeGrant allowance as a grantee in celestia-node","titles":["FeeGrant module for blobs submission"]},"263":{"title":"Checking account balances after submission","titles":["FeeGrant module for blobs submission"]},"264":{"title":"Optional: Revoke permission for a FeeGrant allowance as a granter","titles":["FeeGrant module for blobs submission"]},"265":{"title":"Optional: Submitting a blob from file input","titles":["FeeGrant module for blobs submission","Optional: Revoke permission for a FeeGrant allowance as a granter"]},"266":{"title":"Optional: Granting fee allowances using celestia-appd","titles":["FeeGrant module for blobs submission"]},"267":{"title":"Optional: Checking the granter\'s account","titles":["FeeGrant module for blobs submission","Optional: Granting fee allowances using celestia-appd"]},"268":{"title":"Golang client library tutorial","titles":[]},"269":{"title":"Project setup","titles":["Golang client library tutorial"]},"270":{"title":"Submitting and retrieving blobs","titles":["Golang client library tutorial"]},"271":{"title":"Subscribing to new blobs","titles":["Golang client library tutorial"]},"272":{"title":"Subscribing to new headers","titles":["Golang client library tutorial"]},"273":{"title":"Fetching an Extended Data Square (EDS)","titles":["Golang client library tutorial"]},"274":{"title":"API documentation","titles":["Golang client library tutorial"]},"275":{"title":"Integrate Celestia for service providers","titles":[]},"276":{"title":"Getting started","titles":["Integrate Celestia for service providers"]},"277":{"title":"Celestia service provider notes","titles":["Integrate Celestia for service providers"]},"278":{"title":"Custody and key management","titles":["Integrate Celestia for service providers","Celestia service provider notes"]},"279":{"title":"RPC and querying","titles":["Integrate Celestia for service providers","Celestia service provider notes"]},"280":{"title":"Compatibility","titles":["Integrate Celestia for service providers","Celestia service provider notes"]},"281":{"title":"Syncing","titles":["Integrate Celestia for service providers","Celestia service provider notes"]},"282":{"title":"Notable exceptions relative to other blockchains","titles":["Integrate Celestia for service providers","Celestia service provider notes"]},"283":{"title":"Introduction to OP Stack integration","titles":[]},"284":{"title":"About the integration","titles":["Introduction to OP Stack integration"]},"285":{"title":"GitHub repository","titles":["Introduction to OP Stack integration","About the integration"]},"286":{"title":"Next steps","titles":["Introduction to OP Stack integration"]},"287":{"title":"MultiAccounts feature for blobs submission","titles":[]},"288":{"title":"Overview","titles":["MultiAccounts feature for blobs submission"]},"289":{"title":"Running a node with a different default key name","titles":["MultiAccounts feature for blobs submission"]},"290":{"title":"Submitting blobs with a different signer/key name","titles":["MultiAccounts feature for blobs submission"]},"291":{"title":"Option 1: Submit passing key name","titles":["MultiAccounts feature for blobs submission","Submitting blobs with a different signer/key name"]},"292":{"title":"Option 2: Submit passing signer address","titles":["MultiAccounts feature for blobs submission","Submitting blobs with a different signer/key name"]},"293":{"title":"Key management","titles":["MultiAccounts feature for blobs submission"]},"294":{"title":"Creating a new key","titles":["MultiAccounts feature for blobs submission","Key management"]},"295":{"title":"Importing an existing key","titles":["MultiAccounts feature for blobs submission","Key management"]},"296":{"title":"Optional flags for write transactions","titles":["MultiAccounts feature for blobs submission"]},"297":{"title":"Node API","titles":[]},"298":{"title":"RPC API","titles":["Node API"]},"299":{"title":"Library","titles":["Node API","RPC API"]},"300":{"title":"RPC","titles":["Node API","RPC API"]},"301":{"title":"RPC API tutorial","titles":["Node API","RPC API"]},"302":{"title":"Gateway API","titles":["Node API"]},"303":{"title":"Gateway API tutorial","titles":["Node API","Gateway API"]},"304":{"title":"Celestia-node RPC CLI tutorial","titles":[]},"305":{"title":"Introduction","titles":["Celestia-node RPC CLI tutorial"]},"306":{"title":"Blobs","titles":["Celestia-node RPC CLI tutorial","Introduction"]},"307":{"title":"Namespaces","titles":["Celestia-node RPC CLI tutorial","Introduction"]},"308":{"title":"Hardware requirements","titles":["Celestia-node RPC CLI tutorial"]},"309":{"title":"Setting up dependencies","titles":["Celestia-node RPC CLI tutorial"]},"310":{"title":"Instantiate a Celestia light node","titles":["Celestia-node RPC CLI tutorial","Setting up dependencies"]},"311":{"title":"Connect to a core endpoint","titles":["Celestia-node RPC CLI tutorial","Setting up dependencies"]},"312":{"title":"Keys and wallets","titles":["Celestia-node RPC CLI tutorial","Setting up dependencies"]},"313":{"title":"RPC CLI guide","titles":["Celestia-node RPC CLI tutorial"]},"314":{"title":"Command formatting","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide"]},"315":{"title":"Basic flags","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide"]},"316":{"title":"Auth token 🔐","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Basic flags"]},"317":{"title":"Node store","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Basic flags"]},"318":{"title":"Auth token on custom or private network","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Basic flags","Node store"]},"319":{"title":"Submitting data","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide"]},"320":{"title":"Optional: Submit with curl","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Submitting data"]},"321":{"title":"Retrieving data","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide"]},"322":{"title":"Setting the gas price","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide"]},"323":{"title":"Examples","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide"]},"324":{"title":"Check your balance","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"325":{"title":"Check the balance of another address","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"326":{"title":"Get your node ID","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"327":{"title":"Get your account address","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"328":{"title":"Get block header by height","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"329":{"title":"Combined commands","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"330":{"title":"Get data availability sampler stats","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"331":{"title":"Transfer balance of utia to another account","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"332":{"title":"API version","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"333":{"title":"Help","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide","Examples"]},"334":{"title":"Advanced example","titles":["Celestia-node RPC CLI tutorial","RPC CLI guide"]},"335":{"title":"Additional resources","titles":["Celestia-node RPC CLI tutorial"]},"336":{"title":"Submitting a blob using curl","titles":["Celestia-node RPC CLI tutorial","Additional resources"]},"337":{"title":"Post an SVG as a PFB","titles":["Celestia-node RPC CLI tutorial","Additional resources"]},"338":{"title":"Troubleshooting","titles":["Celestia-node RPC CLI tutorial","Additional resources"]},"339":{"title":"Optimism devnet deep dive","titles":[]},"340":{"title":"Find a transaction","titles":["Optimism devnet deep dive"]},"341":{"title":"Read the transaction call data","titles":["Optimism devnet deep dive"]},"342":{"title":"Find the data on Celestia","titles":["Optimism devnet deep dive"]},"343":{"title":"Span batches","titles":["Optimism devnet deep dive"]},"344":{"title":"Run an OP Stack rollup with Celestia underneath","titles":[]},"345":{"title":"Dependency setup","titles":["Run an OP Stack rollup with Celestia underneath"]},"346":{"title":"Setting up your light node","titles":["Run an OP Stack rollup with Celestia underneath","Dependency setup"]},"347":{"title":"Deploying a devnet to Mocha","titles":["Run an OP Stack rollup with Celestia underneath"]},"348":{"title":"Deploying a testnet to an L1 (or L2) and Mocha","titles":["Run an OP Stack rollup with Celestia underneath"]},"349":{"title":"Congratulations","titles":["Run an OP Stack rollup with Celestia underneath"]},"350":{"title":"Prompt scavenger","titles":[]},"351":{"title":"Dependencies","titles":["Prompt scavenger"]},"352":{"title":"Install Celestia Node and run a light node","titles":["Prompt scavenger","Dependencies"]},"353":{"title":"OpenAI key","titles":["Prompt scavenger","Dependencies"]},"354":{"title":"Building the Prompt Scavenger","titles":["Prompt scavenger"]},"355":{"title":"Initialize your Go project","titles":["Prompt scavenger","Building the Prompt Scavenger"]},"356":{"title":"Build your import statements","titles":["Prompt scavenger","Building the Prompt Scavenger"]},"357":{"title":"Main function","titles":["Prompt scavenger","Building the Prompt Scavenger"]},"358":{"title":"Utility functions","titles":["Prompt scavenger","Building the Prompt Scavenger"]},"359":{"title":"Prompting ChatGPT","titles":["Prompt scavenger","Building the Prompt Scavenger"]},"360":{"title":"Wrapping things up","titles":["Prompt scavenger","Building the Prompt Scavenger"]},"361":{"title":"Next steps","titles":["Prompt scavenger"]},"362":{"title":"Rust client library tutorial","titles":[]},"363":{"title":"Project setup","titles":["Rust client library tutorial"]},"364":{"title":"Submitting and retrieving blobs","titles":["Rust client library tutorial"]},"365":{"title":"Subscribing to new headers","titles":["Rust client library tutorial"]},"366":{"title":"Fetching an Extended Data Square (EDS)","titles":["Rust client library tutorial"]},"367":{"title":"API documentation","titles":["Rust client library tutorial"]},"368":{"title":"New SP1 Blobstream deployments","titles":[]},"369":{"title":"Deploying the contracts","titles":["New SP1 Blobstream deployments"]},"370":{"title":"Submitting data blobs to Celestia","titles":[]},"371":{"title":"Maximum blob size","titles":["Submitting data blobs to Celestia"]},"372":{"title":"Fee market and mempool","titles":["Submitting data blobs to Celestia"]},"373":{"title":"Fees and gas limits","titles":["Submitting data blobs to Celestia","Fee market and mempool"]},"374":{"title":"Estimating PFB gas","titles":["Submitting data blobs to Celestia","Fee market and mempool","Fees and gas limits"]},"375":{"title":"Gas fee calculation","titles":["Submitting data blobs to Celestia","Fee market and mempool","Fees and gas limits"]},"376":{"title":"Estimating gas programmatically","titles":["Submitting data blobs to Celestia","Fee market and mempool","Fees and gas limits"]},"377":{"title":"Submitting multiple transactions in one block from the same account","titles":["Submitting data blobs to Celestia","Fee market and mempool"]},"378":{"title":"API","titles":["Submitting data blobs to Celestia"]},"379":{"title":"The celestia-app consensus node CLI","titles":["Submitting data blobs to Celestia","API"]},"380":{"title":"The celestia-node light node CLI","titles":["Submitting data blobs to Celestia","API"]},"381":{"title":"The celestia-node API golang client","titles":["Submitting data blobs to Celestia","API"]},"382":{"title":"GRPC to a consensus node via the user package","titles":["Submitting data blobs to Celestia","API"]},"383":{"title":"RPC to a celestia-node","titles":["Submitting data blobs to Celestia","API"]},"384":{"title":"Post a blob directly from Celenium","titles":["Submitting data blobs to Celestia","API"]},"385":{"title":"Transaction resubmission","titles":[]},"386":{"title":"Monitoring and resubmission","titles":["Transaction resubmission"]},"387":{"title":"Notes","titles":["Transaction resubmission"]},"388":{"title":"Wallet integrations with Celestia","titles":[]},"389":{"title":"Add Celestia network parameters to Keplr with React","titles":["Wallet integrations with Celestia"]},"390":{"title":"Adding a custom chain to Leap","titles":["Wallet integrations with Celestia"]},"391":{"title":"Adding a custom chain to Cosmostation","titles":["Wallet integrations with Celestia"]},"392":{"title":"Data availability FAQ","titles":[]},"393":{"title":"What is data availability?","titles":["Data availability FAQ"]},"394":{"title":"What is the data availability problem?","titles":["Data availability FAQ"]},"395":{"title":"How do nodes verify data availability in Celestia?","titles":["Data availability FAQ"]},"396":{"title":"What is data availability sampling?","titles":["Data availability FAQ"]},"397":{"title":"What are some of the security assumptions that Celestia makes for data availability sampling?","titles":["Data availability FAQ"]},"398":{"title":"Why is block reconstruction necessary for security?","titles":["Data availability FAQ"]},"399":{"title":"What is data storage?","titles":["Data availability FAQ"]},"400":{"title":"What is the problem around data storage?","titles":["Data availability FAQ"]},"401":{"title":"What is the difference between data availability and data storage?","titles":["Data availability FAQ"]},"402":{"title":"Where does blockchain state fit into this?","titles":["Data availability FAQ"]},"403":{"title":"Why doesn’t Celestia incentivize storage of historical data?","titles":["Data availability FAQ"]},"404":{"title":"Who may store historical data if there is no reward?","titles":["Data availability FAQ"]},"405":{"title":"What are some things blockchains can do to provide stronger assurances of data retrievability?","titles":["Data availability FAQ"]},"406":{"title":"Celestia\'s data availability layer","titles":[]},"407":{"title":"Data availability sampling (DAS)","titles":["Celestia\'s data availability layer"]},"408":{"title":"Scalability","titles":["Celestia\'s data availability layer","Data availability sampling (DAS)"]},"409":{"title":"Fraud proofs of incorrectly extended data","titles":["Celestia\'s data availability layer","Data availability sampling (DAS)"]},"410":{"title":"Namespaced Merkle trees (NMTs)","titles":["Celestia\'s data availability layer"]},"411":{"title":"Building a PoS blockchain for DA","titles":["Celestia\'s data availability layer"]},"412":{"title":"Providing data availability","titles":["Celestia\'s data availability layer","Building a PoS blockchain for DA"]},"413":{"title":"Monolithic vs. modular blockchains","titles":[]},"414":{"title":"Introduction","titles":[]},"415":{"title":"The lifecycle of a celestia-app transaction","titles":[]},"416":{"title":"Checking data availability","titles":["The lifecycle of a celestia-app transaction"]},"417":{"title":"How to stake TIA","titles":[]},"418":{"title":"Select your preferred wallet","titles":["How to stake TIA"]},"419":{"title":"Stake TIA with Keplr wallet","titles":["How to stake TIA"]},"420":{"title":"1️⃣ Open your Keplr browser extension","titles":["How to stake TIA","Stake TIA with Keplr wallet"]},"421":{"title":"2️⃣ Select Celestia network and search for a validator","titles":["How to stake TIA","Stake TIA with Keplr wallet"]},"422":{"title":"3️⃣ Stake your TIA tokens","titles":["How to stake TIA","Stake TIA with Keplr wallet"]},"423":{"title":"4️⃣ Confirm and manage your TIA","titles":["How to stake TIA","Stake TIA with Keplr wallet"]},"424":{"title":"Stake TIA with Leap wallet","titles":["How to stake TIA"]},"425":{"title":"1️⃣ Open your Leap browser extension","titles":["How to stake TIA","Stake TIA with Leap wallet"]},"426":{"title":"2️⃣ Select a validator and stake TIA","titles":["How to stake TIA","Stake TIA with Leap wallet"]},"427":{"title":"3️⃣ Confirm and manage your TIA","titles":["How to stake TIA","Stake TIA with Leap wallet"]},"428":{"title":"Stake TIA with Gem wallet","titles":["How to stake TIA"]},"429":{"title":"1️⃣ Open your Gem Wallet app","titles":["How to stake TIA","Stake TIA with Gem wallet"]},"430":{"title":"2️⃣ Choose the amount of Celestia and search for a validator.","titles":["How to stake TIA","Stake TIA with Gem wallet"]},"431":{"title":"3️⃣ Stake your TIA tokens","titles":["How to stake TIA","Stake TIA with Gem wallet"]},"432":{"title":"4️⃣ Manage your TIA","titles":["How to stake TIA","Stake TIA with Gem wallet"]},"433":{"title":"Paying for blobspace","titles":[]},"434":{"title":"PayForBlobs transactions","titles":["Paying for blobspace"]},"435":{"title":"Fee market overview","titles":["Paying for blobspace"]},"436":{"title":"Data retrievability and pruning","titles":[]},"437":{"title":"Data retrievability and pruning in celestia-node","titles":["Data retrievability and pruning"]},"438":{"title":"Suggested practices for rollups","titles":["Data retrievability and pruning"]},"439":{"title":"Staking, governance, & supply","titles":[]},"440":{"title":"Proof-of-stake on Celestia","titles":["Staking, governance, & supply"]},"441":{"title":"Inflation","titles":["Staking, governance, & supply"]},"442":{"title":"Decentralised governance","titles":["Staking, governance, & supply"]},"443":{"title":"Network parameters","titles":["Staking, governance, & supply","Decentralised governance"]},"444":{"title":"Community pool","titles":["Staking, governance, & supply","Decentralised governance"]},"445":{"title":"TIA allocation at genesis","titles":["Staking, governance, & supply"]},"446":{"title":"Unlocks","titles":["Staking, governance, & supply","TIA allocation at genesis"]},"447":{"title":"Staking on Celestia","titles":[]},"448":{"title":"Mainnet Beta","titles":["Staking on Celestia"]},"449":{"title":"Mocha testnet","titles":["Staking on Celestia"]},"450":{"title":"Overview of TIA","titles":[]},"451":{"title":"TIA at a glance","titles":["Overview of TIA"]},"452":{"title":"Role of TIA","titles":["Overview of TIA"]},"453":{"title":"Paying for blobspace","titles":["Overview of TIA","Role of TIA"]},"454":{"title":"Bootstrapping new rollups","titles":["Overview of TIA","Role of TIA"]},"455":{"title":"Proof-of-stake","titles":["Overview of TIA","Role of TIA"]},"456":{"title":"Decentralised governance","titles":["Overview of TIA","Role of TIA"]},"457":{"title":"Denominations","titles":["Overview of TIA","Role of TIA"]},"458":{"title":"TIA: display token","titles":["Overview of TIA","Role of TIA","Denominations"]},"459":{"title":"utia: staking denomination","titles":["Overview of TIA","Role of TIA","Denominations"]},"460":{"title":"microtia: staking denomination alias","titles":["Overview of TIA","Role of TIA","Denominations"]},"461":{"title":"Arabica devnet","titles":[]},"462":{"title":"Network stability and upgrades","titles":["Arabica devnet"]},"463":{"title":"Network details","titles":["Arabica devnet"]},"464":{"title":"Software version numbers","titles":["Arabica devnet","Network details"]},"465":{"title":"Integrations","titles":["Arabica devnet"]},"466":{"title":"Production RPC endpoints","titles":["Arabica devnet","Integrations"]},"467":{"title":"Community RPC endpoints","titles":["Arabica devnet","Integrations"]},"468":{"title":"Using consensus endpoints with DA nodes","titles":["Arabica devnet","Integrations"]},"469":{"title":"Data availability (DA) RPC endpoints for bridge node sync","titles":["Arabica devnet","Integrations","Using consensus endpoints with DA nodes"]},"470":{"title":"Data availability (DA) gRPC endpoints for state access","titles":["Arabica devnet","Integrations","Using consensus endpoints with DA nodes"]},"471":{"title":"Arabica devnet faucet","titles":["Arabica devnet"]},"472":{"title":"Discord","titles":["Arabica devnet","Arabica devnet faucet"]},"473":{"title":"Web","titles":["Arabica devnet","Arabica devnet faucet"]},"474":{"title":"Explorers","titles":["Arabica devnet"]},"475":{"title":"Network upgrades","titles":["Arabica devnet"]},"476":{"title":"Setting up a Celestia bridge node","titles":[]},"477":{"title":"Overview of bridge nodes","titles":["Setting up a Celestia bridge node"]},"478":{"title":"Hardware requirements","titles":["Setting up a Celestia bridge node"]},"479":{"title":"Setting up your bridge node","titles":["Setting up a Celestia bridge node"]},"480":{"title":"Setup the dependencies","titles":["Setting up a Celestia bridge node","Setting up your bridge node"]},"481":{"title":"Deploy the Celestia bridge node","titles":["Setting up a Celestia bridge node"]},"482":{"title":"Install Celestia Node","titles":["Setting up a Celestia bridge node","Deploy the Celestia bridge node"]},"483":{"title":"Initialize the bridge node","titles":["Setting up a Celestia bridge node","Deploy the Celestia bridge node"]},"484":{"title":"Run the bridge node","titles":["Setting up a Celestia bridge node","Deploy the Celestia bridge node"]},"485":{"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"]},"486":{"title":"Optional: Migrate node id to another server","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":"Jailing and 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 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":"Background","titles":["Troubleshooting","Changing the location of your node store"]},"555":{"title":"Demonstration","titles":["Troubleshooting","Changing the location of your node store"]},"556":{"title":"Examples","titles":["Troubleshooting","Changing the location of your node store"]},"557":{"title":"Mainnet Beta full and Mocha light","titles":["Troubleshooting","Changing the location of your node store","Examples"]},"558":{"title":"Mocha full and Arabica light","titles":["Troubleshooting","Changing the location of your node store","Examples"]},"559":{"title":"Using a custom rpc.config address","titles":["Troubleshooting","Changing the location of your node store","Examples"]},"560":{"title":"Resetting your config","titles":["Troubleshooting"]},"561":{"title":"Clearing the data store","titles":["Troubleshooting"]},"562":{"title":"FATAL headers given to the heightSub are in the wrong order","titles":["Troubleshooting"]},"563":{"title":"Error: "too many open files"","titles":["Troubleshooting"]},"564":{"title":"Syncing a light node from a trusted hash","titles":[]},"565":{"title":"For service operators","titles":["Syncing a light node from a trusted hash"]},"566":{"title":"Install celestia-node","titles":[]},"567":{"title":"Installing from source","titles":["Install celestia-node"]},"568":{"title":"Installing a pre-built binary","titles":["Install celestia-node"]},"569":{"title":"Next steps","titles":["Install celestia-node"]},"570":{"title":"Upgrading your binary","titles":["Install celestia-node"]},"571":{"title":"config.toml guide","titles":[]},"572":{"title":"Pre-requisites","titles":["config.toml guide"]},"573":{"title":"Understanding config.toml","titles":["config.toml guide"]},"574":{"title":"Core","titles":["config.toml guide","Understanding config.toml"]},"575":{"title":"P2P","titles":["config.toml guide","Understanding config.toml"]},"576":{"title":"Bootstrap","titles":["config.toml guide","Understanding config.toml","P2P"]},"577":{"title":"Mutual peers","titles":["config.toml guide","Understanding config.toml","P2P"]},"578":{"title":"Services","titles":["config.toml guide","Understanding config.toml"]},"579":{"title":"TrustedHash and TrustedPeer","titles":["config.toml guide","Understanding config.toml","Services"]},"580":{"title":"Consensus node","titles":[]},"581":{"title":"Minimum hardware requirements","titles":["Consensus node"]},"582":{"title":"Set up a consensus node","titles":["Consensus node"]},"583":{"title":"Set up the dependencies","titles":["Consensus node","Set up a consensus node"]},"584":{"title":"Install celestia-app","titles":["Consensus node","Set up a consensus node"]},"585":{"title":"Set up the P2P networks","titles":["Consensus node","Set up a consensus node"]},"586":{"title":"Storage and pruning configurations","titles":["Consensus node"]},"587":{"title":"Optional: Connect a consensus node to a bridge node","titles":["Consensus node","Storage and pruning configurations"]},"588":{"title":"Enable transaction indexing","titles":["Consensus node","Storage and pruning configurations","Optional: Connect a consensus node to a bridge node"]},"589":{"title":"Retain all block data","titles":["Consensus node","Storage and pruning configurations","Optional: Connect a consensus node to a bridge node"]},"590":{"title":"Query transactions by hash","titles":["Consensus node","Storage and pruning configurations"]},"591":{"title":"Optional: Access historical state","titles":["Consensus node","Storage and pruning configurations"]},"592":{"title":"Save on storage requirements","titles":["Consensus node","Storage and pruning configurations"]},"593":{"title":"Sync types","titles":["Consensus node"]},"594":{"title":"Option 1: Block sync","titles":["Consensus node","Sync types"]},"595":{"title":"Option 2: State sync","titles":["Consensus node","Sync types"]},"596":{"title":"Option 3: Quick sync","titles":["Consensus node","Sync types"]},"597":{"title":"Start the consensus node","titles":["Consensus node"]},"598":{"title":"Extra resources for consensus nodes","titles":["Consensus node"]},"599":{"title":"Optional: Reset network","titles":["Consensus node","Extra resources for consensus nodes"]},"600":{"title":"Optional: Configure an RPC endpoint","titles":["Consensus node","Extra resources for consensus nodes"]},"601":{"title":"Expose RPC","titles":["Consensus node","Extra resources for consensus nodes","Optional: Configure an RPC endpoint"]},"602":{"title":"Note on external-address","titles":["Consensus node","Extra resources for consensus nodes","Optional: Configure an RPC endpoint"]},"603":{"title":"Restart the node","titles":["Consensus node","Extra resources for consensus nodes","Optional: Configure an RPC endpoint"]},"604":{"title":"Optional: Transaction indexer configuration options","titles":["Consensus node","Extra resources for consensus nodes"]},"605":{"title":"Optional: Discard ABCI responses configuration","titles":["Consensus node","Extra resources for consensus nodes"]},"606":{"title":"FAQ","titles":["Consensus node"]},"607":{"title":"+2/3 committed an invalid block: wrong Block.Header.Version","titles":["Consensus node","FAQ"]},"608":{"title":"Deciding which node to run","titles":[]},"609":{"title":"Light node","titles":["Deciding which node to run"]},"610":{"title":"Other DA nodes","titles":["Deciding which node to run","Light node"]},"611":{"title":"Consensus node","titles":["Deciding which node to run"]},"612":{"title":"🐳 Docker setup","titles":[]},"613":{"title":"Prerequisites","titles":["🐳 Docker setup"]},"614":{"title":"Quick start","titles":["🐳 Docker setup"]},"615":{"title":"Light node setup with persistent storage","titles":["🐳 Docker setup"]},"616":{"title":"Initialize the node store and key","titles":["🐳 Docker setup","Light node setup with persistent storage"]},"617":{"title":"Start the node","titles":["🐳 Docker setup","Light node setup with persistent storage"]},"618":{"title":"Video walkthrough","titles":["🐳 Docker setup"]},"619":{"title":"2.5 minute version","titles":["🐳 Docker setup","Video walkthrough"]},"620":{"title":"Troubleshooting","titles":["🐳 Docker setup"]},"621":{"title":"Development environment","titles":[]},"622":{"title":"Install dependencies","titles":["Development environment"]},"623":{"title":"Install Golang","titles":["Development environment"]},"624":{"title":"Setting up a Celestia full storage Node","titles":[]},"625":{"title":"Overview of full storage nodes","titles":["Setting up a Celestia full storage Node"]},"626":{"title":"Hardware requirements","titles":["Setting up a Celestia full storage Node"]},"627":{"title":"Setting up your full storage node","titles":["Setting up a Celestia full storage Node"]},"628":{"title":"Setup the dependencies","titles":["Setting up a Celestia full storage Node","Setting up your full storage node"]},"629":{"title":"Install celestia-node","titles":["Setting up a Celestia full storage Node"]},"630":{"title":"Run the full storage node","titles":["Setting up a Celestia full storage Node","Install celestia-node"]},"631":{"title":"Initialize the full storage node","titles":["Setting up a Celestia full storage Node","Install celestia-node","Run the full storage node"]},"632":{"title":"Start the full storage node","titles":["Setting up a Celestia full storage Node","Install celestia-node","Run the full storage node"]},"633":{"title":"Optional: run the full storage node with a custom key","titles":["Setting up a Celestia full storage Node","Install celestia-node"]},"634":{"title":"Optional: Migrate node id to another server","titles":["Setting up a Celestia full storage Node","Install celestia-node","Optional: run the full storage node with a custom key"]},"635":{"title":"Optional: start the full storage node with SystemD","titles":["Setting up a Celestia full storage Node","Install celestia-node"]},"636":{"title":"Stop the full storage node","titles":["Setting up a Celestia full storage Node","Install celestia-node"]},"637":{"title":"IBC relaying guide","titles":[]},"638":{"title":"Hermes","titles":["IBC relaying guide"]},"639":{"title":"Configuration","titles":["IBC relaying guide","Hermes"]},"640":{"title":"Add relayer wallets","titles":["IBC relaying guide","Hermes"]},"641":{"title":"Verify configuration files","titles":["IBC relaying guide","Hermes"]},"642":{"title":"Create a connection between 2 chains","titles":["IBC relaying guide"]},"643":{"title":"Create clients","titles":["IBC relaying guide"]},"644":{"title":"Open connection over new clients","titles":["IBC relaying guide","Create clients"]},"645":{"title":"Configure channels in Hermes","titles":["IBC relaying guide","Create clients"]},"646":{"title":"Start the relayer","titles":["IBC relaying guide"]},"647":{"title":"Transfer","titles":["IBC relaying guide"]},"648":{"title":"Token filter","titles":["IBC relaying guide","Transfer"]},"649":{"title":"Celestia App network instantiation guide","titles":[]},"650":{"title":"Hardware requirements","titles":["Celestia App network instantiation guide"]},"651":{"title":"Setup dependencies","titles":["Celestia App network instantiation guide"]},"652":{"title":"celestia-app installation","titles":["Celestia App network instantiation guide"]},"653":{"title":"Spin up a Celestia testnet","titles":["Celestia App network instantiation guide"]},"654":{"title":"Optional: Reset working directory","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"655":{"title":"Initialize a working directory","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"656":{"title":"Create a new key","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"657":{"title":"Add genesis account KeyName","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"658":{"title":"Optional: Adding other validators","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"659":{"title":"Create the genesis transaction for new chain","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"660":{"title":"Creating the genesis JSON file","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"661":{"title":"Modify your config file","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"662":{"title":"Add your node as a persistent peer","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"663":{"title":"Instantiate the network","titles":["Celestia App network instantiation guide","Spin up a Celestia testnet"]},"664":{"title":"Setting up a Celestia light node","titles":[]},"665":{"title":"Overview of light nodes","titles":["Setting up a Celestia light node"]},"666":{"title":"Hardware requirements","titles":["Setting up a Celestia light node"]},"667":{"title":"Quickstart: Run a light node in your browser","titles":["Setting up a Celestia light node"]},"668":{"title":"Setting up your light node","titles":["Setting up a Celestia light node"]},"669":{"title":"Install celestia-node","titles":["Setting up a Celestia light node","Setting up your light node"]},"670":{"title":"Initialize the light node","titles":["Setting up a Celestia light node"]},"671":{"title":"Start the light node","titles":["Setting up a Celestia light node"]},"672":{"title":"Keys and wallets","titles":["Setting up a Celestia light node","Start the light node"]},"673":{"title":"Testnet tokens","titles":["Setting up a Celestia light node","Start the light node","Keys and wallets"]},"674":{"title":"Optional: run the light node with a custom key","titles":["Setting up a Celestia light node","Start the light node"]},"675":{"title":"Optional: Migrate node id to another server","titles":["Setting up a Celestia light node","Start the light node","Optional: run the light node with a custom key"]},"676":{"title":"Optional: start light node with SystemD","titles":["Setting up a Celestia light node","Start the light node"]},"677":{"title":"Data availability sampling","titles":["Setting up a Celestia light node"]},"678":{"title":"Mainnet Beta","titles":[]},"679":{"title":"Network stability and upgrades","titles":["Mainnet Beta"]},"680":{"title":"Network details","titles":["Mainnet Beta"]},"681":{"title":"Software version numbers","titles":["Mainnet Beta"]},"682":{"title":"Network parameters","titles":["Mainnet Beta"]},"683":{"title":"Maximum bytes","titles":["Mainnet Beta","Network parameters"]},"684":{"title":"Integrations","titles":["Mainnet Beta"]},"685":{"title":"Production RPC endpoints","titles":["Mainnet Beta","Integrations"]},"686":{"title":"Consensus nodes","titles":["Mainnet Beta","Integrations"]},"687":{"title":"Community consensus RPC endpoints","titles":["Mainnet Beta","Integrations","Consensus nodes"]},"688":{"title":"Community API endpoints","titles":["Mainnet Beta","Integrations","Consensus nodes"]},"689":{"title":"Community gRPC endpoints","titles":["Mainnet Beta","Integrations","Consensus nodes"]},"690":{"title":"Community WebSocket endpoints","titles":["Mainnet Beta","Integrations","Consensus nodes"]},"691":{"title":"Data availability nodes","titles":["Mainnet Beta","Integrations"]},"692":{"title":"Community Data availability (DA) RPC endpoints for bridge node sync","titles":["Mainnet Beta","Integrations","Data availability nodes"]},"693":{"title":"Community Data availability (DA) gRPC endpoints for state access","titles":["Mainnet Beta","Integrations","Data availability nodes"]},"694":{"title":"Archival DA RPC endpoints","titles":["Mainnet Beta","Integrations","Data availability nodes"]},"695":{"title":"Grove archival endpoints","titles":["Mainnet Beta","Integrations","Data availability nodes","Archival DA RPC endpoints"]},"696":{"title":"Explorers","titles":["Mainnet Beta"]},"697":{"title":"Analytics","titles":["Mainnet Beta"]},"698":{"title":"Network upgrades","titles":["Mainnet Beta"]},"699":{"title":"Mocha testnet","titles":[]},"700":{"title":"Network details","titles":["Mocha testnet"]},"701":{"title":"Software version numbers","titles":["Mocha testnet"]},"702":{"title":"RPC for DA bridge, full, and light nodes","titles":["Mocha testnet"]},"703":{"title":"Production RPC endpoints","titles":["Mocha testnet","RPC for DA bridge, full, and light nodes"]},"704":{"title":"Community Data availability (DA) RPC endpoints for bridge node sync","titles":["Mocha testnet","RPC for DA bridge, full, and light nodes"]},"705":{"title":"Community Data availability (DA) gRPC endpoints for state access","titles":["Mocha testnet","RPC for DA bridge, full, and light nodes"]},"706":{"title":"Community RPC endpoints","titles":["Mocha testnet"]},"707":{"title":"Community API endpoints","titles":["Mocha testnet"]},"708":{"title":"Community gRPC endpoints","titles":["Mocha testnet"]},"709":{"title":"Community bridge and full node endpoints","titles":["Mocha testnet"]},"710":{"title":"Mocha testnet faucet","titles":["Mocha testnet"]},"711":{"title":"Analytics","titles":["Mocha testnet"]},"712":{"title":"Explorers","titles":["Mocha testnet"]},"713":{"title":"Network upgrades","titles":["Mocha testnet"]},"714":{"title":"Celestia network upgrade process","titles":[]},"715":{"title":"General process","titles":["Celestia network upgrade process"]},"716":{"title":"Lemongrass network upgrade","titles":["Celestia network upgrade process","General process"]},"717":{"title":"Overview to running nodes on Celestia","titles":[]},"718":{"title":"Recommended Celestia node requirements","titles":["Overview to running nodes on Celestia"]},"719":{"title":"Data availability nodes","titles":["Overview to running nodes on Celestia"]},"720":{"title":"Consensus nodes","titles":["Overview to running nodes on Celestia"]},"721":{"title":"Participate in the Celestia networks","titles":[]},"722":{"title":"Mainnet Beta","titles":["Participate in the Celestia networks"]},"723":{"title":"Compatible software versions for Mainnet Beta","titles":["Participate in the Celestia networks","Mainnet Beta"]},"724":{"title":"Testnets","titles":["Participate in the Celestia networks"]},"725":{"title":"Arabica Devnet","titles":["Participate in the Celestia networks","Testnets"]},"726":{"title":"Compatible software versions for Arabica devnet","titles":["Participate in the Celestia networks","Testnets","Arabica Devnet"]},"727":{"title":"Mocha testnet","titles":["Participate in the Celestia networks","Testnets"]},"728":{"title":"Compatible software versions for Mocha testnet","titles":["Participate in the Celestia networks","Testnets"]},"729":{"title":"Network upgrades","titles":["Participate in the Celestia networks"]},"730":{"title":"Quick start guide","titles":[]},"731":{"title":"Celestia Node","titles":["Quick start guide"]},"732":{"title":"Celestia App","titles":["Quick start guide"]},"733":{"title":"Getting started","titles":["Quick start guide"]},"734":{"title":"Setting up your node as a background process with SystemD","titles":[]},"735":{"title":"Consensus nodes","titles":["Setting up your node as a background process with SystemD"]},"736":{"title":"Start the celestia-app with SystemD","titles":["Setting up your node as a background process with SystemD","Consensus nodes"]},"737":{"title":"Data availability nodes","titles":["Setting up your node as a background process with SystemD"]},"738":{"title":"Celestia full storage node","titles":["Setting up your node as a background process with SystemD","Data availability nodes"]},"739":{"title":"Celestia bridge node","titles":["Setting up your node as a background process with SystemD","Data availability nodes"]},"740":{"title":"Celestia light node","titles":["Setting up your node as a background process with SystemD","Data availability nodes"]},"741":{"title":"Setting up a Celestia validator node","titles":[]},"742":{"title":"Hardware requirements","titles":["Setting up a Celestia validator node"]},"743":{"title":"Setting up a validator node","titles":["Setting up a Celestia validator node"]},"744":{"title":"Wallet","titles":["Setting up a Celestia validator node","Setting up a validator node"]},"745":{"title":"Delegate stake to a validator","titles":["Setting up a Celestia validator node","Setting up a validator node"]},"746":{"title":"Optional: Deploy the celestia-node","titles":["Setting up a Celestia validator node"]},"747":{"title":"Install celestia-node","titles":["Setting up a Celestia validator node","Optional: Deploy the celestia-node"]},"748":{"title":"Initialize the bridge node","titles":["Setting up a Celestia validator node","Optional: Deploy the celestia-node"]},"749":{"title":"Run the bridge node","titles":["Setting up a Celestia validator node","Optional: Deploy the celestia-node"]},"750":{"title":"Optional: start the bridge node with SystemD","titles":["Setting up a Celestia validator node","Optional: Deploy the celestia-node","Run the bridge node"]},"751":{"title":"Run the validator node","titles":["Setting up a Celestia validator node"]},"752":{"title":"Submit your validator information","titles":["Setting up a Celestia validator node"]},"753":{"title":"Optional: Transaction indexer configuration options","titles":["Setting up a Celestia validator node"]},"754":{"title":"Additional resources","titles":["Setting up a Celestia validator node"]},"755":{"title":"FAQ","titles":["Setting up a Celestia validator node"]},"756":{"title":"+2/3 committed an invalid block: wrong Block.Header.Version","titles":["Setting up a Celestia validator node","FAQ"]}},"dirtCount":0,"index":[["∑i=1nssn",{"2":{"375":1}}],["×ss×gcpbb",{"2":{"375":1}}],["|",{"2":{"319":4,"321":2,"329":2,"331":2,"334":2,"341":4,"343":1,"492":4,"585":14,"596":12,"736":4}}],["||",{"2":{"244":2}}],[">",{"2":{"234":1,"235":3}}],["~2",{"2":{"683":1}}],["~20",{"2":{"199":1}}],["~5",{"2":{"593":1}}],["~",{"2":{"253":2,"352":2,"485":1,"486":1,"562":4,"596":18,"601":2,"633":1,"634":1,"674":1,"675":1}}],["~1",{"2":{"593":1}}],["~10",{"2":{"200":1}}],["~1hr",{"2":{"200":1}}],["~30",{"2":{"202":1}}],["~3",{"2":{"200":1,"593":1}}],["$ip",{"2":{"662":2}}],["$input",{"2":{"341":2}}],["$network",{"2":{"614":3,"616":3}}],["$networkdocker",{"2":{"614":3,"616":3}}],["$node",{"2":{"317":3,"614":6,"616":2,"662":2,"739":2}}],["$peer",{"2":{"662":1}}],["$peerpeer=",{"2":{"662":1}}],["$persistent",{"2":{"585":12}}],["$port",{"2":{"662":2}}],["$path",{"2":{"623":1}}],["$private",{"2":{"195":2}}],["$from",{"2":{"522":6,"524":2,"525":2,"529":2}}],["$tia",{"2":{"657":2}}],["$to",{"2":{"522":6,"523":2,"525":2,"529":2}}],["$tx",{"2":{"341":2}}],["$broadcast",{"2":{"507":2}}],["$multisig",{"2":{"507":10}}],["$shell",{"2":{"623":1}}],["$seeds",{"2":{"585":8}}],["$staking",{"2":{"500":2,"659":3}}],["$signer",{"2":{"292":2}}],["$keyring",{"2":{"507":4}}],["$key",{"2":{"500":6,"656":3,"657":3,"659":3}}],["$celes",{"2":{"500":2}}],["$celestia",{"2":{"336":2,"342":2,"519":2,"520":2,"522":6,"523":2,"524":2,"525":4}}],["$chain",{"2":{"500":4,"507":10,"655":4,"659":2}}],["$validator",{"2":{"500":2,"507":6,"655":3,"745":2}}],["$external",{"2":{"602":2}}],["$encoded",{"2":{"342":2}}],["$etherscan",{"2":{"195":2}}],["$address",{"2":{"331":6}}],["$auth",{"2":{"314":2}}],["$request",{"2":{"312":1,"472":1,"673":1,"710":1}}],["$rpc",{"2":{"195":2,"261":2,"262":2,"266":2,"529":2,"614":6,"617":6}}],["$granter",{"2":{"262":2,"266":2,"267":2}}],["$grantee",{"2":{"261":2,"264":2,"266":2}}],["$",{"2":{"253":2,"329":2,"334":2,"540":2,"542":12,"568":2,"585":2,"596":6,"622":2}}],["$home",{"2":{"89":2,"188":2,"193":2,"196":2,"199":2,"200":2,"318":4,"346":2,"347":2,"503":3,"514":3,"515":1,"516":2,"519":2,"539":3,"540":1,"567":2,"568":1,"573":3,"585":11,"595":1,"596":6,"599":2,"602":2,"615":4,"616":8,"617":8,"623":20,"639":2,"654":2,"658":1,"659":1,"660":4,"661":1}}],["^external",{"2":{"602":2}}],["^persistent",{"2":{"585":6}}],["^seeds",{"2":{"585":4}}],["^",{"2":{"188":2,"199":4,"200":4}}],["^0",{"2":{"103":2,"148":2}}],["\\tlog",{"2":{"358":6,"360":4}}],["\\tlatestblocknumber",{"2":{"129":2,"146":2}}],["\\tpromptanswer",{"2":{"360":2}}],["\\tproof",{"2":{"129":2}}],["\\tpayload",{"2":{"358":2}}],["\\theader",{"2":{"273":2}}],["\\theaderchan",{"2":{"272":2}}],["\\thelloworldblob",{"2":{"270":2,"381":2}}],["\\theight",{"2":{"129":2,"145":2,"148":4,"270":2,"358":2,"381":2}}],["\\tns",{"2":{"358":2}}],["\\tnodeip",{"2":{"357":2,"358":2,"360":2}}],["\\tnodeclient",{"2":{"356":2}}],["\\tnonce",{"2":{"129":2,"145":2,"148":4}}],["\\tnamespaceid",{"2":{"358":2,"360":2}}],["\\tnamespacebytes",{"2":{"358":2}}],["\\tnamespace",{"2":{"270":2,"271":2,"272":2,"381":2}}],["\\tretrievedblobs",{"2":{"270":2,"381":2}}],["\\treturn",{"2":{"129":4,"141":8,"142":2,"143":2,"144":2,"145":2,"148":18,"270":2,"273":2,"358":4,"359":4,"381":2}}],["\\trowproofs",{"2":{"144":2,"148":2}}],["\\trowroots",{"2":{"143":2,"148":2}}],["\\tmaxns",{"2":{"141":2,"148":2}}],["\\tmaxsquaresize",{"2":{"135":2}}],["\\tminns",{"2":{"141":2,"148":2}}],["\\tsimplerollup",{"2":{"148":2}}],["\\tsimplerollupwrapper",{"2":{"148":2}}],["\\tsidenodes",{"2":{"129":2,"145":2,"148":2}}],["\\tshareproofs",{"2":{"141":2,"148":2}}],["\\tsharesproof",{"2":{"138":1,"148":4}}],["\\tsubtreerootthreshold",{"2":{"135":2}}],["\\tversion",{"2":{"135":2,"141":4,"148":6}}],["\\tvalid",{"2":{"129":2}}],["\\tvar",{"2":{"129":2,"141":6,"142":2,"146":2,"148":10}}],["\\twrappedproof",{"2":{"129":2}}],["\\twrapper",{"2":{"129":2,"146":2}}],["\\tcreatedblob",{"2":{"358":4,"360":2}}],["\\tclient",{"2":{"270":4,"271":2,"272":2,"273":2,"357":2,"358":4,"360":2,"381":4}}],["\\tcopy",{"2":{"141":6,"142":2,"148":2}}],["\\tcommitted",{"2":{"129":2}}],["\\tctx",{"2":{"128":2,"129":2,"148":4,"357":2,"358":4,"360":2}}],["\\tfetchedblob",{"2":{"358":2,"360":2}}],["\\tfor",{"2":{"129":4,"141":2,"143":2,"144":2,"145":2,"146":2,"148":16,"271":2,"272":2}}],["\\tfmt",{"2":{"128":2,"270":4,"381":4}}],["\\tblobchan",{"2":{"271":2}}],["\\tblobsharerange",{"2":{"135":2}}],["\\tblobstreamxwrapper",{"2":{"129":4,"148":2}}],["\\tblockdataroot",{"2":{"145":2,"148":2}}],["\\tblockres",{"2":{"129":2,"135":2}}],["\\ttmproto",{"2":{"148":2}}],["\\ttuple",{"2":{"129":2}}],["\\ttxhash",{"2":{"135":2}}],["\\ttx",{"2":{"129":2,"135":2,"148":2}}],["\\ttrpc",{"2":{"128":2,"129":2}}],["\\teventsiterator",{"2":{"129":2,"146":2}}],["\\tethclient",{"2":{"129":2,"146":2,"148":2}}],["\\tethcmn",{"2":{"129":2,"148":2}}],["\\terr",{"2":{"121":2,"128":2,"129":6,"146":2,"148":4}}],["\\tdatarootinclusionproof",{"2":{"145":2,"148":4}}],["\\tdataroot",{"2":{"129":2,"148":2}}],["\\tdcproof",{"2":{"128":2,"129":2}}],["\\tdefer",{"2":{"121":2,"129":2,"146":2,"148":2,"357":4,"358":4,"360":4}}],["\\t",{"2":{"121":6,"128":14,"129":90,"133":2,"135":22,"138":2,"141":8,"142":2,"143":2,"144":2,"145":4,"146":27,"147":24,"148":98,"270":28,"271":12,"272":12,"273":8,"356":28,"357":24,"358":54,"360":26,"381":28}}],["\\t\\tlog",{"2":{"357":4,"358":10,"360":12}}],["\\t\\tuint64",{"2":{"148":2}}],["\\t\\tclient",{"2":{"148":2}}],["\\t\\tctx",{"2":{"148":2,"359":2}}],["\\t\\tcase",{"2":{"147":4,"271":4,"272":4}}],["\\t\\tcopy",{"2":{"145":2}}],["\\t\\tproof",{"2":{"145":2,"148":2}}],["\\t\\tvar",{"2":{"145":2,"148":2}}],["\\t\\tversion",{"2":{"141":4,"142":2,"148":6}}],["\\t\\trowproofs",{"2":{"144":2,"148":2}}],["\\t\\trowroots",{"2":{"143":2,"148":2}}],["\\t\\treturn",{"2":{"121":2,"129":30,"135":8,"146":14,"148":8,"270":10,"271":6,"272":6,"273":4,"358":6,"381":10}}],["\\t\\tdcproof",{"2":{"148":2}}],["\\t\\tdigest",{"2":{"141":2,"148":4}}],["\\t\\tdataroot",{"2":{"129":2}}],["\\t\\tmax",{"2":{"141":2,"148":2}}],["\\t\\tmaxsquaresize",{"2":{"135":2}}],["\\t\\tmin",{"2":{"141":2,"148":2}}],["\\t\\tfor",{"2":{"141":2,"144":2,"148":6}}],["\\t\\tfmt",{"2":{"128":6,"129":6,"148":2}}],["\\t\\tselect",{"2":{"271":2,"272":2}}],["\\t\\tsharesproof",{"2":{"148":2}}],["\\t\\tshareproofs",{"2":{"141":2,"148":2}}],["\\t\\tsimplerollupwrapper",{"2":{"148":2}}],["\\t\\tsidenodes",{"2":{"129":4,"141":2,"144":2,"145":2,"148":6}}],["\\t\\tsubtreerootthreshold",{"2":{"135":2}}],["\\t\\tid",{"2":{"141":4,"142":2,"148":12}}],["\\t\\tint",{"2":{"135":2}}],["\\t\\tif",{"2":{"121":2,"129":2,"146":2}}],["\\t\\tblockdataroot",{"2":{"148":2}}],["\\t\\tblockres",{"2":{"135":2,"148":2}}],["\\t\\tbig",{"2":{"129":2}}],["\\t\\twrappedproof",{"2":{"129":2}}],["\\t\\ttuplerootnonce",{"2":{"145":2,"148":2}}],["\\t\\ttuple",{"2":{"129":2,"145":2,"148":2}}],["\\t\\tnumleaves",{"2":{"129":2}}],["\\t\\tnil",{"2":{"129":6,"146":6}}],["\\t\\tkey",{"2":{"129":2}}],["\\t\\theight",{"2":{"129":2}}],["\\t\\tevent",{"2":{"148":2}}],["\\t\\te",{"2":{"129":2,"146":2}}],["\\t\\terr",{"2":{"121":2}}],["\\t\\tos",{"2":{"128":6,"129":2,"148":2}}],["\\t\\t\\tif",{"2":{"272":2}}],["\\t\\t\\tfmt",{"2":{"271":2,"272":2}}],["\\t\\t\\tfor",{"2":{"148":2}}],["\\t\\t\\tattestationproof",{"2":{"148":2}}],["\\t\\t\\trowproofs",{"2":{"148":2}}],["\\t\\t\\trowroots",{"2":{"148":2}}],["\\t\\t\\treturn",{"2":{"147":2,"271":2,"272":2}}],["\\t\\t\\tnamespace",{"2":{"148":2}}],["\\t\\t\\tnumleaves",{"2":{"144":2,"145":2,"148":4}}],["\\t\\t\\tdata",{"2":{"148":2}}],["\\t\\t\\tdataroot",{"2":{"145":2,"148":2}}],["\\t\\t\\theight",{"2":{"145":2,"148":2}}],["\\t\\t\\tkey",{"2":{"144":2,"145":2,"148":4}}],["\\t\\t\\tcopy",{"2":{"144":2}}],["\\t\\t\\tcontext",{"2":{"129":2,"146":2,"147":2,"148":2}}],["\\t\\t\\tvar",{"2":{"144":2,"148":2}}],["\\t\\t\\tblobs",{"2":{"272":2}}],["\\t\\t\\tbzsidenode",{"2":{"148":2}}],["\\t\\t\\tbeginkey",{"2":{"141":2,"148":2}}],["\\t\\t\\tbreak",{"2":{"129":2,"146":2}}],["\\t\\t\\tshareproofs",{"2":{"148":2}}],["\\t\\t\\tsidenodes",{"2":{"141":4,"144":4,"145":2,"148":10}}],["\\t\\t\\tstart",{"2":{"129":2,"146":2}}],["\\t\\t\\t\\tfmt",{"2":{"272":2}}],["\\t\\t\\t\\tbzsidenode",{"2":{"148":2}}],["\\t\\t\\t\\tdatacommitment",{"2":{"129":2,"146":2}}],["\\t\\t\\t\\tendblock",{"2":{"129":2,"146":2}}],["\\t\\t\\t\\tstartblock",{"2":{"129":2,"146":2}}],["\\t\\t\\t\\tproofnonce",{"2":{"129":2,"146":2}}],["\\t\\t\\tevent",{"2":{"129":2,"146":2}}],["\\t\\t\\tendkey",{"2":{"141":2,"148":2}}],["\\t\\t\\tend",{"2":{"129":2,"146":2}}],["\\t\\t\\t",{"2":{"121":2,"129":2,"146":2,"147":2,"148":2,"272":4}}],["\\t\\t",{"2":{"121":4,"129":8,"133":4,"135":2,"138":2,"141":4,"144":4,"145":4,"146":6,"147":4,"148":20,"271":2,"272":2}}],["\\tif",{"2":{"121":4,"128":6,"129":32,"135":8,"138":2,"146":14,"148":10,"270":10,"271":6,"272":6,"273":4,"357":4,"358":16,"359":2,"360":12,"381":10}}],["🤪",{"2":{"111":1}}],["`$home",{"2":{"585":2}}],["`job=",{"2":{"503":2}}],["`json",{"2":{"110":22}}],["```json",{"2":{"328":2}}],["```",{"2":{"328":2}}],["`celestia",{"2":{"382":4}}],["`cel",{"2":{"245":2}}],["`private",{"2":{"187":2}}],["`mnemonic",{"2":{"187":2}}],["`wallet",{"2":{"187":4}}],["`trpc",{"2":{"148":4}}],["`blobstreamdatacommitmentstored`",{"2":{"146":2,"147":2,"148":2}}],["`availabledataroot`",{"2":{"145":2}}],["`",{"2":{"110":22,"245":2,"389":12,"503":2}}],["=1",{"2":{"683":1}}],["==",{"2":{"129":2,"141":6,"144":3,"146":2,"359":2,"374":1,"616":2,"617":2}}],["=",{"2":{"103":2,"113":28,"115":14,"121":12,"125":8,"128":14,"129":74,"133":8,"135":22,"138":4,"141":20,"143":6,"144":12,"145":10,"146":30,"147":18,"148":96,"160":4,"200":2,"201":1,"203":1,"218":1,"270":20,"271":14,"272":18,"273":8,"338":4,"348":29,"357":10,"358":34,"359":8,"360":26,"364":10,"365":20,"366":4,"373":1,"375":4,"376":2,"381":20,"382":44,"389":8,"459":1,"502":8,"531":10,"564":1,"574":1,"576":2,"585":22,"588":2,"589":2,"590":2,"591":3,"592":6,"595":8,"597":1,"602":4,"604":2,"605":2,"639":170,"645":8,"661":2,"716":2,"751":1}}],["zshrcecho",{"2":{"623":1}}],["zshrc",{"2":{"623":3}}],["z20=",{"2":{"321":2}}],["zxrxj96",{"2":{"132":2}}],["zhsjrh",{"2":{"127":2}}],["zero",{"2":{"103":2,"109":1,"206":1}}],["zkvm",{"2":{"206":1,"212":1,"368":1}}],["zkp",{"2":{"103":2}}],["zk",{"0":{"171":1,"172":1,"173":1,"174":1,"175":1,"176":1,"177":1,"178":1,"217":1},"1":{"172":1,"173":2,"174":2,"175":1,"176":2,"177":2,"178":2,"179":1,"180":1,"181":1,"218":1,"219":1,"220":1},"2":{"103":7,"112":2,"124":2,"150":1,"154":1,"171":1,"172":2,"173":1,"175":2,"176":4,"178":1,"179":5,"181":1,"182":1,"206":1,"212":4}}],["zip",{"2":{"73":2}}],["🌉🌉🌉🌉🌉",{"2":{"78":2}}],["⏰⏰⏰⏰⏰⏰",{"2":{"78":2}}],["🐳",{"0":{"612":1},"1":{"613":1,"614":1,"615":1,"616":1,"617":1,"618":1,"619":1,"620":1}}],["😎",{"2":{"346":1,"615":2}}],["🔐",{"0":{"316":1}}],["🙂",{"2":{"109":1}}],["🔃🔃🔃",{"2":{"78":2}}],["🚀🚀🚀🚀",{"2":{"78":2}}],["📝📝📝📝📝",{"2":{"78":2}}],["💰💰💰💰💰💰",{"2":{"78":2}}],["😁",{"2":{"76":1}}],["9390",{"2":{"689":1,"708":1}}],["96112",{"2":{"522":4}}],["999994999800000",{"2":{"525":2}}],["99",{"2":{"396":1}}],["973",{"2":{"371":1,"683":2}}],["9anbaxno1b4x5lr2qy5qwqwrmnozejkctxwzq9bexsg=",{"2":{"328":4}}],["91e04695cf9cf531bc0891e7b1d602b3e8022c86",{"2":{"328":8}}],["9070",{"2":{"705":1}}],["9080",{"2":{"693":1}}],["9099",{"2":{"708":1}}],["9095",{"2":{"689":1}}],["9090",{"2":{"311":1,"336":1,"467":1,"470":1,"483":1,"484":1,"503":2,"541":2,"552":1,"632":1,"671":1,"689":3,"693":7,"705":2,"708":5}}],["90",{"2":{"445":1}}],["9000000utia",{"2":{"659":1}}],["90000",{"2":{"129":2,"146":4}}],["9",{"0":{"78":1}}],["xzf",{"2":{"623":8}}],["xf",{"2":{"596":6}}],["x86",{"2":{"540":1,"568":1}}],["xv92a3qxexport",{"2":{"515":1}}],["xv92a3qx",{"2":{"515":3}}],["x64",{"2":{"479":1,"582":1,"627":1,"668":1,"743":1}}],["xxxxxxxxxxxxx",{"2":{"515":4}}],["xxxxxxxxxxxxxxxxxxxxxxxx",{"2":{"353":1}}],["xxxxxxxxxxxxxxxxxxxxxxxxexport",{"2":{"353":1}}],["xxd",{"2":{"341":2}}],["xyz",{"2":{"203":1,"223":1,"225":1,"226":1,"228":1,"229":1,"466":3,"639":6,"685":3,"687":3,"688":3,"689":3,"693":2,"703":3,"705":2,"706":2,"708":2}}],["xthp6xws=",{"2":{"137":2}}],["xiijl+gqm0fqxiw0z0o",{"2":{"137":2}}],["x",{"0":{"82":1,"96":1,"185":1,"217":1,"218":1,"219":1,"220":1},"1":{"83":1,"84":1,"186":1,"187":1,"188":1,"189":1,"190":1,"191":1,"192":1,"193":1,"194":1,"195":1,"196":1,"197":1,"198":1,"199":1,"200":1,"218":1,"219":1,"220":1},"2":{"76":4,"81":1,"82":1,"93":1,"106":1,"107":7,"148":4,"149":1,"186":1,"188":2,"201":3,"202":1,"203":4,"204":2,"206":3,"207":2,"218":6,"219":3,"220":3,"270":1,"336":2,"342":2,"347":1,"348":1,"357":4,"358":10,"376":2,"382":2,"494":1,"509":1,"596":6,"597":2,"751":2}}],["782",{"2":{"607":2,"756":2}}],["786bytes",{"2":{"683":1}}],["786",{"2":{"371":1,"683":1}}],["770",{"2":{"607":2,"756":2}}],["746",{"2":{"607":2,"756":2}}],["741743z",{"2":{"328":2}}],["79",{"2":{"445":1}}],["75",{"2":{"385":1,"386":1,"387":1,"446":1}}],["767",{"2":{"340":2}}],["76au8rj4vsyvasph5lgqq2kg",{"2":{"132":2}}],["7a5fabb19713d732d967b1da84fa0df5e87a7b62302d783f78743e216c1a3550",{"2":{"328":2}}],["71",{"2":{"326":4}}],["7",{"0":{"76":1},"2":{"132":2,"328":4,"340":2,"445":1,"523":2,"638":1}}],["5s",{"2":{"503":2,"639":4}}],["59",{"2":{"445":1}}],["554+0200\\tinfo\\tnode\\tnodebuilder",{"2":{"352":2}}],["5iieerohbmff+ser3jpvroieejzjby+tre0ntadqll3",{"2":{"328":4}}],["564+0200\\tinfo\\tnode\\tnodebuilder",{"2":{"352":2}}],["56",{"2":{"308":2,"666":2,"719":1}}],["512gb",{"2":{"202":1}}],["5158076079",{"2":{"78":2}}],["53",{"2":{"199":2}}],["57",{"2":{"148":2}}],["58",{"2":{"78":2,"141":4,"148":2,"199":2}}],["5",{"0":{"74":1,"619":1},"2":{"127":2,"132":2,"148":2,"350":2,"351":1,"359":1,"361":2,"377":2,"391":1,"441":1,"451":1,"503":2,"509":1,"639":4}}],["500ms",{"2":{"639":4}}],["5000100000000utia",{"2":{"500":2}}],["500000000",{"2":{"328":4}}],["500",{"2":{"308":1,"371":1,"639":2,"666":1,"719":1}}],["504g",{"2":{"199":1,"200":1}}],["50",{"2":{"20":1,"22":2,"27":1,"308":1}}],["✨",{"2":{"70":1,"78":2}}],["09",{"2":{"716":1}}],["07",{"2":{"644":8}}],["0ns",{"2":{"644":4}}],["010",{"2":{"548":1}}],["01",{"2":{"389":2,"500":2,"639":6,"687":1,"688":1,"689":1,"706":1,"707":1,"708":1,"751":2}}],["011",{"2":{"154":1}}],["0blockhash",{"2":{"341":1}}],["08",{"2":{"331":1,"716":2}}],["025",{"2":{"639":2}}],["02",{"2":{"328":2,"389":2}}],["06",{"2":{"328":4}}],["0mfhykqui2bu+u1jxpzg7qy2bvv1lb3kiu+zak7nuiy=",{"2":{"319":2,"557":2,"558":4,"559":2}}],["0+",{"2":{"314":1,"554":1,"638":1}}],["03",{"2":{"199":2}}],["05",{"2":{"199":2,"352":4}}],["001hermes",{"2":{"643":2,"644":1}}],["001",{"2":{"639":3,"640":2,"642":1,"643":4,"644":7,"645":3}}],["003442",{"2":{"563":4}}],["000",{"2":{"374":2,"375":1,"445":3,"451":3,"459":2,"509":1}}],["000000000000000000000000000000000000000000000000000000000400000000000000000000000000000000000000121312324243243288993946154604701154f739f3d1b5475786ddd960f06d8708d4e870da6501c51750",{"2":{"137":2}}],["00000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000584cb53884b394593052df4039e58c7d51f0e45cace7dd584125f62f9261b7900ac1eeef5e82349",{"2":{"132":2}}],["008",{"2":{"331":1}}],["004celestia",{"2":{"322":1}}],["004",{"2":{"322":3}}],["002",{"2":{"322":1,"336":2}}],["00",{"2":{"199":2,"445":1,"446":2,"716":3}}],["048091bc7ddc283f77bfbf91d73c44da58c3df8a9cbc867405d8b7f3daada22f",{"2":{"328":2}}],["04",{"2":{"199":16,"280":1,"479":1,"582":1,"607":2,"627":1,"668":1,"743":1,"756":2}}],["0xbe",{"2":{"270":2,"271":2,"272":2,"364":2,"365":2,"381":2}}],["0xb6052122545aacd2bdda0ca9fa56416bd968cdbc",{"2":{"86":1}}],["0x",{"2":{"195":1,"319":2}}],["0x0",{"2":{"343":1,"607":2,"756":2}}],["0x0000000000000000",{"2":{"340":2}}],["0x0000000000000000000000000000000000000000",{"2":{"340":2}}],["0x0000000000000000000000000000000000000000000000000000000000000000",{"2":{"340":2}}],["0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000",{"2":{"340":2}}],["0x0000000000000000000000000000000000000000000000000000000000000001",{"2":{"187":2}}],["0x0000000000000000000000000000000000000001",{"2":{"187":2}}],["0x007d0b2a2e2b013612e8",{"2":{"200":2}}],["0x0c",{"2":{"94":1}}],["0x04cae899fc0b7ef45c529f8bf075d54f6fb70ed9",{"2":{"86":1}}],["0x03b43f7b61fa100611191f481ef48aa1fc98f434",{"2":{"86":1}}],["0x01b5905b154f21a393f5b5a0c6d15b53a493c05e",{"2":{"85":1}}],["0x019850732270d8c436585c7921219252422228b5d0f559da0da219f0fa2b7216",{"2":{"78":2}}],["0x8e1ede4ce0865b41d714",{"2":{"200":2}}],["0x89b7c7970c13bb587893a70697ad6d2a335b6a15",{"2":{"86":1}}],["0x8f97cb7c643acd7f79f3b13841b24a243da51242",{"2":{"86":1}}],["0x8dee6e88d3b62b258c1574cbb7005e1c3cf193b60a99b5c2fcfae00819b7ed82",{"2":{"78":2}}],["0xd4b998a35d20d98ed3488221f0c161a0a9572d3de66399482553c8e3d2fae751",{"2":{"340":2}}],["0xd883010d04846765746888676f312e32312e33856c696e7578000000000000006b3afa42dce1f87f1f07a1ef569c4d43e41738ef93c865098bfa1458645f384e2e4498bcfe4ad9353ff1913a2e16162f496fafe5b0939a6c78fb5b503248d6da01",{"2":{"340":2}}],["0xde",{"2":{"270":2,"271":2,"272":2,"364":2,"365":2,"381":2}}],["0xdeadbeef",{"2":{"270":4,"271":3,"272":5,"365":3,"381":4}}],["0xd2d353916b34a877793628049c99858f04123ee1",{"2":{"85":1}}],["0xd0a6699fc7519966685181c80bf98d35afa1fc95",{"2":{"85":1}}],["0xef",{"2":{"270":2,"271":2,"272":2,"364":2,"365":2,"381":2}}],["0xe2662ff9b41f39e63a850e50e013ea66e60a4f37",{"2":{"85":1}}],["0xe24a60b758b51b0a3da5e8f4f6ddf1cd0aff646c",{"2":{"85":1}}],["0xe371afcb8437bf61bd831ef57be7a2496d88488b",{"2":{"85":1}}],["0x3",{"2":{"607":6,"756":6}}],["0x383ed2debf9f9055920cd7340418dda7e2bca6b989eb6992d83d123d4e322f2a",{"2":{"341":2}}],["0x3c44cdddb6a900fa2b585dd299e03d12fa4293bc",{"2":{"341":2}}],["0x3b6e845fb9f0c8ee4e9f6d44781f6547d9c6359a",{"2":{"86":1}}],["0x3bc040eaca40b91fa06cf55ea91842fac88b1af4",{"2":{"85":1}}],["0x3681cbb0e95ab50b63f2fc524fbbcc78adefbd33",{"2":{"85":1}}],["0xaf5c1505c7dfcebca94d9a6a8c0caf99b6c87a8ed6d6c0b3161c9026f270a84f",{"2":{"341":2}}],["0xaf8ff6af1180c8be9e4e8f3a5f882b3b227233f4abbefa479836d3721682a389",{"2":{"340":2}}],["0xaddress",{"2":{"348":11}}],["0xadd3a5dc0b8c605aeac891098e87cbaff43bb642896ebbf74f964c0690e46df2",{"2":{"341":2}}],["0xad",{"2":{"270":2,"271":2,"272":2,"364":2,"365":2,"381":2}}],["0xa83ca7775bc2889825bcdedffa5b758cf69e8794",{"2":{"216":2}}],["0xa2140c9bde000dc5e21e",{"2":{"200":2}}],["0xac3427e621c6f10dc2abdab00188d92690503914",{"2":{"85":1}}],["0xa773e19dc9e822933a7e72df9c87ed1578701d29",{"2":{"85":1}}],["0xaa",{"2":{"195":1}}],["0xaa3b8b63ccca3c98b948fd1d6ed875d378de2c6c",{"2":{"84":1}}],["0xaae3a04931345df5ac6e784bb6bdeb29b1ff0286",{"2":{"83":1}}],["0x28",{"2":{"607":2,"756":2}}],["0x2b30a00",{"2":{"607":2,"756":2}}],["0x20d8153aacc4e6d29558fa3916bff422bede9b5e",{"2":{"86":1}}],["0x2588867f19e2de51f90f0ab852c7ad11228e3d83",{"2":{"85":1}}],["0x22a6580faeca49cf86cbb2f18f2b7f98031fc6ad",{"2":{"83":1}}],["0x55527d53fda37dbf1924482b40acf8625e1caa5b",{"2":{"86":1}}],["0x5a48adf22f526ebd06e3e8856cfea2490923cc55",{"2":{"86":1}}],["0x51d196e07a27dba0f4461dd6cc26108424f196f7",{"2":{"86":1}}],["0x5187a92539bb4a2befe1fc078745c84ab6d37171",{"2":{"85":1}}],["0x53dea3a90fd6c82840a1f7224f799d622f142df4",{"2":{"85":1}}],["0x5810f0916bae1067ca1efcc00aaaf30301af001c",{"2":{"85":1}}],["0x59d2db6c5095b9e329c80211b7a761d20064379e3382d156b69e5cf3b5fe2fc7",{"2":{"78":2}}],["0x664bf4bb4a57dd5768a0a98991d77c58fb7a4e164c2581c79fb33ce9c3d4c250",{"2":{"340":2}}],["0x676d",{"2":{"319":3,"558":2}}],["0x61254e43e5c1e9e801f9c56b47a9ac3eadf6d1e9",{"2":{"86":1}}],["0x61e154128b6a1400ea8090b4431b4aa1dbb80cc4",{"2":{"85":1}}],["0x630093954cbf19fe4532a2edd0bd3b10deca7a4d",{"2":{"85":1}}],["0x6cb49605f10831749c6090ad09918bc61439bace",{"2":{"85":1}}],["0x6c7360a96165c570dcb7ce609d748d612c5fa5b76e229cd81ba5f5c93c00f805",{"2":{"78":2}}],["0x9f4dfae061b5ddd86f95a81be5daa0d7fe32e7f7f770f86dc375e0007d249bd2",{"2":{"341":2}}],["0x9039e58b2089e5f9abbb",{"2":{"200":2}}],["0x9c3f372a2aacc0d7272f56a4664311b7647b0031",{"2":{"199":32}}],["0x99e9d2f04352b42c18f1da5dd93a970f82c08afe",{"2":{"86":1}}],["0x9abc41fefae7e7543a01fa837aec909f96147280",{"2":{"86":1}}],["0x95cbda89325db5529eaf1813e181f66b83a7d65a",{"2":{"85":1}}],["0x95fea00e689e8d1cba909836e1ef1b941d5f21b1",{"2":{"85":1}}],["0x97b50f60b60d0e658fdbf185969db0a0327bd0ae9e57cd65af2a7f9be0eeb5b0",{"2":{"78":2}}],["0x40002a61b0",{"2":{"607":2,"756":2}}],["0x400065ea88",{"2":{"607":10,"756":10}}],["0x400143e048",{"2":{"607":2,"756":2}}],["0x400153b240",{"2":{"607":2,"756":2}}],["0x40f8c63e0a20b399bcd9631a22e57bb988a9400e",{"2":{"86":1}}],["0x42690c204d39600fddd3celestia",{"2":{"321":1,"329":1,"334":1}}],["0x42690c204d39600fddd3",{"2":{"262":2,"291":2,"292":2,"314":4,"319":5,"321":7,"322":2,"329":1,"334":1,"557":2,"558":2,"559":2}}],["0x4270889adcb82338c5ff5e64b45c0a3d31cfd08c",{"2":{"84":1}}],["0x44b412b291fef00398501b2ca353ea912ad0fe13",{"2":{"86":1}}],["0x4888fdf44251d456bbfca92bfc6e180cfe0b096ffbea2f6da2a203a16902214f",{"2":{"78":2}}],["0x1b91180",{"2":{"607":2,"756":2}}],["0x1bb8add5e878b12fa37756392642eb94c53a1cf4",{"2":{"84":1,"86":1}}],["0x1dcc4de8dec75d7aab85b567b6ccd41ad312451b948a7413f0a142fd40d49347",{"2":{"340":2}}],["0x1cb54d2369752ef73511c202ff9cdfd0eadf3a77b7aef0092bea63f2b5d57659",{"2":{"340":2}}],["0x1cc4551922c069a9ade06756bf14bf0410ea44ff",{"2":{"86":1}}],["0x1ae3a8dc1e7efd37f418b2987d3df74c5a917a8b",{"2":{"86":1}}],["0x10c65b27d5031ce2351c719072e58f3153228887f027f9f6d65300d2b5b30152",{"2":{"81":1}}],["0x79a0a7a1b4936aafe7a37dbfb07a6a9e55c145a4ed6fd54f962649b4b7db8de7",{"2":{"340":2}}],["0x7973d0b475e898082df25c1617cbce1917cfed17",{"2":{"85":1}}],["0x79751b011bcc20f413a2c4e3af019b6e2a9738b9",{"2":{"83":1,"85":1}}],["0x7cf3876f681dbb6eda8f6ffc45d66b996df08fae",{"2":{"216":1}}],["0x78f8b2941dde5a8a312814ebd29c2e2a36f25e91",{"2":{"86":1}}],["0x7fc4d9a24949680fad666feee7cd6a100e39c4f0",{"2":{"85":1}}],["0x7fbeb5bc73a11b438891022786feb2c624f275f0",{"2":{"78":2}}],["0xff00000000000000000000000000000000000901",{"2":{"341":3}}],["0xf0c6429ebab2e7dc6e05dafb61128be21f13cb1e",{"2":{"216":1}}],["0xf6759ff933786ddacb92",{"2":{"200":2}}],["0xfill",{"2":{"196":4}}],["0xf889a3174fddd9f78e6cd250ebf4c16f1bdd1b6a",{"2":{"86":1}}],["0xfb612fb83959b8acd3e49540b29c93c5a67e05f1",{"2":{"85":1}}],["0xf47dc66514fd78e4666e35abd12df7d1ae2c79f69f7dfedb8d98e4106142ab7c",{"2":{"78":2}}],["0xf2ddc2dad90e7e2b20a772bf89f989224165659d50824b98d7340e12265abf01",{"2":{"78":2}}],["0xf31fd34f8a9d9057198d8b13e755e583766bd528459733d948d9ffbc980c9506",{"2":{"78":2}}],["0xcryptovestor",{"2":{"687":1}}],["0xce",{"2":{"341":2}}],["0xce3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f",{"2":{"341":2}}],["0xce1636cfaf2bd5497c11",{"2":{"200":2}}],["0xc7535f078cb3880a0fd5e54fa7a3b4eaf09b3924",{"2":{"86":1}}],["0xc99eea0b8e67d5b2226ab6d37882daaf6dd7593b",{"2":{"86":1}}],["0xcd9fca5015b5ce2b06a2266e4a5dd54d9ca39f1a",{"2":{"86":1}}],["0xc61382d5609ab0ece36b2776349c8bdceeafdd13dde9624cdf3d746fb4cf7d79",{"2":{"78":2}}],["0xc3e209eb245fd59c8586777b499d6a665df3abd2",{"2":{"76":2,"82":1,"216":2}}],["0",{"2":{"67":2,"69":2,"76":6,"78":6,"81":1,"102":1,"103":2,"113":2,"124":1,"125":6,"127":2,"132":16,"137":12,"138":3,"141":2,"145":2,"148":6,"159":1,"160":6,"191":1,"262":2,"263":2,"270":2,"300":1,"321":8,"322":5,"324":2,"325":2,"326":10,"327":2,"328":10,"331":3,"336":14,"341":7,"342":8,"352":1,"358":4,"359":2,"364":4,"373":4,"374":1,"377":2,"381":2,"382":8,"389":6,"391":3,"437":2,"502":2,"503":4,"522":10,"523":2,"524":2,"525":4,"540":1,"541":4,"558":2,"559":15,"589":3,"592":4,"595":2,"597":2,"601":20,"607":14,"639":14,"644":14,"645":2,"716":3,"745":8,"751":10,"756":14}}],["qo",{"2":{"602":2}}],["qgb",{"2":{"148":2,"206":1}}],["qmaezrnbtgexcryc8pcvgrbs+umukniborae4qye7osgwcrwbvys",{"2":{"328":2}}],["qm",{"2":{"132":2}}],["q5",{"2":{"42":1}}],["q4",{"2":{"42":1}}],["q3",{"2":{"42":1}}],["q2",{"2":{"42":1}}],["q1",{"2":{"42":1}}],["q",{"2":{"38":1,"53":1,"54":1,"55":1,"58":1,"492":6,"493":4,"494":4,"495":2}}],["qubelabs",{"2":{"596":12}}],["quorum",{"2":{"494":1}}],["quot",{"0":{"563":2},"2":{"41":2,"47":2,"72":2,"75":2,"76":72,"78":12,"89":4,"95":4,"103":12,"110":44,"115":8,"121":8,"127":64,"128":24,"129":80,"132":356,"133":4,"135":24,"137":120,"146":16,"147":12,"148":84,"187":8,"189":12,"191":4,"192":4,"199":4,"228":2,"252":4,"253":24,"257":4,"262":36,"263":20,"266":4,"270":40,"271":4,"272":8,"319":36,"321":128,"322":16,"324":32,"325":32,"326":52,"327":20,"328":364,"329":4,"334":4,"336":76,"338":4,"341":4,"342":60,"343":4,"348":44,"352":2,"356":36,"357":12,"358":44,"359":24,"360":32,"364":32,"365":24,"366":12,"371":2,"376":4,"381":40,"382":64,"384":6,"389":96,"390":2,"391":4,"415":2,"438":2,"459":2,"467":2,"470":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,"557":8,"558":36,"559":12,"562":2,"568":4,"585":58,"588":4,"590":6,"591":6,"592":10,"595":8,"596":36,"601":8,"602":12,"604":4,"607":8,"622":4,"623":68,"639":188,"641":4,"644":64,"657":4,"661":4,"662":4,"716":2,"736":2,"745":28,"751":36,"756":8}}],["quic",{"2":{"326":6}}],["quicker",{"2":{"594":1}}],["quickly",{"2":{"371":1,"454":1}}],["quick",{"0":{"596":1,"614":1,"730":1},"1":{"731":1,"732":1,"733":1},"2":{"281":1,"535":1,"593":1,"596":2,"616":1,"638":1,"653":1,"733":1}}],["quickstart",{"0":{"66":1,"233":1,"667":1},"1":{"67":1,"68":1,"69":1,"70":1,"71":1,"72":1,"73":1,"74":1,"75":1,"76":1,"77":1,"78":1,"79":1,"80":1,"81":1,"82":1,"83":1,"84":1,"85":1,"86":1,"234":1,"235":1,"236":1},"2":{"68":1,"88":1}}],["queried",{"2":{"121":1,"128":1,"131":1,"138":1,"139":1,"145":2,"157":1,"195":2,"325":1}}],["queries",{"0":{"120":1,"122":1},"1":{"121":1,"122":1,"123":2,"124":2,"125":1,"126":1,"127":1,"128":1,"129":1,"130":1,"131":1,"132":1,"133":1,"134":1,"135":1,"136":1,"137":1,"138":1,"139":1,"140":1,"141":1,"142":1,"143":1,"144":1,"145":1,"146":1,"147":1,"148":1,"149":1},"2":{"106":1,"121":1,"125":1,"127":1,"132":1,"137":2,"138":1,"148":1,"167":1,"311":2,"404":1,"407":1,"416":3,"483":2,"604":2,"605":3,"632":2,"708":1}}],["queryable",{"2":{"145":2}}],["querysharerange",{"2":{"135":2}}],["querying",{"0":{"106":1,"146":1,"194":1,"279":1,"492":1},"2":{"111":2,"126":1,"134":1,"136":1,"145":1,"167":1,"193":1,"194":1,"195":2,"214":1,"224":1,"279":1,"470":1,"488":2,"523":1,"595":1,"693":1,"705":1,"706":1}}],["query",{"0":{"127":1,"523":1,"524":1,"525":1,"590":1},"2":{"105":1,"106":1,"122":1,"124":1,"126":1,"129":2,"131":1,"136":2,"140":1,"141":4,"142":1,"144":4,"145":1,"146":2,"148":2,"157":1,"160":2,"168":1,"194":1,"267":2,"324":1,"325":1,"332":1,"382":2,"407":1,"437":1,"488":6,"492":2,"494":1,"523":2,"524":2,"525":4,"537":2,"542":1,"590":1,"591":1,"604":1,"632":1,"643":2,"671":1}}],["question",{"2":{"42":3,"393":1,"642":1,"745":1}}],["questions",{"0":{"42":1},"2":{"16":1,"41":2}}],["quad",{"2":{"581":1,"719":1,"720":1}}],["quadratic",{"2":{"408":1}}],["quantum",{"2":{"206":1}}],["quarterly",{"2":{"36":1}}],["quarter",{"2":{"28":1}}],["quality",{"2":{"23":1,"40":1,"41":1,"58":1}}],["+0x110",{"2":{"607":2,"756":2}}],["+0x1c8",{"2":{"607":2,"756":2}}],["+0x124",{"2":{"607":2,"756":2}}],["+0x2c4",{"2":{"607":2,"756":2}}],["+0x2d8",{"2":{"607":2,"756":2}}],["+0x26c",{"2":{"607":2,"756":2}}],["+0xd30",{"2":{"607":2,"756":2}}],["+0x44",{"2":{"607":2,"756":2}}],["+0x64",{"2":{"607":2,"756":2}}],["+2",{"0":{"607":1,"756":1},"2":{"607":2,"756":2}}],["+stake",{"2":{"425":1}}],["+sgf6mfzmmtdkz5mllh+y7mpv9moo2x5rljle3gbfqo=",{"2":{"127":2}}],["+mhkrh9borgjn3oqt",{"2":{"132":2}}],["+1",{"2":{"113":2,"115":4}}],["+",{"2":{"26":2,"89":1,"113":2,"115":2,"159":1,"160":2,"300":1,"389":10,"555":1,"636":1,"683":1}}],["8affx48xyb",{"2":{"640":1}}],["856",{"2":{"607":2,"756":2}}],["8545",{"2":{"340":1,"341":1}}],["8545cast",{"2":{"340":1,"341":1}}],["87",{"2":{"562":2}}],["80000",{"2":{"331":3}}],["80000celestia",{"2":{"331":1}}],["8000",{"2":{"331":4,"503":1}}],["8dxemrbwrpwj6hi+3uesz+0p5vrf3v8ssaqeabelorfgaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa=",{"2":{"137":2}}],["8woyobnjs3mt1dkfrqid8vc75oja2jnf3wn",{"2":{"132":2}}],["8449",{"2":{"78":3}}],["8",{"0":{"77":1},"2":{"22":2,"81":2,"102":1,"103":2,"137":2,"148":2,"253":2,"371":2,"441":1,"451":1,"515":4,"720":1,"742":1}}],["kv",{"2":{"588":2,"590":3,"604":5}}],["kjnodes",{"2":{"449":1,"687":1,"688":1,"689":1,"697":1}}],["k×k",{"2":{"407":3,"409":1,"415":1}}],["kr",{"2":{"382":6}}],["kbps",{"2":{"308":2,"666":2,"719":1}}],["k",{"2":{"132":2,"148":8,"409":3}}],["kqkhkpjxmiby",{"2":{"132":2}}],["known",{"2":{"136":1,"206":1,"212":1}}],["knowing",{"2":{"42":1}}],["know",{"2":{"38":1,"39":1,"41":2,"94":1,"204":1,"232":1,"238":1,"436":1,"591":1}}],["knowledge",{"2":{"37":1,"40":1,"58":1,"103":2,"109":1,"206":1,"256":1}}],["knowledgeable",{"2":{"36":1,"37":1}}],["kept",{"2":{"394":1}}],["keplrbutton",{"2":{"389":2}}],["keplr",{"0":{"389":1,"419":1,"420":1},"1":{"420":1,"421":1,"422":1,"423":1},"2":{"388":1,"389":18,"417":1,"420":2,"421":1,"422":1,"448":1,"449":1,"506":1}}],["keeping",{"2":{"408":1}}],["keeps",{"2":{"95":1,"124":1}}],["keep",{"2":{"35":1,"36":1,"112":1,"197":1,"246":2,"348":1,"462":1,"504":1,"585":1,"592":3,"661":2,"751":1}}],["keymake",{"2":{"567":1}}],["keyfile",{"2":{"491":2}}],["keyforge",{"2":{"195":1}}],["keybase",{"2":{"491":1}}],["keyname",{"0":{"657":1},"2":{"253":2,"289":2,"312":6,"485":6,"633":6,"672":6,"674":6}}],["keyring",{"2":{"246":6,"247":7,"248":6,"252":2,"253":4,"266":2,"289":2,"294":2,"298":1,"312":10,"382":10,"484":2,"485":7,"489":4,"491":1,"507":6,"527":3,"531":4,"536":2,"555":3,"632":2,"633":7,"656":2,"657":2,"659":2,"672":10,"674":7,"745":2,"751":2}}],["key`",{"2":{"187":4,"245":4}}],["key=sk",{"2":{"353":2}}],["key=",{"2":{"78":2,"187":2}}],["keys`",{"2":{"382":4}}],["keystore",{"2":{"292":1,"293":1}}],["keys",{"0":{"246":1,"247":1,"248":1,"253":1,"312":1,"491":1,"520":1,"672":1},"1":{"673":1},"2":{"26":1,"73":1,"187":2,"244":2,"246":2,"252":1,"253":8,"259":1,"277":1,"293":1,"310":1,"348":2,"485":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,"555":3,"620":1,"633":1,"634":1,"640":6,"641":1,"656":2,"658":1,"674":1,"675":1,"745":2,"756":1}}],["key",{"0":{"22":1,"93":1,"244":1,"249":1,"250":1,"278":1,"289":1,"290":1,"291":1,"293":1,"294":1,"295":1,"353":1,"485":1,"490":1,"519":1,"616":1,"633":1,"656":1,"674":1},"1":{"94":1,"95":1,"96":1,"97":1,"245":1,"246":1,"247":1,"248":1,"249":1,"251":1,"252":1,"253":1,"291":1,"292":1,"294":1,"295":1,"491":1,"634":1,"675":1},"2":{"32":2,"37":1,"49":1,"71":3,"75":1,"78":2,"93":1,"113":2,"132":8,"141":4,"144":5,"160":3,"187":14,"188":8,"189":8,"195":10,"196":2,"203":2,"243":2,"244":2,"245":5,"246":18,"247":13,"248":12,"249":2,"252":5,"253":11,"258":1,"278":1,"288":3,"289":2,"291":3,"293":2,"294":2,"295":2,"312":15,"324":1,"328":4,"346":1,"348":11,"351":1,"352":8,"353":5,"359":5,"382":2,"406":1,"456":1,"484":5,"485":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,"554":2,"555":3,"567":2,"604":1,"612":2,"616":1,"632":5,"633":9,"639":12,"656":2,"657":1,"672":16,"674":9}}],["kib",{"2":{"371":1}}],["kit",{"2":{"51":2,"52":1,"57":1}}],["kickstarting",{"2":{"33":1}}],["kinds",{"2":{"167":1}}],["kindly",{"2":{"46":1}}],["kind",{"2":{"7":1}}],["kindness",{"2":{"7":1}}],["66",{"2":{"446":3}}],["67",{"2":{"445":1,"446":3}}],["64x64",{"2":{"371":1,"683":1}}],["64",{"2":{"369":1,"445":1,"540":1,"568":1,"683":1}}],["64cpu",{"2":{"199":1,"200":1,"202":1}}],["6363c68770c200fd794445668f9b18f5b1dd1125180d6e8d5ab004f7dd7a0f48",{"2":{"328":4}}],["65",{"2":{"326":4,"374":1,"375":1}}],["62562",{"2":{"322":2}}],["6093df76dba90f04ff63d197fc1569f04ed3dbe64081a0bba9bad4e69aa570d2",{"2":{"522":1}}],["6093df76dba90f04ff63d197fc1569f04ed3dbe64081a0bba9bad4e69aa570d2gas",{"2":{"522":1}}],["60",{"2":{"196":1}}],["6962",{"2":{"104":1}}],["6",{"0":{"75":1},"2":{"20":1,"23":1,"26":1,"67":1,"69":1,"389":6,"390":1,"391":1,"451":1,"478":1,"607":2,"626":1,"719":1,"756":2}}],["jemalloc",{"2":{"567":2}}],["jemallocmake",{"2":{"567":1}}],["jail",{"2":{"509":1}}],["jailing",{"0":{"509":1},"2":{"509":1}}],["jailed",{"2":{"23":2,"24":1,"26":1,"509":2}}],["jsimport",{"2":{"389":3}}],["js",{"2":{"389":2}}],["jsx",{"2":{"389":1}}],["jsoncelestia",{"2":{"265":1}}],["jsonrpc",{"2":{"127":2,"132":2,"137":2,"324":2,"325":2,"326":2,"327":2,"328":2,"336":4,"342":4}}],["jsondocker",{"2":{"89":2}}],["json",{"0":{"660":1},"2":{"61":1,"71":2,"73":2,"74":2,"75":1,"76":1,"78":2,"79":1,"89":6,"110":2,"113":2,"127":1,"132":1,"137":1,"262":2,"263":1,"265":1,"279":1,"300":1,"319":2,"321":4,"322":1,"324":1,"325":1,"326":1,"327":1,"336":2,"342":2,"343":1,"383":1,"488":6,"494":9,"495":5,"500":1,"507":18,"585":3,"658":3,"659":3,"660":6,"756":1}}],["jwt",{"2":{"357":2,"358":2,"360":2}}],["jq",{"2":{"329":2,"331":3,"334":3,"585":2,"622":10,"736":4}}],["j",{"2":{"141":4,"144":4,"148":8}}],["j2pyclt4khpiqvh7b6wp9kcdmghlct9edfydr7zaq9o=",{"2":{"132":2}}],["jh0tyrmtrdtha",{"2":{"132":2}}],["journalctl",{"2":{"499":6,"736":2,"738":2,"739":2,"740":2}}],["journey",{"2":{"29":1,"44":1}}],["jotzla2v3g2q7y+18ih5j0uxk",{"2":{"137":2}}],["job",{"2":{"42":1,"237":1,"403":1,"503":8}}],["joining",{"0":{"42":1},"2":{"36":1,"41":1,"47":1,"58":1}}],["join",{"2":{"18":1,"44":1,"475":1}}],["june",{"2":{"27":1}}],["jurisdiction",{"2":{"23":1}}],["just",{"2":{"7":1,"72":1,"78":2,"89":1,"102":1,"111":1,"244":1,"331":1,"371":1,"416":1,"443":1,"448":1,"449":1,"462":1,"503":1,"511":1,"632":1,"727":1,"733":1}}],["4hermes",{"2":{"643":1}}],["4hr",{"2":{"200":1}}],["48pm",{"2":{"607":2,"756":2}}],["48",{"2":{"607":2,"756":2}}],["482",{"2":{"371":1,"683":1}}],["4️⃣",{"0":{"423":1,"432":1}}],["4k",{"2":{"407":1,"408":1,"409":1,"415":1,"416":1}}],["443celestia",{"2":{"493":1}}],["443export",{"2":{"266":1,"529":1}}],["443",{"2":{"257":1,"266":1,"267":2,"467":1,"493":3,"529":1,"531":2,"639":2,"687":2,"688":2,"689":7,"690":2,"705":2,"706":3,"708":4}}],["478",{"2":{"371":1,"683":1}}],["47",{"2":{"199":2}}],["41",{"2":{"199":2,"445":1}}],["46",{"2":{"125":2}}],["49",{"2":{"352":4}}],["49s",{"2":{"78":2}}],["4989526079",{"2":{"78":2}}],["400000",{"2":{"639":4}}],["401",{"2":{"557":4}}],["4094×482bytes",{"2":{"683":1}}],["4094",{"2":{"371":1}}],["4095",{"2":{"371":1,"683":1}}],["4096",{"2":{"371":1,"683":1}}],["40",{"2":{"37":1,"328":2}}],["4",{"0":{"15":1,"73":1},"2":{"21":1,"22":3,"25":1,"26":1,"27":1,"67":1,"69":1,"72":1,"76":2,"78":2,"82":1,"132":2,"148":2,"159":1,"201":1,"213":1,"253":4,"266":2,"341":2,"357":2,"358":2,"360":2,"437":1,"446":1,"449":1,"467":2,"639":3,"640":2,"642":1,"643":5,"644":10,"645":3,"705":3,"706":2,"707":1,"708":4,"709":8,"716":1}}],["3152",{"2":{"644":2}}],["3108",{"2":{"639":2,"645":2}}],["33373",{"2":{"687":1,"690":1}}],["33554432",{"2":{"639":4}}],["33",{"2":{"446":6,"542":2}}],["3️⃣",{"0":{"422":1,"427":1,"431":1}}],["3d96b7d238e7e0456f6af8e7cdf0a67bd6cf9c2089ecb559c659dcaa1f880353",{"2":{"328":2}}],["3dmfzfawzmtzvxhphf5txlcj+ct3evmmfopixfh+id4=",{"2":{"137":2}}],["3+fuhlzufkjnzd8yg",{"2":{"137":2}}],["3wl9cwxs8rqbddgupu9p8jcfj+jc77zsswdcf6pm=",{"2":{"132":2}}],["34",{"2":{"125":2,"607":14,"638":1,"756":14}}],["32cpu",{"2":{"196":1}}],["32",{"2":{"125":2,"129":6,"141":3,"144":4,"145":6,"148":14,"187":2,"189":4}}],["391",{"2":{"607":2,"756":2}}],["39",{"2":{"115":2,"129":2,"146":2,"148":2,"187":10,"253":4,"262":4,"270":2,"291":4,"292":4,"314":8,"319":8,"322":4,"328":2,"329":4,"334":4,"336":4,"342":4,"358":2,"360":4,"364":2,"381":2,"382":8,"389":62,"488":2,"503":2,"519":4,"520":8,"522":8,"523":4,"524":8,"531":2,"585":48,"601":4,"615":2,"645":24,"745":4,"751":4}}],["30s",{"2":{"639":4}}],["3001",{"2":{"639":2}}],["30000",{"2":{"639":2}}],["30000000",{"2":{"340":2}}],["3000",{"2":{"503":1,"639":2}}],["300xzo8tilwpnurey6ojcrkzthq4y6yy6qh0waummrc=",{"2":{"137":2}}],["30t08",{"2":{"328":2}}],["30",{"2":{"78":2,"141":2,"148":2,"437":4,"639":4}}],["3663",{"2":{"638":1}}],["3623",{"2":{"638":1}}],["362101",{"2":{"336":2}}],["36",{"2":{"22":1,"199":2}}],["35",{"2":{"22":1,"199":14,"607":14,"756":14}}],["3",{"0":{"14":1,"71":1,"72":1,"596":1,"607":1,"756":1},"2":{"22":1,"27":1,"37":1,"76":2,"78":4,"81":1,"89":2,"93":1,"124":1,"125":6,"148":2,"253":2,"350":2,"351":1,"357":2,"358":2,"359":4,"360":6,"361":2,"446":1,"467":2,"494":1,"502":2,"507":4,"540":1,"568":1,"607":2,"639":4,"756":2}}],["27090",{"2":{"705":1,"708":1}}],["2727",{"2":{"644":6}}],["27t13",{"2":{"328":2}}],["2382",{"2":{"644":4}}],["2371495",{"2":{"597":1,"716":1,"751":1}}],["2371495celestia",{"2":{"597":1,"751":1}}],["23",{"2":{"558":2}}],["26357",{"2":{"705":1}}],["26557",{"2":{"693":1}}],["26660",{"2":{"502":4,"503":2}}],["26656",{"2":{"467":1,"602":2,"662":1}}],["26659version",{"2":{"253":1}}],["26659",{"2":{"252":4,"253":3,"552":1}}],["26657",{"2":{"76":2,"128":2,"129":2,"336":1,"467":1,"469":1,"484":2,"541":2,"552":1,"601":4,"639":4,"692":1,"693":6,"704":1,"705":1,"706":3,"736":2}}],["26658curl",{"2":{"336":1,"342":1}}],["26658",{"2":{"76":3,"269":2,"315":1,"336":1,"342":1,"360":2,"363":2,"552":1}}],["26",{"2":{"445":1}}],["2️⃣",{"0":{"421":1,"426":1,"430":1}}],["2k",{"2":{"409":1}}],["2k×2k",{"2":{"407":2,"415":1}}],["2585031",{"2":{"597":1,"716":1,"751":1}}],["2585031celestia",{"2":{"597":1,"751":1}}],["25231",{"2":{"559":5}}],["252614",{"2":{"319":3,"321":9,"334":2}}],["252607",{"2":{"319":2}}],["25",{"2":{"391":1,"446":1,"509":1,"607":2,"756":2}}],["25630",{"2":{"326":4}}],["256gb",{"2":{"196":1}}],["2s",{"2":{"238":1}}],["22t14",{"2":{"352":4}}],["2283",{"2":{"340":2}}],["22",{"2":{"148":2,"252":2,"253":2,"318":4}}],["28657",{"2":{"693":1}}],["28090",{"2":{"689":1,"693":1}}],["28",{"2":{"141":5,"142":3,"148":8,"716":1}}],["28217653",{"2":{"78":2}}],["28217647",{"2":{"78":2}}],["2zdir",{"2":{"132":2}}],["299137127z",{"2":{"328":2}}],["29",{"2":{"125":2,"141":7,"142":1,"148":2,"607":14,"756":14}}],["2f",{"2":{"62":1}}],["2e",{"2":{"62":1}}],["2d",{"2":{"62":1}}],["2c",{"2":{"62":1,"65":1}}],["2b",{"2":{"62":1,"65":1}}],["2ae1ddf74ef7",{"2":{"125":2}}],["2a",{"2":{"62":1,"65":1}}],["24",{"2":{"24":1,"341":2,"607":4,"756":4}}],["21090",{"2":{"708":1}}],["214",{"2":{"607":2,"756":2}}],["216\\tnew",{"2":{"352":2}}],["211\\tno",{"2":{"352":2}}],["21572",{"2":{"341":2}}],["21568",{"2":{"340":2}}],["2121",{"2":{"252":2,"253":2,"326":8,"467":4,"541":2,"542":4,"552":2,"739":2}}],["21",{"2":{"22":1,"252":2,"253":2,"509":1}}],["2019",{"2":{"678":1}}],["2001",{"2":{"607":2,"756":2}}],["200",{"2":{"326":4}}],["20000",{"2":{"639":2}}],["20000utia",{"2":{"266":2}}],["2000",{"2":{"261":1}}],["2000celestia",{"2":{"261":1}}],["2048",{"2":{"200":4}}],["20ignition",{"2":{"199":16}}],["2023",{"2":{"328":4}}],["20210819022825",{"2":{"125":2}}],["2025",{"2":{"27":1}}],["2024",{"2":{"20":1,"27":2,"199":14,"352":4,"607":1,"716":3,"756":1}}],["20",{"2":{"22":1,"27":1,"37":1,"127":1,"128":2,"280":1,"445":1,"479":1,"582":1,"627":1,"668":1,"743":1}}],["2",{"0":{"13":1,"62":1,"65":1,"70":1,"130":1,"292":1,"595":1,"619":1,"642":1},"2":{"16":2,"22":2,"23":1,"26":1,"27":1,"35":1,"37":1,"42":1,"64":1,"65":1,"67":1,"78":2,"81":4,"103":2,"124":1,"127":2,"130":1,"132":2,"137":2,"148":3,"160":1,"172":1,"191":1,"300":1,"324":2,"325":2,"326":2,"327":2,"328":4,"336":4,"340":2,"342":4,"357":2,"358":2,"360":2,"371":1,"407":1,"409":3,"410":3,"412":1,"415":1,"444":1,"446":2,"456":1,"467":2,"500":2,"507":4,"509":1,"524":2,"581":1,"597":1,"607":2,"706":1,"707":1,"708":1,"709":6,"720":2,"742":1,"746":1,"751":3,"756":2}}],["1×478bytes",{"2":{"683":1}}],["1️⃣",{"0":{"420":1,"425":1,"429":1}}],["1utia",{"2":{"382":2,"507":2}}],["1usgft4=",{"2":{"132":2}}],["1751707",{"2":{"597":1,"716":1,"751":1}}],["1751707celestia",{"2":{"597":1,"751":1}}],["17",{"2":{"445":1,"716":1}}],["171",{"2":{"326":4}}],["172118057",{"2":{"324":2}}],["1s",{"2":{"237":1,"238":1}}],["11090",{"2":{"689":1}}],["11036",{"2":{"503":1}}],["115",{"2":{"542":2}}],["118",{"2":{"389":2,"390":1,"391":1}}],["11celestia",{"2":{"352":1}}],["11155111",{"2":{"348":2}}],["1117733",{"2":{"348":2}}],["1141",{"2":{"340":2}}],["11",{"2":{"199":2,"289":2,"311":2,"328":4,"352":3,"390":2,"467":11,"470":2,"473":1,"474":1,"484":2,"607":4,"671":2,"672":2,"716":1,"756":4}}],["1hr",{"2":{"196":2}}],["19000000utia",{"2":{"493":2}}],["19l",{"2":{"262":2}}],["196",{"2":{"199":32}}],["19",{"2":{"102":1,"103":2,"199":16,"328":2,"445":1,"716":1}}],["16002",{"2":{"708":1}}],["1606",{"2":{"607":2,"756":2}}],["169",{"2":{"607":2,"756":2}}],["1699638350",{"2":{"340":2}}],["1637",{"2":{"607":2,"756":2}}],["1639397",{"2":{"262":2}}],["1687908352",{"2":{"523":2}}],["1686748051",{"2":{"522":8,"523":2,"529":2}}],["16xlarge",{"2":{"202":1}}],["16",{"2":{"78":2,"148":2,"478":1,"581":1,"596":12,"626":1,"716":1,"719":2,"720":2,"742":1}}],["1443",{"2":{"708":1}}],["1400000",{"2":{"563":1}}],["147105",{"2":{"329":2}}],["14",{"2":{"78":2,"125":2,"314":1,"542":2,"554":1,"607":2,"716":3,"756":2}}],["180000",{"2":{"639":4}}],["18",{"2":{"71":1,"125":2,"716":1}}],["1318129",{"2":{"557":2,"558":2,"559":2}}],["1317",{"2":{"467":1,"707":1}}],["1350632",{"2":{"132":2}}],["13",{"2":{"67":1,"78":2,"81":2,"437":2,"568":1,"682":1}}],["1g",{"2":{"64":1}}],["1f",{"2":{"64":1}}],["1e",{"2":{"64":1}}],["1d",{"2":{"64":1}}],["1container",{"2":{"252":1,"253":1}}],["1c",{"2":{"61":1,"64":1}}],["1b",{"2":{"61":1,"64":1}}],["1a",{"2":{"61":1,"64":1}}],["151",{"2":{"542":2}}],["15s",{"2":{"503":2}}],["1559",{"2":{"373":1}}],["15",{"2":{"27":2,"78":2,"127":1,"128":2,"137":1,"138":2,"145":1,"199":16,"352":4,"445":1,"503":2}}],["15002",{"2":{"689":1}}],["1500",{"2":{"26":1}}],["10790",{"2":{"708":2}}],["10s",{"2":{"639":4}}],["103",{"2":{"494":2}}],["105",{"2":{"494":3}}],["1089",{"2":{"386":1}}],["1080694",{"2":{"132":2}}],["1040676758",{"2":{"341":2}}],["106",{"2":{"326":4}}],["1024cd",{"2":{"200":1}}],["1024",{"2":{"200":14}}],["1095604",{"2":{"132":2}}],["10+",{"2":{"34":1}}],["10",{"2":{"23":3,"76":4,"127":3,"128":2,"137":1,"138":1,"148":2,"319":1,"326":4,"348":1,"374":1,"390":1,"441":1,"451":1,"472":1,"478":1,"626":1,"639":4,"710":1,"719":2}}],["1000",{"2":{"639":2}}],["10001",{"2":{"615":4}}],["1000utia",{"2":{"507":4}}],["100000utia",{"2":{"507":2,"522":12,"529":4}}],["100000",{"2":{"331":4,"522":4,"523":2,"525":2,"529":2,"639":4}}],["10000000utia",{"2":{"657":2}}],["100000000000utia",{"2":{"495":2}}],["1000000000",{"2":{"494":2}}],["1000000000utia",{"2":{"494":2,"495":2}}],["1000000",{"2":{"325":2,"499":2}}],["1000000utia",{"2":{"266":2,"745":2}}],["100",{"2":{"20":1,"76":2,"147":2,"440":2,"639":2,"666":1,"719":1}}],["12d3koowd5wcbjxkqudjhxfjtfmrzoysgvltvht5hmovbslcbv22",{"2":{"739":1}}],["12d3koowd5wcbjxkqudjhxfjtfmrzoysgvltvht5hmovbslcbv22node",{"2":{"739":1}}],["12d3koowkeertzvmpudxyszo2edqps6ms67n6lt5mpdulskpsxbq",{"2":{"542":2}}],["12d3koowlt1yssrd7xwsbjh7tu1hqanf5m64dhv6aum6cyejxmpk",{"2":{"467":1}}],["12d3koowewuqrjulanpukdfgvohw3roeuu53ec9t9v5cww3mkvdq",{"2":{"467":1}}],["12d3koowcmgm5ezwvfcn9zlavigfluwafxp5pcm78nfkb9jpbtua",{"2":{"467":1}}],["12d3koowgqwzdeqm54dce6lxzffr97bnhvm6rn7km7mfwdomfm4s",{"2":{"467":1}}],["12d3koowffhcaaqy56oeqy3plzudlsv4ryafvwkatzrepupdoslp",{"2":{"326":2}}],["127",{"2":{"336":3,"342":2,"601":2,"639":4}}],["1234",{"2":{"187":3}}],["123456",{"2":{"564":1}}],["1234567890",{"2":{"382":2}}],["12345=",{"2":{"187":6}}],["12345",{"2":{"187":9}}],["128gb",{"2":{"369":1}}],["128",{"2":{"132":2}}],["120000",{"2":{"132":2}}],["12",{"2":{"22":2,"282":1,"445":1}}],["1",{"0":{"12":1,"61":1,"64":1,"69":1,"126":1,"291":1,"594":1},"1":{"127":1,"128":1},"2":{"16":2,"21":1,"22":7,"23":1,"24":1,"25":1,"27":4,"35":1,"42":1,"67":1,"69":1,"70":1,"78":4,"81":6,"89":4,"113":4,"115":2,"124":1,"125":4,"127":2,"128":6,"129":2,"130":1,"132":2,"137":5,"138":2,"141":6,"142":2,"145":2,"148":14,"191":1,"201":1,"213":2,"237":2,"238":1,"252":1,"253":1,"311":2,"324":2,"325":2,"326":6,"327":2,"328":13,"331":1,"336":7,"342":6,"343":1,"352":2,"357":2,"358":2,"360":2,"364":2,"371":4,"377":1,"382":2,"389":2,"391":1,"400":1,"409":1,"441":1,"445":1,"446":8,"451":3,"459":2,"467":2,"470":2,"478":2,"484":2,"494":1,"499":2,"500":2,"509":1,"559":5,"581":2,"593":1,"597":1,"601":2,"607":2,"626":2,"639":14,"671":2,"672":2,"683":1,"706":1,"707":1,"708":1,"709":4,"719":2,"720":2,"742":2,"751":3,"756":2}}],["ua",{"2":{"705":2,"706":1,"707":1,"708":1,"712":1}}],["uatom",{"2":{"639":2}}],["u",{"2":{"499":6,"736":2,"738":2,"739":2,"740":2}}],["udp",{"2":{"326":6,"541":1,"552":1}}],["uri",{"2":{"311":6,"312":6,"483":8,"484":2,"485":6,"493":4,"544":4,"545":2,"546":2,"547":8,"550":2,"632":2,"633":6,"674":6,"739":2,"740":2,"748":2}}],["urldocker",{"2":{"617":3}}],["urlcelestia",{"2":{"261":1}}],["url=this",{"2":{"614":2}}],["url=https",{"2":{"266":2,"529":2}}],["url=rpc",{"2":{"259":2}}],["url=",{"2":{"78":4,"194":2}}],["url",{"0":{"75":1},"2":{"41":2,"75":5,"76":4,"78":1,"188":4,"189":12,"195":5,"196":2,"203":2,"226":1,"261":1,"262":2,"266":3,"269":2,"270":4,"271":4,"272":4,"273":4,"314":1,"315":1,"340":2,"341":2,"348":2,"363":2,"364":4,"365":4,"366":4,"381":4,"390":3,"391":4,"466":1,"469":1,"470":3,"503":2,"529":5,"531":1,"614":9,"617":3,"639":4,"685":1,"692":1,"693":3,"703":1,"704":1,"705":1}}],["urls",{"0":{"224":1},"1":{"225":1,"226":1},"2":{"41":2,"75":1,"187":2,"196":1,"224":2}}],["ubuntu",{"2":{"280":1,"479":1,"582":1,"612":1,"622":1,"623":2,"627":1,"668":1,"743":1}}],["ugp0sv9ynei5poiyr7rdodswwlfbh2o3xirsmmnmbks=",{"2":{"137":2}}],["uhlbgdgo=",{"2":{"132":2}}],["uhhb3u6dpayj1pjnefrwq5he=",{"2":{"127":2}}],["utc",{"2":{"716":3}}],["ut8sjtguh+akimnqtkuuu",{"2":{"132":2}}],["utility",{"0":{"244":1,"358":1},"1":{"245":1,"246":1,"247":1,"248":1,"249":1},"2":{"243":1,"244":3,"246":1,"357":1,"358":2,"359":1,"567":1,"672":1}}],["utilizing",{"0":{"41":1}}],["utilize",{"2":{"37":1,"39":2,"40":1,"281":1}}],["utia=tia×10−6",{"2":{"451":1}}],["utiacelestia",{"2":{"267":1}}],["utia",{"0":{"331":1,"459":1},"2":{"76":1,"261":1,"262":2,"263":2,"267":1,"322":1,"324":2,"325":2,"331":6,"389":6,"390":1,"391":1,"459":3,"460":1,"494":2,"522":4,"523":2,"525":4,"639":2}}],["uint32",{"2":{"376":4,"382":4}}],["uint8",{"2":{"142":2,"191":2}}],["uint64",{"2":{"110":8,"111":4,"113":4,"115":4,"129":10,"145":4,"148":14,"204":3,"273":2,"358":2}}],["uint256",{"2":{"103":2,"139":2,"141":4,"144":4,"145":4,"160":3,"204":1}}],["ui",{"2":{"61":1,"64":1,"65":2,"71":1}}],["ultimate",{"2":{"43":1}}],["upper",{"2":{"683":1}}],["upload",{"2":{"282":1,"308":1,"384":1,"478":1,"486":1,"581":1,"626":1,"634":1,"640":1,"666":1,"675":1,"742":1}}],["updocker",{"2":{"253":1}}],["updating",{"2":{"156":1,"195":3,"201":1,"515":2}}],["updatesudo",{"2":{"622":1}}],["updates",{"2":{"195":1,"219":1,"462":1,"679":1,"722":1,"725":1}}],["updateheaders",{"2":{"113":2}}],["updated",{"2":{"78":2,"113":1,"187":1,"201":2,"213":1,"358":1,"462":1,"725":1}}],["update",{"2":{"71":1,"125":1,"156":1,"187":2,"195":5,"201":1,"212":3,"213":1,"219":1,"238":1,"360":1,"377":1,"394":1,"413":1,"437":1,"494":2,"560":4,"622":4,"660":1,"756":1}}],["upgrading",{"0":{"570":1},"2":{"195":2}}],["upgraded",{"2":{"597":1,"607":1,"751":1,"756":1}}],["upgradeexecutor",{"2":{"78":2}}],["upgrade",{"0":{"510":1,"714":1,"716":1},"1":{"715":1,"716":1},"2":{"24":1,"78":4,"81":1,"195":2,"277":1,"475":2,"510":1,"570":1,"597":10,"607":2,"622":3,"698":2,"713":2,"714":4,"715":7,"716":14,"729":2,"751":10,"756":3}}],["upgrades",{"0":{"462":1,"475":1,"679":1,"698":1,"713":1,"729":1},"2":{"23":1,"461":1,"462":2,"475":2,"494":1,"510":1,"678":1,"679":1,"698":2,"713":2,"714":5,"715":1,"716":1,"727":2,"729":2}}],["upon",{"2":{"50":1,"113":1,"119":1,"321":1,"336":1,"446":1,"485":1,"633":1,"674":1}}],["upcoming",{"2":{"47":1,"137":1,"138":1,"151":1,"153":1,"206":1,"714":1}}],["ups",{"2":{"33":1}}],["up",{"0":{"78":1,"309":1,"346":1,"360":1,"476":1,"479":1,"513":1,"518":1,"530":1,"582":1,"583":1,"585":1,"624":1,"627":1,"653":1,"664":1,"668":1,"734":1,"741":1,"743":1},"1":{"310":1,"311":1,"312":1,"477":1,"478":1,"479":1,"480":2,"481":1,"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,"583":1,"584":1,"585":1,"625":1,"626":1,"627":1,"628":2,"629":1,"630":1,"631":1,"632":1,"633":1,"634":1,"635":1,"636":1,"654":1,"655":1,"656":1,"657":1,"658":1,"659":1,"660":1,"661":1,"662":1,"663":1,"665":1,"666":1,"667":1,"668":1,"669":2,"670":1,"671":1,"672":1,"673":1,"674":1,"675":1,"676":1,"677":1,"735":1,"736":1,"737":1,"738":1,"739":1,"740":1,"742":1,"743":1,"744":2,"745":2,"746":1,"747":1,"748":1,"749":1,"750":1,"751":1,"752":1,"753":1,"754":1,"755":1,"756":1},"2":{"20":1,"22":1,"25":1,"30":1,"31":1,"32":1,"41":3,"42":1,"48":1,"54":1,"55":1,"75":2,"77":1,"153":2,"184":1,"238":1,"243":1,"246":1,"252":2,"253":3,"284":3,"310":1,"318":1,"351":1,"353":2,"355":1,"356":2,"361":1,"385":1,"402":1,"403":1,"476":1,"486":1,"487":2,"501":1,"503":1,"504":1,"517":1,"526":2,"529":1,"530":1,"539":1,"547":1,"564":1,"567":1,"577":1,"580":1,"594":2,"621":1,"624":1,"628":1,"634":1,"641":1,"653":1,"654":1,"664":1,"668":2,"675":1,"676":1,"717":1,"720":1,"735":1,"736":1,"741":1,"743":1,"750":2,"751":1,"756":2}}],["uptime",{"2":{"20":1,"22":1,"26":2,"230":1,"544":1,"693":1}}],["unexplored",{"2":{"679":1}}],["unexpectedly",{"2":{"462":1}}],["unordered",{"2":{"644":2}}],["unmarshaling",{"2":{"557":2}}],["unjail",{"2":{"498":3,"509":2}}],["unjailing",{"0":{"498":1}}],["unbonded",{"2":{"509":1}}],["unbond",{"2":{"497":3}}],["unbonding",{"2":{"193":1,"509":1}}],["unused",{"2":{"356":1,"373":1,"375":1}}],["untested",{"2":{"280":1}}],["until",{"2":{"94":1,"115":1,"377":1,"382":2,"402":1,"441":1,"451":1,"462":1,"638":1,"736":1}}],["unstable",{"2":{"725":1}}],["unstake",{"2":{"423":1,"427":1,"432":1}}],["unsafe",{"2":{"561":4,"599":2,"654":2}}],["unsignedtx",{"2":{"507":8}}],["unsubscribe",{"2":{"147":2}}],["unsolicited",{"2":{"13":1,"14":1}}],["unknown",{"2":{"145":1}}],["unauthorized",{"2":{"557":2}}],["unable",{"2":{"377":1,"400":1,"509":1}}],["unavailability",{"2":{"254":1}}],["unavailable",{"0":{"159":1},"2":{"115":1,"164":1}}],["unaffected",{"2":{"241":1}}],["unacceptable",{"2":{"7":1,"10":1}}],["unpacked",{"2":{"96":1}}],["unpealed",{"2":{"95":2}}],["unprofessional",{"2":{"12":1}}],["unlimited",{"2":{"502":2}}],["unlike",{"2":{"112":1,"154":1,"204":1,"298":1}}],["unlocked",{"2":{"446":9}}],["unlocks",{"0":{"446":1},"2":{"446":4}}],["unlock",{"2":{"389":2,"446":3}}],["unlocking",{"2":{"49":1}}],["unless",{"2":{"23":1,"184":1,"336":1,"346":1,"653":1}}],["unix",{"2":{"554":1}}],["unit",{"2":{"382":2,"736":2,"738":2,"739":2,"740":2}}],["unify",{"2":{"258":1}}],["unified",{"2":{"124":1}}],["universal",{"2":{"199":1}}],["university",{"2":{"34":1}}],["universities",{"2":{"34":2,"38":1}}],["unique",{"2":{"41":1,"71":2,"240":1,"407":2,"409":1,"515":1}}],["undelegate",{"0":{"497":1},"2":{"497":1}}],["undelegation",{"0":{"24":1}}],["undergoes",{"2":{"462":1}}],["underneath",{"0":{"79":1,"344":1},"1":{"345":1,"346":1,"347":1,"348":1,"349":1},"2":{"94":2,"95":1,"221":1,"284":1,"349":1}}],["underlying",{"2":{"71":1,"78":1,"94":1,"97":1}}],["understand",{"2":{"41":1,"42":3,"58":1,"76":1,"106":1,"286":1,"435":1,"529":1}}],["understanding",{"0":{"573":1},"1":{"574":1,"575":1,"576":1,"577":1,"578":1,"579":1},"2":{"36":1,"58":1,"231":1,"238":1,"251":1,"441":1,"613":2}}],["under",{"2":{"23":1,"26":1,"61":2,"65":1,"75":1,"81":1,"105":1,"133":1,"179":1,"193":1,"195":1,"199":2,"200":1,"244":1,"246":1,"361":1,"371":2,"373":1,"502":2,"563":1,"595":1,"753":1}}],["unwelcome",{"2":{"12":1}}],["usr",{"2":{"540":1,"568":1,"607":4,"623":21,"756":4}}],["usual",{"2":{"160":1}}],["usually",{"2":{"71":1,"484":1,"577":1,"632":1,"671":1}}],["usable",{"0":{"139":1},"1":{"140":1,"141":1,"142":1,"143":1,"144":1,"145":1,"146":1,"147":1,"148":1},"2":{"106":1,"127":1,"132":1,"137":1}}],["usage",{"0":{"103":1,"179":1,"180":1,"181":1},"1":{"180":1,"181":1},"2":{"181":1,"357":2,"358":2,"360":2,"488":2,"491":1,"492":1,"493":2,"504":1,"567":1}}],["using",{"0":{"131":1,"134":1,"244":1,"259":1,"262":1,"266":1,"336":1,"468":1,"559":1},"1":{"132":1,"133":1,"135":1,"245":1,"246":1,"247":1,"248":1,"249":1,"260":1,"261":1,"267":1,"469":1,"470":1},"2":{"9":1,"30":1,"31":1,"42":1,"66":1,"69":1,"71":1,"75":1,"76":6,"78":1,"95":1,"97":1,"98":1,"102":3,"107":3,"111":1,"112":2,"115":1,"121":1,"122":1,"124":2,"126":1,"128":1,"130":1,"133":1,"134":2,"135":1,"138":1,"140":2,"145":2,"147":1,"148":3,"153":1,"154":1,"155":2,"157":3,"158":2,"160":6,"161":1,"163":1,"164":2,"169":1,"171":1,"173":1,"176":3,"179":1,"182":1,"184":4,"187":7,"189":1,"193":1,"194":1,"199":1,"200":1,"203":2,"206":2,"212":1,"218":1,"242":1,"243":1,"245":1,"252":1,"253":3,"254":1,"260":1,"262":1,"268":1,"271":3,"272":3,"273":1,"277":1,"284":2,"288":1,"306":1,"312":1,"313":1,"315":1,"316":1,"317":2,"318":2,"320":1,"334":1,"336":2,"344":1,"348":1,"350":3,"352":1,"358":8,"359":2,"360":2,"362":1,"365":1,"366":1,"368":1,"369":1,"376":2,"380":1,"382":4,"383":2,"389":2,"409":1,"410":2,"412":3,"415":2,"434":1,"438":2,"454":1,"470":1,"471":1,"483":1,"485":1,"494":1,"495":1,"505":1,"506":1,"511":1,"521":1,"529":1,"534":1,"539":1,"542":1,"555":3,"559":1,"560":1,"565":2,"590":1,"592":1,"594":1,"612":3,"614":1,"616":2,"620":1,"623":1,"632":1,"633":1,"637":1,"639":1,"672":1,"673":1,"674":1,"707":2,"708":1,"710":1,"745":1,"748":1}}],["useful",{"2":{"42":1,"51":1,"77":1,"462":1,"585":1,"707":1,"734":1,"736":1}}],["user=$user",{"2":{"736":2,"738":2,"739":2,"740":2}}],["user",{"0":{"382":1},"2":{"36":1,"41":2,"42":1,"107":2,"109":1,"237":1,"254":1,"277":1,"297":1,"298":1,"315":2,"319":1,"373":2,"375":3,"377":1,"382":8,"384":1,"387":1,"389":4,"455":1,"458":1,"503":1,"508":1,"555":8,"579":1,"615":1,"648":2,"736":2,"738":2,"739":2,"740":2}}],["users",{"2":{"18":1,"71":1,"150":1,"168":1,"206":2,"241":1,"254":1,"300":1,"302":1,"318":1,"347":1,"370":1,"373":3,"375":3,"376":1,"377":2,"378":1,"393":2,"400":2,"403":2,"404":1,"414":1,"415":1,"438":1,"447":1,"469":1,"470":1,"612":1,"620":1,"679":1,"692":1,"693":1,"704":1,"705":1,"706":1,"707":1,"708":1,"722":1}}],["uses",{"0":{"148":1},"2":{"34":1,"81":1,"92":1,"97":1,"111":4,"115":1,"148":4,"163":1,"167":1,"173":1,"182":1,"209":1,"210":1,"212":1,"218":1,"222":1,"244":1,"270":2,"284":1,"301":1,"359":2,"373":1,"375":1,"376":1,"377":1,"381":2,"398":1,"407":1,"435":1,"441":1,"455":1,"502":1,"509":1,"557":1,"558":1,"595":1,"637":1,"648":1}}],["use",{"0":{"164":1,"165":1,"166":1,"167":1,"168":1,"169":1,"170":1,"172":1,"173":1,"174":1,"175":1,"177":1,"178":1,"184":1},"1":{"165":1,"166":1,"167":1,"169":1,"170":1,"173":1,"174":1,"176":1,"177":1,"178":1},"2":{"7":1,"12":1,"18":1,"39":1,"41":1,"42":2,"69":1,"75":2,"76":3,"89":2,"94":1,"109":1,"110":2,"112":2,"113":1,"118":1,"122":1,"124":1,"129":2,"137":1,"138":1,"139":1,"141":2,"146":4,"147":3,"148":4,"150":1,"159":2,"160":2,"171":1,"175":1,"182":1,"184":1,"187":4,"189":3,"195":1,"196":1,"198":1,"200":1,"206":1,"212":3,"218":1,"226":1,"241":1,"245":1,"246":1,"248":1,"256":1,"259":1,"262":1,"266":1,"269":4,"270":2,"277":2,"279":1,"289":1,"292":1,"293":1,"302":1,"304":1,"307":1,"311":2,"314":4,"315":1,"316":1,"317":2,"319":4,"321":5,"322":2,"331":1,"332":1,"333":2,"336":1,"344":1,"356":3,"358":3,"363":4,"364":7,"369":1,"372":1,"373":1,"374":1,"382":6,"384":2,"385":1,"388":1,"389":1,"412":1,"413":1,"438":1,"453":1,"474":1,"483":1,"494":2,"495":1,"503":2,"510":1,"512":1,"515":1,"516":1,"522":1,"527":1,"529":3,"530":1,"531":1,"539":1,"547":1,"552":1,"554":1,"565":1,"567":1,"574":1,"610":1,"612":1,"623":1,"632":1,"636":1,"638":1,"640":1,"642":1,"643":2,"644":1,"645":1,"655":2,"660":1,"696":1,"712":1,"716":1,"733":1}}],["used",{"2":{"3":1,"71":2,"75":1,"76":4,"78":1,"94":3,"95":1,"110":2,"112":1,"113":2,"115":1,"121":2,"132":2,"134":1,"136":1,"141":1,"142":1,"151":2,"159":1,"172":2,"184":2,"187":5,"198":2,"199":3,"200":3,"201":1,"206":1,"212":1,"219":1,"244":1,"258":1,"260":1,"277":1,"280":1,"282":1,"299":1,"315":2,"322":1,"334":1,"358":1,"373":1,"374":2,"375":1,"376":2,"407":1,"412":1,"415":1,"461":1,"466":1,"482":1,"484":1,"494":1,"501":1,"509":1,"522":2,"536":1,"539":1,"541":1,"552":1,"573":1,"602":1,"605":1,"615":1,"621":1,"632":2,"637":1,"640":1,"685":1,"703":1,"708":1,"709":1,"715":1,"745":2,"751":2}}],["us",{"2":{"7":1,"23":1,"26":1,"42":6,"96":3,"241":1,"334":1,"357":1,"359":1,"389":1,"683":2}}],["h",{"2":{"336":4,"342":4}}],["huzwottdmd36n1f75a9bshxnlrascnnpqiwqihdvhcu",{"2":{"328":4}}],["hub",{"2":{"223":1,"228":1,"637":1,"639":1,"640":1}}],["hzfy",{"2":{"132":2}}],["historic",{"2":{"437":1,"438":3}}],["historical",{"0":{"403":1,"404":1,"591":1},"2":{"400":2,"403":2,"404":2,"405":1,"436":1,"438":9,"591":1}}],["history",{"2":{"400":1,"404":1,"470":1,"564":1,"580":1,"693":1,"694":1,"717":1}}],["hidden",{"2":{"111":1,"350":1}}],["hiring",{"2":{"38":1}}],["highly",{"2":{"484":1,"609":1,"632":1}}],["highlighted",{"2":{"122":1}}],["highlight",{"2":{"38":1}}],["highlights",{"2":{"32":1}}],["higher",{"2":{"71":1,"241":1,"322":1,"371":1,"372":1,"377":1,"435":1,"441":1}}],["high",{"2":{"22":1,"40":1,"58":1,"71":1,"95":1,"96":2,"107":1,"109":1,"206":1,"211":1,"212":1,"348":1,"385":1,"389":2,"407":1,"563":1,"708":1}}],["hollow",{"2":{"519":2}}],["holders",{"2":{"443":1,"495":1}}],["holds",{"2":{"73":1}}],["hold",{"2":{"36":1}}],["homebrew",{"2":{"622":4}}],["homecd",{"2":{"519":1}}],["home=",{"2":{"515":2}}],["home",{"0":{"515":1},"2":{"246":1,"315":1,"515":4,"519":3,"520":4,"522":12,"523":4,"524":4,"525":8,"541":1,"552":1,"555":8,"599":2,"607":2,"616":8,"617":8,"654":2,"756":3}}],["hood",{"2":{"81":1,"244":1,"373":1}}],["hosted",{"2":{"203":1,"221":1,"222":1,"547":1}}],["hosts",{"2":{"76":2}}],["host",{"2":{"76":8,"89":2,"237":1,"238":1,"531":2,"559":2,"615":3,"639":4,"643":6}}],["hosting",{"2":{"26":1,"37":1,"40":1,"42":2,"66":1}}],["honesty",{"2":{"400":1}}],["honest",{"2":{"71":1,"397":2,"407":1,"409":1,"416":2}}],["hour",{"2":{"42":1,"201":1,"213":2,"593":1}}],["hours",{"2":{"24":1,"64":1,"65":1,"156":1,"201":1,"213":1,"593":1}}],["hope",{"2":{"42":2}}],["however",{"2":{"38":1,"41":1,"111":1,"115":1,"160":1,"171":1,"182":1,"184":1,"315":1,"373":1,"374":1,"398":1,"408":1,"410":1,"412":1,"494":1,"547":1,"579":1,"605":1}}],["how",{"0":{"214":1,"219":1,"395":1,"417":1,"499":1,"511":1},"1":{"418":1,"419":1,"420":1,"421":1,"422":1,"423":1,"424":1,"425":1,"426":1,"427":1,"428":1,"429":1,"430":1,"431":1,"432":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":{"21":1,"35":1,"36":1,"38":2,"41":1,"42":1,"54":1,"59":1,"71":1,"76":1,"79":1,"96":1,"98":1,"106":1,"107":1,"109":1,"111":1,"112":1,"119":2,"122":1,"124":2,"157":1,"167":1,"201":1,"213":1,"220":1,"242":1,"251":1,"252":1,"279":1,"288":2,"304":1,"313":1,"314":2,"316":1,"344":1,"347":1,"348":1,"350":1,"374":2,"375":1,"376":2,"382":2,"388":2,"389":2,"396":1,"417":1,"434":1,"435":1,"437":1,"440":2,"441":1,"443":1,"444":1,"453":1,"455":1,"465":2,"501":2,"508":1,"510":1,"511":1,"525":1,"526":2,"530":1,"533":2,"534":1,"547":1,"564":1,"580":1,"604":1,"605":1,"637":2,"684":2,"699":3,"717":1,"730":1}}],["http2",{"2":{"707":1}}],["http",{"0":{"127":1,"132":1,"137":1},"2":{"75":1,"76":4,"77":1,"78":3,"121":6,"127":1,"128":4,"129":4,"135":2,"140":1,"141":1,"148":2,"225":1,"269":2,"302":1,"315":1,"342":2,"363":2,"541":1,"552":2,"557":2,"559":2,"614":1,"707":1}}],["https",{"0":{"225":1},"2":{"16":3,"74":2,"78":2,"141":2,"142":2,"145":2,"187":2,"188":2,"193":2,"196":2,"199":18,"200":2,"203":1,"223":1,"225":1,"228":1,"229":1,"245":2,"257":1,"267":2,"285":1,"348":2,"358":2,"369":2,"390":3,"391":4,"448":3,"449":3,"466":4,"467":2,"473":1,"474":3,"493":4,"508":4,"531":2,"539":2,"540":2,"567":2,"568":2,"585":10,"596":12,"614":1,"622":2,"623":8,"639":8,"640":2,"685":4,"696":9,"697":5,"703":4,"707":16,"711":1,"712":9}}],["html",{"2":{"16":1,"141":2,"142":2}}],["halo",{"2":{"477":1}}],["halting",{"2":{"394":1}}],["hamburger",{"2":{"391":1}}],["had",{"2":{"148":2,"314":1}}],["handpicking",{"2":{"503":1}}],["hand",{"2":{"211":1}}],["hands",{"0":{"125":1},"1":{"126":1,"127":1,"128":1,"129":1,"130":1,"131":1,"132":1,"133":1,"134":1,"135":1,"136":1,"137":1,"138":1}}],["handlemsg",{"2":{"607":2,"756":2}}],["handlecompleteproposal",{"2":{"607":2,"756":2}}],["handled",{"2":{"373":1}}],["handle",{"2":{"296":1,"382":4,"414":1}}],["handles",{"2":{"78":1,"204":1,"284":1,"315":1}}],["handling",{"2":{"97":1,"284":1}}],["happen",{"2":{"416":1,"715":1}}],["happens",{"2":{"94":1,"715":1}}],["happened",{"2":{"72":1}}],["happy",{"2":{"41":1,"57":1}}],["having",{"2":{"34":2,"35":1,"71":1,"155":1,"156":1,"168":1,"178":1,"196":1,"203":1,"396":2,"612":1}}],["haven",{"2":{"389":2,"569":1}}],["have",{"2":{"4":1,"8":1,"23":3,"26":1,"34":4,"35":1,"36":1,"37":1,"39":1,"40":2,"41":1,"57":1,"65":1,"66":1,"72":1,"76":1,"77":1,"78":2,"95":2,"101":1,"109":1,"111":1,"125":1,"139":4,"154":1,"168":2,"178":1,"187":4,"193":1,"194":1,"195":1,"196":1,"199":2,"200":1,"201":1,"212":1,"220":1,"231":1,"242":1,"246":1,"261":1,"262":1,"263":1,"267":2,"268":1,"282":2,"296":1,"302":1,"307":2,"309":1,"313":1,"315":1,"338":1,"344":1,"346":2,"348":1,"349":1,"352":1,"353":1,"360":1,"362":1,"364":1,"377":1,"394":1,"395":2,"398":1,"404":1,"410":1,"416":2,"432":1,"438":1,"445":1,"477":1,"487":1,"509":3,"517":1,"518":1,"526":1,"537":2,"540":1,"542":1,"543":1,"552":1,"568":1,"572":1,"608":1,"614":1,"617":1,"623":1,"639":2,"640":1,"642":1,"644":1,"645":1,"654":1,"659":1,"660":1,"663":1,"665":1,"673":1,"683":1,"693":1,"699":1,"736":1,"750":1,"751":1}}],["hash=580b3dff8a7c716968161d91116a1e171f486298d582874e93714e489c9e6e88",{"2":{"542":2}}],["hash=",{"2":{"542":2}}],["hash=0x79a0a7a1b4936aafe7a37dbfb07a6a9e55c145a4ed6fd54f962649b4b7db8de7",{"2":{"340":1}}],["hash=0x79a0a7a1b4936aafe7a37dbfb07a6a9e55c145a4ed6fd54f962649b4b7db8de7export",{"2":{"340":1}}],["hash=0xef9f50bfb39f11b022a6cd7026574eccdc6d596689bdccc7b2c482a1b26b26b8",{"2":{"132":1}}],["hashing",{"2":{"95":1}}],["hashes",{"2":{"95":2,"103":4,"212":1}}],["hash",{"0":{"131":1,"134":1,"184":1,"194":1,"564":1,"590":1},"1":{"132":1,"133":1,"135":1,"565":1},"2":{"78":6,"95":2,"103":10,"110":2,"113":2,"127":2,"129":2,"131":1,"132":5,"133":2,"134":1,"135":4,"137":2,"145":1,"157":1,"184":8,"194":3,"195":5,"328":24,"331":1,"340":4,"341":4,"384":2,"410":1,"542":6,"564":10,"565":1,"567":1,"579":2,"590":1,"595":3,"604":2,"745":3,"751":2}}],["hashtags",{"2":{"39":1}}],["has",{"2":{"27":1,"41":1,"44":1,"71":1,"76":1,"78":3,"81":1,"94":2,"95":1,"97":2,"103":3,"153":1,"154":1,"155":1,"182":1,"184":1,"204":1,"254":1,"270":1,"302":1,"314":1,"338":1,"352":2,"371":1,"374":2,"393":1,"403":1,"408":1,"410":1,"437":1,"438":1,"462":1,"472":1,"477":1,"503":1,"505":1,"523":1,"554":2,"596":1,"597":1,"607":1,"612":1,"632":1,"644":1,"670":1,"682":1,"683":1,"710":1,"724":1,"736":1,"740":1,"751":1,"756":1}}],["hard",{"2":{"683":2}}],["hardhat",{"2":{"78":1}}],["hardware",{"0":{"308":1,"478":1,"581":1,"626":1,"650":1,"666":1,"742":1},"2":{"26":1,"54":1,"55":1,"308":1,"478":1,"581":1,"626":1,"650":1,"666":1,"742":1}}],["harassing",{"2":{"10":1}}],["harassment",{"2":{"6":1,"7":1,"15":1}}],["harmfully",{"2":{"509":1}}],["harmful",{"2":{"8":1,"24":1}}],["hermes",{"0":{"638":1,"645":1},"1":{"639":1,"640":1,"641":1},"2":{"637":2,"638":9,"639":4,"640":4,"641":2,"642":1,"646":1}}],["here",{"2":{"0":1,"21":1,"46":1,"54":1,"55":1,"95":1,"96":1,"110":1,"111":1,"113":1,"115":1,"203":1,"234":1,"316":1,"319":2,"321":3,"325":1,"328":2,"331":1,"348":1,"356":1,"358":2,"382":4,"483":1,"484":1,"494":1,"503":2,"529":1,"545":1,"546":1,"547":1,"548":1,"595":1,"656":2,"657":1,"659":1,"684":1,"735":1,"751":1}}],["hence",{"2":{"374":1,"624":1}}],["helloworldblob",{"2":{"270":4,"381":4}}],["hello",{"2":{"270":4,"319":1,"364":4,"381":4}}],["helpful",{"0":{"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":{"539":1,"540":1}}],["helpcelestia",{"2":{"333":2,"492":1,"493":1,"529":1,"539":1,"647":1}}],["helper",{"2":{"117":1,"141":1,"144":1,"145":1}}],["helping",{"2":{"41":1,"254":1,"488":2,"649":1}}],["helps",{"2":{"2":1,"37":1,"40":1,"42":4,"71":1}}],["help",{"0":{"333":1},"2":{"0":1,"29":2,"30":1,"33":1,"34":1,"35":1,"36":3,"37":2,"38":1,"39":2,"40":2,"41":3,"42":2,"51":2,"107":1,"241":1,"249":2,"333":4,"357":1,"359":1,"455":1,"488":6,"492":1,"493":1,"529":2,"539":2,"548":1,"576":1,"647":2,"699":1,"730":1}}],["height=",{"2":{"194":1}}],["height=15",{"2":{"127":1,"137":1,"145":1}}],["heightsub",{"0":{"562":1},"2":{"562":4}}],["heights",{"2":{"193":1,"202":1,"716":1}}],["height",{"0":{"328":1},"2":{"94":2,"104":1,"110":8,"111":4,"113":2,"115":6,"124":3,"126":2,"127":1,"129":16,"132":2,"135":2,"136":1,"145":17,"146":8,"148":8,"153":1,"157":2,"168":2,"172":1,"193":1,"194":4,"195":2,"197":1,"218":1,"262":3,"270":8,"271":4,"272":11,"273":7,"319":5,"321":6,"322":3,"328":10,"329":2,"334":3,"336":1,"358":17,"360":4,"364":10,"365":13,"366":1,"381":6,"410":1,"441":1,"488":4,"522":4,"564":3,"565":2,"591":1,"594":1,"595":3,"597":9,"604":2,"607":1,"716":7,"745":2,"751":11,"756":2}}],["hexadecimal",{"2":{"319":1,"358":1}}],["hexbytes",{"2":{"143":2,"148":2}}],["hextoaddress",{"2":{"129":2,"146":2,"147":2,"148":2}}],["hex",{"2":{"76":1,"133":2,"135":4,"140":1,"141":1,"142":1,"187":2,"195":1,"319":8,"321":2,"356":2,"358":11,"360":2,"379":4,"380":4,"384":1}}],["health",{"2":{"501":1,"641":4,"693":1}}],["healthy",{"2":{"6":1,"641":1}}],["heavy",{"0":{"179":1,"180":1,"181":1},"1":{"180":1,"181":1},"2":{"181":1}}],["headerclient",{"2":{"364":2}}],["headerchan",{"2":{"272":2}}],["headerrangefunctionid",{"2":{"203":1}}],["header",{"0":{"328":1,"607":1,"756":1},"2":{"76":1,"94":2,"95":1,"103":2,"104":1,"105":1,"110":11,"111":9,"112":1,"113":18,"115":19,"135":2,"145":2,"159":1,"189":2,"193":5,"195":8,"196":10,"200":16,"202":1,"203":2,"212":2,"272":16,"273":7,"328":8,"329":2,"334":4,"359":2,"365":22,"366":13,"407":1,"410":1,"415":1,"416":4,"434":1,"562":2,"595":1,"607":2,"756":2}}],["headers",{"0":{"119":1,"272":1,"365":1,"562":1},"2":{"76":1,"82":1,"94":1,"109":1,"113":1,"115":5,"119":1,"124":1,"206":1,"272":6,"365":6,"407":1,"408":1,"409":1,"416":5,"477":2,"562":3,"564":2,"625":2,"665":3,"739":1,"740":1}}],["head",{"2":{"65":1,"107":1,"111":1,"238":1,"312":1,"366":2,"505":1,"528":1,"594":1,"622":2}}],["heard",{"2":{"238":1}}],["hear",{"2":{"42":2}}],["hesitate",{"2":{"34":1,"39":1}}],["hetzner",{"2":{"23":1}}],["vlog",{"2":{"563":4}}],["vcpu",{"2":{"369":1}}],["vghpcybpcybhbiblegftcgxlig9mihnvbwugymxvyibkyxrh",{"2":{"336":2}}],["vms",{"2":{"242":1}}],["vm",{"2":{"241":1,"242":4}}],["vs",{"0":{"208":1,"413":1},"1":{"209":1,"210":1,"211":1},"2":{"694":1}}],["vk",{"2":{"199":2}}],["vkrarg7fgtz",{"2":{"127":2}}],["v",{"2":{"89":2,"270":2,"272":2,"276":1,"341":2,"357":2,"358":8,"360":8,"381":2,"616":11,"617":10}}],["v1beta1",{"2":{"266":2,"494":4,"522":2,"523":2,"524":2}}],["v1",{"2":{"81":2,"125":8,"132":4,"137":1,"138":1,"148":2,"266":2,"326":6,"373":2,"377":1,"540":1,"597":1,"607":14,"638":1,"751":1,"756":14}}],["v2",{"2":{"81":1,"89":2,"135":2,"137":1,"597":10,"607":1,"716":3,"751":10,"756":2}}],["vulnerability",{"2":{"71":1}}],["v0",{"2":{"67":1,"81":3,"125":4,"314":1,"364":3,"365":2,"437":2,"554":1,"568":1,"607":14,"638":1,"716":1,"756":14}}],["void",{"2":{"519":2}}],["voice",{"2":{"34":1}}],["vote",{"2":{"417":1,"443":2,"444":1,"494":5,"495":1}}],["voting",{"2":{"328":4,"456":1,"494":1,"717":1}}],["volume",{"2":{"615":2,"616":6,"617":2,"620":1}}],["volumes",{"2":{"253":2}}],["volunteer",{"2":{"30":1}}],["volunteers",{"2":{"30":1,"32":1,"46":1}}],["vouch",{"2":{"209":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":{"277":1,"374":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}}],["vec",{"2":{"364":2}}],["venv",{"2":{"347":7}}],["venue",{"0":{"34":1},"2":{"30":2,"31":1,"34":4,"38":2,"39":2}}],["ver=",{"2":{"623":3}}],["very",{"2":{"109":1,"202":2,"282":1}}],["verifiable",{"2":{"212":1}}],["verified",{"2":{"122":1,"148":2,"153":1,"165":1,"168":1,"171":1,"175":1,"179":1,"187":1,"191":1,"393":1,"394":1,"395":1,"403":1}}],["verifiercd",{"2":{"188":1}}],["verifiers",{"0":{"188":1,"200":1},"2":{"187":2,"188":5,"189":4,"199":1}}],["verifier",{"0":{"189":1,"199":1},"1":{"190":1,"191":1,"192":1},"2":{"106":1,"107":1,"117":1,"148":2,"176":1,"179":1,"187":2,"188":7,"189":10,"190":4,"191":1,"195":6,"196":5,"198":1,"199":22,"200":13,"219":1,"369":4}}],["verifies",{"2":{"105":2,"107":3,"212":2,"219":1}}],["verification",{"0":{"210":1},"2":{"76":1,"94":1,"107":1,"124":1,"145":1,"172":2,"176":1,"179":1,"184":3,"187":1,"199":1,"374":1,"595":1}}],["verifydatarootinclusion",{"2":{"129":4}}],["verifymultirowrootstodataroottupleroot",{"2":{"107":1}}],["verifyrowroottodataroottupleroot",{"2":{"107":2}}],["verifysharestodataroottupleroot",{"2":{"107":2,"148":2}}],["verifying",{"0":{"107":1},"2":{"104":2,"111":2,"112":1,"124":1,"149":1,"153":2,"164":1,"277":1,"278":1,"395":1,"401":1,"409":1}}],["verifyzkp",{"2":{"103":4}}],["verifyattestation",{"2":{"97":1,"103":2,"105":1,"129":2}}],["verify",{"0":{"395":1,"641":1},"2":{"76":2,"102":1,"103":4,"107":4,"112":1,"121":1,"129":6,"139":2,"141":6,"144":4,"145":2,"148":10,"149":1,"157":1,"160":1,"163":1,"164":1,"165":1,"167":1,"168":1,"171":1,"172":2,"176":2,"179":3,"184":2,"187":1,"188":4,"189":2,"195":2,"198":1,"218":1,"241":2,"263":1,"393":5,"395":2,"396":1,"397":1,"398":1,"403":1,"406":1,"407":1,"409":1,"410":1,"413":1,"414":2,"519":2,"522":1,"567":1,"595":1,"623":1,"638":1,"642":1}}],["versed",{"2":{"58":1}}],["versioned",{"2":{"683":1}}],["versiongo",{"2":{"623":1}}],["versioncelestia",{"2":{"567":1}}],["versioncd",{"2":{"516":1}}],["versions",{"0":{"723":1,"726":1,"728":1},"2":{"76":1,"81":1,"125":2,"714":1}}],["version",{"0":{"332":1,"464":1,"516":1,"607":1,"619":1,"681":1,"701":1,"756":1},"2":{"16":2,"81":2,"102":1,"132":2,"135":6,"137":2,"141":11,"142":11,"148":6,"200":2,"222":1,"277":1,"319":1,"321":8,"328":2,"332":1,"336":2,"360":1,"369":1,"373":2,"412":1,"437":1,"438":1,"488":6,"512":1,"516":2,"526":1,"529":1,"539":1,"558":2,"567":6,"570":1,"597":1,"607":3,"623":4,"637":1,"638":1,"644":12,"751":1,"756":4}}],["ve",{"2":{"39":1,"58":1,"232":1,"238":1,"350":1,"358":2,"525":1,"533":1,"569":1}}],["vanilla",{"2":{"412":1}}],["vastly",{"2":{"241":1}}],["vault",{"2":{"187":6}}],["var",{"2":{"113":2,"515":4}}],["vary",{"2":{"35":1,"374":1,"441":1,"683":1}}],["variable",{"2":{"78":1,"140":1,"187":2,"196":3,"317":1,"331":1,"340":1,"341":1,"353":1,"359":3,"375":1,"435":1,"515":1,"745":1}}],["variablescd",{"2":{"195":1}}],["variables",{"0":{"521":1},"2":{"76":1,"187":1,"195":3,"196":1,"203":1,"259":1,"375":1,"521":1,"527":1,"616":1}}],["variety",{"2":{"35":1,"242":1}}],["varies",{"2":{"22":1}}],["various",{"2":{"30":1,"41":1,"58":1,"232":1}}],["val",{"2":{"745":2}}],["valconspub",{"2":{"389":2}}],["valcons",{"2":{"389":2}}],["valoperpub",{"2":{"389":2}}],["valoper",{"2":{"389":2,"496":2,"497":4}}],["validao",{"2":{"687":1,"688":1,"689":1}}],["validating",{"2":{"79":1,"90":1,"121":1,"714":1}}],["validation",{"0":{"90":1},"2":{"71":2,"115":2}}],["validatehermes",{"2":{"641":1}}],["validates",{"2":{"412":1,"488":2}}],["validated",{"2":{"26":1}}],["validate",{"2":{"26":1,"168":1,"408":1,"409":1,"477":1,"488":2,"594":1,"641":1,"725":1}}],["validator1",{"2":{"655":1}}],["validatorwalletcreator",{"2":{"85":1,"86":1}}],["validatorutils",{"2":{"85":1,"86":1}}],["validators",{"0":{"658":1},"2":{"18":1,"20":1,"21":1,"23":1,"25":1,"26":1,"28":1,"71":9,"72":1,"95":1,"109":1,"110":1,"193":1,"212":2,"240":1,"282":1,"328":6,"372":1,"398":1,"435":1,"440":1,"462":1,"494":3,"658":3,"659":1,"662":1,"699":1,"714":3,"715":2,"725":2,"727":1}}],["validator",{"0":{"87":1,"421":1,"426":1,"430":1,"496":1,"498":1,"530":1,"741":1,"743":1,"745":1,"751":1,"752":1},"1":{"88":1,"89":1,"90":1,"742":1,"743":1,"744":2,"745":2,"746":1,"747":1,"748":1,"749":1,"750":1,"751":1,"752":1,"753":1,"754":1,"755":1,"756":1},"2":{"18":2,"20":2,"22":2,"23":4,"26":7,"27":5,"71":2,"73":1,"78":1,"90":1,"95":1,"113":1,"240":1,"311":2,"328":4,"352":2,"402":1,"421":1,"426":1,"430":1,"440":3,"447":1,"455":2,"461":3,"467":4,"470":2,"477":1,"484":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,"611":1,"632":3,"656":1,"661":4,"662":2,"671":4,"672":2,"686":1,"699":1,"716":1,"717":1,"720":1,"727":1,"732":1,"733":1,"735":1,"741":2,"742":1,"745":7,"746":2,"751":16,"756":2}}],["validiums",{"2":{"394":1}}],["validity",{"2":{"171":1,"206":1,"241":3}}],["valid",{"2":{"107":2,"129":2,"153":2,"163":1,"164":1,"168":1,"172":2,"176":2,"179":2,"187":1,"195":1,"241":1,"407":1,"413":2,"416":1,"641":2}}],["valuable",{"2":{"34":1,"37":2,"42":1,"58":1}}],["valuelog",{"2":{"563":2}}],["values",{"0":{"542":1},"2":{"23":1,"72":1,"127":1,"132":1,"137":1,"141":1,"203":1,"246":1,"319":1,"443":1,"542":2,"560":3}}],["value",{"2":{"4":1,"38":1,"76":2,"132":8,"141":6,"142":1,"172":1,"189":1,"195":3,"319":2,"322":1,"326":1,"328":4,"341":2,"365":2,"373":1,"374":1,"375":1,"494":4,"547":1,"563":3,"592":2,"604":2,"605":1,"655":2,"656":1}}],["visualize",{"2":{"503":1}}],["visualizer",{"2":{"503":1}}],["visualization",{"0":{"503":1}}],["vision",{"2":{"454":1}}],["visited",{"2":{"389":2}}],["visit",{"2":{"66":1,"70":1,"71":1,"75":1,"77":1,"227":1,"228":1,"229":1,"230":1,"353":1,"612":1}}],["visibility",{"2":{"39":1}}],["visible",{"2":{"6":1}}],["virtual",{"2":{"223":1}}],["virtually",{"2":{"58":1}}],["vim",{"2":{"195":2,"196":2}}],["viewed",{"2":{"548":1}}],["viewing",{"2":{"494":1}}],["view",{"0":{"249":1},"2":{"62":1,"64":1,"77":1,"222":1,"297":1,"333":2,"384":2,"390":1,"391":1,"488":1,"495":1,"529":1,"540":1,"568":1}}],["viewpoints",{"2":{"7":1}}],["vibrant",{"2":{"49":1}}],["video",{"0":{"618":1},"1":{"619":1},"2":{"34":1,"40":1,"54":1,"55":1}}],["videos",{"2":{"31":1,"40":1}}],["violate",{"2":{"22":1}}],["violating",{"2":{"13":1,"14":1,"23":1,"24":1}}],["violation",{"2":{"11":1,"12":1,"13":1,"14":1,"15":1}}],["via",{"0":{"117":1,"118":1,"382":1},"2":{"9":1,"39":2,"110":1,"112":1,"115":1,"122":1,"124":1,"130":1,"141":1,"142":1,"145":1,"159":1,"160":1,"176":1,"188":1,"193":1,"194":1,"204":1,"245":4,"246":1,"382":2,"415":1,"417":1,"494":1,"537":1,"550":1,"607":1,"640":1,"646":1,"647":1,"727":1,"756":1}}],["vitepress",{"2":{"1":1}}],["egrep",{"2":{"596":6}}],["egg",{"2":{"519":2}}],["egeiggrjtkryaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa=",{"2":{"132":2}}],["eye",{"2":{"504":1}}],["eprintln",{"2":{"365":4}}],["ephemeral",{"2":{"241":1}}],["eq",{"2":{"364":6}}],["equally",{"2":{"414":1}}],["equal",{"2":{"270":4,"381":4}}],["equipments",{"2":{"34":1}}],["equipment",{"2":{"30":2,"31":1,"34":2,"40":1}}],["e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855",{"2":{"328":8}}],["eclipse",{"2":{"397":1}}],["ecfg",{"2":{"382":4}}],["echo",{"2":{"341":2,"585":10,"623":1,"662":2}}],["ec2",{"2":{"202":1}}],["ecosystem",{"2":{"7":1,"20":1,"26":3,"37":3,"42":1,"44":1,"47":1,"54":1,"55":1,"58":3,"206":1,"444":1,"445":2,"446":2,"503":1,"678":1}}],["economic",{"2":{"6":1,"23":1}}],["eofsudo",{"2":{"736":1,"738":1,"739":1,"740":1}}],["eof",{"2":{"557":2,"736":3,"738":3,"739":3,"740":3}}],["eoa",{"2":{"187":2}}],["eoujy0=",{"2":{"132":2}}],["ef9f50bfb39f11b022a6cd7026574eccdc6d596689bdccc7b2c482a1b26b26b8",{"2":{"132":3}}],["efficient",{"2":{"241":1,"376":1,"414":1}}],["efficiently",{"2":{"152":1,"393":1}}],["efficiency",{"2":{"206":1}}],["effect",{"2":{"408":1}}],["effects",{"2":{"148":2}}],["effectively",{"2":{"39":1,"51":1,"596":1}}],["effective",{"2":{"38":1,"42":1,"683":1}}],["efforts",{"2":{"41":1}}],["eight",{"2":{"410":1}}],["eip",{"2":{"373":1}}],["eip4844",{"2":{"94":1}}],["eiokkc9jzwxlc3rpys5ibg9ilnyxlk1zz1bheuzvckjsb2jzumvzcg9uc2u=",{"2":{"132":2}}],["either",{"2":{"58":1,"89":1,"154":1,"171":1,"188":2,"189":1,"196":2,"200":1,"245":1,"312":1,"315":1,"336":1,"373":1,"416":1,"446":1,"503":3,"537":1,"540":1,"568":1,"614":1}}],["era",{"2":{"454":1}}],["erasure",{"2":{"122":1,"280":1,"397":1,"398":3,"412":1,"415":1,"434":1,"477":2}}],["err=",{"2":{"607":2,"756":2}}],["err",{"2":{"113":18,"115":12,"121":10,"128":16,"129":78,"133":8,"135":24,"138":4,"146":34,"147":26,"148":30,"270":30,"271":18,"272":24,"273":12,"357":6,"358":42,"359":6,"360":30,"365":4,"381":30,"382":50,"607":2,"756":2}}],["errorcodes",{"2":{"148":2}}],["errorf",{"2":{"115":4,"129":2,"146":2,"358":6,"359":4}}],["error",{"0":{"563":1},"2":{"80":1,"111":10,"113":4,"115":4,"129":6,"135":2,"146":2,"148":4,"254":3,"270":2,"271":2,"272":4,"273":2,"338":3,"358":8,"359":4,"365":4,"381":2,"382":2,"554":1,"557":2,"559":2,"560":3,"563":6,"607":1,"659":1,"756":1}}],["errors",{"2":{"76":1,"346":1,"640":1}}],["erc20outbox",{"2":{"85":1,"86":1}}],["erc20rollupeventinbox",{"2":{"85":1,"86":1}}],["erc20inbox",{"2":{"85":1,"86":1}}],["erc20bridge",{"2":{"85":1,"86":1}}],["erc20tokenbridge",{"2":{"78":2}}],["erc20",{"2":{"78":2}}],["earn",{"2":{"440":1,"509":1}}],["early",{"2":{"206":1,"445":4,"446":2,"519":2}}],["easiest",{"2":{"612":2,"667":1}}],["easier",{"2":{"42":1}}],["easily",{"2":{"260":1,"283":1}}],["ease",{"0":{"240":1}}],["easy2stake",{"2":{"687":1,"688":1}}],["easy",{"2":{"41":1,"222":1,"240":1,"414":1,"454":1}}],["eager",{"2":{"42":1}}],["each",{"2":{"25":1,"27":2,"36":1,"46":1,"58":1,"76":1,"95":1,"104":1,"110":1,"112":1,"115":1,"145":2,"151":1,"156":1,"187":4,"199":1,"258":1,"277":1,"296":1,"314":1,"361":1,"371":1,"373":1,"374":3,"375":2,"376":2,"377":2,"407":1,"408":1,"414":1,"415":2,"416":1,"434":1,"435":1,"438":1,"441":1,"465":1,"509":1,"531":1,"554":2,"555":1,"640":2,"683":1,"684":1,"699":2,"717":2}}],["es",{"2":{"687":1,"688":1,"689":1}}],["estimates",{"2":{"375":1}}],["estimategas",{"2":{"375":2}}],["estimated",{"2":{"375":1,"382":2}}],["estimate",{"2":{"36":1,"78":8,"375":1,"376":1,"522":2}}],["estimating",{"0":{"374":1,"376":1},"2":{"36":2,"375":1,"382":2}}],["established",{"2":{"644":1}}],["establish",{"2":{"36":1,"75":2,"637":1}}],["especially",{"2":{"34":1,"44":1,"385":1}}],["essentially",{"2":{"548":1}}],["essential",{"2":{"34":1,"39":1,"41":1,"47":1,"73":1,"75":1,"224":1,"447":1,"453":1,"541":1,"552":1,"622":5}}],["eliminating",{"2":{"554":1}}],["eligibility",{"0":{"23":1},"2":{"20":1,"22":1}}],["else",{"2":{"129":2,"389":2,"640":1}}],["elastic",{"2":{"33":1}}],["eth0",{"2":{"602":2}}],["etherscan",{"2":{"187":6,"188":6,"193":1,"195":6}}],["ethereumclient",{"2":{"111":6,"113":4,"115":2}}],["ethereum",{"0":{"97":1,"119":1,"205":1,"254":1},"1":{"206":1,"207":1,"208":1,"209":1,"210":1,"211":1,"212":1,"213":1,"214":1,"215":1,"216":1},"2":{"33":1,"67":3,"69":1,"70":1,"76":1,"88":2,"92":1,"93":1,"96":2,"97":4,"105":1,"109":2,"110":4,"111":12,"113":3,"115":4,"119":3,"129":12,"148":12,"201":1,"206":3,"210":1,"211":2,"212":8,"213":2,"216":1,"223":1,"227":1,"254":6,"283":2,"284":7,"348":4,"373":1,"454":1}}],["ethcmn",{"2":{"129":2,"146":2,"147":2,"148":2}}],["ethclient",{"2":{"129":10,"146":8,"147":8,"148":8}}],["eths",{"2":{"78":2}}],["eth",{"0":{"69":1},"2":{"62":1,"67":2,"69":5,"71":1,"72":2,"75":1,"76":6,"78":7,"80":1,"348":1,"454":1}}],["ethnicity",{"2":{"6":1}}],["etc",{"2":{"30":2,"32":1,"42":1,"122":1,"159":1,"314":2,"470":1,"563":2,"693":1,"705":1,"736":4,"738":4,"739":4,"740":4}}],["evolution",{"2":{"412":1,"678":1}}],["evidence",{"2":{"277":1,"328":2}}],["evm",{"2":{"66":1,"92":1,"129":4,"146":6,"147":2,"148":4,"150":1,"187":1,"195":1,"196":2,"206":1,"216":1,"221":2,"223":2,"236":1,"242":1,"347":1,"348":3,"500":6}}],["evaluated",{"2":{"25":1}}],["evaluate",{"2":{"23":1,"32":1}}],["ever",{"2":{"519":2}}],["everything",{"2":{"57":1,"103":2,"171":1,"592":5,"641":1}}],["everyone",{"2":{"6":1,"35":1,"46":2,"653":1,"655":1,"658":1,"660":1,"699":1}}],["every",{"2":{"2":1,"21":1,"25":1,"28":1,"94":1,"196":1,"201":2,"213":2,"241":2,"307":1,"356":1,"371":1,"407":3,"408":1,"410":3,"413":1,"415":5,"416":1,"441":1,"503":4,"515":1,"527":1,"531":1,"594":1,"653":1}}],["even",{"2":{"34":2,"190":1,"385":1,"409":1,"413":1,"615":1,"715":1}}],["eventually",{"2":{"377":1,"416":1}}],["eventpayforblobs",{"2":{"132":2}}],["eventschan",{"2":{"147":6}}],["eventsiterator",{"2":{"129":8,"146":8}}],["events",{"2":{"36":6,"46":1,"48":1,"50":1,"58":1,"75":3,"76":2,"129":4,"132":4,"145":2,"146":6,"147":2,"148":4,"193":2,"522":2,"605":2}}],["event",{"2":{"9":1,"23":1,"30":5,"31":2,"32":3,"34":2,"35":2,"36":1,"37":5,"38":11,"39":9,"40":5,"46":2,"50":1,"76":3,"97":1,"129":12,"146":8,"147":8,"148":2,"193":1,"204":1,"254":2,"605":1,"639":4}}],["e",{"2":{"9":1,"22":1,"32":1,"42":2,"104":1,"129":12,"145":1,"146":12,"159":2,"160":1,"172":1,"191":1,"202":1,"252":4,"257":1,"284":1,"314":1,"365":8,"369":1,"371":1,"377":1,"396":1,"407":3,"408":1,"409":4,"410":4,"412":2,"413":6,"415":4,"416":9,"503":1,"551":1,"564":1,"585":10,"597":1,"602":2,"614":12,"616":12,"617":12,"665":1,"751":1}}],["ed25519",{"2":{"277":1}}],["eds",{"0":{"273":1,"366":1},"2":{"273":8,"366":10}}],["editing",{"2":{"563":1,"601":1,"641":1}}],["edits",{"2":{"8":1}}],["edit",{"2":{"8":1,"196":2,"531":1,"563":1,"639":2,"645":1}}],["education",{"2":{"6":1,"44":1}}],["embedded",{"2":{"477":1}}],["embark",{"2":{"350":1}}],["emvpw0p8nijmvnocp4bv6k+v6gjmwdxuku=",{"2":{"336":2}}],["emitted",{"2":{"204":1}}],["emits",{"2":{"129":2,"146":2,"148":2}}],["employed",{"2":{"509":1}}],["employees",{"2":{"41":1}}],["employ",{"2":{"393":1}}],["empty",{"2":{"195":1,"269":1,"357":2,"358":3,"360":2,"363":1,"576":1}}],["empowering",{"2":{"46":1}}],["empower",{"2":{"44":1}}],["emphasizing",{"2":{"34":1}}],["empathy",{"2":{"7":1}}],["emergency",{"2":{"23":1}}],["emails",{"0":{"48":1},"2":{"48":2}}],["email",{"2":{"7":1,"23":1,"26":1,"30":1,"39":2,"41":2,"47":1,"48":1,"57":1}}],["engine",{"2":{"613":1,"615":1}}],["engages",{"2":{"509":1}}],["engage",{"2":{"38":1,"39":1,"40":1,"44":1,"447":1}}],["engaged",{"2":{"36":1}}],["engagement",{"2":{"32":1,"37":1}}],["engaging",{"2":{"24":1,"35":1,"37":1,"41":1,"49":1,"57":1,"58":1}}],["enabling",{"2":{"256":1,"389":2,"414":1,"547":1,"727":1}}],["enabled=0",{"2":{"199":2}}],["enabled",{"2":{"187":2,"343":1,"502":1,"541":1,"552":1,"639":12,"647":1}}],["enables",{"2":{"97":1,"206":1,"227":1,"241":1,"242":1,"254":1,"406":1,"408":1,"412":2,"413":1,"414":1,"440":1,"544":1,"547":1,"567":1,"746":1}}],["enable",{"0":{"190":1,"588":1},"2":{"18":1,"76":4,"196":2,"283":1,"302":1,"343":1,"348":1,"389":2,"406":1,"409":1,"412":1,"544":1,"547":2,"587":1,"595":2,"637":1,"736":3,"738":3,"739":3,"740":3}}],["envcd",{"2":{"196":1}}],["env",{"2":{"187":4,"195":11,"196":7,"203":1,"369":1}}],["environments",{"2":{"466":1,"685":1,"703":1}}],["environment",{"0":{"621":1},"1":{"622":1,"623":1},"2":{"7":1,"46":1,"187":4,"195":3,"196":10,"223":1,"226":1,"243":1,"253":2,"286":1,"351":1,"353":1,"359":3,"369":1,"413":1,"539":1,"547":1,"567":1,"612":1,"621":2,"668":1,"678":1,"745":1}}],["enough",{"2":{"121":1,"212":1,"407":1}}],["encrypt",{"2":{"491":2}}],["encrypted",{"2":{"247":1,"491":5}}],["encapsulating",{"2":{"73":1}}],["encounter",{"2":{"338":1,"560":1,"563":1,"607":1,"659":1,"679":1,"756":1}}],["encouraging",{"2":{"44":1}}],["encouraged",{"2":{"311":1,"560":1,"699":1}}],["encourages",{"2":{"35":1,"38":1,"41":1,"71":1}}],["encourage",{"2":{"31":1,"37":1,"38":1,"39":1}}],["encode",{"2":{"140":2,"341":1,"382":2,"407":1}}],["encoded",{"2":{"110":1,"112":1,"122":1,"127":1,"132":1,"137":1,"154":2,"168":1,"175":1,"176":1,"179":1,"319":9,"321":2,"341":2,"342":1,"379":4,"380":4,"494":1}}],["encoding",{"2":{"110":1,"135":2,"156":1,"356":2,"382":6,"398":2,"407":2,"409":8,"412":1,"415":1}}],["encompasses",{"2":{"477":1}}],["encompass",{"2":{"41":1}}],["enjoy",{"2":{"46":1,"58":1,"78":2}}],["enjoyable",{"2":{"35":1}}],["enlightening",{"2":{"44":1}}],["endkey",{"2":{"141":3}}],["ending",{"2":{"141":2,"352":1,"413":1}}],["endshare=1",{"2":{"137":1}}],["endblock",{"2":{"129":6,"146":4}}],["end=20",{"2":{"127":1}}],["endpoints",{"0":{"466":1,"467":1,"468":1,"469":1,"470":1,"685":1,"687":1,"688":1,"689":1,"690":1,"692":1,"693":1,"694":1,"695":1,"703":1,"704":1,"705":1,"706":1,"707":1,"708":1,"709":1},"1":{"469":1,"470":1,"695":1},"2":{"224":1,"268":1,"279":1,"302":1,"307":1,"310":1,"362":1,"467":7,"469":1,"470":3,"529":1,"552":1,"595":2,"685":1,"687":1,"692":1,"693":3,"703":1,"704":1,"705":2,"707":2,"709":1,"732":1}}],["endpoint",{"0":{"311":1,"548":1,"600":1},"1":{"601":1,"602":1,"603":1},"2":{"76":3,"94":2,"121":4,"126":1,"127":1,"128":1,"129":4,"132":1,"134":1,"136":1,"137":3,"138":1,"145":1,"146":4,"148":4,"187":1,"195":1,"196":2,"311":5,"319":1,"336":1,"348":1,"465":1,"467":2,"470":1,"477":1,"483":1,"484":1,"503":4,"529":1,"544":6,"545":2,"546":2,"547":4,"548":1,"564":2,"595":1,"600":1,"602":1,"614":1,"632":3,"671":3,"684":1,"693":1,"695":1,"706":1,"707":1,"708":1}}],["end",{"2":{"38":1,"110":2,"126":1,"132":6,"136":1,"137":4,"141":3,"148":4,"157":1,"160":1,"201":2,"204":1,"410":1,"415":1,"522":2,"523":2,"639":4}}],["enhancement",{"2":{"554":1}}],["enhanced",{"2":{"461":1}}],["enhances",{"2":{"35":1,"38":1}}],["enhance",{"2":{"32":1,"416":1}}],["entails",{"2":{"409":1,"413":4,"414":1}}],["entering",{"2":{"745":1}}],["enter",{"2":{"248":1,"422":1,"426":1,"491":2,"503":1,"509":1,"519":2}}],["enters",{"2":{"23":1}}],["entrypoint",{"2":{"187":1,"195":1,"219":2,"252":2,"253":2}}],["entry",{"2":{"71":2,"226":1}}],["enthusiastic",{"2":{"43":1}}],["enthusiasts",{"2":{"37":1,"44":1}}],["entitle",{"2":{"471":1,"710":1}}],["entities",{"2":{"209":1,"403":1}}],["entity",{"2":{"26":2,"547":1,"564":2}}],["entirely",{"2":{"119":1}}],["entire",{"2":{"29":1,"95":1,"253":1,"397":1,"406":1,"416":1,"564":1,"580":1,"596":1,"694":1}}],["ensuring",{"2":{"18":1,"39":1,"58":1}}],["ensures",{"2":{"254":1,"393":1,"397":1,"413":1,"605":1}}],["ensure",{"2":{"2":1,"31":1,"36":1,"37":1,"39":1,"40":2,"41":1,"71":1,"73":1,"75":1,"211":1,"254":1,"338":1,"384":1,"413":1,"436":2,"563":1,"641":1,"660":1,"665":1,"679":1,"715":1}}],["enforced",{"2":{"373":1,"620":1}}],["enforce",{"2":{"46":1}}],["enforcement",{"0":{"8":1,"10":1,"11":1},"1":{"12":1,"13":1,"14":1,"15":1},"2":{"10":1,"16":1}}],["enforcing",{"2":{"8":1,"13":1,"14":1}}],["exit",{"2":{"128":6,"129":2,"148":2}}],["exist",{"2":{"186":2,"262":1,"437":1,"438":1,"448":1,"449":1,"485":1,"557":1,"633":1,"674":1}}],["exists",{"2":{"152":1,"155":1,"643":1}}],["existent",{"2":{"112":1}}],["existing",{"0":{"197":1,"253":1,"295":1},"2":{"22":1,"25":1,"36":1,"41":1,"54":1,"55":1,"76":1,"169":1,"189":1,"193":4,"195":1,"197":2,"202":1,"242":2,"252":1,"253":2,"278":1,"284":1,"346":1,"377":1,"503":1,"539":1,"560":1,"562":1,"567":1,"642":1,"644":1,"645":1,"660":1,"724":1}}],["execstart=$",{"2":{"736":2,"738":2,"739":2,"740":2}}],["exec",{"2":{"252":2,"253":2}}],["executable",{"2":{"199":2,"415":4,"434":2}}],["execution",{"2":{"115":1,"189":1,"221":1,"223":1,"238":2,"241":3,"242":3,"282":1,"284":1,"406":2,"412":1,"413":7,"414":1,"454":1}}],["executing",{"2":{"78":2,"413":1,"605":1}}],["executor",{"2":{"78":6}}],["executes",{"2":{"593":2}}],["executed",{"2":{"413":1,"415":1,"597":1,"716":1,"751":1}}],["execute",{"2":{"51":1,"57":1,"189":1,"238":1,"241":1,"412":1,"522":1,"555":1,"594":1,"622":1}}],["exercises",{"2":{"111":1}}],["exhaustive",{"2":{"94":2}}],["extends",{"2":{"415":1}}],["extendeddatasquare",{"2":{"273":2,"364":2,"366":2}}],["extendedheaders",{"2":{"665":1}}],["extendedheader",{"2":{"272":2}}],["extended",{"0":{"273":1,"366":1,"409":1},"2":{"159":1,"160":1,"273":1,"334":1,"365":4,"366":1,"407":4,"409":6,"415":4,"416":2,"434":1}}],["extension",{"0":{"420":1,"425":1},"2":{"389":2,"522":4}}],["external",{"0":{"602":1},"2":{"13":1,"92":1,"240":1,"302":1,"503":4,"602":3}}],["extradata",{"2":{"340":2}}],["extra",{"0":{"598":1},"1":{"599":1,"600":1,"601":1,"602":1,"603":1,"604":1,"605":1},"2":{"76":2,"80":1,"160":1,"375":1,"754":1}}],["exact",{"2":{"376":1,"441":1,"683":1}}],["exactly",{"2":{"25":1,"277":1,"503":2}}],["example",{"0":{"103":1,"129":1,"148":1,"167":1,"334":1},"2":{"59":1,"75":1,"78":1,"103":1,"127":2,"132":2,"137":2,"140":1,"141":1,"142":1,"144":1,"145":3,"148":5,"157":1,"165":1,"167":1,"187":8,"195":5,"196":3,"203":1,"238":1,"252":1,"253":1,"262":1,"263":1,"266":1,"271":1,"272":1,"289":1,"311":2,"314":1,"316":1,"318":2,"319":4,"321":3,"325":1,"328":1,"331":1,"334":1,"336":2,"348":1,"360":1,"365":1,"374":1,"410":2,"413":1,"416":1,"438":1,"470":2,"483":1,"484":1,"491":1,"493":2,"522":1,"529":1,"536":1,"541":1,"545":1,"546":1,"552":1,"555":1,"557":1,"558":1,"564":2,"585":1,"591":1,"604":1,"605":1,"615":1,"616":1,"642":1,"739":1,"745":1}}],["examples",{"0":{"323":1,"556":1},"1":{"324":1,"325":1,"326":1,"327":1,"328":1,"329":1,"330":1,"331":1,"332":1,"333":1,"557":1,"558":1,"559":1},"2":{"7":2,"9":1,"119":1,"322":1,"336":1,"381":1,"542":1,"547":1}}],["exceed",{"2":{"593":1}}],["exceptions",{"0":{"282":1}}],["exceptional",{"2":{"18":1,"37":1}}],["except",{"2":{"129":2,"385":1}}],["exclusively",{"2":{"461":1}}],["exclusive",{"2":{"47":1,"58":1,"132":1}}],["exchange",{"2":{"23":1,"72":1}}],["exposes",{"2":{"707":1}}],["expose",{"0":{"601":1}}],["exposed",{"2":{"111":1,"279":1,"300":1,"310":1,"484":1,"632":1,"671":1}}],["exporter",{"0":{"504":1},"2":{"504":1}}],["exported",{"2":{"491":2}}],["exporting",{"0":{"247":1,"491":1}}],["export",{"0":{"499":1},"2":{"78":1,"247":7,"252":1,"259":4,"266":4,"316":2,"317":1,"318":2,"336":1,"341":3,"389":10,"488":4,"491":3,"499":1,"503":1,"521":2,"527":2,"542":15,"623":4}}],["expand",{"2":{"40":1,"46":1}}],["express",{"2":{"38":1}}],["expression",{"2":{"6":1}}],["expensive",{"2":{"156":1,"168":1,"170":1,"173":1,"184":2}}],["expenses",{"2":{"38":1}}],["experiment",{"2":{"649":1}}],["experimentation",{"2":{"242":1}}],["experimentalsuggestchain",{"2":{"389":4}}],["experimental",{"2":{"206":1,"567":2,"679":1,"722":1}}],["experimenting",{"2":{"234":1}}],["experiences",{"2":{"7":1}}],["experience",{"2":{"6":2,"7":1,"26":1,"32":1,"35":1,"38":1,"46":1,"298":1,"344":1,"350":1,"722":1}}],["experts",{"2":{"37":2,"58":1}}],["expertise",{"2":{"37":1,"42":1,"58":1}}],["expects",{"2":{"544":1,"620":1}}],["expect",{"2":{"48":1,"58":1,"109":1,"253":1,"364":10,"365":6,"366":6}}],["expectations",{"2":{"36":1}}],["expected",{"2":{"23":1,"30":1,"115":2,"189":1,"438":2,"523":1,"557":1,"558":1,"559":1,"607":2,"756":2}}],["exploring",{"2":{"231":1,"608":1}}],["exploreme",{"2":{"696":1}}],["explored",{"2":{"182":1}}],["explorers",{"0":{"474":1,"696":1,"712":1},"2":{"275":1,"404":1,"474":1,"696":2,"712":3,"751":1}}],["explorer",{"0":{"77":1,"229":1},"2":{"64":1,"77":2,"229":2,"358":2,"384":2,"390":2,"391":1,"449":1,"474":1,"667":1,"696":3,"712":2,"745":1,"751":1}}],["explore",{"2":{"34":1,"50":1,"51":1,"286":1}}],["explains",{"2":{"288":1,"637":1}}],["explained",{"2":{"168":1,"173":1,"182":1}}],["explanation",{"2":{"12":1,"396":1}}],["explicit",{"2":{"7":1}}],["python3",{"2":{"347":2}}],["pwd",{"2":{"253":2}}],["psql",{"2":{"604":2}}],["ps",{"2":{"252":1,"253":1}}],["psdocker",{"2":{"252":1,"253":1}}],["p",{"2":{"200":2,"252":2,"341":2,"596":6}}],["pk",{"2":{"199":2}}],["pkg",{"2":{"129":2,"135":2,"148":2,"382":6,"607":14,"622":4,"756":14}}],["pfbs",{"2":{"154":1,"168":1,"170":1,"376":1}}],["pfb",{"0":{"134":1,"337":1,"374":1},"1":{"135":1},"2":{"78":1,"122":5,"124":1,"130":1,"134":1,"135":2,"154":4,"155":1,"156":1,"176":2,"321":3,"371":1,"374":4,"435":1,"683":2}}],["p2p",{"0":{"575":1,"585":1},"1":{"576":1,"577":1},"2":{"76":4,"111":1,"246":9,"247":6,"248":6,"252":4,"253":3,"261":2,"262":2,"289":2,"294":2,"310":4,"311":8,"312":8,"314":2,"316":4,"318":4,"326":2,"336":2,"348":4,"352":4,"412":1,"467":5,"470":2,"483":4,"484":6,"485":4,"488":2,"541":2,"542":4,"544":2,"546":2,"547":4,"550":4,"551":5,"552":2,"555":6,"560":4,"561":4,"562":4,"564":4,"614":12,"616":12,"617":6,"631":4,"632":2,"633":4,"670":4,"671":6,"672":8,"674":4,"709":1,"739":2}}],["pinnacle",{"2":{"678":1}}],["pinning",{"2":{"438":1}}],["pin",{"2":{"438":1}}],["pings",{"2":{"203":1}}],["pink",{"2":{"122":1}}],["pieces",{"2":{"110":1}}],["pick",{"0":{"70":1,"75":1},"2":{"246":1,"421":1,"536":1,"569":1,"585":1,"751":1}}],["pitches",{"2":{"38":1}}],["pitch",{"2":{"38":2}}],["pizza",{"2":{"35":1}}],["phase",{"2":{"579":1}}],["phrase",{"2":{"519":2,"640":1}}],["photos",{"2":{"31":1,"32":1}}],["physical",{"2":{"7":1}}],["plugin",{"2":{"505":1}}],["plus",{"2":{"69":1,"356":1}}],["plonky2x",{"2":{"199":7,"200":3,"218":1}}],["plonk",{"2":{"187":1,"188":1,"198":1,"199":3,"200":1}}],["plw1gxabnavhwwurqswb0xh25zv9xhielqtvld0xqc4=",{"2":{"132":2}}],["plasma",{"2":{"339":1,"344":1}}],["plastic",{"2":{"35":1}}],["plain",{"2":{"319":1,"321":1,"448":1}}],["plaintext",{"2":{"319":1}}],["play",{"2":{"456":1,"609":1,"663":1}}],["playbook",{"2":{"44":1}}],["playlist",{"2":{"40":1}}],["platform",{"0":{"41":1,"203":1},"2":{"49":1,"188":1,"196":2,"203":3,"204":1,"223":1,"612":1,"678":1}}],["platforms",{"2":{"30":1,"39":1,"40":1,"280":1}}],["planned",{"2":{"36":1,"39":1,"714":1}}],["planning",{"2":{"32":1,"42":1,"569":1,"733":1}}],["plans",{"2":{"35":1}}],["plan",{"2":{"30":1,"35":1,"36":1,"39":1,"40":1,"41":1,"51":1,"57":1,"733":1}}],["placement",{"2":{"22":2,"26":1}}],["place",{"2":{"22":10,"38":1,"232":1,"269":1,"363":1,"503":1,"519":2,"642":1,"716":1}}],["plethora",{"2":{"504":1}}],["pledge",{"0":{"6":1},"2":{"6":2}}],["please",{"2":{"2":1,"4":1,"26":4,"30":1,"47":2,"57":1,"76":1,"109":1,"119":1,"137":1,"138":1,"148":2,"206":1,"246":1,"389":2,"562":2,"572":1,"611":1,"612":1,"638":2,"640":1,"683":1,"716":1,"720":1,"752":1}}],["packet",{"2":{"639":8,"645":4}}],["packets",{"2":{"637":3,"638":1,"639":2}}],["packages",{"2":{"622":1,"730":1}}],["package",{"0":{"382":1},"2":{"89":1,"128":1,"129":1,"148":1,"356":1,"382":2,"489":1,"503":1}}],["panic",{"2":{"607":2,"756":2}}],["panel",{"2":{"432":1}}],["pagination",{"2":{"525":4}}],["page",{"2":{"26":1,"62":1,"72":1,"98":1,"145":1,"230":1,"232":1,"246":1,"252":2,"295":1,"311":1,"312":1,"339":1,"371":1,"388":1,"415":1,"420":1,"475":1,"483":3,"484":1,"503":2,"505":1,"509":1,"529":1,"537":1,"539":1,"540":1,"550":1,"612":2,"614":2,"621":1,"632":2,"663":1,"668":1,"671":1,"698":1,"699":2,"713":1,"729":1,"748":1,"754":1}}],["paired",{"2":{"503":1}}],["paid",{"2":{"72":1,"348":1,"438":1}}],["padded",{"2":{"415":1}}],["paper",{"2":{"407":1,"410":1,"416":1}}],["path=$path",{"2":{"623":4}}],["path",{"0":{"515":1},"2":{"188":2,"196":3,"200":2,"245":2,"315":1,"317":3,"318":5,"485":1,"486":1,"494":4,"495":2,"515":1,"573":1,"616":6,"617":4,"620":1,"633":1,"634":1,"674":1,"675":1}}],["pattern",{"2":{"15":1}}],["payee",{"2":{"639":2}}],["payer",{"2":{"522":2}}],["paying",{"0":{"433":1,"453":1},"1":{"434":1,"435":1},"2":{"454":1}}],["payload",{"2":{"358":4}}],["payment",{"2":{"258":1,"277":1,"434":2}}],["payments",{"2":{"71":1}}],["payforblob",{"2":{"129":6,"135":1,"146":2,"148":2,"484":1,"632":2,"672":1,"677":1}}],["payforblobs",{"0":{"434":1},"2":{"122":1,"254":1,"312":1,"338":1,"346":1,"375":1,"379":2,"415":3,"416":1,"434":3,"453":1,"671":1}}],["pay",{"2":{"72":1,"76":1,"258":2,"266":1,"312":1,"348":1,"375":1,"484":1,"632":1,"672":1}}],["paradigm",{"2":{"413":2}}],["parallel",{"2":{"242":1}}],["param",{"2":{"494":2}}],["paramfilter",{"2":{"494":1}}],["params=",{"2":{"389":6}}],["params",{"2":{"76":1,"277":1,"336":2,"342":3,"389":27,"494":3,"579":1}}],["parameterchangeproposal",{"2":{"494":2}}],["parameters",{"0":{"389":1,"443":1,"682":1},"1":{"683":1},"2":{"78":1,"139":4,"145":2,"189":1,"296":1,"314":1,"371":1,"374":2,"375":1,"388":1,"389":3,"390":1,"391":1,"416":1,"443":2,"456":1,"494":1,"509":1,"682":2}}],["parameter",{"2":{"71":2,"107":1,"132":1,"374":1,"375":1,"376":1,"494":3,"563":2,"683":2,"751":1}}],["parity",{"2":{"159":1,"407":1,"409":1}}],["parsing",{"2":{"154":1,"155":2,"164":1,"168":1,"179":1}}],["parsed",{"2":{"184":1,"195":1,"358":2}}],["parse",{"2":{"148":2,"154":2,"156":1,"168":1,"170":1,"184":1,"334":1,"358":2}}],["parenthash",{"2":{"340":2}}],["parent",{"2":{"71":1,"75":2,"78":18}}],["parts",{"2":{"328":4,"434":1,"456":1}}],["party",{"2":{"256":1,"258":1,"266":1,"275":1,"403":1,"596":1}}],["partitioned",{"2":{"410":1,"415":1}}],["partitions",{"2":{"307":1,"410":1}}],["particularly",{"2":{"280":1,"393":1}}],["particular",{"2":{"212":1}}],["participating",{"2":{"42":1,"46":1,"58":1,"465":1,"500":1,"655":1,"684":1,"699":1,"731":1}}],["participation",{"2":{"6":1,"18":1,"23":1,"32":1,"49":1,"71":1}}],["participates",{"2":{"717":1}}],["participate",{"0":{"721":1},"1":{"722":1,"723":1,"724":1,"725":1,"726":1,"727":1,"728":1,"729":1},"2":{"38":1,"47":1,"58":1,"71":1,"414":1,"467":1,"653":1,"684":1,"699":1,"717":1,"724":1,"725":1,"741":1,"751":1}}],["participant",{"2":{"23":1,"48":1}}],["participants",{"2":{"22":1,"40":1,"42":3,"46":1,"58":1,"658":1,"660":4,"661":2,"662":1}}],["parties",{"2":{"163":1,"403":1}}],["part",{"2":{"47":1,"122":1,"124":2,"125":1,"130":1,"148":6,"159":1,"160":5,"164":1,"168":3,"172":1,"212":1,"240":1,"410":2,"413":1,"446":1,"453":1,"454":1,"638":1,"682":1,"733":1,"746":1}}],["partnering",{"2":{"34":1,"38":1}}],["passphrase",{"2":{"491":2,"745":3}}],["password",{"2":{"491":3,"503":1,"519":2}}],["passed",{"2":{"322":1,"358":2,"485":1,"488":2,"632":1,"633":1,"671":1,"674":1}}],["passes",{"2":{"315":1,"415":1}}],["passing",{"0":{"291":1,"292":1}}],["passionate",{"2":{"58":1}}],["pass",{"2":{"107":1,"269":1,"315":1,"357":3,"358":3,"360":2,"363":1,"389":2,"494":1}}],["pasting",{"2":{"353":1}}],["paste",{"2":{"76":1,"109":1,"352":1}}],["past",{"2":{"38":1,"399":1,"400":3,"401":1,"403":1,"404":2,"522":1,"591":1,"654":1}}],["policy",{"2":{"639":4,"645":4}}],["political",{"2":{"7":1}}],["polypore",{"2":{"639":6}}],["popup",{"2":{"422":1}}],["populating",{"2":{"357":1,"602":1}}],["populates",{"2":{"469":1,"470":1,"692":1,"693":1,"704":1,"705":1}}],["populated",{"2":{"358":1}}],["populate",{"2":{"95":1}}],["popularity",{"2":{"36":1}}],["popular",{"2":{"36":1,"39":1}}],["pops",{"2":{"252":2,"253":2,"311":2,"391":2,"484":2,"493":4,"529":2,"531":2,"555":2,"639":2,"671":4,"672":2,"687":1,"688":1,"689":1,"693":1,"705":1,"707":1,"708":1}}],["point",{"2":{"226":1,"246":1,"377":1,"413":2,"503":2,"519":2,"564":1,"687":1,"688":1,"689":1,"706":1,"707":1,"708":1}}],["pointing",{"2":{"157":1}}],["pointer",{"2":{"94":1,"97":1,"110":2,"112":1,"157":1}}],["points",{"0":{"22":1},"2":{"292":1,"554":1}}],["portid",{"2":{"644":4}}],["portions",{"2":{"396":1,"397":1,"398":1}}],["portion",{"2":{"21":1,"122":1,"408":1,"446":1,"455":1,"526":1,"530":1}}],["ports",{"0":{"541":1,"552":1},"2":{"252":4,"253":4,"311":2,"467":3,"470":2,"483":2,"537":2,"541":7,"550":2,"552":7,"614":2,"632":2,"663":2,"671":2,"748":2}}],["portrays",{"2":{"167":1}}],["port",{"2":{"76":1,"311":5,"336":2,"469":1,"470":5,"483":5,"484":1,"503":3,"504":1,"531":2,"541":2,"552":4,"554":1,"559":2,"632":1,"639":4,"644":8,"661":4,"662":3,"671":1,"692":1,"693":20,"704":1,"705":5,"706":1,"707":1,"708":1}}],["portal",{"2":{"66":1,"70":3,"71":1}}],["poolcelestia",{"2":{"495":1}}],["pool",{"0":{"444":1,"495":1},"2":{"36":1,"444":2,"456":1,"495":9}}],["potentially",{"2":{"36":1,"40":1,"280":1,"374":1,"375":1}}],["potential",{"2":{"30":1,"36":1,"37":2,"38":2,"49":1,"76":1,"211":1}}],["powerful",{"2":{"501":1}}],["power",{"2":{"23":1,"328":4,"350":1,"415":1}}],["pos",{"0":{"411":1},"1":{"412":1},"2":{"412":2}}],["possibly",{"2":{"387":1}}],["possible",{"2":{"39":1,"66":1,"93":1,"115":2,"134":1,"167":1,"193":1,"220":1,"241":1,"260":1,"277":1,"338":1,"407":1,"409":1,"416":1,"436":1,"438":1,"562":1}}],["possibilities",{"2":{"286":1,"413":1}}],["positions",{"2":{"160":1}}],["position",{"2":{"94":1}}],["positiveerror",{"2":{"560":1}}],["positives",{"2":{"416":1}}],["positive",{"2":{"7":1,"46":1,"416":1,"560":3}}],["postgresql",{"2":{"604":1}}],["poster",{"0":{"75":1},"2":{"71":4,"73":1,"75":2,"76":7,"78":3,"94":3}}],["posted",{"2":{"70":3,"71":1,"76":1,"78":2,"94":2,"97":1,"103":1,"107":1,"109":1,"110":2,"111":1,"115":3,"119":1,"121":1,"122":1,"124":1,"139":4,"153":2,"165":1,"171":1,"172":1,"176":1,"179":1,"222":1,"270":1,"284":1,"306":1,"358":1,"364":1,"557":1}}],["post",{"0":{"337":1,"384":1},"2":{"34":1,"37":1,"39":1,"66":1,"71":1,"76":1,"94":2,"109":1,"113":2,"150":2,"163":1,"164":1,"213":1,"258":1,"270":2,"283":1,"336":4,"337":1,"342":2,"346":1,"348":1,"360":1,"381":2,"559":2}}],["posts",{"2":{"30":1,"32":1,"70":1,"284":3,"344":1}}],["postings",{"2":{"73":1}}],["posting",{"0":{"119":1},"2":{"9":1,"71":3,"72":1,"75":2,"76":1,"94":1,"109":1,"113":1,"254":2,"284":4,"361":1,"609":1}}],["peerexchange",{"2":{"577":1}}],["peers=$",{"2":{"585":6}}],["peers",{"0":{"577":1},"2":{"576":2,"579":3,"585":38,"595":1,"602":1,"661":3,"709":1}}],["peer",{"0":{"662":1},"2":{"385":1,"438":6,"579":3,"662":3,"752":1}}],["peerid",{"2":{"326":1}}],["pending",{"2":{"62":1,"64":1}}],["percentage",{"2":{"440":1,"495":1,"509":2}}],["persist",{"2":{"615":1}}],["persisted",{"2":{"605":1,"620":1}}],["persistent",{"0":{"615":1,"662":1},"1":{"616":1,"617":1},"2":{"585":13,"617":1,"661":5,"662":1}}],["perspective",{"2":{"416":1,"477":1}}],["person",{"2":{"40":2,"58":1}}],["personal",{"2":{"6":1,"7":1,"26":1,"37":1,"38":1}}],["per",{"2":{"37":1,"374":2,"375":2,"376":1,"382":2,"435":1,"441":1,"451":1,"472":2,"710":2,"716":1}}],["performs",{"2":{"314":1,"415":1}}],["performed",{"2":{"281":1,"408":1,"641":2,"668":1}}],["performing",{"2":{"224":1,"408":1,"416":2}}],["perform",{"2":{"115":2,"413":2,"416":1,"437":1,"664":1,"665":1}}],["performance",{"2":{"18":1,"20":1,"23":1,"32":1,"206":1,"242":1,"461":1,"501":1,"563":1,"567":1,"679":1,"708":1,"722":1}}],["perfect",{"2":{"34":1}}],["permanently",{"2":{"436":1,"509":1}}],["permanent",{"0":{"15":1},"2":{"13":1,"14":1,"15":1}}],["permissions",{"2":{"300":1,"615":1,"620":2}}],["permissioned",{"2":{"210":1}}],["permissionless",{"2":{"191":1,"406":1,"413":1,"455":1}}],["permissionlss",{"2":{"190":1}}],["permission",{"0":{"261":1,"264":1},"1":{"265":1},"2":{"7":1,"260":1}}],["periods",{"2":{"23":1}}],["period",{"2":{"13":1,"14":2,"22":1,"24":1,"71":2,"193":1,"509":3,"644":2}}],["people",{"2":{"7":1,"13":1,"14":1,"34":1,"36":1,"41":1,"42":1}}],["punish",{"2":{"509":1}}],["put",{"2":{"434":1}}],["pure",{"2":{"103":2}}],["purposes",{"2":{"111":1,"399":1,"620":1,"699":1}}],["purpose",{"2":{"46":1,"245":1,"284":1,"403":1,"436":1,"577":1}}],["purchased",{"2":{"7":1}}],["pub",{"2":{"328":4,"389":2,"523":2,"524":2}}],["pubkey=$",{"2":{"500":2,"751":2}}],["pubkeyed25519",{"2":{"328":4}}],["pubkey",{"2":{"253":4,"519":4,"520":8,"524":2}}],["pubs",{"2":{"36":1}}],["publish",{"2":{"405":1,"434":1,"502":1}}],["published",{"2":{"94":1,"106":1,"124":1,"129":2,"149":1,"210":1,"237":1,"238":1,"393":2,"436":2,"438":1}}],["publishing",{"2":{"7":1,"503":3}}],["publicnode",{"2":{"348":2}}],["publicly",{"2":{"58":1,"601":1}}],["public",{"2":{"3":1,"7":1,"9":1,"12":1,"14":2,"15":1,"37":1,"66":1,"72":1,"103":6,"148":2,"203":2,"302":1,"348":3,"401":1,"437":1,"438":2,"445":1,"446":1,"471":1,"537":2,"600":1,"601":1,"602":1,"658":1,"687":1,"688":1,"689":1,"693":1,"705":1,"706":1,"708":1,"710":1,"751":1}}],["pulled",{"2":{"360":1}}],["pull",{"2":{"2":2,"245":1,"560":1,"660":1,"720":1}}],["push",{"2":{"2":2,"103":2,"639":4}}],["prune",{"2":{"437":1,"592":2,"694":1}}],["pruned",{"2":{"23":1,"27":1,"387":1,"437":1}}],["pruning",{"0":{"436":1,"437":1,"586":1},"1":{"437":1,"438":1,"587":1,"588":1,"589":1,"590":1,"591":1,"592":1},"2":{"436":1,"437":4,"470":1,"591":2,"592":1,"693":1,"694":1}}],["pragma",{"2":{"103":2,"148":1}}],["practices",{"0":{"438":1},"2":{"54":1}}],["precise",{"2":{"683":1}}],["precedence",{"2":{"554":1}}],["precompiles",{"2":{"242":1}}],["preserved",{"2":{"560":1}}],["presence",{"2":{"385":1,"554":1}}],["present",{"2":{"210":1,"550":1,"560":1}}],["presenters",{"2":{"37":1}}],["presenting",{"2":{"37":1}}],["presentation",{"0":{"53":1,"54":1,"55":1,"56":1},"2":{"30":1,"37":1,"38":1,"53":3,"54":3,"55":3,"56":2}}],["presentations",{"2":{"30":1,"31":1,"37":1,"54":1,"55":1,"58":1}}],["presumes",{"2":{"539":1}}],["press",{"2":{"431":1}}],["predetermined",{"2":{"396":1}}],["prefix",{"2":{"341":2,"390":1,"391":1,"639":12}}],["prefixed",{"2":{"319":1,"415":1}}],["preferences",{"2":{"540":1,"568":1}}],["preferred",{"0":{"418":1}}],["prefer",{"2":{"311":1,"483":1}}],["prevent",{"2":{"530":1,"531":1,"648":1}}],["preventing",{"2":{"254":1}}],["previously",{"2":{"284":1,"314":1,"537":1,"751":1}}],["previoushash",{"2":{"110":2,"113":2}}],["previous",{"0":{"217":1},"1":{"218":1,"219":1,"220":1},"2":{"32":1,"72":1,"97":1,"110":2,"189":2,"198":1,"200":3,"201":1,"212":1,"218":1,"399":1,"425":1,"522":1,"526":1,"530":1,"554":1,"562":1,"659":1,"733":1}}],["preimage",{"0":{"95":1},"2":{"93":1,"95":3}}],["pre",{"0":{"257":1,"540":1,"568":1,"572":1},"2":{"71":1,"189":1,"339":1,"538":1,"540":3,"568":3}}],["prerequisites",{"0":{"67":1,"88":1,"101":1,"121":1,"251":1,"535":1,"613":1},"2":{"243":1}}],["prerequisite",{"2":{"47":1,"243":1,"507":2}}],["preparing",{"2":{"31":1}}],["prepared",{"2":{"607":1,"756":1}}],["prepare",{"2":{"30":2,"39":1,"714":1}}],["pro",{"2":{"696":1}}],["probabilistically",{"2":{"416":1}}],["probability",{"2":{"407":1}}],["probably",{"2":{"238":1}}],["problems",{"2":{"400":1,"402":1}}],["problem",{"0":{"394":1,"400":1},"2":{"394":2,"403":2,"406":1,"413":1,"414":1}}],["problematic",{"2":{"393":1}}],["proj",{"2":{"353":2}}],["project",{"0":{"269":1,"355":1,"363":1},"2":{"148":2,"154":1,"269":1,"355":2,"363":1,"495":2}}],["projector",{"2":{"34":1}}],["projectors",{"2":{"30":1,"34":1}}],["projects",{"2":{"23":1,"38":1,"298":1}}],["properly",{"2":{"579":1}}],["property",{"2":{"416":1,"451":1}}],["properties",{"2":{"241":1,"477":1}}],["proposal",{"2":{"444":1,"494":14,"495":5}}],["proposalscelestia",{"2":{"494":1}}],["proposals",{"2":{"417":1,"443":2,"456":1,"494":6,"495":3}}],["proposing",{"2":{"415":1}}],["propose",{"2":{"443":1,"509":1}}],["proposed",{"2":{"394":1}}],["proposer",{"2":{"284":1,"328":8,"348":4,"494":4}}],["proxy",{"2":{"203":1}}],["pros",{"0":{"155":1,"161":1,"165":1,"169":1,"173":1,"177":1,"180":1},"2":{"151":1,"155":1}}],["prospective",{"2":{"20":1}}],["products",{"2":{"462":1}}],["product",{"2":{"374":1,"375":2}}],["production",{"0":{"466":1,"685":1,"703":1},"2":{"71":2,"240":1,"311":2,"385":2,"466":1,"467":2,"477":1,"529":2,"585":1,"678":1,"685":3,"687":2,"703":3,"722":1}}],["producers",{"2":{"409":2,"416":1}}],["producer",{"2":{"394":2,"415":4}}],["produced",{"2":{"271":1,"272":1,"365":1}}],["produceblock",{"2":{"113":2}}],["producing",{"2":{"111":2,"505":1,"717":1}}],["provably",{"2":{"436":1}}],["provision",{"2":{"695":1}}],["provisions",{"2":{"441":1}}],["proving",{"0":{"117":1,"129":1,"159":1,"160":1,"202":1},"2":{"109":1,"130":1,"149":1,"150":1,"152":1,"154":2,"160":2,"164":4,"196":1,"202":2,"203":1,"220":1,"369":4,"393":1}}],["provides",{"2":{"41":1,"107":1,"223":1,"232":1,"256":1,"298":1,"300":1,"311":1,"368":1,"384":1,"406":1,"410":1,"412":1,"413":1,"414":1,"438":1,"483":1,"501":1,"632":1,"683":1}}],["provided",{"2":{"33":1,"37":1,"44":1,"66":1,"78":1,"142":1,"159":1,"168":1,"176":2,"187":2,"244":1,"277":1,"410":3,"576":1,"579":1,"595":2,"715":1}}],["providers",{"0":{"275":1},"1":{"276":1,"277":1,"278":1,"279":1,"280":1,"281":1,"282":1},"2":{"34":1,"240":1,"275":1,"302":1,"438":2,"466":1,"467":1,"685":2,"687":1,"703":2}}],["provider",{"0":{"94":1,"277":1},"1":{"278":1,"279":1,"280":1,"281":1,"282":1},"2":{"26":1,"34":1,"75":1,"93":1,"94":1,"193":1,"235":1,"344":1,"385":1,"466":1,"596":1,"685":1,"703":1}}],["provide",{"0":{"405":1},"2":{"18":1,"26":1,"31":1,"35":1,"40":1,"41":1,"89":1,"96":1,"124":1,"125":1,"126":1,"130":1,"159":1,"160":2,"319":1,"398":1,"403":2,"404":2,"406":1,"410":2,"438":1,"469":1,"470":2,"501":1,"522":1,"576":1,"597":1,"659":1,"692":1,"693":2,"697":1,"704":1,"705":2,"711":1,"720":1,"732":1,"751":2}}],["providing",{"0":{"412":1},"2":{"12":1,"35":1,"44":1,"49":1,"97":1,"184":1,"221":1,"242":1,"300":1,"336":1,"461":1}}],["provers",{"2":{"190":1,"191":1,"369":1}}],["prover=network",{"2":{"369":1}}],["prover=local",{"2":{"369":1}}],["prover=",{"2":{"187":2}}],["prover",{"0":{"190":1,"192":1,"196":1},"2":{"187":7,"190":1,"191":2,"192":3,"193":2,"196":7,"197":1,"213":1,"369":1}}],["proven",{"2":{"165":1,"172":1,"201":1,"218":1}}],["proveshars`",{"2":{"148":2}}],["provesharesv2`",{"2":{"148":2}}],["provesharesv2",{"2":{"138":1}}],["proveshares",{"2":{"134":1,"136":1,"138":3,"148":2,"160":1}}],["proves",{"2":{"141":2,"206":1}}],["prove=true",{"2":{"132":1}}],["proved",{"2":{"112":1}}],["prove",{"2":{"106":1,"110":3,"113":1,"115":1,"122":1,"124":5,"126":1,"130":1,"132":1,"137":3,"139":2,"142":1,"144":1,"149":1,"150":1,"153":5,"159":2,"160":6,"163":1,"168":1,"173":1,"176":1,"196":3,"212":1,"238":1,"410":1,"415":1}}],["proofnonce",{"2":{"129":4,"146":2,"148":2}}],["proof",{"0":{"106":1,"122":1,"126":1,"130":1,"131":1,"134":1,"136":1,"146":1,"153":1,"158":1,"197":1,"440":1,"455":1},"1":{"123":1,"124":1,"127":1,"128":1,"132":1,"133":1,"135":1,"137":1,"138":1},"2":{"95":1,"103":12,"104":2,"105":1,"106":1,"107":5,"109":1,"112":5,"115":2,"117":1,"124":2,"125":1,"126":4,"127":4,"128":2,"129":14,"130":3,"131":1,"132":9,"133":2,"134":1,"136":5,"137":3,"139":9,"140":1,"141":12,"143":1,"144":14,"145":11,"148":40,"149":2,"153":6,"154":1,"155":1,"158":1,"159":6,"160":11,"164":1,"167":1,"168":3,"172":3,"173":2,"176":3,"179":5,"182":1,"184":3,"187":4,"190":1,"191":1,"193":2,"195":2,"196":1,"199":3,"201":1,"202":1,"203":2,"204":3,"209":1,"212":4,"241":3,"398":2,"407":1,"410":1,"416":1,"417":1,"440":3,"447":1,"455":2,"477":1,"509":1}}],["proofs",{"0":{"107":1,"120":1,"139":1,"154":1,"179":1,"180":1,"181":1,"203":1,"204":1,"409":1},"1":{"121":1,"122":1,"123":1,"124":1,"125":1,"126":1,"127":1,"128":1,"129":1,"130":1,"131":1,"132":1,"133":1,"134":1,"135":1,"136":1,"137":1,"138":1,"139":1,"140":2,"141":2,"142":2,"143":2,"144":2,"145":2,"146":2,"147":2,"148":2,"149":1,"180":1,"181":1},"2":{"95":1,"106":1,"107":2,"112":1,"121":1,"122":1,"124":1,"125":1,"130":1,"132":6,"136":4,"137":4,"139":4,"141":8,"144":10,"147":1,"148":15,"149":1,"153":9,"154":5,"156":1,"159":1,"160":5,"161":1,"163":1,"164":3,"165":1,"167":2,"168":4,"170":1,"171":1,"175":1,"176":1,"179":3,"181":1,"187":1,"188":1,"190":1,"191":3,"193":1,"196":6,"197":1,"198":2,"200":3,"201":1,"202":2,"206":1,"212":2,"214":1,"218":2,"219":1,"220":1,"241":1,"397":1,"398":1,"407":1,"409":2,"410":1,"413":1,"416":1,"625":2,"637":1}}],["procedure",{"2":{"224":1,"313":1}}],["proceeding",{"2":{"72":1,"115":1,"638":1}}],["proceed",{"2":{"64":1,"71":2,"149":1,"176":1,"179":1,"243":1,"431":1,"658":1,"733":1}}],["processes",{"2":{"359":2,"386":1,"404":1,"477":1,"637":1,"734":1,"736":1}}],["processed",{"2":{"237":1,"385":1}}],["processing",{"2":{"242":1}}],["process",{"0":{"19":1,"21":1,"28":1,"714":1,"715":1,"734":1},"1":{"20":1,"21":1,"22":1,"23":1,"24":1,"715":1,"716":2,"735":1,"736":1,"737":1,"738":1,"739":1,"740":1},"2":{"21":1,"30":1,"41":1,"66":1,"71":1,"97":1,"112":1,"115":3,"147":2,"171":1,"172":1,"212":1,"227":1,"237":1,"243":1,"258":1,"284":1,"296":1,"350":1,"360":2,"398":1,"413":1,"475":1,"477":3,"487":1,"548":1,"555":1,"597":1,"635":1,"676":1,"682":1,"698":1,"713":1,"714":1,"715":1,"729":1,"735":1,"740":1,"750":1}}],["proto",{"2":{"148":2,"382":2}}],["protobuf",{"0":{"176":1},"2":{"125":4,"154":2,"155":1,"156":1,"168":2,"175":1,"176":2,"178":1,"179":2}}],["protocol",{"2":{"23":1,"37":1,"92":1,"111":1,"212":1,"218":1,"269":1,"363":1,"373":1,"440":1,"445":1,"541":1,"547":1,"552":1,"637":1}}],["protection",{"2":{"300":1}}],["protect",{"2":{"24":1,"76":1,"300":1}}],["prohibited",{"2":{"23":1,"554":1}}],["prometheuslistenaddr",{"2":{"502":2}}],["prometheus",{"2":{"502":9,"503":17,"504":1,"548":1,"638":1}}],["prompts",{"2":{"361":1}}],["promptanswer",{"2":{"360":2}}],["prompting",{"0":{"359":1},"2":{"360":1}}],["prompt",{"0":{"350":1,"354":1},"1":{"351":1,"352":1,"353":1,"354":1,"355":2,"356":2,"357":2,"358":2,"359":2,"360":2,"361":1},"2":{"72":1,"303":1,"350":1,"355":2,"356":2,"357":8,"358":13,"359":2,"360":14}}],["prompted",{"2":{"70":1,"491":2,"751":1}}],["promptly",{"2":{"10":1,"94":1}}],["prominent",{"2":{"58":1}}],["promote",{"2":{"36":1,"39":2,"40":3,"58":1,"242":1}}],["promotions",{"2":{"31":1}}],["promotional",{"2":{"30":1,"31":1}}],["promoting",{"2":{"18":1}}],["profileecho",{"2":{"623":1}}],["profile",{"2":{"623":3}}],["proficient",{"2":{"18":1}}],["professionals",{"2":{"40":1}}],["professional",{"2":{"7":1,"438":2}}],["programs",{"2":{"445":1}}],["programmatic",{"2":{"299":1}}],["programmatically",{"0":{"376":1},"2":{"119":1,"382":2}}],["programming",{"2":{"149":1}}],["program",{"0":{"17":1,"18":1,"20":1,"43":1,"44":1},"1":{"18":1,"19":1,"20":1,"21":1,"22":1,"23":1,"24":1,"25":1,"26":1,"27":1,"28":1,"44":1,"45":1,"46":1,"47":1,"48":1,"49":1,"50":1},"2":{"18":1,"20":2,"22":2,"23":4,"25":1,"26":3,"27":1,"28":2,"33":1,"36":1,"37":1,"44":1,"47":2,"48":1,"50":1,"56":3,"58":3,"213":1,"356":4,"357":5,"358":6,"360":2}}],["priv",{"2":{"756":1}}],["privacy",{"2":{"10":1,"58":1}}],["privatekey",{"2":{"348":10}}],["privatecelestia",{"2":{"318":1}}],["private",{"0":{"318":1},"2":{"7":2,"12":1,"14":1,"26":1,"34":1,"71":2,"73":1,"78":2,"103":4,"187":10,"188":4,"189":8,"195":4,"196":2,"242":1,"247":1,"318":7,"488":2,"491":4,"507":2,"517":1,"522":6,"554":1,"562":4}}],["primitive",{"2":{"395":1,"414":1}}],["primitives",{"2":{"393":1}}],["primarily",{"2":{"71":2,"298":1}}],["primary",{"2":{"18":1,"49":1,"97":1}}],["pricing",{"2":{"211":1}}],["priced",{"2":{"372":1}}],["price",{"0":{"322":1},"2":{"76":3,"270":3,"322":10,"358":2,"364":3,"371":1,"373":4,"375":3,"382":2,"435":1,"639":8}}],["prices",{"2":{"7":1,"372":1}}],["printf",{"2":{"270":4,"271":2,"272":4,"358":6,"360":4,"381":4}}],["printed",{"2":{"187":1,"195":1}}],["println",{"2":{"128":8,"129":6,"147":2,"148":2,"364":2,"365":2,"382":6}}],["print",{"2":{"30":1,"488":2}}],["prioritised",{"2":{"435":2}}],["prioritized",{"2":{"372":2}}],["priority",{"2":{"328":4}}],["prior",{"2":{"25":1,"30":1,"39":1,"507":2,"714":1,"715":1,"716":3}}],["pr",{"2":{"2":1}}],["ncdu",{"2":{"622":4}}],["ncreated",{"2":{"607":2,"756":2}}],["ntfs",{"2":{"620":1}}],["npanic",{"2":{"607":2,"756":2}}],["np5wre24w+e79",{"2":{"132":2}}],["ngithub",{"2":{"607":12,"756":12}}],["nruntime",{"2":{"607":2,"756":2}}],["nw",{"2":{"508":1}}],["n11",{"2":{"410":2}}],["n12",{"2":{"410":2}}],["n14",{"2":{"410":1}}],["n7",{"2":{"410":1}}],["n8",{"2":{"410":1}}],["n2",{"2":{"408":1,"409":1,"410":1}}],["nft",{"2":{"399":1}}],["nfts",{"2":{"37":1}}],["ns",{"2":{"358":2,"382":6}}],["nsoy0rgl7hqt4vwlg441gqkjsz2fbunzxipgns8ov",{"2":{"253":2}}],["n",{"2":{"270":4,"271":2,"272":4,"358":6,"360":4,"381":4,"400":1,"408":1,"409":2,"499":2,"522":2,"541":1,"552":1,"564":2,"585":12,"607":20,"751":2,"756":20}}],["nyzlbfjjnskofrzur8xvjijla+wbptwm0kbyglilxlg=",{"2":{"127":2}}],["nmbwwwhpiphwagai7maqm",{"2":{"127":2}}],["nmts",{"0":{"410":1},"2":{"307":1,"406":2,"410":2,"415":1}}],["nmtproof",{"2":{"141":2,"148":2}}],["nmt",{"2":{"124":2,"136":2,"139":2,"364":2,"410":2}}],["nidstring",{"2":{"358":4}}],["nil",{"2":{"113":10,"115":6,"121":6,"128":6,"129":36,"133":4,"135":8,"138":2,"146":14,"147":12,"148":12,"270":12,"271":8,"272":10,"273":8,"357":2,"358":22,"359":4,"360":10,"381":12,"382":16}}],["nitro",{"0":{"97":1},"2":{"77":1,"81":4,"89":4,"92":1,"93":1,"94":3,"95":2,"97":2,"254":1}}],["nice",{"2":{"34":1}}],["null",{"2":{"522":4,"523":2,"525":4,"604":1,"736":2,"738":2,"739":2,"740":2,"745":2,"751":2}}],["numerator",{"2":{"639":4}}],["numerous",{"2":{"394":1,"413":1}}],["num",{"2":{"639":4}}],["numia",{"2":{"466":4,"685":4,"687":1,"688":1,"689":1,"693":1,"703":4,"705":1,"706":1,"708":1}}],["numleaves",{"2":{"144":3,"160":2}}],["number=26656",{"2":{"662":1}}],["number=26656ip",{"2":{"662":1}}],["numbers",{"0":{"464":1,"681":1,"701":1}}],["number",{"2":{"30":1,"36":1,"71":2,"78":8,"144":2,"150":1,"157":1,"159":2,"195":1,"206":1,"241":1,"340":2,"374":3,"375":1,"377":1,"382":2,"397":1,"408":1,"414":2,"416":2,"502":4,"504":1,"523":2,"524":2,"563":1,"662":2,"683":1,"715":2}}],["nutshell",{"2":{"124":1}}],["nurturing",{"2":{"47":1}}],["necessitate",{"2":{"548":1}}],["necessity",{"2":{"547":1}}],["necessary",{"0":{"398":1},"2":{"24":1,"30":1,"34":3,"73":1,"75":1,"94":2,"95":1,"139":4,"145":2,"159":1,"168":1,"195":3,"210":1,"241":1,"315":1,"385":1,"397":1,"412":1,"416":1,"602":1,"622":1,"639":1,"751":2}}],["neither",{"2":{"412":1,"557":1}}],["net",{"2":{"311":1,"519":2,"672":1,"687":2,"688":2,"689":2,"690":1,"693":2,"696":1,"697":1,"705":1,"706":1,"707":1,"708":1,"712":1}}],["netcelestia",{"2":{"311":1,"672":1}}],["network=$network",{"2":{"614":6,"616":6,"617":6}}],["network=celestia",{"2":{"614":1}}],["network=celestiaexport",{"2":{"614":1}}],["network=custom",{"2":{"542":2}}],["network=",{"2":{"542":2}}],["network=arabicaexport",{"2":{"614":1}}],["network=arabica",{"2":{"289":2,"614":1}}],["network=mochaexport",{"2":{"614":1}}],["network=mochacelestia",{"2":{"262":1}}],["network=mocha",{"2":{"252":2,"262":1,"614":1}}],["networking",{"2":{"31":1,"35":1,"42":2}}],["networks",{"0":{"542":1,"585":1,"721":1},"1":{"722":1,"723":1,"724":1,"725":1,"726":1,"727":1,"728":1,"729":1},"2":{"26":1,"71":1,"111":1,"388":1,"406":1,"455":1,"500":1,"550":1,"551":1,"554":1,"585":9,"595":1,"637":1,"642":1,"644":1,"662":1,"665":1,"673":1,"693":1,"699":1,"714":1,"717":1,"733":1,"751":1,"752":1}}],["network",{"0":{"318":1,"389":1,"421":1,"443":1,"462":1,"463":1,"475":1,"500":1,"550":1,"599":1,"649":1,"663":1,"679":1,"680":1,"682":1,"698":1,"700":1,"713":1,"714":1,"716":1,"729":1},"1":{"464":1,"551":1,"650":1,"651":1,"652":1,"653":1,"654":1,"655":1,"656":1,"657":1,"658":1,"659":1,"660":1,"661":1,"662":1,"663":1,"683":1,"715":1,"716":1},"2":{"18":2,"24":1,"37":1,"38":1,"66":1,"70":2,"71":2,"75":3,"76":4,"78":6,"89":2,"111":1,"115":2,"125":2,"139":2,"212":1,"216":1,"238":1,"241":1,"246":20,"247":12,"248":12,"252":3,"253":3,"261":2,"270":4,"275":1,"283":1,"284":1,"294":2,"302":1,"306":1,"310":4,"311":8,"312":13,"316":8,"318":8,"319":2,"336":3,"348":1,"352":4,"356":2,"357":2,"358":10,"360":2,"364":4,"369":1,"381":4,"385":2,"388":1,"389":4,"393":1,"402":1,"407":1,"408":1,"412":1,"413":3,"414":2,"416":4,"417":1,"421":1,"425":1,"431":1,"437":2,"438":3,"440":1,"443":1,"447":3,"453":2,"455":2,"456":1,"462":1,"465":1,"469":1,"470":3,"475":3,"477":5,"483":4,"484":9,"485":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,"554":2,"555":6,"560":8,"561":6,"562":10,"564":10,"567":1,"576":1,"585":1,"595":1,"597":1,"601":2,"607":3,"614":7,"616":6,"623":1,"631":4,"632":5,"633":4,"649":1,"655":1,"659":1,"664":1,"670":4,"671":6,"672":12,"674":4,"678":2,"679":2,"682":1,"684":1,"692":1,"693":2,"698":3,"699":2,"704":1,"705":1,"706":1,"709":1,"713":3,"714":8,"715":6,"716":7,"717":4,"722":2,"727":2,"729":3,"741":1,"746":2,"750":1,"751":2,"756":4}}],["nearby",{"2":{"39":1}}],["needing",{"2":{"165":1,"395":1,"406":1,"438":1,"522":1,"530":1,"531":1}}],["need",{"2":{"41":1,"57":1,"69":1,"71":1,"73":1,"76":2,"94":2,"95":1,"111":5,"112":1,"121":1,"126":1,"127":1,"130":1,"132":1,"137":1,"153":1,"155":1,"160":2,"165":1,"168":2,"170":1,"172":1,"175":2,"178":1,"179":1,"186":1,"187":1,"193":1,"220":2,"240":2,"241":1,"245":1,"246":1,"252":1,"258":1,"259":1,"260":1,"269":1,"296":1,"312":2,"313":1,"314":1,"315":1,"316":1,"318":1,"336":1,"347":1,"348":1,"356":1,"357":2,"358":2,"359":1,"363":1,"369":2,"389":2,"398":1,"408":1,"409":1,"438":1,"454":1,"469":1,"470":1,"484":2,"486":1,"503":4,"507":2,"518":1,"527":1,"529":2,"532":1,"535":1,"541":2,"544":1,"547":1,"551":1,"552":2,"554":1,"555":1,"563":1,"564":1,"576":1,"577":1,"587":1,"597":1,"601":1,"602":1,"604":1,"607":1,"615":2,"616":1,"632":2,"634":1,"640":2,"644":1,"645":1,"650":1,"651":1,"652":1,"656":1,"658":1,"659":1,"672":1,"675":1,"692":1,"693":1,"704":1,"705":1,"714":1,"716":3,"727":1,"733":1,"745":1,"751":1,"756":1}}],["needs",{"2":{"38":1,"41":1,"42":1,"151":1,"153":1,"168":1,"184":1,"202":1,"219":1,"241":1,"346":1,"400":1,"410":1}}],["needed",{"2":{"30":1,"31":1,"34":1,"37":1,"73":1,"76":1,"94":1,"110":1,"121":1,"139":2,"145":2,"160":2,"195":1,"200":2,"351":1,"353":1,"358":1,"374":1,"375":1,"397":1,"398":1,"399":1,"412":1,"503":1,"506":1,"574":1,"579":1,"597":1,"730":1,"739":1,"751":1}}],["negatives",{"2":{"416":1}}],["negatively",{"2":{"26":1,"38":1}}],["negotiate",{"2":{"34":1}}],["nextheaderfunctionid",{"2":{"203":1}}],["nextheight",{"2":{"115":4}}],["next",{"0":{"98":1,"231":1,"286":1,"361":1,"517":1,"569":1},"2":{"32":1,"70":2,"71":1,"75":1,"79":1,"94":1,"98":1,"107":1,"115":2,"127":1,"129":2,"132":1,"137":1,"146":2,"156":1,"187":1,"189":2,"195":4,"196":5,"200":2,"203":1,"214":1,"252":1,"312":1,"319":1,"328":2,"357":2,"358":7,"360":2,"365":2,"384":1,"414":1,"415":2,"500":1,"503":1,"514":1,"525":6,"555":1,"656":1,"733":1}}],["newer",{"2":{"494":1}}],["newmetric",{"2":{"466":2,"685":2,"687":1,"688":1,"689":1,"693":1,"703":2,"705":1,"706":1,"708":1}}],["newcredentials",{"2":{"382":2}}],["newclient",{"2":{"270":2,"271":2,"272":2,"273":2,"357":2,"358":2,"359":2,"360":2,"381":2}}],["newsubmitoptions",{"2":{"270":4}}],["newsletters",{"2":{"32":1}}],["newblob",{"2":{"382":2}}],["newblobv0",{"2":{"270":3,"358":2,"381":2}}],["newblobnamespacev0",{"2":{"270":3,"271":2,"272":2,"358":2,"381":2}}],["newblobstreamxfilterer",{"2":{"147":2}}],["newblobstreamx",{"2":{"129":2,"146":2}}],["newwrappers",{"2":{"148":2}}],["newint",{"2":{"129":8,"141":4,"144":4,"145":8,"148":16}}],["newly",{"2":{"78":1,"394":1,"645":1}}],["new",{"0":{"147":1,"185":1,"187":1,"271":1,"272":1,"294":1,"365":1,"368":1,"454":1,"500":1,"519":1,"644":1,"656":1,"659":1},"1":{"186":1,"187":1,"188":1,"189":1,"190":1,"191":1,"192":1,"193":1,"194":1,"195":1,"196":1,"197":1,"198":1,"199":1,"200":1,"369":1},"2":{"2":2,"22":2,"23":2,"25":1,"36":1,"37":1,"41":3,"42":1,"75":1,"110":1,"121":2,"128":2,"129":4,"137":1,"138":1,"146":2,"147":3,"148":2,"185":1,"186":1,"193":1,"197":1,"212":2,"213":1,"240":2,"241":2,"242":1,"271":4,"272":11,"293":1,"350":1,"352":2,"355":1,"358":4,"359":1,"361":1,"364":8,"365":9,"366":2,"368":1,"369":1,"377":1,"385":1,"390":3,"391":1,"393":2,"394":1,"395":2,"401":1,"414":1,"420":1,"436":1,"438":2,"486":1,"490":2,"503":1,"509":1,"517":1,"527":1,"555":1,"560":2,"576":1,"585":1,"603":1,"604":1,"605":1,"607":1,"634":1,"637":1,"642":3,"643":1,"644":3,"645":1,"649":2,"654":1,"656":1,"659":1,"660":1,"663":1,"665":1,"675":1,"714":1,"715":2,"725":1,"731":1,"756":1}}],["nor",{"2":{"412":1,"557":1}}],["normal",{"2":{"115":3,"149":1,"374":1,"393":1,"595":1}}],["normally",{"2":{"34":1}}],["novel",{"2":{"406":1}}],["now",{"2":{"72":1,"76":2,"79":1,"94":1,"148":2,"187":1,"189":1,"190":1,"196":1,"200":1,"231":1,"252":1,"253":1,"260":2,"262":1,"284":1,"286":1,"296":1,"310":1,"311":1,"328":1,"340":1,"341":2,"349":1,"352":2,"358":4,"360":4,"389":1,"390":1,"391":1,"402":1,"437":1,"503":1,"517":2,"525":1,"569":1,"608":1,"614":1,"615":1,"617":1,"635":1,"639":1,"640":1,"644":1,"645":1,"660":1,"662":1,"663":1,"693":1,"739":1,"740":1,"751":3}}],["none",{"2":{"162":1,"166":1,"174":1,"177":1,"180":1,"315":1,"644":4}}],["nonce",{"2":{"103":4,"105":1,"129":4,"139":4,"145":10,"146":2,"148":6,"254":1,"340":2,"341":2,"377":2,"382":2}}],["non",{"2":{"23":1,"27":1,"94":2,"112":1,"317":1,"348":1,"443":1,"522":2,"554":1,"648":1,"714":2}}],["no",{"0":{"404":1},"2":{"13":1,"14":1,"22":1,"34":1,"102":2,"156":1,"171":1,"222":1,"240":1,"257":1,"302":1,"322":1,"371":1,"373":1,"393":1,"409":1,"416":1,"454":1,"459":1,"470":1,"471":1,"494":5,"505":1,"551":2,"559":3,"579":1,"693":1,"710":1}}],["nothing",{"2":{"591":3}}],["notably",{"2":{"374":1}}],["notable",{"0":{"282":1},"2":{"397":1}}],["notfound",{"2":{"338":2}}],["notify",{"2":{"665":1}}],["notification",{"2":{"65":1}}],["notice",{"2":{"139":4,"141":6,"142":2,"144":2,"145":4,"319":1,"412":1,"523":1,"524":1,"720":1}}],["noted",{"2":{"282":1,"653":1}}],["notes",{"0":{"277":1,"387":1,"532":1},"1":{"278":1,"279":1,"280":1,"281":1,"282":1},"2":{"48":1,"437":1,"593":1}}],["note",{"0":{"602":1},"2":{"26":1,"47":1,"76":2,"89":1,"90":1,"95":1,"102":1,"110":1,"111":3,"113":1,"115":2,"125":1,"127":1,"132":1,"136":1,"137":1,"148":2,"149":1,"151":1,"153":1,"157":1,"159":1,"189":1,"199":1,"200":1,"201":1,"203":1,"206":1,"212":1,"218":2,"219":1,"246":1,"261":1,"311":1,"314":1,"343":1,"369":1,"374":2,"382":2,"410":1,"413":1,"415":2,"416":1,"472":1,"484":1,"494":1,"503":1,"507":2,"511":1,"515":1,"522":1,"530":1,"535":1,"544":1,"550":2,"557":1,"563":1,"591":1,"611":1,"632":1,"640":1,"659":1,"662":1,"683":1,"710":1,"733":1,"739":1}}],["not",{"2":{"7":1,"8":1,"22":1,"23":7,"26":1,"34":1,"35":2,"58":1,"71":2,"75":2,"76":1,"78":2,"81":1,"95":1,"97":1,"102":1,"109":2,"112":1,"113":1,"129":2,"130":1,"148":2,"153":1,"158":1,"159":1,"163":1,"164":1,"165":1,"168":2,"178":1,"184":1,"186":1,"187":3,"193":1,"195":5,"196":2,"200":2,"212":1,"219":1,"246":2,"252":1,"254":1,"261":2,"262":2,"263":1,"268":1,"279":1,"280":1,"282":1,"307":1,"309":1,"311":1,"312":1,"313":1,"316":3,"322":1,"338":2,"359":2,"362":1,"373":3,"374":1,"375":1,"377":3,"382":2,"385":3,"386":1,"387":1,"397":1,"398":1,"403":1,"409":2,"415":1,"416":2,"436":1,"437":2,"438":2,"443":1,"461":1,"467":1,"470":2,"471":1,"483":1,"484":1,"494":1,"503":1,"509":1,"515":1,"532":1,"535":1,"542":1,"550":1,"551":1,"560":1,"564":1,"579":2,"592":1,"602":1,"605":2,"607":1,"620":1,"623":1,"624":1,"632":1,"636":1,"640":1,"685":1,"687":1,"693":1,"703":1,"705":1,"707":1,"710":1,"714":1,"715":1,"716":1,"717":1,"725":1,"746":2,"756":1}}],["noders",{"2":{"687":1,"688":1,"689":1,"693":1,"705":1,"706":1,"707":1,"708":1}}],["nodebuilder",{"2":{"560":6}}],["nodeclient",{"2":{"357":2,"358":4,"360":2}}],["nodeconfig",{"2":{"73":1,"74":1,"75":1,"76":1,"89":2}}],["nodeip",{"2":{"357":4,"358":4,"360":6}}],["node",{"0":{"54":1,"76":1,"77":1,"87":1,"89":1,"90":1,"114":1,"118":1,"243":1,"246":1,"247":1,"248":1,"252":1,"259":1,"260":1,"262":1,"289":1,"297":1,"304":1,"310":1,"317":1,"326":1,"346":1,"352":2,"379":1,"380":2,"381":1,"382":1,"383":1,"437":1,"469":1,"476":1,"479":1,"481":1,"482":1,"483":1,"484":1,"485":1,"486":1,"487":1,"504":1,"530":1,"543":1,"553":1,"564":1,"566":1,"580":1,"582":1,"587":2,"597":1,"603":1,"608":1,"609":1,"611":1,"615":1,"616":1,"617":1,"624":1,"627":1,"629":1,"630":1,"631":1,"632":1,"633":1,"634":1,"635":1,"636":1,"662":1,"664":1,"667":1,"668":1,"669":1,"670":1,"671":1,"674":1,"675":1,"676":1,"692":1,"704":1,"709":1,"718":1,"731":1,"734":1,"738":1,"739":1,"740":1,"741":1,"743":1,"746":1,"747":1,"748":1,"749":1,"750":1,"751":1},"1":{"88":1,"89":1,"90":1,"115":1,"116":1,"117":1,"118":1,"119":1,"244":1,"245":1,"246":1,"247":1,"248":1,"249":1,"250":1,"251":1,"252":1,"253":1,"260":1,"261":1,"298":1,"299":1,"300":1,"301":1,"302":1,"303":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":2,"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,"338":1,"477":1,"478":1,"479":1,"480":2,"481":1,"482":2,"483":2,"484":2,"485":3,"486":3,"487":2,"544":1,"545":1,"546":1,"547":1,"548":1,"554":1,"555":1,"556":1,"557":1,"558":1,"559":1,"565":1,"567":1,"568":1,"569":1,"570":1,"581":1,"582":1,"583":2,"584":2,"585":2,"586":1,"587":1,"588":3,"589":3,"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,"609":1,"610":2,"611":1,"616":1,"617":1,"625":1,"626":1,"627":1,"628":2,"629":1,"630":2,"631":3,"632":3,"633":2,"634":3,"635":2,"636":2,"665":1,"666":1,"667":1,"668":1,"669":2,"670":1,"671":1,"672":2,"673":2,"674":2,"675":3,"676":2,"677":1,"735":1,"736":1,"737":1,"738":1,"739":1,"740":1,"742":1,"743":1,"744":2,"745":2,"746":1,"747":2,"748":2,"749":2,"750":3,"751":1,"752":1,"753":1,"754":1,"755":1,"756":1},"2":{"3":1,"23":2,"24":1,"26":2,"27":1,"42":1,"54":6,"67":1,"72":1,"73":1,"75":4,"76":10,"77":1,"78":1,"79":1,"81":3,"89":3,"90":3,"94":6,"95":1,"97":1,"111":13,"115":1,"118":1,"121":5,"122":1,"141":16,"148":12,"153":1,"241":1,"243":1,"244":2,"245":5,"246":10,"247":6,"248":6,"251":1,"252":15,"253":9,"256":1,"257":3,"258":6,"259":1,"260":2,"261":1,"262":3,"263":2,"266":3,"267":2,"268":1,"269":3,"270":3,"279":2,"282":1,"284":2,"288":2,"289":3,"294":2,"295":1,"297":2,"298":1,"299":1,"300":1,"301":2,"304":1,"307":1,"308":1,"309":1,"310":4,"311":3,"312":9,"313":2,"314":7,"315":6,"316":6,"317":10,"318":8,"319":7,"321":3,"324":2,"325":1,"326":2,"327":1,"332":3,"336":7,"342":2,"345":1,"346":4,"350":2,"351":1,"352":8,"356":2,"357":3,"358":2,"361":1,"362":1,"363":3,"364":1,"373":3,"376":1,"380":1,"381":3,"382":6,"383":1,"385":1,"393":3,"396":2,"397":4,"399":1,"407":1,"408":2,"410":1,"416":8,"437":3,"438":3,"445":1,"465":3,"467":3,"469":2,"470":7,"476":1,"477":4,"478":1,"482":3,"483":4,"484":8,"485":2,"486":3,"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,"554":7,"555":26,"557":4,"558":3,"559":1,"560":6,"561":2,"562":6,"563":1,"564":5,"567":17,"568":5,"569":6,"570":1,"572":1,"573":4,"574":1,"576":1,"579":7,"580":1,"581":1,"585":10,"587":3,"591":1,"593":1,"594":1,"595":1,"596":2,"597":2,"600":1,"602":1,"604":1,"607":2,"609":2,"610":2,"611":3,"612":7,"614":30,"615":5,"616":27,"617":30,"620":1,"622":1,"623":2,"624":3,"626":1,"629":1,"632":12,"633":2,"634":3,"635":2,"636":2,"639":4,"655":1,"659":1,"662":7,"663":2,"664":1,"666":1,"667":1,"669":1,"670":2,"671":6,"672":9,"674":2,"675":3,"676":1,"677":1,"679":1,"683":1,"684":3,"685":1,"686":2,"691":3,"692":2,"693":5,"699":10,"703":1,"704":2,"705":4,"706":1,"707":1,"708":1,"709":4,"716":4,"717":11,"719":4,"720":2,"730":1,"731":3,"732":2,"733":4,"735":1,"736":1,"738":2,"739":4,"740":6,"741":1,"742":1,"743":1,"746":5,"747":1,"748":2,"750":2,"751":2,"752":2,"754":1,"756":5}}],["nodessection",{"2":{"754":1}}],["nodestake",{"2":{"687":1,"688":1,"689":1,"696":1,"712":1}}],["nodes",{"0":{"395":1,"468":1,"477":1,"598":1,"610":1,"625":1,"665":1,"686":1,"691":1,"702":1,"717":1,"719":1,"720":1,"735":1,"737":1},"1":{"469":1,"470":1,"599":1,"600":1,"601":1,"602":1,"603":1,"604":1,"605":1,"687":1,"688":1,"689":1,"690":1,"692":1,"693":1,"694":1,"695":1,"703":1,"704":1,"705":1,"718":1,"719":1,"720":1,"736":1,"738":1,"739":1,"740":1},"2":{"3":2,"54":1,"72":1,"106":1,"110":2,"111":1,"132":2,"137":2,"141":8,"144":2,"148":4,"164":1,"168":1,"241":2,"260":1,"276":1,"298":1,"302":1,"310":1,"377":1,"393":1,"394":1,"395":2,"396":2,"397":5,"398":5,"400":1,"405":1,"406":1,"407":6,"408":6,"409":3,"410":3,"412":1,"413":3,"414":2,"416":5,"436":1,"437":8,"438":12,"467":3,"469":1,"470":2,"476":1,"477":4,"503":2,"541":3,"548":1,"552":3,"554":1,"561":1,"565":1,"576":1,"577":1,"579":2,"580":1,"585":3,"600":1,"608":1,"621":1,"625":5,"661":2,"665":3,"684":1,"687":1,"688":1,"689":1,"692":1,"693":4,"694":3,"699":1,"704":1,"705":2,"706":3,"707":2,"708":2,"709":1,"715":1,"727":1,"731":4,"732":1,"739":1,"741":1,"746":1}}],["navigating",{"2":{"193":1}}],["navigate",{"2":{"75":1,"384":1,"420":1,"425":1,"429":1}}],["name=$",{"2":{"596":6}}],["name=",{"2":{"507":2}}],["name=validator",{"2":{"500":2,"656":2}}],["name=validator1",{"2":{"500":2,"655":2}}],["namekey",{"2":{"500":1}}],["namevalidator",{"2":{"500":1}}],["named",{"2":{"288":1,"346":1,"539":1,"540":1,"568":1}}],["names",{"2":{"200":1,"252":2,"253":2}}],["namespce",{"2":{"113":2}}],["namespacebytes",{"2":{"358":2}}],["namespacehex",{"2":{"357":2,"358":4,"360":4}}],["namespaced",{"0":{"410":1},"2":{"307":1,"406":1,"410":1,"412":1,"415":3}}],["namespaceid",{"2":{"142":5,"148":8,"358":14,"360":8}}],["namespacenode",{"2":{"139":2,"141":12,"143":4,"148":10}}],["namespacemerklemultiproof",{"2":{"139":2,"141":10,"148":6}}],["namespaces",{"0":{"307":1},"2":{"132":2,"159":1,"270":2,"307":1,"364":2,"410":3,"415":1,"437":1,"438":1}}],["namespace",{"0":{"142":1},"2":{"76":8,"78":1,"94":2,"110":8,"113":2,"124":2,"130":2,"132":4,"137":4,"139":6,"141":31,"142":21,"148":16,"160":2,"222":1,"270":11,"271":6,"272":11,"304":1,"319":7,"321":12,"329":2,"334":3,"336":2,"342":1,"356":2,"357":6,"358":23,"360":10,"361":2,"364":16,"365":13,"379":2,"380":2,"381":10,"382":6,"384":2,"410":6,"415":5,"434":3,"438":1,"502":4,"557":1,"558":2}}],["name",{"0":{"289":1,"290":1,"291":1},"1":{"291":1,"292":1},"2":{"26":1,"30":1,"62":1,"71":2,"187":2,"196":2,"241":1,"246":7,"247":6,"248":6,"252":2,"253":2,"288":1,"289":1,"291":3,"312":8,"352":2,"390":1,"391":1,"485":7,"490":8,"491":4,"500":8,"503":6,"519":2,"520":4,"536":2,"542":2,"585":7,"596":6,"615":2,"633":7,"639":4,"655":4,"656":4,"657":4,"659":3,"672":2,"674":7,"745":2,"751":2}}],["naming",{"2":{"2":1,"41":3}}],["natural",{"2":{"403":1}}],["nature",{"2":{"12":1,"211":1,"402":1,"406":1,"495":1}}],["natively",{"2":{"71":1}}],["native",{"2":{"67":1,"69":1,"78":6,"80":1,"93":1,"390":1,"417":1,"453":1,"459":1,"640":1,"647":1,"648":1}}],["nationality",{"2":{"6":1}}],["natnet",{"2":{"57":1}}],["nat",{"2":{"30":1,"602":1}}],["ignore",{"2":{"410":1}}],["ignition",{"2":{"199":16}}],["icon",{"2":{"391":1}}],["icons",{"2":{"52":1}}],["ixg+08hv5rspf3lle8ph+b2tugsgusbiseflxh6wb5e=celestia",{"2":{"321":1}}],["ixg+08hv5rspf3lle8ph+b2tugsgusbiseflxh6wb5e=",{"2":{"319":2,"321":11,"322":2}}],["ibc",{"0":{"637":1},"1":{"638":1,"639":1,"640":1,"641":1,"642":1,"643":1,"644":1,"645":1,"646":1,"647":1,"648":1},"2":{"637":3,"638":2,"639":4,"644":1,"647":4}}],["ibctransfer",{"2":{"277":1}}],["ibchost",{"2":{"277":1}}],["i++",{"2":{"113":2}}],["illustrate",{"2":{"109":1}}],["illustrations",{"2":{"52":1}}],["i",{"0":{"184":1},"2":{"104":1,"113":6,"129":4,"141":4,"143":4,"144":4,"145":5,"148":32,"159":2,"160":1,"172":1,"191":1,"202":1,"284":1,"377":1,"407":3,"408":1,"409":4,"410":3,"412":2,"413":6,"415":3,"416":8,"551":1,"585":10,"601":2,"602":2,"665":1}}],["io",{"2":{"78":2,"141":2,"142":2,"252":2,"358":2,"391":1,"448":1,"474":1,"508":1,"596":12,"614":6,"616":6,"617":6,"687":1,"688":1,"689":1,"696":3,"697":1,"706":1,"707":1,"708":1,"712":3}}],["ip=",{"2":{"739":2}}],["ip=consensus",{"2":{"289":2}}],["ip6",{"2":{"326":4}}],["ip4",{"2":{"326":8,"542":4,"739":2}}],["ip",{"2":{"76":2,"252":2,"253":2,"261":2,"262":2,"311":14,"312":6,"336":2,"352":2,"357":2,"358":2,"360":2,"467":1,"469":2,"470":6,"483":11,"484":6,"485":6,"542":2,"544":2,"545":2,"546":2,"547":4,"550":2,"555":2,"560":2,"564":2,"601":2,"614":6,"617":6,"632":3,"633":6,"661":4,"671":7,"672":6,"674":6,"692":2,"693":4,"704":2,"705":2,"739":2,"740":2,"748":2}}],["idvalidator",{"2":{"655":1}}],["id=$",{"2":{"662":2}}],["id=",{"2":{"507":2,"745":2,"751":2}}],["id=testnet",{"2":{"500":2,"655":2}}],["ids",{"2":{"189":1,"195":2,"542":1,"551":1}}],["idaoracle",{"2":{"102":1,"103":6,"105":1,"107":1,"148":6}}],["ide",{"2":{"356":1}}],["idea",{"2":{"238":1}}],["ideas",{"2":{"35":1}}],["ideally",{"2":{"37":1}}],["ideal",{"2":{"34":1,"223":1}}],["identically",{"2":{"109":1}}],["identifiers",{"2":{"220":2,"410":1}}],["identifier",{"2":{"71":1,"103":2}}],["identify",{"2":{"35":1,"42":1,"94":1,"110":2,"410":1}}],["identifying",{"2":{"32":1}}],["identity",{"2":{"6":2,"415":1,"434":1}}],["id",{"0":{"326":1,"486":1,"551":1,"634":1,"675":1},"2":{"26":3,"71":2,"76":4,"110":1,"127":2,"132":4,"137":4,"141":19,"142":13,"148":12,"187":4,"189":3,"191":2,"192":2,"194":1,"195":3,"196":10,"203":4,"246":1,"252":9,"253":10,"266":2,"312":1,"321":1,"324":2,"325":2,"326":6,"327":2,"328":10,"336":4,"342":4,"348":6,"356":2,"357":2,"358":2,"390":1,"415":1,"472":1,"484":1,"486":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,"585":6,"632":1,"634":1,"639":4,"644":40,"655":5,"659":4,"662":4,"675":1,"710":1,"716":1,"745":1}}],["ifconfig",{"2":{"662":2}}],["if",{"0":{"404":1},"2":{"4":1,"23":3,"24":1,"25":1,"26":5,"30":2,"31":1,"34":3,"35":1,"36":2,"37":1,"38":2,"39":1,"40":2,"41":3,"42":3,"66":2,"76":4,"90":1,"94":3,"97":2,"102":1,"110":1,"111":1,"112":1,"113":8,"115":10,"124":2,"133":4,"136":1,"139":4,"140":2,"141":1,"145":1,"147":6,"148":4,"153":1,"154":1,"155":1,"157":1,"159":3,"163":1,"164":2,"165":1,"168":1,"170":1,"176":1,"182":1,"184":1,"186":1,"187":12,"190":1,"193":1,"195":2,"200":6,"201":1,"202":1,"210":1,"212":1,"219":1,"232":1,"238":1,"241":2,"246":2,"261":1,"268":1,"269":1,"302":1,"307":2,"309":1,"311":3,"312":1,"313":1,"315":2,"316":2,"317":1,"318":1,"319":2,"322":2,"331":1,"337":1,"338":1,"344":1,"347":1,"348":2,"350":1,"352":2,"356":1,"358":1,"359":2,"361":1,"362":1,"363":1,"369":1,"371":2,"373":1,"374":2,"376":1,"377":4,"382":26,"385":1,"386":1,"387":1,"389":8,"390":1,"393":1,"395":1,"397":1,"398":1,"407":1,"409":2,"410":1,"416":3,"459":1,"461":1,"462":1,"483":2,"484":1,"494":1,"499":1,"502":3,"503":3,"505":1,"509":3,"511":1,"516":1,"519":2,"522":2,"529":1,"531":1,"537":2,"539":1,"541":2,"542":1,"547":1,"550":1,"552":3,"555":1,"560":1,"562":1,"565":1,"567":1,"569":2,"570":1,"576":2,"579":2,"585":3,"587":1,"591":1,"592":2,"597":3,"604":1,"607":1,"611":1,"612":2,"614":1,"615":1,"622":1,"623":1,"632":1,"635":1,"641":1,"642":2,"643":1,"648":1,"649":2,"653":1,"654":1,"658":1,"659":2,"705":1,"707":1,"715":1,"716":2,"720":1,"731":1,"733":3,"735":1,"736":3,"738":1,"739":1,"740":2,"745":3,"746":1,"751":3,"756":2}}],["ini",{"2":{"563":1}}],["initiatives",{"2":{"444":1,"445":2,"446":1}}],["initially",{"2":{"731":1}}],["initialisation",{"2":{"550":1,"576":1}}],["initialise",{"2":{"470":1,"693":1}}],["initialization",{"2":{"288":1,"573":1,"579":1}}],["initializing",{"2":{"193":1,"310":1,"352":1,"483":1}}],["initialize",{"0":{"355":1,"483":1,"616":1,"631":1,"655":1,"670":1,"748":1},"2":{"102":1,"240":1,"258":1,"318":2,"355":1,"356":2,"357":3,"358":2,"488":2,"560":1,"562":1,"564":1,"579":1,"585":1,"616":1,"748":1}}],["initialized",{"2":{"72":1,"102":1,"121":1,"572":1,"647":1,"654":1,"670":1}}],["initial",{"2":{"22":1,"71":1,"413":2,"440":1,"445":1,"446":1,"494":2}}],["initcelestia",{"2":{"310":1,"318":1,"631":1,"670":1}}],["init",{"2":{"102":1,"310":5,"318":5,"352":6,"355":2,"483":8,"488":2,"500":2,"550":3,"555":3,"562":4,"564":2,"585":6,"616":9,"631":5,"655":2,"670":5,"748":2}}],["initgit",{"2":{"102":1}}],["inherently",{"2":{"436":1}}],["inherits",{"2":{"506":1}}],["inherit",{"2":{"240":1}}],["inheriting",{"2":{"206":1}}],["inability",{"2":{"400":1}}],["inappropriate",{"2":{"7":1,"8":1,"12":2,"14":1,"15":1}}],["innovative",{"2":{"241":1}}],["innernode",{"2":{"141":12,"148":12}}],["inputting",{"2":{"745":1,"751":1}}],["input=$",{"2":{"341":2}}],["input=3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f",{"2":{"341":2}}],["inputs",{"2":{"103":2,"292":1}}],["input",{"0":{"265":1},"2":{"103":2,"172":3,"176":2,"179":2,"188":1,"265":3,"296":1,"341":4,"342":2,"384":1,"544":1}}],["inbox",{"2":{"85":1,"86":1,"95":1}}],["indeed",{"2":{"406":1,"415":1,"416":1}}],["independently",{"2":{"211":1,"413":1}}],["indexing",{"0":{"588":1},"2":{"399":1,"587":1,"590":1,"604":1}}],["index`",{"2":{"187":2}}],["index",{"2":{"110":2,"127":2,"129":2,"132":6,"135":8,"136":2,"137":2,"144":3,"145":2,"148":4,"157":1,"159":1,"160":3,"519":2,"558":2,"604":2,"753":1}}],["indexer",{"0":{"604":1,"753":1},"2":{"588":1,"590":2,"604":3,"753":1}}],["indexers",{"2":{"404":1}}],["indexed",{"2":{"105":1,"604":2}}],["indexes",{"2":{"71":1}}],["induced",{"2":{"115":1}}],["industry",{"2":{"38":1}}],["indicates",{"2":{"262":1,"336":1,"563":1}}],["indicated",{"2":{"199":1}}],["indicators",{"2":{"32":1}}],["individual",{"2":{"9":1,"15":1,"26":1,"58":1,"107":1}}],["individuals",{"2":{"7":1,"15":1,"209":1}}],["inflation",{"0":{"441":1},"2":{"441":3,"451":2}}],["infra",{"2":{"302":1}}],["infrastructure",{"2":{"23":2,"26":1,"42":1,"78":4,"438":1,"445":1,"548":1,"699":1,"727":1}}],["infocurl",{"2":{"736":1}}],["infocelestia",{"2":{"326":1,"332":1}}],["infos",{"2":{"522":2}}],["informed",{"2":{"698":1,"713":1,"729":1}}],["inform",{"2":{"94":1}}],["informative",{"2":{"42":1,"57":1,"58":1}}],["information$",{"2":{"488":1}}],["information",{"0":{"27":1,"752":1},"2":{"7":1,"36":1,"42":2,"66":2,"72":1,"73":1,"79":1,"90":1,"94":1,"96":1,"97":1,"110":2,"111":2,"112":2,"121":1,"124":1,"127":1,"132":1,"137":1,"148":1,"157":1,"159":1,"187":1,"195":1,"218":1,"219":1,"220":1,"230":1,"246":1,"252":2,"311":1,"312":1,"326":1,"328":1,"376":1,"382":2,"399":2,"400":1,"402":1,"483":1,"484":1,"488":1,"494":1,"503":2,"505":1,"531":2,"537":1,"550":1,"614":1,"632":3,"647":1,"663":1,"671":2,"748":1}}],["informalsystems",{"2":{"508":1}}],["informal",{"2":{"31":1}}],["info",{"0":{"45":1},"1":{"46":1,"47":1,"48":1,"49":1,"50":1},"2":{"26":1,"132":2,"314":1,"326":1,"332":1,"402":1,"522":4,"639":2,"736":1,"745":2,"751":2}}],["intuitive",{"2":{"184":1}}],["int64",{"2":{"129":8,"141":4,"145":4,"146":4,"148":8}}],["intel",{"2":{"623":1}}],["intensive",{"2":{"591":1}}],["intended",{"2":{"267":1}}],["integrating",{"2":{"214":1,"275":1}}],["integrations",{"0":{"388":1,"465":1,"684":1},"1":{"389":1,"390":1,"391":1,"466":1,"467":1,"468":1,"469":1,"470":1,"685":1,"686":1,"687":1,"688":1,"689":1,"690":1,"691":1,"692":1,"693":1,"694":1,"695":1},"2":{"254":1,"505":1}}],["integration",{"0":{"283":1,"284":1},"1":{"284":1,"285":2,"286":1},"2":{"66":2,"67":1,"81":1,"88":1,"92":2,"93":2,"94":2,"95":1,"96":2,"97":1,"206":1,"222":2,"231":1,"284":1,"285":2,"286":2,"348":1,"505":1,"679":1}}],["integrate",{"0":{"99":1,"108":1,"213":1,"214":1,"275":1},"1":{"100":1,"101":1,"102":1,"103":1,"104":1,"105":1,"106":1,"107":1,"109":1,"110":1,"111":1,"112":1,"113":1,"114":1,"115":1,"116":1,"117":1,"118":1,"119":1,"214":1,"215":1,"216":1,"276":1,"277":1,"278":1,"279":1,"280":1,"281":1,"282":1},"2":{"213":1,"214":2,"242":1,"350":1,"505":1}}],["integrated",{"2":{"96":1}}],["integrates",{"2":{"81":1}}],["integrity",{"2":{"71":2}}],["integer",{"2":{"71":1}}],["interruptions",{"2":{"679":1}}],["inter",{"2":{"637":1}}],["interval",{"2":{"503":4,"560":4,"639":2}}],["intermediate",{"2":{"408":1,"409":1,"415":1,"416":1}}],["interfaces",{"2":{"111":2,"448":1,"449":1,"458":1,"601":2}}],["interface",{"0":{"105":1},"2":{"94":2,"102":1,"105":3,"107":1,"111":4,"204":1,"313":1,"384":2,"412":1,"506":1,"531":2}}],["internally",{"2":{"477":1}}],["internal",{"2":{"76":5,"298":1}}],["interestingly",{"2":{"414":1}}],["interested",{"2":{"38":2,"39":3,"42":3,"339":1}}],["interest",{"2":{"36":2,"37":1,"47":1}}],["interests",{"2":{"36":1,"41":1,"42":1}}],["interactivecelestia",{"2":{"536":1}}],["interactive",{"2":{"58":1,"536":1}}],["interactions",{"2":{"13":1,"111":1,"224":1}}],["interaction",{"2":{"13":2,"14":3,"15":1,"31":1,"75":1,"477":1}}],["interacting",{"2":{"3":1,"119":1,"297":1,"301":1,"314":2,"541":1,"551":1,"552":1}}],["interact",{"2":{"6":1,"107":1,"121":1,"216":1,"224":1,"226":1,"252":2,"253":2,"279":1,"297":1,"313":1,"314":2,"315":1,"316":1,"317":2,"384":1,"620":1,"665":1,"706":1,"707":1,"708":1}}],["intake",{"2":{"41":2}}],["introducing",{"2":{"414":1}}],["introduction",{"0":{"53":1,"56":1,"91":1,"150":1,"258":1,"283":1,"305":1,"414":1},"1":{"92":1,"93":1,"94":1,"95":1,"96":1,"97":1,"98":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,"163":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,"284":1,"285":1,"286":1,"306":1,"307":1},"2":{"37":1,"53":2,"55":1,"56":3,"66":1,"67":1,"88":1,"92":1,"222":1,"276":1}}],["introduces",{"2":{"437":1}}],["introduced",{"2":{"386":1}}],["introduce",{"2":{"31":1}}],["into",{"0":{"402":1},"2":{"22":1,"25":1,"53":1,"62":1,"74":1,"78":1,"81":1,"94":1,"95":1,"96":1,"122":2,"124":1,"193":1,"199":1,"206":1,"246":1,"307":1,"358":2,"373":1,"407":2,"409":1,"410":2,"413":1,"415":5,"434":3,"491":1,"501":1,"505":1,"514":1,"539":1,"544":1,"551":1,"560":1,"567":1,"570":1,"640":1,"679":1,"715":1}}],["inception",{"2":{"678":1}}],["incentive",{"2":{"416":1}}],["incentives",{"2":{"37":1,"403":1}}],["incentivized",{"2":{"445":1}}],["incentivizes",{"2":{"405":1}}],["incentivize",{"0":{"403":1},"2":{"403":1}}],["incomplete",{"2":{"410":1}}],["incoming",{"2":{"365":2}}],["incorrectly",{"0":{"409":1},"2":{"397":1,"398":1,"409":2}}],["incorrect",{"2":{"76":1,"254":1,"398":1}}],["incurred",{"2":{"258":1,"266":1}}],["incubators",{"2":{"34":1}}],["incredible",{"2":{"44":1}}],["increased",{"2":{"78":2}}],["increases",{"2":{"39":1,"71":1,"396":1,"414":1}}],["increase",{"2":{"38":1,"39":1,"413":1,"414":1,"502":2,"563":1}}],["increasing",{"2":{"36":1,"408":1,"414":1,"563":1}}],["incident",{"2":{"10":1,"13":1}}],["inclusion",{"0":{"107":1,"117":1,"126":1,"130":1,"131":1,"134":1,"136":1,"160":1},"1":{"127":1,"128":1,"132":1,"133":1,"135":1,"137":1,"138":1},"2":{"40":1,"102":1,"104":2,"105":1,"107":4,"110":1,"112":3,"117":1,"122":1,"124":1,"126":1,"127":1,"129":2,"131":1,"132":2,"136":1,"140":1,"141":1,"145":1,"148":1,"149":1,"153":3,"154":2,"159":2,"160":7,"173":1,"176":1,"184":2,"212":2,"386":1}}],["inclusive",{"2":{"6":1,"46":1,"201":1}}],["including",{"2":{"13":1,"14":2,"15":1,"26":1,"30":3,"31":1,"41":1,"46":1,"58":1,"71":1,"73":1,"79":1,"296":1,"375":1,"445":1,"745":1}}],["included",{"2":{"105":2,"112":2,"113":3,"122":1,"145":2,"157":1,"159":1,"168":1,"212":1,"270":2,"319":1,"321":1,"322":1,"364":2,"371":2,"377":3,"381":2,"385":1,"386":1,"387":1,"434":1,"683":1}}],["includes",{"2":{"13":1,"51":1,"52":1,"72":1,"81":1,"96":1,"281":1,"402":1,"410":1,"434":1,"446":1,"638":1,"660":1,"716":1}}],["include",{"2":{"7":2,"9":1,"38":1,"157":1,"201":1,"213":1,"242":1,"315":1,"343":1,"404":1,"434":1,"504":1,"661":1}}],["invocations",{"2":{"597":1,"751":1}}],["involves",{"2":{"374":1,"401":1}}],["involve",{"2":{"111":2,"112":1}}],["involvement",{"2":{"37":1}}],["involved",{"2":{"13":1,"14":1,"175":1,"284":1,"374":1,"375":1}}],["investigations",{"2":{"170":1}}],["investigate",{"2":{"36":1,"38":1}}],["investigated",{"2":{"10":1,"155":1,"168":1,"178":1}}],["invalid",{"0":{"607":1,"756":1},"2":{"112":2,"115":2,"148":4,"159":1,"164":2,"168":2,"364":2,"365":2,"409":2,"416":1,"560":2,"607":2,"756":2}}],["invitation",{"2":{"47":1}}],["invitations",{"2":{"30":1}}],["invites",{"2":{"48":1}}],["invite",{"2":{"37":2,"39":2}}],["invisible",{"2":{"6":1}}],["inspect",{"2":{"393":1}}],["inspired",{"2":{"16":1,"163":1,"187":1}}],["insecure",{"2":{"382":4}}],["inside",{"0":{"176":1},"2":{"157":2,"172":1,"173":1,"176":2,"178":1,"179":3,"181":1,"244":1,"246":1,"356":1,"361":1,"485":1,"539":1,"607":1,"633":1,"659":1,"660":1,"661":1,"674":1,"756":2}}],["insight",{"2":{"501":1}}],["insightful",{"2":{"44":1}}],["insights",{"2":{"32":1,"37":1,"58":1,"508":1}}],["instrumentation",{"2":{"502":6}}],["instructions",{"0":{"195":1},"2":{"185":1,"196":2,"199":1,"202":1,"203":1,"347":1,"348":1,"368":1,"369":1,"484":1,"510":1,"511":1,"522":1,"540":1,"568":1,"570":1,"583":1,"612":3,"620":1,"622":2,"632":1,"699":2,"743":1,"751":1,"753":1}}],["instead",{"2":{"89":1,"95":1,"111":1,"115":3,"146":4,"147":4,"148":4,"161":1,"179":1,"254":1,"284":1,"321":1,"373":3,"377":1,"437":1,"454":1}}],["instability",{"2":{"679":1,"722":1}}],["instantaneous",{"2":{"636":1}}],["instantiation",{"0":{"649":1},"1":{"650":1,"651":1,"652":1,"653":1,"654":1,"655":1,"656":1,"657":1,"658":1,"659":1,"660":1,"661":1,"662":1,"663":1}}],["instantiating",{"2":{"90":1,"310":1}}],["instantiate",{"0":{"310":1,"663":1},"2":{"310":1,"413":1,"649":1}}],["instance",{"2":{"62":1,"77":2,"199":1,"200":1,"220":1,"369":1,"479":1,"503":1,"543":1,"562":1,"582":1,"627":1,"668":1,"743":1}}],["instances",{"2":{"10":1}}],["installer",{"2":{"612":1}}],["installed",{"2":{"101":1,"187":1,"193":1,"251":1,"257":1,"351":1,"389":1,"515":1,"526":1,"535":1,"539":1,"569":1,"572":1,"608":1,"623":2,"638":1,"639":1}}],["installmake",{"2":{"539":1,"567":1}}],["installs",{"2":{"245":2,"567":1}}],["installation",{"0":{"245":1,"652":1},"2":{"243":1,"540":2,"568":2}}],["installing",{"0":{"102":1,"540":1,"567":1,"568":1},"2":{"480":1,"482":1,"538":1,"540":1,"567":1,"568":1,"583":1,"584":1,"612":1,"620":1,"622":2,"629":1,"669":1,"730":1,"733":1,"747":1}}],["install",{"0":{"352":1,"482":1,"538":1,"566":1,"584":1,"622":1,"623":1,"629":1,"669":1,"747":1},"1":{"539":1,"540":1,"541":1,"567":1,"568":1,"569":1,"570":1,"630":1,"631":1,"632":1,"633":1,"634":1,"635":1,"636":1},"2":{"74":2,"102":3,"188":2,"195":2,"245":2,"268":1,"309":1,"313":1,"352":1,"362":1,"389":2,"482":1,"503":1,"505":1,"510":1,"512":1,"535":1,"539":3,"540":1,"567":3,"568":1,"570":1,"612":1,"622":15,"623":2,"638":1,"652":1,"669":1,"731":1,"732":1,"733":1,"736":2,"738":2,"739":2,"740":2}}],["insulting",{"2":{"7":1}}],["in",{"0":{"59":1,"60":1,"97":1,"139":1,"189":1,"260":1,"262":1,"377":1,"395":1,"437":1,"562":1,"645":1,"667":1,"721":1},"1":{"60":1,"61":2,"62":2,"63":1,"64":1,"65":1,"140":1,"141":1,"142":1,"143":1,"144":1,"145":1,"146":1,"147":1,"148":1,"190":1,"191":1,"192":1,"722":1,"723":1,"724":1,"725":1,"726":1,"727":1,"728":1,"729":1},"2":{"3":1,"4":2,"6":2,"7":1,"8":1,"9":1,"11":2,"12":1,"13":1,"18":1,"20":1,"22":6,"23":6,"24":3,"26":2,"28":1,"29":1,"31":1,"32":1,"33":1,"34":2,"35":4,"36":3,"37":2,"38":6,"39":7,"40":4,"42":7,"44":2,"46":2,"47":2,"48":1,"50":1,"54":1,"55":1,"58":4,"59":2,"61":3,"62":4,"64":3,"65":3,"69":1,"70":2,"71":12,"72":4,"73":1,"74":1,"75":5,"76":10,"78":10,"79":3,"80":1,"81":1,"82":1,"89":4,"92":1,"93":1,"94":7,"95":10,"96":1,"97":3,"98":1,"104":2,"105":2,"106":1,"107":2,"110":13,"111":3,"112":12,"113":4,"115":1,"121":3,"122":8,"124":9,"125":1,"126":1,"135":2,"137":3,"138":2,"140":3,"141":13,"142":1,"143":1,"144":9,"145":8,"146":2,"147":1,"149":2,"150":1,"151":2,"152":1,"153":3,"154":4,"156":3,"157":4,"158":1,"159":9,"160":9,"164":2,"165":1,"167":2,"168":5,"169":1,"170":1,"171":1,"172":3,"173":1,"175":1,"176":2,"179":1,"182":2,"184":5,"186":1,"187":7,"188":3,"189":5,"190":1,"193":6,"195":6,"196":16,"198":2,"199":2,"200":7,"201":3,"202":2,"204":1,"206":2,"209":1,"211":1,"212":5,"213":1,"215":1,"218":2,"219":1,"220":1,"222":2,"223":1,"226":1,"238":1,"240":1,"241":2,"242":2,"244":1,"245":4,"246":2,"247":1,"252":3,"253":3,"254":4,"258":2,"261":1,"269":1,"270":1,"271":5,"272":6,"277":1,"278":1,"279":4,"282":2,"284":3,"286":1,"289":1,"292":1,"296":1,"297":1,"302":1,"304":1,"307":1,"310":1,"311":1,"312":2,"314":3,"315":1,"316":1,"317":1,"318":1,"319":4,"321":4,"322":1,"326":2,"327":1,"331":6,"334":1,"336":3,"339":1,"343":2,"344":1,"346":2,"348":1,"349":1,"350":1,"352":5,"353":1,"355":1,"356":1,"358":3,"363":1,"364":1,"365":4,"368":1,"369":5,"371":5,"372":1,"373":3,"374":4,"375":4,"376":2,"377":2,"378":1,"380":1,"383":1,"384":5,"385":2,"386":1,"387":1,"389":2,"390":2,"391":2,"395":2,"398":1,"400":1,"401":1,"402":1,"403":1,"406":1,"407":4,"408":3,"410":1,"413":9,"414":1,"415":3,"416":3,"417":1,"420":1,"421":1,"425":1,"434":1,"435":2,"436":3,"437":3,"438":4,"440":1,"441":2,"444":1,"445":1,"446":2,"451":1,"453":1,"454":3,"455":2,"456":1,"458":1,"459":1,"462":2,"465":2,"466":1,"467":2,"470":2,"477":3,"483":1,"485":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,"554":4,"555":2,"557":1,"560":1,"562":2,"563":4,"564":1,"567":1,"568":2,"573":1,"576":1,"577":1,"579":1,"580":1,"585":7,"587":1,"589":1,"590":1,"591":2,"592":2,"595":3,"602":1,"604":2,"605":6,"607":2,"609":1,"611":1,"612":2,"616":3,"620":1,"623":1,"632":1,"633":1,"634":1,"636":2,"637":2,"638":1,"641":2,"642":5,"653":1,"654":1,"655":1,"658":3,"659":2,"660":2,"662":1,"667":1,"670":1,"671":1,"672":2,"673":1,"674":1,"675":1,"678":2,"682":1,"683":6,"684":4,"685":1,"695":1,"699":3,"703":1,"705":1,"707":1,"708":1,"709":1,"715":3,"716":2,"717":2,"724":1,"727":1,"730":2,"731":2,"733":4,"736":3,"740":2,"741":2,"745":2,"751":3,"756":2}}],["imagine",{"2":{"242":1}}],["image",{"2":{"89":1,"252":3,"253":6,"391":1,"614":1}}],["imagery",{"2":{"7":1}}],["images",{"2":{"3":1,"391":1}}],["immutable",{"2":{"71":2,"103":2}}],["improperly",{"2":{"397":1}}],["improve",{"2":{"41":1}}],["improvement",{"2":{"32":1,"408":1}}],["improvements",{"2":{"4":1,"679":1}}],["improved",{"2":{"28":1,"206":1,"501":1}}],["improves",{"2":{"23":1}}],["improving",{"2":{"2":1}}],["impractical",{"2":{"393":1}}],["importing",{"0":{"248":1,"295":1,"491":1},"2":{"146":2,"147":2,"148":2,"542":1}}],["import",{"0":{"356":1},"2":{"103":6,"128":2,"129":4,"135":1,"146":4,"147":4,"148":10,"248":1,"270":1,"295":2,"356":4,"376":1,"381":1,"382":1,"389":7,"477":1,"491":5,"503":2,"542":1,"640":1}}],["importance",{"2":{"58":1}}],["important",{"0":{"45":1},"1":{"46":1,"47":1,"48":1,"49":1,"50":1},"2":{"23":1,"27":1,"95":1,"110":2,"375":1,"385":1,"412":1,"503":1,"504":1,"519":2,"551":1,"714":1}}],["implement",{"2":{"361":1,"413":1,"438":2}}],["implemented",{"2":{"149":1,"369":1,"437":4,"707":1}}],["implements",{"2":{"94":1}}],["implementations",{"0":{"207":1},"2":{"111":2,"637":1}}],["implementation",{"0":{"94":1,"95":1,"96":1,"217":1,"260":1},"1":{"218":1,"219":1,"220":1},"2":{"93":3,"107":1,"109":1,"148":4,"206":3,"207":1,"211":1,"212":2,"218":1,"296":1,"314":1,"368":1,"477":1,"638":1,"715":1}}],["impactful",{"2":{"37":1,"58":1}}],["impact",{"2":{"11":1,"12":1,"13":1,"14":1,"15":1,"16":1,"26":1,"38":1}}],["itrocket",{"2":{"687":1,"688":1,"689":1,"690":1,"693":1,"696":1,"697":1,"705":1,"706":1,"707":1,"708":1,"712":1}}],["ith",{"2":{"375":2}}],["item",{"2":{"360":1}}],["items",{"2":{"357":1,"358":1}}],["itself",{"2":{"81":1,"503":2}}],["its",{"2":{"18":2,"26":2,"71":4,"72":2,"95":1,"124":1,"132":1,"134":1,"141":2,"142":2,"148":2,"151":1,"153":1,"157":1,"159":1,"163":1,"167":1,"168":1,"172":1,"184":1,"187":1,"194":1,"195":1,"210":1,"211":1,"231":2,"324":1,"361":1,"373":1,"374":1,"384":2,"395":1,"396":1,"410":2,"413":1,"447":1,"455":1,"503":1,"563":1,"678":1,"736":1,"738":1,"739":1,"740":1}}],["it",{"2":{"2":2,"23":1,"29":1,"35":1,"36":1,"41":2,"42":1,"47":1,"62":1,"71":5,"73":2,"75":1,"76":3,"77":1,"89":1,"94":6,"95":1,"97":1,"104":1,"105":1,"107":6,"109":1,"111":2,"115":2,"121":1,"122":2,"124":2,"126":2,"136":2,"140":2,"142":2,"145":1,"148":2,"150":1,"153":1,"155":1,"157":1,"159":2,"164":1,"167":1,"168":1,"172":2,"179":1,"184":1,"187":10,"188":4,"189":3,"193":1,"195":8,"196":2,"199":4,"200":5,"201":1,"202":1,"203":1,"204":1,"211":1,"212":2,"218":1,"219":3,"220":1,"222":1,"223":2,"226":1,"237":1,"240":1,"243":1,"245":1,"256":1,"260":1,"270":2,"282":1,"284":1,"296":1,"300":1,"302":1,"311":2,"314":1,"319":1,"328":2,"337":2,"338":1,"340":1,"341":1,"348":1,"350":1,"352":3,"353":1,"358":7,"359":3,"360":1,"361":2,"371":2,"374":2,"375":4,"377":4,"381":2,"382":2,"384":1,"385":1,"386":1,"387":1,"389":4,"393":5,"396":2,"397":1,"402":1,"403":1,"409":1,"410":1,"413":1,"414":1,"415":5,"416":1,"438":2,"441":1,"462":5,"483":2,"484":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,"559":1,"560":2,"562":2,"563":1,"565":1,"568":1,"569":1,"592":1,"596":1,"600":1,"601":2,"605":1,"607":1,"609":1,"615":2,"632":2,"637":1,"638":2,"640":1,"641":1,"648":1,"658":1,"660":2,"661":1,"670":1,"678":1,"679":1,"683":1,"699":1,"705":1,"714":1,"722":1,"725":1,"727":1,"733":1,"736":2,"745":1,"746":2,"756":1}}],["issuing",{"2":{"454":1}}],["issuance",{"2":{"441":2}}],["issues",{"2":{"8":1,"54":1,"76":1,"285":1,"348":3,"570":1}}],["issue",{"2":{"4":1,"78":1,"377":1,"400":1,"441":1,"454":1,"638":1,"720":2}}],["isn",{"2":{"115":1,"369":1,"683":1}}],["isvalid",{"2":{"115":2}}],["isolated",{"2":{"612":1}}],["isolates",{"2":{"241":1}}],["isolate",{"2":{"111":1}}],["is",{"0":{"206":1,"212":1,"218":1,"237":1,"238":1,"393":1,"394":1,"396":1,"398":1,"399":1,"400":1,"401":1,"404":1},"1":{"207":1},"2":{"1":1,"7":1,"9":1,"14":1,"16":1,"20":1,"21":1,"23":1,"25":1,"26":2,"29":1,"31":1,"34":1,"37":1,"38":1,"39":2,"41":3,"42":1,"43":1,"46":1,"47":2,"49":2,"50":1,"51":1,"53":1,"54":2,"55":2,"58":3,"64":1,"69":1,"70":3,"71":8,"72":1,"73":3,"75":3,"76":7,"78":10,"79":1,"81":2,"93":3,"94":7,"95":2,"96":1,"97":2,"102":2,"103":2,"104":3,"105":2,"107":6,"109":3,"110":30,"111":5,"112":6,"113":6,"115":6,"119":1,"121":1,"122":3,"124":5,"126":2,"130":1,"132":2,"133":1,"134":2,"135":1,"136":1,"137":1,"138":1,"140":2,"141":5,"142":3,"144":2,"145":12,"148":6,"149":1,"150":1,"152":1,"153":1,"154":1,"157":3,"158":1,"159":9,"160":12,"163":3,"164":6,"165":1,"168":9,"170":1,"171":1,"172":6,"173":1,"176":6,"178":1,"179":4,"182":1,"184":6,"187":13,"189":2,"190":2,"193":3,"195":4,"196":6,"197":2,"198":1,"199":8,"200":7,"203":1,"204":1,"206":4,"209":1,"210":2,"211":2,"212":3,"218":2,"219":3,"220":1,"221":2,"222":3,"226":1,"227":1,"237":1,"238":1,"240":2,"241":1,"243":1,"244":1,"254":3,"258":1,"260":1,"267":1,"269":1,"275":1,"277":2,"279":1,"280":1,"281":1,"282":3,"283":2,"284":8,"285":1,"286":1,"297":1,"300":1,"301":1,"302":2,"306":1,"310":1,"312":1,"314":5,"315":2,"316":2,"317":1,"318":3,"319":3,"321":3,"322":2,"324":1,"325":1,"326":2,"327":1,"328":1,"331":2,"338":1,"339":1,"346":2,"347":1,"348":1,"352":2,"356":1,"358":5,"360":2,"363":1,"368":1,"371":4,"373":4,"374":4,"375":14,"376":2,"377":2,"382":8,"384":2,"385":3,"386":1,"387":1,"389":5,"393":2,"394":3,"395":1,"396":3,"397":6,"398":3,"399":2,"400":2,"401":2,"402":3,"403":3,"405":1,"406":2,"407":6,"409":7,"410":8,"412":7,"413":3,"414":3,"415":5,"416":9,"417":1,"423":1,"427":1,"432":1,"434":2,"436":5,"437":3,"438":4,"440":1,"446":3,"453":1,"454":1,"458":1,"459":4,"460":1,"461":4,"462":1,"465":1,"472":1,"473":1,"477":5,"479":1,"483":1,"484":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,"554":1,"555":1,"557":1,"559":1,"560":3,"562":2,"563":2,"564":2,"565":1,"567":2,"568":2,"574":1,"576":1,"577":2,"579":4,"582":1,"585":3,"587":1,"589":2,"591":1,"592":2,"594":1,"597":2,"602":3,"604":7,"605":1,"607":1,"609":1,"611":1,"612":3,"614":2,"615":1,"620":2,"623":1,"624":1,"627":1,"632":3,"636":1,"637":1,"638":4,"641":3,"647":1,"648":1,"649":1,"655":2,"657":1,"658":1,"662":2,"665":1,"667":1,"671":1,"673":1,"678":2,"679":2,"683":7,"684":2,"694":1,"695":1,"699":4,"706":2,"707":3,"708":3,"710":1,"714":3,"716":1,"722":2,"725":2,"727":1,"732":1,"733":2,"734":1,"736":3,"739":1,"740":1,"743":1,"746":3,"750":1,"751":3,"756":1}}],["lcd",{"2":{"688":2}}],["l5dhzjldczo",{"2":{"519":2,"520":2}}],["ljj6hspn0kn09qf9fy8kdyh40000gn",{"2":{"515":4}}],["luckily",{"2":{"400":1,"403":1}}],["lumina",{"2":{"362":1,"667":2}}],["lunaroasis",{"2":{"311":2,"672":2,"687":1,"688":1,"689":1,"693":1}}],["ls",{"2":{"199":2,"200":2}}],["ldflags",{"2":{"199":2}}],["lfg",{"2":{"78":2}}],["l2genesisdeltatimeoffset",{"2":{"343":1}}],["l2s",{"2":{"97":1,"206":2,"211":1,"254":2,"284":1}}],["l2",{"0":{"75":1,"348":1},"2":{"69":1,"71":2,"72":1,"73":1,"78":6,"96":1,"105":1,"107":1,"206":1,"210":2,"214":1,"254":1,"283":1,"348":2,"394":2}}],["l1",{"0":{"348":1},"2":{"69":2,"71":1,"78":8,"340":1,"341":1,"348":4,"394":1}}],["lts",{"2":{"280":1,"479":1,"582":1,"627":1,"668":1,"743":1}}],["lt",{"2":{"62":1,"76":8,"113":2,"121":3,"127":1,"129":4,"132":1,"133":2,"135":7,"137":1,"146":4,"147":6,"188":8,"189":30,"191":4,"192":4,"194":6,"200":4,"201":1,"218":1,"246":13,"247":12,"248":12,"252":5,"253":8,"259":4,"266":4,"271":6,"272":6,"311":6,"312":21,"314":6,"316":6,"317":4,"319":8,"321":8,"325":2,"328":4,"331":6,"333":4,"340":1,"357":6,"358":6,"360":12,"379":4,"380":4,"389":14,"470":6,"472":3,"483":8,"484":4,"485":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,"555":10,"560":8,"561":4,"562":8,"564":9,"617":2,"632":4,"633":12,"640":4,"672":6,"673":3,"674":12,"693":6,"710":3,"736":4,"738":4,"739":6,"740":6,"745":4,"748":2,"751":2}}],["l3s",{"2":{"97":1,"206":1,"254":1}}],["l3configuration",{"2":{"78":2}}],["l3",{"2":{"59":1,"62":1,"64":1,"73":2,"78":6}}],["living",{"2":{"682":1}}],["lives",{"2":{"42":1,"95":1,"517":1}}],["live",{"2":{"40":4,"61":1,"286":1,"348":1,"517":1,"679":1}}],["life",{"2":{"434":1,"678":1}}],["lifecycle",{"0":{"415":1},"1":{"416":1}}],["limitnofile=65535",{"2":{"736":2}}],["limitnofile=1400000",{"2":{"563":2,"738":2,"739":2}}],["limitnofile",{"2":{"563":2}}],["limit=fc+∑i=1nssn",{"2":{"375":1}}],["limit×gas",{"2":{"375":1}}],["limits",{"0":{"373":1},"1":{"374":1,"375":1,"376":1},"2":{"322":1,"413":1,"502":2,"563":1}}],["limited",{"2":{"277":1,"408":2,"414":1,"494":1}}],["limit",{"2":{"261":2,"266":2,"319":1,"322":1,"331":1,"373":2,"375":4,"376":1,"382":2,"472":1,"522":2,"563":1,"683":1,"710":1}}],["limitations",{"2":{"156":1}}],["libp2p",{"2":{"416":1,"477":1}}],["libssl",{"2":{"622":4}}],["libs",{"2":{"148":2}}],["library",{"0":{"139":1,"268":1,"299":1,"362":1},"1":{"140":1,"141":1,"142":1,"143":1,"144":1,"145":1,"146":1,"147":1,"148":1,"269":1,"270":1,"271":1,"272":1,"273":1,"274":1,"363":1,"364":1,"365":1,"366":1,"367":1},"2":{"107":3,"139":1,"148":2,"159":1,"168":1,"176":1,"268":1,"293":1,"299":1,"300":1,"362":1}}],["libraries",{"2":{"34":2,"173":1,"278":1,"280":1,"281":1,"356":1,"730":1}}],["lib",{"2":{"103":2,"107":1,"148":2}}],["license",{"2":{"103":2}}],["lightdsudo",{"2":{"740":2}}],["lightd",{"2":{"740":13}}],["lightbridgefull",{"2":{"614":1}}],["light$home",{"2":{"346":1}}],["lightlink",{"2":{"148":1}}],["light",{"0":{"54":1,"76":1,"118":1,"310":1,"346":1,"352":1,"380":1,"557":1,"558":1,"564":1,"609":1,"615":1,"664":1,"665":1,"667":1,"668":1,"670":1,"671":1,"674":1,"676":1,"702":1,"740":1},"1":{"565":1,"610":1,"616":1,"617":1,"665":1,"666":1,"667":1,"668":1,"669":2,"670":1,"671":1,"672":2,"673":2,"674":2,"675":3,"676":2,"677":1,"703":1,"704":1,"705":1},"2":{"54":7,"67":1,"76":7,"96":1,"111":9,"118":1,"119":1,"206":2,"212":2,"241":1,"244":1,"246":2,"247":2,"248":2,"252":5,"253":10,"257":1,"259":1,"261":2,"262":4,"263":2,"282":1,"289":2,"294":2,"302":1,"308":1,"310":8,"311":14,"312":12,"316":3,"317":3,"318":12,"319":1,"336":3,"346":7,"351":1,"352":10,"376":1,"395":1,"396":4,"397":6,"398":2,"406":1,"407":5,"408":8,"409":3,"414":2,"416":9,"437":6,"467":1,"470":3,"477":1,"483":1,"543":2,"548":1,"554":1,"557":1,"558":1,"561":3,"564":7,"565":1,"569":1,"573":2,"579":6,"595":2,"609":2,"616":6,"617":6,"625":2,"632":1,"664":1,"665":2,"666":1,"667":1,"670":6,"671":7,"672":12,"674":9,"675":3,"676":1,"677":1,"691":1,"693":1,"694":2,"699":1,"717":2,"719":1,"731":2,"733":2,"739":1,"740":7}}],["lightning",{"2":{"42":1,"283":1}}],["lisbon",{"2":{"44":1}}],["listcelestia",{"2":{"527":1,"536":1}}],["listing",{"2":{"490":2}}],["listed",{"2":{"369":1,"467":1,"685":1,"687":1,"703":1,"716":1}}],["listens",{"2":{"601":1}}],["listen",{"2":{"204":1,"416":1,"502":4,"503":1,"601":1,"665":1}}],["listening",{"0":{"147":1},"2":{"147":1,"601":1}}],["list",{"0":{"58":1,"520":1},"2":{"26":2,"30":1,"36":1,"37":2,"50":2,"58":6,"71":1,"94":3,"103":2,"141":2,"144":2,"188":1,"190":1,"191":2,"196":1,"203":1,"252":3,"253":3,"274":1,"312":2,"367":1,"407":1,"430":1,"443":1,"483":2,"484":2,"490":2,"494":1,"502":2,"520":2,"527":2,"536":1,"555":2,"585":2,"632":2,"639":4,"645":4,"661":2,"672":2,"684":1,"699":1,"751":1}}],["lines",{"2":{"499":2}}],["linear",{"2":{"375":1}}],["line",{"0":{"507":1},"2":{"311":1,"313":1,"483":1,"506":1,"531":1,"562":1,"574":1,"605":1,"614":1,"661":1}}],["lineup",{"2":{"37":1}}],["link",{"2":{"216":1,"358":2,"465":1,"503":2,"504":1,"684":1,"699":1}}],["linked",{"2":{"118":1}}],["links",{"2":{"26":2,"48":1,"121":1}}],["linuxdocker",{"2":{"615":1}}],["linux",{"2":{"76":1,"280":1,"479":1,"540":1,"568":1,"582":1,"613":1,"620":1,"623":12,"627":1,"668":1,"743":1}}],["little",{"2":{"34":1,"72":1,"282":1,"659":1}}],["likely",{"2":{"111":5,"282":2,"404":1,"560":1,"607":1,"756":1}}],["likelihood",{"2":{"39":1,"416":1}}],["like",{"2":{"13":1,"30":1,"34":2,"35":1,"40":1,"41":2,"42":1,"67":1,"69":1,"71":2,"76":1,"77":1,"89":1,"109":1,"110":1,"115":1,"187":3,"200":2,"232":1,"237":1,"238":2,"241":2,"252":1,"253":1,"269":1,"302":2,"307":1,"311":1,"312":1,"318":1,"321":1,"328":2,"331":3,"337":1,"338":1,"340":1,"344":1,"352":1,"363":1,"389":1,"393":1,"394":1,"396":1,"402":1,"405":1,"440":1,"455":1,"465":1,"475":1,"485":1,"503":1,"511":1,"512":1,"516":1,"522":3,"542":2,"547":1,"552":1,"555":1,"562":1,"563":1,"565":1,"597":1,"607":1,"612":1,"614":2,"622":1,"633":1,"635":1,"641":1,"674":1,"679":1,"684":1,"698":1,"699":2,"713":1,"716":1,"729":1,"731":1,"756":1}}],["laddr",{"2":{"541":1}}],["ladder",{"2":{"16":1}}],["lang",{"2":{"531":2}}],["languages",{"2":{"121":1,"149":1}}],["language",{"2":{"7":1,"12":1}}],["lag",{"2":{"462":1}}],["laid",{"2":{"176":1,"179":1}}],["latter",{"2":{"172":1,"414":1}}],["latency",{"2":{"639":4}}],["latestblock",{"2":{"213":1,"218":5}}],["latestblocknumber",{"2":{"129":4,"146":4}}],["latestrollupheight",{"2":{"111":2,"113":6}}],["latest",{"2":{"80":1,"81":1,"89":1,"111":2,"125":1,"201":2,"206":1,"212":5,"218":1,"252":2,"277":1,"340":2,"366":6,"368":1,"394":1,"438":2,"461":1,"462":2,"526":1,"539":6,"540":1,"567":6,"568":1,"570":1,"607":1,"614":6,"616":6,"617":6,"632":1,"637":1,"725":1,"756":2}}],["later",{"2":{"34":1,"200":1,"400":1,"415":1,"416":1,"539":1,"656":1,"731":1,"733":1}}],["large",{"2":{"156":1,"202":1,"282":2}}],["larger",{"2":{"34":1,"36":1,"282":1,"408":1,"502":2}}],["layout",{"2":{"112":1,"122":2,"153":1}}],["layers",{"2":{"406":2,"412":1,"413":7,"436":2}}],["layer",{"0":{"406":1},"1":{"407":1,"408":1,"409":1,"410":1,"411":1,"412":1},"2":{"76":1,"92":2,"96":1,"163":2,"206":1,"212":1,"221":1,"222":1,"223":1,"237":3,"238":3,"254":1,"276":1,"279":1,"284":5,"304":1,"306":1,"406":3,"408":2,"410":4,"412":4,"413":6,"415":1,"416":1,"434":1,"454":1,"476":2,"477":1,"580":1,"716":1,"732":2}}],["label",{"2":{"503":2,"647":1}}],["labels",{"2":{"503":4}}],["labeled",{"2":{"47":1}}],["labs",{"2":{"23":1,"37":2,"38":2,"39":2,"40":5,"41":10,"58":1,"66":2,"78":1,"221":1,"445":1,"461":1,"714":1,"715":1}}],["lastly",{"2":{"503":1}}],["lastblock",{"2":{"113":6}}],["last",{"2":{"26":1,"35":1,"37":1,"193":1,"201":1,"328":6,"360":1,"499":2,"509":1}}],["law",{"2":{"24":1}}],["launching",{"2":{"36":1}}],["launch",{"0":{"20":1,"73":1},"2":{"72":1,"73":1,"222":1,"414":1,"446":2,"454":1}}],["legacy",{"2":{"494":3,"495":2}}],["lemongrass",{"0":{"716":1},"2":{"475":1,"698":1,"713":1,"716":3,"729":1}}],["lemonade",{"2":{"35":1}}],["ledger",{"2":{"187":2,"393":1}}],["len",{"2":{"113":8,"115":6,"129":2,"141":4,"143":2,"144":4,"145":2,"148":12,"271":2,"272":2,"357":2,"358":2,"360":2,"364":2,"365":2,"382":4}}],["length",{"2":{"110":2}}],["left",{"2":{"111":1,"164":2,"172":1,"176":1,"179":1,"403":1}}],["lets",{"2":{"241":1}}],["let",{"2":{"41":3,"72":1,"107":1,"270":2,"310":1,"311":1,"314":1,"324":1,"325":1,"328":1,"350":1,"352":1,"357":1,"364":12,"365":12,"366":4,"381":2,"573":1}}],["leveraging",{"2":{"644":1}}],["leverages",{"2":{"103":1}}],["leverage",{"2":{"37":1,"38":1,"39":1,"695":1}}],["leveldb",{"2":{"604":1}}],["levels",{"2":{"300":1}}],["level",{"2":{"6":1,"107":1,"109":1,"122":1,"184":1,"200":1,"300":1,"396":1,"447":1,"461":1,"504":1,"639":2}}],["lessons",{"2":{"32":1}}],["less",{"2":{"23":1,"24":1,"154":1,"168":1,"238":1,"438":1,"683":1,"714":1}}],["leave",{"2":{"736":1}}],["leaves",{"2":{"103":2,"141":6,"144":2,"415":1}}],["leaving",{"2":{"683":1}}],["leapwallet",{"2":{"448":1}}],["leap",{"0":{"390":1,"424":1,"425":1},"1":{"425":1,"426":1,"427":1},"2":{"388":1,"390":3,"417":1}}],["leafs",{"2":{"410":1}}],["leaf",{"2":{"127":2,"132":2,"137":2,"144":2}}],["least",{"2":{"23":1,"35":1,"39":1,"67":1,"69":1,"196":1,"348":1,"397":1,"407":1,"409":1,"416":2,"507":2,"595":1,"659":1}}],["lead",{"2":{"13":1,"14":1,"37":1}}],["leaders",{"2":{"6":1,"8":2,"10":2,"11":1,"12":1}}],["learned",{"2":{"32":1,"526":1,"533":1}}],["learning",{"2":{"3":1,"7":1,"40":1,"42":2,"286":1}}],["learn",{"2":{"3":1,"40":1,"46":1,"50":1,"64":1,"66":2,"71":1,"79":1,"92":1,"98":1,"110":1,"122":1,"148":1,"212":1,"222":1,"223":1,"238":1,"252":1,"258":1,"276":1,"278":1,"279":1,"295":1,"314":1,"322":1,"349":1,"350":2,"371":1,"380":1,"383":1,"417":1,"434":1,"440":2,"443":2,"444":1,"455":1,"456":1,"465":1,"475":1,"489":1,"511":1,"525":1,"532":1,"539":1,"540":2,"567":1,"568":1,"612":1,"643":1,"684":1,"695":1,"698":1,"713":1,"717":1,"729":1}}],["lost",{"2":{"560":1,"615":1}}],["lose",{"2":{"509":1,"536":1}}],["loses",{"2":{"23":1}}],["locked",{"2":{"389":2,"446":1}}],["lock",{"2":{"242":1,"554":1}}],["located",{"2":{"112":1,"486":1,"634":1,"675":1}}],["locate",{"2":{"110":1}}],["location",{"0":{"553":1},"1":{"554":1,"555":1,"556":1,"557":1,"558":1,"559":1},"2":{"23":1,"26":1,"35":2,"58":1,"110":2,"112":1,"113":1,"157":1,"158":1,"159":3,"172":1,"318":1,"488":4,"515":1,"555":10,"573":1,"660":1,"670":1}}],["localroot",{"2":{"253":1}}],["localized",{"2":{"143":1}}],["localhost",{"2":{"77":1,"78":3,"128":2,"129":2,"269":2,"315":1,"340":2,"341":2,"352":1,"360":2,"363":2,"502":2,"503":4,"541":3,"552":3,"601":2,"736":2}}],["locally",{"2":{"66":1,"188":2,"197":1,"373":1,"517":1,"547":1}}],["local",{"0":{"196":1,"202":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,"34":1,"36":1,"37":2,"38":2,"39":3,"40":2,"47":1,"62":1,"64":1,"66":1,"72":1,"78":1,"79":1,"193":2,"196":5,"202":2,"220":1,"247":1,"253":1,"366":2,"369":2,"491":2,"511":1,"516":1,"519":2,"520":4,"525":1,"526":1,"533":1,"540":1,"547":1,"568":1,"585":2,"607":4,"616":2,"617":2,"623":21,"705":1,"745":1,"756":4}}],["loop",{"2":{"196":1}}],["looks",{"2":{"71":1,"321":1}}],["looking",{"2":{"42":1,"66":2,"78":1,"145":1,"212":1,"223":1,"341":1,"352":1,"611":1,"612":1}}],["look",{"2":{"40":1,"77":1,"89":1,"107":1,"109":1,"129":2,"146":4,"148":2,"187":1,"252":1,"253":1,"311":1,"318":1,"321":1,"324":1,"331":1,"340":1,"341":1,"342":1,"407":1,"412":1,"416":1,"522":1,"542":2,"563":1,"616":1,"617":1,"660":1}}],["load",{"2":{"62":1,"246":1,"356":2,"357":3,"358":2,"382":2,"603":1,"604":1,"605":1}}],["log=debug",{"2":{"200":2}}],["log",{"2":{"132":2,"352":1,"356":2,"503":1,"522":2,"562":1,"563":2,"597":1,"639":2,"745":2,"751":3}}],["logging",{"2":{"80":1}}],["logged",{"2":{"75":1}}],["logsbloom",{"2":{"340":2}}],["logs",{"0":{"499":1},"2":{"64":1,"78":1,"195":1,"196":1,"343":1,"499":7,"522":2,"557":1,"563":1,"595":1,"736":1,"738":1,"739":1,"740":1,"745":2,"751":2}}],["logos",{"2":{"41":1}}],["logo",{"2":{"38":1,"41":1,"52":1,"390":1}}],["login",{"2":{"503":1}}],["logic",{"2":{"94":2,"97":1,"111":1,"164":1,"199":1,"214":2,"242":1,"412":1,"477":1}}],["logical",{"2":{"37":1}}],["logistics",{"0":{"33":1,"57":1},"1":{"34":1,"35":1,"36":1,"37":1,"38":1,"39":1,"40":1,"41":1,"42":1},"2":{"30":1,"33":1,"37":1}}],["lowers",{"2":{"71":1}}],["lower",{"2":{"71":2}}],["low",{"2":{"26":1,"122":1,"254":1,"283":1,"385":1,"389":2,"391":1}}],["lot",{"2":{"26":1,"33":1,"34":1,"51":1,"184":1}}],["longer",{"2":{"71":1,"282":1,"348":1,"393":1,"409":1,"454":1,"505":1}}],["long",{"2":{"22":1,"72":1,"407":1,"437":1,"438":1,"441":1}}],["love",{"2":{"2":1,"42":1}}],["ll",{"2":{"0":1,"35":1,"50":1,"57":1,"58":1,"61":1,"62":1,"64":1,"69":1,"71":1,"72":1,"79":1,"94":1,"252":1,"253":1,"262":1,"314":1,"340":1,"350":4,"360":1,"369":1,"390":1,"391":1,"393":1,"526":1,"555":1,"597":1,"607":1,"616":1,"656":1,"751":1,"756":1}}],["wget",{"2":{"602":2,"622":10,"623":8}}],["wsl",{"2":{"620":2}}],["ws",{"2":{"226":1,"269":2,"360":2,"363":2,"639":4,"690":2}}],["wss",{"0":{"226":1},"2":{"75":6,"76":6,"226":1,"690":3}}],["w",{"2":{"199":2,"358":4,"359":2}}],["wrong",{"0":{"562":1,"607":1,"756":1},"2":{"562":2,"607":2,"756":2}}],["wrapped",{"2":{"200":1,"665":1}}],["wrappers",{"2":{"148":2}}],["wrapper",{"2":{"129":4,"146":2,"196":2,"199":4,"200":2}}],["wrapping",{"0":{"360":1},"2":{"111":1}}],["wraps",{"2":{"111":2,"199":1}}],["writing",{"2":{"111":2,"119":1}}],["write",{"0":{"296":1},"2":{"33":1,"253":1,"296":2,"350":1,"519":2,"720":1}}],["written",{"2":{"12":1,"206":1,"284":2,"623":1}}],["won",{"2":{"75":1,"76":1,"168":1,"193":1,"346":1,"725":1}}],["wonderful",{"2":{"33":1}}],["worry",{"2":{"94":1,"612":1}}],["world",{"2":{"43":1,"270":4,"350":1,"364":4,"381":4,"678":1}}],["words",{"2":{"105":1,"159":1,"201":1,"400":1,"413":1}}],["word",{"2":{"36":1,"39":1}}],["workshop",{"0":{"53":1,"54":1,"55":1,"56":1}}],["workshops",{"2":{"34":1,"37":1}}],["works",{"0":{"219":1},"2":{"35":2,"81":1,"374":1,"376":1,"377":1,"396":1,"440":1,"455":1,"523":1,"548":1}}],["working",{"0":{"654":1,"655":1},"2":{"30":1,"34":2,"245":2,"267":1,"285":1,"502":1,"567":1,"654":1}}],["work",{"2":{"21":1,"23":1,"35":1,"107":1,"119":1,"154":1,"237":1,"280":1,"352":1,"410":1,"470":1,"714":1}}],["wouldn",{"2":{"398":1}}],["would",{"2":{"34":1,"39":1,"42":5,"89":1,"107":1,"109":1,"111":9,"112":1,"115":3,"168":2,"170":1,"176":2,"179":1,"188":1,"210":1,"211":1,"269":1,"302":1,"307":1,"312":1,"318":1,"331":1,"344":1,"363":1,"369":1,"389":1,"465":1,"494":1,"512":1,"522":1,"542":1,"552":1,"585":1,"597":1,"612":1,"614":2,"635":1,"684":1,"699":2}}],["www",{"2":{"16":3,"466":1,"685":1,"703":1}}],["walkthrough",{"0":{"618":1},"1":{"619":1}}],["wallet=",{"2":{"745":2,"751":2}}],["wallets",{"0":{"312":1,"640":1,"672":1},"1":{"673":1},"2":{"388":1,"417":1,"458":1,"536":1,"640":3}}],["wallet",{"0":{"243":1,"388":1,"418":1,"419":1,"424":1,"428":1,"429":1,"489":1,"527":1,"534":1,"536":1,"537":1,"744":1},"1":{"244":1,"245":1,"246":1,"247":1,"248":1,"249":1,"250":1,"251":1,"252":1,"253":1,"389":1,"390":1,"391":1,"420":1,"421":1,"422":1,"423":1,"425":1,"426":1,"427":1,"429":1,"430":1,"431":1,"432":1,"535":1,"536":1,"537":1},"2":{"62":1,"64":1,"65":1,"67":1,"70":3,"72":2,"187":6,"243":1,"258":1,"295":1,"298":1,"300":1,"302":1,"312":5,"384":2,"389":5,"390":2,"391":2,"448":2,"449":1,"484":1,"493":1,"494":6,"495":2,"496":2,"497":4,"498":2,"500":1,"534":1,"536":3,"537":2,"591":1,"632":1,"641":1,"672":1,"673":2,"744":1,"745":8,"751":4}}],["wanting",{"2":{"300":1}}],["want",{"2":{"187":9,"188":1,"190":1,"193":1,"195":1,"198":1,"204":1,"232":1,"238":1,"241":1,"246":1,"273":2,"302":1,"314":2,"366":2,"377":1,"382":4,"384":1,"390":1,"396":1,"404":1,"494":1,"502":2,"536":1,"548":1,"576":2,"585":1,"591":2,"592":2,"597":1,"610":1,"639":1,"649":2,"653":1,"658":1,"699":1,"725":1,"733":1,"745":2,"751":2}}],["wantedby=multi",{"2":{"736":2,"738":2,"739":2,"740":2}}],["wanted",{"2":{"132":2,"522":2,"745":2,"751":2}}],["wants",{"2":{"121":1,"653":1}}],["waiting",{"2":{"78":6,"111":1,"113":1,"115":3,"156":2,"352":1,"382":2}}],["wait",{"2":{"76":1,"94":1,"115":1,"148":2,"196":1,"352":1,"377":1,"382":2,"426":1,"537":1}}],["watchopts",{"2":{"147":2}}],["watchdatacommitmentstored",{"2":{"147":3}}],["watch",{"2":{"40":1}}],["water",{"2":{"35":2}}],["warn",{"2":{"505":1}}],["warning",{"0":{"13":1},"2":{"12":1,"13":1,"76":1,"137":1,"138":1,"285":1,"302":1,"307":1,"462":1,"467":1,"471":1,"552":1,"564":1,"685":1,"687":1,"703":1,"710":1,"716":1}}],["warehouse",{"2":{"466":1,"685":1,"703":1}}],["warm",{"2":{"39":1}}],["way",{"2":{"38":1,"69":1,"95":1,"125":1,"151":1,"154":2,"155":1,"156":1,"157":1,"168":2,"176":1,"184":1,"188":2,"189":1,"282":1,"286":1,"301":1,"350":1,"403":1,"406":1,"462":1,"494":1,"503":1,"519":2,"536":1,"540":1,"568":1,"612":2,"665":1,"667":1}}],["ways",{"2":{"6":1,"150":1,"215":1,"240":1,"284":1,"297":1,"378":1,"698":1,"713":1,"717":1,"729":1}}],["wasm",{"2":{"81":1}}],["was",{"0":{"129":1},"2":{"12":1,"66":1,"68":1,"78":5,"96":1,"106":1,"107":5,"109":1,"113":1,"115":1,"121":1,"122":1,"124":2,"129":6,"139":6,"140":1,"145":7,"148":2,"149":1,"153":2,"159":1,"165":1,"171":1,"172":1,"176":1,"179":1,"189":2,"190":1,"193":1,"199":1,"200":2,"218":1,"238":1,"252":1,"270":3,"284":1,"319":1,"321":2,"322":1,"358":1,"364":3,"381":2,"382":4,"393":1,"410":1,"415":1,"416":1,"515":1,"517":1,"539":1,"557":1,"596":1,"607":1,"623":1,"641":1,"668":1,"736":1,"738":1,"739":1,"740":1,"745":1,"756":1}}],["windows",{"2":{"613":1,"620":2}}],["window",{"2":{"252":1,"253":1,"352":1,"385":1,"386":1,"387":1,"389":8,"437":5,"509":1,"636":1}}],["wise",{"2":{"415":1}}],["wish",{"2":{"201":1,"212":1,"373":1,"377":1,"438":1}}],["wisdom",{"2":{"50":1}}],["wiki",{"2":{"8":1}}],["willingness",{"2":{"377":1}}],["willing",{"2":{"298":1,"375":1}}],["will",{"2":{"8":2,"10":1,"11":1,"20":2,"21":2,"22":3,"23":2,"25":2,"26":1,"27":1,"28":1,"34":1,"35":1,"36":2,"37":1,"38":1,"41":1,"42":3,"44":1,"46":1,"47":2,"59":1,"64":1,"65":2,"66":2,"69":1,"70":2,"71":1,"72":2,"73":1,"75":3,"76":3,"77":3,"78":1,"89":1,"90":2,"94":5,"102":2,"109":1,"110":1,"111":3,"115":1,"121":1,"124":1,"125":1,"126":1,"129":2,"130":1,"137":1,"138":1,"139":5,"141":1,"146":2,"148":4,"150":1,"151":2,"153":1,"154":4,"156":1,"157":1,"159":4,"160":2,"168":1,"171":1,"172":1,"173":1,"175":1,"176":2,"178":1,"179":3,"185":1,"187":2,"188":2,"189":5,"193":1,"195":3,"199":1,"200":3,"201":1,"204":2,"206":1,"213":4,"220":3,"232":1,"240":2,"243":2,"245":1,"246":4,"252":1,"253":2,"258":4,"259":1,"267":1,"268":1,"269":1,"271":2,"272":2,"282":3,"289":1,"291":1,"302":1,"304":1,"307":1,"310":1,"311":1,"312":2,"313":2,"314":1,"315":1,"316":3,"317":1,"318":1,"319":5,"321":6,"322":3,"324":1,"325":1,"328":3,"331":1,"336":3,"340":1,"341":1,"342":1,"343":1,"344":1,"346":1,"347":1,"348":1,"352":4,"353":1,"355":1,"356":2,"357":2,"358":3,"360":3,"362":1,"363":1,"365":4,"371":2,"372":1,"377":3,"382":4,"384":1,"385":1,"389":4,"390":1,"391":1,"393":2,"394":1,"396":1,"397":1,"403":1,"404":1,"416":1,"417":1,"420":1,"422":1,"423":1,"427":1,"432":1,"435":1,"436":1,"437":5,"438":2,"440":1,"445":1,"446":2,"458":1,"462":2,"465":1,"469":1,"470":2,"476":1,"482":1,"483":1,"484":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,"555":1,"557":2,"558":1,"560":1,"563":1,"564":2,"567":2,"568":3,"573":1,"579":1,"585":1,"587":1,"591":1,"592":3,"594":2,"599":1,"604":2,"605":2,"615":4,"616":1,"617":1,"621":1,"623":1,"624":1,"632":2,"636":1,"639":1,"640":1,"648":1,"650":1,"651":1,"652":1,"655":2,"656":1,"659":3,"660":3,"664":2,"670":2,"672":2,"679":2,"684":1,"692":1,"693":1,"699":1,"704":1,"705":1,"714":1,"715":1,"716":6,"725":1,"727":1,"731":1,"733":2,"736":1,"738":1,"739":2,"740":3,"741":1,"745":1,"751":2}}],["withheld",{"2":{"394":1}}],["withhold",{"2":{"394":1,"416":1}}],["withholding",{"2":{"394":1}}],["withtransportcredentials",{"2":{"382":2}}],["withcancel",{"2":{"357":2,"358":2,"360":2}}],["withfeegranteraddress",{"2":{"270":1}}],["withsigneraddress",{"2":{"270":1}}],["withkeyname",{"2":{"270":1}}],["withgas",{"2":{"270":1}}],["withgasprice",{"2":{"270":2}}],["withdraw",{"0":{"64":1},"2":{"496":2}}],["withdrawalsroot",{"2":{"340":2}}],["withdrawals",{"2":{"71":1}}],["withdrawal",{"0":{"63":1,"65":1},"1":{"64":1,"65":1},"2":{"23":1,"59":2,"64":2,"65":2}}],["within",{"2":{"9":1,"15":1,"20":1,"23":4,"34":1,"37":1,"44":1,"58":1,"71":1,"314":1,"385":1,"386":1,"434":1,"437":3,"620":1}}],["without",{"2":{"7":1,"75":1,"94":1,"122":1,"155":1,"240":1,"256":1,"260":1,"269":1,"317":1,"346":1,"363":1,"395":1,"396":1,"406":1,"414":1,"438":1,"446":1,"522":1,"612":1,"614":1,"694":1}}],["with",{"0":{"79":1,"90":1,"91":1,"99":1,"108":1,"213":1,"214":1,"222":1,"243":1,"289":1,"290":1,"320":1,"344":1,"388":1,"389":1,"419":1,"424":1,"428":1,"468":1,"485":1,"487":1,"499":1,"511":1,"534":1,"615":1,"633":1,"635":1,"674":1,"676":1,"734":1,"736":1,"750":1},"1":{"92":1,"93":1,"94":1,"95":1,"96":1,"97":1,"98":1,"100":1,"101":1,"102":1,"103":1,"104":1,"105":1,"106":1,"107":1,"109":1,"110":1,"111":1,"112":1,"113":1,"114":1,"115":1,"116":1,"117":1,"118":1,"119":1,"214":1,"215":1,"216":1,"244":1,"245":1,"246":1,"247":1,"248":1,"249":1,"250":1,"251":1,"252":1,"253":1,"291":1,"292":1,"345":1,"346":1,"347":1,"348":1,"349":1,"389":1,"390":1,"391":1,"420":1,"421":1,"422":1,"423":1,"425":1,"426":1,"427":1,"429":1,"430":1,"431":1,"432":1,"469":1,"470":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,"616":1,"617":1,"634":1,"675":1,"735":1,"736":1,"737":1,"738":1,"739":1,"740":1},"2":{"1":1,"2":1,"3":2,"13":3,"14":3,"23":2,"24":1,"25":1,"26":2,"30":2,"31":1,"32":2,"34":5,"36":5,"37":7,"38":2,"39":2,"40":5,"41":3,"42":1,"43":1,"44":1,"47":1,"48":2,"49":1,"55":1,"57":1,"58":1,"66":1,"67":1,"69":1,"71":2,"72":1,"75":2,"76":2,"78":6,"81":5,"88":1,"89":2,"92":1,"93":1,"94":3,"95":2,"96":2,"97":1,"107":3,"111":3,"115":1,"119":1,"121":1,"124":1,"127":1,"129":2,"132":1,"135":1,"137":1,"141":2,"142":1,"143":1,"144":1,"145":6,"146":2,"148":2,"149":1,"150":1,"151":1,"157":1,"160":1,"163":1,"168":1,"187":1,"188":1,"193":1,"195":1,"199":1,"201":1,"202":1,"203":2,"204":1,"206":2,"210":1,"212":3,"213":1,"214":3,"216":1,"218":3,"221":3,"222":1,"224":2,"226":1,"235":1,"238":1,"241":1,"242":2,"243":1,"246":1,"251":1,"252":3,"253":5,"254":2,"256":1,"257":2,"258":2,"259":1,"268":1,"270":2,"271":2,"272":2,"277":1,"279":1,"280":1,"283":2,"284":2,"286":1,"288":1,"289":1,"291":1,"295":1,"297":2,"300":2,"301":1,"311":3,"312":4,"313":1,"314":5,"315":1,"316":3,"317":2,"318":2,"319":2,"321":1,"324":1,"331":1,"336":2,"338":1,"340":1,"346":1,"348":4,"349":1,"350":2,"352":2,"353":2,"356":2,"357":5,"358":6,"360":6,"361":1,"362":1,"364":2,"369":1,"371":1,"372":1,"375":1,"377":2,"384":1,"385":1,"394":1,"398":1,"399":1,"400":1,"405":1,"407":2,"408":1,"409":6,"410":2,"412":1,"413":1,"414":1,"415":4,"416":3,"420":1,"434":1,"435":1,"440":1,"447":1,"455":1,"459":1,"461":1,"462":1,"467":3,"470":4,"472":1,"477":3,"483":2,"484":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,"555":3,"560":1,"561":1,"564":2,"567":2,"568":2,"579":1,"585":1,"587":1,"591":1,"597":1,"601":1,"607":1,"609":3,"614":1,"617":1,"620":1,"632":4,"635":1,"638":1,"640":1,"647":1,"649":2,"653":1,"656":1,"658":3,"659":2,"660":2,"662":4,"663":1,"665":1,"667":1,"671":2,"672":3,"676":1,"677":1,"679":2,"683":1,"685":1,"687":1,"693":2,"703":1,"706":1,"707":2,"708":1,"710":1,"714":3,"715":1,"727":1,"730":1,"731":1,"746":1,"750":2,"751":1,"756":2}}],["whole",{"2":{"407":1}}],["whom",{"2":{"136":1}}],["whose",{"2":{"132":1,"187":2}}],["who",{"0":{"404":1},"2":{"21":1,"23":1,"37":1,"39":2,"40":3,"58":1,"222":1,"298":1,"302":1,"318":1,"653":1,"660":1,"725":1}}],["whitelist",{"0":{"191":1},"2":{"190":1,"191":3}}],["whitelisting",{"0":{"190":1},"2":{"187":1,"190":2,"369":1}}],["whitelisted",{"2":{"187":1}}],["whiteboard",{"2":{"34":1}}],["whiteboards",{"2":{"30":1,"34":1}}],["while",{"2":{"18":1,"35":1,"78":1,"141":1,"150":1,"206":1,"212":1,"243":1,"277":1,"282":1,"284":2,"331":1,"352":1,"365":2,"408":1,"413":1,"416":1,"438":1,"563":4,"679":1,"722":1,"732":1}}],["which",{"0":{"608":1},"1":{"609":1,"610":1,"611":1},"2":{"7":1,"26":1,"35":1,"41":1,"42":1,"70":1,"71":3,"73":1,"75":2,"76":1,"77":1,"82":1,"89":1,"93":1,"94":1,"95":3,"96":3,"97":2,"107":1,"110":1,"111":2,"113":1,"122":2,"124":4,"126":1,"127":1,"132":1,"134":1,"136":1,"137":1,"139":2,"141":2,"142":1,"143":1,"145":5,"151":1,"153":1,"159":1,"164":1,"165":1,"169":1,"172":1,"176":1,"178":1,"179":1,"184":1,"187":3,"189":3,"190":1,"191":3,"196":2,"198":2,"199":3,"200":2,"203":1,"204":1,"206":2,"209":1,"210":1,"219":1,"220":1,"246":1,"252":1,"270":1,"277":1,"284":1,"297":1,"301":1,"302":1,"311":1,"319":4,"321":4,"322":1,"324":1,"343":1,"358":2,"359":1,"374":1,"394":1,"402":1,"403":1,"413":1,"434":2,"438":1,"440":1,"456":1,"465":1,"482":1,"483":1,"484":1,"494":1,"503":2,"537":1,"539":1,"542":1,"548":1,"550":1,"594":1,"601":2,"604":2,"608":1,"611":1,"614":1,"624":1,"632":2,"637":1,"638":1,"648":1,"660":2,"662":1,"663":1,"664":1,"671":2,"683":4,"684":1,"699":1,"707":3,"714":3,"715":1,"731":1,"733":1,"736":2,"738":2,"739":2,"740":2,"748":1,"753":1}}],["why",{"0":{"398":1,"403":1},"2":{"12":1,"26":1,"42":5,"58":1,"199":1,"714":1}}],["whenever",{"2":{"39":1,"254":1,"699":1}}],["when",{"2":{"8":1,"9":1,"36":1,"42":1,"44":1,"58":1,"75":1,"76":2,"94":2,"122":1,"124":1,"129":2,"142":1,"146":2,"148":2,"154":1,"160":1,"164":1,"171":1,"172":2,"190":1,"199":1,"204":1,"252":1,"262":1,"269":1,"276":1,"284":1,"312":1,"314":1,"315":1,"348":1,"363":1,"375":1,"376":1,"393":2,"394":1,"410":1,"446":1,"494":1,"502":2,"503":2,"541":1,"547":1,"551":1,"552":1,"555":1,"557":1,"558":1,"559":1,"563":2,"585":2,"602":1,"604":2,"616":1,"659":1,"673":1,"715":1}}],["whereas",{"2":{"71":1}}],["where",{"0":{"402":1},"2":{"7":1,"94":2,"95":1,"112":1,"132":1,"139":2,"143":1,"186":1,"195":1,"196":1,"310":1,"312":1,"314":1,"350":1,"352":1,"369":1,"371":1,"373":1,"374":1,"375":1,"385":1,"398":1,"409":1,"415":1,"423":1,"427":1,"432":1,"437":1,"438":3,"472":1,"477":1,"489":1,"531":2,"559":1,"564":1,"636":1,"673":1,"678":1,"710":1,"715":1}}],["whether",{"2":{"2":1,"36":1,"385":1,"389":3,"394":1,"400":1,"406":1,"605":1}}],["whatever",{"0":{"232":1},"1":{"233":1,"234":1,"235":1,"236":1,"237":1,"238":1,"239":1,"240":1,"241":1,"242":1},"2":{"276":1,"536":1}}],["what",{"0":{"206":1,"212":1,"218":1,"237":1,"238":1,"393":1,"394":1,"396":1,"397":1,"399":1,"400":1,"401":1,"405":1},"1":{"207":1},"2":{"7":1,"21":1,"42":4,"53":1,"54":1,"55":1,"64":1,"72":1,"109":1,"112":1,"115":1,"159":1,"196":2,"232":1,"318":1,"321":1,"331":1,"360":2,"436":1,"461":1,"540":1,"544":1,"568":1,"623":1}}],["went",{"2":{"745":1}}],["we+s5gft6g944xbkvvygqb9oy+u",{"2":{"519":2,"520":2}}],["week",{"2":{"472":1,"710":1}}],["weeks",{"2":{"39":1,"193":1,"593":1,"594":1}}],["weak",{"2":{"400":1,"595":1}}],["weth",{"2":{"78":2}}],["webhook",{"2":{"505":1}}],["web",{"0":{"473":1},"2":{"473":1,"503":1}}],["websocket",{"0":{"690":1},"2":{"75":5,"121":2,"128":2,"129":2,"639":4,"690":3}}],["websites",{"2":{"697":1,"711":1}}],["website",{"2":{"26":1,"40":1,"75":1,"389":2}}],["web3",{"2":{"42":1,"44":1}}],["were",{"2":{"16":1,"107":5,"139":6,"140":1,"201":1,"212":1,"361":1,"446":1}}],["well",{"2":{"13":1,"34":1,"35":2,"58":2,"94":1,"95":1,"187":1,"242":1,"280":1,"298":1,"436":1,"632":1,"638":1}}],["welcoming",{"2":{"6":1,"34":1,"46":1}}],["welcome",{"2":{"0":1,"20":1,"31":1,"41":2,"42":1,"43":1,"48":1,"51":1,"350":1,"678":1}}],["we",{"2":{"2":1,"4":1,"6":2,"23":1,"34":1,"42":8,"46":2,"58":2,"70":2,"71":1,"75":1,"90":1,"94":1,"102":2,"107":1,"110":4,"111":3,"112":1,"115":4,"122":1,"124":5,"126":1,"129":2,"130":1,"139":5,"140":2,"141":6,"144":1,"146":2,"148":4,"149":1,"154":1,"157":1,"158":1,"159":4,"160":5,"171":1,"187":4,"189":4,"190":1,"193":2,"194":2,"195":10,"196":1,"199":1,"200":1,"241":1,"245":1,"252":1,"253":2,"271":1,"272":1,"276":1,"277":2,"278":1,"279":1,"281":1,"282":1,"285":1,"304":1,"314":1,"319":6,"321":2,"324":1,"328":1,"336":1,"340":1,"348":2,"350":1,"352":3,"356":4,"357":8,"358":26,"359":1,"360":11,"365":3,"369":1,"377":1,"382":12,"389":5,"436":1,"461":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,"555":1,"569":2,"574":1,"599":1,"616":2,"623":1,"639":1,"640":2,"642":1,"644":1,"645":2,"655":2,"656":1,"679":1,"699":1,"705":1,"730":1,"733":1,"739":1}}],["dteam",{"2":{"687":1,"688":1,"689":1,"690":1,"693":1,"696":1,"705":2,"706":1,"707":1,"708":1,"712":1}}],["dl",{"2":{"623":8}}],["db",{"2":{"607":1,"756":1}}],["dbs",{"2":{"607":1,"756":1}}],["dbbackend",{"2":{"604":1}}],["d6",{"2":{"410":1}}],["d5",{"2":{"410":2}}],["d4",{"2":{"410":2}}],["d3",{"2":{"410":1}}],["dns4",{"2":{"467":4}}],["dns",{"2":{"336":1}}],["d85c907ce660878a8203ac74baa147ccc1f87114b45b568b72ad207b62afe45e",{"2":{"328":2}}],["dynamic",{"2":{"241":1,"374":1,"375":1}}],["d7wrs",{"2":{"132":2}}],["dcproof",{"2":{"128":2,"129":2}}],["dfcelestia",{"2":{"89":1}}],["drafted",{"2":{"682":1}}],["drawing",{"2":{"50":1}}],["drift",{"2":{"639":4}}],["drinks",{"2":{"35":4}}],["drop",{"2":{"377":1,"445":1}}],["dropped",{"2":{"377":1}}],["dropdown",{"2":{"61":1}}],["d",{"2":{"41":2,"76":1,"77":1,"115":4,"270":2,"271":4,"272":4,"331":1,"337":1,"342":2,"358":4,"381":2,"445":1,"446":2,"511":1,"516":1,"522":2,"596":6}}],["daemon",{"2":{"554":1,"563":3,"734":1,"736":6,"738":3,"739":3,"740":4,"746":1}}],["darwin",{"2":{"540":1,"568":1,"623":12}}],["daring",{"2":{"519":2}}],["dah",{"2":{"328":2,"329":2,"334":2}}],["dac",{"2":{"210":1,"211":1}}],["dacs",{"0":{"208":1},"1":{"209":1,"210":1,"211":1},"2":{"209":1,"211":2}}],["daverifier",{"0":{"139":1,"148":1},"1":{"140":1,"141":1,"142":1,"143":1,"144":1,"145":1,"146":1,"147":1,"148":1},"2":{"107":3,"139":1,"148":11,"159":1}}],["daser",{"2":{"564":1}}],["dashboard",{"2":{"420":2,"421":1,"423":1,"427":1,"470":1,"503":5,"548":1,"693":2}}],["dashboards",{"2":{"23":1,"440":1,"503":1,"751":1}}],["das",{"0":{"407":1},"1":{"408":1,"409":1},"2":{"95":1,"206":1,"330":2,"396":1,"406":2,"407":2,"408":4,"409":1,"416":4,"664":1,"665":1,"740":1}}],["da",{"0":{"91":1,"94":1,"205":1,"411":1,"468":1,"469":1,"470":1,"610":1,"692":1,"693":1,"694":1,"702":1,"704":1,"705":1},"1":{"92":1,"93":1,"94":1,"95":1,"96":1,"97":1,"98":1,"206":1,"207":1,"208":1,"209":1,"210":1,"211":1,"212":1,"213":1,"214":1,"215":1,"216":1,"412":1,"469":1,"470":1,"695":1,"703":1,"704":1,"705":1},"2":{"76":3,"81":1,"90":1,"93":1,"94":1,"97":1,"98":1,"109":1,"111":4,"112":1,"115":1,"163":2,"171":1,"206":2,"212":3,"222":1,"244":1,"254":1,"257":1,"258":4,"276":1,"283":1,"284":8,"298":2,"300":2,"302":1,"306":1,"342":2,"347":1,"348":1,"349":1,"406":3,"408":2,"410":4,"412":3,"413":6,"416":5,"467":5,"469":1,"470":5,"477":1,"564":1,"573":1,"610":1,"664":1,"665":1,"692":1,"693":7,"694":1,"704":1,"705":2,"709":8,"716":1,"731":2,"732":1}}],["dao",{"2":{"71":1}}],["day",{"2":{"39":1,"437":2}}],["days",{"2":{"35":1,"437":2,"509":1}}],["dat2024",{"2":{"199":1}}],["dat",{"2":{"199":31}}],["datalen",{"2":{"157":2,"159":1}}],["datahash",{"2":{"129":2,"148":2}}],["datacommitmentstored",{"2":{"193":1,"204":1}}],["datacommitment",{"2":{"129":2,"146":2}}],["datasharelen",{"2":{"110":4}}],["datasharestart",{"2":{"110":4}}],["datarootinclusionproof",{"2":{"128":2,"129":2,"145":9,"148":10}}],["dataroot",{"2":{"103":4,"124":2,"129":2,"145":5,"148":2}}],["dataroottupleroot",{"2":{"126":1}}],["dataroottuples",{"2":{"104":5,"105":1}}],["dataroottuple",{"2":{"103":4,"104":2,"105":1,"107":1,"126":1,"129":2,"139":2,"145":7,"148":2}}],["dataavailabilityprovider",{"2":{"94":2}}],["data",{"0":{"104":1,"107":1,"112":1,"118":1,"126":1,"140":1,"147":1,"159":1,"160":1,"184":1,"201":1,"208":1,"273":1,"319":1,"321":1,"330":1,"341":1,"342":1,"366":1,"370":1,"392":1,"393":1,"394":1,"395":1,"396":1,"397":1,"399":1,"400":1,"401":2,"403":1,"404":1,"405":1,"406":1,"407":1,"409":1,"412":1,"416":1,"436":1,"437":1,"469":1,"470":1,"561":1,"589":1,"677":1,"691":1,"692":1,"693":1,"704":1,"705":1,"719":1,"737":1},"1":{"127":1,"128":1,"202":1,"203":1,"204":1,"209":1,"210":1,"211":1,"320":1,"371":1,"372":1,"373":1,"374":1,"375":1,"376":1,"377":1,"378":1,"379":1,"380":1,"381":1,"382":1,"383":1,"384":1,"393":1,"394":1,"395":1,"396":1,"397":1,"398":1,"399":1,"400":1,"401":1,"402":1,"403":1,"404":1,"405":1,"407":1,"408":2,"409":2,"410":1,"411":1,"412":1,"437":1,"438":1,"692":1,"693":1,"694":1,"695":1,"738":1,"739":1,"740":1},"2":{"26":1,"37":1,"42":1,"53":1,"66":1,"70":5,"75":3,"76":1,"92":2,"94":10,"95":8,"96":5,"97":6,"103":5,"104":1,"105":1,"106":1,"107":8,"109":2,"110":31,"111":7,"112":7,"113":16,"115":10,"118":1,"121":1,"122":5,"124":9,"126":6,"127":2,"129":10,"130":3,"132":5,"134":1,"136":3,"137":2,"139":14,"140":2,"142":1,"143":1,"144":1,"145":31,"147":2,"148":14,"149":3,"150":4,"152":2,"153":8,"154":3,"155":2,"157":9,"158":1,"159":7,"160":19,"163":2,"164":7,"165":1,"167":2,"168":3,"171":2,"172":8,"175":1,"176":5,"179":5,"184":6,"199":9,"206":3,"209":2,"210":4,"211":2,"212":5,"213":1,"216":1,"218":1,"222":2,"224":1,"238":3,"241":2,"254":2,"256":1,"266":1,"273":1,"279":3,"280":1,"283":2,"284":14,"304":2,"306":1,"307":3,"310":1,"319":9,"321":10,"328":2,"336":4,"341":1,"344":1,"346":2,"350":3,"356":2,"357":2,"358":5,"360":5,"364":2,"366":1,"370":2,"379":2,"380":2,"382":8,"383":1,"384":2,"393":11,"394":7,"395":7,"396":9,"397":2,"398":6,"399":4,"400":6,"401":4,"402":3,"403":9,"404":4,"405":6,"406":5,"407":11,"408":3,"409":11,"410":14,"412":5,"413":2,"414":6,"415":15,"416":9,"434":9,"436":7,"437":7,"438":12,"453":1,"454":1,"466":1,"467":1,"476":1,"477":3,"503":4,"522":2,"543":1,"544":1,"548":2,"558":2,"561":1,"562":1,"563":4,"568":1,"587":1,"589":2,"592":3,"593":2,"596":19,"599":1,"600":2,"607":1,"609":3,"615":3,"624":1,"625":1,"664":2,"665":1,"679":1,"685":1,"694":5,"699":1,"703":1,"717":5,"731":2,"740":1,"745":2,"746":2,"751":2,"756":2}}],["date",{"2":{"25":1,"30":1,"567":1,"716":1}}],["duty",{"2":{"519":2}}],["dubbing",{"2":{"412":1}}],["dumps",{"2":{"245":2}}],["dummy",{"2":{"199":3}}],["dues",{"2":{"41":4}}],["due",{"2":{"23":1,"58":1,"76":1,"254":1,"374":1,"377":1,"406":1,"413":1,"416":1,"550":1}}],["duration",{"2":{"22":3,"27":1}}],["during",{"0":{"31":1},"2":{"14":1,"22":1,"24":1,"31":1,"35":1,"38":1,"145":1,"199":1,"212":1,"260":1,"288":1,"385":1,"397":1,"509":1,"540":1,"568":1,"576":1,"579":1,"597":1,"714":1,"751":1}}],["double",{"2":{"509":2,"560":1}}],["domain",{"2":{"336":1}}],["dos",{"2":{"300":1}}],["doing",{"2":{"115":1,"141":1,"238":1,"339":1,"540":1,"568":1}}],["downside",{"2":{"409":1}}],["down",{"2":{"245":1,"376":1,"390":1,"391":1,"573":1,"715":1}}],["downtime",{"2":{"97":1,"254":2,"509":1}}],["downloads",{"2":{"503":1,"567":1,"593":4,"596":1}}],["downloading",{"0":{"115":1},"2":{"115":1,"188":1,"395":1,"409":2,"622":1}}],["downloaded",{"0":{"198":1},"1":{"199":1,"200":1},"2":{"73":1,"74":1,"188":2,"196":1,"198":1,"199":1,"200":2,"394":1,"438":1,"593":1}}],["download",{"0":{"73":1},"2":{"72":1,"73":3,"111":1,"115":10,"188":2,"196":1,"199":16,"282":1,"307":1,"308":1,"355":1,"393":3,"395":2,"396":1,"406":2,"407":1,"408":3,"409":1,"410":1,"412":1,"438":1,"478":1,"512":1,"540":1,"568":1,"581":1,"585":7,"593":1,"607":1,"623":1,"626":1,"666":1,"705":2,"716":3,"742":1,"756":1}}],["do",{"0":{"395":1,"405":1},"2":{"42":1,"69":1,"109":1,"144":1,"184":1,"189":1,"195":1,"201":1,"238":1,"246":2,"252":1,"254":1,"279":1,"311":1,"312":1,"357":1,"373":1,"390":1,"395":1,"398":1,"436":1,"437":1,"438":1,"462":1,"467":1,"470":1,"483":1,"484":1,"503":2,"535":1,"548":1,"550":1,"592":1,"601":1,"620":1,"632":1,"636":1,"640":1,"642":1,"649":1,"654":1,"685":1,"687":1,"703":1,"705":1,"716":1,"731":1,"740":1}}],["done",{"2":{"78":10,"109":1,"112":1,"130":1,"140":1,"147":2,"160":2,"170":1,"271":2,"272":2,"377":1,"386":1,"479":1,"494":1,"582":1,"627":1,"653":1,"727":1,"743":1}}],["don",{"2":{"26":1,"34":1,"39":1,"40":1,"64":1,"94":2,"187":2,"218":1,"240":1,"241":1,"242":1,"296":1,"302":1,"344":1,"382":2,"398":1,"403":1,"527":1,"597":1,"604":1,"642":1,"733":1,"751":1}}],["docker",{"0":{"250":1,"612":1},"1":{"251":1,"252":1,"253":1,"613":1,"614":1,"615":1,"616":1,"617":1,"618":1,"619":1,"620":1},"2":{"67":2,"76":7,"77":2,"251":2,"252":3,"253":4,"343":2,"612":7,"613":4,"615":1,"616":2,"617":2}}],["docs",{"2":{"23":1,"75":1,"89":2,"117":1,"148":1,"153":1,"213":1,"349":1,"383":1,"466":2,"505":1,"508":1,"540":2,"548":1,"568":2,"595":1,"685":2,"703":2}}],["documented",{"2":{"147":1,"193":2}}],["documents",{"2":{"89":2}}],["document",{"2":{"22":1,"32":1,"124":1,"150":1,"185":1,"275":1,"288":1,"368":1,"436":1,"507":6,"682":1}}],["documentation",{"0":{"0":1,"116":1,"274":1,"367":1},"1":{"1":1,"2":1,"3":1,"4":1,"117":1,"118":1,"119":1},"2":{"0":1,"1":1,"3":1,"4":1,"23":1,"31":1,"66":1,"80":1,"106":1,"107":1,"111":2,"112":3,"148":1,"160":2,"167":1,"187":1,"188":1,"193":1,"197":1,"202":1,"215":1,"219":1,"220":2,"274":1,"278":1,"297":1,"367":1,"465":1,"489":2,"494":1,"502":2,"532":1,"684":1}}],["doesn",{"0":{"403":1},"2":{"121":1,"159":1,"184":1,"242":1,"315":1,"338":1,"624":1,"638":1}}],["does",{"0":{"402":1},"2":{"22":1,"26":1,"95":1,"186":1,"187":1,"261":1,"262":2,"373":1,"375":1,"377":1,"409":1,"415":1,"461":1,"471":1,"515":1,"620":1,"707":1,"710":1,"715":1,"717":1}}],["dir",{"2":{"555":3}}],["directed",{"2":{"632":1,"671":1}}],["direct",{"2":{"467":2,"470":1,"505":1}}],["directories",{"2":{"438":1}}],["directory",{"0":{"3":1,"515":1,"654":1,"655":1},"2":{"61":1,"74":1,"79":1,"102":1,"245":2,"246":4,"253":5,"312":2,"315":1,"346":1,"355":1,"356":1,"485":1,"486":1,"500":1,"514":1,"515":4,"517":1,"539":1,"562":1,"567":1,"593":1,"596":1,"607":2,"615":4,"623":1,"633":1,"634":1,"654":2,"660":3,"672":2,"674":1,"675":1,"756":3}}],["directly",{"0":{"384":1},"2":{"111":1,"115":2,"204":1,"210":1,"279":1,"317":1,"373":1,"375":1,"382":2,"384":1,"389":1}}],["dimensional",{"2":{"407":1,"409":4,"412":1,"415":1}}],["digest",{"2":{"141":9,"148":4,"199":2}}],["digital",{"2":{"7":1,"277":1}}],["dial",{"2":{"129":2,"146":2,"147":2,"148":2,"382":2,"559":2,"602":1}}],["diagram",{"2":{"124":1,"219":1,"441":1}}],["diagrams",{"2":{"3":1,"122":1}}],["did",{"2":{"42":2,"196":2}}],["difficulty",{"2":{"34":1,"36":1,"340":2}}],["difference",{"0":{"401":1},"2":{"89":1,"115":1,"160":1,"168":2,"179":1}}],["different",{"0":{"289":1,"290":1},"1":{"291":1,"292":1},"2":{"33":1,"75":1,"76":1,"107":1,"115":1,"149":1,"158":2,"164":1,"167":1,"171":1,"182":1,"187":5,"201":3,"215":1,"218":1,"227":1,"232":1,"288":2,"289":1,"291":1,"292":1,"300":1,"352":1,"361":1,"377":1,"402":2,"413":1,"446":1,"465":1,"494":1,"503":1,"529":1,"551":2,"555":2,"595":1,"602":1,"684":1,"717":1,"733":1}}],["differing",{"2":{"7":1}}],["div",{"2":{"389":4}}],["dividing",{"2":{"373":1}}],["divided",{"2":{"22":1,"25":1}}],["dive",{"0":{"339":1},"1":{"340":1,"341":1,"342":1,"343":1},"2":{"339":1}}],["dives",{"2":{"42":1,"53":1}}],["diverse",{"2":{"6":1}}],["disk",{"2":{"308":1,"478":1,"504":1,"581":1,"605":2,"626":1,"666":1,"719":1,"720":1,"742":1}}],["disruptions",{"2":{"254":1}}],["disruptive",{"2":{"18":1}}],["disables",{"2":{"547":1,"604":1}}],["disable",{"2":{"328":2,"547":2}}],["disabled",{"2":{"191":1,"357":2,"358":2,"360":2}}],["disability",{"2":{"6":1}}],["distribution",{"2":{"277":1,"471":1,"495":2,"496":2,"710":1}}],["distributed",{"2":{"413":1}}],["distribute",{"2":{"21":1}}],["distinguish",{"2":{"71":1}}],["disputes",{"2":{"71":1,"413":1}}],["dispute",{"2":{"71":1,"238":1}}],["displayed",{"2":{"491":1}}],["displaydenom",{"2":{"458":1}}],["display",{"0":{"458":1},"2":{"71":1,"321":1}}],["displaying",{"2":{"31":1}}],["disparagement",{"2":{"15":1}}],["disqualify",{"2":{"23":1}}],["discard",{"0":{"605":1},"2":{"94":1,"96":1,"605":1}}],["disclaimer",{"2":{"64":1}}],["discussed",{"2":{"156":1}}],["discussion",{"2":{"37":2}}],["discussions",{"2":{"31":1,"44":1}}],["discuss",{"2":{"37":1,"40":1,"51":1,"171":1,"436":1}}],["discover",{"2":{"42":1}}],["discovering",{"2":{"37":1,"595":1}}],["discounts",{"2":{"34":1}}],["discord",{"0":{"49":1,"472":1},"2":{"10":1,"26":1,"39":2,"47":3,"48":1,"49":2,"312":2,"472":2,"537":1,"640":3,"673":1,"698":1,"710":2,"713":1,"729":2}}],["discretion",{"2":{"24":1}}],["discrepancies",{"2":{"4":1}}],["derivation",{"2":{"639":4}}],["derogatory",{"2":{"7":1}}],["debug",{"2":{"488":2,"607":4,"756":4}}],["debugging",{"2":{"77":1,"488":2}}],["denoted",{"2":{"415":2}}],["denominator",{"2":{"639":4}}],["denomination",{"0":{"459":1,"460":1},"2":{"459":2}}],["denominations",{"0":{"457":1},"1":{"458":1,"459":1,"460":1}}],["denominated",{"2":{"453":1}}],["denominate",{"2":{"72":1}}],["denom",{"2":{"262":2,"263":2,"267":2,"324":2,"325":2,"390":1,"391":1,"494":2,"522":4,"523":2,"525":4,"639":4}}],["deal",{"2":{"416":1}}],["dealing",{"2":{"409":1}}],["deadbeef",{"2":{"364":2}}],["deadline",{"2":{"23":2,"25":1}}],["deducting",{"2":{"373":1}}],["deducted",{"2":{"263":2,"267":2,"373":1}}],["dedicated",{"2":{"23":1,"31":1,"47":1,"221":1,"236":1}}],["demosubmitdata",{"2":{"382":2}}],["demo",{"0":{"109":1},"2":{"107":1,"109":1,"110":1,"111":4,"382":2}}],["demonstration",{"0":{"125":1,"555":1},"1":{"126":1,"127":1,"128":1,"129":1,"130":1,"131":1,"132":1,"133":1,"134":1,"135":1,"136":1,"137":1,"138":1},"2":{"656":1}}],["demonstrating",{"2":{"7":1,"15":1}}],["demonstrate",{"2":{"38":1,"389":1}}],["degree",{"2":{"95":1,"96":1}}],["defer",{"2":{"147":4,"382":2}}],["defaultestimategas",{"2":{"375":1,"376":2,"382":2}}],["defaultgasprice",{"2":{"358":2,"381":2}}],["default",{"0":{"289":1},"2":{"97":2,"187":3,"190":3,"191":2,"201":1,"241":1,"246":3,"269":1,"270":2,"277":1,"288":2,"289":2,"296":1,"311":1,"315":2,"317":1,"322":2,"324":1,"336":1,"346":2,"358":2,"363":1,"364":3,"372":1,"374":1,"377":1,"381":2,"389":2,"437":1,"469":1,"470":2,"483":1,"485":1,"486":1,"488":2,"502":2,"503":6,"541":1,"542":1,"550":1,"551":1,"552":1,"554":1,"573":1,"574":1,"576":2,"579":1,"589":3,"592":4,"594":1,"601":1,"604":1,"605":1,"633":1,"634":1,"639":4,"662":1,"674":1,"675":1,"692":1,"693":1,"694":1,"704":1,"705":1,"706":1,"707":1,"708":1}}],["defaults",{"2":{"71":1,"311":1,"375":1,"483":1,"494":1,"604":1}}],["definitions",{"2":{"446":2}}],["defining",{"0":{"110":1},"2":{"111":1}}],["defines",{"2":{"751":1}}],["defined",{"2":{"71":1,"111":1,"127":1,"132":1,"141":1,"154":1,"156":1,"157":1,"179":1,"201":1,"446":2,"715":1,"751":1}}],["define",{"2":{"37":1,"110":2,"136":1,"160":1,"164":2,"172":1,"176":1,"179":1,"190":1,"241":1,"708":1}}],["defi",{"2":{"37":1}}],["deeper",{"2":{"447":1}}],["deepen",{"2":{"58":1}}],["deep",{"0":{"339":1},"1":{"340":1,"341":1,"342":1,"343":1},"2":{"42":1,"53":1,"339":1}}],["deemed",{"2":{"12":1}}],["deem",{"2":{"8":1,"11":1}}],["decline",{"2":{"519":2}}],["declare",{"2":{"246":1,"252":1,"312":1,"484":1,"529":1,"551":1,"632":1}}],["decreasing",{"2":{"451":1}}],["decreases",{"2":{"441":1}}],["decentralised",{"0":{"442":1,"456":1},"1":{"443":1,"444":1},"2":{"456":2}}],["decentralized",{"2":{"211":1,"667":1}}],["decentralization",{"0":{"209":1},"2":{"697":1}}],["decouple",{"2":{"413":1}}],["decoupling",{"2":{"241":1,"242":1,"414":1}}],["decoding",{"2":{"142":1,"358":2,"639":4}}],["decode",{"2":{"184":1,"350":1,"358":2,"360":2,"361":1,"382":2}}],["decoder",{"2":{"178":1}}],["decoded",{"2":{"141":4,"321":1}}],["decodestring",{"2":{"133":2,"135":2,"358":2}}],["deciding",{"0":{"608":1},"1":{"609":1,"610":1,"611":1}}],["decide",{"2":{"604":1}}],["decimal",{"2":{"195":1}}],["decimals",{"2":{"71":1,"390":1,"391":1,"451":1}}],["decisions",{"2":{"8":1}}],["deck",{"2":{"38":2}}],["descendants",{"2":{"410":2}}],["desc",{"2":{"338":2}}],["descriptor",{"2":{"563":1}}],["descriptors",{"2":{"563":1}}],["description=celestia",{"2":{"736":2,"738":2,"739":2,"740":2}}],["descriptions",{"2":{"71":2}}],["description",{"0":{"44":1},"2":{"36":1,"38":1,"41":3,"71":1,"374":1,"445":1,"494":6,"495":2,"541":1,"552":1,"715":1}}],["describing",{"2":{"412":1}}],["describes",{"2":{"110":2,"585":1,"637":1,"746":1}}],["described",{"2":{"20":1,"142":1,"172":1,"415":3,"445":1,"446":1}}],["deserialization",{"0":{"176":1},"2":{"176":1,"179":1}}],["deserialize",{"2":{"175":1}}],["deserialized",{"2":{"94":1,"176":2}}],["desktop",{"2":{"76":1,"612":1,"613":1,"615":1}}],["destination",{"2":{"64":1,"537":1,"637":1}}],["desired",{"2":{"69":1,"219":1,"336":1,"426":1,"539":1,"567":1,"623":1}}],["desire",{"2":{"38":1}}],["designs",{"2":{"182":1}}],["design",{"0":{"548":1},"2":{"156":1,"211":1,"242":1,"503":1,"544":1,"548":1,"579":1}}],["designate",{"2":{"31":1}}],["designed",{"2":{"20":1,"43":1,"44":1,"51":1,"211":1,"299":1,"409":1,"699":2,"725":1}}],["delta",{"2":{"343":1}}],["deleting",{"2":{"490":2}}],["deleted",{"2":{"615":1}}],["delete",{"2":{"244":1,"490":2,"562":1,"599":1,"615":1}}],["delegate",{"0":{"497":1,"745":1},"2":{"497":3,"509":1,"745":5}}],["delegated",{"2":{"25":2,"240":1,"509":1,"523":4,"751":1}}],["delegating",{"2":{"447":1,"455":1}}],["delegation=1000000",{"2":{"751":3}}],["delegation=1",{"2":{"500":2}}],["delegations",{"2":{"20":1,"27":1}}],["delegation",{"0":{"17":1,"19":1},"1":{"18":1,"19":1,"20":2,"21":2,"22":2,"23":2,"24":2,"25":1,"26":1,"27":1,"28":1},"2":{"18":1,"22":4,"23":4,"24":1,"26":2,"440":2,"488":2}}],["delegators",{"2":{"440":1,"509":2}}],["delegator",{"2":{"417":1,"440":1}}],["delayedcelestia",{"2":{"529":1}}],["delayed",{"2":{"511":2,"522":7,"529":1}}],["delay",{"2":{"196":1,"639":4,"644":4}}],["delays",{"2":{"71":1}}],["delivering",{"2":{"58":1}}],["delivery",{"2":{"35":1}}],["delivers",{"2":{"18":1}}],["deprecated",{"2":{"137":1,"138":1,"148":2,"302":1,"307":1,"552":1}}],["depth",{"2":{"71":1,"441":1}}],["depositing",{"2":{"80":1}}],["deposited",{"2":{"78":2}}],["deposit",{"0":{"60":1,"62":1},"1":{"61":1,"62":1},"2":{"59":2,"62":2,"78":5,"494":5,"495":2}}],["deployer",{"2":{"187":2,"348":4}}],["deployed",{"0":{"189":1,"216":1},"1":{"190":1,"191":1,"192":1},"2":{"42":1,"71":2,"72":1,"76":1,"78":3,"189":2,"190":1,"196":2,"203":1,"216":1,"218":1,"219":3,"220":1,"302":1,"369":1,"462":1,"585":1,"715":1}}],["deployhelper",{"2":{"85":1,"86":1}}],["deployments",{"0":{"82":1,"85":1,"86":1,"185":1,"368":1},"1":{"83":1,"84":1,"186":1,"187":1,"188":1,"189":1,"190":1,"191":1,"192":1,"193":1,"194":1,"195":1,"196":1,"197":1,"198":1,"199":1,"200":1,"369":1},"2":{"76":1,"82":2,"83":1,"84":1,"187":3,"200":1,"201":1,"206":1,"212":2,"213":1,"218":1,"467":2,"685":2,"687":2,"703":2}}],["deployment",{"0":{"70":1,"71":1,"195":1,"197":1},"2":{"66":2,"70":3,"71":4,"72":1,"187":1,"193":4,"195":4,"197":1,"200":1,"202":1,"216":6,"219":1,"220":1,"369":1}}],["deploy",{"0":{"55":1,"66":1,"72":1,"187":1,"188":1,"193":1,"220":1,"481":1,"746":1},"1":{"67":1,"68":1,"69":1,"70":1,"71":1,"72":1,"73":1,"74":1,"75":1,"76":1,"77":1,"78":1,"79":1,"80":1,"81":1,"82":1,"83":1,"84":1,"85":1,"86":1,"194":1,"195":1,"482":1,"483":1,"484":1,"485":1,"486":1,"487":1,"747":1,"748":1,"749":1,"750":1},"2":{"55":1,"70":1,"71":2,"72":3,"78":4,"88":1,"92":1,"98":1,"185":1,"186":1,"187":8,"188":2,"189":1,"193":1,"195":6,"197":1,"200":2,"202":2,"206":1,"212":1,"220":2,"221":1,"226":1,"235":1,"236":1,"240":2,"283":1,"347":1,"348":4,"369":2,"462":1,"699":1,"725":1,"727":1}}],["deploying",{"0":{"186":1,"240":1,"347":1,"348":1,"369":1},"1":{"187":1,"188":1,"189":1,"190":1,"191":1,"192":1,"193":1,"194":1,"195":1,"196":1,"197":1},"2":{"3":1,"55":1,"66":1,"69":1,"187":1,"189":1,"190":1,"193":1,"195":2,"232":1,"240":2,"348":1,"368":1,"369":1,"454":2,"678":1,"722":1}}],["depends",{"2":{"593":1,"683":1}}],["dependent",{"2":{"119":1,"375":1,"597":1,"751":1}}],["dependency",{"0":{"345":1},"1":{"346":1},"2":{"102":1,"269":1,"363":1}}],["dependencies",{"0":{"309":1,"351":1,"480":1,"583":1,"622":1,"628":1,"651":1},"1":{"310":1,"311":1,"312":1,"352":1,"353":1},"2":{"74":1,"268":1,"309":1,"313":1,"351":1,"362":1,"480":1,"583":1,"608":1,"612":1,"628":1,"651":1,"668":1}}],["depend",{"2":{"76":1,"82":1,"211":1,"394":1,"397":1}}],["depending",{"2":{"34":1,"35":1,"58":1,"156":1,"200":1,"356":1,"374":1,"465":1,"530":1,"540":1,"568":1,"594":1,"595":1,"604":1,"610":1,"684":1,"699":1}}],["dependable",{"2":{"18":1}}],["deter",{"2":{"509":1}}],["deterministic",{"2":{"413":1}}],["determining",{"2":{"11":1,"374":1}}],["determined",{"2":{"113":1,"375":1,"683":3}}],["determines",{"2":{"71":1}}],["determine",{"2":{"30":1,"35":2,"37":1,"42":1,"270":1,"364":1,"446":1,"465":1,"684":1,"699":1}}],["detecting",{"2":{"595":1}}],["detection",{"2":{"554":1}}],["detects",{"2":{"389":1}}],["detected",{"2":{"115":1}}],["detect",{"2":{"111":1}}],["details",{"0":{"26":1,"153":1,"158":1,"463":1,"523":1,"524":1,"680":1,"700":1},"1":{"464":1},"2":{"20":1,"41":1,"62":2,"64":1,"65":2,"72":1,"81":1,"122":1,"124":1,"125":1,"153":1,"154":1,"160":1,"167":1,"168":1,"172":1,"176":1,"179":4,"182":1,"219":1,"302":1,"303":1,"307":1,"375":1,"384":1,"407":1,"410":1,"412":2,"416":1,"426":1,"451":1,"509":1,"519":1,"523":1,"524":1,"552":1,"554":1}}],["detailed",{"2":{"2":1,"21":1,"41":1,"66":1,"200":1,"434":1,"494":1}}],["devs",{"2":{"445":1}}],["devnetl1",{"2":{"343":1}}],["devnet",{"0":{"339":1,"347":1,"461":1,"471":1,"512":1,"513":1,"514":1,"516":1,"518":1,"522":1,"523":1,"524":1,"525":1,"725":1,"726":1},"1":{"340":1,"341":1,"342":1,"343":1,"462":1,"463":1,"464":1,"465":1,"466":1,"467":1,"468":1,"469":1,"470":1,"471":1,"472":2,"473":2,"474":1,"475":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,"726":1},"2":{"98":1,"311":1,"312":1,"343":1,"344":1,"347":2,"389":2,"390":2,"461":1,"462":1,"465":1,"467":1,"470":1,"472":1,"483":1,"507":2,"511":1,"514":1,"515":1,"516":1,"517":4,"525":1,"526":1,"529":1,"533":1,"574":1,"585":2,"632":2,"671":1,"673":1,"699":1,"715":1,"725":2,"733":1,"748":1}}],["devrel",{"2":{"35":1,"37":1}}],["developing",{"2":{"632":1}}],["development",{"0":{"621":1},"1":{"622":1,"623":1},"2":{"66":1,"71":2,"240":1,"445":2,"567":1,"621":2,"699":1,"722":1}}],["developer",{"2":{"23":1,"39":1,"61":1,"151":1,"232":1,"297":1,"387":1,"548":1,"649":1}}],["developers",{"2":{"3":1,"71":1,"92":2,"206":2,"213":1,"221":1,"223":2,"224":1,"232":1,"234":1,"241":1,"242":3,"256":1,"283":1,"298":1,"388":1,"434":1,"436":1,"438":1,"445":1,"453":2,"454":3,"461":1,"462":1,"699":1,"725":1,"727":1}}],["develop",{"2":{"23":1,"38":1}}],["devops",{"2":{"23":1,"344":1,"547":1}}],["dev",{"2":{"1":1,"89":1,"145":2,"622":4,"736":2,"738":2,"739":2,"740":2}}],["devyarn",{"2":{"1":1}}],["yconfirm",{"2":{"751":1}}],["ycelestia",{"2":{"496":1,"498":1}}],["yum",{"2":{"622":4}}],["ysudo",{"2":{"622":3}}],["y",{"2":{"496":1,"498":1,"522":9,"622":3,"751":4}}],["yparity",{"2":{"341":2}}],["yield",{"2":{"184":1}}],["y33zld2luvedelzzr9cf92+2etaimiwhn9pcaqaszwpqckykhy9jb3ntb3muy3j5chrvlnnly3ayntzrms5qdwjlzxksiwoha36hewmw",{"2":{"137":2}}],["yh2zdyne9u",{"2":{"132":2}}],["yhcdb4cz7z4lrxmvrq5f8=",{"2":{"127":2}}],["y91da3zryfzmc7l",{"2":{"132":2}}],["yamlversion",{"2":{"253":1}}],["yamlextra",{"2":{"76":1}}],["yarn",{"2":{"1":2,"74":1,"78":2}}],["ymlglobal",{"2":{"503":1}}],["yml",{"2":{"76":1,"253":2,"343":1,"503":3,"504":1}}],["yet",{"2":{"78":2,"113":1,"186":1,"338":1,"569":1,"594":1}}],["yesterday",{"2":{"499":2}}],["yescelestia",{"2":{"266":1}}],["yes",{"2":{"42":1,"266":1,"440":1,"494":1,"507":2,"522":1}}],["years",{"2":{"722":1}}],["year",{"2":{"24":1,"25":1,"441":2,"446":11,"451":2}}],["youth",{"2":{"519":2}}],["youtube",{"2":{"34":1,"40":2}}],["yours",{"2":{"515":1}}],["yourself",{"2":{"503":2}}],["yourprivatekey",{"2":{"78":3}}],["your",{"0":{"59":1,"60":1,"61":1,"62":1,"63":1,"64":1,"65":1,"70":1,"71":1,"72":1,"73":2,"74":1,"76":1,"77":1,"78":1,"252":1,"324":1,"326":1,"327":1,"346":1,"355":1,"356":1,"418":1,"420":1,"422":1,"423":1,"425":1,"427":1,"429":1,"431":1,"432":1,"479":1,"520":1,"522":1,"528":1,"531":1,"553":1,"560":1,"570":1,"627":1,"661":1,"662":1,"667":1,"668":1,"734":1,"752":1},"1":{"60":1,"61":2,"62":2,"63":1,"64":2,"65":2,"480":1,"554":1,"555":1,"556":1,"557":1,"558":1,"559":1,"628":1,"669":1,"735":1,"736":1,"737":1,"738":1,"739":1,"740":1},"2":{"2":5,"23":2,"24":1,"26":5,"29":1,"34":4,"35":7,"36":6,"37":11,"38":14,"39":15,"40":7,"41":8,"42":1,"44":1,"47":1,"49":1,"50":2,"51":1,"57":1,"58":3,"59":2,"61":2,"62":7,"64":7,"65":4,"67":1,"69":2,"70":3,"71":11,"72":13,"73":8,"74":1,"75":6,"76":16,"77":3,"78":20,"79":5,"125":1,"188":1,"189":1,"190":1,"191":2,"200":2,"214":4,"219":1,"220":1,"226":1,"235":1,"236":1,"238":1,"243":1,"246":3,"248":1,"251":1,"252":2,"253":3,"259":5,"260":1,"261":1,"262":2,"266":5,"269":6,"270":1,"286":1,"301":1,"310":2,"311":2,"312":6,"314":3,"315":1,"316":3,"317":5,"318":5,"321":3,"326":1,"327":1,"331":1,"332":1,"336":2,"337":1,"338":1,"342":1,"343":2,"344":1,"346":2,"347":1,"348":6,"352":5,"353":1,"355":1,"356":1,"363":6,"369":5,"371":2,"384":3,"386":1,"390":1,"391":1,"421":1,"422":1,"426":1,"432":1,"440":1,"448":1,"449":1,"461":1,"465":1,"467":1,"470":3,"476":1,"483":3,"484":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,"555":6,"560":5,"563":1,"564":1,"567":1,"568":3,"569":2,"570":2,"576":2,"585":4,"587":2,"589":1,"590":1,"591":1,"592":2,"593":1,"597":1,"600":1,"602":1,"604":1,"605":1,"607":5,"610":1,"612":2,"614":2,"615":2,"616":1,"620":1,"621":1,"622":1,"623":3,"628":1,"632":7,"640":1,"649":1,"653":1,"655":1,"656":1,"658":2,"659":3,"662":3,"663":2,"667":1,"670":2,"671":1,"672":3,"673":1,"677":1,"684":1,"685":1,"693":3,"695":2,"699":1,"703":1,"716":1,"733":2,"736":1,"745":2,"748":3,"751":9,"752":2,"753":1,"756":11}}],["you",{"2":{"0":2,"2":1,"4":1,"23":3,"25":1,"26":6,"29":4,"30":1,"33":1,"34":4,"35":7,"36":4,"37":3,"39":1,"40":3,"41":11,"42":8,"47":2,"48":1,"50":2,"51":2,"54":1,"55":1,"57":2,"58":4,"61":1,"62":3,"64":3,"65":3,"66":4,"69":1,"70":1,"71":2,"72":5,"73":1,"74":1,"76":9,"77":2,"78":4,"79":1,"89":4,"90":1,"94":5,"95":2,"96":1,"102":1,"110":1,"111":2,"124":2,"148":3,"153":1,"187":13,"188":3,"189":1,"190":2,"193":1,"195":1,"196":4,"198":1,"199":1,"200":6,"201":3,"204":1,"212":1,"213":1,"216":1,"218":1,"220":2,"224":1,"226":2,"231":2,"232":5,"234":1,"238":3,"243":3,"245":1,"246":6,"247":1,"252":4,"253":2,"258":4,"259":1,"260":1,"262":1,"263":1,"268":2,"269":5,"270":1,"271":1,"272":1,"273":3,"279":1,"286":2,"291":1,"292":1,"296":2,"302":2,"307":4,"309":1,"311":5,"312":11,"313":3,"314":7,"315":1,"316":3,"317":4,"318":2,"319":1,"321":3,"322":3,"325":1,"331":3,"332":1,"336":1,"337":2,"338":2,"341":1,"343":1,"344":2,"346":4,"347":2,"348":5,"349":2,"350":5,"352":4,"353":1,"356":1,"358":1,"360":4,"361":3,"362":2,"363":5,"364":1,"365":1,"366":3,"369":2,"376":1,"384":4,"385":1,"389":2,"390":6,"391":4,"393":1,"417":3,"423":2,"427":2,"432":2,"440":1,"458":1,"461":1,"465":3,"467":2,"471":1,"472":1,"474":1,"483":3,"484":8,"486":1,"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":3,"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,"555":5,"560":2,"562":2,"563":2,"564":5,"565":1,"567":4,"568":3,"569":3,"570":2,"572":1,"573":1,"576":3,"577":1,"580":1,"585":7,"587":1,"591":3,"592":3,"595":2,"597":7,"600":1,"601":2,"602":1,"604":3,"605":2,"607":3,"608":2,"609":1,"610":1,"611":2,"612":5,"614":4,"615":10,"616":3,"617":1,"620":1,"622":1,"623":2,"624":1,"628":1,"629":1,"632":9,"634":1,"635":2,"639":3,"640":1,"641":1,"642":2,"643":1,"644":3,"649":3,"650":1,"651":1,"652":1,"653":2,"654":3,"655":1,"656":1,"658":2,"659":5,"660":4,"662":6,"663":2,"664":2,"667":1,"671":1,"672":6,"673":3,"675":1,"677":1,"684":4,"693":2,"694":1,"695":2,"696":1,"699":8,"705":3,"710":2,"712":1,"716":5,"717":2,"720":1,"724":1,"730":2,"731":3,"732":2,"733":5,"735":1,"736":2,"738":2,"739":2,"740":1,"741":2,"745":8,"746":2,"747":1,"750":1,"751":11,"756":4}}],["ahzu6yr9xmpixlquhgbhj9xl3wiaoz6pe3cvml",{"2":{"520":2,"524":2}}],["ahead",{"2":{"35":1,"115":1}}],["a5jf",{"2":{"519":2,"520":2}}],["away",{"2":{"454":1,"715":1}}],["await",{"2":{"364":6,"365":8,"366":6,"389":4}}],["awareness",{"2":{"30":1}}],["a1",{"2":{"253":2}}],["aztec",{"2":{"199":17}}],["aka",{"2":{"171":1,"193":1,"201":1,"377":1}}],["aaaaaaaaaaaaaaaaaaaaaaaaaaecawqfbgcicra=",{"2":{"336":2,"342":2}}],["aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaba==",{"2":{"132":2,"137":2}}],["aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaap8aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa",{"2":{"132":2}}],["aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaeaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaazz5aj1mijjrowjdcifyjkr0pcroiu2jigmd9bzuhzro",{"2":{"132":2}}],["aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaeaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaqs98k+8wq2ix2bdctfohjtrqbqybtpdb1bufy",{"2":{"132":2}}],["aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaqbaaabxaaaacbaagroagqgaqqdaqogl2nlbgvzdglhlmjsb2iudjeutxnnugf5rm9yqmxvynmseqovy2vszxn0awexdwc1zwt0mmnjn250dzrkdg1zzdlsn3n0ctbzn3z5ztd5btjyzhishqaaaaaaaaaaaaaaaaaaaaaaaaasexiyqkmkmoizggkxaiigrflow1m",{"2":{"137":2}}],["aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaqbaaacugaaacbaagrnagqfaqqcaqogl2nlbgvzdglhlmjsb2iudjeutxnnugf5rm9yqmxvynmseaovy2vszxn0awexywxwngzwbhf5d21jnmn1adl5mzvly2xychf5cwf4mjn2z3zrczushqaaaaaaaaaaaaaaaaaaaaaaaaaaaagicagicagiggfeiiazloeqsgxg3row7sr1rkq7dvjygp3axkaqy4og6hyc0eibabjnclikrgofl2nvc21vcy5jcnlwdg8uc2vjcdi1nmsxllb1yktlerijcieca4ief8fzeabqlvc2wocefs+lhak0mdnmpnsxylkqv7qsbaoccaey5pebehekcwoedxrpyridmtyweltvbbpa7xbbsgyfrwtzcfhq3va1vhtbriczyd0elkajo6klsdooeqcwovaguwtdp55v8btf3wc7",{"2":{"132":2}}],["aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaqaaaaaaeuaa8cpbyigsvxwya9toi+aytu3jja2wki5zlkm72",{"2":{"132":2}}],["aaaaaaaaaaaaaaaaaaaaaaaaaaaabytlu4hlouuaaaaaaaaaaaaaaaaaaaaaaaaaaaafhmtties5rt0t52lwq3l0fnm6kqqka7nrxfnm8zc",{"2":{"132":2}}],["aaaaaaaaaaaaaaaaaaaaaaaaaaaabytlu4hlouuaaaaaaaaaaaaaaaaaaaaaaaaaaaafhmtties5ru6k2enim7thjqycl82hsxpinycelhed9qk+p9zbnide",{"2":{"132":2}}],["aaaaaaaaaaaaaaaaaaaaaaaaaaaabytlu4hlouu=",{"2":{"132":2}}],["aaaaaaaaaaaaaaaaaaaaaaaaaejpdcbnowap3dm=",{"2":{"321":8,"558":2}}],["aaaaaaaaaaaaaaaaaaaaaaaaabitejjcqyqyijkaaaaaaaaaaaaaaaaaaaaaaaaaehmsmkjdjdkimbiwnpodwizbfr0uifhpkwgy",{"2":{"137":2}}],["axtrw6s+qsnuzfgfeg37da6igop2zqck+04egqkaggbgaisewoncgr1dglheguymtawmbdq6awaqclylqpnrfod6h8mgmwxjfenhwhru39ecrvkmfknq8+hhuodhdoqig",{"2":{"137":2}}],["airdrop",{"2":{"471":1,"710":1}}],["ai",{"2":{"350":1,"448":1,"449":1,"687":1,"688":1,"689":1,"693":1,"697":1}}],["aiming",{"2":{"211":1}}],["aims",{"2":{"44":1,"277":1}}],["aim",{"2":{"37":1,"211":1}}],["aid",{"2":{"29":1,"398":1}}],["aura",{"2":{"508":2}}],["augments",{"2":{"416":1,"477":1}}],["aunt",{"2":{"129":4}}],["aunts",{"2":{"127":2,"129":4,"132":2,"137":2,"144":5,"145":4,"148":8}}],["authcelestia",{"2":{"352":1}}],["authorize",{"2":{"506":1}}],["authorization",{"2":{"315":1,"336":2,"342":2}}],["authority",{"2":{"71":1}}],["authentication",{"2":{"300":1,"315":1,"316":5,"317":1,"359":2,"554":2}}],["authenticated",{"2":{"160":1,"184":1}}],["auth",{"0":{"316":1,"318":1},"2":{"76":10,"269":5,"277":1,"314":2,"315":5,"316":9,"317":2,"318":7,"319":4,"321":2,"324":1,"336":7,"342":2,"352":1,"357":4,"358":4,"360":4,"363":5,"374":1,"494":2,"522":2,"524":2}}],["automate",{"2":{"554":1}}],["automated",{"2":{"386":1}}],["automatically",{"2":{"23":1,"71":1,"270":1,"296":1,"322":1,"364":1,"376":1,"382":2,"509":1}}],["autogenerated",{"2":{"252":1}}],["auto",{"2":{"71":2,"496":2,"498":2,"522":6,"639":2}}],["audible",{"2":{"34":1}}],["audience",{"0":{"36":1},"2":{"34":1,"36":3,"37":1,"38":1,"39":1,"40":1}}],["amd",{"2":{"623":1}}],["amd64",{"2":{"540":1,"568":1,"623":12}}],["amanda",{"2":{"491":2}}],["amazonaws",{"2":{"199":16}}],["amountceles",{"2":{"500":1}}],["amount=1000000utia",{"2":{"751":2}}],["amount=9000000utia",{"2":{"659":2}}],["amount=5000000000000utia",{"2":{"500":2}}],["amount=",{"2":{"500":2,"657":2}}],["amount",{"0":{"64":1,"430":1},"2":{"35":1,"62":1,"71":1,"202":1,"261":3,"262":2,"263":2,"324":2,"325":2,"331":3,"374":1,"375":1,"397":1,"405":1,"414":1,"422":1,"426":1,"430":1,"446":2,"493":2,"494":2,"495":2,"497":4,"500":3,"522":8,"523":2,"525":4,"605":2,"657":2,"658":1,"659":3,"751":1}}],["among",{"2":{"31":1,"35":1,"412":1,"413":1,"477":1,"714":1}}],["ample",{"2":{"39":1}}],["amp",{"0":{"4":1,"72":1,"439":1,"497":1},"1":{"440":1,"441":1,"442":1,"443":1,"444":1,"445":1,"446":1},"2":{"1":4,"2":2,"38":1,"53":1,"54":1,"55":1,"58":1,"72":1,"127":2,"129":14,"132":1,"135":2,"137":2,"141":6,"142":2,"146":10,"147":2,"148":10,"200":8,"203":1,"364":10,"365":8,"366":6,"445":3,"446":5,"477":1,"503":1,"527":4,"564":2,"622":10,"738":4,"739":4}}],["affect",{"2":{"386":1,"563":1}}],["affected",{"2":{"7":1}}],["affordable",{"2":{"40":1}}],["affiliated",{"2":{"38":1}}],["after=network",{"2":{"736":2,"738":2,"739":2,"740":2}}],["afterwards",{"2":{"24":1}}],["after",{"0":{"32":1,"263":1},"2":{"22":2,"23":2,"34":1,"39":1,"40":1,"62":1,"64":2,"65":2,"66":1,"71":2,"73":1,"75":1,"77":1,"81":1,"149":1,"157":1,"188":1,"189":1,"195":1,"196":1,"263":1,"267":1,"311":1,"321":1,"358":2,"360":1,"385":1,"423":1,"427":1,"432":1,"437":1,"483":1,"491":1,"509":2,"540":1,"560":1,"568":1,"573":1,"593":1,"597":1,"603":1,"604":1,"605":1,"615":1,"622":2,"639":1,"641":1,"660":1,"714":1,"740":1,"745":1,"751":2,"752":1,"756":1}}],["avril14th",{"2":{"687":1,"688":1,"689":1,"706":1,"707":1,"708":1}}],["average",{"2":{"389":2,"391":1}}],["avoid",{"2":{"35":1,"38":1,"39":1,"76":1,"348":2,"375":1,"503":1,"615":1,"640":1}}],["avoiding",{"2":{"13":1,"18":1}}],["availability",{"0":{"208":1,"330":1,"392":1,"393":1,"394":1,"395":1,"396":1,"397":1,"401":1,"406":1,"407":1,"412":1,"416":1,"469":1,"470":1,"677":1,"691":1,"692":1,"693":1,"704":1,"705":1,"719":1,"737":1},"1":{"209":1,"210":1,"211":1,"393":1,"394":1,"395":1,"396":1,"397":1,"398":1,"399":1,"400":1,"401":1,"402":1,"403":1,"404":1,"405":1,"407":1,"408":2,"409":2,"410":1,"411":1,"412":1,"692":1,"693":1,"694":1,"695":1,"738":1,"739":1,"740":1},"2":{"30":1,"36":1,"37":1,"53":1,"70":1,"92":2,"96":1,"97":2,"105":1,"107":1,"150":3,"164":1,"167":1,"179":1,"206":2,"209":2,"210":1,"211":2,"222":1,"238":1,"241":2,"254":2,"256":1,"266":1,"279":2,"283":1,"284":2,"304":1,"348":1,"385":1,"393":7,"394":3,"395":5,"396":4,"397":1,"398":1,"401":1,"402":1,"403":1,"406":4,"409":1,"413":1,"414":5,"415":1,"434":1,"436":2,"437":1,"453":1,"454":1,"467":1,"476":1,"477":2,"543":1,"548":1,"568":1,"600":2,"609":2,"664":2,"665":1,"679":1,"694":1,"699":1,"717":4,"731":2,"740":1,"746":2}}],["available",{"2":{"16":2,"21":1,"29":1,"31":1,"40":2,"50":1,"96":1,"107":1,"159":1,"163":1,"164":1,"212":2,"234":1,"237":1,"270":1,"274":1,"367":1,"396":2,"401":1,"403":1,"406":1,"407":2,"412":1,"413":1,"415":3,"416":4,"434":2,"438":2,"446":2,"473":1,"488":2,"494":1,"502":2,"540":3,"568":3,"601":3,"604":1,"605":2,"683":1,"694":1}}],["agreeing",{"2":{"413":1}}],["agree",{"2":{"413":1}}],["agreement",{"2":{"238":1}}],["agnostic",{"2":{"412":1}}],["again",{"2":{"338":1,"352":1,"377":1,"416":1,"555":1,"560":1,"645":1}}],["against",{"2":{"76":1,"104":1,"107":2,"212":1,"219":1,"562":1}}],["ago",{"2":{"252":2,"253":2}}],["aggression",{"2":{"15":1}}],["agendas",{"2":{"30":1}}],["agenda",{"2":{"30":1}}],["age",{"2":{"6":1}}],["aria2",{"2":{"622":4}}],["aria2c",{"2":{"596":6}}],["arise",{"2":{"402":1}}],["arises",{"2":{"394":1}}],["arm",{"2":{"623":1}}],["arm64",{"2":{"540":1,"568":1,"623":12}}],["armored",{"2":{"247":1,"491":2}}],["arrived",{"2":{"537":1}}],["array",{"2":{"364":3,"579":1}}],["arranging",{"2":{"31":1}}],["arranges",{"2":{"415":1}}],["arrangement",{"2":{"413":1}}],["arranged",{"2":{"407":1,"413":1}}],["arrange",{"2":{"30":2,"36":1,"37":1}}],["architectnodes",{"2":{"639":2,"706":1,"707":1,"708":1}}],["architecture",{"2":{"548":1}}],["architectures",{"2":{"414":1,"540":1,"568":1}}],["archives",{"2":{"687":1,"688":1,"689":1}}],["archive",{"2":{"470":1,"591":1,"687":1,"688":1,"689":1,"693":1}}],["archival",{"0":{"694":1,"695":1},"1":{"695":1},"2":{"23":1,"27":1,"437":3,"438":5,"694":2,"695":1}}],["arg",{"2":{"488":2}}],["args",{"2":{"314":3,"317":2,"357":8,"358":8,"360":8,"616":4,"617":4}}],["argument",{"2":{"176":1,"179":1}}],["arguments",{"2":{"172":1,"314":1,"356":2,"357":5,"358":5,"360":2}}],["arabicaversions",{"2":{"539":2,"567":2,"614":2,"616":2,"617":2}}],["arabicaresturl",{"2":{"389":2}}],["arabicarpcurl",{"2":{"389":2}}],["arabicachainid",{"2":{"321":1,"346":2,"389":2,"390":1,"551":1,"585":6,"596":2}}],["arabicacelestia",{"2":{"310":1,"311":2,"312":1,"352":1,"483":1,"484":1,"485":1,"631":1,"633":1,"670":1,"671":1,"672":1,"674":1}}],["arabica",{"0":{"461":1,"471":1,"558":1,"725":1,"726":1},"1":{"462":1,"463":1,"464":1,"465":1,"466":1,"467":1,"468":1,"469":1,"470":1,"471":1,"472":2,"473":2,"474":1,"475":1,"726":1},"2":{"246":2,"289":2,"294":2,"310":1,"311":5,"312":3,"316":3,"328":2,"352":9,"358":3,"385":1,"389":6,"390":6,"461":2,"462":3,"465":1,"467":12,"470":5,"472":2,"473":1,"474":3,"483":2,"484":5,"485":1,"537":1,"551":2,"554":1,"558":1,"564":1,"585":3,"614":1,"631":1,"632":4,"633":1,"670":1,"671":4,"672":3,"673":1,"674":1,"699":1,"715":1,"716":3,"725":3,"727":1,"733":2,"748":1}}],["artifacts",{"0":{"198":1},"1":{"199":1,"200":1},"2":{"188":1}}],["arfge6qnxuiz18vlglgewtw",{"2":{"132":4}}],["arbitrary",{"2":{"111":2,"495":1}}],["arbitrum",{"0":{"66":1,"69":1,"83":1,"85":1,"91":1},"1":{"67":1,"68":1,"69":1,"70":1,"71":1,"72":1,"73":1,"74":1,"75":1,"76":1,"77":1,"78":1,"79":1,"80":1,"81":1,"82":1,"83":1,"84":1,"85":1,"86":1,"92":1,"93":1,"94":1,"95":1,"96":1,"97":1,"98":1},"2":{"59":1,"61":1,"62":3,"64":4,"66":4,"67":3,"68":1,"69":3,"70":1,"72":5,"75":6,"76":1,"78":2,"80":1,"82":1,"83":1,"88":3,"89":2,"92":5,"93":1,"94":1,"96":1,"97":2,"98":1,"201":1,"213":1,"216":2,"254":1}}],["around",{"0":{"400":1},"2":{"12":1,"43":1,"394":1,"400":1,"402":1,"548":1,"609":1,"663":1}}],["areas",{"2":{"32":1,"37":1,"42":1}}],["area",{"2":{"31":1,"36":1,"37":1,"38":1,"39":1,"42":1}}],["aren",{"2":{"29":1,"529":1}}],["are",{"0":{"397":1,"405":1,"562":1},"2":{"8":2,"10":1,"16":1,"18":1,"20":2,"22":2,"23":3,"25":1,"29":1,"30":1,"33":2,"35":1,"36":2,"42":3,"44":1,"46":1,"53":1,"58":1,"59":1,"71":3,"72":1,"75":1,"76":4,"78":2,"94":1,"95":1,"103":2,"104":2,"110":1,"111":3,"112":1,"113":1,"115":3,"119":1,"121":1,"122":3,"124":3,"127":1,"129":2,"132":1,"136":1,"137":1,"141":1,"143":2,"144":2,"146":2,"148":2,"149":1,"153":2,"158":1,"159":1,"163":1,"164":1,"165":1,"170":1,"171":1,"172":2,"173":1,"176":1,"179":1,"182":1,"184":1,"187":5,"188":1,"189":1,"195":1,"196":2,"198":1,"200":5,"203":2,"209":1,"210":1,"216":1,"218":1,"224":2,"232":1,"234":2,"237":2,"238":1,"240":1,"241":1,"242":1,"246":3,"252":2,"263":1,"270":3,"271":1,"272":1,"277":1,"279":1,"280":2,"282":2,"284":2,"285":1,"296":1,"297":1,"298":1,"308":1,"310":1,"311":3,"314":1,"315":1,"316":1,"317":1,"318":2,"338":1,"341":1,"342":1,"347":1,"348":2,"351":1,"365":1,"373":2,"374":3,"375":1,"377":1,"381":2,"384":1,"385":2,"387":2,"395":1,"397":4,"400":1,"402":1,"403":1,"404":1,"406":3,"407":3,"408":1,"409":2,"412":1,"413":3,"415":4,"416":1,"434":1,"435":2,"437":1,"438":2,"441":1,"446":4,"461":1,"465":1,"466":1,"471":1,"474":1,"478":1,"483":1,"484":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":2,"531":2,"537":1,"540":1,"541":3,"547":1,"550":1,"552":3,"554":4,"560":2,"562":3,"568":1,"574":1,"576":1,"581":1,"585":1,"594":1,"595":2,"597":2,"604":1,"605":6,"611":1,"612":1,"614":1,"622":4,"623":1,"625":1,"626":1,"632":2,"635":1,"637":1,"642":1,"658":1,"663":1,"666":1,"671":1,"684":1,"685":1,"696":1,"698":1,"699":3,"703":1,"705":1,"709":1,"710":1,"712":1,"713":1,"714":2,"715":2,"716":2,"717":1,"727":1,"729":1,"731":2,"733":1,"735":2,"742":1,"746":1,"748":1,"751":6}}],["abstain",{"2":{"494":1}}],["abbreviation",{"2":{"451":1}}],["abci",{"0":{"605":1},"2":{"412":1,"605":8}}],["abci++",{"2":{"412":1,"415":1}}],["abi",{"2":{"129":2,"140":2,"148":2}}],["abigen",{"2":{"121":1}}],["ability",{"2":{"97":1,"154":1,"394":1,"398":1,"399":1,"509":1,"632":2,"671":2}}],["able",{"2":{"34":2,"35":1,"58":1,"64":2,"65":1,"75":1,"76":1,"110":2,"124":1,"154":1,"168":1,"170":1,"175":1,"189":1,"196":1,"240":1,"346":1,"358":1,"361":1,"377":1,"390":1,"391":1,"395":1,"398":1,"410":2,"437":1,"505":1,"548":1,"725":1,"736":1,"738":1,"739":1,"740":1,"751":1}}],["abusive",{"2":{"10":1}}],["about",{"0":{"284":1},"1":{"285":1},"2":{"3":1,"7":1,"16":1,"20":1,"30":1,"36":1,"37":1,"42":5,"43":1,"58":1,"64":1,"66":2,"71":1,"79":1,"92":1,"94":1,"95":1,"110":2,"122":1,"196":2,"219":1,"222":1,"238":1,"258":1,"286":1,"306":1,"307":1,"314":2,"321":1,"322":1,"349":1,"376":1,"400":1,"401":1,"402":2,"456":1,"465":1,"475":1,"488":2,"519":1,"540":1,"567":1,"568":1,"612":2,"684":1,"694":1,"695":1,"698":2,"713":2,"717":1,"729":2}}],["above",{"2":{"2":1,"23":1,"70":1,"76":1,"90":1,"115":1,"118":1,"124":3,"139":1,"140":1,"142":1,"147":1,"148":1,"149":1,"154":1,"157":1,"171":1,"176":1,"182":1,"187":1,"195":1,"318":1,"319":1,"321":3,"324":1,"374":1,"382":2,"412":1,"415":2,"467":1,"470":1,"477":1,"495":1,"522":1,"570":1,"615":1,"658":1,"659":1,"672":1,"756":1}}],["atomic",{"2":{"414":1}}],["atlas",{"2":{"81":1}}],["atmosphere",{"2":{"34":1,"35":1}}],["at",{"0":{"445":1,"451":1},"1":{"446":1},"2":{"9":1,"10":1,"16":3,"23":1,"26":1,"34":2,"35":2,"36":2,"37":1,"38":3,"39":1,"40":1,"42":1,"67":1,"69":1,"78":1,"82":1,"107":2,"109":1,"139":2,"145":4,"165":1,"184":1,"196":1,"197":1,"200":3,"201":2,"202":1,"203":1,"206":1,"212":1,"223":1,"237":1,"270":3,"271":2,"272":7,"273":3,"285":1,"321":1,"348":1,"358":4,"364":3,"365":5,"366":1,"371":2,"377":1,"381":2,"397":1,"400":1,"407":3,"409":1,"412":1,"415":1,"416":3,"440":1,"441":2,"444":1,"445":1,"446":6,"447":1,"451":1,"469":1,"470":1,"473":1,"485":1,"486":1,"488":4,"503":2,"505":1,"507":2,"515":1,"532":1,"540":1,"541":1,"548":1,"552":1,"564":1,"568":1,"591":1,"595":1,"633":1,"634":1,"638":1,"640":1,"658":1,"659":1,"674":1,"675":1,"679":1,"692":1,"693":1,"704":1,"705":1,"715":1,"716":2,"725":1,"739":1}}],["attach",{"2":{"503":2}}],["attack",{"2":{"394":3,"397":1}}],["attacks",{"2":{"7":1,"71":1}}],["attempting",{"2":{"642":1}}],["attempt",{"2":{"393":1}}],["attesting",{"2":{"145":1}}],["attestations",{"2":{"210":2,"254":1}}],["attestation",{"2":{"139":2,"145":4}}],["attestationproof",{"0":{"145":1},"2":{"139":6,"145":6,"148":4}}],["attests",{"2":{"124":1}}],["attested",{"2":{"103":2,"216":1}}],["attended",{"2":{"39":1}}],["attendees",{"2":{"30":3,"31":4,"32":3,"34":2,"35":2,"36":2,"37":2,"39":1,"40":1,"42":1,"46":1,"58":2}}],["attending",{"2":{"39":2}}],["attend",{"2":{"36":1,"40":3}}],["attendance",{"2":{"30":1,"32":1,"36":3,"39":2}}],["attention",{"2":{"7":1,"348":1}}],["attributes",{"2":{"132":4}}],["attribution",{"0":{"16":1}}],["attracting",{"2":{"39":1}}],["attract",{"2":{"36":1,"37":1}}],["advertise",{"2":{"602":1}}],["adventure",{"2":{"350":1}}],["advise",{"2":{"705":1}}],["advised",{"2":{"550":1,"585":1}}],["advisable",{"2":{"371":1}}],["advantages",{"2":{"184":1}}],["advanced",{"0":{"334":1},"2":{"361":1}}],["advance",{"2":{"35":1,"39":1,"716":1}}],["advances",{"2":{"7":1,"212":1}}],["ad5ezbg0",{"2":{"336":2}}],["adjusted",{"2":{"374":1,"375":1}}],["adjust",{"2":{"242":1,"564":1}}],["adr019",{"2":{"441":1}}],["adrs",{"2":{"412":1,"477":1}}],["adr",{"2":{"154":1,"548":2}}],["adapt",{"2":{"124":1,"139":1}}],["adapted",{"2":{"16":1,"68":1,"446":1}}],["admin",{"2":{"76":2,"316":4,"318":2,"336":2,"348":4,"503":1}}],["administrative",{"2":{"71":1,"298":1}}],["adhere",{"2":{"46":1}}],["adhering",{"2":{"23":1}}],["addnetworkkeplr",{"2":{"389":17}}],["adds",{"2":{"374":1,"415":1,"564":1}}],["addcustomprover",{"2":{"191":1,"192":2}}],["addblock",{"2":{"113":2,"115":4}}],["addrs",{"2":{"326":2}}],["addr",{"2":{"76":1,"382":4,"502":2,"560":2,"639":8}}],["addresscelestia",{"2":{"262":1,"264":1,"292":1,"327":1}}],["address=$",{"2":{"602":2,"662":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":{"331":1}}],["address=celestia1c425ckmve2489atttx022qpc02gxspa29wmh0dexport",{"2":{"331":1}}],["address=",{"2":{"259":4,"266":4,"500":2,"503":1}}],["addresses",{"2":{"71":2,"76":3,"79":1,"266":1,"293":1,"369":1}}],["address",{"0":{"292":1,"325":1,"327":1,"559":1,"602":1},"2":{"7":1,"9":1,"23":1,"26":3,"38":1,"71":8,"76":6,"78":4,"85":1,"86":1,"110":1,"129":2,"146":2,"147":2,"148":2,"187":9,"188":2,"189":21,"191":1,"192":4,"194":2,"195":6,"196":4,"203":2,"216":1,"253":4,"259":4,"261":2,"262":3,"264":1,"266":8,"267":2,"289":1,"291":1,"292":3,"311":1,"312":7,"315":1,"325":9,"327":2,"328":8,"331":2,"341":1,"352":4,"369":2,"382":2,"390":1,"391":1,"472":5,"483":1,"484":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,"554":1,"559":2,"585":2,"601":1,"602":8,"632":2,"639":4,"656":1,"661":8,"662":4,"672":2,"673":5,"710":5,"745":2}}],["added",{"2":{"71":1,"78":2,"97":1,"293":1,"361":1,"393":1,"503":2,"506":1,"658":1,"660":1}}],["add",{"0":{"61":1,"74":1,"192":1,"389":1,"640":1,"657":1,"662":1},"2":{"39":1,"41":1,"61":1,"70":1,"76":2,"111":2,"115":4,"190":1,"192":1,"244":1,"246":6,"248":6,"269":1,"279":1,"294":2,"311":1,"312":2,"356":1,"363":3,"388":2,"389":6,"390":4,"391":3,"438":1,"446":1,"483":1,"488":4,"490":2,"500":5,"503":1,"507":8,"511":1,"519":2,"527":5,"536":2,"544":1,"555":1,"576":1,"623":1,"639":1,"640":6,"645":3,"656":2,"657":2,"660":1,"661":1,"672":2,"715":1}}],["addition",{"2":{"34":1,"403":1,"416":1,"454":1,"577":1}}],["additionally",{"2":{"39":1,"76":1,"93":1,"94":2,"97":1,"111":1,"407":1,"443":1}}],["additional",{"0":{"85":1,"86":1,"335":1,"754":1},"1":{"336":1,"337":1,"338":1},"2":{"32":1,"38":1,"40":1,"83":1,"84":1,"92":1,"110":1,"173":1,"300":1,"314":1,"374":1,"409":1,"423":1,"427":1,"432":1,"754":1}}],["adding",{"0":{"390":1,"391":1,"658":1},"2":{"2":1,"78":2,"242":1,"490":2,"504":1,"640":1,"641":1}}],["apt",{"2":{"622":6}}],["aptyummac",{"2":{"622":1}}],["aptyum",{"2":{"622":1}}],["apache",{"2":{"103":2}}],["api",{"0":{"274":1,"297":1,"298":1,"301":1,"302":1,"303":1,"332":1,"367":1,"378":1,"381":1,"688":1,"707":1},"1":{"298":1,"299":2,"300":2,"301":2,"302":1,"303":2,"379":1,"380":1,"381":1,"382":1,"383":1,"384":1},"2":{"42":1,"75":1,"187":6,"188":4,"195":6,"203":3,"274":1,"279":2,"297":5,"298":4,"299":1,"300":1,"302":2,"303":1,"304":1,"313":1,"315":1,"316":1,"317":2,"332":1,"350":3,"351":1,"353":3,"356":2,"357":3,"358":2,"361":1,"367":1,"381":1,"383":2,"390":1,"391":1,"404":1,"438":2,"466":1,"467":3,"494":1,"543":1,"600":1,"609":1,"638":1,"685":1,"688":21,"703":1,"707":18}}],["apology",{"2":{"12":1}}],["apologizing",{"2":{"7":1}}],["appcelestia",{"2":{"599":1,"654":1}}],["appcd",{"2":{"539":1}}],["appconsts",{"2":{"135":6,"382":4}}],["appdsudo",{"2":{"736":2}}],["appd",{"0":{"266":1},"1":{"267":1},"2":{"266":2,"267":2,"379":2,"382":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,"585":12,"597":9,"599":2,"603":1,"604":1,"605":1,"607":1,"647":3,"654":3,"655":2,"656":2,"657":2,"659":2,"660":3,"662":2,"663":2,"735":1,"736":15,"745":4,"751":13,"756":2}}],["append",{"2":{"103":2,"115":2}}],["appendix",{"0":{"80":1},"1":{"81":1,"82":1,"83":1,"84":1,"85":1,"86":1},"2":{"76":1}}],["appear",{"2":{"78":1,"422":1,"503":1}}],["appearance",{"2":{"6":1}}],["appeal",{"2":{"36":1}}],["app",{"0":{"379":1,"415":1,"429":1,"511":1,"534":1,"538":1,"584":1,"649":1,"652":1,"732":1,"736":1},"1":{"416":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,"650":1,"651":1,"652":1,"653":1,"654":1,"655":1,"656":1,"657":1,"658":1,"659":1,"660":1,"661":1,"662":1,"663":1},"2":{"75":2,"129":2,"135":4,"137":1,"138":1,"141":4,"142":4,"148":6,"279":1,"328":4,"373":2,"376":2,"377":1,"382":20,"385":1,"412":6,"415":1,"416":2,"448":1,"449":1,"466":1,"477":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":12,"540":4,"541":1,"584":1,"585":13,"589":1,"591":1,"592":2,"595":1,"596":18,"597":4,"599":1,"601":2,"602":2,"604":1,"605":1,"607":1,"624":1,"637":1,"649":1,"652":1,"654":1,"658":1,"659":1,"660":3,"661":1,"682":1,"683":1,"685":1,"703":1,"716":1,"717":1,"732":3,"733":2,"736":1,"739":1,"746":1,"751":3,"756":1}}],["appreciate",{"2":{"42":1,"46":1}}],["appreciation",{"2":{"37":2}}],["approximate",{"2":{"716":1}}],["approximately",{"2":{"64":1,"65":1}}],["approximation",{"2":{"374":1,"683":1}}],["approving",{"2":{"64":1}}],["approvedprovers",{"2":{"369":1}}],["approve",{"2":{"62":1,"65":1,"70":1,"384":1,"422":1}}],["approval",{"2":{"47":1,"422":1}}],["approaching",{"2":{"371":1}}],["approaches",{"2":{"149":1}}],["approach",{"2":{"36":1,"37":1,"38":1,"171":1,"178":1,"184":1,"465":1,"684":1,"699":1}}],["appropriate",{"2":{"8":2,"111":2,"270":1,"364":1,"434":1,"563":1,"639":1}}],["apple",{"2":{"540":1,"568":1,"623":1}}],["applicable",{"2":{"24":2,"26":1,"30":1}}],["applicant",{"2":{"22":1}}],["applicants",{"2":{"22":8,"23":2,"25":2}}],["applications",{"2":{"23":1,"25":1,"27":3,"221":1,"223":2,"226":1,"237":1,"241":2,"242":2,"286":1,"307":2,"404":1,"410":1,"436":1,"438":1,"461":1,"678":1,"722":1,"734":1,"736":1}}],["application",{"0":{"25":1,"26":1},"1":{"26":1},"2":{"20":3,"23":3,"26":1,"241":2,"242":1,"307":1,"311":1,"336":2,"342":2,"369":1,"373":2,"377":1,"385":1,"410":7,"412":3,"454":1,"477":1,"488":10,"504":1,"529":1,"536":1,"563":1,"596":1,"604":1,"607":1,"756":1}}],["applies",{"2":{"9":2,"90":1,"296":1}}],["applying",{"2":{"26":1,"32":1,"407":1}}],["apply",{"2":{"20":1,"24":1,"25":1,"413":1,"544":1,"595":1}}],["appointed",{"2":{"9":1}}],["acelestia",{"2":{"745":1}}],["achieve",{"2":{"292":1}}],["achieved",{"2":{"44":1,"504":1,"587":1}}],["acquire",{"0":{"69":1},"2":{"69":1}}],["acquaintances",{"2":{"38":1,"39":1}}],["across",{"2":{"41":2,"94":1,"141":1,"241":1,"445":1,"648":1}}],["acknowledging",{"2":{"38":1}}],["accurately",{"2":{"375":1,"559":1}}],["accurate",{"2":{"373":1}}],["accomplish",{"2":{"253":1}}],["accommodate",{"2":{"30":1}}],["accordingly",{"2":{"42":1,"76":1,"289":1,"564":1}}],["according",{"2":{"35":1,"212":1}}],["accountexport",{"2":{"527":1}}],["accounted",{"2":{"374":1}}],["accountname",{"2":{"382":2}}],["accountaddress",{"2":{"314":1}}],["accounts",{"0":{"525":1},"2":{"78":5,"129":2,"148":2,"258":1,"259":1,"374":1,"506":4,"507":2,"525":1,"532":1}}],["account",{"0":{"263":1,"267":1,"327":1,"331":1,"377":1,"511":1,"518":1,"522":1,"523":1,"524":1,"528":1,"529":1,"657":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":{"9":1,"71":2,"75":3,"78":9,"187":5,"256":2,"257":2,"258":6,"259":4,"262":1,"263":2,"266":7,"267":1,"324":2,"327":3,"338":4,"346":2,"348":10,"352":4,"353":2,"374":3,"377":1,"382":6,"390":1,"391":1,"402":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,"632":1,"639":4,"640":1,"657":2,"671":1}}],["accessed",{"2":{"601":1}}],["accessing",{"2":{"400":1,"401":1}}],["accessible",{"2":{"30":1,"58":1,"212":1,"245":4,"400":1,"541":2,"552":2}}],["access",{"0":{"470":1,"591":1,"693":1,"705":1},"2":{"36":1,"37":1,"47":2,"50":1,"58":1,"77":1,"121":1,"155":1,"169":1,"178":1,"187":2,"206":1,"257":1,"298":1,"299":1,"311":1,"336":1,"374":1,"389":2,"395":1,"399":1,"400":2,"404":2,"432":1,"438":4,"466":4,"470":2,"483":1,"632":2,"671":1,"685":4,"693":1,"703":4,"705":1}}],["accept",{"2":{"172":1,"373":1,"377":1,"416":1,"502":2,"600":1}}],["accepts",{"2":{"96":1,"196":1,"416":2}}],["accepted",{"2":{"22":1,"95":1}}],["acceptable",{"2":{"8":1}}],["accepting",{"2":{"7":2}}],["actors",{"2":{"404":1}}],["acts",{"2":{"187":1}}],["actually",{"2":{"282":1,"393":1}}],["actual",{"2":{"111":3,"113":1,"122":1,"158":1,"199":1,"374":1,"441":1}}],["activating",{"2":{"714":1,"715":1}}],["activation",{"2":{"343":1}}],["activate",{"2":{"347":1,"576":1}}],["activatecd",{"2":{"347":1}}],["activity",{"2":{"77":1}}],["activities",{"2":{"24":1,"30":1,"31":1}}],["active",{"2":{"20":1,"23":3,"26":2,"49":1,"252":1,"253":1,"477":1}}],["acting",{"2":{"9":1}}],["actions",{"2":{"13":1,"111":4}}],["action",{"2":{"8":1,"11":1,"132":2,"314":1}}],["act",{"2":{"6":1,"509":1}}],["aside",{"2":{"446":1}}],["async",{"2":{"364":2,"365":1,"366":1,"389":2}}],["ascii",{"2":{"247":1,"491":2}}],["aspects",{"2":{"41":1,"58":1}}],["asked",{"2":{"46":1,"184":1,"699":1}}],["ask",{"2":{"34":1,"37":1,"38":1,"389":2,"416":1}}],["assurances",{"0":{"405":1}}],["assumption",{"2":{"375":1,"397":2,"400":1,"564":1}}],["assumptions",{"0":{"397":1},"2":{"209":1,"400":1,"554":1}}],["assumed",{"2":{"397":1,"459":1}}],["assumes",{"2":{"200":2,"336":1,"375":1,"397":1,"543":1,"567":1}}],["assume",{"2":{"111":1,"187":1,"241":1,"352":1}}],["assign",{"2":{"71":1,"500":1}}],["assist",{"2":{"40":1,"41":1}}],["assistance",{"2":{"39":1,"41":1}}],["associated",{"2":{"23":1,"39":1,"145":2,"284":1,"291":1,"324":1,"415":2}}],["asset",{"2":{"453":1}}],["assets",{"2":{"7":1,"227":1}}],["assert",{"2":{"176":1,"364":6}}],["asserted",{"2":{"172":2}}],["assertions",{"2":{"71":3}}],["assessing",{"2":{"32":1}}],["assess",{"2":{"20":1,"36":1}}],["as",{"0":{"91":1,"235":1,"261":1,"262":1,"264":1,"337":1,"662":1,"734":1},"1":{"92":1,"93":1,"94":1,"95":1,"96":1,"97":1,"98":1,"265":1,"735":1,"736":1,"737":1,"738":1,"739":1,"740":1},"2":{"6":1,"7":2,"9":1,"13":2,"22":1,"23":2,"30":1,"34":2,"36":1,"37":3,"38":2,"39":2,"41":1,"44":1,"48":1,"50":1,"51":1,"58":3,"64":1,"71":1,"72":2,"73":2,"76":1,"78":1,"94":2,"95":2,"98":1,"102":1,"104":1,"107":1,"110":5,"111":2,"112":2,"115":1,"118":1,"121":1,"124":1,"132":1,"137":2,"138":2,"139":1,"140":1,"141":2,"142":3,"143":1,"144":1,"147":2,"148":5,"154":1,"160":1,"163":1,"165":2,"168":1,"169":1,"171":1,"172":3,"173":1,"176":6,"179":2,"182":1,"184":1,"187":3,"188":2,"189":1,"193":1,"195":3,"196":2,"198":1,"200":2,"206":1,"209":1,"211":1,"212":2,"218":2,"219":1,"222":1,"226":1,"240":6,"254":2,"259":2,"267":1,"269":3,"271":1,"272":1,"275":1,"277":2,"278":1,"284":5,"298":2,"299":1,"300":2,"310":1,"314":3,"318":1,"319":2,"331":1,"336":1,"340":1,"341":2,"344":1,"353":1,"356":1,"357":2,"358":2,"360":2,"361":1,"363":3,"364":1,"365":1,"373":2,"374":1,"375":2,"377":1,"382":2,"385":1,"393":2,"396":2,"397":1,"399":1,"400":1,"403":2,"407":4,"409":3,"410":4,"412":1,"413":5,"415":3,"416":3,"417":1,"436":3,"437":4,"438":4,"440":1,"444":1,"446":2,"454":3,"455":1,"456":1,"462":1,"470":1,"477":1,"484":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,"554":2,"558":1,"559":1,"579":1,"589":1,"595":1,"596":1,"597":1,"620":1,"632":4,"635":1,"636":1,"638":3,"647":1,"649":1,"657":1,"661":1,"662":1,"671":2,"676":1,"678":1,"679":1,"682":2,"683":1,"699":1,"707":1,"725":1,"727":1,"733":1,"734":1,"735":1,"736":1,"740":1,"745":2,"746":1,"750":1,"752":1}}],["anger",{"2":{"519":2}}],["annual",{"2":{"441":2}}],["annually",{"2":{"441":1,"451":1}}],["announced",{"2":{"39":1,"716":1}}],["announce",{"2":{"39":1}}],["announcements",{"2":{"698":1,"713":1,"729":2}}],["announcement",{"2":{"39":1,"475":1,"698":1,"699":1,"713":1,"729":1}}],["announcing",{"2":{"39":2}}],["analytics",{"0":{"697":1,"711":1},"2":{"697":2,"711":1}}],["analysis",{"2":{"168":1}}],["analyze",{"2":{"32":1}}],["analogously",{"2":{"105":1}}],["analogous",{"2":{"104":1}}],["analog",{"2":{"103":2}}],["another",{"0":{"325":1,"331":1,"486":1,"634":1,"675":1},"2":{"71":1,"78":2,"155":1,"168":1,"252":1,"253":1,"254":1,"258":1,"259":1,"311":1,"325":1,"373":1,"415":1,"483":1,"493":1,"509":1,"579":1,"625":1,"648":1,"746":1}}],["answer",{"2":{"42":1,"359":1,"522":1}}],["answers",{"2":{"16":1,"41":1,"393":1}}],["anticipate",{"2":{"36":1}}],["an",{"0":{"64":1,"66":1,"75":1,"197":1,"261":1,"273":1,"295":1,"337":1,"344":1,"348":1,"366":1,"600":1,"607":1,"756":1},"1":{"67":1,"68":1,"69":1,"70":1,"71":1,"72":1,"73":1,"74":1,"75":1,"76":1,"77":1,"78":1,"79":1,"80":1,"81":1,"82":1,"83":1,"84":1,"85":1,"86":1,"345":1,"346":1,"347":1,"348":1,"349":1,"601":1,"602":1,"603":1},"2":{"4":1,"6":1,"9":5,"12":1,"15":1,"21":1,"23":4,"26":1,"31":1,"34":1,"36":1,"37":2,"38":1,"41":1,"46":1,"47":2,"49":1,"53":1,"54":1,"55":1,"58":1,"62":2,"64":1,"69":1,"71":2,"72":1,"75":4,"78":1,"79":1,"88":1,"92":2,"94":1,"95":1,"96":1,"97":2,"98":1,"102":1,"104":1,"107":2,"109":1,"111":4,"112":2,"121":1,"126":1,"129":2,"136":2,"140":1,"141":2,"142":1,"144":1,"145":3,"146":2,"148":4,"156":1,"157":2,"160":1,"163":2,"164":1,"167":1,"184":2,"187":6,"188":1,"193":3,"195":3,"197":1,"202":1,"203":1,"206":2,"207":1,"213":1,"221":1,"223":3,"241":1,"253":2,"254":2,"269":2,"270":1,"273":1,"300":1,"311":3,"316":1,"317":1,"318":2,"319":3,"321":2,"325":1,"326":1,"327":1,"328":1,"331":1,"338":1,"348":1,"349":1,"350":1,"351":1,"352":1,"353":6,"357":2,"358":3,"360":2,"363":2,"364":2,"366":1,"369":2,"373":1,"374":3,"376":1,"377":1,"382":2,"394":1,"397":3,"406":1,"410":4,"412":1,"413":6,"414":1,"415":1,"416":1,"440":1,"441":1,"447":1,"451":1,"453":1,"460":1,"470":2,"477":2,"479":1,"483":2,"484":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":2,"558":1,"560":3,"563":1,"568":1,"579":1,"582":1,"591":1,"604":1,"605":1,"607":3,"612":1,"614":3,"616":1,"627":1,"632":1,"638":2,"644":1,"659":1,"668":1,"683":1,"693":1,"716":1,"743":1,"745":2,"748":1,"751":1,"756":3}}],["anyway",{"2":{"302":1,"307":1,"552":1}}],["anyone",{"2":{"283":1,"393":1,"414":1,"637":1}}],["anything",{"2":{"165":1,"346":1,"640":1}}],["anytrust",{"2":{"70":1,"71":1,"92":1,"94":1}}],["any",{"2":{"4":1,"7":1,"8":1,"10":1,"11":1,"14":1,"15":1,"23":4,"24":1,"26":1,"30":3,"31":2,"39":2,"42":1,"76":1,"109":1,"159":1,"184":1,"186":1,"191":1,"193":1,"206":1,"238":1,"240":1,"242":2,"244":1,"262":1,"279":1,"281":1,"282":1,"314":1,"346":1,"369":1,"371":1,"393":1,"408":1,"455":1,"471":2,"488":2,"503":4,"539":1,"560":1,"567":1,"570":1,"573":1,"579":1,"595":1,"647":1,"679":1,"694":1,"710":2,"720":1,"745":1}}],["and",{"0":{"33":1,"35":1,"39":1,"59":1,"73":1,"74":1,"77":1,"87":1,"200":1,"209":1,"211":1,"222":1,"250":1,"270":1,"278":1,"279":1,"312":1,"348":1,"352":1,"364":1,"372":1,"373":1,"386":1,"401":1,"421":1,"423":1,"426":1,"427":1,"430":1,"436":1,"437":1,"462":1,"491":1,"509":1,"542":1,"557":1,"558":1,"579":1,"586":1,"616":1,"672":1,"679":1,"702":1,"709":1},"1":{"34":1,"35":1,"36":1,"37":1,"38":1,"39":1,"40":1,"41":1,"42":1,"60":1,"61":1,"62":1,"63":1,"64":1,"65":1,"88":1,"89":1,"90":1,"251":1,"252":1,"253":1,"373":1,"374":2,"375":2,"376":2,"377":1,"437":1,"438":1,"587":1,"588":1,"589":1,"590":1,"591":1,"592":1,"673":1,"703":1,"704":1,"705":1},"2":{"0":1,"1":1,"2":4,"3":3,"6":5,"7":8,"8":6,"9":1,"10":3,"12":1,"18":3,"20":2,"21":1,"22":3,"23":11,"25":1,"26":10,"27":2,"30":12,"31":7,"32":6,"33":1,"34":6,"35":9,"36":7,"37":6,"38":8,"39":11,"40":11,"41":4,"42":3,"44":4,"46":4,"47":2,"48":1,"50":2,"51":3,"52":1,"53":1,"54":1,"55":1,"56":1,"57":2,"58":10,"59":3,"64":3,"65":1,"66":5,"67":3,"70":2,"71":13,"72":4,"73":3,"74":1,"75":6,"76":1,"77":2,"78":5,"79":2,"81":2,"88":1,"90":1,"92":2,"94":9,"95":3,"96":1,"104":1,"106":1,"107":9,"109":3,"110":3,"111":8,"112":3,"113":1,"115":10,"117":1,"121":1,"122":5,"124":4,"125":2,"126":1,"129":2,"130":2,"132":2,"135":1,"136":2,"141":5,"142":2,"144":3,"145":4,"146":4,"147":2,"148":4,"149":3,"150":1,"151":2,"153":3,"154":1,"157":3,"158":2,"159":2,"160":5,"163":3,"164":3,"165":2,"167":1,"168":5,"170":1,"171":2,"172":3,"176":4,"179":2,"181":1,"182":2,"184":4,"187":8,"188":4,"189":3,"190":3,"193":3,"194":1,"195":6,"196":3,"199":2,"200":5,"201":3,"202":2,"203":1,"204":1,"206":4,"211":5,"212":4,"213":1,"214":1,"218":4,"219":1,"220":4,"223":1,"224":1,"226":1,"227":1,"228":1,"230":1,"231":2,"232":3,"234":1,"237":2,"238":4,"241":4,"242":2,"244":1,"246":1,"247":1,"254":3,"258":1,"259":1,"262":1,"266":2,"268":1,"269":1,"270":3,"272":2,"273":1,"275":1,"277":5,"278":2,"279":1,"281":2,"282":1,"283":3,"284":4,"285":1,"286":2,"288":1,"289":1,"293":1,"296":2,"297":2,"298":4,"299":1,"300":1,"302":1,"304":1,"307":3,"309":1,"310":2,"311":1,"313":1,"314":2,"315":1,"316":2,"318":1,"319":4,"321":1,"322":2,"331":2,"333":1,"336":1,"337":1,"338":1,"340":1,"341":1,"343":2,"344":3,"346":4,"347":1,"348":1,"350":3,"352":4,"353":2,"355":1,"356":4,"357":5,"358":19,"359":4,"360":7,"361":1,"362":1,"363":2,"364":2,"366":1,"369":1,"370":1,"371":1,"373":2,"374":3,"375":6,"376":3,"377":2,"378":1,"382":14,"384":6,"386":1,"387":2,"388":3,"389":2,"390":3,"391":4,"393":2,"394":3,"397":1,"398":1,"399":3,"400":2,"401":2,"402":2,"403":4,"405":2,"406":3,"407":4,"408":1,"409":3,"410":8,"412":3,"413":9,"414":2,"415":7,"416":3,"417":2,"420":1,"421":1,"422":1,"425":1,"426":2,"429":1,"430":1,"431":1,"434":4,"435":1,"436":4,"437":1,"438":4,"440":2,"441":2,"443":4,"445":6,"446":3,"447":1,"454":1,"456":1,"458":1,"459":1,"461":2,"467":1,"470":1,"476":1,"477":5,"483":1,"484":1,"486":1,"488":6,"491":2,"494":5,"495":2,"501":3,"503":9,"504":1,"507":2,"509":8,"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":7,"557":3,"558":2,"559":2,"560":1,"561":1,"562":1,"563":2,"564":2,"565":1,"567":7,"569":1,"570":1,"572":1,"574":1,"576":1,"585":1,"587":1,"589":1,"591":1,"593":2,"594":1,"595":3,"596":1,"600":1,"602":1,"604":2,"605":4,"609":1,"612":2,"613":2,"615":1,"616":1,"621":1,"622":4,"623":2,"625":2,"632":1,"634":1,"637":4,"638":1,"639":2,"640":1,"641":2,"642":5,"645":2,"648":1,"649":1,"660":1,"665":1,"669":1,"670":1,"671":1,"675":1,"678":2,"679":4,"683":4,"693":2,"699":2,"705":1,"706":1,"708":2,"709":1,"714":1,"715":2,"716":4,"717":2,"720":1,"722":4,"725":1,"730":1,"731":2,"732":1,"733":3,"736":1,"738":1,"739":3,"740":1,"746":1,"751":3,"752":1,"756":2}}],["alias",{"0":{"460":1},"2":{"460":1,"492":2}}],["align",{"2":{"38":1}}],["aligned",{"2":{"8":1,"23":1}}],["algorithm",{"2":{"412":1}}],["alertmanager",{"2":{"503":2}}],["alerts",{"0":{"505":1},"2":{"503":2,"505":1}}],["alert",{"2":{"389":4}}],["alerting",{"2":{"26":1}}],["although",{"2":{"416":3}}],["alt",{"2":{"347":1,"348":1,"349":1}}],["alternatives",{"2":{"594":1}}],["alternatively",{"2":{"115":2,"168":1,"175":1,"188":1,"194":1,"202":1,"272":1,"292":1,"314":1,"358":1,"409":1,"416":1}}],["alternative",{"2":{"97":1,"156":1,"157":1,"197":1}}],["along",{"2":{"111":1,"124":1,"145":1,"148":2,"188":1,"213":1,"218":1,"311":1,"350":1,"434":1}}],["alongside",{"2":{"42":1,"92":1,"637":1}}],["alphab",{"2":{"448":1,"449":1,"687":1,"688":1,"689":1,"693":1,"697":1}}],["alpha",{"2":{"81":1,"125":2,"203":1}}],["alchemy",{"2":{"75":3}}],["alcohol",{"2":{"35":1}}],["already",{"2":{"23":1,"40":1,"41":1,"44":1,"76":1,"136":1,"186":1,"193":1,"201":2,"268":1,"278":1,"307":1,"309":1,"313":1,"362":1,"377":1,"410":1,"505":1,"518":1,"543":1,"576":1,"579":1,"623":1,"642":1,"643":1,"645":1,"654":1}}],["also",{"2":{"9":1,"23":1,"35":3,"38":1,"39":1,"40":2,"42":1,"62":1,"71":2,"96":1,"112":1,"115":2,"125":1,"128":1,"132":2,"153":1,"167":1,"171":1,"178":1,"193":1,"196":2,"199":1,"200":1,"212":1,"220":1,"246":1,"252":1,"253":1,"254":1,"260":1,"300":1,"311":1,"314":1,"317":1,"319":1,"348":1,"350":1,"358":1,"384":1,"389":2,"410":1,"415":1,"416":1,"456":1,"477":2,"501":1,"504":1,"509":1,"517":1,"529":1,"547":1,"574":1,"595":2,"609":1,"610":1,"611":1,"658":1,"659":1,"667":1,"670":1,"699":1,"705":1,"727":1,"731":1,"732":1}}],["allocation",{"0":{"445":1},"1":{"446":1},"2":{"445":1,"446":1}}],["allocating",{"2":{"40":1}}],["allocated",{"2":{"445":1,"446":1,"495":1}}],["allocate",{"2":{"35":1}}],["allotting",{"2":{"37":1}}],["allowance",{"0":{"261":1,"262":1,"264":1},"1":{"265":1},"2":{"259":1,"267":1}}],["allowances",{"0":{"259":1,"266":1},"1":{"260":1,"261":1,"267":1},"2":{"266":1}}],["allowing",{"2":{"41":1,"266":1,"647":1,"699":1}}],["allow",{"2":{"38":2,"71":1,"77":1,"154":1,"182":1,"190":1,"191":1,"206":2,"212":1,"224":1,"283":1,"317":1,"389":2,"393":1,"438":3,"469":1,"541":1,"542":1,"548":2,"552":1,"580":1,"615":1,"639":4,"645":4,"664":1,"692":1,"704":1,"706":1,"707":1,"708":1,"741":1}}],["allows",{"2":{"22":1,"40":2,"41":1,"71":1,"92":1,"96":2,"97":1,"105":1,"107":1,"126":1,"136":1,"150":1,"152":1,"155":1,"157":1,"159":1,"161":1,"163":2,"191":2,"195":1,"202":1,"258":1,"269":1,"279":1,"284":1,"298":1,"307":1,"322":1,"346":1,"363":1,"384":1,"389":1,"408":1,"417":1,"456":1,"505":1,"541":2,"552":2,"554":1,"563":1,"600":1,"605":1,"612":1,"662":1,"707":1,"731":2,"732":2}}],["allowed",{"2":{"14":1,"266":2}}],["all",{"0":{"249":1,"589":1},"2":{"3":1,"9":1,"10":2,"26":2,"41":1,"42":1,"46":3,"78":2,"110":2,"149":1,"153":1,"187":1,"200":1,"206":1,"212":3,"238":1,"240":3,"241":2,"242":1,"271":1,"272":5,"293":1,"296":3,"310":1,"315":1,"321":6,"356":1,"358":1,"364":2,"365":5,"374":1,"375":1,"376":1,"382":2,"387":1,"393":3,"395":3,"396":1,"402":1,"410":3,"413":3,"416":1,"437":1,"438":1,"444":1,"446":1,"462":1,"470":1,"488":1,"492":1,"519":2,"532":1,"536":1,"541":1,"552":1,"565":1,"587":1,"589":2,"592":1,"593":2,"596":1,"599":3,"601":2,"612":1,"615":1,"624":1,"625":1,"641":2,"654":2,"658":2,"660":4,"678":1,"693":1,"694":1,"716":1,"717":1,"751":1}}],["always",{"2":{"2":1,"34":1,"94":1,"189":1,"246":1,"374":1,"462":1,"585":2,"604":2,"727":1}}],["a",{"0":{"42":1,"54":1,"55":1,"87":1,"89":1,"90":1,"110":1,"129":1,"164":1,"165":1,"166":1,"167":1,"176":1,"187":1,"196":1,"197":1,"234":1,"235":1,"237":1,"238":1,"240":1,"243":1,"261":1,"262":2,"264":2,"265":1,"289":2,"290":1,"294":1,"310":1,"311":1,"336":1,"337":1,"340":1,"347":1,"348":1,"352":1,"382":1,"383":1,"384":1,"390":1,"391":1,"411":1,"415":1,"421":1,"426":1,"430":1,"451":1,"476":1,"485":1,"489":1,"500":1,"511":1,"519":1,"527":1,"529":1,"530":1,"534":1,"536":1,"537":1,"540":1,"559":1,"564":2,"568":1,"582":1,"587":2,"624":1,"633":1,"642":1,"653":1,"655":1,"656":1,"662":1,"664":1,"667":1,"674":1,"734":1,"741":1,"743":1,"745":1},"1":{"88":1,"89":1,"90":1,"165":1,"166":1,"167":1,"244":1,"245":1,"246":1,"247":1,"248":1,"249":1,"250":1,"251":1,"252":1,"253":1,"265":2,"291":1,"292":1,"412":1,"416":1,"477":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,"565":2,"583":1,"584":1,"585":1,"588":2,"589":2,"625":1,"626":1,"627":1,"628":1,"629":1,"630":1,"631":1,"632":1,"633":1,"634":2,"635":1,"636":1,"654":1,"655":1,"656":1,"657":1,"658":1,"659":1,"660":1,"661":1,"662":1,"663":1,"665":1,"666":1,"667":1,"668":1,"669":1,"670":1,"671":1,"672":1,"673":1,"674":1,"675":2,"676":1,"677":1,"735":1,"736":1,"737":1,"738":1,"739":1,"740":1,"742":1,"743":1,"744":2,"745":2,"746":1,"747":1,"748":1,"749":1,"750":1,"751":1,"752":1,"753":1,"754":1,"755":1,"756":1},"2":{"2":4,"3":3,"6":1,"7":3,"12":2,"13":5,"14":4,"15":2,"18":3,"20":1,"21":2,"22":2,"23":6,"24":1,"25":1,"26":2,"27":1,"28":1,"29":2,"30":6,"31":1,"33":3,"34":12,"35":3,"36":8,"37":7,"38":7,"39":3,"40":3,"41":8,"42":4,"44":1,"46":2,"47":2,"48":1,"50":1,"51":1,"53":1,"54":7,"55":5,"58":5,"59":2,"64":1,"65":1,"66":3,"67":3,"69":2,"70":2,"71":15,"72":3,"73":1,"75":7,"76":10,"77":1,"78":2,"79":1,"88":1,"89":2,"90":3,"92":1,"93":1,"94":21,"95":5,"96":3,"97":2,"102":1,"103":3,"104":8,"105":4,"107":13,"109":3,"110":15,"111":19,"112":3,"113":3,"115":6,"121":3,"122":10,"124":17,"126":5,"130":5,"131":1,"134":1,"136":7,"139":2,"141":4,"142":3,"144":2,"145":8,"148":2,"150":1,"152":1,"153":2,"154":5,"155":1,"156":1,"157":7,"158":1,"159":6,"160":12,"161":1,"163":3,"164":3,"165":3,"168":7,"171":1,"172":1,"173":1,"176":1,"178":2,"179":1,"182":1,"184":3,"185":1,"186":3,"187":17,"188":1,"189":6,"190":1,"191":2,"192":2,"193":11,"194":1,"195":4,"196":7,"197":2,"199":10,"200":10,"201":4,"202":5,"203":3,"204":3,"206":2,"209":2,"210":1,"211":3,"212":8,"213":2,"215":1,"218":1,"219":3,"220":3,"221":1,"222":3,"223":2,"226":1,"227":1,"231":1,"232":1,"234":1,"235":1,"237":3,"238":5,"240":7,"241":3,"242":2,"243":1,"244":1,"246":5,"247":1,"248":1,"252":2,"253":1,"254":2,"256":2,"257":1,"258":4,"261":1,"262":2,"263":1,"265":1,"266":2,"269":1,"270":7,"271":6,"272":5,"273":1,"277":1,"279":1,"282":1,"283":2,"284":4,"285":1,"286":2,"288":4,"289":2,"291":2,"292":1,"293":1,"295":1,"297":1,"298":2,"299":1,"300":1,"302":1,"307":1,"308":1,"310":2,"311":4,"312":2,"313":1,"314":1,"316":2,"317":3,"318":4,"319":7,"320":1,"322":2,"331":3,"333":1,"336":3,"338":1,"339":1,"340":4,"341":1,"344":2,"346":2,"347":3,"348":4,"350":1,"352":3,"355":2,"356":5,"357":4,"358":21,"359":3,"360":4,"361":3,"363":1,"364":4,"365":1,"366":1,"368":1,"369":4,"370":1,"371":6,"372":1,"373":6,"374":10,"375":13,"376":4,"377":7,"381":4,"382":10,"384":3,"385":4,"386":2,"387":1,"389":2,"390":1,"391":1,"393":2,"394":5,"395":3,"396":6,"397":5,"398":3,"399":2,"400":4,"401":1,"402":2,"403":4,"405":1,"406":5,"407":9,"408":3,"409":7,"410":4,"412":5,"413":10,"414":3,"415":13,"416":9,"417":2,"420":1,"421":1,"422":1,"426":1,"430":1,"432":1,"434":5,"435":3,"437":3,"438":2,"440":4,"443":2,"444":1,"445":2,"446":1,"447":4,"453":1,"454":6,"455":3,"456":1,"461":2,"462":2,"469":3,"470":4,"472":2,"477":6,"483":1,"484":5,"485":2,"486":1,"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":9,"510":2,"511":4,"515":1,"517":4,"518":1,"519":3,"522":3,"523":1,"525":2,"526":4,"527":3,"529":4,"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,"554":2,"555":3,"557":4,"558":1,"559":1,"562":4,"563":1,"564":7,"567":2,"568":2,"569":1,"573":2,"576":1,"577":1,"579":5,"580":1,"581":1,"585":4,"587":1,"591":2,"592":1,"593":2,"594":1,"595":3,"596":2,"597":2,"600":1,"601":1,"602":2,"604":1,"605":2,"607":1,"609":2,"610":2,"611":4,"612":2,"613":2,"614":1,"615":6,"616":1,"617":2,"620":3,"624":3,"632":8,"633":2,"634":1,"635":2,"637":1,"638":2,"640":1,"642":1,"643":3,"644":17,"648":3,"649":2,"653":1,"654":2,"656":2,"662":1,"663":1,"664":1,"666":1,"667":1,"671":5,"672":1,"674":2,"675":1,"676":1,"678":1,"679":1,"682":2,"683":9,"684":1,"692":3,"693":3,"695":1,"698":1,"699":2,"704":3,"705":5,"708":2,"710":2,"713":1,"714":1,"715":3,"716":7,"717":1,"720":3,"725":1,"727":1,"729":1,"730":1,"731":1,"732":1,"733":3,"734":1,"735":2,"736":1,"741":1,"742":1,"743":1,"744":1,"745":3,"746":4,"750":2,"751":4,"752":1,"756":2}}],["bzsidenode",{"2":{"144":6,"145":6,"148":8}}],["bitszn",{"2":{"687":1,"688":1,"689":1,"706":1,"707":1,"708":1}}],["bitter",{"2":{"519":2}}],["bidirectional",{"2":{"577":1}}],["billion",{"2":{"446":1}}],["bip44",{"2":{"389":2}}],["bip39",{"2":{"248":1}}],["bi",{"2":{"375":2}}],["bigger",{"2":{"393":1,"397":1}}],["big",{"2":{"129":8,"141":4,"144":4,"145":8,"148":18}}],["bincd",{"2":{"199":1}}],["bin",{"2":{"194":2,"196":2,"199":5,"200":2,"252":2,"253":2,"347":2,"507":2,"539":1,"540":1,"568":1,"622":2,"623":9}}],["binaries",{"2":{"188":2,"196":1,"540":2,"568":2,"621":1}}],["binarymerkleproof",{"2":{"103":4,"104":1,"129":2,"139":4,"144":9,"145":5,"148":8}}],["binary",{"0":{"539":1,"540":1,"568":1,"570":1},"2":{"81":1,"89":1,"95":2,"103":2,"104":1,"107":1,"124":1,"126":1,"130":1,"136":2,"139":2,"145":2,"159":2,"160":1,"196":5,"199":1,"200":1,"245":4,"257":1,"352":1,"482":1,"488":2,"510":2,"515":1,"538":1,"539":5,"540":4,"567":4,"568":4,"570":1,"607":1,"612":1,"669":1,"715":4,"716":5,"756":2}}],["binding",{"2":{"129":2,"146":2,"148":2}}],["bindings",{"2":{"121":7,"129":4,"146":2,"147":2,"148":4,"213":1,"218":1}}],["bind",{"2":{"129":6,"146":2,"147":2,"148":4,"601":1}}],["blind",{"2":{"519":2}}],["blue",{"2":{"122":1}}],["blobtypes",{"2":{"376":4,"382":6}}],["blobtxs",{"2":{"378":1}}],["blobtx",{"2":{"370":1,"382":4}}],["blobclient",{"2":{"364":2}}],["blobchan",{"2":{"271":2}}],["blobresponse",{"2":{"271":2}}],["blobpointer",{"2":{"94":1}}],["blobspace",{"0":{"433":1,"453":1},"1":{"434":1,"435":1}}],["blobsharerange",{"2":{"135":2,"148":4}}],["blobs",{"0":{"255":1,"270":1,"271":1,"287":1,"290":1,"306":1,"364":1,"370":1},"1":{"256":1,"257":1,"258":1,"259":1,"260":1,"261":1,"262":1,"263":1,"264":1,"265":1,"266":1,"267":1,"288":1,"289":1,"290":1,"291":2,"292":2,"293":1,"294":1,"295":1,"296":1,"371":1,"372":1,"373":1,"374":1,"375":1,"376":1,"377":1,"378":1,"379":1,"380":1,"381":1,"382":1,"383":1,"384":1},"2":{"94":1,"122":1,"135":1,"256":1,"258":2,"270":5,"271":11,"272":13,"288":1,"304":1,"321":1,"342":1,"358":1,"364":13,"365":17,"370":1,"371":1,"374":1,"375":3,"381":2,"384":1,"434":2,"437":1,"470":1,"693":1,"705":1}}],["blobstreamgit",{"2":{"369":1}}],["blobstreamwrapper",{"2":{"147":4}}],["blobstreamxcd",{"2":{"193":1}}],["blobstreamxdatacommitmentstored",{"2":{"129":4,"146":4,"147":2}}],["blobstreamxwrapper",{"2":{"129":14,"146":6,"147":4}}],["blobstreamxsp1",{"2":{"121":1,"129":1,"146":1,"147":1,"148":1}}],["blobstreamx",{"0":{"193":1},"1":{"194":1,"195":1},"2":{"76":4,"94":3,"96":2,"103":1,"105":1,"121":4,"129":16,"139":2,"146":8,"147":4,"148":10,"185":1,"187":2,"188":5,"189":2,"193":5,"195":4,"196":9,"197":2,"200":7,"202":3,"218":4,"219":1}}],["blobstream",{"0":{"82":1,"96":1,"99":1,"102":1,"108":1,"109":1,"117":1,"120":1,"129":1,"150":1,"185":1,"205":1,"206":1,"207":1,"208":1,"212":1,"213":1,"214":1,"215":1,"217":2,"218":1,"219":1,"220":1,"368":1},"1":{"83":1,"84":1,"100":1,"101":1,"102":1,"103":1,"104":1,"105":1,"106":1,"107":1,"109":1,"110":1,"111":1,"112":1,"113":1,"114":1,"115":1,"116":1,"117":1,"118":1,"119":1,"121":1,"122":1,"123":1,"124":1,"125":1,"126":1,"127":1,"128":1,"129":1,"130":1,"131":1,"132":1,"133":1,"134":1,"135":1,"136":1,"137":1,"138":1,"139":1,"140":1,"141":1,"142":1,"143":1,"144":1,"145":1,"146":1,"147":1,"148":1,"149":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,"163":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,"186":1,"187":1,"188":1,"189":1,"190":1,"191":1,"192":1,"193":1,"194":1,"195":1,"196":1,"197":1,"198":1,"199":1,"200":1,"206":1,"207":2,"208":1,"209":2,"210":2,"211":2,"212":1,"213":1,"214":2,"215":2,"216":2,"218":2,"219":2,"220":2,"369":1},"2":{"26":1,"75":3,"76":6,"81":1,"82":1,"93":1,"95":1,"96":1,"97":1,"102":4,"103":22,"106":1,"107":9,"109":3,"110":2,"111":4,"112":3,"113":1,"115":2,"117":1,"121":4,"122":2,"124":5,"126":2,"129":7,"139":4,"145":6,"146":5,"147":5,"148":13,"149":1,"150":2,"151":2,"153":2,"156":1,"159":1,"160":2,"163":2,"171":1,"186":1,"188":2,"197":1,"201":3,"202":1,"203":4,"204":2,"206":11,"207":3,"209":2,"210":2,"211":3,"212":12,"213":6,"214":4,"215":2,"216":8,"218":8,"219":3,"220":3,"254":1,"368":3,"369":6}}],["blob",{"0":{"134":1,"152":1,"153":1,"154":1,"155":1,"156":1,"168":1,"169":1,"170":1,"175":1,"177":1,"178":1,"265":1,"336":1,"371":1,"384":1},"1":{"135":1,"153":1,"154":1,"155":1,"156":1,"169":1,"170":1,"176":1,"177":1,"178":1},"2":{"76":1,"78":1,"94":1,"95":2,"121":4,"122":2,"124":1,"129":4,"130":1,"132":6,"134":1,"135":7,"140":1,"141":1,"145":2,"146":2,"148":4,"151":1,"152":1,"153":8,"154":2,"155":1,"158":1,"160":3,"161":1,"168":5,"169":1,"171":1,"173":1,"175":1,"176":5,"179":7,"184":2,"262":4,"263":1,"265":5,"266":2,"270":30,"271":5,"272":2,"291":3,"292":3,"296":1,"314":8,"319":12,"320":1,"321":10,"322":3,"336":5,"356":8,"357":6,"358":51,"360":11,"361":1,"364":34,"365":2,"370":2,"371":2,"374":6,"375":6,"376":6,"379":2,"380":3,"381":24,"382":14,"383":1,"384":6,"434":1,"435":1,"437":1,"557":4,"558":2,"559":2,"683":4}}],["blocktimes",{"2":{"282":1}}],["blockdataroot",{"2":{"145":3,"148":8}}],["blockres",{"2":{"129":2,"135":2}}],["blocknumber",{"2":{"129":2,"146":2,"341":2}}],["block",{"0":{"77":1,"115":1,"118":1,"129":1,"328":1,"377":1,"398":1,"589":1,"594":1,"607":2,"756":2},"2":{"76":1,"78":12,"103":16,"104":2,"105":1,"107":5,"109":1,"110":34,"111":2,"112":4,"113":17,"115":15,"122":4,"124":3,"126":3,"129":10,"134":1,"135":6,"136":1,"137":1,"145":16,"146":4,"148":6,"153":3,"156":1,"157":3,"159":1,"168":1,"172":1,"184":1,"194":4,"200":2,"201":6,"206":1,"212":2,"213":1,"218":3,"266":2,"273":2,"282":1,"284":1,"307":1,"319":1,"321":6,"322":1,"328":10,"336":1,"340":3,"358":2,"366":2,"371":2,"377":2,"382":2,"390":1,"393":3,"394":4,"395":2,"396":4,"397":4,"398":2,"401":1,"404":1,"406":1,"407":8,"408":3,"409":4,"410":3,"412":2,"414":2,"415":10,"416":13,"434":3,"436":1,"438":4,"441":3,"444":1,"456":1,"470":1,"477":2,"495":1,"507":2,"564":2,"587":1,"589":2,"592":3,"593":1,"594":2,"595":1,"605":3,"607":4,"625":2,"639":4,"665":1,"667":1,"683":3,"693":1,"715":2,"716":1,"745":1,"751":1,"756":5}}],["blocks",{"0":{"113":1},"2":{"71":2,"77":1,"110":3,"111":4,"112":1,"113":12,"115":13,"124":4,"130":1,"146":2,"157":2,"163":2,"193":1,"201":1,"282":2,"284":3,"377":2,"393":1,"397":2,"398":1,"401":1,"408":1,"409":1,"437":3,"438":3,"441":1,"469":1,"477":3,"509":2,"517":1,"589":3,"592":4,"593":2,"595":1,"692":1,"704":1,"705":2,"716":1,"717":2,"739":1}}],["blockscout",{"2":{"62":1,"77":2}}],["blockchains",{"0":{"239":1,"282":1,"405":1,"413":1},"1":{"240":1,"241":1,"242":1},"2":{"53":3,"56":1,"58":1,"112":1,"227":1,"232":2,"238":3,"240":2,"241":2,"242":1,"393":2,"395":1,"403":1,"405":1,"413":6,"414":1,"509":1}}],["blockchain",{"0":{"238":1,"402":1,"411":1},"1":{"412":1},"2":{"18":1,"34":1,"51":1,"53":1,"110":3,"111":3,"115":1,"119":1,"224":3,"237":1,"238":3,"240":3,"282":2,"283":2,"350":3,"360":3,"382":2,"393":1,"394":1,"400":1,"402":1,"403":1,"405":1,"406":2,"412":3,"413":1,"414":2,"417":1,"440":2,"453":1,"454":2,"580":1,"594":2,"596":1,"637":1,"714":1,"717":1}}],["blog",{"2":{"30":1,"32":1}}],["b",{"2":{"115":10,"148":24,"364":4,"369":1,"445":1,"446":1,"567":1,"644":8}}],["brew",{"2":{"503":1,"622":4}}],["brevity",{"2":{"413":1}}],["break",{"2":{"462":1,"573":1}}],["breaking",{"2":{"376":1,"714":4,"715":1,"716":1,"727":1}}],["breaks",{"2":{"31":1,"37":1}}],["brush",{"2":{"238":1}}],["broken",{"2":{"519":2,"715":1}}],["broadcasting",{"2":{"470":1,"522":2,"531":2,"693":1,"705":1,"706":1,"751":2}}],["broadcasttx",{"2":{"382":2}}],["broadcast",{"2":{"195":2,"266":2,"414":1,"507":4,"531":2,"708":1}}],["broader",{"2":{"37":1,"40":1,"58":1,"679":1}}],["browsers",{"2":{"302":1}}],["browser",{"0":{"420":1,"425":1,"667":1},"2":{"67":1,"420":1,"503":1,"667":1,"695":1}}],["branding",{"0":{"52":1}}],["brand",{"2":{"41":1,"52":1}}],["branch",{"2":{"2":3}}],["brightlystake",{"2":{"687":1,"688":1,"689":1,"706":1,"707":1,"708":1}}],["brightest",{"2":{"58":1}}],["bridging",{"0":{"59":1},"1":{"60":1,"61":1,"62":1,"63":1,"64":1,"65":1},"2":{"59":1,"72":1,"212":2,"227":1}}],["bridge=",{"2":{"542":4}}],["bridgefulllight",{"2":{"246":1,"247":1,"248":1}}],["bridgesudo",{"2":{"563":1}}],["bridges",{"2":{"212":1,"717":1}}],["bridgecreator",{"2":{"85":1,"86":1}}],["bridge",{"0":{"60":1,"63":1,"227":1,"469":1,"476":1,"477":1,"479":1,"481":1,"483":1,"484":1,"485":1,"487":1,"587":1,"692":1,"702":1,"704":1,"709":1,"739":1,"748":1,"749":1,"750":1},"1":{"61":1,"62":1,"64":1,"65":1,"477":1,"478":1,"479":1,"480":2,"481":1,"482":2,"483":2,"484":2,"485":3,"486":3,"487":2,"588":1,"589":1,"703":1,"704":1,"705":1,"750":1},"2":{"23":2,"26":3,"27":1,"59":2,"61":1,"62":1,"64":1,"65":2,"69":2,"73":1,"78":8,"79":1,"85":1,"86":1,"90":1,"111":2,"148":4,"206":1,"227":2,"244":1,"246":2,"247":2,"248":2,"254":1,"310":1,"413":1,"467":6,"469":1,"470":2,"476":2,"477":4,"478":1,"482":1,"483":9,"484":12,"485":9,"486":3,"487":2,"541":2,"542":4,"552":2,"554":1,"561":1,"562":4,"563":10,"573":2,"574":1,"577":1,"579":3,"587":1,"610":1,"611":1,"691":1,"692":1,"693":2,"699":1,"704":1,"705":2,"709":7,"717":1,"719":1,"731":1,"739":20,"740":1,"746":3,"748":2,"749":2,"750":2}}],["briefly",{"2":{"38":1,"72":1}}],["bringing",{"2":{"503":1}}],["bring",{"2":{"35":1,"58":1}}],["bound",{"2":{"683":1}}],["bounds",{"2":{"159":2,"164":1}}],["bot",{"2":{"640":1}}],["both",{"2":{"22":1,"23":1,"27":1,"73":1,"78":1,"82":1,"111":1,"171":1,"189":1,"211":1,"284":1,"292":1,"375":1,"406":1,"410":1,"413":1,"434":1,"443":1,"503":1,"511":1,"531":1,"538":1,"640":1,"733":1}}],["bonded",{"2":{"509":2}}],["bonddenomalias",{"2":{"460":1}}],["bonddenom",{"2":{"459":1}}],["bonus",{"2":{"361":1,"381":2}}],["box",{"2":{"242":1}}],["boxes",{"2":{"64":1}}],["bootstrappeers",{"2":{"576":2}}],["bootstrappers",{"2":{"576":1,"693":1}}],["bootstrapper",{"2":{"542":1,"576":3,"709":1}}],["bootstrapping",{"0":{"454":1},"2":{"438":1}}],["bootstrap",{"0":{"576":1},"2":{"454":1,"579":1}}],["boolean",{"2":{"544":2,"547":1}}],["bool",{"2":{"103":2,"129":2,"148":2}}],["boost",{"2":{"40":1}}],["book",{"2":{"34":1,"119":1}}],["board",{"2":{"37":2}}],["borrowing",{"2":{"34":1}}],["body",{"2":{"6":1,"522":2}}],["bak",{"2":{"585":10,"602":2}}],["bad",{"2":{"398":2}}],["bare",{"2":{"614":1}}],["bar",{"2":{"384":1}}],["barrier",{"2":{"71":2}}],["balances",{"0":{"263":1,"525":1},"2":{"267":2,"311":1,"402":2,"470":1,"483":1,"493":5,"522":1,"525":6,"537":2,"632":1,"693":1,"705":1}}],["balancecelestia",{"2":{"262":1,"263":1,"324":1}}],["balance",{"0":{"324":1,"325":1,"331":1},"2":{"78":4,"211":1,"254":1,"262":3,"263":2,"267":1,"324":2,"325":7,"352":3,"390":1,"391":1,"495":1,"525":3,"591":1,"632":1,"671":1}}],["batcher",{"2":{"284":5,"341":1,"343":1,"348":4}}],["batched",{"2":{"210":1}}],["batches",{"0":{"343":1},"2":{"71":2,"75":1,"94":1,"96":2,"104":1,"129":2,"146":2,"148":2,"156":1,"200":5,"218":2,"284":3,"343":2,"415":1}}],["batchposter",{"2":{"97":1}}],["batch",{"0":{"75":1},"2":{"71":4,"72":1,"73":2,"75":2,"76":5,"78":3,"80":1,"94":8,"105":2,"124":4,"145":1,"193":1,"201":1,"343":4,"639":8}}],["backward",{"2":{"714":2}}],["backed",{"2":{"604":2}}],["backers",{"2":{"445":2,"446":2}}],["backend=test",{"2":{"751":2}}],["backend=",{"2":{"507":2}}],["backend",{"2":{"246":6,"247":6,"248":6,"252":2,"253":2,"266":2,"294":2,"312":4,"484":2,"489":4,"507":8,"527":3,"531":4,"536":2,"547":1,"632":2,"656":2,"657":2,"659":2,"672":4}}],["background",{"0":{"554":1,"734":1},"1":{"735":1,"736":1,"737":1,"738":1,"739":1,"740":1},"2":{"128":2,"129":4,"135":4,"148":2,"357":2,"358":2,"360":2,"487":1,"597":1,"635":1,"676":1,"734":1,"735":1,"736":1,"740":1,"750":1}}],["back",{"2":{"65":1,"75":1,"94":1,"97":1,"115":1,"164":3,"184":1,"187":1,"254":1,"270":2,"284":1,"356":1,"358":2,"360":2,"361":2,"364":2,"381":2,"486":1,"634":1,"638":1,"648":1,"675":1,"756":2}}],["basis",{"2":{"716":1}}],["basic",{"0":{"315":1},"1":{"316":1,"317":1,"318":1},"2":{"314":1,"315":1,"503":1,"604":1,"608":1,"613":2}}],["basically",{"2":{"242":1}}],["bashmoniker=",{"2":{"751":1}}],["bashmake",{"2":{"539":1,"567":4}}],["bashhermes",{"2":{"640":1,"641":1,"643":3,"644":2}}],["bashvalidator",{"2":{"745":1}}],["bashvim",{"2":{"639":1}}],["bashver=",{"2":{"623":3}}],["bashzsh",{"2":{"623":1}}],["bashpersistent",{"2":{"585":3}}],["bashprometheus",{"2":{"503":1}}],["bashprivate",{"2":{"78":1}}],["bashsuccess",{"2":{"641":1,"644":2}}],["bashsudo",{"2":{"563":2,"615":1,"622":4}}],["bashseeds=$",{"2":{"585":2}}],["bashnano",{"2":{"563":1}}],["bashenter",{"2":{"745":1}}],["bashecho",{"2":{"623":2}}],["basherror",{"2":{"563":1}}],["bashexport",{"2":{"259":1,"266":1,"316":2,"317":1,"331":1,"336":1,"340":1,"341":1,"515":1,"521":1,"527":1,"529":1,"542":2,"614":7}}],["bashfatal",{"2":{"562":1}}],["bashfunding",{"2":{"78":1}}],["bashbash",{"2":{"540":1,"568":1}}],["bashbalances",{"2":{"525":2}}],["bash$home",{"2":{"346":3}}],["bashgas",{"2":{"522":1}}],["bashgrafana",{"2":{"503":1}}],["bashgo",{"2":{"269":1,"623":1}}],["bashgit",{"2":{"74":1,"539":3,"567":3}}],["bashroot",{"2":{"253":1}}],["bashdocker",{"2":{"252":4,"253":4,"614":3,"616":3,"617":3}}],["bash",{"2":{"246":3,"247":3,"248":3,"252":2,"253":1,"312":2,"318":2,"336":1,"502":1,"507":1,"515":1,"519":1,"520":2,"522":3,"523":2,"524":2,"525":2,"540":1,"547":1,"555":2,"557":1,"558":1,"559":1,"568":1,"585":1,"615":1,"616":1,"617":1,"622":3,"623":4,"645":2}}],["bashcargo",{"2":{"363":1}}],["bashcast",{"2":{"340":1,"341":1}}],["bashcurl",{"2":{"336":1,"342":1,"502":1}}],["bashcontainer",{"2":{"252":1,"253":1}}],["bashcd",{"2":{"200":1,"514":1,"516":1,"519":1,"539":1,"567":1,"596":3,"615":1,"623":4}}],["bashcelestia",{"2":{"76":2,"261":2,"262":3,"263":1,"264":1,"265":1,"266":1,"267":1,"310":3,"311":6,"312":3,"314":3,"317":1,"318":2,"319":4,"321":5,"322":1,"324":1,"325":2,"326":1,"327":1,"328":2,"329":1,"330":1,"331":3,"332":1,"333":2,"334":1,"379":1,"380":1,"470":2,"527":3,"529":2,"542":1,"555":2,"560":2,"561":2,"567":1,"585":6,"647":1,"693":1,"745":2,"748":1,"749":1}}],["bash2024",{"2":{"199":1,"607":1,"756":1}}],["bashyarn",{"2":{"1":1}}],["basechainhandle",{"2":{"644":8}}],["baseaccount",{"2":{"524":3}}],["basefeepergas",{"2":{"340":1}}],["base64=truecelestia",{"2":{"321":2}}],["base64=true",{"2":{"321":3}}],["base64",{"2":{"127":1,"132":1,"137":1,"141":1,"142":1,"319":4,"321":4,"337":1,"341":3,"342":1}}],["base",{"0":{"84":1,"86":1,"524":1},"2":{"69":2,"71":13,"72":6,"73":1,"76":2,"78":12,"79":1,"82":1,"84":1,"170":1,"201":1,"213":1,"216":2,"413":2,"523":4}}],["based",{"2":{"22":2,"23":1,"67":1,"107":1,"199":1,"209":1,"277":1,"282":2,"322":1,"350":1,"371":2,"374":2,"375":3,"387":1,"405":1,"417":1,"435":1,"440":1,"441":1,"454":1,"539":1,"567":1}}],["bandwidth",{"2":{"282":2,"308":1,"408":1,"478":1,"581":1,"626":1,"666":1,"719":1,"720":1,"742":1}}],["bank",{"2":{"266":2,"267":2,"277":1,"493":10,"507":4,"525":4,"537":2}}],["banner",{"2":{"38":1}}],["banners",{"2":{"38":1}}],["ban",{"0":{"14":1,"15":1},"2":{"13":1,"14":2,"15":1}}],["buckets",{"2":{"639":8}}],["bug",{"2":{"562":2,"720":1}}],["burned",{"2":{"494":1}}],["bubs",{"0":{"221":1,"223":1},"1":{"222":1,"223":1,"224":2,"225":2,"226":2,"227":2,"228":2,"229":2,"230":2,"231":1},"2":{"221":2,"222":3,"223":2,"224":1,"225":1,"226":3,"227":2,"228":2,"229":1,"230":1,"231":1,"286":1}}],["businesses",{"2":{"38":1}}],["budget",{"2":{"35":3,"40":1}}],["button",{"2":{"72":1,"73":1,"384":2,"389":4,"425":1,"503":1}}],["but",{"2":{"7":1,"23":1,"26":1,"29":1,"34":1,"35":3,"38":1,"41":1,"42":1,"58":1,"71":4,"94":2,"95":1,"110":1,"111":2,"146":2,"147":2,"148":2,"155":1,"193":1,"195":1,"204":1,"211":1,"212":2,"238":1,"279":1,"280":1,"292":1,"348":1,"377":1,"402":1,"415":1,"437":1,"446":2,"462":1,"504":1,"515":1,"518":1,"592":2,"605":1,"607":1,"624":1,"640":1,"655":1,"679":1,"699":1,"715":1,"717":1,"746":1,"756":1}}],["buildmake",{"2":{"567":1}}],["builds",{"2":{"515":1}}],["builders",{"2":{"240":1}}],["builder",{"2":{"199":1}}],["buildcd",{"2":{"187":1}}],["build",{"0":{"199":1,"200":1,"232":1,"356":1},"1":{"233":1,"234":1,"235":1,"236":1,"237":1,"238":1,"239":1,"240":1,"241":1,"242":1},"2":{"37":1,"150":1,"151":1,"163":1,"168":1,"187":2,"188":2,"196":3,"198":2,"199":14,"200":22,"206":1,"215":1,"223":1,"240":1,"276":1,"453":1,"514":2,"515":3,"516":2,"519":2,"539":1,"567":10,"622":4,"623":1,"649":1}}],["building",{"0":{"1":1,"223":1,"233":1,"354":1,"411":1,"539":1},"1":{"224":1,"225":1,"226":1,"227":1,"228":1,"229":1,"230":1,"234":1,"235":1,"236":1,"355":1,"356":1,"357":1,"358":1,"359":1,"360":1,"412":1},"2":{"151":1,"200":1,"234":1,"298":1,"352":1,"567":1,"621":1,"669":1}}],["built",{"0":{"222":1,"540":1,"568":1},"2":{"1":1,"95":1,"151":1,"153":1,"163":1,"199":2,"209":1,"218":1,"245":1,"283":2,"284":1,"286":1,"412":2,"438":1,"455":1,"477":2,"509":1,"538":1,"540":3,"568":3,"637":2,"647":1}}],["bypassing",{"2":{"554":1}}],["bytes=",{"2":{"683":1}}],["bytes28",{"2":{"141":2,"142":2}}],["bytes1",{"2":{"141":2,"142":2}}],["bytes",{"0":{"683":1},"2":{"103":4,"139":2,"140":2,"141":5,"142":1,"143":4,"148":6,"159":1,"189":4,"270":4,"319":1,"358":2,"371":4,"374":1,"381":4,"382":2,"408":2,"409":3,"415":1,"682":1,"683":4}}],["bytes32",{"2":{"103":8,"127":1,"132":1,"137":1,"141":2,"144":3,"145":2,"148":2,"189":11,"191":3,"192":2,"204":1}}],["byte",{"2":{"76":3,"110":8,"129":10,"141":17,"142":7,"144":4,"145":6,"148":36,"187":2,"211":1,"270":4,"271":2,"272":2,"358":4,"375":1,"376":1,"381":4,"382":4}}],["by",{"0":{"129":1,"328":1,"590":1},"2":{"7":1,"16":1,"18":1,"20":1,"22":2,"23":2,"32":1,"35":1,"36":1,"37":1,"40":1,"41":1,"42":1,"54":1,"58":2,"66":1,"70":1,"71":2,"72":1,"74":1,"75":1,"76":2,"78":3,"94":1,"97":1,"103":2,"105":1,"107":7,"109":1,"110":1,"111":5,"113":2,"115":2,"124":3,"125":1,"126":1,"127":1,"129":6,"132":1,"139":4,"142":1,"145":2,"148":6,"149":1,"153":2,"156":1,"157":1,"159":2,"160":2,"163":1,"168":1,"172":1,"187":1,"189":2,"190":1,"191":2,"197":1,"199":2,"200":1,"201":3,"203":2,"204":1,"206":2,"218":1,"219":1,"221":1,"222":1,"240":1,"241":1,"242":1,"244":1,"252":1,"253":2,"254":1,"258":1,"266":1,"277":1,"281":2,"282":1,"284":2,"288":1,"291":2,"292":1,"302":1,"304":1,"306":1,"312":3,"319":3,"328":4,"329":4,"334":6,"343":2,"346":1,"352":2,"356":1,"358":1,"365":2,"369":1,"372":2,"373":6,"374":5,"375":6,"376":2,"377":2,"390":1,"394":1,"395":1,"396":1,"397":1,"398":1,"407":2,"408":1,"410":2,"412":3,"414":2,"415":5,"416":3,"435":1,"437":3,"438":3,"441":1,"446":1,"447":1,"454":1,"455":1,"484":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,"567":1,"574":2,"576":2,"579":2,"592":1,"594":1,"595":1,"601":2,"604":2,"607":2,"615":1,"632":2,"637":2,"638":1,"644":1,"651":1,"652":1,"653":1,"654":1,"661":1,"663":1,"669":1,"672":3,"683":4,"694":1,"706":1,"715":2,"717":2,"727":1,"745":1,"756":2}}],["bearer",{"2":{"336":2,"342":2}}],["bedrock",{"2":{"222":1}}],["belong",{"2":{"139":2,"153":2,"159":1}}],["below",{"2":{"20":1,"21":1,"41":1,"59":1,"71":1,"76":1,"82":1,"83":1,"84":1,"112":1,"122":1,"145":1,"156":1,"173":1,"218":1,"219":1,"269":1,"341":1,"342":1,"363":1,"412":1,"415":2,"438":1,"441":1,"445":1,"446":1,"467":1,"537":1,"539":1,"540":1,"568":1,"616":1,"617":1,"632":1,"671":1,"685":1,"687":1,"703":1,"705":1,"709":1,"745":1,"751":2}}],["bech",{"2":{"745":2}}],["bech32prefixconspub",{"2":{"389":2}}],["bech32prefixconsaddr",{"2":{"389":2}}],["bech32prefixvalpub",{"2":{"389":2}}],["bech32prefixvaladdr",{"2":{"389":2}}],["bech32prefixaccpub",{"2":{"389":2}}],["bech32prefixaccaddr",{"2":{"389":2}}],["bech32config",{"2":{"389":2}}],["because",{"2":{"95":1,"113":1,"158":1,"242":1,"282":1,"352":1,"358":1,"393":1,"403":1,"557":1,"593":1,"638":1}}],["becomes",{"2":{"289":1,"393":2,"579":1}}],["become",{"2":{"47":1,"50":1}}],["behave",{"2":{"506":1}}],["behavior",{"2":{"7":2,"8":2,"10":1,"12":2,"13":1,"14":1,"15":1,"314":1,"509":1,"554":1,"595":1,"665":1}}],["behalf",{"2":{"260":1}}],["behind",{"2":{"94":1,"282":1,"462":1,"602":1,"727":1}}],["beneficial",{"2":{"42":1}}],["benefit",{"2":{"38":2,"40":1,"96":1}}],["benefits",{"0":{"239":1},"1":{"240":1,"241":1,"242":1},"2":{"34":1,"38":1,"53":1,"56":1,"58":2,"232":2}}],["better",{"2":{"41":1,"42":1,"111":1,"151":1,"182":1,"184":1,"231":1,"438":1}}],["between",{"0":{"401":1,"642":1},"2":{"36":1,"37":1,"72":1,"115":1,"160":1,"168":1,"187":1,"211":1,"227":2,"413":1,"441":1,"541":1,"552":1,"625":1,"637":2,"638":1,"639":1,"642":1,"644":1,"717":1,"739":1}}],["betaarabicamocha",{"2":{"674":1}}],["betaarabica",{"2":{"346":1}}],["betamochaarabica",{"2":{"310":1,"311":2,"312":1,"389":1,"483":1,"485":1,"539":1,"567":1,"585":4,"596":1,"597":1,"614":2,"616":1,"617":1,"623":1,"631":1,"633":1,"670":1,"671":1,"672":1,"751":1}}],["beta",{"0":{"448":1,"545":1,"557":1,"678":1,"722":1,"723":1},"1":{"679":1,"680":1,"681":1,"682":1,"683":1,"684":1,"685":1,"686":1,"687":1,"688":1,"689":1,"690":1,"691":1,"692":1,"693":1,"694":1,"695":1,"696":1,"697":1,"698":1,"723":1},"2":{"23":4,"24":1,"26":4,"27":1,"97":1,"121":1,"216":3,"246":3,"252":1,"253":1,"254":1,"285":1,"311":1,"312":1,"336":1,"371":1,"385":1,"448":1,"484":1,"545":1,"550":1,"551":3,"554":1,"557":3,"564":1,"585":1,"614":1,"632":2,"671":1,"678":3,"679":2,"684":2,"693":1,"696":1,"698":2,"715":1,"716":3,"722":2,"729":1,"748":1}}],["beefy",{"2":{"196":1,"199":1,"200":1}}],["beer",{"2":{"35":1}}],["been",{"2":{"26":1,"78":2,"81":1,"94":1,"97":2,"103":3,"153":1,"204":1,"263":1,"267":2,"302":1,"307":1,"352":1,"393":1,"402":1,"403":1,"438":2,"509":1,"523":1,"537":1,"552":1,"554":1,"644":1,"670":1,"682":1,"736":1,"740":1}}],["beginkey",{"2":{"141":3,"160":2}}],["beginning",{"2":{"38":2,"141":2,"145":1,"193":1,"441":1}}],["begin",{"2":{"32":1,"115":1,"160":1,"259":1,"357":1,"503":1}}],["be",{"0":{"139":1},"1":{"140":1,"141":1,"142":1,"143":1,"144":1,"145":1,"146":1,"147":1,"148":1},"2":{"7":2,"10":2,"12":1,"20":1,"22":3,"25":3,"26":1,"28":1,"34":3,"35":3,"37":1,"38":3,"39":4,"42":4,"47":1,"58":1,"64":2,"65":1,"70":1,"71":5,"72":1,"75":1,"76":4,"77":2,"82":1,"90":1,"94":3,"95":2,"96":2,"97":1,"102":1,"107":1,"110":4,"111":5,"112":2,"113":3,"115":3,"121":8,"122":2,"124":3,"125":1,"126":1,"127":2,"128":1,"130":1,"131":1,"132":2,"134":1,"136":1,"137":3,"138":2,"140":2,"141":2,"142":2,"143":1,"144":1,"145":4,"146":2,"147":1,"148":3,"149":1,"151":3,"153":3,"154":4,"155":1,"156":2,"157":4,"159":2,"160":6,"163":2,"165":3,"167":2,"168":5,"170":1,"171":1,"172":4,"173":2,"175":2,"176":2,"178":1,"179":2,"182":1,"186":1,"187":11,"188":2,"189":1,"191":2,"193":3,"194":1,"195":3,"196":2,"199":1,"200":4,"201":3,"202":1,"203":1,"204":2,"206":1,"213":2,"215":1,"218":4,"219":2,"220":1,"222":1,"240":4,"242":2,"244":1,"245":1,"246":2,"252":1,"253":1,"254":3,"258":1,"260":1,"270":2,"277":1,"281":1,"282":3,"291":1,"293":1,"299":1,"302":2,"307":1,"310":1,"311":1,"312":1,"314":1,"315":2,"316":2,"319":6,"322":2,"325":1,"336":1,"343":3,"346":3,"348":1,"350":2,"351":1,"352":4,"353":1,"356":1,"358":1,"364":2,"369":1,"371":2,"372":1,"373":1,"374":3,"375":1,"376":2,"377":3,"382":4,"385":2,"386":2,"390":1,"391":1,"394":1,"397":1,"398":3,"400":2,"403":2,"404":1,"407":1,"408":1,"409":3,"410":1,"413":3,"415":2,"416":1,"434":2,"435":1,"436":1,"437":4,"438":3,"441":2,"446":2,"454":1,"462":2,"466":1,"470":1,"482":1,"483":1,"484":1,"485":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,"560":4,"563":1,"568":1,"574":1,"576":1,"579":2,"587":1,"590":1,"595":5,"600":1,"601":1,"604":2,"605":1,"614":1,"615":2,"621":1,"632":4,"633":1,"636":2,"637":1,"639":1,"640":2,"644":2,"647":3,"648":1,"653":1,"658":1,"659":1,"660":1,"663":1,"671":2,"672":1,"674":1,"679":1,"682":1,"683":4,"685":1,"703":1,"707":1,"709":1,"714":1,"715":3,"716":3,"725":3,"727":2,"731":1,"736":1,"738":2,"739":2,"740":1,"748":1,"751":3}}],["best",{"2":{"7":1,"54":1,"58":1,"241":1,"465":1,"504":1,"684":1,"699":1,"731":1}}],["being",{"2":{"7":1,"23":1,"25":1,"71":1,"76":2,"89":1,"95":1,"113":1,"135":2,"137":1,"138":1,"141":2,"142":1,"143":1,"144":1,"145":3,"172":1,"187":1,"199":1,"277":1,"322":1,"387":1,"400":1,"416":1,"477":1,"550":1,"587":1,"593":1,"648":1}}],["before",{"0":{"30":1},"2":{"2":1,"23":2,"26":1,"36":2,"37":1,"39":1,"72":1,"76":2,"97":1,"112":1,"115":3,"140":1,"156":1,"196":1,"218":1,"219":1,"314":1,"347":1,"389":3,"415":1,"517":1,"522":2,"550":1,"595":1,"615":1,"638":1,"654":1,"657":1,"716":1,"736":1,"751":2}}],["rf",{"2":{"539":2,"562":4,"567":2,"596":6,"623":8}}],["rfc",{"2":{"104":1}}],["rs",{"2":{"667":1}}],["rsmt2d",{"2":{"273":2}}],["rsvp",{"2":{"30":1}}],["rigorous",{"2":{"678":1}}],["right",{"2":{"8":1,"61":1,"76":1,"232":1,"384":1,"425":1,"454":1,"715":1}}],["richer",{"2":{"298":1}}],["risk",{"2":{"206":1}}],["r6a",{"2":{"202":1}}],["r",{"2":{"200":2,"329":2,"334":2,"341":4,"445":1,"446":2,"585":2}}],["r1cs",{"2":{"199":2}}],["r1",{"2":{"122":1}}],["r0",{"2":{"122":1}}],["rm",{"2":{"89":2,"539":2,"562":4,"567":2,"596":6,"623":16}}],["rc0",{"2":{"716":1}}],["rc",{"2":{"81":1,"89":2}}],["rpcs",{"2":{"470":1,"693":1}}],["rpc",{"0":{"75":1,"224":1,"279":1,"298":1,"300":1,"301":1,"304":1,"313":1,"383":1,"466":1,"467":1,"469":1,"559":1,"600":1,"601":1,"685":1,"687":1,"692":1,"694":1,"702":1,"703":1,"704":1,"706":1},"1":{"225":1,"226":1,"299":1,"300":1,"301":1,"305":1,"306":1,"307":1,"308":1,"309":1,"310":1,"311":1,"312":1,"313":1,"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":2,"334":2,"335":1,"336":1,"337":1,"338":1,"601":1,"602":1,"603":1,"695":1,"703":1,"704":1,"705":1},"2":{"75":3,"76":18,"78":7,"94":1,"111":4,"118":1,"121":5,"122":1,"127":1,"128":2,"129":8,"132":1,"135":2,"137":1,"146":4,"147":2,"148":6,"187":7,"188":4,"189":12,"193":1,"194":5,"195":4,"196":5,"203":2,"224":2,"225":1,"226":1,"252":2,"253":2,"257":1,"259":2,"266":5,"267":2,"268":1,"269":1,"279":2,"297":1,"298":3,"300":3,"301":1,"304":1,"307":1,"310":1,"311":4,"313":1,"314":2,"315":4,"316":1,"317":1,"326":1,"327":1,"336":1,"338":2,"340":2,"341":2,"348":7,"352":2,"357":2,"358":2,"360":2,"362":1,"363":4,"364":4,"365":2,"366":2,"383":1,"385":1,"389":10,"390":2,"391":2,"466":3,"467":4,"469":2,"470":1,"477":1,"483":1,"484":2,"493":4,"529":8,"530":1,"531":5,"541":2,"542":1,"552":2,"554":1,"555":2,"557":2,"559":3,"564":2,"569":1,"595":4,"600":1,"601":3,"602":1,"605":3,"614":5,"632":1,"639":16,"671":5,"672":3,"685":3,"687":25,"688":1,"689":2,"690":1,"692":2,"693":12,"703":3,"704":2,"705":6,"706":16,"708":1,"732":1,"748":1}}],["rblocks",{"2":{"73":1}}],["rounds",{"2":{"396":2}}],["round",{"2":{"328":2}}],["roughly",{"2":{"282":1,"377":1}}],["route",{"2":{"559":2}}],["router",{"2":{"541":1,"552":1}}],["routes",{"2":{"195":1,"559":1}}],["routine",{"2":{"36":1}}],["robusta",{"2":{"318":2}}],["robust",{"2":{"223":1}}],["rowproof",{"2":{"143":1,"144":1,"148":4}}],["rowproofs",{"0":{"144":1},"2":{"139":2,"144":2,"145":1,"148":2,"160":4}}],["rowroots",{"0":{"143":1},"2":{"139":4,"143":3,"148":4}}],["row",{"2":{"95":3,"107":4,"122":1,"124":6,"130":3,"132":8,"136":6,"137":8,"139":2,"141":1,"144":1,"145":1,"153":2,"159":2,"160":7,"328":2,"409":1,"415":4}}],["rows",{"2":{"95":2,"107":3,"124":1,"136":1,"139":8,"141":1,"143":1,"144":1,"153":2,"159":2,"407":1}}],["ro",{"2":{"89":2}}],["room",{"2":{"371":1}}],["rooms",{"2":{"34":1}}],["roots",{"2":{"94":1,"95":5,"96":1,"107":2,"113":1,"122":1,"124":4,"132":2,"136":3,"137":2,"139":2,"141":1,"143":8,"145":1,"148":6,"153":6,"210":1,"212":2,"328":4,"407":3,"408":1,"409":1,"415":2,"416":1,"477":1}}],["root",{"0":{"126":1},"1":{"127":1,"128":1},"2":{"61":1,"74":2,"77":1,"78":1,"81":1,"95":2,"96":2,"97":1,"103":4,"104":4,"107":10,"110":1,"112":5,"113":1,"124":9,"126":8,"127":1,"129":12,"130":5,"132":1,"136":6,"139":14,"143":4,"144":1,"145":26,"148":10,"149":2,"153":8,"154":2,"159":5,"160":15,"172":4,"176":3,"179":2,"184":2,"206":1,"212":4,"253":2,"315":1,"407":1,"409":1,"410":1,"415":2,"434":1}}],["role",{"0":{"452":1},"1":{"453":1,"454":1,"455":1,"456":1,"457":1,"458":1,"459":1,"460":1},"2":{"54":1,"55":1,"71":1,"359":2,"456":1}}],["roles",{"2":{"30":1}}],["rolling",{"2":{"509":1}}],["rollback",{"2":{"488":8}}],["roll",{"2":{"344":1,"347":3}}],["rollkit",{"2":{"42":1,"55":2}}],["rollupinclusionproofs",{"2":{"140":1,"141":1,"142":1,"144":1,"145":1,"167":1}}],["rollupuserlogic",{"2":{"85":1,"86":1}}],["rollupadminlogic",{"2":{"85":1,"86":1}}],["rollupeventinbox",{"2":{"85":1,"86":1}}],["rollupcreator",{"2":{"76":1,"83":1,"84":1,"85":1,"86":1}}],["rollup",{"0":{"55":1,"59":1,"60":1,"62":1,"63":1,"64":1,"66":1,"109":1,"111":1,"114":1,"148":1,"184":1,"237":1,"344":1},"1":{"60":1,"61":2,"62":2,"63":1,"64":2,"65":2,"67":1,"68":1,"69":1,"70":1,"71":1,"72":1,"73":1,"74":1,"75":1,"76":1,"77":1,"78":1,"79":1,"80":1,"81":1,"82":1,"83":1,"84":1,"85":1,"86":1,"112":1,"113":1,"115":1,"116":1,"117":1,"118":1,"119":1,"345":1,"346":1,"347":1,"348":1,"349":1},"2":{"38":1,"55":4,"59":2,"62":4,"64":1,"66":3,"69":1,"70":3,"73":2,"75":1,"77":1,"78":8,"79":2,"88":1,"98":1,"103":15,"107":1,"109":3,"110":11,"111":15,"115":4,"119":1,"122":2,"124":2,"125":1,"130":2,"134":1,"142":1,"143":1,"145":2,"148":4,"150":1,"151":1,"154":4,"155":2,"156":1,"157":7,"160":3,"163":3,"164":7,"165":2,"167":1,"168":7,"171":2,"172":5,"175":2,"176":4,"179":4,"182":1,"184":2,"212":1,"214":1,"215":1,"219":1,"222":1,"223":1,"235":1,"237":1,"238":1,"240":2,"241":1,"242":5,"284":4,"346":1,"348":2,"349":1,"410":1,"436":2,"438":7,"445":1,"453":1,"565":1,"632":2}}],["rollups",{"0":{"91":1,"150":1,"163":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,"177":1,"178":1,"215":1,"235":1,"438":1,"454":1},"1":{"92":1,"93":1,"94":1,"95":1,"96":1,"97":1,"98":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,"163":1,"164":2,"165":3,"166":3,"167":3,"168":2,"169":3,"170":3,"171":1,"172":2,"173":3,"174":3,"175":2,"176":3,"177":3,"178":3,"179":2,"180":2,"181":2,"182":1,"183":1,"184":1},"2":{"3":1,"37":1,"42":1,"53":1,"55":1,"69":1,"90":1,"109":1,"112":3,"149":1,"150":2,"151":2,"156":1,"163":3,"164":1,"171":2,"176":1,"182":2,"212":4,"215":1,"222":1,"232":1,"236":1,"237":1,"238":1,"240":2,"241":6,"242":3,"283":1,"339":1,"344":1,"394":1,"404":1,"413":1,"436":1,"438":3,"454":1,"461":1,"462":1,"678":1,"699":1,"722":1,"725":1,"727":1}}],["roster",{"2":{"26":1}}],["rustasync",{"2":{"365":1,"366":1}}],["rustuse",{"2":{"364":1}}],["rust",{"0":{"362":1},"1":{"363":1,"364":1,"365":1,"366":1,"367":1},"2":{"200":2,"206":1,"212":1,"362":1,"368":1,"637":1,"638":1}}],["rules",{"2":{"22":1,"111":1,"190":1,"413":1}}],["runners",{"2":{"470":1}}],["running",{"0":{"87":1,"89":1,"90":1,"252":1,"289":1,"544":1,"717":1},"1":{"88":1,"89":1,"90":1,"545":1,"546":1,"547":1,"718":1,"719":1,"720":1},"2":{"23":1,"54":1,"67":1,"74":1,"75":1,"76":1,"78":6,"79":1,"90":1,"94":2,"102":1,"111":1,"196":1,"197":1,"203":1,"212":1,"253":1,"257":1,"270":2,"276":1,"307":1,"308":1,"311":1,"312":3,"315":1,"316":1,"336":1,"347":1,"348":2,"349":1,"352":4,"373":1,"381":2,"385":1,"397":1,"465":1,"478":1,"484":2,"499":1,"501":1,"502":1,"503":1,"504":1,"507":2,"517":1,"529":1,"530":1,"543":2,"548":1,"554":3,"559":1,"563":1,"567":1,"579":1,"581":1,"585":2,"597":2,"607":3,"609":1,"611":2,"614":1,"615":1,"617":1,"621":1,"622":2,"626":1,"632":2,"635":1,"636":1,"638":1,"640":1,"654":1,"663":1,"666":1,"672":3,"677":1,"684":1,"699":1,"705":1,"727":1,"731":1,"732":1,"733":1,"734":1,"735":1,"736":2,"742":1,"746":3,"751":2,"756":4}}],["runs",{"2":{"375":1,"503":2,"515":1}}],["runtime",{"2":{"260":1,"607":4,"644":8,"756":4}}],["rundown",{"2":{"232":1}}],["run",{"0":{"54":1,"76":1,"77":1,"196":1,"197":1,"344":1,"352":1,"484":1,"485":1,"514":1,"608":1,"630":1,"633":1,"667":1,"674":1,"749":1,"751":1},"1":{"345":1,"346":1,"347":1,"348":1,"349":1,"485":1,"486":1,"609":1,"610":1,"611":1,"631":1,"632":1,"634":1,"675":1,"750":1},"2":{"1":1,"23":2,"26":1,"51":1,"54":2,"76":1,"77":1,"78":3,"79":1,"89":4,"90":1,"111":1,"187":1,"189":1,"193":4,"194":2,"195":1,"196":6,"199":3,"200":3,"202":2,"203":2,"220":1,"242":2,"251":1,"252":3,"258":1,"259":1,"264":1,"267":1,"269":1,"288":1,"298":1,"300":1,"302":1,"311":1,"313":1,"319":1,"344":1,"346":1,"347":1,"352":1,"355":1,"357":2,"358":2,"360":8,"363":1,"369":1,"465":1,"467":1,"470":1,"477":3,"482":1,"483":1,"485":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,"560":1,"562":1,"564":1,"568":1,"569":2,"570":1,"591":1,"596":1,"597":1,"608":1,"610":1,"611":2,"612":6,"614":9,"616":9,"617":9,"621":1,"623":2,"631":1,"633":1,"635":1,"637":1,"641":1,"655":1,"656":1,"657":1,"659":2,"660":1,"662":2,"667":2,"670":1,"674":1,"684":2,"693":1,"699":3,"705":1,"716":2,"717":1,"730":1,"731":1,"732":1,"733":3,"745":2,"746":1,"748":1,"749":1,"751":1}}],["rate=0",{"2":{"500":6,"751":6}}],["rates",{"2":{"441":1}}],["rate",{"2":{"391":3,"441":1}}],["rather",{"2":{"211":1,"238":1,"441":1,"461":1}}],["raas",{"2":{"235":1,"344":2}}],["ram",{"2":{"196":1,"199":1,"200":1,"202":1,"308":1,"369":1,"478":1,"567":1,"581":1,"626":1,"666":1,"719":3,"720":2,"742":1}}],["rawlog",{"2":{"382":4}}],["raw",{"2":{"140":1,"391":1,"477":2,"522":2,"585":10,"622":2,"665":1,"745":2,"751":2}}],["rawmessage",{"2":{"110":2,"113":2}}],["rand",{"2":{"76":1}}],["randomly",{"2":{"407":1}}],["randomness",{"2":{"199":1}}],["random",{"2":{"76":1,"168":1,"199":1,"396":1,"414":1,"416":2}}],["ranges",{"0":{"201":1},"1":{"202":1,"203":1,"204":1},"2":{"201":2,"218":3}}],["range",{"0":{"136":1},"1":{"137":1,"138":1},"2":{"76":1,"94":3,"127":1,"129":6,"132":2,"134":1,"136":3,"141":4,"143":2,"144":4,"145":2,"146":6,"148":26,"159":1,"189":1,"195":2,"196":5,"200":14,"201":1,"202":1,"203":1,"204":2,"212":1,"213":2,"410":2}}],["rapidly",{"2":{"44":1}}],["race",{"2":{"6":1}}],["raises",{"2":{"71":1}}],["raise",{"2":{"4":1,"30":1}}],["reinitializing",{"2":{"654":1}}],["reinitialize",{"2":{"550":1}}],["reindexing",{"2":{"605":2}}],["reindex",{"2":{"605":1}}],["rejoin",{"2":{"509":1}}],["rejected",{"2":{"416":1,"648":1}}],["rejects",{"2":{"416":1}}],["reject",{"2":{"8":1,"409":1}}],["redelegate",{"2":{"423":1,"427":1,"432":1}}],["reducing",{"2":{"414":1}}],["reduces",{"2":{"296":1}}],["reduce",{"2":{"240":1,"284":1}}],["reduced",{"2":{"34":1,"165":1,"240":1,"416":1,"679":1,"722":1}}],["redundant",{"2":{"398":1}}],["reed",{"2":{"407":2,"409":5,"412":1,"415":1}}],["revoke",{"0":{"264":1},"1":{"265":1},"2":{"260":1,"264":3}}],["revert",{"2":{"148":2}}],["reveal",{"2":{"95":1}}],["revisit",{"2":{"40":1}}],["reviewing",{"2":{"32":1}}],["reviewed",{"2":{"10":1,"20":1,"23":1}}],["review",{"0":{"72":1},"2":{"2":1,"30":1,"47":1,"71":1,"72":2,"426":2,"431":1}}],["retain",{"0":{"589":1},"2":{"587":1,"589":5,"592":4}}],["returning",{"2":{"270":2,"364":2}}],["returned",{"2":{"189":3,"321":1,"410":1,"509":1,"648":1,"745":1}}],["return",{"2":{"103":2,"113":10,"115":12,"147":8,"188":2,"189":3,"200":2,"321":3,"331":1,"359":2,"382":18,"389":2}}],["returns",{"2":{"76":1,"103":2,"111":4,"157":1,"189":1,"271":1,"272":1,"273":1,"319":1,"358":1,"359":2,"365":1,"366":1}}],["retrial",{"2":{"416":1}}],["retried",{"2":{"145":1}}],["retrievable",{"2":{"403":1,"436":1,"694":1}}],["retrievability",{"0":{"405":1,"436":1,"437":1},"1":{"437":1,"438":1},"2":{"402":1,"436":1,"437":1}}],["retrieval",{"2":{"399":1}}],["retrieving",{"0":{"270":1,"321":1,"364":1},"2":{"94":1,"399":1,"403":1,"595":1,"609":1}}],["retrievedblobs",{"2":{"270":2,"381":2}}],["retrieved",{"2":{"121":1,"140":1,"145":1,"315":1,"356":2,"357":2,"358":2,"364":8,"400":1,"438":1}}],["retrieve",{"2":{"94":2,"136":1,"304":1,"321":1,"337":1,"346":2,"356":2,"357":2,"358":2,"359":1,"364":2,"400":1}}],["retryable",{"2":{"78":6}}],["retryables",{"2":{"78":4}}],["rebasing",{"2":{"71":1}}],["rename",{"2":{"490":2}}],["renaming",{"2":{"490":2}}],["rentals",{"2":{"40":1}}],["renewals",{"2":{"22":1}}],["renewal",{"2":{"22":1,"25":1}}],["rewards",{"0":{"496":1},"2":{"423":1,"427":1,"432":1,"440":2,"444":1,"446":1,"447":1,"455":1,"456":1,"495":1,"496":3,"509":1}}],["reward",{"0":{"404":1},"2":{"37":1,"405":1}}],["regenerate",{"0":{"199":1}}],["regenerated",{"2":{"188":1}}],["regeneration",{"2":{"196":1}}],["regenerating",{"0":{"198":1},"1":{"199":1,"200":1},"2":{"188":1}}],["regen",{"2":{"125":4}}],["regions",{"2":{"41":1}}],["registering",{"2":{"195":1}}],["registerfunction",{"2":{"189":4}}],["register",{"0":{"189":1},"1":{"190":1,"191":1,"192":1},"2":{"189":3,"200":1,"639":2}}],["registered",{"2":{"30":1,"187":1,"190":2,"196":2,"220":1}}],["registry",{"2":{"187":1,"391":1,"585":3,"595":1}}],["registration",{"2":{"30":1,"31":1,"47":1,"189":1,"195":1}}],["regular",{"2":{"36":1,"42":1,"69":1,"412":1,"506":1}}],["regardless",{"2":{"6":1,"154":1,"385":1,"387":1}}],["refresh",{"2":{"639":2}}],["refreshments",{"0":{"35":1},"2":{"30":1,"35":6,"38":1}}],["refund",{"2":{"373":1,"375":1}}],["refunded",{"2":{"373":1}}],["refers",{"2":{"159":2,"477":1}}],["referred",{"2":{"121":1,"413":1,"415":2}}],["referrals",{"2":{"37":2}}],["referencing",{"2":{"112":1,"155":2,"157":1,"158":1,"184":2}}],["references",{"2":{"172":1}}],["referenced",{"2":{"153":1,"157":1,"168":1,"172":1,"176":1,"179":1}}],["reference",{"0":{"184":1},"2":{"0":1,"3":1,"36":1,"124":1,"164":1,"167":1,"171":1,"175":1,"176":1,"184":3,"296":1,"643":6}}],["refer",{"2":{"71":1,"121":1,"187":1,"193":1,"246":1,"252":2,"311":1,"312":1,"320":1,"361":1,"375":2,"381":1,"410":1,"441":1,"467":1,"483":1,"484":1,"509":1,"537":1,"550":1,"570":1,"612":1,"614":2,"632":2,"663":1,"671":1,"699":1,"748":1,"754":1}}],["rec",{"2":{"382":4}}],["recipient",{"2":{"331":1,"495":2}}],["recency",{"2":{"437":2}}],["recent",{"2":{"94":1,"340":1,"694":1}}],["receipt",{"2":{"446":1}}],["receiptsroot",{"2":{"340":2}}],["receiving",{"2":{"26":1,"365":2}}],["receiveroutine",{"2":{"607":4,"756":4}}],["receiver",{"2":{"78":4}}],["received",{"2":{"23":1,"147":1,"407":1,"416":2,"665":1}}],["receives",{"2":{"22":3,"393":1,"444":1,"456":1}}],["receive",{"2":{"20":1,"22":1,"26":1,"28":1,"47":1,"48":1,"187":1,"271":1,"272":1,"322":1,"365":1,"397":1,"407":1,"447":1,"679":1,"722":1}}],["recap",{"2":{"48":1}}],["recruiting",{"2":{"42":2}}],["reconstructing",{"2":{"409":1}}],["reconstruction",{"0":{"398":1}}],["reconstruct",{"2":{"397":1,"398":1,"438":1}}],["recoverable",{"2":{"409":1}}],["recovered",{"2":{"407":1}}],["recover",{"2":{"248":6,"519":2,"536":1}}],["recompute",{"2":{"95":1}}],["recommend",{"2":{"71":1,"90":1,"276":1,"348":2,"369":1,"461":1,"569":2,"699":1}}],["recommends",{"2":{"41":1}}],["recommendations",{"2":{"35":1}}],["recommended",{"0":{"718":1},"2":{"23":1,"29":1,"75":1,"184":1,"187":2,"196":1,"243":1,"301":1,"308":1,"385":1,"389":2,"416":1,"478":1,"484":1,"494":1,"565":1,"567":1,"581":1,"609":1,"620":1,"626":1,"632":1,"666":1,"733":1,"742":1,"746":1}}],["recognizable",{"2":{"71":1}}],["recognize",{"2":{"47":1}}],["recognition",{"2":{"37":1}}],["recorded",{"2":{"40":4}}],["record",{"2":{"34":1}}],["recordings",{"2":{"34":1,"40":3}}],["recording",{"0":{"40":1},"2":{"34":2,"40":6}}],["reactjs",{"2":{"389":1}}],["react",{"0":{"389":1},"2":{"388":1,"389":4}}],["reaching",{"2":{"451":1}}],["reaches",{"2":{"396":1,"441":1}}],["reached",{"2":{"382":2}}],["reach",{"2":{"23":1,"30":1,"37":1,"38":1,"39":1,"40":3}}],["reaped",{"2":{"377":1}}],["reapplying",{"2":{"25":1}}],["reapply",{"2":{"23":1}}],["real",{"2":{"75":1,"678":1,"736":1,"740":1}}],["readme",{"2":{"347":1,"348":1,"369":1,"510":1}}],["reads",{"2":{"284":2,"359":1}}],["reader",{"2":{"111":1,"121":1}}],["reading",{"2":{"111":2,"284":4,"311":1,"399":1,"483":1,"569":1,"632":1}}],["read",{"0":{"341":1},"2":{"64":1,"66":2,"81":1,"94":1,"95":1,"110":1,"111":1,"193":1,"302":1,"306":1,"307":1,"314":1,"321":1,"341":1,"374":1,"620":1,"694":1,"699":1}}],["ready",{"2":{"26":1,"125":1,"234":1,"240":1,"350":1,"517":1,"595":1,"715":1,"751":1}}],["reasonable",{"2":{"202":1}}],["reasonably",{"2":{"7":1}}],["reason",{"2":{"24":1,"282":1}}],["reasons",{"2":{"8":1,"184":1}}],["resubmission",{"0":{"385":1,"386":1},"1":{"386":1,"387":1},"2":{"385":1}}],["resubmitted",{"2":{"386":1}}],["resubmit",{"2":{"377":2,"387":1}}],["results",{"2":{"124":1,"125":1,"328":2,"509":1,"605":4}}],["resulting",{"2":{"124":1,"409":1,"413":1,"554":1}}],["result",{"2":{"23":2,"127":2,"132":4,"137":2,"140":1,"141":1,"194":1,"262":4,"263":2,"292":1,"319":6,"321":8,"322":2,"324":2,"325":2,"326":2,"327":2,"328":2,"329":2,"331":2,"334":2,"336":3,"342":3,"407":1,"410":2,"413":3,"557":2,"558":3,"559":2,"620":1,"736":2}}],["resolved",{"2":{"638":1}}],["resolve",{"2":{"413":1}}],["resolving",{"2":{"285":1}}],["resolution",{"2":{"238":1,"562":1}}],["resources",{"0":{"335":1,"508":1,"598":1,"754":1},"1":{"336":1,"337":1,"338":1,"599":1,"600":1,"601":1,"602":1,"603":1,"604":1,"605":1},"2":{"39":1,"40":2,"43":1,"44":2,"50":1,"51":1,"80":1,"119":1,"276":1,"754":2}}],["resource",{"2":{"37":1,"50":1,"58":1,"211":1,"408":2,"414":1,"591":1}}],["resetting",{"0":{"560":1}}],["reset",{"0":{"599":1,"654":1},"2":{"550":1,"561":4,"599":2,"607":1,"654":2,"756":3}}],["reserved",{"2":{"159":1,"415":2,"683":1}}],["research",{"2":{"36":1,"38":1,"445":1}}],["restartsec=3",{"2":{"736":2,"738":2,"739":2,"740":2}}],["restart=on",{"2":{"736":2,"738":2,"739":2,"740":2}}],["restart",{"0":{"603":1},"2":{"260":1,"502":1,"544":1,"563":3,"570":1,"603":1,"604":1,"605":1,"607":1,"756":2}}],["restaurant",{"2":{"34":1}}],["restaurants",{"2":{"34":2}}],["restrictions",{"2":{"446":1}}],["restrict",{"2":{"190":1,"242":1}}],["rest",{"2":{"112":1,"199":1,"302":1,"352":1,"389":10,"390":1,"391":1,"552":1,"638":1,"639":2,"688":3,"707":4}}],["resides",{"2":{"95":1}}],["resilience",{"2":{"26":1}}],["resp",{"2":{"271":6,"359":4,"382":20}}],["respectively",{"2":{"203":1}}],["respective",{"2":{"27":1,"111":1,"115":1,"122":1,"469":1,"470":1,"500":1,"595":2,"692":1,"693":1,"699":1,"704":1,"705":1}}],["respect",{"2":{"10":1}}],["respectfully",{"2":{"46":1}}],["respectful",{"2":{"7":1}}],["responses",{"0":{"605":1},"2":{"42":1,"350":1,"361":1,"605":9}}],["response",{"2":{"8":1,"42":1,"127":1,"132":1,"137":1,"139":1,"141":4,"142":1,"144":4,"145":1,"252":1,"253":1,"262":1,"321":3,"324":1,"325":1,"326":1,"327":1,"359":2,"360":2,"382":4,"407":1,"557":2}}],["responsible",{"2":{"8":1,"10":1,"71":1,"72":1,"75":1,"111":2,"284":1,"414":1,"438":2}}],["responsibilities",{"0":{"8":1}}],["responsibility",{"2":{"7":1,"8":1,"107":2,"387":1,"403":1}}],["requiring",{"2":{"414":1}}],["requires",{"2":{"104":1,"151":1,"156":1,"178":1,"187":1,"193":1,"202":1,"203":1,"204":1,"212":1,"214":1,"343":1,"403":1}}],["require",{"2":{"90":1,"103":4,"149":1,"154":1,"156":1,"212":1,"241":1,"314":1,"403":1,"404":1,"409":2,"413":2,"554":1,"591":1,"714":2}}],["requirement",{"2":{"35":1,"409":1,"506":1}}],["requirements",{"0":{"308":1,"478":1,"581":1,"592":1,"626":1,"650":1,"666":1,"718":1,"742":1},"2":{"23":1,"30":1,"37":1,"54":1,"55":1,"71":1,"182":1,"282":2,"308":1,"478":1,"581":1,"592":1,"626":1,"650":1,"666":1,"742":1}}],["required",{"2":{"29":1,"30":1,"71":1,"134":1,"173":1,"187":1,"196":1,"222":1,"240":1,"252":1,"311":1,"317":1,"355":1,"356":1,"375":1,"409":1,"483":1,"537":1,"541":1,"550":1,"551":2,"552":1,"595":1,"612":1,"614":1,"632":1,"663":1,"671":1,"746":1,"748":1}}],["requisites",{"0":{"257":1,"572":1}}],["requestheaderrange",{"2":{"204":1,"218":1}}],["requesting",{"0":{"201":1},"1":{"202":1,"203":1,"204":1},"2":{"203":1,"204":1,"422":1}}],["requests",{"2":{"187":2,"203":1,"204":1,"315":1,"405":2,"410":1,"416":1,"438":1,"477":1,"600":1}}],["requested",{"2":{"12":1,"201":1,"203":1,"204":1,"218":1}}],["request",{"0":{"132":1,"137":1,"203":1,"204":1},"2":{"2":2,"38":1,"94":1,"127":1,"132":1,"137":1,"140":1,"141":1,"201":1,"203":1,"204":1,"312":1,"314":1,"316":1,"352":1,"389":2,"415":1,"472":1,"484":1,"509":1,"554":1,"557":2,"558":2,"594":1,"632":1,"673":1,"710":1,"720":1}}],["remainder",{"2":{"259":1,"509":1,"515":1,"522":1}}],["remain",{"2":{"241":1,"371":1,"436":1,"501":1,"589":1,"655":1,"716":1}}],["remaining",{"2":{"141":2,"142":1,"238":1,"267":1,"446":4,"525":1,"683":1}}],["remains",{"2":{"18":1,"184":1,"206":1,"679":1,"722":1}}],["removing",{"2":{"509":1}}],["removed",{"2":{"78":2,"137":1,"138":1,"302":1,"307":1,"356":1,"509":2,"552":1}}],["removechainowner",{"2":{"78":2}}],["remove",{"2":{"8":1,"341":2,"539":1,"561":1,"567":1,"607":2,"756":2}}],["remote",{"2":{"40":1,"224":1,"313":1,"369":1,"477":1,"488":2,"503":2,"574":3,"579":1}}],["remember",{"2":{"34":1,"38":1,"226":1,"242":1,"341":1,"604":1,"605":1,"679":1}}],["reminders",{"2":{"39":1}}],["reminder",{"2":{"30":1,"39":1,"64":1}}],["repeat",{"2":{"658":1}}],["replicated",{"2":{"413":1}}],["replaced",{"2":{"146":2}}],["replace",{"2":{"129":2,"168":1,"316":1,"377":1,"515":1,"660":1,"671":1}}],["replaces",{"2":{"125":1,"199":1,"412":1,"601":1}}],["replacing",{"2":{"78":2,"252":1,"253":1,"537":1}}],["replaying",{"2":{"196":1,"197":1,"438":1}}],["replayer",{"0":{"197":1},"2":{"193":2}}],["replay",{"2":{"81":1,"95":1}}],["repo",{"2":{"102":2,"187":1,"188":1,"212":1,"477":2,"500":1,"585":1,"660":1,"662":1}}],["report",{"2":{"27":1,"562":2}}],["reporting",{"2":{"23":1}}],["reporter",{"2":{"10":1}}],["reported",{"2":{"10":1}}],["repository",{"0":{"74":1,"285":1},"2":{"0":1,"1":1,"2":2,"4":1,"74":3,"77":1,"78":1,"102":1,"193":1,"196":2,"197":1,"213":1,"218":1,"222":1,"244":1,"245":1,"285":1,"369":1,"539":1,"567":1,"585":2,"752":1}}],["represent",{"2":{"122":1}}],["representation",{"2":{"111":1,"141":2,"142":2}}],["representative",{"2":{"9":1}}],["represents",{"2":{"94":2,"314":1}}],["representing",{"2":{"9":2}}],["relies",{"2":{"595":1,"637":1}}],["religion",{"2":{"6":1}}],["reloadsudo",{"2":{"563":1}}],["reload",{"2":{"563":2}}],["relying",{"2":{"209":1,"438":1}}],["rely",{"2":{"200":1,"210":1,"278":1,"438":1,"467":2,"685":2,"687":2,"703":2}}],["released",{"2":{"638":1}}],["release",{"2":{"196":1,"200":4,"437":1,"637":1,"715":1}}],["releasesource",{"2":{"196":1}}],["releases",{"2":{"137":1,"138":1,"540":1,"568":1}}],["relevant",{"2":{"37":2,"39":2,"42":1,"382":2,"406":1,"416":1,"465":2,"607":1,"665":1,"684":2,"699":2,"756":2}}],["relevantly",{"2":{"2":1}}],["relaying",{"0":{"637":1},"1":{"638":1,"639":1,"640":1,"641":1,"642":1,"643":1,"644":1,"645":1,"646":1,"647":1,"648":1},"2":{"254":1,"638":1,"640":3}}],["relayers",{"2":{"637":1}}],["relayer",{"0":{"640":1,"646":1},"2":{"191":1,"213":1,"220":1,"637":3,"638":2,"646":1}}],["relayed",{"2":{"94":2,"104":1,"105":1,"109":1,"112":1,"113":1,"204":1}}],["relay",{"2":{"94":1,"196":1,"220":1,"637":1,"639":1}}],["relays",{"2":{"82":1,"96":1,"206":1}}],["relatively",{"2":{"282":1,"385":1}}],["relative",{"0":{"282":1},"2":{"178":1,"282":1}}],["relations",{"2":{"39":1}}],["relationships",{"2":{"32":1,"36":1}}],["related",{"2":{"38":1,"42":1,"311":1,"402":1,"483":1,"632":2,"671":1}}],["re",{"2":{"2":1,"34":1,"35":1,"46":1,"66":2,"71":2,"76":1,"102":1,"124":1,"148":2,"158":1,"163":1,"168":1,"187":3,"195":6,"200":2,"212":1,"232":1,"234":1,"241":1,"253":1,"350":1,"361":1,"369":1,"382":4,"385":1,"390":1,"391":1,"505":1,"550":1,"560":1,"562":1,"565":1,"567":1,"569":1,"607":1,"615":2,"642":1,"756":2}}],["sql",{"2":{"466":1,"685":1,"703":1}}],["squaresizeupperbound",{"2":{"135":2}}],["square",{"0":{"123":1,"273":1,"366":1},"2":{"94":1,"95":2,"107":3,"112":1,"122":2,"129":2,"135":6,"148":2,"153":1,"157":2,"158":1,"159":9,"160":1,"273":1,"366":1,"415":3,"683":4}}],["snaps",{"2":{"596":12}}],["snapshots",{"2":{"438":4,"592":1,"595":3}}],["snapshot",{"2":{"402":1,"593":2,"596":1}}],["snap",{"2":{"596":24}}],["snippets",{"2":{"121":1,"125":1,"356":1}}],["ss",{"2":{"375":1}}],["ssd",{"2":{"308":1,"478":1,"581":1,"626":1,"666":1,"719":3,"720":2,"742":1}}],["sgvsbg8gd28ybgqh",{"2":{"342":2}}],["svg",{"0":{"337":1},"2":{"337":2,"391":1}}],["s3",{"2":{"199":16}}],["skipauth",{"2":{"554":1}}],["skip",{"2":{"186":1,"200":2,"269":1,"307":1,"352":2,"357":2,"358":2,"360":2,"363":1,"642":1,"733":1,"746":1}}],["skillsets",{"2":{"46":1}}],["src",{"2":{"145":2,"188":2,"389":8,"607":4,"756":4}}],["srl59gitsiwc9lqdyaszfc6tvusyy7njx8",{"2":{"137":2}}],["sdk",{"2":{"81":4,"125":7,"242":1,"244":1,"277":4,"278":2,"279":1,"281":1,"282":1,"369":1,"370":1,"374":1,"382":2,"412":1,"417":1,"440":3,"444":1,"455":1,"465":1,"477":1,"488":2,"494":2,"506":1,"509":1,"638":1,"684":1,"707":1,"708":1}}],["switching",{"2":{"550":1}}],["switch",{"2":{"70":1,"148":2,"284":1,"389":2,"391":1,"503":1}}],["swag",{"0":{"57":1},"2":{"37":1}}],["symbol",{"2":{"391":2}}],["sync|async|block",{"2":{"531":2}}],["sync",{"0":{"469":1,"593":1,"594":1,"595":1,"596":1,"692":1,"704":1},"1":{"594":1,"595":1,"596":1},"2":{"281":3,"346":1,"400":1,"438":3,"469":1,"530":1,"531":2,"564":1,"580":1,"593":4,"594":2,"595":4,"596":2,"692":1,"704":1,"717":1,"736":4,"740":1,"756":1}}],["syncing",{"0":{"281":1,"564":1},"1":{"565":1},"2":{"281":1,"399":1,"436":1,"487":1,"564":1,"594":1,"738":1,"739":2,"740":1,"750":1,"756":1}}],["synced",{"2":{"67":1,"346":1}}],["syndtr",{"2":{"125":4}}],["systemctl",{"2":{"563":4,"736":6,"738":4,"739":4,"740":6}}],["systemd",{"0":{"487":1,"499":1,"635":1,"676":1,"734":1,"736":1,"750":1},"1":{"735":1,"736":1,"737":1,"738":1,"739":1,"740":1},"2":{"487":1,"499":7,"563":2,"597":1,"635":1,"676":1,"734":1,"736":6,"738":5,"739":5,"740":4,"750":1}}],["systems",{"2":{"278":1,"503":2,"540":1,"568":1}}],["system",{"2":{"22":1,"30":1,"209":1,"300":1,"374":1,"375":2,"413":2,"501":2,"504":1,"505":1,"563":5,"567":1,"612":1,"641":1,"736":4,"738":4,"739":4,"740":4}}],["sausage",{"2":{"519":2}}],["sandbox",{"2":{"695":1}}],["sanity",{"2":{"357":1}}],["sanctions",{"2":{"23":1}}],["sashabaranov",{"2":{"355":2,"356":2}}],["salt=",{"2":{"187":2}}],["salt",{"2":{"187":5,"189":4,"195":2}}],["sales",{"2":{"38":1,"471":1,"710":1}}],["save",{"0":{"515":1,"592":1},"2":{"187":1,"195":1,"352":2,"356":1,"494":1,"503":1,"536":1,"560":1,"592":1,"604":1,"605":1,"656":1,"694":1}}],["saved",{"2":{"122":1,"124":1,"154":1,"169":1,"172":1}}],["saving",{"2":{"37":1,"156":1,"494":1}}],["same",{"0":{"377":1},"2":{"97":1,"110":1,"124":1,"155":1,"160":1,"169":1,"172":1,"176":2,"187":2,"189":1,"209":1,"220":1,"279":1,"292":1,"300":1,"382":2,"413":2,"503":4,"509":1,"554":1,"565":3,"585":1,"655":1,"657":1,"751":1}}],["sampling",{"0":{"396":1,"397":1,"407":1,"677":1},"1":{"408":1,"409":1},"2":{"53":1,"206":1,"241":1,"282":1,"330":2,"395":1,"396":5,"397":1,"398":1,"406":1,"407":3,"409":1,"414":2,"437":3,"477":1,"664":1,"679":1,"717":1,"731":1,"740":1}}],["samplefrom",{"2":{"564":2}}],["samples",{"2":{"408":1}}],["sampled",{"2":{"397":1,"416":1}}],["sampler",{"0":{"330":1}}],["sample",{"0":{"53":1,"54":1,"55":1,"56":1},"2":{"53":2,"54":2,"55":2,"56":2,"148":1,"414":1,"437":1,"564":1}}],["said",{"2":{"81":1}}],["safely",{"2":{"96":1,"414":1,"555":1}}],["safe",{"2":{"46":1,"352":2,"508":1,"519":2}}],["smartstake",{"2":{"697":1}}],["smart",{"0":{"236":1},"2":{"66":1,"72":2,"75":1,"107":7,"109":1,"121":1,"124":1,"126":1,"127":1,"132":1,"137":1,"139":5,"145":2,"149":1,"154":1,"159":1,"160":2,"201":1,"212":2,"213":1,"214":1,"216":1,"218":1,"219":1,"226":1,"236":1,"238":1,"240":1,"242":1,"369":1,"402":1,"454":1}}],["smartphone",{"2":{"34":1}}],["smaller",{"2":{"371":1}}],["small",{"2":{"40":1,"94":1,"110":1,"182":1,"241":1,"396":1,"408":1,"414":1}}],["smooth",{"2":{"37":1}}],["shnode",{"2":{"739":1}}],["shpeer=",{"2":{"662":1}}],["ship",{"2":{"662":1}}],["shirts",{"2":{"37":1}}],["shtia",{"2":{"657":1}}],["shutdown",{"2":{"636":1}}],["shsed",{"2":{"601":1}}],["shstaking",{"2":{"500":1,"659":1}}],["shsudo",{"2":{"499":1,"736":4,"738":2,"739":2,"740":4}}],["shkey",{"2":{"500":1,"656":1}}],["shvalidator",{"2":{"500":1,"655":1}}],["shgo",{"2":{"360":2}}],["shgit",{"2":{"102":1,"245":1}}],["shmkdir",{"2":{"355":1}}],["shexternal",{"2":{"602":1}}],["shexport",{"2":{"353":1}}],["sherror",{"2":{"560":1}}],["shevm",{"2":{"500":1}}],["shellsource",{"2":{"196":1}}],["shellforge",{"2":{"195":1}}],["shelltendermint",{"2":{"194":1}}],["shellcast",{"2":{"189":3,"191":1,"192":1}}],["shellcd",{"2":{"187":1,"188":1,"193":1,"195":1,"196":1,"199":1}}],["shell",{"2":{"187":2,"623":1}}],["shellgit",{"2":{"187":1,"369":1}}],["shcurl",{"2":{"736":1}}],["shcat",{"2":{"736":1,"738":1,"739":1,"740":1}}],["shceles",{"2":{"500":1}}],["shcelestia",{"2":{"289":1,"291":1,"292":1,"352":3,"483":4,"484":3,"485":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,"564":2,"597":4,"599":1,"631":3,"632":1,"633":3,"654":1,"660":1,"663":1,"670":3,"671":3,"672":3,"674":3,"751":4}}],["shcd",{"2":{"347":1,"514":1}}],["sha3uncles",{"2":{"340":2}}],["sharing",{"2":{"40":1,"58":1,"438":1}}],["shareversionzero",{"2":{"382":2}}],["shareclient",{"2":{"364":2}}],["shareindexinrowmajororder",{"2":{"160":2}}],["shareindexinrow",{"2":{"160":4}}],["shareproof",{"2":{"148":2,"160":6}}],["shareproofs",{"0":{"141":1},"2":{"139":2,"141":3,"148":4,"160":2}}],["sharesize",{"2":{"375":1}}],["sharesproof",{"2":{"139":3,"141":1,"142":1,"143":1,"144":1,"148":18}}],["shares",{"2":{"103":2,"107":5,"110":3,"112":1,"122":4,"124":3,"130":1,"136":3,"137":4,"139":14,"140":1,"141":2,"142":1,"143":1,"144":1,"148":4,"154":1,"157":1,"160":4,"165":1,"184":4,"209":1,"321":2,"371":2,"374":2,"375":2,"376":1,"407":5,"409":4,"410":3,"414":1,"415":4,"416":3,"477":1,"625":2,"683":4}}],["shared",{"2":{"58":1,"659":1}}],["share",{"0":{"136":1,"152":1,"153":1,"154":1,"155":1,"156":1,"168":1,"169":1,"170":1,"175":1,"177":1,"178":1},"1":{"137":1,"138":1,"153":1,"154":1,"155":1,"156":1,"169":1,"170":1,"176":1,"177":1,"178":1},"2":{"26":1,"30":1,"32":1,"34":1,"37":1,"39":4,"40":3,"42":1,"110":6,"122":2,"124":5,"132":5,"134":1,"136":6,"137":2,"148":2,"151":1,"152":1,"153":11,"154":2,"155":1,"157":2,"158":1,"159":2,"160":9,"161":1,"165":2,"168":7,"169":1,"171":1,"173":1,"175":1,"176":5,"179":7,"184":3,"270":7,"271":2,"272":4,"273":3,"319":1,"321":8,"329":2,"334":3,"336":2,"356":2,"358":7,"366":3,"371":2,"374":1,"375":2,"381":6,"407":1,"415":1,"416":1,"438":2,"447":1,"500":1,"558":2,"560":4,"658":1,"660":1,"662":3,"683":3}}],["sh",{"2":{"187":2,"245":2,"249":1,"252":2,"253":2,"294":1,"295":1,"338":1,"484":1,"490":1,"507":4,"514":1,"515":2,"540":2,"562":1,"568":2,"622":2,"632":1,"672":2}}],["shforge",{"2":{"102":1}}],["shove",{"2":{"519":2}}],["shoutout",{"2":{"38":1}}],["shouldn",{"2":{"403":1}}],["should",{"0":{"184":1},"2":{"26":1,"34":1,"38":1,"50":1,"71":1,"72":1,"121":1,"140":1,"171":1,"172":1,"184":1,"187":3,"189":1,"193":1,"195":1,"196":1,"199":5,"200":2,"201":2,"206":1,"246":1,"282":2,"319":1,"343":2,"348":1,"352":1,"373":1,"374":1,"386":1,"438":1,"454":1,"467":1,"503":1,"540":1,"555":1,"563":1,"568":1,"589":1,"591":1,"595":3,"641":1,"644":2,"649":1,"655":1,"659":1,"660":2,"685":1,"687":1,"703":1,"738":1,"739":1,"745":2,"751":2}}],["showing",{"2":{"263":1}}],["shown",{"2":{"219":1,"558":1,"751":1}}],["shows",{"2":{"124":1,"167":1,"334":1,"382":2,"410":1,"412":1}}],["showcase",{"2":{"38":1}}],["show",{"2":{"37":1,"38":1,"159":1,"267":1,"336":1,"344":1,"389":1,"465":1,"500":2,"525":2,"555":1,"557":1,"567":1,"623":1,"662":2,"670":2,"684":1,"699":1,"730":1,"745":2,"751":3}}],["shortly",{"2":{"62":1}}],["short",{"2":{"37":1,"616":1,"715":1}}],["scan",{"2":{"637":1}}],["scattered",{"2":{"350":1}}],["scavenger",{"0":{"350":1,"354":1},"1":{"351":1,"352":1,"353":1,"354":1,"355":2,"356":2,"357":2,"358":2,"359":2,"360":2,"361":1},"2":{"303":1,"350":1,"355":6,"360":1}}],["scaling",{"0":{"241":1},"2":{"241":2,"393":1,"394":1,"406":1}}],["scalable",{"2":{"211":1,"284":1,"403":1,"406":1,"414":1}}],["scalability",{"0":{"211":1,"408":1},"2":{"96":1,"211":2}}],["scale",{"2":{"241":2,"408":1}}],["scaleway",{"2":{"199":1,"200":1}}],["scales",{"2":{"150":1,"206":1,"414":2}}],["scraped",{"2":{"503":2}}],["scrape",{"2":{"503":16}}],["scratches",{"2":{"242":1}}],["scratch",{"2":{"240":1}}],["scroll",{"2":{"391":1}}],["scrolling",{"2":{"390":1}}],["scripts",{"2":{"507":2,"514":2,"515":2}}],["script",{"0":{"74":1},"2":{"61":1,"74":5,"77":1,"78":7,"79":1,"187":2,"195":4,"203":2,"352":1,"360":1,"507":2,"514":1,"540":1,"568":1}}],["screen",{"2":{"23":1,"34":1,"384":2,"422":1,"426":1}}],["scheme",{"0":{"124":1},"2":{"153":1,"407":1,"409":1,"412":1,"415":1}}],["schemes",{"2":{"52":1,"277":1}}],["schedules",{"2":{"446":1}}],["schedule",{"2":{"30":1,"31":1,"35":1,"37":1,"42":1,"446":2,"451":1}}],["schools",{"2":{"38":1}}],["scene",{"2":{"36":1}}],["scope",{"0":{"9":1},"2":{"56":1}}],["sl",{"2":{"540":2,"568":2,"585":8}}],["slack",{"2":{"505":1}}],["slas",{"2":{"467":1,"685":1,"687":1,"703":1}}],["slashboard",{"2":{"697":1}}],["slashing",{"0":{"509":1},"2":{"277":1,"498":2,"509":3}}],["slashed",{"2":{"23":1,"24":1,"26":1,"193":1,"509":4}}],["slower",{"2":{"727":1}}],["slowest",{"2":{"594":1}}],["slowly",{"2":{"348":3}}],["slots",{"2":{"20":1,"22":1}}],["slightly",{"2":{"683":1}}],["slice",{"2":{"270":3,"358":2}}],["slides",{"2":{"37":1}}],["soundness",{"2":{"416":1}}],["sourcing",{"2":{"240":1}}],["sources",{"2":{"503":1}}],["source",{"0":{"539":1,"567":1},"2":{"76":1,"124":1,"195":2,"347":2,"495":2,"503":4,"538":1,"548":1,"623":4,"638":1,"639":4,"708":1}}],["software",{"0":{"464":1,"681":1,"701":1,"723":1,"726":1,"728":1},"2":{"54":1,"55":1,"97":1,"206":1,"462":1,"621":1,"699":1,"714":2,"732":1,"733":1}}],["sovereign",{"0":{"55":1},"2":{"53":1,"55":5,"462":1,"632":2,"699":1,"725":1,"727":1}}],["soda",{"2":{"35":1}}],["sometimes",{"2":{"477":1,"625":1}}],["something",{"2":{"89":1,"95":1,"200":2,"328":2,"352":1,"503":1,"542":1}}],["somewhere",{"2":{"352":2,"656":1}}],["some",{"0":{"160":1,"397":1,"405":1},"2":{"34":1,"40":1,"69":1,"77":1,"110":3,"111":1,"113":1,"139":4,"145":2,"151":1,"237":1,"241":1,"242":1,"259":1,"277":1,"280":1,"319":1,"350":1,"352":1,"357":2,"360":1,"364":2,"365":4,"366":2,"382":2,"404":1,"405":1,"415":1,"436":1,"438":1,"446":1,"494":1,"495":1,"503":1,"507":4,"530":1,"532":1,"548":1,"573":1,"644":8,"715":1}}],["someone",{"2":{"31":1,"39":1}}],["solomon",{"2":{"407":2,"409":5,"412":1,"415":1}}],["solution",{"2":{"115":1,"150":1,"182":1,"206":1,"211":1,"406":1,"409":1,"413":1,"414":1}}],["solutions",{"2":{"96":1,"377":1,"394":1,"406":1}}],["solidityuint256",{"2":{"160":1}}],["soliditypragma",{"2":{"148":1}}],["solidity",{"2":{"102":1,"103":4,"127":1,"132":1,"137":1,"139":1,"141":6,"142":1,"144":4,"145":2,"148":2,"149":1,"153":1,"198":1,"216":1}}],["soliciting",{"2":{"32":1}}],["sol",{"2":{"96":1,"103":7,"105":1,"107":1,"140":1,"141":1,"142":1,"144":1,"145":1,"148":4,"188":3,"195":2,"199":4,"200":4,"213":1,"218":1,"219":1}}],["solely",{"2":{"438":1}}],["sole",{"2":{"24":1}}],["so",{"2":{"22":1,"23":2,"28":1,"35":2,"39":1,"64":1,"76":1,"102":1,"110":2,"124":2,"134":1,"135":1,"141":3,"158":1,"159":1,"172":1,"184":1,"195":1,"200":1,"234":1,"246":1,"311":1,"315":1,"350":1,"390":1,"395":1,"397":1,"398":2,"410":1,"416":1,"436":1,"462":1,"483":1,"485":1,"522":1,"527":1,"560":1,"592":1,"597":1,"599":1,"620":1,"623":1,"633":1,"649":1,"654":1,"662":2,"674":1,"693":1,"694":1,"751":1}}],["sorted",{"2":{"387":1}}],["sort",{"2":{"14":1,"15":1}}],["social",{"2":{"9":1,"13":1,"30":1,"32":1,"39":4,"40":1,"494":1}}],["socio",{"2":{"6":1}}],["spidey",{"2":{"687":1,"688":1,"689":1,"706":1,"707":1,"708":1}}],["spin",{"0":{"653":1},"1":{"654":1,"655":1,"656":1,"657":1,"658":1,"659":1,"660":1,"661":1,"662":1,"663":1},"2":{"653":1}}],["spinning",{"2":{"547":1}}],["splits",{"2":{"415":1}}],["split",{"2":{"122":1,"407":1,"409":1,"415":1,"434":1,"445":1}}],["sp1blobstreamwrapper",{"2":{"129":2,"146":2,"147":2,"148":2}}],["sp1blobstream",{"2":{"121":2,"213":1}}],["sp1",{"0":{"212":1,"213":1,"368":1},"1":{"214":1,"215":1,"216":1,"369":1},"2":{"121":2,"129":4,"146":4,"147":4,"148":4,"206":3,"207":1,"212":10,"213":6,"216":8,"368":3,"369":14}}],["sparse",{"2":{"683":2}}],["sparsesharesneeded",{"2":{"375":1}}],["spans",{"0":{"157":1,"158":1,"159":1,"160":1,"161":1,"162":1,"164":1,"165":1,"166":1,"167":1,"172":1,"173":1,"174":1},"1":{"158":1,"159":1,"160":1,"161":1,"162":1,"165":1,"166":1,"167":1,"173":1,"174":1},"2":{"132":1,"136":1,"151":1,"157":3,"158":1,"159":3,"160":7,"161":1,"164":4,"167":1,"168":2,"171":1,"172":5,"182":3,"184":2}}],["span",{"0":{"343":1},"2":{"110":12,"111":4,"112":2,"113":8,"115":3,"139":4,"141":1,"159":1,"343":2}}],["space",{"2":{"46":1,"504":1,"604":1,"605":2,"687":1,"688":1,"689":1,"694":1}}],["spaces",{"2":{"9":2,"13":1,"34":5,"38":1}}],["spdx",{"2":{"103":2}}],["sponsoring",{"2":{"38":1}}],["sponsorship",{"2":{"38":5}}],["sponsors",{"0":{"38":1},"2":{"38":9,"39":1,"46":1}}],["spread",{"2":{"36":1,"39":1}}],["speed",{"2":{"386":1,"593":1}}],["spend",{"2":{"261":2,"266":2,"375":1,"444":1,"494":1,"495":7}}],["spec",{"2":{"695":1}}],["specialization",{"2":{"413":1}}],["specialized",{"2":{"413":3}}],["specialize",{"2":{"238":1}}],["specif",{"2":{"139":2,"145":2}}],["specifying",{"2":{"291":1,"292":1,"373":1,"503":1}}],["specify",{"2":{"71":1,"187":4,"246":1,"296":1,"311":2,"322":1,"377":1,"483":2,"554":1,"616":2}}],["specifies",{"2":{"71":1,"246":1,"261":1}}],["specified",{"2":{"13":1,"14":1,"72":1,"187":1,"188":1,"196":2,"198":1,"200":2,"203":1,"246":1,"261":1,"314":1,"322":1,"336":1,"373":2,"375":1,"437":1,"459":1,"488":2,"548":1}}],["specifications",{"2":{"110":1,"509":1,"682":1,"715":1}}],["specifically",{"2":{"58":1,"75":1,"393":1,"402":1,"595":1}}],["specific",{"0":{"136":1},"1":{"137":1,"138":1},"2":{"35":1,"37":1,"42":1,"71":1,"94":1,"105":2,"110":2,"111":4,"113":1,"192":1,"199":2,"200":3,"202":2,"209":1,"211":1,"218":1,"314":2,"333":1,"389":1,"475":1,"504":1,"541":1,"552":1,"554":1,"658":1,"698":1,"713":1,"715":1,"729":1}}],["specs",{"2":{"122":2,"124":1,"141":2,"142":2,"145":4,"321":1,"376":1,"494":2}}],["speak",{"2":{"37":1}}],["speaking",{"2":{"36":1,"42":1}}],["speaker",{"0":{"58":1},"2":{"30":1,"34":1,"36":1,"37":3,"50":1,"58":5}}],["speakers",{"0":{"37":1},"2":{"30":3,"31":1,"32":1,"34":1,"36":5,"37":13,"38":2,"39":3,"42":2,"46":1,"56":1,"58":2}}],["sed",{"2":{"585":12,"602":2}}],["semantic",{"2":{"567":1}}],["semi",{"2":{"209":1}}],["severe",{"2":{"509":1}}],["severity",{"2":{"394":1}}],["several",{"2":{"94":1,"201":1,"446":1,"683":1,"712":1,"715":1,"716":1,"717":1}}],["sequential",{"2":{"242":1,"639":4}}],["sequence",{"0":{"157":1,"158":1,"159":1,"160":1,"161":1,"162":1,"164":1,"165":1,"166":1,"167":1,"172":1,"173":1,"174":1},"1":{"158":1,"159":1,"160":1,"161":1,"162":1,"165":1,"166":1,"167":1,"173":1,"174":1},"2":{"151":1,"157":3,"158":1,"159":3,"160":7,"161":1,"164":4,"167":1,"168":2,"171":1,"172":5,"182":3,"184":2,"254":1,"374":1,"377":1,"382":2,"413":2,"523":2,"524":2,"640":1}}],["sequencers",{"2":{"147":1,"168":1,"240":1}}],["sequencersignature",{"2":{"110":2,"113":2,"115":2}}],["sequenceraddress",{"2":{"110":2,"115":2}}],["sequencer",{"0":{"111":1},"1":{"112":1,"113":1},"2":{"95":1,"97":2,"110":3,"111":7,"112":2,"113":8,"115":8,"254":1,"348":4}}],["sequencerinbox",{"2":{"76":2,"85":2,"86":2,"96":1}}],["self",{"2":{"202":1,"488":2,"500":2,"751":4}}],["selecting",{"2":{"34":1,"75":1,"92":1,"512":1}}],["selection",{"0":{"550":1},"1":{"551":1},"2":{"26":1,"58":1}}],["select",{"0":{"418":1,"421":1,"426":1},"2":{"30":1,"61":2,"70":1,"147":2,"246":1,"384":1,"420":1,"421":1,"422":2,"425":2,"426":1,"429":1,"430":1,"522":1,"539":1,"550":1,"592":1,"604":1,"699":3,"753":1}}],["selected",{"2":{"23":1,"27":1,"438":1}}],["separation",{"2":{"413":1}}],["separating",{"2":{"241":1}}],["separately",{"2":{"168":1}}],["separate",{"2":{"141":1,"259":1,"300":1,"407":1,"416":1,"477":2,"547":1}}],["separated",{"2":{"122":1,"196":1,"585":2,"661":2}}],["sepoliaeth",{"2":{"348":1}}],["sepoliafaucet",{"2":{"69":1}}],["sepolia",{"0":{"69":1,"83":1,"84":1,"85":1,"86":1},"2":{"62":2,"64":2,"66":1,"67":1,"69":3,"70":1,"72":1,"75":4,"78":2,"82":2,"83":1,"84":1,"216":3,"223":1,"225":1,"226":1,"227":1,"228":1,"229":1,"348":5}}],["session",{"2":{"53":1,"54":1,"55":1}}],["sessions",{"2":{"38":1,"58":1}}],["seamless",{"2":{"679":1}}],["seamlessly",{"2":{"254":1}}],["search",{"0":{"421":1,"430":1},"2":{"384":2}}],["sealfields",{"2":{"340":2}}],["seasoned",{"2":{"50":1}}],["seating",{"2":{"31":1,"34":1}}],["seats",{"2":{"27":4}}],["sentry",{"2":{"585":1,"639":6}}],["sent",{"2":{"78":2,"124":1,"238":1,"374":1,"537":1}}],["sensitive",{"2":{"73":1,"298":1}}],["sense",{"2":{"36":1,"40":1}}],["sendrequest",{"2":{"557":2,"559":2}}],["sender",{"2":{"415":1,"434":1,"644":16}}],["sending",{"2":{"196":1,"224":1,"254":1,"331":1,"415":1,"537":1}}],["sends",{"2":{"97":1}}],["send",{"2":{"30":2,"32":1,"39":3,"41":1,"187":1,"189":2,"266":1,"493":6,"495":1,"507":8,"509":1,"529":1,"625":1,"648":1}}],["serving",{"2":{"399":1,"405":1,"438":1}}],["servicecat",{"2":{"736":1,"738":1,"739":1,"740":1}}],["servicenano",{"2":{"563":1}}],["service",{"0":{"235":1,"275":1,"277":1,"565":1},"1":{"276":1,"277":1,"278":2,"279":2,"280":2,"281":2,"282":2},"2":{"275":1,"300":1,"344":1,"467":1,"499":7,"563":9,"601":3,"679":1,"685":1,"687":1,"703":1,"734":1,"736":8,"738":7,"739":7,"740":7}}],["services",{"0":{"578":1},"1":{"579":1},"2":{"23":1,"34":1,"253":2,"565":2,"687":2,"688":2,"689":2,"693":1,"697":2,"705":1,"706":2,"707":2,"708":2}}],["served",{"2":{"502":2}}],["serves",{"2":{"226":1,"416":1,"477":1,"648":1,"678":1}}],["servergrafana",{"2":{"503":1}}],["server",{"0":{"486":1,"634":1,"675":1},"2":{"47":1,"382":2,"472":1,"486":1,"503":6,"541":2,"548":1,"552":2,"634":1,"640":1,"675":1,"710":1}}],["servers",{"2":{"26":1,"595":2}}],["serve",{"2":{"44":1,"403":1,"405":1,"600":1}}],["serialized",{"2":{"94":1}}],["serious",{"2":{"14":1}}],["series",{"2":{"13":1,"445":1,"446":1,"503":2}}],["secp256k1",{"2":{"253":2,"277":1,"519":2,"520":4,"524":2}}],["sections",{"2":{"182":1,"214":1,"465":1,"573":1,"684":1,"699":1,"731":1,"733":1}}],["section",{"2":{"36":1,"68":1,"71":2,"75":1,"94":1,"107":1,"112":1,"127":1,"132":1,"137":1,"141":1,"145":1,"148":4,"151":1,"156":1,"159":1,"168":2,"171":1,"172":1,"176":1,"179":2,"184":1,"188":1,"189":2,"193":2,"195":4,"196":2,"198":2,"200":5,"232":1,"246":1,"252":3,"268":1,"307":1,"311":1,"312":1,"313":1,"318":1,"320":1,"356":1,"362":1,"412":1,"470":1,"483":1,"484":1,"515":1,"526":1,"537":1,"539":1,"542":1,"544":1,"550":1,"555":1,"563":1,"567":1,"570":1,"574":1,"604":1,"605":1,"611":1,"612":1,"614":1,"616":1,"632":2,"637":1,"642":1,"663":1,"671":1,"730":1,"733":2,"746":1,"748":1}}],["securing",{"2":{"36":1,"37":1,"38":2}}],["security",{"0":{"209":1,"397":1,"398":1},"2":{"10":1,"26":2,"206":2,"209":1,"211":1,"220":1,"240":2,"393":1,"400":1,"620":1}}],["securely",{"2":{"73":1,"150":1,"206":1,"254":1,"414":1,"715":1}}],["secure",{"2":{"24":1,"30":1,"211":1,"403":1,"417":1,"447":1,"455":2,"547":1,"594":1}}],["seconds",{"2":{"78":2,"252":4,"253":4,"282":1,"352":1,"503":4}}],["second",{"2":{"22":4,"141":1,"168":1,"218":1,"377":1,"385":1,"386":1,"387":1,"397":1,"643":1}}],["setgaslimitandfee",{"2":{"382":2}}],["setwhiteliststatus",{"2":{"191":2}}],["sets",{"2":{"132":1,"316":1,"389":1}}],["settling",{"2":{"212":1}}],["settlement",{"2":{"122":1,"150":1,"156":1,"157":1,"164":1,"165":1,"168":2,"170":1,"171":2,"172":3,"175":1,"176":1,"179":1,"212":1,"238":1,"284":1,"406":2,"412":1,"413":4}}],["settles",{"2":{"92":1}}],["settle",{"2":{"66":1,"72":1,"96":1,"212":1,"283":1}}],["settled",{"2":{"62":1,"65":1}}],["settingpruning",{"2":{"592":1}}],["settingmin",{"2":{"589":1}}],["settings",{"2":{"61":1,"503":1,"554":1,"587":1,"604":1,"605":1}}],["setting",{"0":{"78":1,"309":1,"322":1,"346":1,"476":1,"479":1,"513":1,"518":1,"624":1,"627":1,"664":1,"668":1,"734":1,"741":1,"743":1},"1":{"310":1,"311":1,"312":1,"477":1,"478":1,"479":1,"480":2,"481":1,"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,"625":1,"626":1,"627":1,"628":2,"629":1,"630":1,"631":1,"632":1,"633":1,"634":1,"635":1,"636":1,"665":1,"666":1,"667":1,"668":1,"669":2,"670":1,"671":1,"672":1,"673":1,"674":1,"675":1,"676":1,"677":1,"735":1,"736":1,"737":1,"738":1,"739":1,"740":1,"742":1,"743":1,"744":2,"745":2,"746":1,"747":1,"748":1,"749":1,"750":1,"751":1,"752":1,"753":1,"754":1,"755":1,"756":1},"2":{"7":1,"41":2,"54":1,"55":1,"78":8,"196":1,"223":1,"243":1,"310":1,"317":1,"331":1,"343":1,"351":1,"382":4,"476":1,"487":1,"539":1,"563":1,"567":1,"579":1,"585":1,"589":2,"592":1,"621":1,"624":1,"628":1,"664":1,"668":1,"676":1,"735":1,"741":1,"743":1,"750":1}}],["setupprivate",{"2":{"78":1}}],["setup",{"0":{"68":1,"74":1,"269":1,"345":1,"363":1,"480":1,"502":1,"612":1,"615":1,"628":1,"651":1},"1":{"69":1,"70":1,"71":1,"72":1,"73":1,"74":1,"75":1,"76":1,"77":1,"78":1,"79":1,"346":1,"613":1,"614":1,"615":1,"616":2,"617":2,"618":1,"619":1,"620":1},"2":{"26":2,"30":1,"61":1,"71":1,"74":5,"77":1,"78":2,"187":2,"190":1,"191":1,"199":1,"222":1,"313":1,"356":1,"365":2,"382":2,"503":2,"505":1,"518":1,"540":1,"543":1,"568":1,"595":1,"637":1,"651":1,"746":1}}],["setupsigner",{"2":{"382":2}}],["setups",{"2":{"23":1,"554":1}}],["set",{"0":{"191":1,"521":1,"530":1,"582":1,"583":1,"585":1},"1":{"583":1,"584":1,"585":1},"2":{"18":2,"20":1,"22":1,"23":1,"26":1,"30":1,"31":1,"36":1,"41":2,"64":1,"75":1,"76":4,"78":10,"107":1,"110":4,"113":1,"136":1,"153":1,"163":1,"187":7,"190":2,"191":1,"195":6,"196":5,"209":1,"240":2,"242":1,"259":1,"266":1,"270":1,"298":1,"300":1,"314":4,"315":2,"316":5,"317":4,"318":4,"319":1,"321":1,"322":3,"328":2,"331":1,"336":1,"340":1,"341":1,"343":1,"353":1,"355":1,"356":1,"359":4,"364":1,"369":3,"375":4,"402":1,"407":1,"413":1,"440":2,"446":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,"563":1,"564":2,"574":1,"577":1,"580":1,"585":6,"590":1,"595":2,"604":1,"605":1,"614":3,"615":1,"616":1,"623":1,"641":1,"656":1,"668":1,"717":1,"750":1}}],["seeing",{"2":{"644":2,"738":1,"739":1}}],["seeds",{"2":{"585":13}}],["seed",{"2":{"445":1,"446":1,"585":2,"640":5,"752":1}}],["sees",{"2":{"394":1}}],["seeking",{"2":{"567":1}}],["seek",{"2":{"34":1,"37":1}}],["see",{"2":{"16":1,"61":1,"62":4,"64":3,"65":1,"72":1,"76":1,"78":1,"79":1,"95":1,"96":1,"112":2,"119":1,"141":2,"142":2,"195":1,"196":1,"199":1,"212":1,"219":1,"230":1,"258":1,"262":1,"269":1,"274":2,"296":1,"321":1,"344":1,"347":1,"348":1,"351":1,"352":1,"363":1,"367":2,"382":4,"384":1,"390":1,"391":1,"393":1,"412":1,"423":1,"427":1,"434":1,"443":1,"458":1,"475":1,"477":3,"492":1,"493":1,"502":1,"503":1,"510":1,"515":1,"519":1,"531":2,"537":1,"539":1,"540":1,"568":1,"595":1,"597":1,"604":1,"638":2,"641":1,"683":1,"693":1,"698":1,"713":1,"716":1,"729":1,"736":1,"738":1,"739":2,"740":1,"745":2,"751":2}}],["sexualized",{"2":{"7":1}}],["sexual",{"2":{"6":1,"7":1}}],["sex",{"2":{"6":1}}],["sudo",{"2":{"499":4,"622":2,"623":16,"736":2,"738":4,"739":4,"740":2}}],["sudden",{"2":{"18":1}}],["supply",{"0":{"439":1},"1":{"440":1,"441":1,"442":1,"443":1,"444":1,"445":1,"446":1},"2":{"441":1,"445":1,"446":6,"451":1,"477":1}}],["supporters",{"2":{"445":2}}],["supported",{"2":{"153":1,"254":1,"281":1,"384":1,"695":1}}],["supports",{"2":{"278":1,"440":1}}],["supportive",{"2":{"46":1}}],["supporting",{"2":{"38":1,"41":1,"56":1}}],["support",{"2":{"30":1,"33":1,"35":1,"38":4,"40":1,"43":1,"44":1,"57":2,"66":2,"75":1,"95":1,"221":1,"241":1,"242":2,"438":2,"440":1,"501":1,"506":1,"620":1,"637":1,"638":1,"707":1,"746":1}}],["superior",{"2":{"298":1}}],["suits",{"2":{"151":1}}],["suitable",{"2":{"30":1,"604":2}}],["summarize",{"2":{"122":1}}],["summarizes",{"2":{"111":4}}],["summary",{"2":{"53":1,"54":1,"55":1,"211":1}}],["surface",{"2":{"242":1}}],["sure",{"2":{"76":2,"101":1,"102":1,"124":2,"125":2,"149":1,"153":1,"160":1,"168":2,"172":2,"176":1,"179":1,"187":2,"189":1,"196":3,"199":1,"200":2,"246":1,"316":1,"352":1,"501":1,"502":2,"526":1,"539":1,"541":3,"552":3,"572":1,"636":1,"736":1}}],["surveys",{"2":{"31":1}}],["suggest",{"2":{"389":2}}],["suggested",{"0":{"438":1},"2":{"377":1,"438":1}}],["suggests",{"2":{"41":1}}],["suggestions",{"0":{"4":1},"2":{"4":1,"34":1}}],["sustainable",{"2":{"38":1}}],["sustained",{"2":{"14":1,"15":1}}],["sufficiently",{"2":{"416":1}}],["sufficient",{"2":{"34":1,"159":1,"257":1,"409":2}}],["succinct",{"0":{"203":1},"2":{"187":4,"188":2,"189":6,"190":1,"196":5,"203":7,"204":1,"206":1,"212":1,"218":1,"220":1,"369":1}}],["succinctxgit",{"2":{"187":1}}],["succinctx",{"2":{"187":4,"199":4,"200":2}}],["succinctgateway",{"0":{"187":1,"189":1},"1":{"190":1,"191":1,"192":1},"2":{"187":7,"189":1,"190":1,"195":3,"200":1,"219":2}}],["succinctlabs",{"2":{"121":4,"129":4,"146":2,"147":2,"148":4,"187":2,"188":2,"193":2,"196":2,"199":2,"200":2,"369":2}}],["succeeds",{"2":{"94":1}}],["success",{"2":{"32":1,"44":1,"62":1,"321":1,"641":3,"644":2}}],["successful",{"2":{"29":1,"36":1,"38":1,"39":1,"51":1,"57":1,"78":1,"189":1,"336":1,"358":1,"382":4,"416":1,"745":1}}],["successfully",{"2":{"23":1,"75":1,"107":2,"153":1,"204":1,"270":1,"352":1,"358":2,"364":1,"393":1,"396":1,"400":1,"487":1,"537":2,"539":1,"639":1,"640":1,"644":1,"736":1,"738":1,"739":1,"740":1,"750":1}}],["such",{"2":{"7":1,"30":1,"36":1,"37":2,"38":1,"71":1,"110":2,"111":1,"254":1,"269":1,"275":1,"310":1,"314":2,"361":1,"363":1,"374":1,"382":2,"394":1,"397":1,"399":1,"400":1,"403":1,"409":1,"413":1,"415":1,"416":2,"436":1,"456":1,"504":1,"506":1,"541":1,"552":1,"632":1,"671":1,"682":1}}],["subcommands",{"0":{"492":1},"2":{"488":6}}],["sub",{"2":{"365":4,"382":2}}],["subdomain",{"2":{"336":1}}],["subtract",{"2":{"279":1}}],["subtrees",{"2":{"415":1}}],["subtree",{"2":{"153":6}}],["subtreerootthreshold",{"2":{"135":2}}],["subspace",{"2":{"494":4}}],["subset",{"2":{"443":1}}],["subsequently",{"2":{"109":1}}],["subsequent",{"2":{"22":1}}],["subscribing",{"0":{"271":1,"272":1,"365":1},"2":{"361":1,"365":2}}],["subscribes",{"2":{"272":2}}],["subscribeblobs",{"2":{"271":2}}],["subscribeheaders",{"2":{"269":1,"272":4,"363":1}}],["subscribe",{"2":{"75":3,"76":1,"271":6,"272":6,"365":6}}],["subscription",{"2":{"147":6,"269":1,"363":1,"365":1}}],["subscriptions",{"2":{"75":1}}],["submitpayforblob",{"2":{"382":2,"383":1}}],["submitdata",{"2":{"382":2}}],["submitblob",{"2":{"270":4,"381":4}}],["submitblockdata",{"2":{"111":2,"113":2}}],["submitfraudproof",{"2":{"148":8}}],["submitter",{"2":{"494":1}}],["submitted",{"2":{"72":1,"110":8,"129":2,"140":2,"146":2,"148":2,"157":2,"193":1,"201":1,"321":1,"322":1,"358":2,"360":1,"377":3,"384":1,"495":1,"639":2,"660":1}}],["submitting",{"0":{"118":1,"265":1,"270":1,"290":1,"319":1,"336":1,"364":1,"370":1,"377":1},"1":{"291":1,"292":1,"320":1,"371":1,"372":1,"373":1,"374":1,"375":1,"376":1,"377":1,"378":1,"379":1,"380":1,"381":1,"382":1,"383":1,"384":1},"2":{"119":1,"142":1,"165":1,"168":1,"171":1,"172":1,"196":1,"202":1,"263":1,"296":1,"311":1,"319":1,"320":1,"321":1,"361":1,"364":2,"374":1,"376":1,"384":2,"435":1,"483":1,"494":1,"632":1,"677":1}}],["submits",{"2":{"111":2,"270":2,"358":3,"381":2,"637":1}}],["submitheader",{"2":{"111":4,"113":2}}],["submitrollupblock",{"2":{"103":2}}],["submit",{"0":{"291":1,"292":1,"320":1,"752":1},"2":{"72":1,"118":1,"148":5,"156":1,"187":1,"190":1,"191":3,"193":1,"196":1,"200":1,"202":1,"256":1,"260":1,"262":3,"265":3,"270":5,"288":1,"291":3,"292":3,"296":1,"304":1,"314":6,"319":12,"322":2,"336":2,"338":1,"346":1,"350":1,"356":2,"357":2,"358":11,"360":2,"361":2,"364":7,"370":2,"371":1,"375":1,"377":2,"378":1,"380":3,"381":4,"382":6,"383":2,"384":3,"416":1,"434":2,"443":1,"444":1,"453":1,"494":10,"495":4,"632":2,"671":2,"752":1}}],["submission",{"0":{"255":1,"263":1,"287":1},"1":{"256":1,"257":1,"258":1,"259":1,"260":1,"261":1,"262":1,"263":1,"264":1,"265":1,"266":1,"267":1,"288":1,"289":1,"290":1,"291":1,"292":1,"293":1,"294":1,"295":1,"296":1},"2":{"47":1,"171":1,"190":2,"191":1,"196":1,"284":1,"322":1,"336":1,"495":1}}],["submissions",{"2":{"20":1,"258":1}}],["subjectivity",{"2":{"595":1}}],["subject",{"2":{"23":1,"37":1,"277":1,"282":1,"371":1,"387":1,"446":2,"462":1}}],["s",{"0":{"71":1,"73":1,"77":1,"146":1,"267":1,"406":1},"1":{"407":1,"408":1,"409":1,"410":1,"411":1,"412":1},"2":{"7":1,"16":1,"18":1,"20":2,"21":1,"22":1,"23":2,"24":2,"26":1,"27":1,"35":1,"36":3,"38":1,"40":1,"42":1,"47":1,"50":1,"58":1,"66":1,"67":1,"68":1,"71":9,"72":9,"73":4,"75":4,"76":1,"77":1,"78":3,"79":1,"88":1,"92":1,"95":1,"96":2,"107":4,"110":1,"113":28,"115":3,"122":1,"124":1,"150":1,"159":1,"164":1,"172":3,"184":1,"187":3,"190":1,"195":4,"196":1,"199":2,"201":1,"203":1,"206":2,"211":2,"212":4,"218":2,"222":1,"237":1,"238":1,"242":3,"254":1,"256":2,"258":2,"263":1,"267":1,"270":2,"276":1,"279":1,"284":1,"289":1,"292":1,"297":1,"306":1,"310":1,"311":1,"313":1,"314":1,"316":1,"324":1,"325":1,"326":1,"327":1,"328":1,"331":1,"332":1,"341":2,"342":2,"344":1,"350":6,"352":1,"357":1,"358":2,"360":5,"362":1,"364":2,"369":1,"374":2,"375":2,"381":2,"382":2,"384":2,"387":1,"389":1,"391":1,"400":1,"402":1,"403":1,"406":1,"407":1,"408":1,"415":1,"416":1,"434":1,"444":1,"446":2,"448":1,"449":1,"453":1,"455":1,"456":1,"462":1,"465":1,"469":1,"470":2,"472":1,"484":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,"557":1,"560":1,"563":2,"569":1,"573":1,"585":14,"596":12,"601":2,"602":2,"604":1,"620":1,"632":2,"637":1,"639":2,"641":1,"664":1,"671":2,"678":2,"679":1,"684":1,"692":1,"693":2,"697":1,"699":1,"704":1,"705":2,"706":2,"710":1,"716":1,"722":1,"725":1,"736":2}}],["sig",{"2":{"508":1}}],["signal",{"2":{"163":1}}],["signatures",{"2":{"328":2,"506":1,"522":2}}],["signature",{"2":{"110":4,"113":4,"115":4,"328":2,"374":1,"415":1,"434":1}}],["signers",{"2":{"288":1}}],["signer",{"0":{"290":1,"292":1},"1":{"291":1,"292":1},"2":{"132":2,"187":2,"270":2,"292":4,"381":2,"382":12,"522":2}}],["signedtx",{"2":{"507":2}}],["signed",{"2":{"113":1,"291":1,"507":4}}],["signbytes",{"2":{"113":2}}],["signs",{"2":{"112":1}}],["sign",{"2":{"75":1,"113":2,"187":2,"353":2,"382":6,"507":6}}],["signifies",{"2":{"554":1}}],["significant",{"2":{"437":1,"591":1,"605":1}}],["significantly",{"2":{"38":1,"240":1,"282":1,"371":1}}],["signing",{"0":{"500":1},"2":{"73":1,"277":1,"278":1,"300":1,"302":1,"509":2,"522":2,"751":2}}],["signup",{"0":{"47":1}}],["situation",{"2":{"398":1}}],["site",{"0":{"0":1,"1":1},"1":{"1":1,"2":1,"3":1,"4":1},"2":{"1":1,"389":8}}],["six",{"2":{"378":1}}],["sidebar",{"2":{"503":2}}],["sidenode",{"2":{"144":4,"145":4,"148":8}}],["sidenodes",{"2":{"129":2,"141":6,"144":6,"145":2,"148":6}}],["side",{"2":{"141":2,"144":2,"153":2,"644":8}}],["simultaneous",{"2":{"502":2}}],["simulated",{"2":{"254":1}}],["simulates",{"2":{"189":1}}],["simplifies",{"2":{"296":1}}],["simplicity",{"2":{"157":1}}],["simply",{"2":{"111":2,"284":1,"477":1,"503":1,"648":1}}],["simpler",{"2":{"161":1,"396":1}}],["simplerollup",{"2":{"148":8}}],["simple",{"2":{"109":1,"112":1,"148":1,"165":1,"173":1,"188":1,"189":1,"219":1,"240":1}}],["simplest",{"2":{"69":1,"115":1,"604":1}}],["similarly",{"2":{"172":1,"196":1,"210":1,"412":1,"425":1,"454":1,"506":1}}],["similar",{"2":{"76":1,"78":1,"129":2,"145":1,"146":2,"147":2,"148":2,"171":1,"179":1,"187":1,"195":1,"282":1,"321":1,"324":1,"341":1,"342":1,"373":1,"414":1,"495":1,"563":1,"565":1,"616":1,"617":1,"644":2,"745":2,"751":1}}],["since",{"2":{"75":1,"76":1,"94":1,"104":1,"129":2,"146":2,"148":2,"154":1,"171":1,"184":1,"195":6,"281":1,"357":2,"358":2,"360":2,"373":1,"375":1,"394":1,"398":1,"408":1,"409":1,"416":1,"441":1,"505":1,"529":1,"678":1,"716":1}}],["single",{"2":{"13":1,"73":1,"105":1,"111":1,"122":1,"157":1,"165":3,"308":1,"400":1,"403":1,"413":1,"507":4,"514":3,"515":2,"666":1,"719":1}}],["sizeofdatainbytes",{"2":{"376":2}}],["sizes",{"2":{"132":2,"182":1,"375":1}}],["size",{"0":{"371":1},"2":{"6":1,"36":2,"76":4,"107":3,"142":1,"154":1,"156":1,"159":5,"314":1,"322":1,"340":2,"371":2,"374":4,"375":3,"376":1,"397":2,"402":1,"408":1,"414":1,"415":2,"434":1,"435":1,"440":1,"639":8,"683":9}}],["style",{"2":{"620":1}}],["styles",{"2":{"389":6}}],["stuck",{"2":{"254":1}}],["stub",{"2":{"103":1}}],["stick",{"2":{"157":1}}],["sticking",{"2":{"71":1}}],["still",{"2":{"151":1,"153":2,"196":2,"348":1,"352":1,"389":1,"408":1,"412":1,"462":1,"554":1,"574":1,"592":1,"632":1,"679":1,"756":1}}],["stops",{"2":{"254":1}}],["stop",{"0":{"636":1},"2":{"121":2,"494":1,"555":1,"636":1}}],["storing",{"2":{"97":1,"400":1,"401":1,"403":1,"405":1,"438":2,"739":1}}],["stories",{"2":{"42":1}}],["storecd",{"2":{"615":1}}],["storecelestia",{"2":{"317":1}}],["storesudo",{"2":{"615":1}}],["stores",{"2":{"412":1,"624":1,"717":1}}],["store=$home",{"2":{"317":2}}],["store",{"0":{"317":1,"404":1,"553":1,"561":1,"616":1},"1":{"318":1,"554":1,"555":1,"556":1,"557":1,"558":1,"559":1},"2":{"73":1,"94":1,"97":1,"310":1,"314":4,"315":7,"317":9,"318":8,"319":4,"321":2,"346":1,"352":4,"374":1,"399":1,"403":2,"404":1,"405":1,"408":1,"412":1,"436":1,"438":1,"477":1,"554":3,"555":9,"561":5,"562":4,"564":1,"605":3,"615":4,"616":9,"617":8,"625":1,"639":8,"670":2,"694":1}}],["stored",{"2":{"71":3,"94":1,"124":1,"145":1,"147":1,"204":1,"212":1,"310":1,"397":1,"398":1,"400":1,"436":1,"437":2,"489":1,"531":2,"605":2}}],["storage",{"0":{"399":1,"400":1,"401":1,"403":1,"586":1,"592":1,"615":1,"624":1,"625":1,"627":1,"630":1,"631":1,"632":1,"633":1,"635":1,"636":1,"738":1},"1":{"587":1,"588":1,"589":1,"590":1,"591":1,"592":1,"616":1,"617":1,"625":1,"626":1,"627":1,"628":2,"629":1,"630":1,"631":2,"632":2,"633":1,"634":2,"635":1,"636":1},"2":{"42":1,"97":1,"203":1,"308":1,"399":2,"400":2,"401":1,"403":2,"405":2,"437":1,"478":1,"503":2,"581":1,"591":1,"592":1,"604":1,"605":1,"617":1,"624":1,"625":2,"626":2,"632":3,"633":2,"634":1,"635":2,"636":1,"666":1,"691":1,"694":1,"699":1,"717":1,"719":1,"731":1,"738":2,"742":1}}],["straight",{"2":{"695":1}}],["straightforward",{"2":{"178":1}}],["strategy",{"2":{"562":1}}],["stronger",{"0":{"405":1}}],["strong",{"2":{"403":1}}],["str",{"2":{"364":4,"365":4,"366":4}}],["strike",{"2":{"211":1}}],["string",{"2":{"128":2,"187":2,"199":1,"269":1,"270":4,"271":4,"272":4,"273":4,"315":3,"316":1,"319":2,"336":1,"357":2,"358":17,"359":4,"360":8,"363":1,"381":4,"382":2,"467":1,"469":1,"470":1,"483":1,"541":2,"551":2,"552":2,"632":1,"692":1,"693":1,"704":1,"705":1}}],["struct",{"2":{"94":1,"110":10,"111":6,"139":4,"141":9,"142":2,"144":5,"145":4}}],["structures",{"0":{"104":1},"2":{"95":1,"110":2,"122":1,"145":2,"382":2}}],["structure",{"0":{"3":1},"2":{"22":1,"110":2,"141":2,"144":2}}],["streaming",{"0":{"205":1},"1":{"206":1,"207":1,"208":1,"209":1,"210":1,"211":1,"212":1,"213":1,"214":1,"215":1,"216":1},"2":{"40":4}}],["stress",{"2":{"35":1}}],["strengthen",{"2":{"32":1}}],["steal",{"2":{"394":1}}],["steady",{"2":{"18":1,"22":1}}],["step",{"0":{"61":1,"62":1,"64":1,"65":1,"69":1,"70":1,"71":1,"72":1,"73":1,"74":1,"75":1,"76":1,"77":1,"78":1},"2":{"54":2,"64":1,"70":1,"71":1,"72":2,"75":1,"110":1,"113":1,"115":1,"148":1,"160":1,"186":1,"189":1,"200":1,"312":1,"358":1,"425":1,"508":2,"642":1,"653":1,"658":1,"659":1,"679":1}}],["steps",{"0":{"98":1,"231":1,"246":1,"247":1,"248":1,"286":1,"361":1,"517":1,"569":1},"2":{"29":1,"75":1,"89":1,"90":1,"184":1,"186":1,"200":1,"369":1,"384":1,"476":1,"539":2,"540":1,"567":1,"568":1,"638":1,"640":1,"649":1,"653":1,"735":1,"751":3,"756":1}}],["stay",{"2":{"505":1,"698":1,"713":1,"729":1}}],["stands",{"2":{"459":1}}],["standard",{"2":{"41":1,"277":2,"279":1,"370":1,"372":1,"409":4,"435":1,"567":1,"637":1}}],["standards",{"0":{"7":1},"2":{"8":1,"14":1,"15":1}}],["stage",{"2":{"206":1,"679":1}}],["stack=",{"2":{"607":2,"756":2}}],["stack",{"0":{"222":1,"283":1,"344":1},"1":{"284":1,"285":1,"286":1,"345":1,"346":1,"347":1,"348":1,"349":1},"2":{"92":1,"221":1,"223":1,"231":1,"240":2,"242":1,"283":1,"284":1,"286":1,"344":1,"349":1,"413":2,"607":4,"756":4}}],["stakin",{"2":{"687":1,"688":1,"689":1}}],["staking",{"0":{"439":1,"447":1,"459":1,"460":1},"1":{"440":1,"441":1,"442":1,"443":1,"444":1,"445":1,"446":1,"448":1,"449":1},"2":{"71":1,"72":1,"277":1,"417":1,"440":3,"446":1,"447":2,"448":1,"449":1,"455":1,"456":1,"459":2,"494":6,"497":4,"500":1,"745":2,"751":2}}],["stakeflow",{"2":{"687":1,"688":1,"689":1,"696":1,"706":1,"707":1,"708":1,"712":1}}],["staked",{"2":{"420":1,"446":1}}],["stakecurrency",{"2":{"389":2}}],["stakers",{"2":{"443":1,"444":1}}],["staker",{"2":{"78":3,"687":1,"688":1,"689":1}}],["stakes",{"2":{"71":2}}],["stake",{"0":{"417":1,"419":1,"422":1,"424":1,"426":1,"428":1,"431":1,"440":1,"455":1,"745":1},"1":{"418":1,"419":1,"420":2,"421":2,"422":2,"423":2,"424":1,"425":2,"426":2,"427":2,"428":1,"429":2,"430":2,"431":2,"432":2},"2":{"18":1,"21":1,"23":1,"71":6,"209":1,"417":3,"420":1,"422":1,"423":1,"425":1,"426":1,"427":1,"429":1,"432":1,"440":4,"447":1,"448":1,"455":2,"477":1,"494":1,"509":4,"745":1}}],["static",{"2":{"374":4,"375":2,"503":2}}],["statscelestia",{"2":{"330":1}}],["stats",{"0":{"330":1},"2":{"330":1}}],["statesync",{"2":{"595":3}}],["stateroot",{"2":{"340":2}}],["stated",{"2":{"195":1}}],["stateless",{"2":{"107":1}}],["state",{"0":{"402":1,"470":1,"591":1,"595":1,"693":1,"705":1},"2":{"71":4,"112":2,"115":1,"148":4,"164":3,"168":1,"172":1,"176":1,"179":1,"187":1,"195":3,"212":1,"238":1,"261":2,"262":2,"263":2,"264":2,"281":1,"282":1,"284":1,"296":2,"311":2,"314":2,"324":2,"325":4,"327":2,"331":6,"352":2,"373":1,"374":1,"375":1,"377":1,"383":1,"394":1,"402":4,"412":3,"413":8,"415":2,"436":2,"438":2,"470":3,"477":2,"483":2,"488":6,"530":1,"591":1,"592":1,"593":2,"595":5,"596":1,"605":3,"607":29,"632":3,"647":1,"671":1,"693":2,"705":2,"706":1,"708":1,"756":31}}],["statements",{"0":{"356":1},"2":{"356":2}}],["statement",{"2":{"26":1}}],["status",{"0":{"191":1,"230":1},"2":{"6":1,"62":1,"190":2,"191":6,"230":2,"252":2,"253":2,"386":1,"467":1,"488":4,"510":1,"557":2,"693":2,"736":4,"740":2}}],["stable",{"2":{"22":1,"679":1,"722":1}}],["stability",{"0":{"462":1,"679":1},"2":{"18":1}}],["startcelestia",{"2":{"597":1,"663":1,"749":1,"751":1}}],["starts",{"2":{"195":1,"202":1,"441":1}}],["startshare=0",{"2":{"137":1}}],["startindex",{"2":{"157":2,"159":2}}],["starting",{"2":{"20":1,"44":1,"110":1,"126":1,"136":1,"269":1,"347":1,"363":1,"413":1,"444":1,"484":1,"555":1,"565":1,"659":1,"752":1}}],["startblock",{"2":{"129":6,"146":4,"193":1}}],["start=10",{"2":{"127":1}}],["start",{"0":{"487":1,"597":1,"614":1,"617":1,"632":1,"635":1,"646":1,"671":1,"676":1,"730":1,"736":1,"750":1},"1":{"672":1,"673":1,"674":1,"675":1,"676":1,"731":1,"732":1,"733":1},"2":{"76":3,"77":1,"110":2,"111":1,"121":2,"128":2,"129":4,"132":5,"137":2,"141":3,"148":4,"157":1,"201":1,"231":1,"234":1,"240":1,"252":2,"253":3,"261":3,"262":3,"269":1,"286":1,"289":3,"311":12,"312":8,"314":1,"352":4,"357":1,"363":1,"440":1,"466":1,"470":6,"483":1,"484":8,"485":7,"486":1,"488":4,"507":2,"523":2,"535":1,"537":2,"542":3,"544":3,"545":2,"546":2,"547":4,"550":2,"555":3,"560":3,"564":2,"595":1,"597":9,"599":1,"608":1,"614":6,"616":1,"617":10,"632":5,"633":7,"634":1,"638":1,"639":6,"646":2,"663":2,"671":7,"672":8,"674":7,"675":1,"685":1,"693":3,"703":1,"716":1,"733":1,"736":5,"738":5,"739":6,"740":7,"749":1,"751":9,"756":1}}],["startup",{"2":{"34":1,"739":1}}],["started",{"0":{"100":1,"276":1,"733":1},"1":{"101":1,"102":1,"103":1},"2":{"1":1,"26":1,"33":1,"41":1,"77":1,"252":1,"259":1,"276":1,"301":1,"348":1,"350":1,"448":1,"449":1,"540":1,"555":1,"568":1,"609":1,"615":1,"730":1,"731":1,"732":1,"733":1,"736":1,"740":1}}],["gprc",{"2":{"689":1}}],["gpt3dot5turbo",{"2":{"359":2}}],["gpt3",{"2":{"359":4,"360":2}}],["gpt",{"2":{"350":2,"351":1,"359":4,"360":4,"361":2}}],["gg",{"2":{"640":2}}],["gz",{"2":{"623":24}}],["global",{"2":{"503":3,"639":2}}],["glance",{"0":{"451":1}}],["gcpbb",{"2":{"375":1}}],["gbps",{"2":{"478":2,"581":2,"626":2,"719":2,"720":2,"742":2}}],["gb",{"2":{"308":1,"478":1,"581":1,"626":1,"666":1,"719":3,"720":2,"742":1}}],["gm",{"2":{"262":2,"291":2,"292":2,"314":4,"319":3,"321":6,"322":2}}],["ghcr",{"2":{"252":2,"614":6,"616":6,"617":6}}],["gnark",{"2":{"199":1}}],["gklcgflcsrm9caqasaapsckykhy9jb3ntb3muy3j5chrvlnnly3ayntzrms5qdwjlzxksiwohaqdvbvupglandgtlocgsoherbafsfbb5l0wdvbjjesehegqkaggbgjjcariscgwkbhv0awesbdixotiqto9cgkcjrjcoyijj81bttfb2gudg7o8auawf0bscbhw9ppd99xpq1slpemfyq0y1joj",{"2":{"132":4}}],["gt",{"2":{"62":1,"76":8,"113":2,"121":3,"125":8,"127":1,"132":1,"133":2,"135":7,"137":1,"188":8,"189":30,"191":4,"192":4,"194":6,"200":2,"218":1,"246":13,"247":12,"248":12,"252":5,"253":8,"259":4,"266":4,"311":6,"312":21,"314":6,"316":6,"317":4,"319":8,"321":8,"325":2,"328":4,"331":6,"333":4,"340":1,"357":6,"358":6,"360":12,"365":8,"366":2,"379":4,"380":4,"389":14,"470":6,"472":3,"483":8,"484":4,"485":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,"555":10,"560":8,"561":4,"562":8,"564":9,"596":12,"597":1,"617":2,"623":8,"632":4,"633":12,"640":4,"672":6,"673":3,"674":12,"693":6,"710":3,"716":2,"736":2,"738":2,"739":4,"740":4,"745":4,"748":2,"751":3}}],["go$ver",{"2":{"623":24}}],["goroutine",{"2":{"607":4,"756":4}}],["gone",{"2":{"535":1}}],["good",{"2":{"503":1,"615":2}}],["google",{"2":{"382":4}}],["govcelestia",{"2":{"494":1}}],["governing",{"2":{"456":1}}],["governance",{"0":{"439":1,"442":1,"456":1,"494":1},"1":{"440":1,"441":1,"442":1,"443":2,"444":2,"445":1,"446":1,"495":1},"2":{"371":1,"374":1,"375":2,"412":1,"417":1,"443":2,"444":1,"446":1,"456":3,"494":3,"495":1,"683":3}}],["governs",{"2":{"220":1}}],["gov",{"2":{"277":1,"494":12,"495":2}}],["gobin",{"2":{"245":2}}],["go\\t",{"2":{"146":1}}],["go\\tsharesproof",{"2":{"138":1}}],["goimport",{"2":{"135":1,"270":1,"376":1,"381":1,"382":1}}],["going",{"2":{"64":1,"517":1,"574":1,"717":1,"736":1}}],["gopackage",{"2":{"128":1,"129":1,"148":1,"356":1}}],["goleveldb",{"2":{"125":4}}],["golangnodearabica",{"2":{"623":2}}],["golangnodemocha",{"2":{"623":2}}],["golangnodemainnet",{"2":{"623":2}}],["golang",{"0":{"128":1,"133":1,"135":1,"138":1,"268":1,"381":1,"623":1},"1":{"269":1,"270":1,"271":1,"272":1,"273":1,"274":1},"2":{"121":2,"128":1,"133":1,"134":1,"135":1,"138":1,"141":1,"143":1,"144":1,"145":1,"148":1,"213":1,"218":1,"268":1,"299":1,"350":2,"351":1,"360":1,"381":2,"382":4,"567":1,"623":11}}],["gogo",{"2":{"125":2}}],["gohttps",{"2":{"121":2}}],["gotten",{"2":{"140":1,"160":1}}],["got",{"2":{"115":2,"321":1,"607":2,"756":2}}],["gotype",{"2":{"110":2,"111":1}}],["gossiped",{"2":{"373":1,"385":1,"407":1}}],["gossiping",{"2":{"115":2,"386":1}}],["gossip",{"2":{"115":1,"625":1}}],["gofunc",{"2":{"113":2,"115":1,"141":1,"142":1,"143":1,"144":1,"145":1,"271":1,"358":1,"360":1}}],["goals",{"2":{"240":1}}],["goal",{"2":{"110":1}}],["goes",{"2":{"54":1,"65":1,"164":3,"184":1,"279":1,"564":1,"567":1}}],["go",{"0":{"355":1},"2":{"44":1,"94":1,"110":1,"111":1,"119":1,"121":3,"125":5,"129":7,"133":1,"135":2,"146":1,"147":2,"148":7,"151":1,"185":1,"199":4,"213":1,"228":1,"229":1,"243":2,"272":1,"273":1,"314":1,"352":4,"355":10,"356":3,"357":5,"358":6,"359":1,"360":10,"413":1,"476":1,"489":1,"503":1,"515":2,"534":1,"539":1,"544":1,"557":1,"558":1,"562":2,"607":36,"615":2,"621":1,"623":18,"637":4,"642":1,"733":1,"756":36}}],["gather",{"2":{"544":1}}],["gateways",{"2":{"187":1}}],["gateway",{"0":{"302":1,"303":1},"1":{"303":1},"2":{"76":2,"78":2,"186":1,"187":3,"189":7,"190":1,"195":3,"196":4,"220":1,"297":1,"298":1,"302":3,"303":1,"307":1,"552":3,"707":1}}],["game",{"2":{"350":2}}],["gaming",{"2":{"37":1,"241":1}}],["gas=220000",{"2":{"751":1}}],["gas=220000moniker=",{"2":{"751":1}}],["gasperblobbyte",{"2":{"374":1}}],["gasperbyte",{"2":{"374":1,"375":2}}],["gaspricestep",{"2":{"389":2}}],["gasprice",{"2":{"341":2,"364":5,"373":1}}],["gastoconsume",{"2":{"374":2,"375":1}}],["gasused",{"2":{"340":2}}],["gaslimit",{"2":{"319":4,"331":1,"340":2,"373":1,"376":2,"382":4}}],["gas",{"0":{"322":1,"373":1,"374":1,"375":1,"376":1},"1":{"374":1,"375":1,"376":1},"2":{"67":1,"69":1,"71":3,"72":2,"76":4,"132":4,"204":1,"211":1,"270":3,"296":1,"319":2,"322":12,"331":7,"341":2,"358":2,"364":3,"371":1,"372":2,"373":9,"374":9,"375":18,"376":5,"382":6,"391":3,"435":1,"454":1,"496":2,"498":2,"522":12,"529":2,"639":16,"745":4,"751":4}}],["gaining",{"2":{"394":1}}],["gain",{"2":{"42":1,"50":1,"58":1,"350":1,"400":1}}],["gained",{"2":{"32":1}}],["gap",{"2":{"36":1}}],["gauge",{"2":{"36":1}}],["grid",{"2":{"371":1}}],["grpcaddr",{"2":{"382":4}}],["grpc",{"0":{"382":1,"470":1,"689":1,"693":1,"705":1,"708":1},"2":{"311":3,"336":1,"382":10,"467":3,"470":4,"483":2,"484":1,"503":3,"541":2,"632":2,"639":12,"671":2,"689":21,"693":12,"705":8,"707":3,"708":19}}],["grzxooejihch93+g3mldibq6ff+nroakrbgupfu8muo=",{"2":{"132":2}}],["groundbreaking",{"2":{"679":1}}],["group",{"2":{"36":2,"39":1,"40":1,"41":13,"42":7}}],["groups",{"2":{"34":1,"36":2,"39":5,"41":2}}],["grove",{"0":{"695":1},"2":{"466":2,"685":2,"695":2,"703":2}}],["growth",{"2":{"58":1}}],["grow",{"2":{"44":1}}],["growing",{"2":{"44":1}}],["grep",{"2":{"343":1}}],["grepping",{"2":{"343":1}}],["green",{"2":{"122":1}}],["greet",{"2":{"31":1}}],["greater",{"2":{"182":1,"595":1}}],["greatly",{"2":{"40":1}}],["great",{"2":{"35":1,"286":1}}],["graphical",{"2":{"506":1}}],["graphics",{"2":{"30":1}}],["grafana",{"2":{"503":3,"505":3,"548":1,"693":1}}],["grant",{"0":{"261":1},"2":{"260":1,"261":3,"264":2,"266":3}}],["grantee",{"0":{"262":1},"2":{"257":1,"258":2,"259":5,"261":3,"266":6}}],["granter",{"0":{"261":1,"264":1,"267":1},"1":{"265":1},"2":{"257":1,"258":2,"259":5,"262":2,"266":6,"267":1,"522":2}}],["granted",{"2":{"47":1,"267":1}}],["granting",{"0":{"259":1,"266":1},"1":{"260":1,"261":1,"267":1},"2":{"256":1,"259":1}}],["gravity",{"2":{"206":1}}],["gradients",{"2":{"122":1}}],["grading",{"2":{"22":1}}],["grassroots",{"2":{"43":1}}],["gratitude",{"2":{"37":1,"38":1}}],["gracefully",{"2":{"7":1,"636":1}}],["guru",{"2":{"696":1,"706":2,"707":2,"708":2,"712":1}}],["guardian",{"2":{"187":1,"195":1}}],["guardian=",{"2":{"187":2}}],["guaranteeing",{"2":{"414":1}}],["guaranteed",{"2":{"280":1,"416":1,"438":2}}],["guarantees",{"2":{"206":2,"220":1,"438":1}}],["guarantee",{"2":{"26":1,"403":1,"404":1,"407":1,"409":1,"436":1,"470":1,"693":1,"705":1}}],["guidance",{"0":{"33":1},"1":{"34":1,"35":1,"36":1,"37":1,"38":1,"39":1,"40":1,"41":1,"42":1}}],["guide",{"0":{"29":1,"313":1,"571":1,"637":1,"649":1,"730":1},"1":{"30":1,"31":1,"32":1,"33":1,"34":1,"35":1,"36":1,"37":1,"38":1,"39":1,"40":1,"41":1,"42":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,"572":1,"573":1,"574":1,"575":1,"576":1,"577":1,"578":1,"579":1,"638":1,"639":1,"640":1,"641":1,"642":1,"643":1,"644":1,"645":1,"646":1,"647":1,"648":1,"650":1,"651":1,"652":1,"653":1,"654":1,"655":1,"656":1,"657":1,"658":1,"659":1,"660":1,"661":1,"662":1,"663":1,"731":1,"732":1,"733":1},"2":{"43":1,"50":1,"54":1,"59":2,"66":3,"70":1,"78":1,"202":1,"245":1,"251":1,"256":1,"259":1,"268":1,"269":1,"307":1,"344":1,"361":1,"362":1,"363":1,"465":2,"504":1,"508":2,"511":1,"534":1,"538":1,"548":1,"555":1,"564":1,"580":1,"605":1,"624":1,"637":1,"649":2,"651":1,"652":1,"664":1,"678":1,"684":2,"699":2,"717":1,"733":1,"741":1}}],["guidelines",{"0":{"2":1,"11":1,"52":1},"1":{"12":1,"13":1,"14":1,"15":1},"2":{"2":1,"11":1,"16":1,"33":2}}],["guides",{"2":{"0":1,"3":1,"23":2,"33":1,"79":1,"232":1,"604":1,"699":1,"720":1}}],["gi",{"2":{"127":2}}],["git",{"2":{"74":1,"102":1,"188":2,"193":2,"196":2,"199":2,"200":4,"245":3,"539":7,"567":7,"622":4}}],["gitgit",{"2":{"74":1}}],["githubusercontent",{"2":{"391":1,"585":10,"622":2}}],["github",{"0":{"285":1},"2":{"26":2,"74":2,"121":4,"125":18,"128":2,"129":16,"135":6,"141":2,"142":2,"145":2,"146":2,"147":2,"148":24,"187":2,"188":2,"193":2,"196":2,"199":2,"200":2,"245":2,"269":2,"270":6,"285":1,"302":1,"307":1,"355":4,"356":8,"369":2,"376":2,"381":6,"382":16,"508":2,"531":2,"539":2,"552":1,"567":2,"607":16,"720":1,"756":16}}],["given",{"0":{"562":1},"2":{"94":1,"131":1,"156":1,"182":1,"187":1,"194":1,"270":1,"273":3,"336":1,"358":1,"359":2,"364":1,"366":1,"397":1,"410":1,"462":1,"560":2,"562":2,"591":1,"683":1,"725":1,"727":1}}],["gives",{"2":{"39":1,"58":1,"96":1,"169":1,"440":1,"683":1}}],["give",{"2":{"38":1,"232":1,"260":1,"615":1,"716":1}}],["giving",{"2":{"7":1,"34":1,"96":1}}],["gifts",{"2":{"37":1}}],["g",{"2":{"22":1,"32":1,"42":2,"257":1,"314":1,"369":1,"371":1,"396":1,"410":1,"416":1,"503":1,"564":1,"597":1,"601":2,"751":1}}],["gem",{"0":{"428":1,"429":1},"1":{"429":1,"430":1,"431":1,"432":1},"2":{"417":1}}],["gentx",{"2":{"488":2,"500":4,"659":5,"660":7}}],["gentxscelestia",{"2":{"660":1}}],["gentxs",{"2":{"488":2,"660":1}}],["gentle",{"2":{"66":1,"67":1,"88":1}}],["genutil",{"2":{"277":1}}],["genuine",{"2":{"47":1,"58":1}}],["genuinely",{"2":{"46":1}}],["genesis",{"0":{"445":1,"500":1,"657":1,"659":1,"660":1},"1":{"446":1},"2":{"145":2,"194":2,"195":4,"400":1,"437":1,"444":1,"445":2,"446":1,"451":1,"488":20,"500":3,"532":1,"542":10,"579":1,"585":7,"593":1,"657":2,"658":3,"659":2,"660":5,"716":1,"756":1}}],["generating",{"0":{"246":1},"2":{"153":1,"159":1,"188":1,"202":1,"352":2}}],["generation",{"2":{"125":1,"158":1,"414":1}}],["generate",{"2":{"153":1,"184":1,"187":1,"188":1,"196":2,"198":2,"199":4,"200":3,"202":1,"212":1,"218":1,"220":1,"243":1,"246":1,"321":1,"350":1,"353":2,"398":2,"409":1,"488":2,"500":1,"507":4,"534":1}}],["generates",{"2":{"126":1}}],["generated",{"2":{"71":4,"109":1,"121":1,"136":1,"160":1,"184":1,"196":1,"199":1,"200":4,"204":2,"270":2,"312":2,"319":1,"324":1,"352":2,"358":1,"364":2,"409":1,"472":1,"484":1,"632":1,"656":1,"659":1,"672":1,"673":1,"710":1}}],["generally",{"2":{"374":1}}],["generalized",{"2":{"157":1}}],["general",{"0":{"715":1},"1":{"716":1},"2":{"26":1,"182":1,"238":1,"244":1,"374":1,"407":1,"446":1,"715":1}}],["gender",{"2":{"6":1}}],["geolocation",{"2":{"23":1}}],["getaddress",{"2":{"382":2}}],["getall",{"2":{"270":3,"272":2,"358":1,"364":1,"381":2,"438":2}}],["getenv",{"2":{"359":2}}],["geteds",{"2":{"273":7,"366":1}}],["geth",{"2":{"284":1}}],["getheader",{"2":{"111":4,"115":2}}],["getbyheight",{"2":{"273":2,"328":1}}],["getblockdata",{"2":{"111":2,"115":2}}],["getlatestblock",{"2":{"115":2}}],["getlatestrollupheight",{"2":{"111":2}}],["gets",{"2":{"94":1,"95":1,"122":1,"124":2,"157":1,"199":1,"509":1}}],["getting",{"0":{"100":1,"276":1,"733":1},"1":{"101":1,"102":1,"103":1},"2":{"24":2,"78":2,"80":1,"188":1,"194":1,"276":1,"351":1,"393":1,"509":1}}],["get",{"0":{"326":1,"327":1,"328":1,"330":1},"2":{"1":1,"26":1,"33":1,"36":1,"41":1,"94":1,"95":2,"129":8,"132":1,"146":2,"148":4,"160":2,"189":2,"203":1,"237":1,"238":1,"241":1,"254":1,"259":1,"269":3,"273":2,"301":1,"319":1,"321":15,"326":1,"327":1,"328":6,"329":4,"333":1,"334":7,"342":2,"348":1,"350":1,"355":4,"357":2,"358":5,"360":4,"361":1,"363":1,"364":2,"365":2,"366":8,"371":1,"375":1,"377":1,"382":2,"393":1,"448":1,"449":1,"484":1,"493":1,"494":1,"504":1,"509":1,"537":1,"540":1,"557":2,"558":2,"559":2,"564":2,"568":1,"585":1,"609":1,"632":2,"640":1,"671":1,"673":1,"730":1,"731":1,"732":1,"733":1,"745":1}}],["oil",{"2":{"519":2}}],["ok",{"2":{"365":4}}],["okpajw+kyhnqkq5vch",{"2":{"132":2}}],["o",{"2":{"199":2,"408":1,"409":2,"596":12}}],["older",{"2":{"193":1,"437":1,"714":1}}],["old",{"2":{"137":1,"138":1,"401":1,"550":1,"560":1}}],["ostrich",{"2":{"519":2}}],["os|file|kwallet|pass|test|memory",{"2":{"489":1,"531":2}}],["oss",{"2":{"280":1}}],["os",{"2":{"128":2,"129":2,"148":2,"356":2,"357":8,"358":8,"359":2,"360":8,"502":2,"622":1}}],["occupied",{"2":{"683":1}}],["occurs",{"2":{"394":1}}],["occurring",{"2":{"282":1}}],["occur",{"2":{"115":1,"494":1}}],["occasional",{"2":{"679":1,"722":1}}],["october",{"2":{"27":1}}],["omitempty",{"2":{"110":2}}],["own",{"2":{"71":1,"76":1,"202":1,"206":1,"240":1,"242":1,"270":1,"283":1,"298":1,"337":1,"344":1,"353":1,"369":1,"410":1,"413":1,"414":1,"438":1,"440":1,"454":1,"455":1,"467":1,"470":1,"483":1,"525":1,"529":1,"537":1,"539":1,"576":1,"632":1,"649":1,"655":1,"685":1,"693":1,"695":1,"703":1,"745":1,"748":1}}],["owners",{"2":{"78":6}}],["ownership",{"2":{"78":2}}],["owner",{"2":{"71":1,"78":5,"187":3,"189":4,"195":1}}],["ovh",{"2":{"23":1}}],["override",{"2":{"503":2}}],["overpaying",{"2":{"375":1}}],["overpay",{"2":{"373":1}}],["overhead",{"2":{"283":1,"284":1}}],["over",{"0":{"644":1},"2":{"53":1,"54":1,"110":4,"112":1,"113":1,"124":3,"151":1,"152":1,"184":1,"185":1,"196":1,"213":1,"238":1,"243":2,"302":1,"312":1,"314":1,"438":1,"456":1,"476":1,"534":1,"544":1,"564":1,"567":1,"621":1,"644":2,"733":1}}],["overview",{"0":{"92":1,"122":1,"256":1,"288":1,"435":1,"450":1,"477":1,"625":1,"665":1,"717":1},"1":{"123":1,"124":1,"451":1,"452":1,"453":1,"454":1,"455":1,"456":1,"457":1,"458":1,"459":1,"460":1,"718":1,"719":1,"720":1},"2":{"21":1,"31":1,"41":1,"53":1,"54":1,"55":1,"62":1,"64":1,"66":1,"67":1,"81":1,"88":1,"107":2,"276":1,"423":1,"427":1,"435":1,"466":2,"548":1,"569":1,"685":2,"703":2}}],["overall",{"2":{"7":1,"35":1,"81":1}}],["otel",{"2":{"23":1,"545":2,"546":2,"548":3}}],["otherwise",{"2":{"10":1,"76":1,"94":1,"96":1,"134":1,"168":1,"187":1,"193":1,"195":1,"198":1,"319":1,"336":1,"358":1,"494":1,"653":1,"736":1}}],["others",{"2":{"7":1,"195":1,"662":1}}],["other",{"0":{"282":1,"610":1,"658":1},"2":{"3":1,"7":2,"8":1,"12":1,"23":2,"24":1,"26":1,"34":1,"36":5,"37":2,"39":2,"41":1,"42":3,"46":1,"71":1,"92":1,"94":1,"105":1,"110":2,"119":1,"121":1,"149":1,"163":1,"182":1,"200":1,"201":1,"211":1,"224":1,"241":1,"246":1,"260":1,"280":2,"282":1,"296":1,"298":1,"307":1,"311":1,"371":2,"372":1,"375":1,"393":1,"400":1,"403":1,"410":1,"413":2,"438":3,"444":1,"455":1,"471":1,"477":1,"483":1,"554":1,"564":1,"601":1,"632":1,"647":1,"658":2,"659":1,"660":2,"661":2,"662":2,"710":1,"731":1}}],["observe",{"2":{"562":1}}],["observer",{"2":{"545":2}}],["observed",{"2":{"184":1}}],["object",{"2":{"75":2}}],["objective",{"2":{"26":1}}],["objectives",{"0":{"18":1},"2":{"18":1,"30":1}}],["obtained",{"2":{"194":1,"195":1,"351":1}}],["obtain",{"2":{"38":1,"75":1}}],["obligated",{"2":{"10":1}}],["onstart",{"2":{"607":2,"756":2}}],["onto",{"2":{"405":1}}],["oncall",{"2":{"505":2}}],["onclick=",{"2":{"389":2}}],["onchain",{"0":{"204":1},"2":{"96":1,"97":1,"187":1,"188":2,"196":1,"198":1,"200":2,"206":2,"214":1,"219":1,"241":1,"284":1,"394":1,"446":1,"751":1}}],["once",{"2":{"24":1,"72":1,"75":1,"94":1,"113":1,"172":1,"179":1,"195":1,"237":1,"243":1,"258":1,"312":1,"314":1,"353":1,"384":1,"396":1,"403":1,"415":1,"436":1,"437":2,"438":1,"484":1,"503":2,"522":1,"595":1,"632":1,"636":1,"658":1,"660":2,"672":1}}],["only",{"2":{"76":3,"81":2,"94":2,"95":3,"96":1,"115":2,"134":1,"151":1,"154":1,"157":1,"160":1,"165":1,"190":1,"191":2,"199":1,"212":1,"241":1,"242":1,"246":1,"277":1,"279":1,"302":1,"307":1,"317":1,"358":1,"371":1,"375":1,"382":2,"398":1,"400":1,"403":1,"406":1,"407":1,"408":1,"409":2,"410":2,"412":1,"413":3,"414":2,"415":2,"437":2,"438":1,"495":1,"507":2,"519":2,"536":1,"541":1,"554":1,"585":1,"597":1,"620":1,"636":1,"649":1,"693":1,"709":1,"733":1,"751":2}}],["online",{"2":{"9":1,"31":1,"38":1,"39":1,"736":2,"738":2,"739":2,"740":2}}],["onboarding",{"0":{"42":1}}],["ones",{"2":{"154":1,"172":1,"198":1,"642":1,"645":1}}],["onestepproofentry",{"2":{"85":1,"86":1}}],["onestepproverhostio",{"2":{"85":1,"86":1}}],["onestepprovermath",{"2":{"85":1,"86":1}}],["onestepprovermemory",{"2":{"85":1,"86":1}}],["onestepprover0",{"2":{"85":1,"86":1}}],["one",{"0":{"377":1},"2":{"39":1,"40":1,"41":1,"42":1,"59":2,"75":1,"76":1,"89":1,"92":1,"94":1,"95":1,"119":1,"137":1,"138":1,"151":1,"154":1,"163":1,"169":1,"172":1,"176":3,"184":1,"193":1,"195":1,"197":1,"199":1,"201":2,"213":1,"216":1,"238":1,"240":1,"241":1,"242":1,"246":1,"252":2,"253":2,"257":2,"258":1,"259":1,"307":1,"311":2,"356":2,"359":1,"361":1,"370":1,"371":1,"374":1,"377":1,"391":2,"397":1,"409":2,"410":1,"413":2,"415":2,"416":2,"483":1,"484":2,"488":4,"493":5,"494":3,"503":3,"515":1,"517":1,"518":2,"527":1,"529":2,"531":2,"554":1,"555":2,"562":1,"579":2,"625":1,"632":1,"637":2,"639":2,"660":1,"671":4,"672":2,"683":2,"687":1,"688":1,"689":1,"693":1,"705":1,"707":1,"708":1,"746":1,"748":1}}],["onwards",{"2":{"22":1,"148":2}}],["on",{"0":{"125":1,"223":1,"233":1,"318":1,"342":1,"440":1,"447":1,"509":1,"518":1,"529":1,"592":1,"602":1,"717":1},"1":{"126":1,"127":1,"128":1,"129":1,"130":1,"131":1,"132":1,"133":1,"134":1,"135":1,"136":1,"137":1,"138":1,"224":1,"225":1,"226":1,"227":1,"228":1,"229":1,"230":1,"234":1,"235":1,"236":1,"448":1,"449":1,"519":1,"520":1,"521":1,"522":1,"523":1,"524":1,"525":1,"718":1,"719":1,"720":1},"2":{"7":2,"22":1,"23":5,"26":5,"30":1,"33":1,"34":3,"35":2,"37":2,"38":1,"39":4,"40":5,"41":1,"42":1,"47":1,"53":1,"54":1,"55":2,"58":1,"61":1,"62":2,"64":3,"66":1,"67":2,"69":1,"71":10,"72":2,"73":1,"75":1,"76":5,"77":1,"78":51,"82":1,"92":1,"94":3,"95":1,"96":4,"97":3,"105":1,"109":2,"111":6,"113":1,"121":1,"122":1,"124":2,"148":1,"150":1,"152":1,"153":3,"154":1,"155":1,"156":2,"159":2,"160":2,"163":1,"165":1,"167":1,"187":2,"189":1,"199":3,"200":4,"201":2,"203":1,"204":1,"206":2,"209":2,"210":4,"211":2,"212":5,"213":2,"215":1,"216":8,"219":1,"221":1,"222":2,"223":1,"232":2,"234":1,"236":1,"238":2,"242":1,"243":1,"246":1,"251":2,"252":4,"253":1,"254":4,"256":1,"257":1,"260":1,"266":1,"270":2,"276":1,"278":1,"280":1,"282":1,"283":1,"284":6,"285":1,"295":1,"298":2,"302":1,"307":1,"310":1,"311":2,"312":3,"316":3,"317":1,"321":1,"322":1,"336":1,"339":1,"340":1,"341":1,"346":1,"347":3,"348":1,"350":2,"352":1,"356":1,"369":1,"371":3,"374":3,"375":6,"381":2,"382":2,"384":4,"385":1,"387":1,"394":1,"396":1,"397":1,"405":1,"406":1,"407":1,"408":1,"410":1,"412":2,"413":3,"416":1,"417":2,"422":1,"426":1,"434":1,"435":3,"437":2,"438":6,"440":3,"441":1,"443":2,"453":2,"454":2,"456":1,"461":3,"462":2,"465":1,"467":2,"470":1,"472":2,"477":2,"479":1,"483":4,"484":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,"555":1,"557":1,"567":1,"568":1,"580":1,"582":1,"583":1,"584":1,"590":1,"592":1,"593":1,"594":1,"595":2,"597":1,"601":3,"604":4,"605":1,"607":1,"610":1,"612":1,"614":3,"615":5,"620":1,"622":1,"627":1,"632":4,"637":3,"639":2,"648":1,"663":2,"664":1,"665":1,"667":1,"668":2,"671":3,"676":1,"677":1,"683":3,"684":1,"685":2,"687":2,"693":1,"694":1,"695":2,"698":1,"699":7,"703":2,"706":1,"710":2,"713":1,"714":1,"715":2,"716":3,"717":3,"720":1,"725":3,"727":3,"730":1,"733":2,"741":1,"743":2,"744":1,"748":2,"750":1,"751":2,"756":2}}],["opqym",{"2":{"520":2,"524":2}}],["oprexqlg9er1oey1de4mkwvmjlfnqoocg==",{"2":{"328":2}}],["op",{"0":{"222":1,"283":1,"344":1},"1":{"284":1,"285":1,"286":1,"345":1,"346":1,"347":1,"348":1,"349":1},"2":{"221":1,"223":1,"231":1,"283":1,"284":10,"286":1,"339":1,"343":1,"344":3,"347":3,"349":1}}],["ops",{"2":{"197":1}}],["opt",{"2":{"196":1,"454":1,"563":4}}],["opted",{"2":{"187":2}}],["optimal",{"2":{"567":1}}],["optimally",{"2":{"413":1}}],["optimized",{"2":{"413":1}}],["optimize",{"2":{"282":1}}],["optimism",{"0":{"339":1},"1":{"340":1,"341":1,"342":1,"343":1},"2":{"222":2,"283":1,"284":2,"285":1,"339":1,"349":1,"446":1}}],["optimised",{"2":{"206":1}}],["optimistically",{"2":{"163":1,"164":1}}],["optimistic",{"0":{"163":1,"164":1,"165":1,"166":1,"167":1,"168":1,"169":1,"170":1},"1":{"164":1,"165":2,"166":2,"167":2,"168":1,"169":2,"170":2},"2":{"112":1,"115":1,"150":1,"156":1,"163":4,"164":1,"167":1,"171":1,"182":1,"212":1}}],["option",{"0":{"291":1,"292":1,"594":1,"595":1,"596":1},"2":{"92":1,"384":1,"477":1,"494":2,"503":1,"567":2,"574":1,"604":4,"605":1,"733":1,"751":1}}],["options",{"0":{"249":1,"604":1,"753":1},"2":{"34":2,"35":1,"40":1,"70":1,"201":1,"232":1,"234":1,"240":1,"242":1,"270":1,"292":1,"382":6,"488":1,"489":1,"492":1,"493":1,"494":2,"502":2,"522":4,"595":2,"604":1,"684":1,"699":1,"717":1,"733":1,"753":1}}],["optionally",{"2":{"62":1,"64":2,"319":1,"585":1}}],["optional",{"0":{"198":1,"264":1,"265":1,"266":1,"267":1,"296":1,"320":1,"485":1,"486":1,"487":1,"530":1,"531":1,"587":1,"591":1,"599":1,"600":1,"604":1,"605":1,"633":1,"634":1,"635":1,"654":1,"658":1,"674":1,"675":1,"676":1,"746":1,"750":1,"753":1},"1":{"199":1,"200":1,"265":1,"267":1,"588":1,"589":1,"601":1,"602":1,"603":1,"634":1,"675":1,"747":1,"748":1,"749":1,"750":1},"2":{"23":1,"34":1,"39":1,"296":1,"319":10,"390":1,"391":2,"416":1,"522":1,"567":1,"585":1,"597":1,"756":2}}],["opportunities",{"2":{"42":1,"58":1}}],["opportunity",{"2":{"18":1}}],["operating",{"2":{"540":1,"563":1,"568":1,"612":1}}],["operations",{"2":{"374":1,"415":1,"459":1,"461":1,"484":1,"632":1}}],["operation",{"2":{"73":1}}],["operators",{"0":{"565":1},"2":{"284":1,"445":1,"548":1,"679":1,"716":1,"717":1}}],["operator",{"2":{"196":1,"200":1,"202":1,"203":3,"204":1,"501":1,"567":1,"604":1,"716":2}}],["operates",{"2":{"211":1}}],["operate",{"2":{"94":1,"413":1}}],["operated",{"2":{"23":1}}],["opened",{"2":{"645":1}}],["opening",{"2":{"563":2}}],["opentelemetry",{"2":{"547":1,"548":1}}],["openaikey",{"2":{"359":6}}],["openaimkdir",{"2":{"355":1}}],["openai",{"0":{"353":1},"2":{"350":2,"351":1,"353":5,"355":1,"356":4,"359":15,"360":1}}],["openrpcgo",{"2":{"269":1}}],["openrpc",{"2":{"81":1,"269":2,"270":6,"300":1,"355":2,"356":6,"381":6}}],["openssl",{"2":{"76":1}}],["open",{"0":{"420":1,"425":1,"429":1,"563":1,"644":1},"2":{"6":1,"22":1,"25":1,"27":3,"252":1,"285":1,"311":1,"420":1,"467":3,"470":1,"483":1,"495":2,"502":2,"503":2,"537":1,"541":3,"548":1,"550":1,"552":2,"563":9,"564":1,"614":1,"632":1,"638":1,"661":1,"663":1,"671":1,"708":1,"748":1}}],["opinions",{"2":{"7":1}}],["our",{"0":{"6":1,"7":1},"2":{"6":1,"7":2,"8":1,"9":1,"42":1,"46":2,"47":2,"49":1,"58":2,"319":1,"324":1,"325":1,"352":4,"356":2,"357":2,"358":5,"360":4,"475":1,"517":1,"529":2,"536":1,"569":1,"616":1,"623":1,"640":1,"656":1}}],["outbound",{"2":{"637":1}}],["outbox",{"2":{"85":1,"86":1}}],["outcome",{"2":{"415":1}}],["outsource",{"2":{"240":1}}],["outsourcing",{"2":{"240":1}}],["outside",{"2":{"181":1,"374":1,"403":1,"437":1}}],["output",{"2":{"263":1,"267":1,"319":3,"321":4,"328":2,"334":1,"340":1,"341":1,"352":1,"360":1,"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,"567":1,"568":2,"622":2,"623":1,"641":1,"644":2,"656":1,"662":2,"670":1,"745":2,"751":1}}],["outputs",{"2":{"199":1,"499":2}}],["outputinfo",{"2":{"61":1,"79":1}}],["outline",{"2":{"109":2,"115":1,"150":1}}],["outlined",{"2":{"22":1,"89":1,"90":1,"115":1,"168":1,"184":1}}],["out",{"0":{"59":1,"63":1},"1":{"60":1,"61":1,"62":1,"63":1,"64":2,"65":2},"2":{"0":1,"30":3,"37":1,"38":1,"39":1,"40":1,"59":2,"79":1,"94":1,"103":2,"106":1,"111":1,"154":1,"159":2,"164":1,"176":1,"179":1,"206":1,"238":1,"242":1,"276":1,"303":1,"314":1,"337":1,"377":1,"396":1,"484":1,"502":2,"503":1,"539":1,"548":1,"567":1,"623":1,"649":1,"677":1,"693":1,"699":2,"727":1,"731":1}}],["oracle",{"0":{"95":1},"2":{"93":1,"95":3,"105":1}}],["originate",{"2":{"385":1}}],["originally",{"2":{"373":1,"555":1}}],["original",{"2":{"66":1,"159":1,"206":1,"377":1,"407":1,"409":3,"410":1,"415":2,"416":1,"523":2}}],["origin",{"2":{"64":1,"387":1,"527":4,"528":1}}],["orientation",{"2":{"6":1}}],["orbitsetupscriptconfig",{"2":{"73":1,"74":1}}],["orbit",{"0":{"59":1,"62":1,"66":1,"71":1,"72":1},"1":{"60":1,"61":1,"62":1,"63":1,"64":1,"65":1,"67":1,"68":1,"69":1,"70":1,"71":1,"72":1,"73":1,"74":1,"75":1,"76":1,"77":1,"78":1,"79":1,"80":1,"81":1,"82":1,"83":1,"84":1,"85":1,"86":1},"2":{"59":2,"61":2,"66":5,"67":2,"68":1,"69":2,"70":1,"71":8,"72":7,"74":5,"76":1,"77":1,"78":28,"79":1,"81":4,"82":1,"88":3,"92":4,"93":1,"95":1}}],["ordered",{"2":{"410":1,"413":2,"554":2}}],["order",{"0":{"562":1},"2":{"35":1,"37":1,"75":1,"94":1,"95":2,"121":1,"124":1,"159":1,"160":1,"237":1,"315":1,"316":1,"317":1,"326":1,"327":1,"336":1,"346":1,"353":1,"371":1,"373":1,"408":1,"413":1,"436":1,"438":1,"465":1,"467":1,"485":1,"562":2,"616":1,"632":1,"633":1,"636":1,"658":1,"671":1,"674":1,"684":2,"705":1,"715":2,"733":1,"751":1}}],["ordering",{"2":{"35":1,"238":1,"414":1,"644":2}}],["organizing",{"2":{"33":1,"36":2,"41":2,"46":1,"50":1,"57":1}}],["organizers",{"2":{"34":1,"37":1,"38":1,"42":2,"43":1,"44":2,"46":1,"47":1,"49":1,"50":2,"58":1}}],["organizer",{"2":{"33":1,"37":1,"41":4,"50":1,"57":1,"58":1}}],["organize",{"2":{"29":2,"36":1,"37":1}}],["organization",{"2":{"26":1}}],["org",{"0":{"5":1,"46":1},"1":{"6":1,"7":1,"8":1,"9":1,"10":1,"11":1,"12":1,"13":1,"14":1,"15":1,"16":1},"2":{"6":1,"10":1,"16":3,"24":1,"40":1,"41":2,"46":1,"57":1,"71":1,"382":4,"540":2,"568":2,"623":8,"687":2,"688":2,"689":2,"706":1,"707":1,"708":1}}],["or",{"0":{"87":1,"318":1,"348":1,"530":1},"1":{"88":1,"89":1,"90":1},"2":{"2":1,"4":1,"6":2,"7":8,"8":2,"9":2,"10":1,"12":2,"13":2,"14":2,"15":2,"22":1,"23":7,"24":4,"26":3,"30":3,"31":4,"32":1,"34":4,"35":3,"36":3,"37":5,"38":4,"39":3,"40":2,"41":1,"42":2,"58":1,"69":1,"70":1,"71":3,"76":2,"78":5,"79":1,"81":1,"89":1,"92":1,"97":2,"107":1,"109":1,"111":1,"112":1,"115":2,"119":2,"121":1,"122":2,"124":1,"145":1,"150":1,"154":1,"159":1,"171":1,"188":2,"189":1,"190":1,"193":1,"196":3,"198":1,"200":1,"209":2,"210":1,"212":1,"238":2,"240":1,"241":1,"242":3,"244":1,"245":1,"246":3,"253":1,"254":5,"279":1,"302":2,"310":1,"311":2,"312":2,"314":1,"315":1,"318":2,"319":1,"336":2,"351":1,"370":1,"373":1,"374":1,"377":1,"382":4,"385":2,"386":1,"387":1,"394":2,"400":1,"404":1,"409":1,"413":1,"416":1,"423":1,"427":1,"432":1,"434":1,"438":2,"440":1,"446":2,"454":1,"459":1,"461":1,"467":1,"469":1,"470":3,"471":1,"477":1,"483":3,"488":4,"489":1,"500":1,"503":6,"505":3,"506":1,"509":2,"529":1,"530":2,"537":1,"547":2,"560":2,"564":4,"585":1,"592":1,"595":1,"601":1,"610":1,"612":1,"613":1,"614":2,"616":1,"632":3,"645":1,"649":1,"659":1,"662":1,"671":2,"672":1,"679":2,"683":1,"685":1,"692":1,"693":3,"695":1,"703":1,"704":1,"705":1,"707":1,"710":1,"714":1,"716":1,"720":2,"722":1,"735":1,"748":2}}],["often",{"2":{"34":1,"36":1,"37":1,"39":1,"373":1,"375":1,"714":1}}],["offloads",{"2":{"237":1}}],["offline",{"2":{"9":1,"509":1,"716":1}}],["off",{"2":{"204":1,"358":1,"503":1}}],["offchain",{"2":{"66":2,"78":1,"211":1,"214":1,"394":1}}],["offense",{"2":{"509":1}}],["offensive",{"2":{"8":1}}],["offers",{"2":{"51":1,"70":1,"206":1,"211":1,"223":1,"733":1}}],["offered",{"2":{"36":1}}],["offering",{"2":{"35":1,"92":1}}],["offer",{"2":{"34":4,"35":1,"37":1,"38":2,"211":1}}],["officially",{"2":{"9":1}}],["official",{"2":{"0":1,"9":2,"76":1}}],["of",{"0":{"5":1,"18":1,"46":1,"59":1,"122":1,"129":1,"157":1,"158":1,"159":1,"160":2,"161":1,"162":1,"164":1,"165":1,"166":1,"167":1,"172":1,"173":1,"174":1,"207":1,"210":1,"217":1,"239":1,"240":1,"325":1,"331":1,"397":1,"403":1,"405":1,"409":1,"415":1,"430":1,"440":1,"450":1,"452":1,"455":1,"477":1,"516":1,"525":1,"553":1,"625":1,"665":1},"1":{"6":1,"7":1,"8":1,"9":1,"10":1,"11":1,"12":1,"13":1,"14":1,"15":1,"16":1,"60":1,"61":1,"62":1,"63":1,"64":1,"65":1,"123":1,"124":1,"158":1,"159":1,"160":1,"161":1,"162":1,"165":1,"166":1,"167":1,"173":1,"174":1,"218":1,"219":1,"220":1,"240":1,"241":1,"242":1,"416":1,"451":1,"452":1,"453":2,"454":2,"455":2,"456":2,"457":2,"458":2,"459":2,"460":2,"554":1,"555":1,"556":1,"557":1,"558":1,"559":1},"2":{"0":1,"2":1,"6":2,"7":6,"8":2,"9":2,"10":3,"11":2,"12":3,"13":3,"14":4,"15":6,"16":4,"18":3,"20":1,"21":3,"22":5,"23":7,"24":1,"25":2,"26":3,"27":1,"30":2,"31":1,"32":1,"33":3,"34":3,"35":1,"36":8,"37":5,"38":6,"39":1,"40":2,"41":5,"42":4,"43":1,"44":1,"46":5,"47":2,"49":1,"50":2,"53":2,"54":1,"55":1,"56":1,"58":7,"59":3,"61":1,"62":3,"64":1,"65":1,"66":3,"69":2,"71":6,"72":3,"74":2,"75":1,"76":3,"77":1,"78":5,"79":3,"81":1,"89":1,"92":1,"93":2,"94":8,"95":2,"96":2,"97":2,"104":3,"107":5,"110":28,"111":10,"112":3,"113":2,"115":8,"119":2,"121":1,"122":4,"124":11,"125":2,"126":2,"127":1,"130":3,"134":1,"135":3,"136":4,"137":3,"138":2,"139":8,"140":1,"141":12,"142":2,"143":1,"144":10,"145":10,"146":4,"147":2,"148":10,"149":1,"150":1,"151":1,"153":2,"154":4,"155":2,"156":1,"157":9,"158":1,"159":10,"160":20,"161":2,"163":2,"164":6,"165":1,"167":2,"168":10,"170":2,"171":1,"172":9,"173":1,"176":4,"179":4,"181":1,"182":3,"184":6,"186":1,"187":12,"188":2,"189":13,"190":2,"191":2,"193":3,"195":6,"196":8,"199":2,"200":2,"201":1,"202":1,"203":1,"204":1,"206":6,"207":1,"209":2,"210":1,"211":3,"212":5,"213":1,"218":1,"222":2,"227":1,"231":2,"232":1,"237":1,"238":6,"240":7,"241":6,"242":3,"243":1,"245":2,"246":1,"251":1,"252":1,"254":3,"259":1,"262":2,"269":1,"270":3,"272":5,"273":2,"274":1,"277":1,"280":1,"282":2,"284":2,"286":1,"296":1,"298":3,"300":1,"307":1,"311":1,"313":1,"314":1,"315":2,"316":2,"318":1,"319":8,"321":3,"322":3,"324":1,"325":4,"328":2,"331":3,"334":1,"340":1,"348":2,"350":2,"352":1,"360":2,"363":1,"364":3,"365":3,"366":2,"367":1,"368":1,"369":2,"370":1,"372":2,"373":5,"374":10,"375":8,"376":3,"377":3,"382":4,"384":1,"385":2,"386":1,"387":1,"390":1,"391":1,"393":2,"394":4,"396":3,"397":3,"398":1,"399":1,"400":2,"402":2,"403":4,"404":2,"405":1,"406":2,"407":5,"408":5,"409":8,"410":7,"412":7,"413":11,"414":4,"415":15,"416":4,"417":1,"421":1,"422":1,"426":1,"430":1,"434":3,"435":2,"436":3,"437":6,"438":4,"440":5,"441":3,"443":2,"444":1,"445":4,"446":5,"447":2,"451":1,"453":1,"454":2,"455":3,"456":2,"465":1,"467":2,"470":1,"471":2,"472":1,"477":3,"483":3,"484":1,"485":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,"554":3,"555":1,"562":1,"563":1,"564":3,"567":2,"569":1,"573":2,"576":1,"577":1,"579":1,"585":2,"589":1,"591":1,"592":3,"593":1,"594":2,"595":1,"605":4,"612":1,"613":2,"614":1,"620":1,"623":2,"632":3,"633":7,"637":2,"638":2,"640":1,"656":3,"660":2,"661":2,"663":1,"665":1,"670":1,"671":2,"674":7,"678":3,"682":1,"683":2,"684":4,"693":2,"694":1,"699":5,"705":1,"710":3,"714":1,"715":3,"716":1,"717":1,"722":2,"731":1,"733":1,"738":1,"739":1,"745":2,"746":1,"748":2,"751":5,"754":1,"756":1}}],["mzonder",{"2":{"687":1,"688":1,"689":1}}],["m",{"2":{"347":2,"448":1,"697":1}}],["mb",{"2":{"308":1,"666":1,"719":1}}],["my",{"2":{"253":4,"288":1,"346":1,"352":2,"615":5,"616":8,"617":8,"672":6}}],["myrollup",{"2":{"103":2}}],["mnemonic",{"2":{"248":2,"352":2,"519":3,"536":1,"640":4}}],["mv",{"2":{"200":2}}],["mkdir",{"2":{"199":2,"200":2,"596":6,"615":2}}],["md013",{"2":{"328":2}}],["md",{"2":{"145":2}}],["mtgbzqhrqol61okkzmhrfyq5bk6goklgwrvparpyxve=",{"2":{"132":2}}],["msgcreatevestingaccount",{"2":{"522":2}}],["msgsubmitproposal",{"2":{"494":2}}],["msgsend",{"2":{"266":2}}],["msgpayforblobs",{"2":{"132":2,"152":1,"266":2,"306":2,"370":1,"375":2}}],["msg",{"2":{"132":2,"359":4,"415":1,"494":1,"639":4}}],["mutual",{"0":{"577":1}}],["mutually",{"2":{"36":1}}],["mut",{"2":{"365":2}}],["multiaddress",{"2":{"739":1}}],["multiaddresses",{"2":{"576":1}}],["multiaccounts",{"0":{"287":1},"1":{"288":1,"289":1,"290":1,"291":1,"292":1,"293":1,"294":1,"295":1,"296":1}}],["multi",{"2":{"508":1}}],["multiplier",{"2":{"639":4}}],["multiplies",{"2":{"375":1}}],["multiple",{"0":{"377":1},"2":{"37":1,"90":1,"107":1,"124":1,"135":1,"136":3,"139":8,"141":4,"157":2,"241":1,"307":1,"396":1,"399":1,"404":1,"407":1,"410":1,"413":2,"415":1,"474":1,"503":1,"554":1,"565":1,"696":1}}],["multiproof",{"2":{"141":2}}],["multisign",{"2":{"507":2}}],["multisig",{"0":{"506":1},"1":{"507":1,"508":1},"2":{"71":1,"220":1,"506":3,"507":18,"508":3}}],["mustnewv0",{"2":{"382":2}}],["must",{"2":{"71":2,"76":1,"269":1,"293":1,"346":2,"348":1,"363":1,"373":1,"377":3,"406":1,"408":1,"410":3,"413":2,"485":2,"494":1,"560":4,"590":1,"595":1,"611":1,"623":1,"632":1,"633":2,"653":1,"654":1,"655":1,"660":2,"671":1,"674":2,"683":1,"751":1}}],["much",{"2":{"35":1,"76":1,"238":1,"240":1,"241":1,"659":1}}],["milestone",{"2":{"699":1}}],["million",{"2":{"499":2}}],["middleware",{"2":{"648":1}}],["migrate",{"0":{"486":1,"634":1,"675":1},"2":{"486":1,"488":4,"634":1,"675":1}}],["might",{"2":{"34":3,"156":1,"238":1,"311":1,"314":1,"409":1,"494":1,"563":1,"585":1,"591":1,"693":1,"725":1}}],["microtia",{"0":{"460":1},"2":{"460":1}}],["micro",{"2":{"459":1}}],["microphone",{"2":{"34":2}}],["microphones",{"2":{"30":1,"34":1}}],["mib",{"2":{"371":3,"683":1}}],["mixhash",{"2":{"340":2}}],["mirrors",{"2":{"223":1}}],["miner",{"2":{"340":2}}],["mined",{"2":{"78":4}}],["mintscan",{"2":{"391":1,"696":1,"712":1}}],["mint",{"2":{"277":1}}],["minority",{"2":{"494":1}}],["minor",{"2":{"277":1}}],["mins",{"2":{"196":1}}],["minns",{"2":{"141":2,"148":2}}],["minnamespace",{"2":{"141":4,"148":4}}],["min",{"2":{"141":4,"500":2,"589":1,"592":4,"639":4,"751":3}}],["minimized",{"2":{"406":1}}],["minimal",{"2":{"103":1,"165":1,"277":1,"283":1,"567":1,"679":1}}],["minimum",{"0":{"581":1},"2":{"23":1,"71":1,"78":4,"102":1,"141":2,"237":1,"308":2,"373":1,"397":1,"478":2,"581":1,"626":2,"666":2,"742":1}}],["minutes",{"2":{"37":1,"94":1,"196":1,"199":1,"200":1,"202":1,"595":1}}],["minute",{"0":{"619":1},"2":{"35":1,"42":1,"377":1,"509":1}}],["mind",{"2":{"35":1,"246":2,"585":1,"751":1}}],["mindful",{"2":{"35":1}}],["misbehaviour",{"2":{"639":2}}],["mismatch",{"2":{"409":1}}],["misuse",{"2":{"76":1}}],["mission",{"2":{"18":1}}],["mistakes",{"2":{"7":1}}],["moonli",{"2":{"687":1,"688":1}}],["mock",{"2":{"347":1}}],["mocha4",{"2":{"705":1,"706":1,"708":1}}],["mocharesturl",{"2":{"389":2}}],["mocharpcurl",{"2":{"389":2}}],["mochamainnet",{"2":{"346":1}}],["mochachainid",{"2":{"317":3,"346":2,"389":2,"551":1,"585":8,"596":2,"745":2,"751":2}}],["mochacelestia",{"2":{"76":1,"310":1,"311":2,"312":1,"483":1,"484":1,"485":1,"493":1,"546":1,"555":1,"561":1,"631":1,"633":1,"670":1,"671":1,"672":1,"674":1}}],["mochadocker",{"2":{"252":1}}],["mochaversions",{"2":{"252":2,"539":2,"567":2,"614":2,"616":2,"617":2}}],["mocha",{"0":{"76":1,"347":1,"348":1,"449":1,"526":1,"529":1,"546":1,"557":1,"558":1,"699":1,"710":1,"727":1,"728":1},"1":{"527":1,"528":1,"529":1,"530":1,"531":1,"700":1,"701":1,"702":1,"703":1,"704":1,"705":1,"706":1,"707":1,"708":1,"709":1,"710":1,"711":1,"712":1,"713":1},"2":{"23":3,"26":1,"27":1,"66":1,"67":2,"76":7,"82":1,"94":1,"121":2,"216":3,"222":2,"246":3,"252":4,"253":8,"256":1,"257":3,"259":2,"261":2,"266":5,"267":2,"310":1,"311":5,"312":3,"344":1,"346":2,"347":1,"385":1,"389":6,"391":5,"449":3,"461":2,"462":2,"483":2,"484":6,"485":1,"493":5,"511":1,"517":1,"525":1,"526":2,"529":7,"531":4,"533":1,"537":1,"546":4,"551":2,"554":1,"555":7,"557":2,"558":2,"561":1,"564":1,"585":1,"614":1,"631":1,"632":3,"633":1,"637":1,"639":9,"640":2,"642":1,"643":6,"644":6,"645":3,"670":1,"671":4,"672":3,"673":1,"674":1,"699":7,"705":6,"706":8,"707":8,"708":13,"709":16,"710":2,"711":1,"712":2,"713":2,"715":1,"716":3,"727":4,"729":1,"733":2,"748":1,"751":1}}],["mounted",{"2":{"253":1}}],["mounting",{"0":{"253":1},"2":{"253":1,"615":1}}],["mount",{"2":{"252":1,"253":1,"346":1,"615":2,"616":1,"620":1}}],["moment",{"2":{"165":1,"168":1,"548":1}}],["moved",{"2":{"413":1,"660":1}}],["move",{"2":{"62":2,"64":1,"74":1,"312":1}}],["moniker",{"2":{"751":3}}],["moniker=$moniker",{"2":{"500":2,"751":2}}],["moniker=validator",{"2":{"500":2}}],["monitors",{"2":{"510":1}}],["monitor",{"0":{"510":1},"2":{"386":1,"387":1,"503":4,"510":1,"548":1}}],["monitoring",{"0":{"386":1},"2":{"23":1,"26":1,"501":1,"505":1,"622":1}}],["monolithic",{"0":{"413":1},"2":{"53":1,"238":2,"276":1,"284":1,"413":3}}],["monthly",{"2":{"36":1,"48":1}}],["month",{"2":{"23":1,"25":1}}],["months",{"2":{"21":1,"22":6,"23":1,"25":1,"26":1,"151":1,"153":1}}],["more",{"0":{"116":1},"1":{"117":1,"118":1,"119":1},"2":{"20":1,"24":1,"34":1,"35":1,"36":1,"38":1,"64":1,"66":3,"71":3,"72":1,"77":1,"79":1,"92":1,"95":1,"97":1,"110":2,"111":2,"112":2,"121":1,"122":2,"124":1,"127":1,"132":1,"137":1,"148":2,"153":1,"154":1,"159":1,"160":2,"167":1,"175":1,"179":1,"181":1,"184":2,"195":1,"197":1,"211":2,"212":1,"215":1,"218":1,"219":2,"220":1,"222":1,"223":1,"238":1,"241":3,"242":1,"246":1,"252":1,"258":1,"278":1,"284":1,"295":1,"302":1,"303":1,"306":1,"307":2,"312":1,"314":2,"321":1,"322":1,"348":1,"349":1,"359":1,"361":1,"370":1,"371":1,"375":1,"376":1,"380":1,"381":1,"383":1,"393":1,"396":1,"397":1,"407":1,"408":3,"410":1,"412":3,"414":1,"416":1,"443":1,"456":1,"475":1,"484":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,"563":1,"567":1,"568":1,"595":1,"612":1,"632":1,"694":1,"695":1,"698":1,"713":1,"717":1,"729":1,"745":2}}],["modifying",{"2":{"661":1}}],["modify",{"0":{"661":1},"2":{"314":1,"348":1,"563":1,"661":1}}],["modifiable",{"2":{"494":1}}],["modifications",{"2":{"277":1}}],["modified",{"2":{"222":1,"282":1,"410":1,"412":1,"477":1}}],["module=consensus",{"2":{"607":2,"756":2}}],["moduleencodingregisters",{"2":{"382":2}}],["modules",{"2":{"277":2,"355":1,"374":1}}],["module",{"0":{"255":1,"260":1},"1":{"256":1,"257":1,"258":1,"259":1,"260":1,"261":1,"262":1,"263":1,"264":1,"265":1,"266":1,"267":1},"2":{"256":1,"258":1,"267":1,"296":2,"314":5,"317":2,"333":2,"376":1,"389":2,"494":4,"509":1,"647":1,"648":1}}],["modularity",{"0":{"53":1},"2":{"42":1,"53":1}}],["modular",{"0":{"29":1,"42":1,"43":1,"51":1,"56":1,"205":1,"238":1,"239":1,"413":1},"1":{"30":1,"31":1,"32":1,"33":1,"34":1,"35":1,"36":1,"37":1,"38":1,"39":1,"40":1,"41":1,"42":1,"44":1,"45":1,"46":1,"47":1,"48":1,"49":1,"50":1,"52":1,"53":1,"54":1,"55":1,"56":1,"57":1,"206":1,"207":1,"208":1,"209":1,"210":1,"211":1,"212":1,"213":1,"214":1,"215":1,"216":1,"240":1,"241":1,"242":1},"2":{"18":1,"29":2,"33":1,"34":1,"35":2,"36":3,"37":4,"39":1,"40":1,"41":2,"42":2,"43":2,"44":2,"47":3,"48":1,"50":2,"51":2,"53":4,"56":3,"57":1,"58":9,"212":1,"232":2,"238":4,"240":2,"241":2,"242":1,"276":2,"283":1,"360":2,"393":1,"413":3,"414":2,"453":1,"454":1,"696":1}}],["mod",{"2":{"125":3,"355":2,"515":2,"607":14,"637":1,"756":14}}],["modern",{"2":{"708":1}}],["moderation",{"2":{"8":1}}],["mode=",{"2":{"507":2}}],["model",{"2":{"359":2,"375":1,"456":1}}],["mode",{"2":{"61":2,"196":2,"266":2,"454":1,"507":4,"531":4,"593":1,"639":12}}],["mozilla",{"2":{"16":1}}],["mostly",{"2":{"632":1}}],["most",{"2":{"0":1,"37":1,"42":5,"44":1,"75":1,"184":1,"268":1,"280":1,"362":1,"395":1,"403":1,"573":1,"594":1,"612":1,"665":1}}],["mesa",{"2":{"687":1,"688":1,"689":1,"693":1,"705":1,"706":1,"708":1}}],["messages",{"2":{"30":1,"32":1,"175":1,"266":2,"277":1,"350":1,"359":2,"494":2,"522":2}}],["message",{"2":{"2":1,"41":3,"94":1,"97":1,"132":2,"321":1,"359":4,"360":2,"361":1,"375":1,"415":2,"532":1,"537":1}}],["me",{"2":{"602":2,"662":2,"687":1,"688":1,"706":1,"707":1,"708":1}}],["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":{"501":3,"502":8,"503":3,"504":2,"543":2,"544":12,"545":6,"546":6,"547":15,"548":2,"638":1}}],["met",{"2":{"373":1}}],["metadata",{"2":{"145":2,"241":1,"416":1,"665":1}}],["metamask",{"2":{"67":1}}],["methods",{"2":{"269":2,"274":1,"314":1,"363":2,"367":1,"383":1,"438":1}}],["method",{"2":{"105":1,"107":2,"138":2,"142":1,"151":1,"159":1,"195":1,"218":1,"270":2,"271":2,"272":2,"273":2,"281":1,"284":1,"314":5,"317":2,"333":3,"334":1,"336":2,"342":2,"364":2,"365":2,"366":2,"389":2,"438":3}}],["mechanisms",{"2":{"72":1,"97":1,"115":1,"374":1,"375":1}}],["mechanism",{"0":{"97":1,"210":1},"2":{"72":1,"93":1,"97":4,"149":1,"184":1,"212":1,"254":2,"395":1,"396":1,"406":1,"438":1,"440":1,"447":1,"505":1,"509":1,"594":1}}],["meant",{"2":{"302":1,"466":1,"685":1,"703":1}}],["meaningful",{"2":{"282":1}}],["meaning",{"2":{"153":2,"282":1,"477":1,"596":1}}],["mean",{"2":{"111":2}}],["means",{"2":{"21":1,"124":1,"184":1,"190":1,"241":2,"277":1,"296":1,"310":1,"371":1,"373":1,"377":1,"408":2,"413":2,"417":1,"435":1,"437":1,"564":1,"601":2,"611":1,"612":1}}],["measured",{"2":{"71":1}}],["menu",{"2":{"61":1,"333":2,"344":1,"391":1,"529":1,"539":1,"540":1,"568":1}}],["mentioned",{"2":{"89":1,"705":1}}],["mention",{"2":{"38":2}}],["memo",{"2":{"522":2,"639":4}}],["memory",{"2":{"112":1,"148":2,"308":1,"478":1,"504":1,"581":1,"626":1,"666":1,"719":1,"720":1,"742":1}}],["memorable",{"2":{"58":1,"71":1}}],["mempool",{"0":{"372":1},"1":{"373":1,"374":1,"375":1,"376":1,"377":1},"2":{"254":1,"371":1,"372":2,"373":2,"377":2,"382":2,"435":1}}],["member",{"2":{"41":1}}],["members",{"0":{"42":1},"2":{"6":1,"22":1,"30":1,"39":3,"41":2,"42":2,"210":1,"377":1,"445":1,"495":1}}],["merge",{"2":{"560":1}}],["merkelized",{"2":{"434":1}}],["merkelizes",{"2":{"212":1}}],["merkleized",{"2":{"104":1,"124":1}}],["merkle",{"0":{"179":1,"180":1,"181":1,"410":1},"1":{"180":1,"181":1},"2":{"95":1,"104":4,"105":1,"107":1,"112":1,"124":3,"126":2,"129":4,"130":3,"136":1,"139":2,"141":4,"144":4,"145":4,"148":8,"153":6,"159":2,"160":5,"173":1,"175":1,"181":1,"210":1,"212":2,"307":1,"406":1,"407":6,"408":1,"409":2,"410":2,"412":2,"415":4,"416":3}}],["merchandise",{"2":{"57":2}}],["merely",{"2":{"26":1}}],["mev",{"2":{"42":1}}],["meetups",{"2":{"32":1,"34":4,"36":5,"37":1,"38":1,"39":1,"40":4,"41":6,"42":2,"43":1,"51":3,"57":2,"58":1}}],["meetup",{"0":{"29":1,"30":1,"31":1,"32":1,"41":1,"42":1,"43":1,"51":1,"56":1},"1":{"30":1,"31":1,"32":1,"33":1,"34":1,"35":1,"36":1,"37":1,"38":1,"39":1,"40":1,"41":1,"42":1,"44":1,"45":1,"46":1,"47":1,"48":1,"49":1,"50":1,"52":1,"53":1,"54":1,"55":1,"56":1,"57":1},"2":{"29":2,"30":2,"32":1,"33":3,"34":4,"35":2,"36":7,"37":9,"38":5,"39":8,"40":4,"41":10,"42":5,"43":1,"44":4,"46":1,"47":2,"48":1,"50":4,"56":2,"57":1,"58":6}}],["meet",{"2":{"21":1}}],["media",{"2":{"3":1,"9":1,"13":1,"30":1,"32":1,"39":4,"40":1}}],["mac",{"2":{"613":1,"615":1,"623":2}}],["macos",{"2":{"347":2}}],["machines",{"2":{"413":1,"503":1,"601":1}}],["machine",{"2":{"2":1,"67":1,"196":2,"199":1,"200":1,"202":1,"223":1,"251":1,"252":1,"310":1,"311":1,"352":1,"373":1,"375":1,"412":3,"413":1,"415":2,"477":1,"479":1,"483":1,"503":1,"517":2,"537":1,"541":1,"550":1,"552":1,"582":1,"612":1,"614":1,"615":3,"627":1,"632":1,"647":1,"663":1,"668":1,"671":1,"743":1,"748":1}}],["magnitude",{"2":{"282":1}}],["majority",{"2":{"409":1}}],["major",{"2":{"160":1,"412":1}}],["master",{"2":{"145":2,"391":1,"585":10}}],["malicious",{"2":{"115":1,"193":2,"409":1,"509":1}}],["mapping",{"2":{"95":1,"141":2,"144":1,"369":1}}],["matters",{"2":{"159":1}}],["math",{"2":{"129":2,"148":2}}],["matching",{"2":{"369":1}}],["match",{"2":{"110":1,"125":1,"365":4}}],["matches",{"2":{"76":1,"184":1,"253":1}}],["matrix",{"0":{"81":1},"2":{"76":1,"407":4,"409":1,"415":5,"416":1}}],["materials",{"0":{"50":1},"2":{"0":1,"3":1,"30":2,"31":1,"51":1}}],["made",{"2":{"66":1,"96":1,"212":1,"242":1,"297":1,"397":1,"412":1,"415":2,"416":1,"434":2,"438":1,"517":1,"525":1,"554":1}}],["manually",{"2":{"314":1,"377":1,"382":2,"386":1,"438":1,"554":1,"576":1,"579":1,"585":2}}],["manual",{"2":{"296":1,"438":1}}],["managing",{"2":{"258":1}}],["manager",{"2":{"503":1}}],["management",{"0":{"278":1,"293":1,"490":1,"493":1},"1":{"294":1,"295":1,"491":1},"2":{"278":1,"612":1}}],["manages",{"2":{"187":1}}],["managed",{"2":{"153":1,"284":1,"437":1}}],["manage",{"0":{"423":1,"427":1,"432":1},"2":{"30":1,"36":1,"244":1,"382":2,"488":2,"605":1}}],["many",{"0":{"563":1},"2":{"39":1,"110":1,"119":1,"241":1,"278":1,"374":1,"376":1,"413":1,"441":1,"444":1,"462":1,"503":1,"563":2,"622":1,"717":1}}],["manner",{"2":{"24":1,"620":1}}],["maxvalidators",{"2":{"494":4}}],["maximize",{"2":{"242":1}}],["maximum",{"0":{"371":1,"683":1},"2":{"141":2,"314":1,"371":2,"375":1,"502":2,"683":8}}],["maxns",{"2":{"141":2,"148":2}}],["maxnamespace",{"2":{"141":4,"148":4}}],["max",{"2":{"26":1,"141":4,"218":1,"494":3,"500":4,"502":2,"639":20,"682":1,"751":4}}],["marking",{"2":{"437":1}}],["market",{"0":{"372":1,"435":1},"1":{"373":1,"374":1,"375":1,"376":1,"377":1}}],["marketing",{"0":{"39":1},"2":{"39":1}}],["markdownlint",{"2":{"328":2}}],["marks",{"2":{"92":1,"678":1,"722":1}}],["mark",{"2":{"26":1}}],["may",{"0":{"404":1},"2":{"10":1,"12":1,"13":1,"14":1,"22":1,"23":1,"34":2,"38":2,"39":3,"70":1,"72":1,"76":1,"149":1,"356":1,"374":1,"385":1,"386":1,"404":1,"416":2,"437":1,"438":1,"444":1,"446":1,"454":1,"530":1,"541":1,"552":1,"563":1,"595":1,"604":1,"610":1,"615":1,"679":1,"722":1}}],["mail",{"2":{"9":1}}],["maintenance",{"2":{"445":1}}],["maintained",{"2":{"213":1}}],["maintaining",{"2":{"46":1,"54":1,"71":1,"206":1,"377":1}}],["maintains",{"2":{"22":1}}],["maintain",{"2":{"18":1,"22":1,"23":2,"27":1,"32":1,"36":1,"37":1,"39":1,"220":1,"554":1}}],["mainly",{"2":{"200":2}}],["mainnetversions",{"2":{"539":2,"567":2,"614":2,"616":2,"617":2}}],["mainnetresturl",{"2":{"389":2}}],["mainnetrpcurl",{"2":{"389":2}}],["mainnetchainid",{"2":{"389":2,"551":2,"585":8,"596":2}}],["mainnet",{"0":{"448":1,"545":1,"557":1,"678":1,"722":1,"723":1},"1":{"679":1,"680":1,"681":1,"682":1,"683":1,"684":1,"685":1,"686":1,"687":1,"688":1,"689":1,"690":1,"691":1,"692":1,"693":1,"694":1,"695":1,"696":1,"697":1,"698":1,"723":1},"2":{"23":4,"24":1,"26":4,"27":1,"97":1,"121":1,"213":2,"216":4,"246":3,"252":1,"253":1,"254":1,"310":1,"311":3,"312":2,"336":1,"371":1,"385":1,"389":5,"448":1,"471":2,"483":1,"484":1,"485":1,"539":1,"545":1,"550":1,"551":3,"554":1,"557":3,"564":1,"567":1,"585":5,"593":1,"596":1,"597":1,"614":3,"616":1,"617":1,"623":1,"631":1,"632":2,"633":1,"670":1,"671":2,"672":1,"674":1,"678":3,"679":2,"684":2,"687":3,"688":3,"689":3,"690":2,"693":4,"696":4,"697":3,"698":2,"710":2,"715":1,"716":3,"722":2,"729":1,"748":1,"751":1}}],["main",{"0":{"357":1},"2":{"2":1,"79":1,"121":4,"128":4,"129":4,"148":4,"199":16,"200":2,"314":1,"356":6,"357":5,"358":5,"360":9,"412":1,"508":1}}],["making",{"2":{"2":1,"41":1,"42":1,"78":1,"212":1,"223":1,"315":1,"413":1,"414":1,"557":1,"558":1,"601":1,"602":1,"603":1,"604":1,"605":1}}],["makeconfig",{"2":{"382":2}}],["makes",{"0":{"397":1},"2":{"172":1,"222":1,"372":1}}],["make",{"2":{"0":1,"2":1,"6":1,"44":1,"75":1,"76":1,"94":1,"101":1,"102":1,"106":1,"125":3,"129":2,"141":5,"143":2,"144":4,"145":2,"147":2,"148":12,"153":1,"168":1,"187":3,"189":1,"196":3,"199":1,"200":2,"212":1,"240":1,"241":2,"245":5,"246":1,"352":1,"407":1,"415":1,"438":1,"501":1,"502":2,"511":1,"522":1,"533":1,"541":3,"552":3,"572":1,"601":1,"622":4,"720":1,"736":1}}],["ccv",{"2":{"639":4}}],["c",{"2":{"540":2,"555":1,"568":2,"596":6,"622":2,"623":8,"636":1}}],["cycle",{"2":{"434":1}}],["css",{"2":{"389":2}}],["c4ibesqxgzc5zxj3vtugbiadqamienbyjkegcac=",{"2":{"262":2}}],["cgo",{"2":{"199":2,"567":1}}],["cpu",{"2":{"308":1,"478":1,"581":1,"626":1,"666":1,"719":1,"720":1,"742":1}}],["cp",{"2":{"188":2,"195":2,"196":2,"200":2}}],["cd",{"2":{"187":2,"188":2,"193":2,"196":2,"199":4,"200":2,"245":2,"355":2,"369":2,"539":2,"567":3,"596":3,"623":4}}],["cips",{"2":{"716":1}}],["cip",{"2":{"437":1,"682":2,"716":1}}],["circulation",{"2":{"446":1}}],["circulating",{"2":{"446":4}}],["circuits",{"0":{"200":1},"2":{"175":1,"188":2,"189":3,"196":2,"198":1,"199":2,"200":5,"218":3}}],["circuit",{"0":{"176":1},"2":{"154":1,"172":1,"173":1,"176":3,"178":1,"179":4,"181":1,"187":1,"189":1,"195":3,"196":2,"199":13,"200":23,"219":1}}],["ciruit",{"2":{"188":1}}],["city",{"2":{"36":1,"38":1,"39":1,"466":1,"685":1,"703":1}}],["cmd",{"2":{"133":2}}],["cqebcp4bciavy2vszxn0aweuymxvyi52ms5nc2dqyxlgb3jcbg9icxj6ci9jzwxlc3rpytf2ogvzy3u3znu5bhy4nzlrenu1dwvndwv1cnrxnxn3nmhzbtnuzridaaaaaaaaaaaaaaaaaaaaaaaaaaaabytlu4hlouuaa8cpbyigsvxwya9toi+aytu3jja2wki5zlkm72",{"2":{"132":2}}],["ctx",{"2":{"128":2,"129":10,"138":2,"146":4,"147":6,"148":4,"270":8,"271":8,"272":8,"273":8,"357":2,"358":9,"359":2,"360":8,"381":8}}],["cfg",{"2":{"75":1,"76":2,"89":1}}],["cherry",{"2":{"505":1}}],["cheap",{"2":{"199":1}}],["cheaper",{"2":{"197":1}}],["checking",{"0":{"263":1,"267":1,"416":1},"2":{"276":1,"410":1}}],["checkout",{"2":{"148":1,"200":2,"539":6,"567":6}}],["checks",{"2":{"103":2,"107":3,"357":1}}],["check",{"0":{"324":1,"325":1,"516":1},"2":{"31":1,"64":1,"97":1,"103":1,"106":1,"107":1,"111":1,"115":2,"124":1,"127":1,"132":1,"137":1,"145":1,"148":5,"159":2,"179":1,"188":1,"197":1,"200":1,"210":1,"220":2,"237":1,"252":1,"253":1,"262":1,"263":1,"303":1,"315":1,"337":1,"340":1,"352":1,"358":1,"382":4,"396":1,"406":1,"410":1,"502":3,"516":1,"522":1,"523":1,"524":1,"525":1,"536":1,"537":1,"539":2,"547":1,"548":1,"560":1,"567":2,"637":1,"641":4,"677":1,"693":1,"736":3,"740":2,"745":1}}],["checklist",{"2":{"29":1}}],["chown",{"2":{"615":2}}],["chose",{"2":{"462":1}}],["chosen",{"2":{"37":1,"72":1,"604":2}}],["choice",{"2":{"421":1,"426":1,"751":2}}],["choices",{"2":{"359":2}}],["choosing",{"2":{"193":1,"656":1}}],["chooses",{"2":{"407":1}}],["choose",{"0":{"64":1,"234":1,"430":1},"2":{"62":1,"64":1,"71":1,"151":1,"188":1,"426":1,"430":1,"438":1,"503":1,"522":1,"530":1,"540":2,"547":1,"555":1,"568":2,"604":1,"655":1}}],["chart",{"2":{"445":1}}],["charge",{"2":{"440":1}}],["charged",{"2":{"375":1}}],["characters",{"2":{"26":1}}],["characteristics",{"2":{"6":1}}],["chatmessageroleuser",{"2":{"359":2}}],["chatcompletion",{"2":{"359":2}}],["chatcompletionmessage",{"2":{"359":2}}],["chatcompletionrequest",{"2":{"359":2}}],["chatgpt",{"0":{"359":1},"2":{"356":2,"357":2,"358":3,"360":1}}],["chainode",{"2":{"687":1,"688":1,"689":1,"690":1}}],["chain=celestia",{"2":{"448":1}}],["chainname",{"2":{"389":12}}],["chainid",{"2":{"187":2,"389":16,"644":8}}],["chainlist",{"2":{"71":1}}],["chain",{"0":{"61":1,"71":1,"72":1,"73":2,"77":1,"78":1,"110":1,"240":1,"390":1,"391":1,"551":1,"659":1},"2":{"61":2,"64":2,"66":1,"69":1,"70":1,"71":26,"72":12,"73":7,"75":2,"76":3,"77":1,"78":56,"79":2,"92":1,"94":6,"95":1,"97":1,"105":1,"110":3,"111":1,"121":1,"156":1,"157":1,"165":1,"185":1,"186":2,"187":12,"189":1,"195":2,"196":5,"199":1,"200":1,"203":2,"204":2,"206":1,"212":2,"219":1,"238":1,"246":1,"252":1,"256":1,"266":2,"277":1,"279":1,"281":1,"282":1,"312":1,"328":2,"347":1,"348":8,"368":1,"369":2,"389":2,"390":5,"391":4,"393":2,"394":1,"414":1,"436":1,"454":1,"470":1,"484":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,"564":2,"585":11,"595":1,"632":1,"637":3,"638":1,"639":4,"640":6,"643":12,"644":22,"647":1,"648":1,"655":4,"659":3,"693":1,"694":1,"705":1,"716":2,"745":2,"751":2,"756":1}}],["chains",{"0":{"642":1},"2":{"26":1,"61":1,"66":1,"67":2,"71":3,"78":1,"82":2,"88":1,"92":1,"115":1,"150":1,"187":1,"213":1,"216":1,"282":1,"284":1,"348":1,"391":1,"440":1,"444":1,"448":1,"449":1,"466":1,"637":1,"638":1,"639":30,"640":1,"641":2,"642":1,"645":4,"685":1,"703":1}}],["challengemanager",{"2":{"85":1,"86":1}}],["challenge",{"2":{"71":1,"72":1,"96":1,"361":1}}],["challenges",{"2":{"42":1}}],["challenging",{"2":{"38":1}}],["changing",{"0":{"553":1},"1":{"554":1,"555":1,"556":1,"557":1,"558":1,"559":1},"2":{"242":1,"592":1}}],["changeable",{"2":{"443":2}}],["change",{"0":{"531":1},"2":{"187":2,"196":2,"277":1,"279":1,"282":1,"289":1,"346":1,"371":1,"374":1,"437":1,"443":1,"494":6,"500":2,"503":1,"514":1,"539":1,"555":1,"567":1,"577":1,"585":1,"716":1,"751":2}}],["changed",{"2":{"78":2,"314":1,"346":1}}],["changes",{"2":{"2":4,"18":1,"37":1,"39":1,"81":2,"377":1,"386":1,"412":2,"462":2,"477":1,"494":4,"603":1,"604":1,"605":1,"632":1,"679":1,"725":1}}],["chan",{"2":{"147":2,"271":2,"272":2}}],["channelid",{"2":{"644":4}}],["channel",{"2":{"34":1,"47":2,"48":1,"76":4,"271":3,"272":3,"472":1,"475":1,"537":1,"639":4,"644":13,"645":4,"698":1,"710":1,"713":1,"729":1}}],["channelside",{"2":{"644":4}}],["channels",{"0":{"645":1},"2":{"13":1,"30":1,"40":2,"42":1,"312":1,"639":2,"642":1,"645":2,"716":1}}],["chances",{"2":{"38":1}}],["chance",{"2":{"26":1,"595":1}}],["cumulo",{"2":{"687":1,"688":1,"689":1,"706":1,"707":1,"708":1}}],["culmination",{"2":{"678":1,"722":1}}],["custody",{"0":{"278":1}}],["custodians",{"2":{"275":1}}],["custodian",{"2":{"23":1}}],["custom=",{"2":{"542":4}}],["custom=robusta",{"2":{"318":2}}],["custom=private",{"2":{"318":2}}],["customtargetblock",{"2":{"218":3}}],["customizations",{"2":{"242":1}}],["customization",{"2":{"242":1}}],["customizability",{"0":{"242":1},"2":{"242":1}}],["customizing",{"2":{"71":1}}],["customized",{"2":{"242":1}}],["customize",{"2":{"41":2,"71":1,"246":1}}],["custom",{"0":{"61":1,"192":1,"318":1,"390":1,"391":1,"485":1,"542":1,"559":1,"633":1,"674":1},"1":{"634":1,"675":1},"2":{"41":3,"61":1,"67":1,"69":1,"71":1,"76":1,"190":1,"191":3,"192":3,"218":1,"242":1,"246":1,"317":3,"318":9,"369":1,"388":1,"390":1,"391":3,"485":9,"542":3,"554":1,"559":2,"560":2,"564":1,"614":1,"633":9,"674":9}}],["currency",{"2":{"391":1,"454":1}}],["currencies",{"2":{"389":2}}],["current",{"2":{"37":1,"82":1,"200":1,"212":2,"245":2,"374":1,"402":1,"490":2,"683":1}}],["currently",{"2":{"22":1,"42":1,"134":1,"151":1,"200":1,"216":1,"234":1,"254":1,"262":1,"279":1,"280":1,"284":1,"302":1,"373":1,"378":1,"448":1,"449":1,"562":1,"637":1,"638":1,"724":1}}],["curl",{"0":{"320":1,"336":1},"2":{"320":1,"336":1,"540":2,"568":2,"585":10,"596":6,"622":6,"662":2}}],["curated",{"2":{"37":1,"58":1}}],["cups",{"2":{"35":1}}],["ce1e5714",{"2":{"360":2,"361":1}}],["cel",{"0":{"244":1,"249":1,"250":1},"1":{"245":1,"246":1,"247":1,"248":1,"249":1,"251":1,"252":1,"253":1},"2":{"243":2,"244":1,"245":5,"246":7,"247":6,"248":6,"249":2,"252":3,"253":3,"258":1,"293":1,"294":2,"295":2,"312":4,"484":3,"554":1,"555":3,"567":3,"612":1,"632":3,"672":5}}],["celesvaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u43cv6hd",{"2":{"745":1}}],["celesvaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u43cv6hdenter",{"2":{"745":1}}],["celesvaloper",{"2":{"745":2}}],["celes",{"2":{"253":4,"288":1,"346":1,"352":2,"672":6}}],["celestiavaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u4q4gx4p",{"2":{"745":2}}],["celestiavaloper",{"2":{"745":1}}],["celestiacelestia",{"2":{"671":1}}],["celestiascan",{"2":{"474":1,"712":1}}],["celestiatestnet",{"2":{"391":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":{"325":1}}],["celestia10rtd9lhel2cuh6c659l25yncl6atcyt37umardcelestia",{"2":{"325":1}}],["celestia1czpgn3hdh9sodm06d5qk23xzgpq2uyc8ggdqgw",{"2":{"493":2}}],["celestia1hn25k7gkfq0fy5a0vmphs6mjma2de74gsn36ef",{"2":{"352":2}}],["celestia1krkle0n547u0znz3unnln8paft2dq4z3rznv86",{"2":{"338":2}}],["celestia1znk24rh52pgcd9z5x2x42jztjh6raaaphuvrt3",{"2":{"327":2}}],["celestia1",{"2":{"312":1,"472":1,"673":1,"710":1}}],["celestia1wkhyhr7ngf0ayqlpnsnxg4d72hfs5453dvunm9",{"2":{"253":2}}],["celestia1vdjkcetnw35kzvtk8pjhxcm4xan82wtvwcurwwtt0f6n2at9va6k2atjw3cn2umhxe58xmfndejs40vqs9",{"2":{"132":2}}],["celestialightnodeclient",{"2":{"111":6,"113":2,"115":2}}],["celestiaheight",{"2":{"110":4}}],["celestiaorg",{"2":{"74":2,"97":1,"102":2,"125":6,"129":2,"135":4,"141":2,"142":2,"145":2,"148":4,"222":1,"245":2,"252":2,"269":2,"270":6,"285":1,"339":1,"355":2,"356":6,"376":2,"381":6,"382":12,"386":1,"539":2,"567":2,"585":9,"607":14,"614":6,"616":6,"617":6,"756":14}}],["celestia",{"0":{"0":1,"5":1,"17":1,"43":1,"46":1,"52":1,"54":1,"79":1,"91":1,"118":1,"123":1,"129":1,"184":1,"222":1,"233":1,"243":1,"259":1,"260":1,"262":1,"266":1,"275":1,"277":1,"304":1,"310":1,"342":1,"344":1,"352":1,"370":1,"379":1,"380":1,"381":1,"383":1,"388":1,"389":1,"395":1,"397":1,"403":1,"406":1,"415":1,"421":1,"430":1,"437":1,"440":1,"447":1,"476":1,"481":1,"482":1,"509":1,"511":1,"534":1,"538":1,"543":1,"566":1,"584":1,"624":1,"629":1,"649":1,"652":1,"653":1,"664":1,"669":1,"714":1,"717":1,"718":1,"721":1,"731":1,"732":1,"736":1,"738":1,"739":1,"740":1,"741":1,"746":1,"747":1},"1":{"1":1,"2":1,"3":1,"4":1,"6":1,"7":1,"8":1,"9":1,"10":1,"11":1,"12":1,"13":1,"14":1,"15":1,"16":1,"18":1,"19":1,"20":1,"21":1,"22":1,"23":1,"24":1,"25":1,"26":1,"27":1,"28":1,"44":1,"45":1,"46":1,"47":1,"48":1,"49":1,"50":1,"92":1,"93":1,"94":1,"95":1,"96":1,"97":1,"98":1,"234":1,"235":1,"236":1,"244":1,"245":1,"246":1,"247":1,"248":1,"249":1,"250":1,"251":1,"252":1,"253":1,"260":1,"261":1,"267":1,"276":1,"277":1,"278":2,"279":2,"280":2,"281":2,"282":2,"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,"338":1,"345":1,"346":1,"347":1,"348":1,"349":1,"371":1,"372":1,"373":1,"374":1,"375":1,"376":1,"377":1,"378":1,"379":1,"380":1,"381":1,"382":1,"383":1,"384":1,"389":1,"390":1,"391":1,"407":1,"408":1,"409":1,"410":1,"411":1,"412":1,"416":1,"448":1,"449":1,"477":1,"478":1,"479":1,"480":1,"481":1,"482":2,"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,"567":1,"568":1,"569":1,"570":1,"625":1,"626":1,"627":1,"628":1,"629":1,"630":2,"631":2,"632":2,"633":2,"634":2,"635":2,"636":2,"650":1,"651":1,"652":1,"653":1,"654":2,"655":2,"656":2,"657":2,"658":2,"659":2,"660":2,"661":2,"662":2,"663":2,"665":1,"666":1,"667":1,"668":1,"669":1,"670":1,"671":1,"672":1,"673":1,"674":1,"675":1,"676":1,"677":1,"715":1,"716":1,"718":1,"719":1,"720":1,"722":1,"723":1,"724":1,"725":1,"726":1,"727":1,"728":1,"729":1,"742":1,"743":1,"744":1,"745":1,"746":1,"747":2,"748":2,"749":2,"750":2,"751":1,"752":1,"753":1,"754":1,"755":1,"756":1},"2":{"0":2,"3":3,"6":1,"7":1,"10":1,"18":3,"20":3,"21":1,"23":3,"24":2,"26":5,"34":2,"35":2,"36":1,"37":4,"38":2,"39":2,"40":6,"41":15,"42":5,"43":1,"44":1,"46":2,"48":1,"50":1,"51":2,"53":2,"54":4,"55":2,"56":1,"57":2,"58":4,"66":3,"67":1,"70":5,"75":1,"76":12,"81":5,"88":1,"89":4,"92":2,"93":1,"94":12,"95":6,"96":4,"97":3,"98":1,"103":1,"104":1,"105":1,"106":2,"107":6,"109":4,"110":20,"111":11,"112":4,"113":3,"115":4,"118":2,"121":2,"122":4,"124":4,"125":3,"129":2,"132":4,"135":2,"137":1,"138":1,"139":2,"140":1,"141":4,"142":5,"145":8,"148":6,"149":2,"150":2,"152":1,"153":5,"156":1,"157":4,"159":3,"160":2,"163":2,"164":1,"165":1,"168":2,"169":1,"171":1,"172":2,"176":1,"179":1,"184":3,"194":3,"196":1,"201":2,"203":1,"206":5,"209":2,"210":2,"211":1,"212":12,"216":1,"218":2,"221":2,"222":3,"223":1,"231":1,"232":1,"234":1,"237":1,"241":1,"242":1,"243":1,"244":1,"245":5,"246":3,"251":1,"252":11,"253":17,"254":5,"256":1,"257":3,"258":2,"259":2,"260":1,"266":5,"267":2,"268":1,"269":3,"270":7,"275":1,"276":4,"277":2,"278":1,"279":3,"282":2,"283":2,"284":7,"286":1,"288":1,"289":3,"295":1,"297":2,"298":1,"301":1,"304":1,"306":1,"307":1,"309":1,"310":2,"311":5,"312":7,"313":2,"314":5,"315":1,"316":4,"317":2,"318":13,"319":4,"321":2,"324":1,"325":1,"328":2,"329":2,"331":1,"334":4,"336":6,"337":1,"339":1,"344":2,"345":1,"346":7,"347":2,"348":1,"349":1,"350":3,"351":1,"352":5,"355":2,"356":6,"360":1,"361":3,"362":1,"363":6,"364":7,"370":1,"372":1,"373":2,"376":3,"377":2,"379":1,"380":1,"381":7,"382":16,"383":1,"384":1,"385":3,"386":1,"388":1,"389":22,"390":4,"391":5,"393":1,"395":1,"397":1,"398":2,"403":1,"406":3,"407":3,"408":4,"409":1,"410":2,"412":12,"414":4,"415":3,"416":7,"417":2,"421":1,"425":1,"429":1,"430":1,"434":3,"435":2,"436":2,"437":2,"438":5,"440":3,"441":1,"444":2,"445":6,"446":1,"447":1,"448":2,"449":3,"453":2,"454":1,"455":2,"456":2,"461":1,"462":1,"466":1,"467":11,"469":1,"470":5,"471":2,"472":4,"473":1,"474":1,"476":1,"477":11,"482":2,"483":3,"484":3,"485":3,"486":2,"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,"555":9,"557":2,"558":2,"559":2,"560":2,"561":1,"562":10,"563":15,"564":4,"567":11,"568":12,"569":1,"572":1,"573":3,"574":1,"577":1,"579":6,"580":2,"584":1,"585":19,"587":1,"591":1,"595":1,"596":42,"597":4,"599":2,"601":2,"602":2,"603":1,"604":2,"605":2,"607":17,"612":6,"614":15,"616":22,"617":22,"620":1,"621":1,"623":1,"624":3,"625":1,"629":1,"632":2,"633":3,"634":2,"635":1,"637":2,"638":1,"639":9,"640":1,"647":3,"648":2,"649":2,"652":1,"654":3,"655":2,"656":2,"657":2,"658":1,"659":3,"660":4,"661":1,"662":2,"663":2,"664":2,"665":2,"667":1,"669":2,"671":6,"672":4,"673":3,"674":3,"675":2,"678":3,"679":1,"682":1,"683":2,"685":1,"687":26,"688":25,"689":23,"690":3,"692":1,"693":11,"695":2,"696":8,"697":6,"699":1,"703":1,"704":1,"705":10,"706":16,"707":12,"708":16,"709":8,"710":6,"711":1,"712":7,"714":1,"715":1,"716":4,"717":3,"722":1,"724":1,"725":1,"730":2,"731":3,"732":4,"733":4,"735":1,"736":16,"738":14,"739":18,"740":16,"741":2,"746":4,"747":1,"748":2,"751":7,"756":19}}],["cele",{"2":{"252":2,"253":2}}],["celenium",{"0":{"384":1},"2":{"78":1,"216":7,"336":1,"358":2,"384":4,"474":1,"564":2,"667":1,"696":1,"712":1}}],["centric",{"2":{"211":1}}],["centralized",{"2":{"111":1,"209":2,"211":1}}],["center",{"2":{"26":1,"389":2}}],["certainty",{"2":{"95":1,"96":1}}],["certain",{"2":{"21":1,"159":1,"193":1,"404":1,"494":1}}],["cryptech",{"2":{"705":2,"706":1,"707":1,"708":1,"712":1}}],["cryptography",{"2":{"280":1}}],["crypto",{"2":{"129":2,"148":2,"253":2,"382":2,"519":2,"520":4,"524":2}}],["crate",{"2":{"638":1}}],["crashed",{"2":{"505":1}}],["cross",{"2":{"637":1}}],["crisis",{"2":{"277":1}}],["critical",{"2":{"39":1,"393":1,"456":1,"501":1,"522":2,"746":1}}],["criteria",{"0":{"23":1,"24":1},"2":{"20":1,"21":1,"22":1,"23":3}}],["crs",{"2":{"199":1}}],["credentials",{"2":{"382":2}}],["credibility",{"2":{"38":1}}],["creation",{"2":{"358":1,"364":2}}],["creating",{"0":{"113":1,"294":1,"489":1,"660":1},"2":{"34":1,"75":1,"78":2,"111":1,"112":2,"113":1,"149":1,"153":1,"364":2,"365":2,"366":2,"454":1,"517":2,"563":2,"642":1,"744":1}}],["creators",{"2":{"66":1}}],["createpayforblob",{"2":{"382":2}}],["createchatcompletion",{"2":{"359":2}}],["createandsubmitblob",{"2":{"358":6,"360":2}}],["createnamespaceid",{"2":{"358":7,"360":2}}],["creates",{"2":{"288":1,"358":2}}],["create2",{"2":{"187":5,"195":2}}],["createdblob",{"2":{"358":6,"360":2}}],["created",{"2":{"58":1,"126":1,"147":1,"160":1,"252":2,"253":2,"312":2,"353":1,"358":3,"506":1,"517":1,"523":1,"532":1,"537":1,"639":1,"644":1,"645":2,"658":1,"660":1,"672":1,"673":1,"736":1,"738":1,"739":1,"740":1}}],["create",{"0":{"243":1,"511":1,"519":1,"522":1,"527":1,"529":1,"534":1,"536":1,"642":1,"643":1,"656":1,"659":1},"1":{"244":1,"245":1,"246":1,"247":1,"248":1,"249":1,"250":1,"251":1,"252":1,"253":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,"644":1,"645":1},"2":{"2":3,"30":1,"38":2,"40":2,"41":2,"58":2,"75":1,"111":1,"113":1,"124":1,"145":1,"148":2,"154":1,"159":1,"163":1,"168":1,"188":2,"195":1,"196":2,"199":1,"218":2,"220":1,"258":1,"270":2,"271":2,"272":2,"283":1,"293":1,"295":1,"312":1,"315":1,"337":1,"356":5,"357":7,"358":15,"359":1,"360":4,"364":2,"378":1,"381":2,"382":4,"389":1,"484":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,"615":2,"616":1,"632":1,"637":1,"642":1,"643":7,"644":6,"656":1,"659":1,"660":1,"672":1,"682":1,"736":1,"738":1,"739":1,"745":1,"751":3}}],["crucial",{"2":{"36":1,"38":1,"71":2,"73":2,"76":1,"679":1}}],["cautious",{"2":{"563":1}}],["cause",{"2":{"400":1,"441":1,"683":1}}],["cake",{"2":{"505":1}}],["cadence",{"2":{"201":2,"202":1,"203":1}}],["car",{"2":{"519":2}}],["carrying",{"2":{"488":2}}],["carry",{"2":{"238":1}}],["carries",{"2":{"122":1}}],["cargo",{"2":{"194":2,"196":2,"199":2,"200":2}}],["care",{"2":{"94":1}}],["capacities",{"2":{"277":1}}],["capacity",{"2":{"241":3,"408":1}}],["capability",{"2":{"277":1,"637":1}}],["capabilities",{"2":{"231":1,"298":1,"563":1}}],["capable",{"2":{"66":1}}],["capture",{"2":{"31":1}}],["catching",{"2":{"736":1}}],["catch",{"2":{"48":1,"389":2}}],["categories",{"2":{"445":1}}],["category",{"2":{"3":3,"344":1,"445":1,"446":2}}],["cater",{"2":{"42":1}}],["catering",{"0":{"35":1},"2":{"35":1,"38":1}}],["caldera",{"2":{"221":1,"222":1,"223":1,"225":1,"226":1,"228":1,"229":1}}],["calculation",{"0":{"375":1}}],["calculating",{"2":{"107":1,"159":1,"373":1,"376":1}}],["calculates",{"2":{"374":1,"375":1}}],["calculated",{"2":{"322":1,"373":1,"375":3,"435":1,"441":1,"683":1}}],["calculate",{"2":{"141":2,"144":2,"159":1,"375":1,"441":1}}],["calendar",{"2":{"39":1,"48":1,"64":1}}],["calling",{"2":{"189":1,"191":1}}],["callopts",{"2":{"129":2}}],["called",{"2":{"110":1,"200":1,"206":1,"244":1,"253":1,"370":1,"376":1,"394":1,"395":1,"416":1,"714":1}}],["calldata",{"2":{"97":1,"103":8,"191":2,"192":2,"254":2,"284":4}}],["calls",{"2":{"48":1,"187":1,"219":1,"707":1}}],["call",{"0":{"341":1},"2":{"37":1,"47":1,"48":1,"97":1,"189":5,"204":1,"224":1,"313":1,"326":1,"327":1,"341":1,"358":1}}],["cafes",{"2":{"36":1}}],["cast",{"2":{"189":5,"191":1,"192":1}}],["caste",{"2":{"6":1}}],["cases",{"2":{"164":1,"385":1,"412":1}}],["case",{"2":{"62":1,"69":1,"96":1,"121":1,"141":1,"147":2,"156":1,"160":1,"168":1,"170":1,"171":1,"173":1,"182":1,"193":3,"196":2,"198":1,"254":1,"284":1,"356":1,"374":3,"405":1,"536":1,"577":1,"610":1,"642":1,"745":1}}],["casual",{"2":{"36":2,"42":1}}],["cancel",{"2":{"357":4,"358":4,"360":4}}],["canceling",{"2":{"39":1}}],["canonical",{"2":{"95":1,"369":1,"377":1,"716":1}}],["cannot",{"2":{"24":1,"40":1,"201":1,"394":2,"400":1,"413":1}}],["can",{"0":{"405":1},"2":{"7":1,"23":2,"28":1,"29":2,"30":1,"34":3,"35":5,"37":3,"38":5,"39":3,"40":5,"41":3,"42":2,"48":1,"54":1,"55":1,"58":3,"62":2,"65":1,"71":4,"75":1,"77":1,"82":1,"89":3,"94":6,"95":5,"96":4,"97":1,"107":1,"109":1,"110":5,"111":5,"113":3,"121":4,"122":3,"124":3,"126":1,"128":1,"131":1,"134":1,"135":1,"136":2,"138":1,"140":2,"141":4,"142":2,"143":1,"144":1,"145":4,"146":2,"147":2,"148":2,"149":2,"151":3,"153":1,"154":1,"157":4,"159":2,"160":5,"163":3,"164":4,"165":2,"167":2,"168":2,"171":2,"172":2,"173":1,"176":1,"179":1,"182":1,"187":1,"188":3,"190":3,"191":2,"193":3,"194":2,"195":2,"196":3,"199":1,"200":5,"201":1,"202":2,"203":1,"206":1,"210":1,"213":2,"215":1,"216":1,"218":5,"219":1,"220":1,"222":1,"224":1,"226":1,"231":1,"240":4,"241":1,"242":4,"243":1,"244":1,"245":1,"246":1,"247":1,"252":1,"254":5,"260":2,"263":1,"269":1,"270":3,"271":1,"272":1,"273":1,"281":1,"284":1,"286":1,"291":1,"292":1,"296":1,"297":1,"299":1,"302":2,"307":2,"311":1,"312":5,"314":2,"315":2,"316":1,"317":2,"319":3,"321":1,"322":1,"331":1,"332":1,"336":1,"337":1,"343":2,"349":1,"352":2,"358":7,"360":2,"361":1,"363":1,"364":3,"365":1,"366":1,"374":1,"376":3,"377":1,"378":1,"382":6,"384":1,"386":1,"388":1,"389":2,"390":2,"391":1,"393":3,"394":1,"397":3,"400":2,"407":1,"408":2,"410":3,"413":2,"414":1,"415":1,"416":2,"417":1,"423":1,"427":1,"432":1,"434":1,"436":1,"438":2,"441":2,"443":1,"447":1,"454":1,"455":1,"462":3,"467":2,"470":1,"472":1,"474":1,"477":1,"483":1,"484":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,"560":1,"563":1,"568":1,"570":1,"574":1,"579":2,"585":4,"587":1,"592":1,"593":1,"595":2,"597":1,"599":1,"600":1,"601":2,"604":1,"605":2,"608":1,"609":1,"612":2,"614":1,"615":3,"616":1,"620":1,"621":1,"628":1,"629":1,"632":5,"637":2,"640":1,"642":1,"643":1,"647":2,"653":1,"654":1,"658":1,"662":5,"663":1,"667":1,"671":1,"672":3,"673":1,"677":1,"682":1,"683":3,"684":1,"693":2,"694":1,"695":2,"696":1,"699":1,"707":1,"709":1,"710":1,"712":1,"714":1,"715":2,"717":3,"724":1,"727":1,"733":1,"739":1,"745":3,"746":1,"747":1,"756":1}}],["clean",{"2":{"654":1}}],["clearing",{"0":{"561":1}}],["clears",{"2":{"385":1}}],["clear",{"2":{"2":1,"37":1,"639":4}}],["clock",{"2":{"639":4}}],["cloud",{"2":{"202":1,"503":1,"541":1,"547":1,"552":1,"696":1}}],["closely",{"2":{"223":1}}],["close",{"2":{"129":4,"146":4,"147":2,"148":2,"357":2,"358":2,"360":2,"382":2}}],["closer",{"2":{"107":1}}],["cloned",{"2":{"74":1}}],["clone",{"0":{"74":1},"2":{"1":1,"2":2,"74":3,"187":3,"188":2,"193":3,"196":4,"199":2,"200":2,"245":2,"364":2,"369":3,"539":3,"567":3}}],["cli",{"0":{"304":1,"313":1,"379":1,"380":1,"488":1},"1":{"305":1,"306":1,"307":1,"308":1,"309":1,"310":1,"311":1,"312":1,"313":1,"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":2,"334":2,"335":1,"336":1,"337":1,"338":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":{"301":1,"307":1,"313":1,"314":2,"315":3,"316":1,"317":1,"333":1,"488":2,"531":2,"536":1,"539":1,"540":1,"542":1,"559":1,"569":1,"638":2,"647":1,"672":1}}],["clientid",{"2":{"644":8}}],["clients",{"0":{"643":1,"644":1},"1":{"644":1,"645":1},"2":{"111":1,"282":1,"639":2,"642":2,"643":2,"644":2,"717":1,"733":1}}],["client",{"0":{"108":1,"128":1,"133":1,"135":1,"138":1,"268":1,"362":1,"381":1,"531":1},"1":{"109":1,"110":1,"111":1,"112":1,"113":1,"114":1,"115":1,"116":1,"117":1,"118":1,"119":1,"269":1,"270":1,"271":1,"272":1,"273":1,"274":1,"363":1,"364":1,"365":1,"366":1,"367":1},"2":{"76":1,"96":1,"111":2,"119":1,"121":1,"125":1,"128":3,"129":4,"133":1,"134":1,"135":3,"138":1,"141":20,"142":4,"143":4,"144":6,"145":8,"148":50,"206":2,"212":2,"214":2,"268":1,"270":6,"271":4,"272":6,"273":6,"279":1,"356":2,"357":7,"358":16,"359":5,"360":8,"361":1,"362":1,"364":12,"365":10,"366":10,"376":1,"381":7,"531":4,"557":2,"559":2,"595":1,"643":8,"644":8,"707":1,"731":1}}],["clicking",{"2":{"240":1,"390":2}}],["click",{"2":{"61":1,"62":3,"64":4,"65":1,"70":1,"72":1,"73":1,"228":1,"384":2,"390":1,"391":3,"426":1,"503":3}}],["clubs",{"2":{"34":2}}],["clang",{"2":{"622":4}}],["classname=",{"2":{"389":4}}],["classes",{"2":{"15":1}}],["claim",{"0":{"65":1,"496":1},"2":{"64":1,"65":3,"423":1,"427":1,"432":1,"496":1}}],["clarifying",{"2":{"8":1}}],["clarity",{"2":{"2":1,"12":1}}],["coin",{"2":{"390":1,"391":1,"396":1}}],["coingeckoid",{"2":{"389":6}}],["coindecimals",{"2":{"389":6}}],["coindenom",{"2":{"389":6}}],["coinminimaldenom",{"2":{"389":6}}],["cointype",{"2":{"389":2}}],["copied",{"2":{"596":1}}],["copies",{"2":{"200":2}}],["copy",{"2":{"76":1,"109":1,"188":3,"340":1,"352":1,"400":1,"539":1,"567":1}}],["cosmovisor",{"2":{"716":1}}],["cosmoslist",{"2":{"697":1,"711":1}}],["cosmosnetwork",{"2":{"640":1}}],["cosmossdk",{"2":{"639":4}}],["cosmostation",{"0":{"391":1},"2":{"388":1,"391":2}}],["cosmos",{"2":{"125":7,"242":1,"244":1,"253":2,"266":2,"277":4,"278":2,"279":1,"281":1,"282":1,"370":1,"374":1,"382":4,"390":1,"391":1,"412":1,"417":1,"440":3,"444":1,"448":1,"455":3,"465":1,"477":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,"585":3,"637":1,"638":1,"639":9,"640":1,"684":1,"707":1,"708":1,"736":2,"738":2,"739":2}}],["costly",{"2":{"154":1}}],["cost",{"2":{"34":1,"69":1,"168":1,"170":1,"240":1,"283":1,"374":6,"375":5,"414":1}}],["costs",{"2":{"34":2,"35":1,"38":3,"40":1,"155":1,"178":1,"211":1,"374":3,"376":1}}],["coral",{"2":{"519":2}}],["corner",{"2":{"384":1,"390":1,"391":1}}],["correspond",{"2":{"193":1}}],["corresponds",{"2":{"179":1,"319":1}}],["corresponding",{"0":{"134":1},"1":{"135":1},"2":{"121":1,"122":1,"124":2,"126":1,"129":2,"134":2,"145":1,"146":2,"148":2,"152":1,"153":3,"157":1,"186":1,"187":2,"188":1,"189":2,"194":1,"195":1,"196":2,"199":1,"218":1,"407":1,"416":2,"512":1,"637":1}}],["correctly",{"2":{"172":1,"176":1,"179":1,"187":1,"413":1,"623":1,"638":1,"641":1,"736":1,"740":1}}],["correct",{"2":{"95":1,"172":1,"200":1,"246":1,"407":1,"485":1,"486":1,"539":1,"541":2,"552":2,"623":1,"633":1,"634":1,"649":1,"674":1,"675":1,"699":1}}],["correction",{"0":{"12":1}}],["corrective",{"2":{"8":1}}],["cores",{"2":{"478":1,"626":1,"719":1,"720":1,"742":1}}],["core",{"0":{"311":1,"574":1},"2":{"53":1,"76":3,"94":1,"125":3,"238":1,"252":2,"253":2,"261":2,"262":2,"279":1,"289":2,"306":1,"308":1,"311":16,"312":6,"336":2,"352":2,"386":1,"412":4,"445":3,"446":1,"454":1,"467":1,"469":1,"470":7,"477":8,"483":12,"484":6,"485":6,"544":2,"545":2,"546":2,"547":4,"550":2,"555":2,"560":2,"564":2,"574":2,"579":1,"581":1,"607":14,"614":6,"617":6,"632":5,"633":6,"649":1,"666":1,"671":8,"672":6,"674":6,"692":1,"693":5,"699":1,"704":1,"705":1,"719":2,"720":1,"740":2,"748":2,"756":14}}],["cooperation",{"2":{"46":1}}],["coordinated",{"2":{"679":1,"716":1}}],["coordinates",{"2":{"407":2}}],["coordinate",{"2":{"30":1,"40":2,"714":1}}],["coordination",{"2":{"30":1,"462":1,"714":2,"727":1}}],["coworking",{"2":{"38":1}}],["coding",{"2":{"37":1,"398":2,"412":1,"434":1}}],["codelab",{"2":{"503":2}}],["coded",{"2":{"397":1,"398":1,"477":1,"683":1}}],["codespace",{"2":{"132":2,"382":4,"522":2,"745":2,"751":2}}],["codebase",{"2":{"94":1,"725":1}}],["code",{"0":{"5":1,"46":1},"1":{"6":1,"7":1,"8":1,"9":1,"10":1,"11":1,"12":1,"13":1,"14":1,"15":1,"16":1},"2":{"8":2,"9":1,"11":1,"13":1,"14":1,"16":4,"24":1,"46":4,"94":2,"95":1,"109":1,"132":2,"213":1,"218":1,"222":1,"338":2,"350":1,"356":1,"382":18,"389":1,"477":1,"503":1,"522":2,"683":1,"715":1,"745":1,"751":1}}],["covers",{"2":{"53":1,"54":1,"55":1,"56":1,"59":1,"66":1,"213":1,"388":1,"580":1,"732":2}}],["cover",{"2":{"38":1,"40":1,"41":1,"59":1,"69":1,"214":1,"232":1,"256":1,"304":1,"314":1,"526":1,"542":1}}],["covering",{"2":{"38":1,"41":2}}],["covered",{"2":{"36":1,"611":1,"731":1,"733":1}}],["covenant",{"2":{"16":4}}],["co",{"2":{"34":2,"38":2,"39":1,"41":1,"50":1,"697":1,"711":1}}],["course",{"2":{"241":1}}],["counterparty",{"2":{"639":2}}],["count",{"2":{"80":1}}],["country",{"2":{"23":1}}],["couldn",{"2":{"40":2,"129":2,"146":2}}],["could",{"2":{"7":1,"109":1,"110":1,"112":1,"115":3,"130":1,"159":1,"160":1,"242":1,"279":1,"314":1,"358":1,"360":1,"374":1,"375":1,"398":2,"409":1,"416":1,"438":3,"505":1}}],["columns",{"2":{"107":1,"124":1,"159":1,"407":1}}],["column",{"2":{"95":1,"107":3,"122":1,"124":2,"159":1,"328":2,"409":1,"415":3}}],["collide",{"2":{"503":1}}],["collaboration",{"2":{"38":1,"44":1}}],["collaborate",{"2":{"34":1}}],["collectively",{"2":{"408":1}}],["collection",{"2":{"43":1,"50":1,"110":1}}],["collect",{"2":{"31":1,"488":4,"660":2}}],["collector",{"2":{"23":1,"78":4,"502":2,"547":2,"548":3}}],["color",{"2":{"6":1,"52":1}}],["cohorts",{"2":{"21":1,"22":3,"25":1}}],["cohort",{"0":{"21":1,"27":1},"2":{"21":3,"22":12,"23":1,"24":1,"25":1,"27":5}}],["conjured",{"2":{"503":1}}],["conn",{"2":{"382":6}}],["connects",{"2":{"548":1}}],["connectivity",{"2":{"416":1}}],["connecting",{"2":{"111":2,"311":1,"483":1,"632":1,"693":1,"733":1,"751":1}}],["connectionid",{"2":{"644":8}}],["connection",{"0":{"642":1,"644":1},"2":{"38":1,"75":5,"90":1,"94":1,"111":6,"311":1,"382":2,"477":1,"484":1,"547":5,"632":1,"642":1,"644":30,"671":1}}],["connectionside",{"2":{"644":4}}],["connections",{"0":{"547":1},"2":{"34":1,"36":1,"37":1,"58":1,"75":1,"502":6,"541":3,"552":3,"600":1,"637":2,"639":2,"642":2,"645":1,"661":2,"739":1}}],["connect",{"0":{"311":1,"587":1},"1":{"588":1,"589":1},"2":{"32":1,"36":1,"37":1,"49":1,"70":1,"94":1,"129":2,"146":2,"148":2,"224":1,"226":1,"336":1,"384":1,"389":2,"416":1,"448":1,"449":1,"465":2,"476":1,"477":1,"559":2,"585":4,"624":1,"684":2,"699":4,"717":1,"751":2}}],["connected",{"2":{"23":1,"390":1,"391":1,"397":2,"412":1,"587":1,"727":1}}],["congrats",{"2":{"347":1,"349":1}}],["congratulations",{"0":{"79":1,"349":1},"2":{"517":1,"525":1,"533":1,"614":1,"617":1,"644":1}}],["congested",{"2":{"254":1}}],["congests",{"2":{"241":1}}],["congestion",{"2":{"241":1,"385":1,"416":1}}],["conditions",{"2":{"176":1,"509":1}}],["conducting",{"2":{"397":1}}],["conduct",{"0":{"5":1,"46":1},"1":{"6":1,"7":1,"8":1,"9":1,"10":1,"11":1,"12":1,"13":1,"14":1,"15":1,"16":1},"2":{"7":1,"8":1,"9":1,"11":1,"13":1,"14":1,"16":4,"24":1,"30":1,"46":3,"396":1,"717":1}}],["concerned",{"2":{"399":1}}],["concepts",{"0":{"151":1},"1":{"152":1,"153":1,"154":1,"155":1,"156":1,"157":1,"158":1,"159":1,"160":1,"161":1,"162":1}}],["concept",{"2":{"53":1,"437":1}}],["concrete",{"2":{"358":2,"360":2}}],["conclusion",{"0":{"149":1,"182":1,"533":1}}],["concise",{"2":{"2":1}}],["conf",{"2":{"89":2}}],["confidence",{"2":{"96":1,"396":2}}],["configs",{"2":{"89":2,"503":4}}],["configurable",{"2":{"97":1}}],["configurations",{"0":{"586":1},"1":{"587":1,"588":1,"589":1,"590":1,"591":1,"592":1},"2":{"73":1,"315":1,"554":1,"603":1,"639":1,"641":1}}],["configuration",{"0":{"73":1,"74":1,"604":1,"605":1,"639":1,"641":1,"753":1},"2":{"71":4,"72":1,"73":3,"296":3,"346":1,"348":1,"488":4,"502":2,"503":3,"531":3,"536":1,"542":1,"554":2,"560":2,"591":1,"595":2,"602":1,"604":2,"605":1,"639":1,"641":2,"753":1}}],["configuring",{"2":{"71":2,"73":1,"612":1,"638":1}}],["configures",{"2":{"489":1}}],["configured",{"2":{"76":1,"343":1,"373":1,"640":1}}],["configure",{"0":{"71":1,"600":1,"645":1},"1":{"601":1,"602":1,"603":1},"2":{"70":1,"71":2,"76":1,"78":3,"600":1,"604":1,"605":1,"642":1,"753":1}}],["config",{"0":{"61":1,"559":1,"560":1,"571":1,"573":1,"661":1},"1":{"572":1,"573":1,"574":2,"575":2,"576":2,"577":2,"578":2,"579":2},"2":{"61":1,"73":4,"74":1,"75":1,"76":3,"89":5,"348":1,"382":2,"488":2,"489":2,"502":1,"503":8,"527":2,"531":2,"536":2,"550":2,"559":1,"560":7,"562":1,"564":3,"573":4,"577":1,"579":1,"585":27,"587":1,"590":1,"595":2,"601":5,"602":4,"604":2,"605":2,"622":4,"638":1,"639":3,"641":5,"645":1,"658":1,"659":1,"660":4,"661":2,"670":1,"753":1}}],["confirming",{"2":{"267":1}}],["confirmed",{"2":{"95":1,"96":1,"382":2,"423":1,"427":1,"432":1,"639":2}}],["confirmations",{"2":{"210":1}}],["confirmation",{"2":{"47":1,"113":1,"537":1,"639":2,"670":1}}],["confirm",{"0":{"423":1,"427":1},"2":{"30":1,"107":1,"210":1,"267":1,"431":1,"503":1,"522":2,"751":1}}],["conferences",{"2":{"39":1}}],["conference",{"2":{"39":1}}],["conventions",{"2":{"554":1}}],["convenient",{"2":{"30":1}}],["conversion",{"2":{"141":1,"451":1}}],["conversations",{"2":{"7":1,"31":1}}],["converts",{"2":{"358":2}}],["convert",{"2":{"140":1,"141":1,"142":1,"144":1,"358":3,"360":2}}],["converting",{"0":{"139":1},"1":{"140":1,"141":1,"142":1,"143":1,"144":1,"145":1,"146":1,"147":1,"148":1}}],["converted",{"2":{"127":1,"132":1,"137":1,"140":1,"143":1,"319":1}}],["consoleconfirm",{"2":{"751":1}}],["consolecode",{"2":{"745":1,"751":1}}],["console$request",{"2":{"673":1}}],["console$",{"2":{"488":1}}],["console",{"2":{"342":1}}],["consoleblockhash",{"2":{"341":1}}],["consolebasefeepergas",{"2":{"340":1}}],["cons",{"0":{"156":1,"162":1,"166":1,"170":1,"174":1,"178":1,"181":1},"2":{"151":1}}],["consumer",{"2":{"639":4}}],["consume",{"2":{"375":1,"605":1}}],["consumes",{"2":{"375":1}}],["consumed",{"2":{"125":1,"374":2}}],["consuming",{"2":{"94":1}}],["consensus",{"0":{"379":1,"382":1,"468":1,"530":1,"580":1,"582":1,"587":1,"597":1,"598":1,"611":1,"686":1,"687":1,"720":1,"735":1},"1":{"469":1,"470":1,"581":1,"582":1,"583":2,"584":2,"585":2,"586":1,"587":1,"588":2,"589":2,"590":1,"591":1,"592":1,"593":1,"594":1,"595":1,"596":1,"597":1,"598":1,"599":2,"600":2,"601":2,"602":2,"603":2,"604":2,"605":2,"606":1,"607":1,"687":1,"688":1,"689":1,"690":1,"736":1},"2":{"76":2,"94":1,"106":1,"109":1,"111":1,"121":1,"122":1,"212":3,"218":1,"238":1,"241":1,"242":1,"277":1,"282":1,"283":1,"311":2,"328":2,"336":2,"373":1,"382":4,"409":1,"412":1,"413":6,"414":2,"440":1,"455":1,"461":1,"467":2,"469":1,"470":2,"476":1,"477":3,"494":1,"503":1,"517":1,"526":1,"530":2,"540":1,"541":1,"564":4,"580":3,"581":1,"587":1,"594":1,"600":1,"607":32,"611":2,"624":1,"671":1,"672":2,"686":1,"692":1,"693":7,"699":2,"704":1,"705":8,"706":2,"708":4,"716":2,"717":5,"720":1,"732":2,"735":1,"741":1,"743":1,"746":1,"751":1,"754":2,"756":32}}],["consequence",{"2":{"12":1,"13":1,"14":1,"15":1,"409":1}}],["consequences",{"2":{"11":1,"13":1,"394":2}}],["const",{"2":{"389":8}}],["construction",{"2":{"156":1,"159":1,"168":2}}],["constructions",{"2":{"149":1,"163":1,"171":1}}],["constructing",{"2":{"145":2}}],["constructive",{"2":{"7":1}}],["constructed",{"2":{"142":1,"218":1,"397":1}}],["construct",{"2":{"139":1,"141":4,"144":1,"361":1}}],["constructs",{"2":{"122":1,"151":1,"167":1,"358":1}}],["constructor",{"2":{"103":2}}],["constants",{"2":{"317":3,"321":1,"346":4,"389":18,"390":1,"551":4,"585":22,"596":6,"623":6,"745":2,"751":2}}],["constantly",{"2":{"256":1,"637":1}}],["constant",{"2":{"76":1,"375":1,"683":2}}],["consist",{"2":{"111":1,"376":1}}],["consists",{"2":{"110":1,"412":1,"413":1,"415":3,"434":1}}],["consistency",{"2":{"41":1}}],["consistent",{"2":{"41":2}}],["considerable",{"2":{"605":1}}],["considerations",{"0":{"548":1},"2":{"94":2,"544":1,"548":1}}],["consider",{"2":{"34":1,"36":2,"37":1,"40":3,"319":1,"396":1,"592":1,"756":1}}],["considered",{"2":{"7":1}}],["context",{"2":{"128":4,"129":10,"133":2,"135":6,"148":8,"270":6,"271":4,"272":6,"273":4,"356":2,"357":4,"358":8,"359":4,"360":4,"381":6,"382":8}}],["contents",{"2":{"361":1}}],["content",{"2":{"2":1,"40":2,"58":1,"336":2,"342":2,"359":4,"494":2,"736":1,"738":1,"739":1,"740":1}}],["containerization",{"2":{"612":1}}],["container",{"0":{"253":1},"2":{"252":9,"253":11,"615":5,"616":4,"617":2}}],["containers",{"2":{"252":1,"253":1}}],["contained",{"2":{"147":1,"152":1}}],["containing",{"2":{"124":1,"129":4,"130":1,"135":1,"141":1,"143":1,"145":3,"146":2,"148":4,"157":3,"165":1,"168":1,"184":1,"187":1,"195":1,"270":2,"381":2,"415":1,"503":2}}],["contain",{"2":{"73":1,"122":1,"124":1,"130":1,"135":1,"136":1,"141":1,"160":1,"258":1,"370":1,"407":1,"415":1,"438":1,"495":1}}],["contains",{"2":{"73":1,"76":1,"79":1,"110":2,"129":4,"132":1,"134":1,"139":4,"145":3,"146":2,"148":2,"258":1,"340":1,"371":1,"465":1,"495":1,"562":1,"684":1,"699":1}}],["contacts",{"2":{"39":1,"40":1}}],["contact",{"2":{"35":1,"42":1}}],["contrastingly",{"2":{"409":1}}],["contrast",{"2":{"209":1,"401":1}}],["contract",{"0":{"82":1,"129":1,"193":1},"1":{"83":1,"84":1,"194":1,"195":1},"2":{"71":1,"72":2,"75":1,"76":4,"78":4,"85":1,"86":1,"94":2,"96":1,"103":6,"105":1,"106":1,"107":7,"109":1,"111":4,"121":3,"122":1,"124":4,"125":1,"126":2,"127":1,"129":12,"132":1,"137":1,"139":6,"145":5,"146":8,"147":6,"148":14,"149":1,"150":1,"154":2,"156":1,"157":1,"159":1,"160":2,"164":1,"168":2,"170":1,"171":2,"172":3,"175":1,"176":1,"179":1,"184":2,"186":1,"187":7,"188":3,"189":8,"190":1,"193":2,"195":9,"196":5,"197":2,"198":1,"199":4,"200":1,"201":3,"202":2,"203":4,"204":2,"210":1,"212":2,"213":5,"214":1,"216":2,"218":5,"219":5,"220":3,"240":1,"348":4,"369":1,"402":1,"454":1}}],["contracts",{"0":{"99":1,"102":1,"186":1,"216":1,"236":1,"369":1},"1":{"100":1,"101":1,"102":1,"103":1,"104":1,"105":1,"106":1,"107":1,"187":1,"188":1,"189":1,"190":1,"191":1,"192":1,"193":1,"194":1,"195":1,"196":1,"197":1},"2":{"66":1,"69":1,"71":4,"72":4,"73":2,"76":2,"78":4,"79":1,"81":3,"82":1,"102":4,"103":6,"105":1,"107":1,"112":2,"113":4,"117":1,"119":2,"139":1,"165":1,"187":5,"188":4,"195":3,"206":1,"214":1,"216":2,"226":1,"236":1,"238":1,"242":1,"369":2}}],["control",{"2":{"187":1,"432":1,"555":1,"605":1,"636":1}}],["controlled",{"2":{"71":1}}],["contributor",{"2":{"16":4,"445":1}}],["contributors",{"2":{"6":1,"445":1,"446":1}}],["contributing",{"2":{"7":1}}],["contributiondao",{"2":{"687":1,"688":1,"689":1}}],["contributions",{"2":{"2":1,"8":1,"20":1,"26":3}}],["contribution",{"0":{"2":1},"2":{"2":2,"37":1,"92":1}}],["contributes",{"2":{"7":1,"374":1}}],["contribute",{"2":{"6":1,"23":1,"26":2,"38":1}}],["continuing",{"2":{"751":1}}],["continuousvestingaccount",{"2":{"523":3}}],["continuous",{"2":{"511":1,"522":1}}],["continuously",{"2":{"446":4,"462":1}}],["continues",{"2":{"501":1,"722":1}}],["continue",{"2":{"64":1,"72":1,"254":1,"365":2,"384":1,"437":1,"438":1,"714":1}}],["continued",{"2":{"13":1,"38":1}}],["continually",{"2":{"28":1}}],["coming",{"2":{"738":1,"739":1}}],["comfortable",{"2":{"567":1}}],["comcelestia",{"2":{"470":1}}],["combines",{"2":{"350":1}}],["combined",{"0":{"329":1},"2":{"241":1}}],["comexport",{"2":{"259":1,"614":1}}],["cometbft",{"2":{"209":1,"212":1,"440":1,"683":1}}],["come",{"2":{"109":1,"212":1,"232":1,"356":2,"678":1}}],["comes",{"2":{"94":1}}],["com",{"0":{"41":1},"2":{"39":2,"41":6,"69":1,"74":2,"76":2,"121":4,"125":18,"128":2,"129":16,"135":6,"145":2,"146":2,"147":2,"148":24,"187":2,"188":2,"193":2,"196":2,"199":18,"200":2,"245":2,"257":1,"259":1,"266":2,"267":2,"269":2,"270":6,"285":1,"289":2,"311":2,"348":2,"352":2,"355":4,"356":8,"369":2,"376":2,"381":6,"382":16,"390":3,"391":1,"449":1,"467":11,"470":1,"473":1,"474":2,"484":2,"508":2,"531":2,"539":2,"546":2,"567":2,"585":10,"607":28,"614":1,"622":2,"639":4,"671":2,"672":2,"687":10,"688":9,"689":8,"697":1,"705":4,"706":7,"707":7,"708":9,"709":8,"712":2,"756":28}}],["compiling",{"2":{"622":1}}],["compile",{"2":{"199":6}}],["compiler",{"2":{"102":1}}],["computing",{"2":{"282":1}}],["computation",{"2":{"211":1}}],["computed",{"2":{"160":1,"374":1,"407":1}}],["computesquaresizefromshareproof",{"2":{"107":1}}],["computesquaresizefromrowproof",{"2":{"107":1,"159":1}}],["computes",{"2":{"107":2,"374":1,"415":1}}],["compute",{"2":{"95":1,"415":1}}],["comprised",{"2":{"435":1}}],["comprises",{"2":{"145":1}}],["comprising",{"2":{"184":1}}],["comprehensive",{"2":{"0":1}}],["compatmode",{"2":{"638":2}}],["compatibility",{"0":{"81":1,"280":1},"2":{"76":1}}],["compatible",{"0":{"723":1,"726":1,"728":1},"2":{"66":1,"81":1,"221":1,"223":1,"236":1,"280":1,"638":1,"714":2,"716":2}}],["compared",{"2":{"211":1}}],["compare",{"2":{"160":1}}],["comparing",{"2":{"154":1}}],["compact",{"0":{"154":1},"2":{"154":4,"155":1,"168":1}}],["companion",{"2":{"50":1}}],["companies",{"2":{"38":3}}],["company",{"2":{"38":1}}],["compelling",{"2":{"38":1}}],["component",{"2":{"81":1,"242":1,"389":2}}],["components",{"0":{"93":1},"1":{"94":1,"95":1,"96":1,"97":1},"2":{"26":1,"33":1,"93":1,"184":1,"186":1,"214":1,"370":1,"376":2,"389":8,"412":1,"678":1,"715":1}}],["compose",{"2":{"67":1,"76":1,"77":1,"252":2,"253":6,"343":2}}],["composition",{"2":{"27":1}}],["complex",{"2":{"604":1}}],["completing",{"2":{"66":1,"751":1}}],["completes",{"2":{"396":1}}],["completely",{"2":{"318":1}}],["completed",{"2":{"243":1,"539":1,"567":1}}],["complete",{"2":{"23":1,"47":1,"72":1,"78":2,"199":1,"200":1,"243":1,"410":1,"443":1}}],["compliant",{"2":{"104":1}}],["compliance",{"2":{"23":1}}],["comply",{"2":{"24":1}}],["complaints",{"2":{"10":1}}],["comma",{"2":{"196":1,"585":2,"661":2}}],["commands",{"0":{"329":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":{"245":1,"266":1,"315":2,"355":1,"382":2,"488":2,"494":1,"500":1,"539":1,"540":1,"560":1,"564":1,"567":1,"585":1,"622":2}}],["command",{"0":{"314":1,"507":1},"2":{"76":2,"78":1,"102":1,"157":1,"245":1,"248":1,"252":2,"253":5,"262":1,"289":1,"296":1,"311":2,"312":4,"313":1,"314":3,"316":1,"317":1,"318":1,"319":1,"321":3,"325":1,"328":1,"332":1,"333":2,"334":1,"352":2,"353":1,"360":1,"470":1,"472":1,"483":1,"484":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,"555":1,"560":2,"561":1,"567":1,"568":1,"574":1,"596":1,"601":1,"605":1,"614":1,"616":2,"617":2,"622":2,"631":1,"632":1,"643":2,"644":1,"654":1,"655":1,"656":2,"657":1,"658":1,"659":2,"660":2,"662":2,"663":1,"670":1,"672":3,"673":1,"710":1,"745":1}}],["commission",{"2":{"23":1,"431":1,"496":2,"500":6,"751":6}}],["commitment",{"0":{"124":1,"152":1,"153":1,"154":1,"155":1,"156":1,"201":1},"1":{"153":1,"154":1,"155":1,"156":1,"202":1,"203":1,"204":1},"2":{"124":5,"127":1,"139":2,"145":3,"147":1,"151":1,"152":3,"153":9,"154":2,"155":2,"158":2,"160":3,"161":1,"168":6,"169":1,"171":1,"172":2,"173":1,"176":5,"179":7,"184":3,"204":1,"213":1,"218":1,"262":1,"270":4,"284":5,"319":5,"321":10,"322":3,"336":2,"358":4,"360":2,"364":4,"381":4,"407":1,"409":1,"415":6,"434":2,"558":2}}],["commitments",{"0":{"147":1,"168":1,"169":1,"170":1,"175":1,"177":1,"178":1},"1":{"169":1,"170":1,"176":1,"177":1,"178":1},"2":{"96":1,"110":6,"112":1,"113":1,"147":1,"154":1,"165":1,"168":2,"171":1,"172":1,"175":1,"206":1,"262":2,"407":1,"416":1}}],["committing",{"0":{"112":1},"2":{"129":2,"146":2,"148":2}}],["committees",{"0":{"208":1},"1":{"209":1,"210":1,"211":1},"2":{"209":1}}],["committee",{"2":{"70":1}}],["committedto",{"2":{"148":4}}],["committed",{"0":{"129":1,"607":1,"756":1},"2":{"41":1,"104":1,"107":9,"110":1,"115":1,"122":2,"124":4,"126":2,"129":8,"139":8,"145":6,"148":4,"149":1,"153":1,"156":1,"157":1,"159":1,"160":2,"201":1,"218":1,"382":4,"415":1,"607":2,"756":2}}],["commitforge",{"2":{"102":1}}],["commits",{"2":{"8":1,"107":1,"112":1,"139":2,"145":2,"163":1,"179":1}}],["commit",{"2":{"2":3,"81":1,"102":1,"107":1,"112":1,"124":1,"168":1,"200":2,"328":4,"567":1}}],["commonly",{"2":{"394":1,"503":1}}],["common",{"2":{"16":1,"54":1,"129":2,"148":2,"268":1,"362":1,"413":2,"547":1,"665":1}}],["communities",{"2":{"58":1}}],["community",{"0":{"42":1,"46":1,"444":1,"467":1,"495":1,"687":1,"688":1,"689":1,"690":1,"692":1,"693":1,"704":1,"705":1,"706":1,"707":1,"708":1,"709":1},"2":{"2":1,"3":2,"4":1,"6":2,"7":2,"8":2,"9":3,"10":2,"11":2,"12":3,"13":2,"14":3,"15":3,"16":1,"24":1,"26":1,"30":1,"32":1,"33":1,"34":4,"37":2,"39":2,"41":1,"44":1,"46":3,"47":1,"49":1,"51":1,"57":1,"58":1,"377":1,"440":1,"444":2,"456":2,"467":1,"470":1,"494":1,"495":11,"678":1,"679":1,"685":1,"687":1,"693":1,"703":1,"722":1,"727":1}}],["communicating",{"2":{"503":2}}],["communications",{"0":{"39":1}}],["communication",{"2":{"14":1,"39":1,"49":1,"541":1,"547":1,"552":1,"577":1,"637":1,"707":1,"739":1}}],["communicate",{"2":{"8":1,"36":1,"76":1,"353":1,"746":1}}],["comments",{"2":{"7":1,"8":1}}],["fsudo",{"2":{"736":1,"738":1,"739":1,"740":1}}],["fssl",{"2":{"622":2}}],["f5nodes",{"2":{"687":1,"688":1,"689":1,"706":1,"707":1,"708":1}}],["fc",{"2":{"375":2}}],["fn",{"2":{"364":2,"365":2,"366":2}}],["fmt",{"2":{"115":4,"128":2,"129":4,"146":2,"147":2,"148":2,"270":2,"356":2,"358":6,"359":4,"381":2,"382":8}}],["f",{"2":{"115":22,"343":1,"736":1,"738":1,"739":1,"740":1}}],["floor",{"2":{"451":1}}],["flow",{"2":{"22":1,"37":1}}],["flipping",{"2":{"396":1}}],["flat",{"2":{"373":3,"376":1,"435":1}}],["flag",{"2":{"94":1,"199":1,"261":1,"269":1,"311":1,"314":2,"315":1,"316":1,"317":1,"321":1,"322":3,"328":2,"336":2,"357":2,"358":2,"360":2,"363":1,"483":1,"511":1,"522":2,"527":1,"531":1,"539":1,"541":1,"544":1,"547":3,"551":1,"552":1,"554":1,"555":1,"564":1,"574":1,"597":2,"616":1,"751":2}}],["flags",{"0":{"296":1,"315":1,"544":1},"1":{"316":1,"317":1,"318":1,"545":1,"546":1,"547":1},"2":{"76":1,"89":2,"246":2,"296":1,"314":4,"315":2,"379":2,"492":4,"529":2,"542":2,"544":2,"554":2,"614":1}}],["flexibility",{"0":{"211":1},"2":{"182":1,"211":1,"242":1,"413":1}}],["flexible",{"2":{"35":1}}],["further",{"2":{"246":1}}],["furthermore",{"2":{"26":1,"47":1,"410":1,"413":1}}],["fullfill",{"2":{"187":2}}],["fullnodes",{"2":{"115":1}}],["fullnode",{"2":{"111":6,"115":4}}],["full",{"0":{"87":1,"89":1,"90":1,"114":1,"129":1,"557":1,"558":1,"624":1,"625":1,"627":1,"630":1,"631":1,"632":1,"633":1,"635":1,"636":1,"702":1,"709":1,"738":1},"1":{"88":1,"89":1,"90":1,"115":1,"116":1,"117":1,"118":1,"119":1,"625":1,"626":1,"627":1,"628":2,"629":1,"630":1,"631":2,"632":2,"633":1,"634":2,"635":1,"636":1,"703":1,"704":1,"705":1},"2":{"49":1,"75":1,"76":3,"79":2,"89":1,"90":1,"94":1,"95":1,"111":4,"115":1,"121":2,"164":1,"168":1,"187":1,"244":1,"246":2,"247":2,"248":2,"274":1,"282":1,"310":1,"367":1,"397":3,"398":4,"407":3,"416":1,"467":2,"470":1,"488":2,"527":1,"529":1,"554":1,"555":4,"557":1,"558":2,"561":1,"573":2,"610":1,"617":1,"624":1,"625":2,"626":1,"631":6,"632":7,"633":9,"634":3,"635":2,"636":1,"682":1,"687":1,"688":1,"689":1,"691":1,"693":2,"699":1,"705":4,"706":1,"707":1,"708":3,"709":7,"717":2,"719":1,"731":1,"738":17}}],["fully",{"2":{"23":1,"27":1,"67":1,"346":1,"437":2,"446":1,"695":1}}],["fun",{"2":{"350":1}}],["func2",{"2":{"607":2,"756":2}}],["func",{"2":{"113":2,"115":3,"121":2,"128":2,"129":6,"135":2,"141":7,"142":1,"143":1,"144":1,"145":1,"148":22,"270":2,"271":1,"272":2,"273":2,"356":2,"357":2,"358":5,"359":2,"360":1,"381":2,"382":2}}],["functionalities",{"2":{"699":1}}],["functionality",{"2":{"30":1,"81":1,"111":2,"260":1,"279":1,"302":1}}],["functional",{"2":{"679":1}}],["functionids",{"2":{"195":1}}],["functionid",{"2":{"189":3,"191":2,"192":1,"195":1,"196":2,"203":1}}],["functionverifier",{"2":{"188":5,"200":4}}],["function",{"0":{"188":1,"189":1,"200":1,"357":1},"1":{"190":1,"191":1,"192":1},"2":{"97":1,"103":4,"107":5,"148":2,"187":3,"188":10,"189":16,"190":3,"191":4,"192":2,"195":8,"196":12,"199":1,"200":4,"203":2,"220":2,"356":2,"357":1,"358":2,"359":1,"360":1,"374":2,"375":7,"376":3,"382":8,"389":5,"410":1,"413":1}}],["functions",{"0":{"358":1},"2":{"95":1,"107":4,"187":2,"238":3,"240":1,"357":1,"358":1,"413":5,"509":1}}],["fund",{"0":{"528":1,"537":1},"2":{"78":1,"312":2,"346":1,"352":1,"444":1,"484":1,"495":2,"528":1,"537":1,"632":1,"672":1}}],["funded",{"2":{"67":1,"307":1,"312":1,"338":1,"346":2,"348":1,"352":2,"640":1}}],["funds",{"2":{"40":1,"62":2,"64":1,"69":1,"254":1,"257":2,"258":1,"262":1,"312":1,"394":1,"444":1,"494":1,"495":3,"507":4,"509":1,"673":1}}],["funding",{"2":{"38":1,"78":3,"256":1,"352":1}}],["future",{"2":{"21":1,"31":1,"32":2,"36":1,"37":1,"42":1,"81":1,"242":1,"279":1,"302":1,"307":1,"438":1,"445":1,"446":1,"548":1,"552":1,"597":1,"715":1,"739":1,"751":1}}],["feasible",{"2":{"408":1}}],["feature",{"0":{"287":1},"1":{"288":1,"289":1,"290":1,"291":1,"292":1,"293":1,"294":1,"295":1,"296":1},"2":{"71":1,"93":1,"242":1,"254":1}}],["features",{"2":{"58":1,"241":1,"298":1,"300":1,"361":1,"406":2,"567":1,"649":1,"714":1,"715":6}}],["federation",{"2":{"503":2}}],["fed",{"2":{"199":1}}],["fek44boeswewectexicgqiabelorfjcagrqagqhaqqeaqogl2nlbgvzdglhlmjsb2iudjeutxnnugf5rm9yqmxvynmsegovy2vszxn0awexdjhlc2n1n2z1owx2odc5a3p1nxvlz3vldxj0ctvzdzzoc20zbmushqaaaaaaaaaaaaaaaaaaaaaaaaaaaawey1oiszk=",{"2":{"132":2}}],["few",{"2":{"115":1,"150":1,"156":1,"234":1,"240":2,"241":1,"284":1,"352":1,"529":1,"595":1,"698":1,"713":1,"729":1}}],["fetches",{"2":{"272":2,"273":2,"358":1}}],["fetchedblob",{"2":{"358":2,"360":4}}],["fetched",{"2":{"95":1,"358":3,"360":3,"554":1}}],["fetcher",{"2":{"94":1}}],["fetch",{"2":{"94":1,"187":1,"270":2,"271":1,"272":3,"273":5,"350":1,"358":5,"360":4,"364":2,"365":3,"366":3,"381":2}}],["fetching",{"0":{"273":1,"366":1},"2":{"75":1,"272":2,"365":2,"366":2}}],["feecurrencies",{"2":{"389":2}}],["fee=gas",{"2":{"375":1}}],["fees=21000utia",{"2":{"745":1,"751":2}}],["fees=21000utiacelestia",{"2":{"745":1}}],["fees",{"0":{"373":1},"1":{"374":1,"375":1,"376":1},"2":{"256":1,"258":1,"263":2,"266":3,"267":2,"322":1,"373":3,"377":1,"385":1,"387":1,"435":3,"507":4,"522":6,"529":2,"639":4}}],["feegrant",{"0":{"255":1,"260":1,"262":1,"264":1},"1":{"256":1,"257":1,"258":1,"259":1,"260":1,"261":1,"262":1,"263":1,"264":1,"265":2,"266":1,"267":1},"2":{"256":1,"258":1,"259":2,"260":1,"262":1,"264":1,"266":4,"267":1,"277":1,"374":1}}],["feel",{"2":{"186":1,"720":1}}],["fee",{"0":{"259":1,"266":1,"372":1,"375":1,"435":1},"1":{"260":1,"261":1,"267":1,"373":1,"374":1,"375":1,"376":1,"377":1},"2":{"71":1,"72":1,"78":20,"187":4,"258":1,"259":1,"261":3,"264":2,"266":1,"267":1,"296":1,"319":5,"331":6,"373":5,"375":3,"435":2,"440":1,"453":1,"522":2}}],["feedback",{"0":{"4":1,"28":1},"2":{"4":1,"7":1,"28":1,"30":1,"31":1,"32":2,"678":1,"720":1}}],["fellow",{"2":{"34":1,"44":1,"49":1}}],["february",{"2":{"20":1,"27":1}}],["fatalf",{"2":{"357":2,"358":6,"360":8}}],["fatal",{"0":{"562":1},"2":{"357":2,"358":4,"360":4,"562":2}}],["faster",{"2":{"576":1}}],["fastest",{"2":{"540":1,"568":1}}],["fast",{"2":{"281":1,"283":1}}],["favourite",{"2":{"188":1}}],["favor",{"2":{"137":1,"138":1}}],["falsediscard",{"2":{"605":1}}],["false",{"2":{"129":2,"195":4,"416":3,"522":2,"541":1,"547":2,"552":1,"574":1,"576":1,"577":1,"605":3,"639":24,"736":1}}],["falls",{"2":{"638":1}}],["fallback",{"0":{"97":1,"254":1},"2":{"93":1,"97":6,"254":5}}],["fall",{"2":{"75":1,"97":1,"115":1,"254":1}}],["faults",{"2":{"111":1}}],["faulty",{"2":{"96":1,"165":1}}],["faucets",{"2":{"640":1}}],["faucet",{"0":{"228":1,"471":1,"710":1},"1":{"472":1,"473":1},"2":{"67":1,"69":1,"228":2,"312":2,"338":1,"352":1,"471":1,"472":3,"473":2,"528":1,"537":2,"710":4}}],["familiarity",{"2":{"67":1,"88":1}}],["family",{"2":{"39":2}}],["fact",{"2":{"95":1,"122":1,"135":1,"154":1,"160":1}}],["factories",{"2":{"76":1}}],["factory",{"2":{"72":1}}],["factors",{"2":{"36":1,"683":2}}],["facilitating",{"2":{"72":1}}],["facilitated",{"2":{"58":1}}],["facilitate",{"2":{"31":1,"412":1}}],["failed",{"2":{"357":2,"358":10,"360":8,"364":8,"365":4,"366":6,"389":2,"557":2,"559":2}}],["fail",{"2":{"196":1,"316":1}}],["failure",{"2":{"115":4,"607":2,"736":2,"738":2,"739":2,"740":2,"756":2}}],["fails",{"2":{"97":1,"416":1,"559":1}}],["failing",{"2":{"24":1}}],["fairly",{"2":{"10":1,"277":1}}],["fair",{"2":{"8":1,"18":1,"462":1}}],["faq",{"0":{"183":1,"392":1,"606":1,"755":1},"1":{"184":1,"393":1,"394":1,"395":1,"396":1,"397":1,"398":1,"399":1,"400":1,"401":1,"402":1,"403":1,"404":1,"405":1,"607":1,"756":1},"2":{"16":2}}],["frequently",{"2":{"725":1}}],["frequency",{"2":{"196":1,"200":3}}],["fresh",{"2":{"599":1}}],["freeze",{"2":{"195":1}}],["free",{"2":{"6":1,"34":1,"186":1,"438":2,"467":1,"523":2,"685":1,"687":1,"703":1,"720":1}}],["fraudulent",{"2":{"115":1,"595":1}}],["fraud",{"0":{"107":1,"409":1},"2":{"95":2,"107":1,"112":1,"115":3,"148":5,"149":1,"163":1,"164":4,"165":1,"168":2,"171":1,"241":2,"397":1,"398":3,"409":1,"413":1,"625":2}}],["frameworks",{"2":{"242":2}}],["framework",{"0":{"234":1},"2":{"440":1,"708":1}}],["frame",{"2":{"71":1}}],["friends",{"2":{"38":1,"39":2,"653":1}}],["friendly",{"2":{"35":2,"384":1}}],["from=$validator",{"2":{"745":2,"751":2}}],["from=",{"2":{"496":2}}],["from",{"0":{"63":1,"64":1,"197":1,"203":1,"265":1,"377":1,"384":1,"539":1,"564":1,"567":1},"1":{"64":1,"65":1,"565":1},"2":{"2":1,"4":1,"7":1,"12":1,"14":1,"15":1,"16":1,"26":2,"30":1,"31":1,"32":1,"37":1,"40":1,"41":1,"43":1,"44":1,"46":1,"48":1,"58":3,"59":1,"61":1,"62":1,"68":1,"71":2,"72":1,"74":1,"75":2,"76":1,"77":1,"78":8,"82":1,"94":3,"95":1,"96":1,"102":1,"106":1,"107":4,"111":8,"113":1,"115":8,"121":1,"131":1,"140":2,"141":2,"142":2,"145":2,"148":4,"157":1,"158":1,"159":1,"164":1,"171":1,"184":1,"187":2,"188":1,"189":1,"193":3,"196":2,"197":1,"199":1,"201":1,"203":4,"206":1,"210":1,"212":1,"213":1,"221":1,"240":2,"241":1,"242":2,"247":1,"248":1,"252":1,"265":1,"267":1,"270":2,"273":2,"279":1,"284":6,"304":1,"312":1,"315":3,"317":1,"318":1,"319":1,"321":3,"326":1,"328":1,"338":1,"340":1,"341":2,"350":1,"352":1,"356":2,"357":4,"358":7,"360":5,"361":1,"364":2,"366":4,"369":1,"373":3,"377":2,"381":2,"382":6,"384":2,"385":1,"389":4,"394":1,"397":1,"398":1,"400":1,"401":1,"402":2,"410":1,"413":1,"414":2,"416":2,"430":1,"437":2,"438":2,"440":1,"446":5,"461":1,"462":1,"469":2,"470":1,"472":1,"477":2,"483":1,"484":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,"560":1,"562":1,"564":3,"570":1,"577":1,"585":1,"593":2,"595":2,"596":3,"597":1,"600":1,"601":1,"607":1,"614":1,"622":2,"632":2,"640":1,"648":2,"671":1,"673":1,"692":2,"693":1,"704":2,"705":3,"710":1,"725":1,"739":2,"740":1,"748":1,"751":3,"756":2}}],["firewall",{"2":{"541":2,"552":2}}],["first",{"2":{"22":4,"37":1,"44":1,"51":1,"53":1,"71":1,"76":2,"78":1,"92":1,"110":3,"111":1,"112":1,"113":1,"115":1,"129":2,"141":2,"142":1,"145":2,"146":2,"148":6,"150":1,"157":1,"168":1,"189":1,"200":2,"206":2,"213":1,"220":1,"221":1,"243":1,"245":1,"261":1,"273":2,"276":1,"284":2,"331":1,"341":1,"352":1,"358":5,"359":1,"366":2,"374":2,"377":1,"445":1,"451":1,"453":1,"465":1,"494":3,"495":1,"500":1,"503":1,"512":1,"519":1,"526":1,"536":1,"555":1,"569":1,"615":1,"622":1,"667":1,"683":1,"684":1,"699":1,"715":1,"716":1,"743":1}}],["figment",{"2":{"508":1}}],["figure",{"2":{"410":1,"412":1,"415":2,"623":1}}],["figures",{"2":{"58":1}}],["five",{"2":{"445":1}}],["fit",{"0":{"402":1}}],["fits",{"2":{"241":1}}],["fields",{"2":{"595":1}}],["field",{"2":{"132":1,"140":2,"141":1,"145":1,"194":1,"358":1,"384":1,"577":1,"602":2}}],["fine",{"2":{"416":1}}],["finished",{"2":{"740":1}}],["finish",{"0":{"78":1},"2":{"360":1}}],["financial",{"2":{"38":1}}],["finality",{"2":{"238":1}}],["finalizecommit",{"2":{"607":2,"756":2}}],["finalize",{"2":{"30":1,"426":1}}],["finally",{"2":{"90":1,"96":1,"145":1,"156":1,"160":1,"374":1,"384":1,"503":1}}],["final",{"2":{"30":1,"199":1,"360":1,"413":3,"505":1,"507":2,"660":2}}],["finger",{"2":{"35":1}}],["finding",{"2":{"34":1,"38":1,"40":1,"42":2}}],["find",{"0":{"340":1,"342":1},"2":{"0":1,"4":1,"37":1,"38":1,"41":1,"54":1,"55":1,"75":2,"78":1,"83":1,"84":1,"94":3,"107":1,"112":2,"129":2,"146":2,"155":1,"200":2,"213":1,"218":1,"219":1,"285":1,"302":1,"307":1,"311":1,"312":1,"467":1,"470":1,"484":1,"505":1,"515":1,"529":1,"552":1,"563":1,"573":1,"576":1,"620":1,"632":1,"658":1,"659":1,"672":1,"693":2,"695":1,"709":1,"733":1}}],["fix",{"2":{"377":1,"563":1,"607":1,"756":1}}],["fixed",{"2":{"374":2,"375":2}}],["fixes",{"2":{"23":1}}],["fixing",{"2":{"2":1}}],["fill",{"2":{"195":2,"503":1}}],["filled",{"2":{"22":1}}],["filter",{"0":{"648":1},"2":{"271":2,"272":2,"365":2,"639":8,"645":4,"648":1}}],["filteropts",{"2":{"129":2,"146":2}}],["filterdatacommitmentstored",{"2":{"129":2,"146":2}}],["file=",{"2":{"503":2}}],["filecoin",{"2":{"405":1}}],["file",{"0":{"265":1,"660":1,"661":1},"2":{"71":3,"73":3,"79":1,"89":3,"187":1,"195":2,"196":2,"200":1,"203":1,"265":3,"343":1,"356":2,"369":1,"382":2,"384":1,"488":6,"491":2,"494":5,"495":4,"500":1,"502":1,"503":1,"504":1,"531":2,"536":1,"539":1,"540":1,"560":1,"562":2,"563":9,"568":1,"579":1,"585":5,"601":1,"604":1,"605":1,"640":9,"658":1,"659":2,"660":4,"661":2,"736":2,"738":2,"739":2,"740":1,"753":1}}],["filesystem",{"2":{"620":1}}],["fileserror",{"2":{"563":1}}],["files",{"0":{"73":1,"74":1,"563":1,"641":1},"2":{"3":1,"52":1,"72":1,"73":3,"74":1,"486":2,"488":2,"563":2,"622":1,"634":2,"660":4,"675":2}}],["four",{"2":{"153":1,"238":1,"413":3,"494":1}}],["foundry",{"2":{"101":1,"187":2,"189":1,"193":2}}],["found",{"2":{"71":2,"82":1,"89":1,"95":1,"97":1,"107":1,"111":1,"121":1,"122":1,"141":1,"142":1,"144":1,"145":1,"153":1,"154":1,"157":1,"160":2,"167":2,"193":1,"195":1,"202":1,"215":1,"218":1,"222":1,"270":1,"271":2,"272":2,"316":1,"336":1,"338":2,"350":1,"352":2,"364":1,"365":2,"376":2,"441":1,"470":1,"494":2,"595":3,"647":1,"682":1}}],["foundation",{"0":{"17":1,"19":1},"1":{"18":1,"19":1,"20":2,"21":2,"22":2,"23":2,"24":2,"25":1,"26":1,"27":1,"28":1},"2":{"18":2,"20":1,"21":2,"23":2,"24":1,"26":2,"27":1,"445":1}}],["folders",{"2":{"515":4,"599":1,"639":1}}],["folder",{"2":{"79":1,"95":1,"199":2,"200":4,"539":1}}],["followed",{"2":{"112":1,"186":1,"197":1}}],["follows",{"2":{"23":1,"111":1,"139":1,"141":2,"142":2,"143":1,"144":1,"147":1,"148":1,"154":1,"160":1,"176":1,"179":1,"314":1,"503":1,"554":1,"647":1,"683":1}}],["follow",{"2":{"11":1,"29":2,"32":1,"42":1,"75":1,"89":1,"184":1,"198":1,"202":1,"203":1,"369":2,"384":1,"413":1,"480":1,"482":1,"487":1,"504":1,"540":1,"568":1,"583":1,"584":1,"597":1,"611":1,"612":1,"622":2,"628":1,"629":1,"635":1,"638":1,"640":1,"649":1,"650":1,"653":1,"676":1,"699":1,"743":1,"744":1,"747":1,"750":1,"751":1,"753":1}}],["following",{"2":{"1":1,"26":1,"35":1,"36":1,"46":1,"47":1,"48":1,"70":1,"78":2,"82":1,"94":2,"101":1,"110":1,"121":1,"125":1,"139":1,"141":2,"144":1,"153":1,"157":1,"163":1,"164":1,"172":1,"176":1,"184":1,"186":1,"187":2,"196":3,"199":1,"201":1,"213":2,"216":1,"224":1,"243":1,"245":1,"246":1,"248":1,"253":1,"262":1,"266":1,"269":1,"289":1,"308":1,"312":4,"315":1,"316":1,"317":1,"318":1,"319":3,"321":3,"332":1,"333":2,"351":1,"352":1,"353":1,"355":1,"360":1,"363":1,"383":1,"389":1,"390":1,"391":1,"410":1,"412":1,"413":1,"415":1,"422":1,"423":1,"426":2,"427":1,"448":1,"449":1,"472":1,"477":1,"478":1,"479":1,"483":1,"484":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,"555":1,"560":2,"567":1,"573":1,"581":1,"582":1,"585":1,"587":1,"596":1,"617":1,"626":1,"627":1,"631":1,"632":2,"637":1,"639":1,"641":1,"643":1,"644":1,"649":1,"651":1,"652":1,"654":1,"655":1,"656":1,"657":1,"658":1,"659":1,"660":3,"661":2,"662":3,"663":1,"665":1,"666":1,"670":1,"672":3,"673":1,"683":1,"697":1,"710":1,"711":1,"742":1,"743":1,"748":1,"749":1,"751":1}}],["foster",{"2":{"46":1,"58":1}}],["fosters",{"2":{"37":1}}],["fostering",{"2":{"18":1,"35":1,"43":1}}],["focus",{"2":{"454":1,"461":1,"543":1}}],["focuses",{"2":{"298":1}}],["focused",{"2":{"36":2,"211":1,"461":1,"725":1,"727":1}}],["focusing",{"2":{"7":2,"37":1}}],["foods",{"2":{"35":1}}],["food",{"2":{"35":2}}],["fortunately",{"2":{"503":1}}],["forest",{"2":{"415":1}}],["forever",{"2":{"403":1}}],["forward",{"2":{"187":1,"736":1}}],["forget",{"2":{"64":1,"519":2}}],["forge",{"2":{"36":1,"187":2,"188":6,"195":2}}],["forces",{"2":{"36":1}}],["former",{"2":{"414":1}}],["formerly",{"2":{"206":1}}],["formula",{"2":{"376":1}}],["formatting",{"0":{"314":1}}],["format",{"2":{"139":1,"140":1,"187":2,"247":1,"314":1,"319":3,"321":1,"325":1,"328":1,"331":1,"358":1,"384":2,"491":1,"494":1,"531":2}}],["formal",{"2":{"36":1}}],["forms",{"2":{"30":1,"31":1,"110":1}}],["form",{"0":{"47":1},"2":{"26":1,"28":1,"47":1,"71":1,"203":1,"446":1}}],["fork",{"2":{"2":3,"377":1}}],["for",{"0":{"42":1,"75":1,"76":1,"107":1,"147":1,"246":1,"247":1,"248":1,"249":1,"255":1,"261":1,"264":1,"275":1,"287":1,"296":1,"397":1,"398":1,"411":1,"421":1,"430":1,"433":1,"438":1,"453":1,"469":1,"470":1,"500":1,"565":1,"598":1,"659":1,"692":1,"693":1,"702":1,"704":1,"705":1,"723":1,"726":1,"728":1},"1":{"256":1,"257":1,"258":1,"259":1,"260":1,"261":1,"262":1,"263":1,"264":1,"265":2,"266":1,"267":1,"276":1,"277":1,"278":1,"279":1,"280":1,"281":1,"282":1,"288":1,"289":1,"290":1,"291":1,"292":1,"293":1,"294":1,"295":1,"296":1,"412":1,"434":1,"435":1,"599":1,"600":1,"601":1,"602":1,"603":1,"604":1,"605":1,"703":1,"704":1,"705":1},"2":{"0":1,"2":1,"3":4,"4":1,"6":1,"7":3,"8":2,"10":1,"11":1,"13":2,"14":1,"16":1,"18":1,"21":2,"22":3,"23":5,"24":2,"25":3,"26":3,"27":2,"30":3,"31":1,"32":4,"33":1,"34":13,"35":6,"36":4,"37":6,"38":4,"39":2,"40":5,"41":8,"42":3,"43":2,"44":1,"46":2,"47":2,"50":1,"51":1,"54":2,"55":1,"57":3,"58":1,"62":3,"66":1,"67":1,"69":2,"71":15,"72":1,"73":5,"75":6,"76":13,"77":1,"78":14,"81":4,"89":1,"92":2,"94":11,"95":5,"97":3,"102":1,"103":1,"105":1,"107":1,"109":1,"110":2,"111":12,"112":4,"113":3,"115":3,"119":2,"121":8,"122":1,"124":2,"125":1,"127":2,"129":2,"132":2,"136":1,"137":3,"138":1,"140":1,"145":2,"146":4,"147":4,"148":13,"150":1,"155":1,"156":2,"157":1,"159":1,"161":1,"164":1,"165":1,"172":1,"179":1,"182":1,"187":17,"189":2,"190":2,"191":1,"192":1,"193":1,"195":3,"196":10,"197":1,"199":1,"200":7,"201":2,"203":2,"204":2,"206":3,"209":1,"211":1,"212":2,"213":2,"214":1,"218":3,"219":3,"220":2,"222":1,"223":2,"224":2,"230":1,"232":3,"234":1,"241":1,"242":2,"244":1,"245":1,"246":5,"252":5,"253":1,"254":2,"256":1,"258":2,"259":2,"260":1,"261":1,"266":2,"275":1,"277":2,"278":1,"279":2,"280":1,"284":4,"285":1,"296":4,"297":1,"299":1,"300":1,"302":1,"303":1,"307":1,"308":3,"311":3,"312":5,"314":3,"315":1,"317":1,"318":1,"319":1,"321":2,"322":2,"324":1,"325":6,"333":1,"336":1,"339":1,"340":1,"341":1,"343":1,"346":1,"347":2,"348":2,"350":1,"351":1,"352":3,"353":2,"356":2,"358":1,"360":1,"361":2,"368":1,"369":4,"371":3,"373":2,"374":3,"375":12,"376":4,"377":1,"381":1,"382":4,"384":3,"385":1,"389":2,"393":2,"394":1,"395":3,"396":4,"397":3,"399":1,"400":1,"401":1,"403":2,"404":2,"405":2,"406":1,"407":4,"408":3,"409":2,"410":7,"412":2,"413":4,"414":2,"415":3,"416":5,"422":1,"426":1,"436":1,"437":1,"438":8,"441":2,"445":2,"446":2,"448":1,"449":1,"453":2,"454":1,"455":1,"459":1,"460":1,"461":2,"465":1,"466":2,"467":1,"469":1,"470":5,"474":1,"475":1,"478":3,"480":1,"482":1,"483":1,"484":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,"554":2,"561":1,"562":1,"563":2,"564":1,"565":2,"567":1,"568":2,"573":4,"574":2,"577":1,"579":3,"581":3,"585":5,"591":1,"592":1,"593":1,"594":1,"595":1,"597":1,"600":1,"602":2,"604":2,"605":3,"607":3,"609":1,"611":1,"612":3,"613":2,"614":2,"615":3,"616":1,"620":3,"621":1,"622":2,"623":1,"626":3,"628":1,"629":1,"632":11,"637":2,"638":1,"639":2,"640":4,"641":2,"645":2,"647":1,"649":1,"654":1,"655":3,"656":1,"658":2,"659":1,"660":1,"663":1,"665":1,"666":3,"671":3,"672":3,"678":2,"682":1,"683":1,"684":1,"685":3,"687":1,"692":1,"693":5,"695":1,"696":1,"697":1,"699":3,"703":3,"704":1,"705":3,"707":1,"709":1,"711":1,"712":1,"714":1,"715":2,"722":1,"725":1,"734":1,"736":1,"739":1,"742":3,"745":1,"746":1,"747":1,"748":1,"751":1,"754":2,"756":4}}],["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":{"336":1}}],["tb",{"2":{"478":1,"581":1,"593":1,"626":1,"719":2,"720":2,"742":1}}],["tbd",{"2":{"277":1}}],["tcp",{"2":{"128":2,"129":2,"252":2,"253":2,"326":6,"467":4,"531":2,"541":2,"542":4,"552":1,"559":2,"601":4,"739":2}}],["tj9wswcyhm0r0i8t+p3hpftdieuyr9wspvkl1r2s=",{"2":{"127":2}}],["tmproto",{"2":{"141":2,"148":2,"382":6}}],["tm",{"2":{"125":2,"277":1,"607":14,"756":14}}],["txtsudo",{"2":{"499":1}}],["txt",{"2":{"491":1,"499":5,"585":8}}],["txtcelestia",{"2":{"491":1}}],["txoption",{"2":{"382":2}}],["txhashcelestia",{"2":{"331":1}}],["txhash",{"2":{"133":4,"135":2,"331":1,"522":2,"745":3,"751":2}}],["txsizecost",{"2":{"375":1}}],["txsizecostperbyte",{"2":{"374":1}}],["txsharerange",{"2":{"157":1}}],["txs",{"2":{"110":4,"113":6,"135":2,"488":2}}],["tx",{"2":{"78":6,"129":14,"132":5,"133":5,"135":6,"146":4,"148":2,"266":2,"340":2,"341":2,"371":1,"373":1,"374":1,"379":2,"415":1,"488":4,"493":6,"494":6,"495":2,"496":2,"497":4,"498":2,"507":10,"522":8,"529":4,"604":4,"639":10,"647":2,"745":7,"751":6}}],["ts",{"2":{"76":1}}],["tuning",{"2":{"416":1}}],["tuplerootnonce",{"0":{"146":1},"2":{"139":2,"145":5}}],["tuple",{"2":{"103":10,"104":1,"112":1,"113":1,"124":4,"126":3,"129":2,"139":14,"145":17,"149":1,"153":3,"154":1,"159":1,"160":4,"184":1}}],["tutorial",{"0":{"268":1,"301":1,"303":1,"304":1,"362":1},"1":{"269":1,"270":1,"271":1,"272":1,"273":1,"274":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,"338":1,"363":1,"364":1,"365":1,"366":1,"367":1},"2":{"66":1,"243":3,"268":1,"279":1,"301":1,"303":1,"304":1,"313":1,"337":1,"350":2,"352":2,"361":1,"362":1,"380":1,"381":1,"417":1,"476":1,"479":1,"480":1,"482":1,"487":1,"511":1,"526":2,"530":1,"535":1,"538":1,"539":1,"543":5,"567":1,"569":1,"582":1,"584":1,"597":1,"609":1,"611":1,"614":1,"624":1,"627":1,"628":1,"629":1,"635":1,"639":1,"664":1,"668":1,"676":1,"677":1,"717":1,"741":1,"743":1,"744":1,"747":1,"750":1,"751":1}}],["tutorials",{"2":{"0":1,"3":1,"23":1,"232":1,"720":1}}],["turned",{"2":{"590":1}}],["turn",{"2":{"42":1,"61":1,"184":1}}],["twitch",{"2":{"40":1}}],["twitter",{"2":{"26":1,"39":2,"40":1}}],["two",{"2":{"39":1,"59":1,"64":1,"122":2,"130":1,"136":1,"151":1,"153":1,"154":1,"160":1,"189":2,"200":2,"214":1,"238":1,"259":1,"284":1,"297":1,"370":1,"406":1,"410":1,"415":1,"434":1,"477":1,"486":1,"494":1,"594":1,"595":1,"634":1,"637":1,"644":1,"673":1,"675":1,"715":1,"724":1,"733":1}}],["tar",{"2":{"596":24,"622":4,"623":32}}],["targets",{"2":{"503":6}}],["targetblock",{"2":{"204":1}}],["target",{"2":{"125":1,"186":1,"187":1,"189":1,"195":1,"196":3,"200":3,"201":2,"206":1,"441":1,"488":2,"562":1,"736":4,"738":4,"739":4,"740":4}}],["targeting",{"2":{"75":1}}],["tasked",{"2":{"398":1}}],["tasks",{"2":{"78":1,"413":1,"622":1}}],["taking",{"2":{"94":1,"141":1,"142":1,"145":1,"154":1,"594":1}}],["taken",{"2":{"199":1}}],["takes",{"2":{"94":1,"126":1,"136":1,"199":2,"200":1,"202":1,"270":2,"273":1,"319":1,"358":1,"364":2,"366":1}}],["takeaways",{"2":{"32":1}}],["take",{"2":{"8":1,"47":1,"94":1,"107":1,"110":1,"139":1,"140":1,"144":1,"160":1,"176":1,"179":1,"184":1,"188":1,"407":1,"412":1,"416":1,"530":1,"551":1,"554":1,"579":1,"595":1,"716":1}}],["tab=staking",{"2":{"448":1}}],["table",{"2":{"71":1,"445":1,"446":1,"470":1}}],["tab",{"2":{"62":2,"64":2,"193":1,"228":1,"246":1}}],["tag",{"2":{"39":1,"252":2,"539":6,"567":6,"614":6,"616":6,"617":6}}],["tags",{"2":{"30":1,"539":6,"567":6}}],["talented",{"2":{"37":1}}],["talks",{"2":{"34":1,"37":1,"40":1,"42":5}}],["talk",{"2":{"34":1,"37":1}}],["tailored",{"2":{"47":1,"58":1}}],["tailor",{"2":{"36":1,"38":1,"42":1}}],["t",{"0":{"403":1},"2":{"26":1,"29":1,"34":1,"37":1,"39":1,"40":3,"64":1,"75":1,"76":1,"94":2,"115":1,"121":1,"129":2,"140":1,"141":1,"142":1,"144":1,"145":1,"146":2,"159":1,"168":1,"184":1,"187":3,"193":1,"218":2,"240":1,"241":1,"242":2,"296":1,"302":2,"315":1,"338":1,"344":1,"346":1,"369":1,"382":2,"389":2,"397":1,"398":2,"403":2,"449":1,"515":4,"527":1,"529":1,"569":1,"597":1,"601":1,"604":1,"607":18,"624":1,"638":1,"642":1,"683":1,"706":1,"707":1,"725":1,"733":1,"751":1,"756":19}}],["title",{"2":{"494":6,"495":2}}],["tiny",{"2":{"391":1}}],["tia",{"0":{"417":1,"419":1,"422":1,"423":1,"424":1,"426":1,"427":1,"428":1,"431":1,"432":1,"445":1,"450":1,"451":1,"452":1,"458":1},"1":{"418":1,"419":1,"420":2,"421":2,"422":2,"423":2,"424":1,"425":2,"426":2,"427":2,"428":1,"429":2,"430":2,"431":2,"432":2,"446":1,"451":1,"452":1,"453":2,"454":2,"455":2,"456":2,"457":2,"458":2,"459":2,"460":2},"2":{"331":2,"389":6,"391":1,"417":3,"422":1,"440":1,"441":3,"443":1,"444":1,"445":1,"446":3,"451":2,"453":2,"454":1,"455":1,"456":1,"458":1,"459":2}}],["ti",{"2":{"252":2,"253":2}}],["tips",{"2":{"436":1}}],["tip",{"2":{"76":1,"201":1,"202":1,"246":1,"252":1,"307":1,"310":1,"311":1,"312":1,"314":1,"318":1,"341":2,"346":1,"347":1,"348":1,"352":1,"356":1,"358":1,"484":1,"522":2,"537":1,"550":1,"560":2,"593":2,"623":1,"632":1,"638":1,"663":1,"671":1,"693":1,"748":1,"756":1}}],["timing",{"2":{"58":1,"377":1}}],["timeseries",{"2":{"503":2}}],["times",{"2":{"407":1,"714":1,"725":1}}],["timestamp",{"2":{"328":2,"340":2,"441":1,"522":3,"745":2,"751":2}}],["timeout",{"2":{"382":2,"522":2,"639":4}}],["timely",{"2":{"24":1}}],["time",{"2":{"13":1,"14":1,"30":1,"39":1,"71":2,"75":1,"115":1,"196":2,"202":1,"218":1,"240":1,"282":1,"296":1,"328":2,"343":1,"356":1,"360":1,"374":1,"377":2,"382":2,"400":1,"415":1,"441":1,"503":2,"509":1,"515":1,"522":2,"523":4,"530":1,"569":1,"593":2,"639":4,"641":1,"715":1,"716":2,"736":1,"740":1}}],["tiers",{"2":{"22":1}}],["tier",{"2":{"22":1,"58":1}}],["tr",{"2":{"585":10,"596":6}}],["tryfinalizecommit",{"2":{"607":2,"756":2}}],["try",{"2":{"314":1,"338":1,"352":1,"361":1,"389":2,"585":1,"648":1}}],["trying",{"2":{"94":1,"338":1,"562":1,"563":1}}],["treated",{"2":{"195":1,"374":1}}],["trees",{"0":{"410":1},"2":{"307":1,"406":1,"410":1}}],["tree",{"2":{"95":2,"103":2,"104":1,"124":2,"141":6,"144":6,"173":1,"410":2,"412":2,"415":1}}],["trpc",{"2":{"121":11,"128":4,"129":8,"133":2,"135":4,"138":2,"148":2}}],["tries",{"2":{"416":1}}],["triggered",{"2":{"254":3}}],["triggering",{"2":{"164":1}}],["trigger",{"2":{"164":1}}],["trick",{"2":{"95":1}}],["tripods",{"2":{"34":1}}],["truth",{"2":{"124":1}}],["true",{"2":{"76":6,"103":2,"129":2,"132":1,"133":2,"135":2,"195":1,"196":2,"321":1,"348":3,"502":4,"541":2,"547":1,"552":2,"576":1,"577":1,"595":3,"605":1,"639":10}}],["trusts",{"2":{"579":1}}],["trustedpeers",{"2":{"579":1}}],["trustedpeer",{"0":{"579":1}}],["trustedhash",{"0":{"579":1},"2":{"579":1}}],["trusted",{"0":{"194":1,"564":1},"1":{"565":1},"2":{"50":1,"193":3,"194":6,"195":6,"199":1,"477":2,"554":1,"564":8,"579":5,"595":2,"639":4,"687":1,"688":1,"689":1,"706":1,"707":1,"708":1}}],["trust",{"2":{"39":1,"406":1,"416":1,"564":3,"595":4,"639":4}}],["trustworthy",{"2":{"18":1}}],["troubles",{"2":{"693":1}}],["troubleshooting",{"0":{"338":1,"549":1,"620":1},"1":{"550":1,"551":1,"552":1,"553":1,"554":1,"555":1,"556":1,"557":1,"558":1,"559":1,"560":1,"561":1,"562":1,"563":1},"2":{"54":1,"80":1,"246":1,"252":2,"311":1,"312":1,"314":1,"483":1,"484":1,"537":1,"570":1,"614":1,"632":2,"663":1,"671":1,"748":1}}],["trolling",{"2":{"7":1}}],["traditional",{"2":{"173":1,"284":1,"413":1}}],["tracking",{"2":{"565":1}}],["trackers",{"2":{"504":1}}],["track",{"2":{"30":1,"95":1,"112":1,"544":1}}],["transcript15",{"2":{"199":4}}],["transcript14",{"2":{"199":4}}],["transcript13",{"2":{"199":4}}],["transcript12",{"2":{"199":4}}],["transcript11",{"2":{"199":4}}],["transcript06",{"2":{"199":4}}],["transcript05",{"2":{"199":4}}],["transcript04",{"2":{"199":4}}],["transact",{"2":{"254":1,"448":1}}],["transactopts",{"2":{"148":2}}],["transactionindex",{"2":{"341":2}}],["transaction",{"0":{"130":1,"131":2,"134":1,"184":1,"340":1,"341":1,"385":1,"415":1,"588":1,"604":1,"659":1,"753":1},"1":{"132":2,"133":2,"135":1,"386":1,"387":1,"416":1},"2":{"62":4,"64":3,"65":3,"70":3,"71":3,"72":4,"78":14,"112":1,"115":1,"122":3,"129":8,"130":4,"131":3,"132":3,"133":2,"134":1,"135":4,"140":1,"141":1,"146":2,"148":4,"149":1,"152":1,"157":2,"160":1,"165":2,"176":1,"184":13,"187":3,"189":2,"238":2,"241":2,"242":1,"254":1,"256":1,"266":2,"291":1,"296":1,"319":3,"321":3,"322":1,"331":1,"336":1,"340":4,"341":2,"370":1,"371":5,"373":2,"374":3,"375":10,"376":1,"377":8,"382":14,"384":1,"386":2,"393":2,"394":3,"395":1,"399":3,"400":3,"401":2,"402":2,"404":2,"405":2,"406":1,"413":1,"415":3,"422":1,"423":1,"426":2,"427":1,"432":1,"434":4,"435":2,"506":1,"507":4,"511":1,"522":3,"529":1,"531":2,"587":1,"590":1,"659":1,"683":4,"751":3,"753":1}}],["transactions",{"0":{"296":1,"377":1,"434":1,"590":1},"2":{"59":1,"62":2,"64":1,"65":1,"66":1,"72":2,"73":1,"77":1,"112":1,"115":1,"119":1,"122":1,"135":1,"212":1,"224":1,"237":3,"238":2,"241":1,"254":1,"260":1,"277":1,"278":1,"296":3,"306":1,"311":1,"312":1,"340":3,"350":1,"370":2,"371":2,"372":2,"377":1,"382":4,"385":3,"386":1,"387":2,"390":1,"391":1,"393":1,"400":1,"406":1,"407":1,"408":1,"412":1,"413":6,"414":1,"415":6,"416":2,"434":2,"435":2,"453":1,"459":1,"470":1,"483":1,"484":1,"488":2,"508":1,"590":1,"604":5,"605":1,"632":3,"671":1,"672":1,"677":1,"693":1,"705":1,"706":1,"708":1,"753":1}}],["transferred",{"2":{"647":1}}],["transferring",{"2":{"78":2}}],["transferhermes",{"2":{"644":1}}],["transfer",{"0":{"331":1,"647":1},"1":{"648":1},"2":{"71":1,"227":1,"331":7,"446":1,"493":1,"637":1,"639":4,"644":7,"645":4,"647":6,"648":1}}],["transitions",{"2":{"115":1,"164":1,"172":1,"176":1,"179":1,"413":1}}],["transition",{"2":{"18":1,"148":4,"164":2,"168":1}}],["translations",{"2":{"16":2}}],["tee",{"2":{"736":2,"738":2,"739":2,"740":2}}],["territories",{"2":{"679":1}}],["term",{"2":{"441":1}}],["terminal",{"2":{"187":1,"336":1,"352":1,"384":1,"540":2,"568":2,"636":1,"670":1}}],["terms",{"2":{"13":1,"14":1,"431":1}}],["teach",{"2":{"313":1}}],["team",{"2":{"26":1,"30":1,"35":1,"37":1,"39":1,"46":1,"66":1,"78":1,"155":1,"176":1,"179":1,"182":1,"202":1,"203":1,"477":1,"714":1,"715":1}}],["text|json",{"2":{"531":2}}],["text",{"2":{"319":2,"321":1,"494":1,"495":1,"531":2,"661":1}}],["text$request",{"2":{"312":1,"472":1,"537":1,"710":1}}],["texthttps",{"2":{"121":2}}],["teststaking",{"2":{"659":1}}],["testtia",{"2":{"657":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}}],["testkey2",{"2":{"291":2,"292":1}}],["testkey2celestia",{"2":{"291":1}}],["testkey",{"2":{"289":2,"294":2,"656":1}}],["testkeycelestia",{"2":{"289":1}}],["testing",{"2":{"226":1,"461":3,"462":1,"678":1,"695":1,"715":1,"722":1}}],["test",{"2":{"141":1,"142":1,"144":1,"145":1,"199":4,"223":2,"246":6,"247":6,"248":6,"252":4,"253":4,"266":2,"286":1,"294":2,"312":4,"355":4,"484":3,"485":1,"489":1,"503":1,"507":6,"517":1,"527":1,"531":2,"536":1,"632":3,"633":1,"639":4,"641":1,"649":2,"656":1,"657":1,"659":1,"672":4,"674":1,"699":2,"727":1,"731":1,"733":1}}],["tested",{"2":{"81":1,"280":1,"343":1}}],["testnetwork",{"2":{"649":1}}],["testnets",{"0":{"724":1},"1":{"725":1,"726":1,"727":1,"728":1},"2":{"67":1,"88":1,"242":1,"389":1,"391":1,"640":1,"715":2,"724":1}}],["testnet",{"0":{"76":1,"221":1,"348":1,"449":1,"546":1,"653":1,"673":1,"699":1,"710":1,"727":1,"728":1},"1":{"222":1,"223":1,"224":1,"225":1,"226":1,"227":1,"228":1,"229":1,"230":1,"231":1,"654":1,"655":1,"656":1,"657":1,"658":1,"659":1,"660":1,"661":1,"662":1,"663":1,"700":1,"701":1,"702":1,"703":1,"704":1,"705":1,"706":1,"707":1,"708":1,"709":1,"710":1,"711":1,"712":1,"713":1},"2":{"23":3,"26":1,"61":1,"66":2,"67":3,"69":5,"72":5,"75":1,"82":1,"121":1,"216":3,"221":2,"222":5,"223":1,"224":1,"226":3,"227":2,"228":1,"231":1,"252":1,"256":1,"257":1,"286":2,"311":1,"312":2,"338":2,"344":2,"346":1,"348":2,"352":1,"389":2,"391":5,"445":1,"449":4,"461":3,"462":1,"483":1,"484":4,"511":1,"517":1,"525":1,"526":2,"529":1,"533":1,"546":1,"585":1,"632":4,"637":2,"639":11,"640":3,"642":1,"643":6,"644":8,"645":3,"649":1,"653":2,"655":1,"658":2,"663":1,"671":1,"672":1,"673":2,"699":1,"705":6,"706":8,"707":8,"708":8,"710":1,"711":2,"712":8,"713":1,"715":1,"727":2,"733":1,"748":1}}],["temp",{"2":{"540":1,"568":1}}],["temporarily",{"2":{"509":1}}],["temporary",{"0":{"14":1},"2":{"13":1,"14":1,"254":1,"517":1}}],["template",{"2":{"198":1,"199":2,"343":1}}],["templates",{"2":{"76":1}}],["tendermintrpc",{"2":{"94":1}}],["tendermint",{"2":{"76":3,"121":1,"125":4,"127":1,"128":4,"129":10,"132":1,"135":4,"137":1,"148":22,"194":1,"196":1,"203":1,"277":2,"278":1,"279":1,"281":1,"282":2,"328":4,"382":6,"412":6,"477":1,"488":8,"500":2,"531":2,"599":2,"607":29,"638":2,"644":8,"654":2,"662":2,"751":2,"756":30}}],["telemetry",{"2":{"639":6}}],["telegram",{"2":{"39":2,"475":1,"505":1,"698":1,"713":1,"729":1}}],["tell",{"2":{"42":1}}],["tech",{"2":{"36":2,"38":1,"92":1,"687":2,"688":2,"689":2,"690":2,"693":1,"696":1,"705":2,"706":1,"707":1,"708":1,"712":1}}],["technologies",{"2":{"42":1,"53":1,"241":1,"679":1}}],["technology",{"2":{"7":1,"36":1,"350":1}}],["technical",{"2":{"3":1,"26":1,"30":1,"42":1}}],["typically",{"2":{"209":1,"458":1}}],["typical",{"2":{"112":1,"238":1,"282":2}}],["type=$node",{"2":{"614":6,"616":6,"617":6}}],["type=full",{"2":{"614":1}}],["type=fullexport",{"2":{"614":1}}],["type=bridge",{"2":{"614":1}}],["type=bridgeexport",{"2":{"614":1}}],["type=1",{"2":{"343":1}}],["type=spanbatch",{"2":{"343":1}}],["type=lightexport",{"2":{"614":1}}],["type=light",{"2":{"252":2,"253":2,"614":1}}],["type=ledger`",{"2":{"187":2}}],["type=private",{"2":{"187":4}}],["type",{"0":{"70":1},"2":{"110":8,"111":7,"132":4,"141":2,"144":2,"163":1,"187":4,"237":1,"244":1,"246":7,"247":6,"248":6,"252":2,"253":6,"294":2,"312":4,"316":4,"328":4,"336":2,"342":2,"343":2,"358":3,"360":2,"390":1,"391":1,"394":2,"465":1,"467":2,"470":2,"484":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,"554":2,"555":12,"560":4,"561":2,"562":4,"569":1,"573":1,"614":13,"616":8,"617":8,"623":1,"632":2,"639":12,"672":4,"684":1,"693":2,"699":3,"717":1,"719":1,"720":1}}],["typescargo",{"2":{"363":1}}],["types",{"0":{"593":1},"1":{"594":1,"595":1,"596":1},"2":{"3":1,"36":1,"148":6,"158":1,"168":1,"241":1,"270":4,"310":1,"356":4,"363":2,"364":4,"376":2,"381":4,"382":4,"404":1,"465":1,"467":1,"541":1,"552":1,"554":1,"569":1,"620":1,"684":2,"699":1,"731":1}}],["typography",{"2":{"52":1}}],["typos",{"2":{"2":1}}],["though",{"2":{"413":1,"415":1}}],["thought",{"2":{"110":2}}],["thorough",{"2":{"170":1}}],["those",{"2":{"7":1,"13":1,"14":1,"33":1,"34":1,"40":3,"43":1,"110":1,"111":1,"124":1,"153":1,"184":1,"189":2,"237":2,"238":1,"277":1,"281":1,"296":1,"339":1,"402":1,"404":1,"407":1,"494":1,"660":1,"699":1,"715":3}}],["thus",{"2":{"75":1,"76":1,"94":1,"96":1,"409":1,"413":3,"415":2}}],["thank",{"2":{"38":1,"66":1}}],["thanks",{"2":{"33":1,"93":1}}],["thanking",{"2":{"32":1}}],["than",{"2":{"24":1,"184":1,"211":1,"238":1,"246":1,"282":1,"284":1,"371":2,"372":1,"402":1,"437":1,"438":1,"441":2,"502":2,"509":1,"529":1,"555":1,"563":1,"683":1}}],["that",{"0":{"129":1,"148":1,"164":1,"165":1,"166":1,"167":1,"168":1,"169":1,"170":1,"172":1,"173":1,"174":1,"175":1,"177":1,"178":1,"397":1},"1":{"165":1,"166":1,"167":1,"169":1,"170":1,"173":1,"174":1,"176":1,"177":1,"178":1},"2":{"6":1,"7":1,"8":2,"18":1,"21":1,"23":4,"29":1,"30":1,"31":1,"34":2,"35":1,"36":1,"38":1,"42":1,"47":2,"58":3,"71":3,"72":2,"74":1,"76":6,"78":2,"81":1,"89":2,"90":1,"92":1,"94":3,"95":4,"96":4,"102":2,"103":4,"105":2,"106":1,"107":13,"109":2,"110":18,"111":18,"112":4,"113":2,"115":6,"121":3,"122":3,"124":14,"126":1,"129":6,"130":2,"134":1,"139":13,"140":2,"141":1,"142":1,"145":9,"146":2,"148":12,"149":2,"150":3,"151":2,"152":1,"153":7,"154":1,"155":1,"157":1,"159":7,"160":8,"163":6,"164":6,"165":2,"167":1,"168":4,"170":1,"171":1,"172":4,"173":2,"176":3,"179":6,"182":2,"184":1,"187":6,"188":1,"189":3,"190":3,"191":1,"193":1,"194":1,"195":3,"196":1,"197":1,"199":1,"200":7,"201":1,"202":1,"203":1,"204":1,"206":1,"210":1,"212":4,"213":2,"218":3,"219":1,"220":1,"222":1,"223":1,"224":1,"227":1,"231":1,"232":1,"234":1,"237":2,"238":2,"240":1,"241":3,"242":3,"243":1,"244":1,"246":3,"252":1,"254":1,"258":3,"261":1,"262":1,"263":1,"267":3,"271":1,"272":1,"277":1,"279":1,"281":1,"282":4,"283":1,"284":1,"286":1,"311":1,"312":1,"314":5,"315":1,"316":1,"321":1,"331":1,"338":1,"340":1,"343":1,"344":1,"346":2,"350":2,"356":2,"357":1,"358":2,"360":1,"365":1,"371":1,"372":1,"373":3,"374":1,"375":5,"377":2,"382":6,"389":5,"390":1,"391":1,"393":4,"395":1,"396":1,"397":8,"398":2,"400":1,"401":1,"402":1,"403":1,"404":6,"405":1,"406":2,"407":3,"408":2,"409":1,"410":6,"412":4,"413":9,"414":4,"415":5,"416":4,"417":1,"426":1,"435":1,"436":5,"437":2,"438":2,"446":1,"454":1,"458":1,"461":2,"469":1,"470":1,"477":1,"483":1,"484":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,"557":1,"560":2,"562":1,"563":2,"564":4,"567":1,"568":1,"569":1,"572":1,"579":1,"585":2,"591":1,"593":1,"597":1,"605":1,"607":1,"608":1,"611":1,"612":2,"615":1,"622":1,"623":1,"624":1,"625":1,"632":2,"635":1,"637":2,"638":2,"640":3,"642":2,"644":1,"645":1,"660":1,"662":2,"665":1,"670":1,"672":1,"678":1,"679":1,"683":3,"692":1,"693":1,"699":1,"704":1,"705":3,"714":1,"716":1,"724":1,"732":1,"733":1,"736":1,"745":1,"750":1,"751":4,"756":1}}],["think",{"2":{"110":1}}],["things",{"0":{"360":1,"405":1},"2":{"78":2,"319":1}}],["third",{"2":{"22":4,"256":1,"258":1,"266":1,"275":1,"596":1}}],["this",{"0":{"402":1},"2":{"1":1,"2":2,"4":1,"8":1,"9":1,"11":1,"13":1,"14":1,"16":2,"22":4,"23":1,"26":1,"36":2,"37":3,"38":2,"39":1,"40":2,"41":1,"42":7,"43":1,"44":1,"46":1,"47":1,"49":1,"50":2,"51":2,"53":1,"54":2,"55":2,"57":1,"58":1,"59":2,"62":1,"66":4,"68":1,"69":2,"70":4,"71":7,"72":4,"73":2,"75":3,"76":1,"77":2,"78":2,"79":1,"81":3,"89":2,"90":1,"94":3,"95":3,"96":1,"102":1,"107":5,"109":4,"110":8,"111":6,"112":3,"113":3,"115":1,"124":4,"125":1,"126":2,"130":1,"132":1,"136":1,"140":3,"141":3,"145":3,"148":4,"150":1,"151":1,"152":1,"154":1,"156":2,"159":5,"160":3,"164":3,"168":2,"171":1,"172":2,"178":1,"184":3,"185":1,"187":2,"189":1,"190":1,"193":2,"198":1,"199":5,"200":5,"202":1,"203":1,"204":1,"206":2,"212":2,"222":1,"226":1,"232":2,"241":1,"242":1,"243":3,"245":1,"246":1,"252":1,"253":3,"254":2,"256":1,"262":1,"267":1,"268":1,"269":2,"271":2,"272":2,"273":1,"275":1,"277":1,"279":2,"281":1,"282":1,"284":3,"285":2,"286":2,"288":1,"289":1,"291":1,"296":2,"304":1,"307":1,"310":1,"311":1,"313":1,"314":2,"317":2,"318":2,"319":1,"321":3,"326":1,"327":1,"328":2,"331":1,"334":1,"336":1,"339":1,"340":1,"343":2,"344":1,"346":2,"350":2,"352":5,"355":1,"356":2,"358":1,"361":1,"362":1,"363":2,"365":2,"366":1,"368":1,"371":2,"373":1,"374":3,"375":3,"376":3,"377":4,"382":10,"385":1,"386":1,"388":1,"389":4,"393":1,"394":1,"395":1,"396":1,"397":2,"405":1,"408":2,"409":2,"410":2,"412":1,"413":4,"414":1,"415":2,"416":2,"417":2,"420":1,"435":1,"436":1,"437":2,"438":4,"446":1,"454":1,"459":1,"465":1,"470":1,"471":1,"476":1,"477":2,"484":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,"555":2,"557":3,"558":1,"560":1,"561":1,"563":5,"564":3,"567":4,"568":1,"574":2,"577":2,"580":1,"585":2,"587":1,"589":2,"591":1,"592":3,"594":1,"595":2,"597":1,"599":1,"600":1,"601":2,"602":1,"604":7,"605":3,"607":1,"609":1,"611":1,"612":2,"615":4,"616":1,"620":1,"621":2,"622":2,"624":1,"636":1,"637":2,"639":1,"642":2,"644":2,"649":2,"653":1,"655":1,"656":2,"659":1,"660":5,"664":1,"665":1,"668":1,"677":1,"678":1,"679":1,"683":1,"684":1,"693":1,"699":2,"707":1,"710":1,"714":1,"717":3,"722":1,"730":1,"733":1,"739":1,"741":1,"746":1,"751":2,"756":1}}],["thread",{"2":{"396":1}}],["threatening",{"2":{"8":1}}],["threshold",{"2":{"373":1,"506":1,"507":2,"639":4}}],["three",{"2":{"193":1,"214":1,"322":1,"410":2,"413":1}}],["throughput",{"2":{"206":1,"211":1,"212":1,"408":1,"413":2}}],["throughout",{"2":{"46":1,"350":1}}],["through",{"2":{"13":1,"32":1,"39":1,"65":1,"76":1,"78":2,"187":1,"219":1,"268":1,"350":1,"362":1,"374":1,"375":1,"386":1,"447":1,"456":1,"502":1,"503":1,"535":1,"538":1,"555":1,"624":1,"647":1,"664":1,"717":1,"738":1,"739":1,"741":1,"745":1}}],["theta",{"2":{"639":9,"640":2,"642":1,"643":6,"644":8,"645":3}}],["then",{"2":{"61":1,"62":1,"71":1,"72":1,"77":1,"94":1,"95":1,"109":2,"110":1,"112":2,"115":1,"124":5,"126":1,"133":1,"136":3,"141":1,"142":1,"147":1,"148":1,"149":1,"153":1,"159":1,"160":1,"164":2,"168":1,"176":3,"179":1,"182":1,"184":2,"187":3,"190":1,"193":1,"195":1,"196":2,"197":1,"199":1,"200":2,"201":1,"219":1,"220":2,"237":1,"248":1,"261":1,"266":1,"284":2,"317":1,"321":1,"331":1,"338":1,"357":1,"358":3,"360":2,"361":1,"371":1,"374":2,"375":1,"376":1,"377":1,"382":2,"384":1,"390":1,"391":1,"393":1,"407":2,"416":3,"426":1,"431":1,"434":1,"435":1,"491":2,"494":2,"500":3,"503":3,"542":1,"548":2,"555":1,"560":1,"562":1,"576":1,"593":1,"595":1,"597":1,"607":1,"622":2,"660":1,"716":2,"751":1,"756":1}}],["therefore",{"2":{"111":1,"382":2,"393":1,"408":1,"462":1}}],["there",{"0":{"404":1},"2":{"22":1,"36":1,"39":1,"44":1,"71":1,"115":1,"154":1,"168":1,"171":1,"184":1,"189":1,"200":1,"226":1,"238":1,"240":1,"241":1,"279":1,"284":1,"297":1,"319":1,"369":1,"371":1,"373":1,"397":1,"398":2,"403":1,"404":1,"407":1,"408":1,"413":1,"471":1,"474":1,"494":3,"551":1,"594":1,"637":1,"683":2,"695":1,"696":1,"698":1,"710":1,"712":1,"713":1,"717":1,"729":1}}],["these",{"2":{"11":1,"13":1,"14":1,"29":1,"33":1,"34":1,"35":1,"39":1,"44":1,"72":1,"73":1,"75":1,"76":1,"107":2,"110":1,"111":2,"122":1,"124":1,"127":2,"132":1,"137":1,"141":1,"144":1,"153":1,"154":2,"156":1,"160":2,"167":1,"172":2,"179":1,"193":1,"196":2,"200":1,"203":1,"214":1,"218":1,"238":1,"241":1,"276":1,"282":1,"315":2,"322":1,"358":1,"369":1,"374":2,"376":1,"384":1,"407":1,"413":2,"415":2,"438":1,"466":1,"467":1,"469":1,"470":1,"495":1,"503":2,"529":1,"540":1,"542":1,"568":1,"585":1,"595":1,"603":1,"605":1,"620":1,"622":2,"637":1,"653":1,"682":1,"683":1,"685":1,"692":1,"693":2,"703":1,"704":1,"705":1,"751":1}}],["they",{"2":{"7":1,"8":1,"11":1,"22":1,"34":1,"38":4,"39":2,"40":1,"41":1,"58":1,"71":2,"73":1,"115":1,"127":1,"132":1,"137":1,"144":1,"154":1,"163":2,"164":2,"168":2,"201":1,"202":1,"224":1,"237":2,"238":1,"240":1,"242":1,"271":1,"272":1,"280":1,"283":1,"314":1,"356":1,"365":1,"373":1,"375":1,"377":3,"385":1,"387":1,"389":2,"395":2,"397":2,"398":1,"404":1,"405":2,"408":1,"413":1,"438":1,"462":1,"469":1,"470":1,"495":1,"509":3,"642":1,"659":1,"660":1,"662":1,"665":2,"692":1,"693":1,"704":1,"705":1,"709":1,"727":1}}],["their",{"2":{"7":1,"27":1,"30":1,"32":1,"34":2,"36":2,"37":2,"38":6,"39":3,"40":2,"41":2,"42":1,"47":1,"58":1,"71":2,"109":1,"124":1,"149":1,"150":3,"151":1,"158":1,"163":1,"164":1,"187":1,"202":1,"206":1,"221":1,"223":1,"232":1,"240":2,"241":2,"242":3,"283":1,"284":1,"298":1,"300":1,"304":1,"307":1,"339":1,"373":1,"377":1,"387":1,"400":1,"404":1,"413":1,"414":2,"436":1,"438":4,"443":1,"454":3,"455":2,"461":1,"462":1,"469":1,"470":1,"503":2,"509":3,"548":1,"554":1,"590":1,"595":1,"658":1,"660":3,"692":1,"693":1,"699":2,"704":1,"705":1,"714":1,"715":2,"727":1}}],["theme",{"2":{"37":2}}],["them",{"2":{"2":1,"32":1,"34":1,"38":4,"39":1,"40":3,"41":3,"89":1,"94":1,"106":1,"124":1,"141":3,"144":1,"149":1,"153":1,"157":1,"167":2,"188":2,"193":1,"196":2,"200":1,"202":1,"220":1,"238":1,"284":2,"302":1,"307":1,"356":1,"406":1,"440":1,"465":1,"509":1,"552":1,"560":1,"565":1,"595":1,"659":1,"660":1,"684":1,"699":1,"705":1}}],["the",{"0":{"1":1,"17":1,"18":1,"30":1,"31":1,"32":1,"74":1,"75":1,"106":1,"115":1,"122":1,"123":1,"124":1,"131":1,"134":1,"139":2,"146":1,"148":1,"184":2,"186":1,"188":1,"189":2,"193":1,"194":1,"198":1,"199":1,"200":1,"203":1,"217":1,"222":1,"244":1,"267":1,"284":1,"322":1,"325":1,"341":1,"342":1,"354":1,"369":1,"377":1,"379":1,"380":1,"381":1,"382":1,"394":1,"397":1,"400":1,"401":1,"415":1,"430":1,"480":1,"481":1,"483":1,"484":1,"485":1,"487":1,"498":1,"513":1,"514":1,"515":1,"516":2,"523":1,"524":1,"525":2,"553":1,"561":1,"562":2,"583":1,"585":1,"597":1,"603":1,"616":1,"617":1,"628":1,"630":1,"631":1,"632":1,"633":1,"635":1,"636":1,"646":1,"659":1,"660":1,"663":1,"670":1,"671":1,"674":1,"721":1,"736":1,"746":1,"748":1,"749":1,"750":1,"751":1},"1":{"18":1,"19":1,"20":1,"21":1,"22":1,"23":1,"24":1,"25":1,"26":1,"27":1,"28":1,"123":1,"124":1,"132":1,"133":1,"135":1,"140":2,"141":2,"142":2,"143":2,"144":2,"145":2,"146":2,"147":2,"148":2,"187":1,"188":1,"189":1,"190":3,"191":3,"192":3,"193":1,"194":2,"195":2,"196":1,"197":1,"199":1,"200":1,"218":1,"219":1,"220":1,"245":1,"246":1,"247":1,"248":1,"249":1,"285":1,"355":1,"356":1,"357":1,"358":1,"359":1,"360":1,"416":1,"482":1,"483":1,"484":1,"485":2,"486":2,"487":1,"514":1,"515":1,"516":1,"517":1,"554":1,"555":1,"556":1,"557":1,"558":1,"559":1,"631":1,"632":1,"634":1,"672":1,"673":1,"674":1,"675":2,"676":1,"722":1,"723":1,"724":1,"725":1,"726":1,"727":1,"728":1,"729":1,"747":1,"748":1,"749":1,"750":2},"2":{"0":2,"1":2,"2":3,"3":2,"4":2,"7":4,"8":1,"9":1,"10":3,"11":1,"12":4,"13":2,"14":3,"15":1,"16":2,"18":5,"20":7,"21":3,"22":9,"23":17,"24":4,"25":3,"26":12,"27":3,"28":2,"29":1,"30":8,"31":4,"32":6,"33":2,"34":12,"35":4,"36":13,"37":15,"38":14,"39":9,"40":8,"41":7,"42":15,"43":2,"44":5,"46":5,"47":5,"48":2,"49":1,"50":2,"51":3,"53":4,"54":3,"55":3,"56":4,"57":1,"58":15,"61":5,"62":7,"64":9,"65":6,"66":8,"69":6,"70":6,"71":40,"72":13,"73":7,"74":6,"75":12,"76":31,"77":2,"78":46,"79":3,"81":8,"82":5,"89":10,"90":2,"92":5,"93":3,"94":31,"95":18,"96":5,"97":14,"98":1,"101":1,"102":6,"103":7,"104":2,"105":4,"106":4,"107":34,"109":2,"110":54,"111":37,"112":25,"113":17,"115":34,"117":1,"118":2,"119":4,"121":11,"122":15,"124":27,"125":9,"126":8,"127":6,"128":2,"129":40,"130":10,"131":2,"132":10,"133":2,"134":5,"135":12,"136":9,"137":8,"138":6,"139":58,"140":9,"141":46,"142":12,"143":5,"144":24,"145":58,"146":26,"147":11,"148":65,"149":6,"150":3,"151":4,"152":4,"153":23,"154":15,"155":6,"156":8,"157":23,"158":3,"159":30,"160":47,"161":1,"163":4,"164":12,"165":6,"167":3,"168":23,"169":2,"170":5,"171":10,"172":29,"173":3,"175":2,"176":28,"178":1,"179":24,"181":1,"182":8,"184":31,"185":1,"186":4,"187":75,"188":23,"189":25,"190":9,"191":5,"193":16,"194":3,"195":51,"196":58,"197":6,"198":6,"199":24,"200":37,"201":9,"202":3,"203":13,"204":9,"206":11,"209":1,"210":4,"211":3,"212":20,"213":12,"214":2,"215":2,"216":3,"218":14,"219":9,"220":9,"221":1,"222":8,"224":4,"226":4,"227":2,"228":2,"229":1,"230":2,"231":1,"232":3,"238":4,"240":6,"241":5,"242":6,"243":3,"244":4,"245":4,"246":13,"247":1,"248":1,"251":1,"252":13,"253":11,"254":6,"256":4,"257":2,"258":9,"259":8,"260":4,"261":6,"262":7,"263":2,"264":1,"266":4,"267":5,"268":2,"269":5,"270":23,"271":2,"272":14,"273":15,"274":2,"275":1,"277":5,"278":2,"279":7,"280":2,"281":2,"282":5,"283":1,"284":15,"285":1,"286":3,"289":3,"291":1,"292":4,"293":2,"295":1,"296":6,"297":5,"298":6,"299":1,"300":4,"301":3,"302":5,"303":1,"304":2,"306":1,"307":5,"308":1,"310":2,"311":9,"312":16,"313":2,"314":18,"315":10,"316":8,"317":6,"318":4,"319":21,"320":1,"321":19,"322":13,"324":4,"325":6,"326":3,"328":4,"331":4,"332":1,"333":4,"334":4,"336":8,"337":1,"338":2,"340":2,"341":8,"342":2,"343":1,"344":2,"346":5,"347":1,"348":3,"350":7,"351":1,"352":11,"353":2,"355":2,"356":14,"357":15,"358":53,"359":6,"360":21,"361":4,"362":1,"363":5,"364":19,"365":12,"366":9,"367":2,"368":2,"369":10,"371":5,"372":1,"373":10,"374":26,"375":43,"376":6,"377":13,"380":1,"381":15,"382":48,"383":3,"384":15,"385":2,"386":1,"387":1,"389":19,"390":3,"391":3,"393":14,"394":9,"395":2,"396":2,"397":4,"398":7,"399":2,"400":3,"402":5,"403":6,"405":2,"406":4,"407":18,"408":9,"409":17,"410":32,"412":26,"413":24,"414":10,"415":47,"416":32,"417":3,"420":1,"421":2,"422":2,"423":2,"425":1,"426":4,"427":2,"430":2,"431":1,"434":16,"435":3,"436":4,"437":6,"438":10,"440":3,"441":9,"443":1,"445":3,"446":10,"447":2,"448":2,"449":2,"451":1,"453":2,"454":2,"455":1,"456":2,"458":1,"459":2,"460":1,"461":2,"462":3,"465":5,"467":2,"469":1,"470":7,"472":2,"473":1,"475":2,"476":3,"477":13,"478":2,"479":1,"480":2,"482":3,"483":13,"484":9,"485":5,"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,"554":8,"555":4,"557":5,"558":3,"559":2,"560":4,"561":1,"562":8,"563":12,"564":13,"565":3,"567":15,"568":9,"569":2,"570":3,"573":2,"574":3,"576":5,"577":3,"579":5,"580":2,"581":1,"582":1,"583":1,"584":1,"585":9,"587":1,"589":1,"590":1,"591":3,"592":5,"593":5,"594":6,"595":10,"596":5,"597":7,"600":1,"601":5,"602":3,"603":1,"604":6,"605":10,"607":6,"608":1,"609":1,"611":1,"612":8,"614":9,"615":7,"616":9,"617":2,"620":3,"622":5,"623":4,"624":1,"625":2,"626":2,"627":1,"628":1,"629":1,"631":1,"632":14,"633":5,"634":5,"635":2,"636":4,"637":5,"638":3,"639":5,"640":4,"641":5,"642":2,"643":2,"644":3,"645":2,"646":1,"647":5,"648":3,"649":1,"651":1,"652":1,"654":2,"655":5,"656":4,"657":2,"658":6,"659":6,"660":10,"661":4,"662":6,"663":3,"665":3,"666":1,"667":2,"668":1,"669":1,"670":4,"671":7,"672":10,"673":3,"674":5,"675":5,"676":2,"678":6,"679":2,"682":2,"683":17,"684":6,"685":1,"687":1,"692":1,"693":6,"694":2,"695":1,"697":1,"698":2,"699":10,"703":1,"704":1,"705":3,"706":4,"707":5,"708":3,"709":2,"710":2,"711":1,"713":2,"714":8,"715":8,"716":14,"717":6,"720":1,"722":3,"725":1,"727":2,"729":2,"730":1,"731":1,"732":3,"733":4,"735":1,"736":1,"738":2,"739":4,"740":5,"741":1,"742":1,"743":2,"744":1,"745":10,"746":3,"747":1,"748":3,"749":1,"750":3,"751":17,"752":1,"753":1,"754":2,"756":14}}],["too",{"0":{"563":1},"2":{"563":3,"659":2}}],["tools",{"2":{"284":1}}],["toolset",{"2":{"189":1}}],["toolkit",{"0":{"51":1},"1":{"52":1,"53":1,"54":1,"55":1,"56":1,"57":1},"2":{"50":1,"51":1}}],["tool",{"2":{"49":1,"189":1,"488":2,"501":1,"605":1,"716":1}}],["toolings",{"2":{"548":1}}],["tooling",{"2":{"23":1,"151":1,"153":1,"169":1}}],["tomlvim",{"2":{"639":1}}],["tomldiscard",{"2":{"605":1}}],["tomlexternal",{"2":{"602":1}}],["tomlsed",{"2":{"601":1}}],["tomlseeds=$",{"2":{"585":2}}],["tomlrpc",{"2":{"595":1}}],["tomlpruning",{"2":{"591":1,"592":1}}],["tomlpersistent",{"2":{"585":3}}],["tomlmin",{"2":{"589":1}}],["tomlindexer",{"2":{"588":1,"590":1,"604":1}}],["toml`",{"2":{"585":2}}],["toml",{"0":{"531":1,"571":1,"573":1},"1":{"572":1,"573":1,"574":2,"575":2,"576":2,"577":2,"578":2,"579":2},"2":{"348":2,"502":1,"503":1,"531":8,"564":3,"573":4,"585":7,"587":1,"589":1,"590":1,"591":1,"592":2,"595":2,"601":2,"602":1,"604":2,"605":2,"639":3,"641":1,"645":1,"661":1,"753":1}}],["tombstoned",{"2":{"24":1}}],["todo",{"2":{"272":2,"356":2,"357":3,"358":3,"360":1,"382":6}}],["today",{"2":{"216":1,"499":2}}],["toattestationproof",{"2":{"145":2,"148":4}}],["torowproofs",{"2":{"144":2,"148":4}}],["torowroots",{"2":{"143":2,"148":4}}],["tonamespacenode",{"2":{"141":4,"143":2,"148":6}}],["tonamespacemerklemultiproofs",{"2":{"141":2,"148":4}}],["tosliceofbytes",{"2":{"135":2}}],["token=$",{"2":{"316":4,"318":2,"336":2}}],["tokencelestia",{"2":{"314":1}}],["tokenbridgeretryablesender",{"2":{"83":1,"84":1}}],["tokenbridgecreator",{"2":{"83":1,"84":1}}],["tokenbridgedeployment",{"2":{"78":2}}],["token",{"0":{"316":1,"318":1,"458":1,"493":1,"648":1},"2":{"67":1,"69":2,"71":8,"73":1,"76":8,"78":13,"80":1,"269":4,"270":4,"271":4,"272":4,"273":4,"314":7,"315":9,"316":6,"317":3,"318":2,"319":4,"324":1,"336":3,"342":2,"357":2,"358":2,"360":2,"363":4,"364":4,"365":4,"366":4,"381":4,"417":1,"454":3,"493":1,"495":1,"554":1,"647":1,"648":3}}],["tokens",{"0":{"422":1,"431":1,"497":1,"673":1},"2":{"7":1,"37":1,"67":1,"71":1,"72":1,"312":2,"331":1,"338":2,"352":2,"417":1,"422":1,"423":1,"427":1,"430":1,"432":1,"445":1,"446":6,"471":2,"472":1,"484":3,"493":1,"497":2,"522":1,"537":2,"632":3,"637":1,"640":2,"648":1,"672":1,"673":1,"710":3,"745":1,"751":1}}],["touch",{"2":{"58":1}}],["together",{"2":{"46":1,"283":1,"407":1,"413":1,"415":1,"416":1,"503":1}}],["topology",{"2":{"595":1}}],["topic",{"2":{"30":1,"36":1,"37":1,"402":1,"494":1}}],["topics",{"2":{"2":1,"30":1,"37":2,"42":2,"214":1}}],["top",{"2":{"23":2,"58":1,"61":1,"298":1,"384":1,"390":1,"391":1,"412":1,"413":1,"425":1,"515":1,"687":1,"688":1,"689":1,"696":1,"712":1}}],["totaldifficulty",{"2":{"340":2}}],["total",{"2":{"20":1,"21":1,"127":2,"129":2,"132":2,"137":2,"144":3,"145":2,"148":4,"159":1,"328":4,"371":2,"373":1,"374":2,"375":6,"441":1,"445":1,"451":1,"525":4,"683":5}}],["towards",{"2":{"18":1,"24":1,"38":1,"509":1}}],["toward",{"2":{"7":1,"15":1}}],["to",{"0":{"53":1,"60":1,"62":1,"64":1,"91":1,"112":1,"118":1,"119":1,"129":1,"139":1,"150":1,"184":1,"205":1,"214":1,"253":1,"271":1,"272":1,"282":1,"283":1,"311":1,"331":1,"347":1,"348":1,"365":1,"370":1,"382":1,"383":1,"389":1,"390":1,"391":1,"405":1,"417":1,"486":1,"499":1,"511":1,"562":1,"587":1,"608":1,"634":1,"675":1,"717":1,"745":1},"1":{"61":1,"62":1,"92":1,"93":1,"94":1,"95":1,"96":1,"97":1,"98":1,"140":1,"141":1,"142":1,"143":1,"144":1,"145":1,"146":1,"147":1,"148":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,"163":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,"206":1,"207":1,"208":1,"209":1,"210":1,"211":1,"212":1,"213":1,"214":1,"215":1,"216":1,"284":1,"285":1,"286":1,"371":1,"372":1,"373":1,"374":1,"375":1,"376":1,"377":1,"378":1,"379":1,"380":1,"381":1,"382":1,"383":1,"384":1,"418":1,"419":1,"420":1,"421":1,"422":1,"423":1,"424":1,"425":1,"426":1,"427":1,"428":1,"429":1,"430":1,"431":1,"432":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,"588":1,"589":1,"609":1,"610":1,"611":1,"718":1,"719":1,"720":1},"2":{"0":2,"1":1,"2":4,"6":3,"7":3,"8":3,"10":2,"13":1,"14":1,"16":1,"18":5,"20":5,"21":1,"22":1,"23":7,"24":3,"25":4,"26":5,"27":1,"29":3,"30":7,"31":1,"32":3,"33":2,"34":7,"35":4,"36":8,"37":14,"38":14,"39":10,"40":7,"41":14,"42":13,"43":1,"44":3,"46":4,"47":4,"48":2,"49":2,"50":2,"51":4,"53":2,"54":1,"55":1,"56":2,"57":1,"58":6,"59":1,"62":3,"64":7,"65":3,"66":3,"69":5,"70":7,"71":19,"72":10,"73":5,"75":15,"76":21,"77":2,"78":21,"79":1,"81":1,"82":1,"89":2,"90":3,"92":2,"93":1,"94":15,"95":6,"96":4,"97":8,"98":1,"101":1,"102":1,"103":6,"104":2,"105":1,"106":4,"107":21,"109":8,"110":23,"111":21,"112":8,"113":8,"115":13,"118":2,"119":1,"121":8,"122":5,"124":17,"125":5,"126":8,"127":4,"129":16,"130":5,"132":5,"136":8,"137":3,"139":27,"140":3,"141":12,"142":4,"144":7,"145":19,"146":10,"147":2,"148":22,"149":3,"150":4,"151":1,"153":24,"154":6,"155":3,"156":4,"157":5,"159":10,"160":18,"163":6,"164":6,"165":5,"167":2,"168":14,"169":1,"170":2,"171":3,"172":8,"173":1,"175":5,"176":5,"178":3,"179":8,"184":8,"185":2,"186":4,"187":44,"188":9,"189":11,"190":3,"191":4,"193":7,"195":14,"196":17,"197":3,"198":3,"199":7,"200":9,"201":5,"202":5,"203":1,"204":2,"206":5,"209":1,"210":1,"211":4,"212":17,"213":1,"215":1,"216":1,"218":3,"219":3,"220":5,"221":2,"222":2,"223":2,"224":2,"226":3,"227":1,"228":2,"229":2,"230":1,"232":2,"234":1,"237":3,"238":5,"240":6,"241":6,"242":4,"243":1,"244":1,"245":1,"246":6,"248":1,"251":1,"252":7,"253":4,"254":7,"256":3,"257":1,"258":4,"259":4,"260":3,"261":1,"262":1,"263":1,"264":1,"265":1,"266":2,"267":1,"269":7,"270":7,"271":5,"272":7,"273":2,"274":1,"276":2,"277":3,"279":2,"280":1,"282":3,"283":4,"284":9,"286":1,"288":2,"289":1,"292":1,"293":2,"296":2,"298":2,"300":3,"301":1,"302":4,"304":2,"306":2,"307":3,"311":9,"312":8,"313":2,"314":10,"315":3,"316":5,"317":6,"318":1,"319":8,"320":1,"321":9,"322":4,"324":1,"326":1,"327":1,"331":1,"332":1,"333":2,"334":2,"336":4,"337":2,"338":1,"341":5,"342":1,"343":1,"344":3,"346":7,"347":2,"348":7,"350":7,"351":1,"352":5,"353":3,"355":1,"356":2,"357":6,"358":25,"359":3,"360":12,"361":8,"363":7,"364":7,"365":3,"366":4,"367":1,"368":1,"369":7,"370":2,"371":3,"373":3,"374":4,"375":9,"377":8,"381":7,"382":26,"384":6,"385":1,"387":2,"388":4,"389":14,"390":4,"391":3,"393":7,"394":3,"395":4,"396":2,"397":3,"398":7,"399":1,"400":5,"403":7,"404":4,"406":7,"407":4,"408":3,"409":6,"410":6,"412":9,"413":8,"414":8,"415":11,"416":8,"417":2,"420":1,"425":2,"426":1,"429":1,"431":1,"432":1,"434":6,"436":2,"437":3,"438":18,"440":3,"441":4,"443":3,"444":3,"445":2,"446":8,"447":2,"448":1,"449":1,"453":1,"454":5,"455":2,"456":1,"462":3,"465":9,"466":1,"467":2,"469":3,"470":6,"471":1,"475":1,"476":1,"477":5,"482":1,"483":7,"484":7,"485":1,"486":3,"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,"554":2,"555":3,"557":2,"558":1,"559":3,"560":3,"562":4,"563":6,"564":5,"565":1,"567":1,"568":4,"569":2,"570":2,"574":1,"576":4,"577":3,"579":1,"580":2,"585":9,"587":2,"590":2,"591":2,"592":3,"593":2,"594":2,"595":4,"596":1,"597":4,"600":2,"601":3,"602":2,"603":1,"604":12,"605":9,"607":3,"608":1,"609":1,"610":1,"611":1,"612":10,"614":5,"615":9,"616":6,"617":2,"620":3,"621":1,"622":1,"623":4,"624":1,"625":1,"632":14,"633":1,"634":3,"635":1,"636":2,"637":3,"638":2,"639":2,"640":7,"641":1,"642":2,"643":2,"644":4,"645":1,"647":2,"648":4,"649":4,"650":1,"651":1,"652":1,"653":2,"656":1,"658":3,"659":1,"660":4,"661":6,"662":2,"663":3,"664":1,"665":1,"667":1,"671":6,"672":2,"673":2,"674":1,"675":3,"678":2,"679":2,"682":1,"683":2,"684":10,"685":1,"692":3,"693":7,"694":1,"698":2,"699":21,"703":1,"704":3,"705":4,"706":2,"707":2,"708":3,"709":1,"710":1,"713":2,"714":4,"715":6,"716":10,"717":3,"720":1,"722":1,"725":3,"727":3,"729":2,"730":2,"731":4,"732":2,"733":9,"735":1,"736":2,"738":1,"739":1,"740":1,"741":1,"745":7,"746":4,"748":2,"751":13,"752":1,"753":3,"754":1,"756":6}}]],"serializationVersion":2}';export{e as default}; diff --git a/assets/chunks/ArabicaVersionTags.f0aaf832.js b/assets/chunks/ArabicaVersionTags.f0aaf832.js new file mode 100644 index 00000000000..a20201cd201 --- /dev/null +++ b/assets/chunks/ArabicaVersionTags.f0aaf832.js @@ -0,0 +1 @@ +import{a as r}from"./arabica_versions.1930378b.js";import{c as s}from"./constants.fa173a21.js";import{_ as o,o as n,c as l,k as e,t as a}from"./framework.1a91c06a.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.4fd38854.js b/assets/chunks/MainnetVersionTags.4fd38854.js new file mode 100644 index 00000000000..798453e66cf --- /dev/null +++ b/assets/chunks/MainnetVersionTags.4fd38854.js @@ -0,0 +1 @@ +import{m as s}from"./mainnet_versions.955428b6.js";import{c as o}from"./constants.fa173a21.js";import{_ as r,o as a,c as l,k as e,t as n}from"./framework.1a91c06a.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.290e3e99.js b/assets/chunks/MochaVersionTags.290e3e99.js new file mode 100644 index 00000000000..34a4a1ad6a3 --- /dev/null +++ b/assets/chunks/MochaVersionTags.290e3e99.js @@ -0,0 +1 @@ +import{m as s}from"./mocha_versions.7704b055.js";import{c as r}from"./constants.fa173a21.js";import{_ as a,o as n,c as l,k as e,t as o}from"./framework.1a91c06a.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.43aac247.js b/assets/chunks/VPLocalSearchBox.43aac247.js new file mode 100644 index 00000000000..b5c4ee088c0 --- /dev/null +++ b/assets/chunks/VPLocalSearchBox.43aac247.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 de,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 nt,a as he,t as ve,at as Ht,p as Gt,m as qt,au as at,av as Qt,a7 as Yt,ad as Zt,_ as Xt}from"./framework.1a91c06a.js";import{u as er,c as tr}from"./theme.9c6b4fd0.js";const rr={root:()=>pt(()=>import("./@localSearchIndexroot.b53f000d.js"),[])};/*! +* tabbable 6.2.0 +* @license MIT, https://github.com/focus-trap/tabbable/blob/master/LICENSE +*/var mt=["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=mt.join(","),yt=typeof Element>"u",se=yt?function(){}:Element.prototype.matches||Element.prototype.msMatchesSelector||Element.prototype.webkitMatchesSelector,Te=!yt&&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},Ce=function o(e,t){var r;t===void 0&&(t=!0);var a=e==null||(r=e.getAttribute)===null||r===void 0?void 0:r.call(e,"inert"),n=a===""||a==="true",i=n||t&&e&&o(e.parentNode);return i},nr=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(Ce(e))return[];var a=Array.prototype.slice.apply(e.querySelectorAll(Ae));return t&&se.call(e,Ae)&&a.unshift(e),a=a.filter(r),a},bt=function o(e,t,r){for(var a=[],n=Array.from(e);n.length;){var i=n.shift();if(!Ce(i,!1))if(i.tagName==="SLOT"){var s=i.assignedElements(),u=s.length?s:i.children,l=o(u,!0,r);r.flatten?a.push.apply(a,l):a.push({scopeParent:i,candidates:l})}else{var d=se.call(i,Ae);d&&r.filter(i)&&(t||!e.includes(i))&&a.push(i);var h=i.shadowRoot||typeof r.getShadowRoot=="function"&&r.getShadowRoot(i),v=!Ce(h,!1)&&(!r.shadowRootFilter||r.shadowRootFilter(i));if(h&&v){var y=o(h===!0?i.children:h.children,!0,r);r.flatten?a.push.apply(a,y):a.push({scopeParent:i,candidates:y})}else n.unshift.apply(n,i.children)}}return a},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)||nr(e))&&!wt(e)?0:e.tabIndex},ar=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=n?e.parentElement:e;if(se.call(i,"details:not([open]) *"))return!0;if(!r||r==="full"||r==="legacy-full"){if(typeof a=="function"){for(var s=e;e;){var u=e.parentElement,l=Te(e);if(u&&!u.shadowRoot&&a(u)===!0)return it(e);e.assignedSlot?e=e.assignedSlot:!u&&l!==e.ownerDocument?e=l.host:e=u}e=s}if(dr(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)},mr=function o(e){var t=[],r=[];return e.forEach(function(a,n){var i=!!a.scopeParent,s=i?a.scopeParent:a,u=ar(s,i),l=i?o(a.candidates):s;u===0?i?t.push.apply(t,l):t.push(s):r.push({documentOrder:n,tabIndex:u,item:a,isScope:i,content:l})}),r.sort(ir).reduce(function(a,n){return n.isScope?a.push.apply(a,n.content):a.push(n.content),a},[]).concat(t)},yr=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)),mr(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=mt.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(a){return Object.getOwnPropertyDescriptor(o,a).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 a=e.indexOf(t);a===-1||e.splice(a,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},ye=function(e){return(e==null?void 0:e.key)==="Tab"||(e==null?void 0:e.keyCode)===9},Ar=function(e){return ye(e)&&!e.shiftKey},Tr=function(e){return ye(e)&&e.shiftKey},lt=function(e){return setTimeout(e,0)},ct=function(e,t){var r=-1;return e.every(function(a,n){return t(a)?(r=n,!1):!0}),r},pe=function(e){for(var t=arguments.length,r=new Array(t>1?t-1:0),a=1;a1?p-1:0),k=1;k=0)c=r.activeElement;else{var f=i.tabbableGroups[0],p=f&&f.firstTabbableNode;c=p||d("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=yr(c,n.tabbableOptions),p=gr(c,n.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)}),m=!!f.find(function(w){return oe(w)>0});return{container:c,tabbableNodes:f,focusableNodes:p,posTabIndexesFound:m,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&&!d("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.")},y=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!==y(document)){if(!c||!c.focus){x(h());return}c.focus({preventScroll:!!n.preventScroll}),i.mostRecentlyFocusedNode=c,Er(c)&&c.select()}},E=function(c){var f=d("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),m=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&&(m.container===f||_e(f,n.tabbableOptions)&&!ue(f,n.tabbableOptions)&&!m.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 ye(p)||(M=m.nextTabbableNode(f,!1))}else{var j=ct(i.tabbableGroups,function(J){var q=J.lastTabbableNode;return f===q});if(j<0&&(m.container===f||_e(f,n.tabbableOptions)&&!ue(f,n.tabbableOptions)&&!m.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 ye(p)||(M=m.nextTabbableNode(f))}}else M=d("fallbackFocus");return M},S=function(c){var f=Ee(c);if(!(l(f,c)>=0)){if(pe(n.clickOutsideDeactivates,c)){s.deactivate({returnFocus:n.returnFocusOnDeactivate});return}pe(n.allowOutsideClick,c)||c.preventDefault()}},C=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 m=z.findIndex(function(w){return w===i.mostRecentlyFocusedNode});m>=0&&(n.isKeyForward(i.recentNavEvent)?m+1=0&&(N=z[m-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:n.isKeyBackward(i.recentNavEvent)})),b(N||i.mostRecentlyFocusedNode||h())}i.recentNavEvent=void 0},T=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&&(ye(c)&&c.preventDefault(),b(p))},R=function(c){if(Sr(c)&&pe(n.escapeDeactivates,c)!==!1){c.preventDefault(),s.deactivate();return}(n.isKeyForward(c)||n.isKeyBackward(c))&&T(c,n.isKeyBackward(c))},L=function(c){var f=Ee(c);l(f,c)>=0||pe(n.clickOutsideDeactivates,c)||pe(n.allowOutsideClick,c)||(c.preventDefault(),c.stopImmediatePropagation())},P=function(){if(i.active)return ut.activateTrap(a,s),i.delayInitialFocusTimer=n.delayInitialFocus?lt(function(){b(h())}):b(h()),r.addEventListener("focusin",C,!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",C,!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(h())},_=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:n.onDeactivate,onPostDeactivate:n.onPostDeactivate,checkCanReturnFocus:n.checkCanReturnFocus},c);clearTimeout(i.delayInitialFocusTimer),i.delayInitialFocusTimer=void 0,I(),i.active=!1,i.paused=!1,O(),ut.deactivateTrap(a,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,...a}=e,n=ie(!1),i=ie(!1),s=h=>t&&t.activate(h),u=h=>t&&t.deactivate(h),l=()=>{t&&(t.pause(),i.value=!0)},d=()=>{t&&(t.unpause(),i.value=!1)};return Be(()=>kt(o),h=>{h&&(t=kr(h,{...a,onActivate(){n.value=!0,e.onActivate&&e.onActivate()},onDeactivate(){n.value=!1,e.onDeactivate&&e.onDeactivate()}}),r&&s())},{flush:"post"}),Nt(()=>u()),{hasFocus:n,isPaused:i,activate:s,deactivate:u,pause:l,unpause:d}}class ce{constructor(e,t=!0,r=[],a=5e3){this.ctx=e,this.iframes=t,this.exclude=r,this.iframesTimeout=a}static matches(e,t){const r=typeof t=="string"?[t]:t,a=e.matches||e.matchesSelector||e.msMatchesSelector||e.mozMatchesSelector||e.oMatchesSelector||e.webkitMatchesSelector;if(a){let n=!1;return r.every(i=>a.call(e,i)?(n=!0,!1):!0),n}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 a=t.filter(n=>n.contains(r)).length>0;t.indexOf(r)===-1&&!a&&t.push(r)}),t}getIframeContents(e,t,r=()=>{}){let a;try{const n=e.contentWindow;if(a=n.document,!n||!a)throw new Error("iframe inaccessible")}catch{r()}a&&t(a)}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 a=!1,n=null;const i=()=>{if(!a){a=!0,clearTimeout(n);try{this.isIframeBlank(e)||(e.removeEventListener("load",i),this.getIframeContents(e,t,r))}catch{r()}}};e.addEventListener("load",i),n=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,a=>{r++,this.waitForIframes(a.querySelector("html"),()=>{--r||t()})},a=>{a||t()})}forEachIframe(e,t,r,a=()=>{}){let n=e.querySelectorAll("iframe"),i=n.length,s=0;n=Array.prototype.slice.call(n);const u=()=>{--i<=0&&a(s)};i||u(),n.forEach(l=>{ce.matches(l,this.exclude)?u():this.onIframeReady(l,d=>{t(l)&&(s++,r(d)),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 a=e.compareDocumentPosition(r),n=Node.DOCUMENT_POSITION_PRECEDING;if(a&n)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,a){let n=!1,i=!1;return a.forEach((s,u)=>{s.val===r&&(n=u,i=s.handled)}),this.compareNodeIframe(e,t,r)?(n===!1&&!i?a.push({val:r,handled:!0}):n!==!1&&!i&&(a[n].handled=!0),!0):(n===!1&&a.push({val:r,handled:!1}),!1)}handleOpenIframes(e,t,r,a){e.forEach(n=>{n.handled||this.getIframeContents(n.val,i=>{this.createInstanceOnIframe(i).forEachNode(t,r,a)})})}iterateThroughNodes(e,t,r,a,n){const i=this.createIterator(t,e,a);let s=[],u=[],l,d,h=()=>({prevNode:d,node:l}=this.getIteratorNode(i),l);for(;h();)this.iframes&&this.forEachIframe(t,v=>this.checkIframeFilter(l,d,v,s),v=>{this.createInstanceOnIframe(v).forEachNode(e,y=>u.push(y),a)}),u.push(l);u.forEach(v=>{r(v)}),this.iframes&&this.handleOpenIframes(s,e,r,a),n()}forEachNode(e,t,r,a=()=>{}){const n=this.getContexts();let i=n.length;i||a(),n.forEach(s=>{const u=()=>{this.iterateThroughNodes(e,s,t,r,()=>{--i<=0&&a()})};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",a=this.opt.ignoreJoiners||this.opt.ignorePunctuation.length?"\0":"";for(let n in t)if(t.hasOwnProperty(n)){const i=t[n],s=this.opt.wildcards!=="disabled"?this.setupWildcardsRegExp(n):this.escapeStr(n),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}`),a+`(${this.processSynomyms(s)}|${this.processSynomyms(u)})`+a))}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,a)=>{let n=a.charAt(r+1);return/[(|)\\]/.test(n)||n===""?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 a=[];return e.split("").forEach(n=>{r.every(i=>{if(i.indexOf(n)!==-1){if(a.indexOf(i)>-1)return!1;e=e.replace(new RegExp(`[${i}]`,`gm${t}`),`[${i}]`),a.push(i)}return!0})}),e}createMergedBlanksRegExp(e){return e.replace(/[\s]+/gmi,"[\\s]+")}createAccuracyRegExp(e){const t="!\"#$%&'()*+,-./:;<=>?@[\\]^_`{|}~¡¿";let r=this.opt.accuracy,a=typeof r=="string"?r:r.value,n=typeof r=="string"?[]:r.limiters,i="";switch(n.forEach(s=>{i+=`|${this.escapeStr(s)}`}),a){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(a=>{a.trim()&&t.indexOf(a)===-1&&t.push(a)}):r.trim()&&t.indexOf(r)===-1&&t.push(r)}),{keywords:t.sort((r,a)=>a.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((a,n)=>a.start-n.start).forEach(a=>{let{start:n,end:i,valid:s}=this.callNoMatchOnInvalidRanges(a,r);s&&(a.start=n,a.length=i-n,t.push(a),r=i)}),t}callNoMatchOnInvalidRanges(e,t){let r,a,n=!1;return e&&typeof e.start<"u"?(r=parseInt(e.start,10),a=r+parseInt(e.length,10),this.isNumeric(e.start)&&this.isNumeric(e.length)&&a-t>0&&a-r>0?n=!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:a,valid:n}}checkWhitespaceRanges(e,t,r){let a,n=!0,i=r.length,s=t-i,u=parseInt(e.start,10)-s;return u=u>i?i:u,a=u+parseInt(e.length,10),a>i&&(a=i,this.log(`End range automatically set to the max value of ${i}`)),u<0||a-u<0||u>i||a>i?(n=!1,this.log(`Invalid range: ${JSON.stringify(e)}`),this.opt.noMatch(e)):r.substring(u,a).replace(/\s+/g,"")===""&&(n=!1,this.log("Skipping whitespace only range: "+JSON.stringify(e)),this.opt.noMatch(e)),{start:u,end:a,valid:n}}getTextNodes(e){let t="",r=[];this.iterator.forEachNode(NodeFilter.SHOW_TEXT,a=>{r.push({start:t.length,end:(t+=a.textContent).length,node:a})},a=>this.matchesExclude(a.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 a=this.opt.element?this.opt.element:"mark",n=e.splitText(t),i=n.splitText(r-t);let s=document.createElement(a);return s.setAttribute("data-markjs","true"),this.opt.className&&s.setAttribute("class",this.opt.className),s.textContent=n.textContent,n.parentNode.replaceChild(s,n),i}wrapRangeInMappedTextNode(e,t,r,a,n){e.nodes.every((i,s)=>{const u=e.nodes[s+1];if(typeof u>"u"||u.start>t){if(!a(i.node))return!1;const l=t-i.start,d=(r>i.end?i.end:r)-i.start,h=e.value.substr(0,i.start),v=e.value.substr(d+i.start);if(i.node=this.wrapRangeInTextNode(i.node,l,d),e.value=h+v,e.nodes.forEach((y,b)=>{b>=s&&(e.nodes[b].start>0&&b!==s&&(e.nodes[b].start-=d),e.nodes[b].end-=d)}),r-=d,n(i.node.previousSibling,i.start),r>i.end)t=i.end;else return!1}return!0})}wrapMatches(e,t,r,a,n){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 d=l.index;if(i!==0)for(let h=1;h{let u;for(;(u=e.exec(s.value))!==null&&u[i]!=="";){let l=u.index;if(i!==0)for(let h=1;hr(u[i],h),(h,v)=>{e.lastIndex=v,a(h)})}n()})}wrapRangeFromIndex(e,t,r,a){this.getTextNodes(n=>{const i=n.value.length;e.forEach((s,u)=>{let{start:l,end:d,valid:h}=this.checkWhitespaceRanges(s,i,n.value);h&&this.wrapRangeInMappedTextNode(n,l,d,v=>t(v,s,n.value.substring(l,d),u),v=>{r(v,s)})}),a()})}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,a="wrapMatches";const n=i=>{r++,this.opt.each(i)};this.opt.acrossElements&&(a="wrapMatchesAcrossElements"),this[a](e,this.opt.ignoreGroups,(i,s)=>this.opt.filter(s,i,r),n,()=>{r===0&&this.opt.noMatch(e),this.opt.done(r)})}mark(e,t){this.opt=t;let r=0,a="wrapMatches";const{keywords:n,length:i}=this.getSeparatedKeywords(typeof e=="string"?[e]:e),s=this.opt.caseSensitive?"":"i",u=l=>{let d=new RegExp(this.createRegExp(l),`gm${s}`),h=0;this.log(`Searching with expression "${d}"`),this[a](d,1,(v,y)=>this.opt.filter(y,l,r,h),v=>{h++,r++,this.opt.each(v)},()=>{h===0&&this.opt.noMatch(l),n[i-1]===l?this.opt.done(r):u(n[n.indexOf(l)+1])})};this.opt.acrossElements&&(a="wrapMatchesAcrossElements"),i===0?this.opt.done(r):u(n[0])}markRanges(e,t){this.opt=t;let r=0,a=this.checkRanges(e);a&&a.length?(this.log("Starting to mark with the following ranges: "+JSON.stringify(a)),this.wrapRangeFromIndex(a,(n,i,s,u)=>this.opt.filter(n,i,s,u),(n,i)=>{r++,this.opt.each(n,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 a=ce.matches(r,t),n=this.matchesExclude(r);return!a||n?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,a=arguments.length;r0&&n[n.length-1])&&(l[0]===6||l[0]===2)){t=0;continue}if(l[0]===3&&(!n||l[1]>n[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),a,n=[],i;try{for(;(e===void 0||e-- >0)&&!(a=r.next()).done;)n.push(a.value)}catch(s){i={error:s}}finally{try{a&&!a.done&&(t=r.return)&&t.call(r)}finally{if(i)throw i.error}}return n}var Rr="ENTRIES",Ft="KEYS",Et="VALUES",H="",Oe=function(){function o(e,t){var r=e._tree,a=Array.from(r.keys());this.set=e,this._type=t,this._path=a.length>0?[{node:r,keys:a}]:[]}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 a=t.get(le(r));return this._path.push({node:a,keys:Array.from(a.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 a=e.length+1,n=a+t,i=new Uint8Array(n*a).fill(t+1),s=0;st)continue e}St(o.get(y),e,t,r,a,E,i,s+y)}}}catch(f){u={error:f}}finally{try{v&&!v.done&&(l=h.return)&&l.call(h)}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 a=W(Ne(this._tree,e.slice(this._prefix.length)),2),n=a[0],i=a[1];if(n===void 0){var s=W(Ke(i),2),u=s[0],l=s[1];try{for(var d=D(u.keys()),h=d.next();!h.done;h=d.next()){var v=h.value;if(v!==H&&v.startsWith(l)){var y=new Map;return y.set(v.slice(l.length),u.get(v)),new o(y,e)}}}catch(b){t={error:b}}finally{try{h&&!h.done&&(r=d.return)&&r.call(d)}finally{if(t)throw t.error}}}return new o(n,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 a=D(this),n=a.next();!n.done;n=a.next()){var i=W(n.value,2),s=i[0],u=i[1];e(s,u,this)}}catch(l){t={error:l}}finally{try{n&&!n.done&&(r=a.return)&&r.call(a)}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),a=r.get(H);return a===void 0&&r.set(H,a=t()),a},o.prototype.values=function(){return new Oe(this,Et)},o.prototype[Symbol.iterator]=function(){return this.entries()},o.from=function(e){var t,r,a=new o;try{for(var n=D(e),i=n.next();!i.done;i=n.next()){var s=W(i.value,2),u=s[0],l=s[1];a.set(u,l)}}catch(d){t={error:d}}finally{try{i&&!i.done&&(r=n.return)&&r.call(n)}finally{if(t)throw t.error}}return a},o.fromObject=function(e){return o.from(Object.entries(e))},o}(),Ne=function(o,e,t){var r,a;if(t===void 0&&(t=[]),e.length===0||o==null)return[o,t];try{for(var n=D(o.keys()),i=n.next();!i.done;i=n.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&&(a=n.return)&&a.call(n)}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 a=D(o.keys()),n=a.next();!n.done;n=a.next()){var i=n.value;if(i!==H&&e.startsWith(i))return $e(o.get(i),e.slice(i.length))}}catch(s){t={error:s}}finally{try{n&&!n.done&&(r=a.return)&&r.call(a)}finally{if(t)throw t.error}}},Me=function(o,e){var t,r,a=e.length;e:for(var n=0;o&&n0)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(a,n){t.removeFieldLength(r,n,t._documentCount,a)}),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,a=e.batchSize,n=e.batchWait;this.conditionalVacuum({batchSize:a,batchWait:n},{minDirtCount:r,minDirtFactor:t})}},o.prototype.discardAll=function(e){var t,r,a=this._options.autoVacuum;try{this._options.autoVacuum=!1;try{for(var n=D(e),i=n.next();!i.done;i=n.next()){var s=i.value;this.discard(s)}}catch(u){t={error:u}}finally{try{i&&!i.done&&(r=n.return)&&r.call(n)}finally{if(t)throw t.error}}}finally{this._options.autoVacuum=a}this.maybeAutoVacuum()},o.prototype.replace=function(e){var t=this._options,r=t.idField,a=t.extractField,n=a(e,r);this.discard(n),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 a=r._enqueuedVacuumConditions;return r._enqueuedVacuumConditions=je,r.performVacuuming(e,a)}),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,a,n,i,s,u,l,d,h,v,y,b,E,g,S,C,T,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];a=e.batchSize||We.batchSize,n=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),d=l[0],h=l[1];try{for(v=(A=void 0,D(h)),y=v.next();!y.done;y=v.next()){b=W(y.value,2),E=b[0],g=b[1];try{for(S=(O=void 0,D(g)),C=S.next();!C.done;C=S.next())T=W(C.value,1),R=T[0],!this._documentIds.has(R)&&(g.size<=1?h.delete(E):g.delete(R))}catch(f){O={error:f}}finally{try{C&&!C.done&&(x=S.return)&&x.call(S)}finally{if(O)throw O.error}}}}catch(f){A={error:f}}finally{try{y&&!y.done&&(_=v.return)&&_.call(v)}finally{if(A)throw A.error}}return this._index.get(d).size===0&&this._index.delete(d),i%a!==0?[3,4]:[4,new Promise(function(f){return setTimeout(f,n)})];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,a;t===void 0&&(t={});var n=this.executeQuery(e,t),i=[];try{for(var s=D(n),u=s.next();!u.done;u=s.next()){var l=W(u.value,2),d=l[0],h=l[1],v=h.score,y=h.terms,b=h.match,E=y.length||1,g={id:this._documentIds.get(d),score:v*E,terms:Object.keys(b),queryTerms:y,match:b};Object.assign(g,this._storedFields.get(d)),(t.filter==null||t.filter(g))&&i.push(g)}}catch(S){r={error:S}}finally{try{u&&!u.done&&(a=s.return)&&a.call(s)}finally{if(r)throw r.error}}return e===o.wildcard&&t.boostDocument==null&&this._options.searchOptions.boostDocument==null||i.sort(ht),i},o.prototype.autoSuggest=function(e,t){var r,a,n,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 d=l.value,h=d.score,v=d.terms,y=v.join(" "),b=s.get(y);b!=null?(b.score+=h,b.count+=1):s.set(y,{score:h,terms:v,count:1})}}catch(L){r={error:L}}finally{try{l&&!l.done&&(a=u.return)&&a.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 C=W(S.value,2),b=C[0],T=C[1],h=T.score,v=T.terms,R=T.count;E.push({suggestion:b,terms:v,score:h/R})}}catch(L){n={error:L}}finally{try{S&&!S.done&&(i=g.return)&&i.call(g)}finally{if(n)throw n.error}}return E.sort(ht),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,a,n,i,s,u,l=e.index,d=e.documentCount,h=e.nextId,v=e.documentIds,y=e.fieldIds,b=e.fieldLength,E=e.averageFieldLength,g=e.storedFields,S=e.dirtCount,C=e.serializationVersion;if(C!==1&&C!==2)throw new Error("MiniSearch: cannot deserialize an index created with an incompatible version");var T=new o(t);T._documentCount=d,T._nextId=h,T._documentIds=Se(v),T._idToShortId=new Map,T._fieldIds=y,T._fieldLength=Se(b),T._avgFieldLength=E,T._storedFields=Se(g),T._dirtCount=S||0,T._index=new Re;try{for(var R=D(T._documentIds),L=R.next();!L.done;L=R.next()){var P=W(L.value,2),I=P[0],A=P[1];T._idToShortId.set(A,I)}}catch(m){r={error:m}}finally{try{L&&!L.done&&(a=R.return)&&a.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];C===1&&(z=z.ds),p.set(parseInt(M,10),Se(z))}}catch(m){s={error:m}}finally{try{k&&!k.done&&(u=N.return)&&u.call(N)}finally{if(s)throw s.error}}T._index.set(c,p)}}catch(m){n={error:m}}finally{try{O&&!O.done&&(i=_.return)&&i.call(_)}finally{if(n)throw n.error}}return T},o.prototype.executeQuery=function(e,t){var r=this;if(t===void 0&&(t={}),e===o.wildcard)return this.executeWildcardQuery(t);if(typeof e!="string"){var a=$($($({},t),e),{queries:void 0}),n=e.queries.map(function(g){return r.executeQuery(g,a)});return this.combineResults(n,a.combineWith)}var i=this._options,s=i.tokenize,u=i.processTerm,l=i.searchOptions,d=$($({tokenize:s,processTerm:u},l),t),h=d.tokenize,v=d.processTerm,y=h(e).flatMap(function(g){return v(g)}).filter(function(g){return!!g}),b=y.map(Wr(d)),E=b.map(function(g){return r.executeQuerySpec(g,d)});return this.combineResults(E,d.combineWith)},o.prototype.executeQuerySpec=function(e,t){var r,a,n,i,s=$($({},this._options.searchOptions),t),u=(s.fields||this._options.fields).reduce(function(M,z){var m;return $($({},M),(m={},m[z]=Le(s.boost,z)||1,m))},{}),l=s.boostDocument,d=s.weights,h=s.maxFuzzy,v=s.bm25,y=$($({},ft.weights),d),b=y.fuzzy,E=y.prefix,g=this._index.get(e.term),S=this.termResults(e.term,e.term,1,g,u,l,v),C,T;if(e.prefix&&(C=this._index.atPrefix(e.term)),e.fuzzy){var R=e.fuzzy===!0?.2:e.fuzzy,L=R<1?Math.min(h,Math.round(e.term.length*R)):R;L&&(T=this._index.fuzzyGet(e.term,L))}if(C)try{for(var P=D(C),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){T==null||T.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&&(a=P.return)&&a.call(P)}finally{if(r)throw r.error}}if(T)try{for(var f=D(T.keys()),p=f.next();!p.done;p=f.next()){var _=p.value,N=W(T.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){n={error:M}}finally{try{p&&!p.done&&(i=f.return)&&i.call(f)}finally{if(n)throw n.error}}return S},o.prototype.executeWildcardQuery=function(e){var t,r,a=new Map,n=$($({},this._options.searchOptions),e);try{for(var i=D(this._documentIds),s=i.next();!s.done;s=i.next()){var u=W(s.value,2),l=u[0],d=u[1],h=n.boostDocument?n.boostDocument(d,"",this._storedFields.get(l)):1;a.set(l,{score:h,terms:[],match:{}})}}catch(v){t={error:v}}finally{try{s&&!s.done&&(r=i.return)&&r.call(i)}finally{if(t)throw t.error}}return a},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,a,n=[];try{for(var i=D(this._index),s=i.next();!s.done;s=i.next()){var u=W(s.value,2),l=u[0],d=u[1],h={};try{for(var v=(r=void 0,D(d)),y=v.next();!y.done;y=v.next()){var b=W(y.value,2),E=b[0],g=b[1];h[E]=Object.fromEntries(g)}}catch(S){r={error:S}}finally{try{y&&!y.done&&(a=v.return)&&a.call(v)}finally{if(r)throw r.error}}n.push([l,h])}}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:n,serializationVersion:2}},o.prototype.termResults=function(e,t,r,a,n,i,s,u){var l,d,h,v,y;if(u===void 0&&(u=new Map),a==null)return u;try{for(var b=D(Object.keys(n)),E=b.next();!E.done;E=b.next()){var g=E.value,S=n[g],C=this._fieldIds[g],T=a.get(C);if(T!=null){var R=T.size,L=this._avgFieldLength[C];try{for(var P=(h=void 0,D(T.keys())),I=P.next();!I.done;I=P.next()){var A=I.value;if(!this._documentIds.has(A)){this.removeTerm(C,A,t),R-=1;continue}var _=i?i(this._documentIds.get(A),t,this._storedFields.get(A)):1;if(_){var O=T.get(A),x=this._fieldLength.get(A)[C],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:(y={},y[t]=[g],y)})}}}catch(k){h={error:k}}finally{try{I&&!I.done&&(v=P.return)&&v.call(P)}finally{if(h)throw h.error}}}}}catch(k){l={error:k}}finally{try{E&&!E.done&&(d=b.return)&&d.call(b)}finally{if(l)throw l.error}}return u},o.prototype.addTerm=function(e,t,r){var a=this._index.fetch(r,vt),n=a.get(e);if(n==null)n=new Map,n.set(t,1),a.set(e,n);else{var i=n.get(t);n.set(t,(i||0)+1)}},o.prototype.removeTerm=function(e,t,r){if(!this._index.has(r)){this.warnDocumentChanged(t,e,r);return}var a=this._index.fetch(r,vt),n=a.get(e);n==null||n.get(t)==null?this.warnDocumentChanged(t,e,r):n.get(t)<=1?n.size<=1?a.delete(e):n.delete(t):n.set(t,n.get(t)-1),this._index.get(r).size===0&&this._index.delete(r)},o.prototype.warnDocumentChanged=function(e,t,r){var a,n;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){a={error:l}}finally{try{s&&!s.done&&(n=i.return)&&n.call(i)}finally{if(a)throw a.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)),en=[Xr],tn=["placeholder"],rn={class:"search-actions"},nn=["title"],an=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)),on=[an],sn=["disabled","title"],un=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)),ln=[un],cn=["id","role","aria-labelledby"],fn=["aria-selected"],dn=["href","aria-label","onMouseenter","onFocusin"],hn={class:"titles"},vn=Y(()=>F("span",{class:"title-icon"},"#",-1)),pn=["innerHTML"],mn=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)),yn={class:"title main"},gn=["innerHTML"],bn={key:0,class:"excerpt-wrapper"},wn={key:0,class:"excerpt",inert:""},xn=["innerHTML"],Fn=Y(()=>F("div",{class:"excerpt-gradient-bottom"},null,-1)),En=Y(()=>F("div",{class:"excerpt-gradient-top"},null,-1)),Sn={key:0,class:"no-results"},An={class:"search-keyboard-shortcuts"},Tn=["aria-label"],Cn=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)),kn=[Cn],Nn=["aria-label"],In=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)),Dn=[In],_n=["aria-label"],On=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)),Rn=[On],Mn=["aria-label"],Ln=It({__name:"VPLocalSearchBox",emits:["close"],setup(o,{emit:e}){var M,z;const t=e,r=be(),a=be(),n=be(rr),i=er(),{activate:s}=Nr(r,{immediate:!0,allowOutsideClick:!0,clickOutsideDeactivates:!0,escapeDeactivates:!0}),{localeIndex:u,theme:l}=i,d=et(async()=>{var m,w,B,G,j,K,V,J,q;return at(zr.loadJSON((B=await((w=(m=n.value)[u.value])==null?void 0:w.call(m)))==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 m,w;return((m=l.value.search)==null?void 0:m.provider)==="local"&&((w=l.value.search.options)==null?void 0:w.disableQueryPersistence)===!0}).value?ie(""):Dt("vitepress:local-search-filter",""),y=_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 m,w,B;return((m=l.value.search)==null?void 0:m.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 m=((w=l.value.search)==null?void 0:w.options)??l.value.algolia;return((K=(j=(G=(B=m==null?void 0:m.locales)==null?void 0:B[u.value])==null?void 0:G.translations)==null?void 0:j.button)==null?void 0:K.buttonText)||((J=(V=m==null?void 0:m.translations)==null?void 0:V.button)==null?void 0:J.buttonText)||"Search"});Ot(()=>{b.value&&(y.value=!1)});const g=be([]),S=ie(!1);Be(v,()=>{S.value=!1});const C=et(async()=>{if(a.value)return at(new Dr(a.value))},null);Rt(()=>[d.value,v.value,y.value],async([m,w,B],G,j)=>{var Ue,He,Ge,qe;let K=!1;if(j(()=>{K=!0}),!m)return;g.value=m.search(w).slice(0,16),S.value=!0;const V=B?await Promise.all(g.value.map(Q=>T(Q.id))):[];if(K)return;const J=new Map;for(const{id:Q,mod:re}of V){const ne=Q.slice(0,Q.indexOf("#"));let ee=J.get(ne);if(ee)continue;ee=new Map,J.set(ne,ee);const Z=re.default??re;if(Z!=null&&Z.render||Z!=null&&Z.setup){const ae=Qt(Z);ae.config.warnHandler=()=>{},ae.provide(Yt,i),Object.defineProperties(ae.config.globalProperties,{$frontmatter:{get(){return i.frontmatter.value}},$params:{get(){return i.page.value.params}}});const Qe=document.createElement("div");ae.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)}),ae.unmount()}if(K)return}const q=new Set;if(g.value=g.value.map(Q=>{const[re,ne]=Q.id.split("#"),ee=J.get(re),Z=(ee==null?void 0:ee.get(ne))??"";for(const ae in Q.match)q.add(ae);return{...Q,text:Z}}),await de(),K)return;await new Promise(Q=>{var re;(re=C.value)==null||re.unmark({done:()=>{var ne;(ne=C.value)==null||ne.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=a.value)==null?void 0:Ge.firstElementChild)==null||qe.scrollIntoView({block:"start"})},{debounce:200,immediate:!0});async function T(m){const w=Zt(m.slice(0,m.indexOf("#")));try{if(!w)throw new Error(`Cannot find file for id: ${m}`);return{id:m,mod:await pt(()=>import(w),[])}}catch(B){return console.error(B),{id:m,mod:{}}}}const R=ie(),L=we(()=>{var m;return((m=v.value)==null?void 0:m.length)<=0});function P(m=!0){var w,B;(w=R.value)==null||w.focus(),m&&((B=R.value)==null||B.select())}De(()=>{P()});function I(m){m.pointerType==="mouse"&&P()}const A=ie(-1),_=ie(!1);Be(g,m=>{A.value=m.length?0:-1,O()});function O(){de(()=>{const m=document.querySelector(".result.selected");m&&m.scrollIntoView({block:"nearest"})})}xe("ArrowUp",m=>{m.preventDefault(),A.value--,A.value<0&&(A.value=g.value.length-1),_.value=!0,O()}),xe("ArrowDown",m=>{m.preventDefault(),A.value++,A.value>=g.value.length&&(A.value=0),_.value=!0,O()});const x=Mt();xe("Enter",m=>{if(m.target instanceof HTMLButtonElement&&m.target.type!=="submit")return;const w=g.value[A.value];if(m.target instanceof HTMLInputElement&&!w){m.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 m;return(m=l.value.search)==null?void 0:m.options}),c);De(()=>{window.history.pushState(null,"",null)}),zt("popstate",m=>{m.preventDefault(),t("close")});const p=Bt(Vt?document.body:null);De(()=>{de(()=>{p.value=!0,de().then(()=>s())})}),$t(()=>{p.value=!1});function N(){v.value="",de().then(()=>P(!1))}function k(m){return new RegExp([...m].sort((w,B)=>B.length-w.length).map(w=>`(${w.replace(/[|\\{}()[\]^$+*?.]/g,"\\$&").replace(/-/g,"\\x2d")})`).join("|"),"gi")}return(m,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=>m.$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=>m.$emit("close"))},en,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,tn),[[Jt,U(v)]]),F("div",rn,[b.value?Fe("",!0):(X(),te("button",{key:0,class:tt(["toggle-layout-button",{"detailed-list":U(y)}]),type:"button",title:U(f)("modal.displayDetails"),onClick:w[3]||(w[3]=V=>A.value>-1&&(y.value=!U(y)))},on,10,nn)),F("button",{class:"clear-button",type:"reset",disabled:L.value,title:U(f)("modal.resetButtonTitle"),onClick:N},ln,8,sn)])],32),F("ul",{ref_key:"resultsEl",ref:a,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(nt,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=>m.$emit("close"))},[F("div",null,[F("div",hn,[vn,(X(!0),te(nt,null,rt(V.titles,(q,Ie)=>(X(),te("span",{key:Ie,class:"title"},[F("span",{class:"text",innerHTML:q},null,8,pn),mn]))),128)),F("span",yn,[F("span",{class:"text",innerHTML:V.title},null,8,gn)])]),U(y)?(X(),te("div",bn,[V.text?(X(),te("div",wn,[F("div",{class:"vp-doc",innerHTML:V.text},null,8,xn)])):Fe("",!0),Fn,En])):Fe("",!0)])],42,dn)],8,fn))),128)),U(v)&&!g.value.length&&S.value?(X(),te("li",Sn,[he(ve(U(f)("modal.noResultsText"))+' "',1),F("strong",null,ve(U(v)),1),he('" ')])):Fe("",!0)],40,cn),F("div",An,[F("span",null,[F("kbd",{"aria-label":U(f)("modal.footer.navigateUpKeyAriaLabel")},kn,8,Tn),F("kbd",{"aria-label":U(f)("modal.footer.navigateDownKeyAriaLabel")},Dn,8,Nn),he(" "+ve(U(f)("modal.footer.navigateText")),1)]),F("span",null,[F("kbd",{"aria-label":U(f)("modal.footer.selectKeyAriaLabel")},Rn,8,_n),he(" "+ve(U(f)("modal.footer.selectText")),1)]),F("span",null,[F("kbd",{"aria-label":U(f)("modal.footer.closeKeyAriaLabel")},"esc",8,Mn),he(" "+ve(U(f)("modal.footer.closeText")),1)])])])],8,Ur)])}}});const $n=Xt(Ln,[["__scopeId","data-v-2813d7e3"]]);export{$n as default}; diff --git a/assets/chunks/arabica_versions.1930378b.js b/assets/chunks/arabica_versions.1930378b.js new file mode 100644 index 00000000000..f1a9d0e16fc --- /dev/null +++ b/assets/chunks/arabica_versions.1930378b.js @@ -0,0 +1 @@ +const e=Object.freeze({"app-latest-tag":"v2.1.2","app-latest-sha":"48173df3dc78f9348eedb3796f29ef9e9e5dc584","core-latest-tag":"v1.40.1-tm-v0.34.29-rc0","core-latest-sha":"0d2b63836d0f4587e162bfded58f53fba238e69c","node-latest-tag":"v0.16.0","node-latest-sha":"6744f648649ebb5fee1b27faf7aca96ecf4519b2"});export{e as a}; diff --git a/assets/chunks/constants.fa173a21.js b/assets/chunks/constants.fa173a21.js new file mode 100644 index 00000000000..a03f060a285 --- /dev/null +++ b/assets/chunks/constants.fa173a21.js @@ -0,0 +1 @@ +const a=Object.freeze({golangNodeMainnet:"1.23.0",golangNodeMocha:"1.23.0",golangNodeArabica:"1.23.0",golangApp:"1.22.6",golangCore:"1.22.6",golang:"1.22.0",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.1a91c06a.js b/assets/chunks/framework.1a91c06a.js new file mode 100644 index 00000000000..bf9379e8782 --- /dev/null +++ b/assets/chunks/framework.1a91c06a.js @@ -0,0 +1,33 @@ +/** +* @vue/shared v3.4.26 +* (c) 2018-present Yuxi (Evan) You and Vue contributors +* @license MIT +**//*! #__NO_SIDE_EFFECTS__ */function Tt(e,t){const n=new Set(e.split(","));return t?r=>n.has(r.toLowerCase()):r=>n.has(r)}const te={},yt=[],we=()=>{},Po=()=>!1,Yt=e=>e.charCodeAt(0)===111&&e.charCodeAt(1)===110&&(e.charCodeAt(2)>122||e.charCodeAt(2)<97),wr=e=>e.startsWith("onUpdate:"),oe=Object.assign,xr=(e,t)=>{const n=e.indexOf(t);n>-1&&e.splice(n,1)},No=Object.prototype.hasOwnProperty,X=(e,t)=>No.call(e,t),B=Array.isArray,vt=e=>Rn(e)==="[object Map]",Qs=e=>Rn(e)==="[object Set]",W=e=>typeof e=="function",Q=e=>typeof e=="string",ht=e=>typeof e=="symbol",ee=e=>e!==null&&typeof e=="object",ei=e=>(ee(e)||W(e))&&W(e.then)&&W(e.catch),ti=Object.prototype.toString,Rn=e=>ti.call(e),Fo=e=>Rn(e).slice(8,-1),ni=e=>Rn(e)==="[object Object]",Cr=e=>Q(e)&&e!=="NaN"&&e[0]!=="-"&&""+parseInt(e,10)===e,bt=Tt(",key,ref,ref_for,ref_key,onVnodeBeforeMount,onVnodeMounted,onVnodeBeforeUpdate,onVnodeUpdated,onVnodeBeforeUnmount,onVnodeUnmounted"),Mn=e=>{const t=Object.create(null);return n=>t[n]||(t[n]=e(n))},Ho=/-(\w)/g,Fe=Mn(e=>e.replace(Ho,(t,n)=>n?n.toUpperCase():"")),Do=/\B([A-Z])/g,nt=Mn(e=>e.replace(Do,"-$1").toLowerCase()),On=Mn(e=>e.charAt(0).toUpperCase()+e.slice(1)),pn=Mn(e=>e?`on${On(e)}`:""),et=(e,t)=>!Object.is(e,t),gn=(e,t)=>{for(let n=0;n{Object.defineProperty(e,t,{configurable:!0,enumerable:!1,writable:r,value:n})},or=e=>{const t=parseFloat(e);return isNaN(t)?e:t},$o=e=>{const t=Q(e)?Number(e):NaN;return isNaN(t)?e:t};let Xr;const Er=()=>Xr||(Xr=typeof globalThis<"u"?globalThis:typeof self<"u"?self:typeof window<"u"?window:typeof global<"u"?global:{});function In(e){if(B(e)){const t={};for(let n=0;n{if(n){const r=n.split(jo);r.length>1&&(t[r[0].trim()]=r[1].trim())}}),t}function Bo(e){let t="";if(!e||Q(e))return t;for(const n in e){const r=e[n],s=n.startsWith("--")?n:nt(n);(Q(r)||typeof r=="number")&&(t+=`${s}:${r};`)}return t}function Ln(e){let t="";if(Q(e))t=e;else if(B(e))for(let n=0;nQ(e)?e:e==null?"":B(e)||ee(e)&&(e.toString===ti||!W(e.toString))?JSON.stringify(e,ii,2):String(e),ii=(e,t)=>t&&t.__v_isRef?ii(e,t.value):vt(t)?{[`Map(${t.size})`]:[...t.entries()].reduce((n,[r,s],i)=>(n[Yn(r,i)+" =>"]=s,n),{})}:Qs(t)?{[`Set(${t.size})`]:[...t.values()].map(n=>Yn(n))}:ht(t)?Yn(t):ee(t)&&!B(t)&&!ni(t)?String(t):t,Yn=(e,t="")=>{var n;return ht(e)?`Symbol(${(n=e.description)!=null?n:t})`:e};/** +* @vue/reactivity v3.4.26 +* (c) 2018-present Yuxi (Evan) You and Vue contributors +* @license MIT +**/let be;class Yo{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=4))break}this._dirtyLevel===1&&(this._dirtyLevel=0),Ue()}return this._dirtyLevel>=4}set dirty(t){this._dirtyLevel=t?4:0}run(){if(this._dirtyLevel=0,!this.active)return this.fn();let t=Ze,n=ut;try{return Ze=!0,ut=this,this._runnings++,Qr(this),this.fn()}finally{es(this),this._runnings--,ut=n,Ze=t}}stop(){this.active&&(Qr(this),es(this),this.onStop&&this.onStop(),this.active=!1)}}function Xo(e){return e.value}function Qr(e){e._trackId++,e._depsLength=0}function es(e){if(e.deps.length>e._depsLength){for(let t=e._depsLength;t{const n=new Map;return n.cleanup=e,n.computed=t,n},bn=new WeakMap,ft=Symbol(""),ar=Symbol("");function ye(e,t,n){if(Ze&&ut){let r=bn.get(e);r||bn.set(e,r=new Map);let s=r.get(n);s||r.set(n,s=fi(()=>r.delete(n))),ai(ut,s)}}function ke(e,t,n,r,s,i){const o=bn.get(e);if(!o)return;let l=[];if(t==="clear")l=[...o.values()];else if(n==="length"&&B(e)){const c=Number(r);o.forEach((a,f)=>{(f==="length"||!ht(f)&&f>=c)&&l.push(a)})}else switch(n!==void 0&&l.push(o.get(n)),t){case"add":B(e)?Cr(n)&&l.push(o.get("length")):(l.push(o.get(ft)),vt(e)&&l.push(o.get(ar)));break;case"delete":B(e)||(l.push(o.get(ft)),vt(e)&&l.push(o.get(ar)));break;case"set":vt(e)&&l.push(o.get(ft));break}Ar();for(const c of l)c&&ui(c,4);Rr()}function Zo(e,t){const n=bn.get(e);return n&&n.get(t)}const Qo=Tt("__proto__,__v_isRef,__isVue"),di=new Set(Object.getOwnPropertyNames(Symbol).filter(e=>e!=="arguments"&&e!=="caller").map(e=>Symbol[e]).filter(ht)),ts=el();function el(){const e={};return["includes","indexOf","lastIndexOf"].forEach(t=>{e[t]=function(...n){const r=J(this);for(let i=0,o=this.length;i{e[t]=function(...n){Ve(),Ar();const r=J(this)[t].apply(this,n);return Rr(),Ue(),r}}),e}function tl(e){ht(e)||(e=String(e));const t=J(this);return ye(t,"has",e),t.hasOwnProperty(e)}class hi{constructor(t=!1,n=!1){this._isReadonly=t,this._isShallow=n}get(t,n,r){const s=this._isReadonly,i=this._isShallow;if(n==="__v_isReactive")return!s;if(n==="__v_isReadonly")return s;if(n==="__v_isShallow")return i;if(n==="__v_raw")return r===(s?i?pl:_i:i?mi:gi).get(t)||Object.getPrototypeOf(t)===Object.getPrototypeOf(r)?t:void 0;const o=B(t);if(!s){if(o&&X(ts,n))return Reflect.get(ts,n,r);if(n==="hasOwnProperty")return tl}const l=Reflect.get(t,n,r);return(ht(n)?di.has(n):Qo(n))||(s||ye(t,"get",n),i)?l:fe(l)?o&&Cr(n)?l:l.value:ee(l)?s?Fn(l):Nn(l):l}}class pi extends hi{constructor(t=!1){super(!1,t)}set(t,n,r,s){let i=t[n];if(!this._isShallow){const c=jt(i);if(!wn(r)&&!jt(r)&&(i=J(i),r=J(r)),!B(t)&&fe(i)&&!fe(r))return c?!1:(i.value=r,!0)}const o=B(t)&&Cr(n)?Number(n)e,Pn=e=>Reflect.getPrototypeOf(e);function en(e,t,n=!1,r=!1){e=e.__v_raw;const s=J(e),i=J(t);n||(et(t,i)&&ye(s,"get",t),ye(s,"get",i));const{has:o}=Pn(s),l=r?Mr:n?Lr:Vt;if(o.call(s,t))return l(e.get(t));if(o.call(s,i))return l(e.get(i));e!==s&&e.get(t)}function tn(e,t=!1){const n=this.__v_raw,r=J(n),s=J(e);return t||(et(e,s)&&ye(r,"has",e),ye(r,"has",s)),e===s?n.has(e):n.has(e)||n.has(s)}function nn(e,t=!1){return e=e.__v_raw,!t&&ye(J(e),"iterate",ft),Reflect.get(e,"size",e)}function ns(e){e=J(e);const t=J(this);return Pn(t).has.call(t,e)||(t.add(e),ke(t,"add",e,e)),this}function rs(e,t){t=J(t);const n=J(this),{has:r,get:s}=Pn(n);let i=r.call(n,e);i||(e=J(e),i=r.call(n,e));const o=s.call(n,e);return n.set(e,t),i?et(t,o)&&ke(n,"set",e,t):ke(n,"add",e,t),this}function ss(e){const t=J(this),{has:n,get:r}=Pn(t);let s=n.call(t,e);s||(e=J(e),s=n.call(t,e)),r&&r.call(t,e);const i=t.delete(e);return s&&ke(t,"delete",e,void 0),i}function is(){const e=J(this),t=e.size!==0,n=e.clear();return t&&ke(e,"clear",void 0,void 0),n}function rn(e,t){return function(r,s){const i=this,o=i.__v_raw,l=J(o),c=t?Mr:e?Lr:Vt;return!e&&ye(l,"iterate",ft),o.forEach((a,f)=>r.call(s,c(a),c(f),i))}}function sn(e,t,n){return function(...r){const s=this.__v_raw,i=J(s),o=vt(i),l=e==="entries"||e===Symbol.iterator&&o,c=e==="keys"&&o,a=s[e](...r),f=n?Mr:t?Lr:Vt;return!t&&ye(i,"iterate",c?ar:ft),{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:e==="clear"?void 0:this}}function ol(){const e={get(i){return en(this,i)},get size(){return nn(this)},has:tn,add:ns,set:rs,delete:ss,clear:is,forEach:rn(!1,!1)},t={get(i){return en(this,i,!1,!0)},get size(){return nn(this)},has:tn,add:ns,set:rs,delete:ss,clear:is,forEach:rn(!1,!0)},n={get(i){return en(this,i,!0)},get size(){return nn(this,!0)},has(i){return tn.call(this,i,!0)},add:Ke("add"),set:Ke("set"),delete:Ke("delete"),clear:Ke("clear"),forEach:rn(!0,!1)},r={get(i){return en(this,i,!0,!0)},get size(){return nn(this,!0)},has(i){return tn.call(this,i,!0)},add:Ke("add"),set:Ke("set"),delete:Ke("delete"),clear:Ke("clear"),forEach:rn(!0,!0)};return["keys","values","entries",Symbol.iterator].forEach(i=>{e[i]=sn(i,!1,!1),n[i]=sn(i,!0,!1),t[i]=sn(i,!1,!0),r[i]=sn(i,!0,!0)}),[e,n,t,r]}const[ll,cl,al,ul]=ol();function Or(e,t){const n=t?e?ul:al:e?cl:ll;return(r,s,i)=>s==="__v_isReactive"?!e:s==="__v_isReadonly"?e:s==="__v_raw"?r:Reflect.get(X(n,s)&&s in r?n:r,s,i)}const fl={get:Or(!1,!1)},dl={get:Or(!1,!0)},hl={get:Or(!0,!1)};const gi=new WeakMap,mi=new WeakMap,_i=new WeakMap,pl=new WeakMap;function gl(e){switch(e){case"Object":case"Array":return 1;case"Map":case"Set":case"WeakMap":case"WeakSet":return 2;default:return 0}}function ml(e){return e.__v_skip||!Object.isExtensible(e)?0:gl(Fo(e))}function Nn(e){return jt(e)?e:Ir(e,!1,rl,fl,gi)}function _l(e){return Ir(e,!1,il,dl,mi)}function Fn(e){return Ir(e,!0,sl,hl,_i)}function Ir(e,t,n,r,s){if(!ee(e)||e.__v_raw&&!(t&&e.__v_isReactive))return e;const i=s.get(e);if(i)return i;const o=ml(e);if(o===0)return e;const l=new Proxy(e,o===2?r:n);return s.set(e,l),l}function It(e){return jt(e)?It(e.__v_raw):!!(e&&e.__v_isReactive)}function jt(e){return!!(e&&e.__v_isReadonly)}function wn(e){return!!(e&&e.__v_isShallow)}function yi(e){return e?!!e.__v_raw:!1}function J(e){const t=e&&e.__v_raw;return t?J(t):e}function mn(e){return Object.isExtensible(e)&&ri(e,"__v_skip",!0),e}const Vt=e=>ee(e)?Nn(e):e,Lr=e=>ee(e)?Fn(e):e;class vi{constructor(t,n,r,s){this.getter=t,this._setter=n,this.dep=void 0,this.__v_isRef=!0,this.__v_isReadonly=!1,this.effect=new Sr(()=>t(this._value),()=>Lt(this,this.effect._dirtyLevel===2?2:3)),this.effect.computed=this,this.effect.active=this._cacheable=!s,this.__v_isReadonly=r}get value(){const t=J(this);return(!t._cacheable||t.effect.dirty)&&et(t._value,t._value=t.effect.run())&&Lt(t,4),Pr(t),t.effect._dirtyLevel>=2&&Lt(t,2),t._value}set value(t){this._setter(t)}get _dirty(){return this.effect.dirty}set _dirty(t){this.effect.dirty=t}}function yl(e,t,n=!1){let r,s;const i=W(e);return i?(r=e,s=we):(r=e.get,s=e.set),new vi(r,s,i||!s,n)}function Pr(e){var t;Ze&&ut&&(e=J(e),ai(ut,(t=e.dep)!=null?t:e.dep=fi(()=>e.dep=void 0,e instanceof vi?e:void 0)))}function Lt(e,t=4,n){e=J(e);const r=e.dep;r&&ui(r,t)}function fe(e){return!!(e&&e.__v_isRef===!0)}function le(e){return bi(e,!1)}function Nr(e){return bi(e,!0)}function bi(e,t){return fe(e)?e:new vl(e,t)}class vl{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:Vt(t)}get value(){return Pr(this),this._value}set value(t){const n=this.__v_isShallow||wn(t)||jt(t);t=n?t:J(t),et(t,this._rawValue)&&(this._rawValue=t,this._value=n?t:Vt(t),Lt(this,4))}}function Fr(e){return fe(e)?e.value:e}const bl={get:(e,t,n)=>Fr(Reflect.get(e,t,n)),set:(e,t,n,r)=>{const s=e[t];return fe(s)&&!fe(n)?(s.value=n,!0):Reflect.set(e,t,n,r)}};function wi(e){return It(e)?e:new Proxy(e,bl)}class wl{constructor(t){this.dep=void 0,this.__v_isRef=!0;const{get:n,set:r}=t(()=>Pr(this),()=>Lt(this));this._get=n,this._set=r}get value(){return this._get()}set value(t){this._set(t)}}function xl(e){return new wl(e)}class Cl{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 Zo(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 fe(e)?e:W(e)?new El(e):ee(e)&&arguments.length>1?Sl(e,t,n):le(e)}function Sl(e,t,n){const r=e[t];return fe(r)?r:new Cl(e,t,n)}/** +* @vue/runtime-core v3.4.26 +* (c) 2018-present Yuxi (Evan) You and Vue contributors +* @license MIT +**/const Pt=[];function ze(e,...t){Ve();const n=Pt.length?Pt[Pt.length-1].component:null,r=n&&n.appContext.config.warnHandler,s=Al();if(r)je(r,n,11,[e+t.map(i=>{var o,l;return(l=(o=i.toString)==null?void 0:o.call(i))!=null?l:JSON.stringify(i)}).join(""),n&&n.proxy,s.map(({vnode:i})=>`at <${lo(n,i.type)}>`).join(` +`),s]);else{const i=[`[Vue warn]: ${e}`,...t];s.length&&i.push(` +`,...Rl(s)),console.warn(...i)}Ue()}function Al(){let e=Pt[Pt.length-1];if(!e)return[];const t=[];for(;e;){const n=t[0];n&&n.vnode===e?n.recurseCount++:t.push({vnode:e,recurseCount:0});const r=e.component&&e.component.parent;e=r&&r.vnode}return t}function Rl(e){const t=[];return e.forEach((n,r)=>{t.push(...r===0?[]:[` +`],...Ml(n))}),t}function Ml({vnode:e,recurseCount:t}){const n=t>0?`... (${t} recursive calls)`:"",r=e.component?e.component.parent==null:!1,s=` at <${lo(e.component,e.type,r)}`,i=">"+n;return e.props?[s,...Ol(e.props),i]:[s+i]}function Ol(e){const t=[],n=Object.keys(e);return n.slice(0,3).forEach(r=>{t.push(...xi(r,e[r]))}),n.length>3&&t.push(" ..."),t}function xi(e,t,n){return Q(t)?(t=JSON.stringify(t),n?t:[`${e}=${t}`]):typeof t=="number"||typeof t=="boolean"||t==null?n?t:[`${e}=${t}`]:fe(t)?(t=xi(e,J(t.value),!0),n?t:[`${e}=Ref<`,t,">"]):W(t)?[`${e}=fn${t.name?`<${t.name}>`:""}`]:(t=J(t),n?t:[`${e}=`,t])}function je(e,t,n,r){try{return r?e(...r):e()}catch(s){Gt(s,t,n)}}function Te(e,t,n,r){if(W(e)){const s=je(e,t,n,r);return s&&ei(s)&&s.catch(i=>{Gt(i,t,n)}),s}if(B(e)){const s=[];for(let i=0;i>>1,s=he[r],i=Bt(s);iNe&&he.splice(t,1)}function Nl(e){B(e)?wt.push(...e):(!Ye||!Ye.includes(e,e.allowRecurse?lt+1:lt))&&wt.push(e),Ei()}function os(e,t,n=Ut?Ne+1:0){for(;nBt(n)-Bt(r));if(wt.length=0,Ye){Ye.push(...t);return}for(Ye=t,lt=0;lte.id==null?1/0:e.id,Fl=(e,t)=>{const n=Bt(e)-Bt(t);if(n===0){if(e.pre&&!t.pre)return-1;if(t.pre&&!e.pre)return 1}return n};function Ti(e){ur=!1,Ut=!0,he.sort(Fl);const t=we;try{for(Ne=0;NeQ(y)?y.trim():y)),h&&(s=n.map(or))}let l,c=r[l=pn(t)]||r[l=pn(Fe(t))];!c&&i&&(c=r[l=pn(nt(t))]),c&&Te(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,Te(a,e,6,s)}}function Si(e,t,n=!1){const r=t.emitsCache,s=r.get(e);if(s!==void 0)return s;const i=e.emits;let o={},l=!1;if(!W(e)){const c=a=>{const f=Si(a,t,!0);f&&(l=!0,oe(o,f))};!n&&t.mixins.length&&t.mixins.forEach(c),e.extends&&c(e.extends),e.mixins&&e.mixins.forEach(c)}return!i&&!l?(ee(e)&&r.set(e,null),null):(B(i)?i.forEach(c=>o[c]=null):oe(o,i),ee(e)&&r.set(e,o),o)}function $n(e,t){return!e||!Yt(t)?!1:(t=t.slice(2).replace(/Once$/,""),X(e,t[0].toLowerCase()+t.slice(1))||X(e,nt(t))||X(e,t))}let ce=null,kn=null;function Cn(e){const t=ce;return ce=e,kn=e&&e.type.__scopeId||null,t}function gu(e){kn=e}function mu(){kn=null}function Dl(e,t=ce,n){if(!t||e._n)return e;const r=(...s)=>{r._d&&xs(-1);const i=Cn(t);let o;try{o=e(...s)}finally{Cn(i),r._d&&xs(1)}return o};return r._n=!0,r._c=!0,r._d=!0,r}function Gn(e){const{type:t,vnode:n,proxy:r,withProxy:s,propsOptions:[i],slots:o,attrs:l,emit:c,render:a,renderCache:f,props:h,data:g,setupState:y,ctx:C,inheritAttrs:I}=e,k=Cn(e);let K,j;try{if(n.shapeFlag&4){const _=s||r,L=_;K=Ae(a.call(L,_,f,h,y,g,C)),j=l}else{const _=t;K=Ae(_.length>1?_(h,{attrs:l,slots:o,emit:c}):_(h,null)),j=t.props?l:$l(l)}}catch(_){$t.length=0,Gt(_,e,1),K=ie(_e)}let p=K;if(j&&I!==!1){const _=Object.keys(j),{shapeFlag:L}=p;_.length&&L&7&&(i&&_.some(wr)&&(j=kl(j,i)),p=tt(p,j,!1,!0))}return n.dirs&&(p=tt(p,null,!1,!0),p.dirs=p.dirs?p.dirs.concat(n.dirs):n.dirs),n.transition&&(p.transition=n.transition),K=p,Cn(k),K}const $l=e=>{let t;for(const n in e)(n==="class"||n==="style"||Yt(n))&&((t||(t={}))[n]=e[n]);return t},kl=(e,t)=>{const n={};for(const r in e)(!wr(r)||!(r.slice(9)in t))&&(n[r]=e[r]);return n};function jl(e,t,n){const{props:r,children:s,component:i}=e,{props:o,children:l,patchFlag:c}=t,a=i.emitsOptions;if(t.dirs||t.transition)return!0;if(n&&c>=0){if(c&1024)return!0;if(c&16)return r?ls(r,o,a):!!o;if(c&8){const f=t.dynamicProps;for(let h=0;he.__isSuspense;function Mi(e,t){t&&t.pendingBranch?B(e)?t.effects.push(...e):t.effects.push(e):Nl(e)}const Bl=Symbol.for("v-scx"),Kl=()=>Ct(Bl);function $r(e,t){return jn(e,null,t)}function vu(e,t){return jn(e,null,{flush:"post"})}const on={};function Qe(e,t,n){return jn(e,t,n)}function jn(e,t,{immediate:n,deep:r,flush:s,once:i,onTrack:o,onTrigger:l}=te){if(t&&i){const M=t;t=(...N)=>{M(...N),L()}}const c=ue,a=M=>r===!0?M:at(M,r===!1?1:void 0);let f,h=!1,g=!1;if(fe(e)?(f=()=>e.value,h=wn(e)):It(e)?(f=()=>a(e),h=!0):B(e)?(g=!0,h=e.some(M=>It(M)||wn(M)),f=()=>e.map(M=>{if(fe(M))return M.value;if(It(M))return a(M);if(W(M))return je(M,c,2)})):W(e)?t?f=()=>je(e,c,2):f=()=>(y&&y(),Te(e,c,3,[C])):f=we,t&&r){const M=f;f=()=>at(M())}let y,C=M=>{y=p.onStop=()=>{je(M,c,4),y=p.onStop=void 0}},I;if(Zt)if(C=we,t?n&&Te(t,c,3,[f(),g?[]:void 0,C]):f(),s==="sync"){const M=Kl();I=M.__watcherHandles||(M.__watcherHandles=[])}else return we;let k=g?new Array(e.length).fill(on):on;const K=()=>{if(!(!p.active||!p.dirty))if(t){const M=p.run();(r||h||(g?M.some((N,S)=>et(N,k[S])):et(M,k)))&&(y&&y(),Te(t,c,3,[M,k===on?void 0:g&&k[0]===on?[]:k,C]),k=M)}else p.run()};K.allowRecurse=!!t;let j;s==="sync"?j=K:s==="post"?j=()=>me(K,c&&c.suspense):(K.pre=!0,c&&(K.id=c.uid),j=()=>Dn(K));const p=new Sr(f,we,j),_=oi(),L=()=>{p.stop(),_&&xr(_.effects,p)};return t?n?K():k=p.run():s==="post"?me(p.run.bind(p),c&&c.suspense):p.run(),I&&I.push(L),L}function Wl(e,t,n){const r=this.proxy,s=Q(e)?e.includes(".")?Oi(r,e):()=>r[e]:e.bind(r,r);let i;W(t)?i=t:(i=t.handler,n=t);const o=Xt(this),l=jn(s,i.bind(r),n);return o(),l}function Oi(e,t){const n=t.split(".");return()=>{let r=e;for(let s=0;s{at(r,t,n)});else if(ni(e))for(const r in e)at(e[r],t,n);return e}function bu(e,t){if(ce===null)return e;const n=Kn(ce)||ce.proxy,r=e.dirs||(e.dirs=[]);for(let s=0;s{e.isMounted=!0}),Fi(()=>{e.isUnmounting=!0}),e}const xe=[Function,Array],Ii={mode:String,appear:Boolean,persisted:Boolean,onBeforeEnter:xe,onEnter:xe,onAfterEnter:xe,onEnterCancelled:xe,onBeforeLeave:xe,onLeave:xe,onAfterLeave:xe,onLeaveCancelled:xe,onBeforeAppear:xe,onAppear:xe,onAfterAppear:xe,onAppearCancelled:xe},zl={name:"BaseTransition",props:Ii,setup(e,{slots:t}){const n=Bn(),r=ql();return()=>{const s=t.default&&Pi(t.default(),!0);if(!s||!s.length)return;let i=s[0];if(s.length>1){for(const g of s)if(g.type!==_e){i=g;break}}const o=J(e),{mode:l}=o;if(r.isLeaving)return Jn(i);const c=as(i);if(!c)return Jn(i);const a=fr(c,o,r,n);dr(c,a);const f=n.subTree,h=f&&as(f);if(h&&h.type!==_e&&!ct(c,h)){const g=fr(h,o,r,n);if(dr(h,g),l==="out-in"&&c.type!==_e)return r.isLeaving=!0,g.afterLeave=()=>{r.isLeaving=!1,n.update.active!==!1&&(n.effect.dirty=!0,n.update())},Jn(i);l==="in-out"&&c.type!==_e&&(g.delayLeave=(y,C,I)=>{const k=Li(r,h);k[String(h.key)]=h,y[Ge]=()=>{C(),y[Ge]=void 0,delete a.delayedLeave},a.delayedLeave=I})}return i}}},Yl=zl;function Li(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 fr(e,t,n,r){const{appear:s,mode:i,persisted:o=!1,onBeforeEnter:l,onEnter:c,onAfterEnter:a,onEnterCancelled:f,onBeforeLeave:h,onLeave:g,onAfterLeave:y,onLeaveCancelled:C,onBeforeAppear:I,onAppear:k,onAfterAppear:K,onAppearCancelled:j}=t,p=String(e.key),_=Li(n,e),L=(S,F)=>{S&&Te(S,r,9,F)},M=(S,F)=>{const w=F[1];L(S,F),B(S)?S.every(H=>H.length<=1)&&w():S.length<=1&&w()},N={mode:i,persisted:o,beforeEnter(S){let F=l;if(!n.isMounted)if(s)F=I||l;else return;S[Ge]&&S[Ge](!0);const w=_[p];w&&ct(e,w)&&w.el[Ge]&&w.el[Ge](),L(F,[S])},enter(S){let F=c,w=a,H=f;if(!n.isMounted)if(s)F=k||c,w=K||a,H=j||f;else return;let R=!1;const z=S[ln]=ne=>{R||(R=!0,ne?L(H,[S]):L(w,[S]),N.delayedLeave&&N.delayedLeave(),S[ln]=void 0)};F?M(F,[S,z]):z()},leave(S,F){const w=String(e.key);if(S[ln]&&S[ln](!0),n.isUnmounting)return F();L(h,[S]);let H=!1;const R=S[Ge]=z=>{H||(H=!0,F(),z?L(C,[S]):L(y,[S]),S[Ge]=void 0,_[w]===e&&delete _[w])};_[w]=e,g?M(g,[S,R]):R()},clone(S){return fr(S,t,n,r)}};return N}function Jn(e){if(Jt(e))return e=tt(e),e.children=null,e}function as(e){if(!Jt(e))return e;const{shapeFlag:t,children:n}=e;if(n){if(t&16)return n[0];if(t&32&&W(n.default))return n.default()}}function dr(e,t){e.shapeFlag&6&&e.component?dr(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 Pi(e,t=!1,n){let r=[],s=0;for(let i=0;i1)for(let i=0;ioe({name:e.name},t,{setup:e}))():e}const xt=e=>!!e.type.__asyncLoader;/*! #__NO_SIDE_EFFECTS__ */function wu(e){W(e)&&(e={loader:e});const{loader:t,loadingComponent:n,errorComponent:r,delay:s=200,timeout:i,suspensible:o=!0,onError:l}=e;let c=null,a,f=0;const h=()=>(f++,c=null,g()),g=()=>{let y;return c||(y=c=t().catch(C=>{if(C=C instanceof Error?C:new Error(String(C)),l)return new Promise((I,k)=>{l(C,()=>I(h()),()=>k(C),f+1)});throw C}).then(C=>y!==c&&c?c:(C&&(C.__esModule||C[Symbol.toStringTag]==="Module")&&(C=C.default),a=C,C)))};return kr({name:"AsyncComponentWrapper",__asyncLoader:g,get __asyncResolved(){return a},setup(){const y=ue;if(a)return()=>Xn(a,y);const C=j=>{c=null,Gt(j,y,13,!r)};if(o&&y.suspense||Zt)return g().then(j=>()=>Xn(j,y)).catch(j=>(C(j),()=>r?ie(r,{error:j}):null));const I=le(!1),k=le(),K=le(!!s);return s&&setTimeout(()=>{K.value=!1},s),i!=null&&setTimeout(()=>{if(!I.value&&!k.value){const j=new Error(`Async component timed out after ${i}ms.`);C(j),k.value=j}},i),g().then(()=>{I.value=!0,y.parent&&Jt(y.parent.vnode)&&(y.parent.effect.dirty=!0,Dn(y.parent.update))}).catch(j=>{C(j),k.value=j}),()=>{if(I.value&&a)return Xn(a,y);if(k.value&&r)return ie(r,{error:k.value});if(n&&!K.value)return ie(n)}}})}function Xn(e,t){const{ref:n,props:r,children:s,ce:i}=t.vnode,o=ie(e,r,s);return o.ref=n,o.ce=i,delete t.vnode.ce,o}const Jt=e=>e.type.__isKeepAlive;function Gl(e,t){Ni(e,"a",t)}function Jl(e,t){Ni(e,"da",t)}function Ni(e,t,n=ue){const r=e.__wdc||(e.__wdc=()=>{let s=n;for(;s;){if(s.isDeactivated)return;s=s.parent}return e()});if(Vn(t,r,n),n){let s=n.parent;for(;s&&s.parent;)Jt(s.parent.vnode)&&Xl(r,t,n,s),s=s.parent}}function Xl(e,t,n,r){const s=Vn(t,e,r,!0);Un(()=>{xr(r[t],s)},n)}function Vn(e,t,n=ue,r=!1){if(n){const s=n[e]||(n[e]=[]),i=t.__weh||(t.__weh=(...o)=>{if(n.isUnmounted)return;Ve();const l=Xt(n),c=Te(t,n,e,o);return l(),Ue(),c});return r?s.unshift(i):s.push(i),i}}const Be=e=>(t,n=ue)=>(!Zt||e==="sp")&&Vn(e,(...r)=>t(...r),n),Zl=Be("bm"),St=Be("m"),Ql=Be("bu"),ec=Be("u"),Fi=Be("bum"),Un=Be("um"),tc=Be("sp"),nc=Be("rtg"),rc=Be("rtc");function sc(e,t=ue){Vn("ec",e,t)}function xu(e,t,n,r){let s;const i=n&&n[r];if(B(e)||Q(e)){s=new Array(e.length);for(let o=0,l=e.length;ot(o,l,void 0,i&&i[l]));else{const o=Object.keys(e);s=new Array(o.length);for(let l=0,c=o.length;lSn(t)?!(t.type===_e||t.type===ge&&!Hi(t.children)):!0)?e:null}function Eu(e,t){const n={};for(const r in e)n[t&&/[A-Z]/.test(r)?`on:${r}`:pn(r)]=e[r];return n}const hr=e=>e?ro(e)?Kn(e)||e.proxy:hr(e.parent):null,Nt=oe(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=>hr(e.parent),$root:e=>hr(e.root),$emit:e=>e.emit,$options:e=>jr(e),$forceUpdate:e=>e.f||(e.f=()=>{e.effect.dirty=!0,Dn(e.update)}),$nextTick:e=>e.n||(e.n=Hn.bind(e.proxy)),$watch:e=>Wl.bind(e)}),Zn=(e,t)=>e!==te&&!e.__isScriptSetup&&X(e,t),ic={get({_:e},t){if(t==="__v_skip")return!0;const{ctx:n,setupState:r,data:s,props:i,accessCache:o,type:l,appContext:c}=e;let a;if(t[0]!=="$"){const y=o[t];if(y!==void 0)switch(y){case 1:return r[t];case 2:return s[t];case 4:return n[t];case 3:return i[t]}else{if(Zn(r,t))return o[t]=1,r[t];if(s!==te&&X(s,t))return o[t]=2,s[t];if((a=e.propsOptions[0])&&X(a,t))return o[t]=3,i[t];if(n!==te&&X(n,t))return o[t]=4,n[t];pr&&(o[t]=0)}}const f=Nt[t];let h,g;if(f)return t==="$attrs"&&ye(e.attrs,"get",""),f(e);if((h=l.__cssModules)&&(h=h[t]))return h;if(n!==te&&X(n,t))return o[t]=4,n[t];if(g=c.config.globalProperties,X(g,t))return g[t]},set({_:e},t,n){const{data:r,setupState:s,ctx:i}=e;return Zn(s,t)?(s[t]=n,!0):r!==te&&X(r,t)?(r[t]=n,!0):X(e.props,t)||t[0]==="$"&&t.slice(1)in e?!1:(i[t]=n,!0)},has({_:{data:e,setupState:t,accessCache:n,ctx:r,appContext:s,propsOptions:i}},o){let l;return!!n[o]||e!==te&&X(e,o)||Zn(t,o)||(l=i[0])&&X(l,o)||X(r,o)||X(Nt,o)||X(s.config.globalProperties,o)},defineProperty(e,t,n){return n.get!=null?e._.accessCache[t]=0:X(n,"value")&&this.set(e,t,n.value,null),Reflect.defineProperty(e,t,n)}};function Tu(){return oc().slots}function oc(){const e=Bn();return e.setupContext||(e.setupContext=io(e))}function us(e){return B(e)?e.reduce((t,n)=>(t[n]=null,t),{}):e}let pr=!0;function lc(e){const t=jr(e),n=e.proxy,r=e.ctx;pr=!1,t.beforeCreate&&fs(t.beforeCreate,e,"bc");const{data:s,computed:i,methods:o,watch:l,provide:c,inject:a,created:f,beforeMount:h,mounted:g,beforeUpdate:y,updated:C,activated:I,deactivated:k,beforeDestroy:K,beforeUnmount:j,destroyed:p,unmounted:_,render:L,renderTracked:M,renderTriggered:N,errorCaptured:S,serverPrefetch:F,expose:w,inheritAttrs:H,components:R,directives:z,filters:ne}=t;if(a&&cc(a,r,null),o)for(const G in o){const D=o[G];W(D)&&(r[G]=D.bind(n))}if(s){const G=s.call(n,n);ee(G)&&(e.data=Nn(G))}if(pr=!0,i)for(const G in i){const D=i[G],Oe=W(D)?D.bind(n,n):W(D.get)?D.get.bind(n,n):we,pt=!W(D)&&W(D.set)?D.set.bind(n):we,rt=se({get:Oe,set:pt});Object.defineProperty(r,G,{enumerable:!0,configurable:!0,get:()=>rt.value,set:Ie=>rt.value=Ie})}if(l)for(const G in l)Di(l[G],r,n,G);if(c){const G=W(c)?c.call(n):c;Reflect.ownKeys(G).forEach(D=>{pc(D,G[D])})}f&&fs(f,e,"c");function V(G,D){B(D)?D.forEach(Oe=>G(Oe.bind(n))):D&&G(D.bind(n))}if(V(Zl,h),V(St,g),V(Ql,y),V(ec,C),V(Gl,I),V(Jl,k),V(sc,S),V(rc,M),V(nc,N),V(Fi,j),V(Un,_),V(tc,F),B(w))if(w.length){const G=e.exposed||(e.exposed={});w.forEach(D=>{Object.defineProperty(G,D,{get:()=>n[D],set:Oe=>n[D]=Oe})})}else e.exposed||(e.exposed={});L&&e.render===we&&(e.render=L),H!=null&&(e.inheritAttrs=H),R&&(e.components=R),z&&(e.directives=z)}function cc(e,t,n=we){B(e)&&(e=gr(e));for(const r in e){const s=e[r];let i;ee(s)?"default"in s?i=Ct(s.from||r,s.default,!0):i=Ct(s.from||r):i=Ct(s),fe(i)?Object.defineProperty(t,r,{enumerable:!0,configurable:!0,get:()=>i.value,set:o=>i.value=o}):t[r]=i}}function fs(e,t,n){Te(B(e)?e.map(r=>r.bind(t.proxy)):e.bind(t.proxy),t,n)}function Di(e,t,n,r){const s=r.includes(".")?Oi(n,r):()=>n[r];if(Q(e)){const i=t[e];W(i)&&Qe(s,i)}else if(W(e))Qe(s,e.bind(n));else if(ee(e))if(B(e))e.forEach(i=>Di(i,t,n,r));else{const i=W(e.handler)?e.handler.bind(n):t[e.handler];W(i)&&Qe(s,i,e)}}function jr(e){const t=e.type,{mixins:n,extends:r}=t,{mixins:s,optionsCache:i,config:{optionMergeStrategies:o}}=e.appContext,l=i.get(t);let c;return l?c=l:!s.length&&!n&&!r?c=t:(c={},s.length&&s.forEach(a=>En(c,a,o,!0)),En(c,t,o)),ee(t)&&i.set(t,c),c}function En(e,t,n,r=!1){const{mixins:s,extends:i}=t;i&&En(e,i,n,!0),s&&s.forEach(o=>En(e,o,n,!0));for(const o in t)if(!(r&&o==="expose")){const l=ac[o]||n&&n[o];e[o]=l?l(e[o],t[o]):t[o]}return e}const ac={data:ds,props:hs,emits:hs,methods:Ot,computed:Ot,beforeCreate:pe,created:pe,beforeMount:pe,mounted:pe,beforeUpdate:pe,updated:pe,beforeDestroy:pe,beforeUnmount:pe,destroyed:pe,unmounted:pe,activated:pe,deactivated:pe,errorCaptured:pe,serverPrefetch:pe,components:Ot,directives:Ot,watch:fc,provide:ds,inject:uc};function ds(e,t){return t?e?function(){return oe(W(e)?e.call(this,this):e,W(t)?t.call(this,this):t)}:t:e}function uc(e,t){return Ot(gr(e),gr(t))}function gr(e){if(B(e)){const t={};for(let n=0;n1)return n&&W(t)?t.call(r&&r.proxy):t}}const ki={},ji=()=>Object.create(ki),Vi=e=>Object.getPrototypeOf(e)===ki;function gc(e,t,n,r=!1){const s={},i=ji();e.propsDefaults=Object.create(null),Ui(e,t,s,i);for(const o in e.propsOptions[0])o in s||(s[o]=void 0);n?e.props=r?s:_l(s):e.type.props?e.props=s:e.props=i,e.attrs=i}function mc(e,t,n,r){const{props:s,attrs:i,vnode:{patchFlag:o}}=e,l=J(s),[c]=e.propsOptions;let a=!1;if((r||o>0)&&!(o&16)){if(o&8){const f=e.vnode.dynamicProps;for(let h=0;h{c=!0;const[g,y]=Bi(h,t,!0);oe(o,g),y&&l.push(...y)};!n&&t.mixins.length&&t.mixins.forEach(f),e.extends&&f(e.extends),e.mixins&&e.mixins.forEach(f)}if(!i&&!c)return ee(e)&&r.set(e,yt),yt;if(B(i))for(let f=0;f-1,y[1]=I<0||C-1||X(y,"default"))&&l.push(h)}}}const a=[o,l];return ee(e)&&r.set(e,a),a}function ps(e){return e[0]!=="$"&&!bt(e)}function gs(e){return e===null?"null":typeof e=="function"?e.name||"":typeof e=="object"&&e.constructor&&e.constructor.name||""}function ms(e,t){return gs(e)===gs(t)}function _s(e,t){return B(t)?t.findIndex(n=>ms(n,e)):W(t)&&ms(t,e)?0:-1}const Ki=e=>e[0]==="_"||e==="$stable",Vr=e=>B(e)?e.map(Ae):[Ae(e)],_c=(e,t,n)=>{if(t._n)return t;const r=Dl((...s)=>Vr(t(...s)),n);return r._c=!1,r},Wi=(e,t,n)=>{const r=e._ctx;for(const s in e){if(Ki(s))continue;const i=e[s];if(W(i))t[s]=_c(s,i,r);else if(i!=null){const o=Vr(i);t[s]=()=>o}}},qi=(e,t)=>{const n=Vr(t);e.slots.default=()=>n},yc=(e,t)=>{const n=e.slots=ji();if(e.vnode.shapeFlag&32){const r=t._;r?(oe(n,t),ri(n,"_",r,!0)):Wi(t,n)}else t&&qi(e,t)},vc=(e,t,n)=>{const{vnode:r,slots:s}=e;let i=!0,o=te;if(r.shapeFlag&32){const l=t._;l?n&&l===1?i=!1:(oe(s,t),!n&&l===1&&delete s._):(i=!t.$stable,Wi(t,s)),o=t}else t&&(qi(e,t),o={default:1});if(i)for(const l in s)!Ki(l)&&o[l]==null&&delete s[l]};function Tn(e,t,n,r,s=!1){if(B(e)){e.forEach((g,y)=>Tn(g,t&&(B(t)?t[y]:t),n,r,s));return}if(xt(r)&&!s)return;const i=r.shapeFlag&4?Kn(r.component)||r.component.proxy:r.el,o=s?null:i,{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&&(Q(a)?(f[a]=null,X(h,a)&&(h[a]=null)):fe(a)&&(a.value=null)),W(c))je(c,l,12,[o,f]);else{const g=Q(c),y=fe(c);if(g||y){const C=()=>{if(e.f){const I=g?X(h,c)?h[c]:f[c]:c.value;s?B(I)&&xr(I,i):B(I)?I.includes(i)||I.push(i):g?(f[c]=[i],X(h,c)&&(h[c]=f[c])):(c.value=[i],e.k&&(f[e.k]=c.value))}else g?(f[c]=o,X(h,c)&&(h[c]=o)):y&&(c.value=o,e.k&&(f[e.k]=o))};o?(C.id=-1,me(C,n)):C()}}}let $e=!1;const bc=e=>e.namespaceURI.includes("svg")&&e.tagName!=="foreignObject",wc=e=>e.namespaceURI.includes("MathML"),cn=e=>{if(bc(e))return"svg";if(wc(e))return"mathml"},Rt=e=>e.nodeType===8;function xc(e){const{mt:t,p:n,o:{patchProp:r,createText:s,nextSibling:i,parentNode:o,remove:l,insert:c,createComment:a}}=e,f=(p,_)=>{if(!_.hasChildNodes()){__VUE_PROD_HYDRATION_MISMATCH_DETAILS__&&ze("Attempting to hydrate existing markup but container is empty. Performing full mount instead."),n(null,p,_),xn(),_._vnode=p;return}$e=!1,h(_.firstChild,p,null,null,null),xn(),_._vnode=p,$e&&console.error("Hydration completed but contains mismatches.")},h=(p,_,L,M,N,S=!1)=>{S=S||!!_.dynamicChildren;const F=Rt(p)&&p.data==="[",w=()=>I(p,_,L,M,N,F),{type:H,ref:R,shapeFlag:z,patchFlag:ne}=_;let de=p.nodeType;_.el=p,ne===-2&&(S=!1,_.dynamicChildren=null);let V=null;switch(H){case Et:de!==3?_.children===""?(c(_.el=s(""),o(p),p),V=p):V=w():(p.data!==_.children&&($e=!0,__VUE_PROD_HYDRATION_MISMATCH_DETAILS__&&ze("Hydration text mismatch in",p.parentNode,` + - rendered on server: ${JSON.stringify(p.data)} + - expected on client: ${JSON.stringify(_.children)}`),p.data=_.children),V=i(p));break;case _e:j(p)?(V=i(p),K(_.el=p.content.firstChild,p,L)):de!==8||F?V=w():V=i(p);break;case Dt:if(F&&(p=i(p),de=p.nodeType),de===1||de===3){V=p;const G=!_.children.length;for(let D=0;D<_.staticCount;D++)G&&(_.children+=V.nodeType===1?V.outerHTML:V.data),D===_.staticCount-1&&(_.anchor=V),V=i(V);return F?i(V):V}else w();break;case ge:F?V=C(p,_,L,M,N,S):V=w();break;default:if(z&1)(de!==1||_.type.toLowerCase()!==p.tagName.toLowerCase())&&!j(p)?V=w():V=g(p,_,L,M,N,S);else if(z&6){_.slotScopeIds=N;const G=o(p);if(F?V=k(p):Rt(p)&&p.data==="teleport start"?V=k(p,p.data,"teleport end"):V=i(p),t(_,G,null,L,M,cn(G),S),xt(_)){let D;F?(D=ie(ge),D.anchor=V?V.previousSibling:G.lastChild):D=p.nodeType===3?no(""):ie("div"),D.el=p,_.component.subTree=D}}else z&64?de!==8?V=w():V=_.type.hydrate(p,_,L,M,N,S,e,y):z&128?V=_.type.hydrate(p,_,L,M,cn(o(p)),N,S,e,h):__VUE_PROD_HYDRATION_MISMATCH_DETAILS__&&ze("Invalid HostVNode type:",H,`(${typeof H})`)}return R!=null&&Tn(R,null,M,_),V},g=(p,_,L,M,N,S)=>{S=S||!!_.dynamicChildren;const{type:F,props:w,patchFlag:H,shapeFlag:R,dirs:z,transition:ne}=_,de=F==="input"||F==="option";if(de||H!==-1){z&&Pe(_,null,L,"created");let V=!1;if(j(p)){V=Yi(M,ne)&&L&&L.vnode.props&&L.vnode.props.appear;const D=p.content.firstChild;V&&ne.beforeEnter(D),K(D,p,L),_.el=p=D}if(R&16&&!(w&&(w.innerHTML||w.textContent))){let D=y(p.firstChild,_,p,L,M,N,S),Oe=!1;for(;D;){$e=!0,__VUE_PROD_HYDRATION_MISMATCH_DETAILS__&&!Oe&&(ze("Hydration children mismatch on",p,` +Server rendered element contains more child nodes than client vdom.`),Oe=!0);const pt=D;D=D.nextSibling,l(pt)}}else R&8&&p.textContent!==_.children&&($e=!0,__VUE_PROD_HYDRATION_MISMATCH_DETAILS__&&ze("Hydration text content mismatch on",p,` + - rendered on server: ${p.textContent} + - expected on client: ${_.children}`),p.textContent=_.children);if(w)if(__VUE_PROD_HYDRATION_MISMATCH_DETAILS__||de||!S||H&48)for(const D in w)__VUE_PROD_HYDRATION_MISMATCH_DETAILS__&&Cc(p,D,w[D],_,L)&&($e=!0),(de&&(D.endsWith("value")||D==="indeterminate")||Yt(D)&&!bt(D)||D[0]===".")&&r(p,D,null,w[D],void 0,void 0,L);else w.onClick&&r(p,"onClick",null,w.onClick,void 0,void 0,L);let G;(G=w&&w.onVnodeBeforeMount)&&Ce(G,L,_),z&&Pe(_,null,L,"beforeMount"),((G=w&&w.onVnodeMounted)||z||V)&&Mi(()=>{G&&Ce(G,L,_),V&&ne.enter(p),z&&Pe(_,null,L,"mounted")},M)}return p.nextSibling},y=(p,_,L,M,N,S,F)=>{F=F||!!_.dynamicChildren;const w=_.children,H=w.length;let R=!1;for(let z=0;z{const{slotScopeIds:F}=_;F&&(N=N?N.concat(F):F);const w=o(p),H=y(i(p),_,w,L,M,N,S);return H&&Rt(H)&&H.data==="]"?i(_.anchor=H):($e=!0,c(_.anchor=a("]"),w,H),H)},I=(p,_,L,M,N,S)=>{if($e=!0,__VUE_PROD_HYDRATION_MISMATCH_DETAILS__&&ze(`Hydration node mismatch: +- rendered on server:`,p,p.nodeType===3?"(text)":Rt(p)&&p.data==="["?"(start of fragment)":"",` +- expected on client:`,_.type),_.el=null,S){const H=k(p);for(;;){const R=i(p);if(R&&R!==H)l(R);else break}}const F=i(p),w=o(p);return l(p),n(null,_,w,F,L,M,cn(w),N),F},k=(p,_="[",L="]")=>{let M=0;for(;p;)if(p=i(p),p&&Rt(p)&&(p.data===_&&M++,p.data===L)){if(M===0)return i(p);M--}return p},K=(p,_,L)=>{const M=_.parentNode;M&&M.replaceChild(p,_);let N=L;for(;N;)N.vnode.el===_&&(N.vnode.el=N.subTree.el=p),N=N.parent},j=p=>p.nodeType===1&&p.tagName.toLowerCase()==="template";return[f,h]}function Cc(e,t,n,r,s){var i;let o,l,c,a;if(t==="class")c=e.getAttribute("class"),a=Ln(n),Ec(ys(c||""),ys(a))||(o=l="class");else if(t==="style"){c=e.getAttribute("style"),a=Q(n)?n:Bo(In(n));const f=vs(c),h=vs(a);if(r.dirs)for(const{dir:y,value:C}of r.dirs)y.name==="show"&&!C&&h.set("display","none");const g=s==null?void 0:s.subTree;if(r===g||(g==null?void 0:g.type)===ge&&g.children.includes(r)){const y=(i=s==null?void 0:s.getCssVars)==null?void 0:i.call(s);for(const C in y)h.set(`--${C}`,String(y[C]))}Tc(f,h)||(o=l="style")}else(e instanceof SVGElement&&qo(t)||e instanceof HTMLElement&&(Zr(t)||Wo(t)))&&(Zr(t)?(c=e.hasAttribute(t),a=Tr(n)):n==null?(c=e.hasAttribute(t),a=!1):(e.hasAttribute(t)?c=e.getAttribute(t):t==="value"&&e.tagName==="TEXTAREA"?c=e.value:c=!1,a=zo(n)?String(n):!1),c!==a&&(o="attribute",l=t));if(o){const f=y=>y===!1?"(not rendered)":`${l}="${y}"`,h=`Hydration ${o} mismatch on`,g=` + - rendered on server: ${f(c)} + - expected on client: ${f(a)} + Note: this mismatch is check-only. The DOM will not be rectified in production due to performance overhead. + You should fix the source of the mismatch.`;return ze(h,e,g),!0}return!1}function ys(e){return new Set(e.trim().split(/\s+/))}function Ec(e,t){if(e.size!==t.size)return!1;for(const n of e)if(!t.has(n))return!1;return!0}function vs(e){const t=new Map;for(const n of e.split(";")){let[r,s]=n.split(":");r=r==null?void 0:r.trim(),s=s==null?void 0:s.trim(),r&&s&&t.set(r,s)}return t}function Tc(e,t){if(e.size!==t.size)return!1;for(const[n,r]of e)if(r!==t.get(n))return!1;return!0}function Sc(){typeof __VUE_PROD_HYDRATION_MISMATCH_DETAILS__!="boolean"&&(Er().__VUE_PROD_HYDRATION_MISMATCH_DETAILS__=!1)}const me=Mi;function Ac(e){return zi(e)}function Rc(e){return zi(e,xc)}function zi(e,t){Sc();const n=Er();n.__VUE__=!0;const{insert:r,remove:s,patchProp:i,createElement:o,createText:l,createComment:c,setText:a,setElementText:f,parentNode:h,nextSibling:g,setScopeId:y=we,insertStaticContent:C}=e,I=(u,d,m,v=null,b=null,T=null,O=void 0,E=null,A=!!d.dynamicChildren)=>{if(u===d)return;u&&!ct(u,d)&&(v=Qt(u),Ie(u,b,T,!0),u=null),d.patchFlag===-2&&(A=!1,d.dynamicChildren=null);const{type:x,ref:P,shapeFlag:U}=d;switch(x){case Et:k(u,d,m,v);break;case _e:K(u,d,m,v);break;case Dt:u==null&&j(d,m,v,O);break;case ge:R(u,d,m,v,b,T,O,E,A);break;default:U&1?L(u,d,m,v,b,T,O,E,A):U&6?z(u,d,m,v,b,T,O,E,A):(U&64||U&128)&&x.process(u,d,m,v,b,T,O,E,A,gt)}P!=null&&b&&Tn(P,u&&u.ref,T,d||u,!d)},k=(u,d,m,v)=>{if(u==null)r(d.el=l(d.children),m,v);else{const b=d.el=u.el;d.children!==u.children&&a(b,d.children)}},K=(u,d,m,v)=>{u==null?r(d.el=c(d.children||""),m,v):d.el=u.el},j=(u,d,m,v)=>{[u.el,u.anchor]=C(u.children,d,m,v,u.el,u.anchor)},p=({el:u,anchor:d},m,v)=>{let b;for(;u&&u!==d;)b=g(u),r(u,m,v),u=b;r(d,m,v)},_=({el:u,anchor:d})=>{let m;for(;u&&u!==d;)m=g(u),s(u),u=m;s(d)},L=(u,d,m,v,b,T,O,E,A)=>{d.type==="svg"?O="svg":d.type==="math"&&(O="mathml"),u==null?M(d,m,v,b,T,O,E,A):F(u,d,b,T,O,E,A)},M=(u,d,m,v,b,T,O,E)=>{let A,x;const{props:P,shapeFlag:U,transition:$,dirs:q}=u;if(A=u.el=o(u.type,T,P&&P.is,P),U&8?f(A,u.children):U&16&&S(u.children,A,null,v,b,Qn(u,T),O,E),q&&Pe(u,null,v,"created"),N(A,u,u.scopeId,O,v),P){for(const Z in P)Z!=="value"&&!bt(Z)&&i(A,Z,null,P[Z],T,u.children,v,b,De);"value"in P&&i(A,"value",null,P.value,T),(x=P.onVnodeBeforeMount)&&Ce(x,v,u)}q&&Pe(u,null,v,"beforeMount");const Y=Yi(b,$);Y&&$.beforeEnter(A),r(A,d,m),((x=P&&P.onVnodeMounted)||Y||q)&&me(()=>{x&&Ce(x,v,u),Y&&$.enter(A),q&&Pe(u,null,v,"mounted")},b)},N=(u,d,m,v,b)=>{if(m&&y(u,m),v)for(let T=0;T{for(let x=A;x{const E=d.el=u.el;let{patchFlag:A,dynamicChildren:x,dirs:P}=d;A|=u.patchFlag&16;const U=u.props||te,$=d.props||te;let q;if(m&&st(m,!1),(q=$.onVnodeBeforeUpdate)&&Ce(q,m,d,u),P&&Pe(d,u,m,"beforeUpdate"),m&&st(m,!0),x?w(u.dynamicChildren,x,E,m,v,Qn(d,b),T):O||D(u,d,E,null,m,v,Qn(d,b),T,!1),A>0){if(A&16)H(E,d,U,$,m,v,b);else if(A&2&&U.class!==$.class&&i(E,"class",null,$.class,b),A&4&&i(E,"style",U.style,$.style,b),A&8){const Y=d.dynamicProps;for(let Z=0;Z{q&&Ce(q,m,d,u),P&&Pe(d,u,m,"updated")},v)},w=(u,d,m,v,b,T,O)=>{for(let E=0;E{if(m!==v){if(m!==te)for(const E in m)!bt(E)&&!(E in v)&&i(u,E,m[E],null,O,d.children,b,T,De);for(const E in v){if(bt(E))continue;const A=v[E],x=m[E];A!==x&&E!=="value"&&i(u,E,x,A,O,d.children,b,T,De)}"value"in v&&i(u,"value",m.value,v.value,O)}},R=(u,d,m,v,b,T,O,E,A)=>{const x=d.el=u?u.el:l(""),P=d.anchor=u?u.anchor:l("");let{patchFlag:U,dynamicChildren:$,slotScopeIds:q}=d;q&&(E=E?E.concat(q):q),u==null?(r(x,m,v),r(P,m,v),S(d.children||[],m,P,b,T,O,E,A)):U>0&&U&64&&$&&u.dynamicChildren?(w(u.dynamicChildren,$,m,b,T,O,E),(d.key!=null||b&&d===b.subTree)&&Ur(u,d,!0)):D(u,d,m,P,b,T,O,E,A)},z=(u,d,m,v,b,T,O,E,A)=>{d.slotScopeIds=E,u==null?d.shapeFlag&512?b.ctx.activate(d,m,v,O,A):ne(d,m,v,b,T,O,A):de(u,d,A)},ne=(u,d,m,v,b,T,O)=>{const E=u.component=kc(u,v,b);if(Jt(u)&&(E.ctx.renderer=gt),jc(E),E.asyncDep){if(b&&b.registerDep(E,V),!u.el){const A=E.subTree=ie(_e);K(null,A,d,m)}}else V(E,u,d,m,b,T,O)},de=(u,d,m)=>{const v=d.component=u.component;if(jl(u,d,m))if(v.asyncDep&&!v.asyncResolved){G(v,d,m);return}else v.next=d,Pl(v.update),v.effect.dirty=!0,v.update();else d.el=u.el,v.vnode=d},V=(u,d,m,v,b,T,O)=>{const E=()=>{if(u.isMounted){let{next:P,bu:U,u:$,parent:q,vnode:Y}=u;{const mt=Gi(u);if(mt){P&&(P.el=Y.el,G(u,P,O)),mt.asyncDep.then(()=>{u.isUnmounted||E()});return}}let Z=P,re;st(u,!1),P?(P.el=Y.el,G(u,P,O)):P=Y,U&&gn(U),(re=P.props&&P.props.onVnodeBeforeUpdate)&&Ce(re,q,P,Y),st(u,!0);const ae=Gn(u),Se=u.subTree;u.subTree=ae,I(Se,ae,h(Se.el),Qt(Se),u,b,T),P.el=ae.el,Z===null&&Vl(u,ae.el),$&&me($,b),(re=P.props&&P.props.onVnodeUpdated)&&me(()=>Ce(re,q,P,Y),b)}else{let P;const{el:U,props:$}=d,{bm:q,m:Y,parent:Z}=u,re=xt(d);if(st(u,!1),q&&gn(q),!re&&(P=$&&$.onVnodeBeforeMount)&&Ce(P,Z,d),st(u,!0),U&&zn){const ae=()=>{u.subTree=Gn(u),zn(U,u.subTree,u,b,null)};re?d.type.__asyncLoader().then(()=>!u.isUnmounted&&ae()):ae()}else{const ae=u.subTree=Gn(u);I(null,ae,m,v,u,b,T),d.el=ae.el}if(Y&&me(Y,b),!re&&(P=$&&$.onVnodeMounted)){const ae=d;me(()=>Ce(P,Z,ae),b)}(d.shapeFlag&256||Z&&xt(Z.vnode)&&Z.vnode.shapeFlag&256)&&u.a&&me(u.a,b),u.isMounted=!0,d=m=v=null}},A=u.effect=new Sr(E,we,()=>Dn(x),u.scope),x=u.update=()=>{A.dirty&&A.run()};x.id=u.uid,st(u,!0),x()},G=(u,d,m)=>{d.component=u;const v=u.vnode.props;u.vnode=d,u.next=null,mc(u,d.props,v,m),vc(u,d.children,m),Ve(),os(u),Ue()},D=(u,d,m,v,b,T,O,E,A=!1)=>{const x=u&&u.children,P=u?u.shapeFlag:0,U=d.children,{patchFlag:$,shapeFlag:q}=d;if($>0){if($&128){pt(x,U,m,v,b,T,O,E,A);return}else if($&256){Oe(x,U,m,v,b,T,O,E,A);return}}q&8?(P&16&&De(x,b,T),U!==x&&f(m,U)):P&16?q&16?pt(x,U,m,v,b,T,O,E,A):De(x,b,T,!0):(P&8&&f(m,""),q&16&&S(U,m,v,b,T,O,E,A))},Oe=(u,d,m,v,b,T,O,E,A)=>{u=u||yt,d=d||yt;const x=u.length,P=d.length,U=Math.min(x,P);let $;for($=0;$P?De(u,b,T,!0,!1,U):S(d,m,v,b,T,O,E,A,U)},pt=(u,d,m,v,b,T,O,E,A)=>{let x=0;const P=d.length;let U=u.length-1,$=P-1;for(;x<=U&&x<=$;){const q=u[x],Y=d[x]=A?Je(d[x]):Ae(d[x]);if(ct(q,Y))I(q,Y,m,null,b,T,O,E,A);else break;x++}for(;x<=U&&x<=$;){const q=u[U],Y=d[$]=A?Je(d[$]):Ae(d[$]);if(ct(q,Y))I(q,Y,m,null,b,T,O,E,A);else break;U--,$--}if(x>U){if(x<=$){const q=$+1,Y=q$)for(;x<=U;)Ie(u[x],b,T,!0),x++;else{const q=x,Y=x,Z=new Map;for(x=Y;x<=$;x++){const ve=d[x]=A?Je(d[x]):Ae(d[x]);ve.key!=null&&Z.set(ve.key,x)}let re,ae=0;const Se=$-Y+1;let mt=!1,Yr=0;const At=new Array(Se);for(x=0;x=Se){Ie(ve,b,T,!0);continue}let Le;if(ve.key!=null)Le=Z.get(ve.key);else for(re=Y;re<=$;re++)if(At[re-Y]===0&&ct(ve,d[re])){Le=re;break}Le===void 0?Ie(ve,b,T,!0):(At[Le-Y]=x+1,Le>=Yr?Yr=Le:mt=!0,I(ve,d[Le],m,null,b,T,O,E,A),ae++)}const Gr=mt?Mc(At):yt;for(re=Gr.length-1,x=Se-1;x>=0;x--){const ve=Y+x,Le=d[ve],Jr=ve+1{const{el:T,type:O,transition:E,children:A,shapeFlag:x}=u;if(x&6){rt(u.component.subTree,d,m,v);return}if(x&128){u.suspense.move(d,m,v);return}if(x&64){O.move(u,d,m,gt);return}if(O===ge){r(T,d,m);for(let U=0;UE.enter(T),b);else{const{leave:U,delayLeave:$,afterLeave:q}=E,Y=()=>r(T,d,m),Z=()=>{U(T,()=>{Y(),q&&q()})};$?$(T,Y,Z):Z()}else r(T,d,m)},Ie=(u,d,m,v=!1,b=!1)=>{const{type:T,props:O,ref:E,children:A,dynamicChildren:x,shapeFlag:P,patchFlag:U,dirs:$}=u;if(E!=null&&Tn(E,null,m,u,!0),P&256){d.ctx.deactivate(u);return}const q=P&1&&$,Y=!xt(u);let Z;if(Y&&(Z=O&&O.onVnodeBeforeUnmount)&&Ce(Z,d,u),P&6)Lo(u.component,m,v);else{if(P&128){u.suspense.unmount(m,v);return}q&&Pe(u,null,d,"beforeUnmount"),P&64?u.type.remove(u,d,m,b,gt,v):x&&(T!==ge||U>0&&U&64)?De(x,d,m,!1,!0):(T===ge&&U&384||!b&&P&16)&&De(A,d,m),v&&qr(u)}(Y&&(Z=O&&O.onVnodeUnmounted)||q)&&me(()=>{Z&&Ce(Z,d,u),q&&Pe(u,null,d,"unmounted")},m)},qr=u=>{const{type:d,el:m,anchor:v,transition:b}=u;if(d===ge){Io(m,v);return}if(d===Dt){_(u);return}const T=()=>{s(m),b&&!b.persisted&&b.afterLeave&&b.afterLeave()};if(u.shapeFlag&1&&b&&!b.persisted){const{leave:O,delayLeave:E}=b,A=()=>O(m,T);E?E(u.el,T,A):A()}else T()},Io=(u,d)=>{let m;for(;u!==d;)m=g(u),s(u),u=m;s(d)},Lo=(u,d,m)=>{const{bum:v,scope:b,update:T,subTree:O,um:E}=u;v&&gn(v),b.stop(),T&&(T.active=!1,Ie(O,u,d,m)),E&&me(E,d),me(()=>{u.isUnmounted=!0},d),d&&d.pendingBranch&&!d.isUnmounted&&u.asyncDep&&!u.asyncResolved&&u.suspenseId===d.pendingId&&(d.deps--,d.deps===0&&d.resolve())},De=(u,d,m,v=!1,b=!1,T=0)=>{for(let O=T;Ou.shapeFlag&6?Qt(u.component.subTree):u.shapeFlag&128?u.suspense.next():g(u.anchor||u.el);let Wn=!1;const zr=(u,d,m)=>{u==null?d._vnode&&Ie(d._vnode,null,null,!0):I(d._vnode||null,u,d,null,null,null,m),Wn||(Wn=!0,os(),xn(),Wn=!1),d._vnode=u},gt={p:I,um:Ie,m:rt,r:qr,mt:ne,mc:S,pc:D,pbc:w,n:Qt,o:e};let qn,zn;return t&&([qn,zn]=t(gt)),{render:zr,hydrate:qn,createApp:hc(zr,qn)}}function Qn({type:e,props:t},n){return n==="svg"&&e==="foreignObject"||n==="mathml"&&e==="annotation-xml"&&t&&t.encoding&&t.encoding.includes("html")?void 0:n}function st({effect:e,update:t},n){e.allowRecurse=t.allowRecurse=n}function Yi(e,t){return(!e||e&&!e.pendingBranch)&&t&&!t.persisted}function Ur(e,t,n=!1){const r=e.children,s=t.children;if(B(r)&&B(s))for(let i=0;i>1,e[n[l]]0&&(t[r]=n[i-1]),n[i]=r)}}for(i=n.length,o=n[i-1];i-- >0;)n[i]=o,o=t[o];return n}function Gi(e){const t=e.subTree.component;if(t)return t.asyncDep&&!t.asyncResolved?t:Gi(t)}const Oc=e=>e.__isTeleport,Ht=e=>e&&(e.disabled||e.disabled===""),bs=e=>typeof SVGElement<"u"&&e instanceof SVGElement,ws=e=>typeof MathMLElement=="function"&&e instanceof MathMLElement,_r=(e,t)=>{const n=e&&e.to;return Q(n)?t?t(n):null:n},Ic={name:"Teleport",__isTeleport:!0,process(e,t,n,r,s,i,o,l,c,a){const{mc:f,pc:h,pbc:g,o:{insert:y,querySelector:C,createText:I,createComment:k}}=a,K=Ht(t.props);let{shapeFlag:j,children:p,dynamicChildren:_}=t;if(e==null){const L=t.el=I(""),M=t.anchor=I("");y(L,n,r),y(M,n,r);const N=t.target=_r(t.props,C),S=t.targetAnchor=I("");N&&(y(S,N),o==="svg"||bs(N)?o="svg":(o==="mathml"||ws(N))&&(o="mathml"));const F=(w,H)=>{j&16&&f(p,w,H,s,i,o,l,c)};K?F(n,M):N&&F(N,S)}else{t.el=e.el;const L=t.anchor=e.anchor,M=t.target=e.target,N=t.targetAnchor=e.targetAnchor,S=Ht(e.props),F=S?n:M,w=S?L:N;if(o==="svg"||bs(M)?o="svg":(o==="mathml"||ws(M))&&(o="mathml"),_?(g(e.dynamicChildren,_,F,s,i,o,l),Ur(e,t,!0)):c||h(e,t,F,w,s,i,o,l,!1),K)S?t.props&&e.props&&t.props.to!==e.props.to&&(t.props.to=e.props.to):an(t,n,L,a,1);else if((t.props&&t.props.to)!==(e.props&&e.props.to)){const H=t.target=_r(t.props,C);H&&an(t,H,null,a,0)}else S&&an(t,M,N,a,1)}Ji(t)},remove(e,t,n,r,{um:s,o:{remove:i}},o){const{shapeFlag:l,children:c,anchor:a,targetAnchor:f,target:h,props:g}=e;if(h&&i(f),o&&i(a),l&16){const y=o||!Ht(g);for(let C=0;C0?Re||yt:null,Pc(),Kt>0&&Re&&Re.push(e),e}function Au(e,t,n,r,s,i){return Zi(to(e,t,n,r,s,i,!0))}function Qi(e,t,n,r,s){return Zi(ie(e,t,n,r,s,!0))}function Sn(e){return e?e.__v_isVNode===!0:!1}function ct(e,t){return e.type===t.type&&e.key===t.key}const eo=({key:e})=>e??null,_n=({ref:e,ref_key:t,ref_for:n})=>(typeof e=="number"&&(e=""+e),e!=null?Q(e)||fe(e)||W(e)?{i:ce,r:e,k:t,f:!!n}:e:null);function to(e,t=null,n=null,r=0,s=null,i=e===ge?0:1,o=!1,l=!1){const c={__v_isVNode:!0,__v_skip:!0,type:e,props:t,key:t&&eo(t),ref:t&&_n(t),scopeId:kn,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:i,patchFlag:r,dynamicProps:s,dynamicChildren:null,appContext:null,ctx:ce};return l?(Br(c,n),i&128&&e.normalize(c)):n&&(c.shapeFlag|=Q(n)?8:16),Kt>0&&!o&&Re&&(c.patchFlag>0||i&6)&&c.patchFlag!==32&&Re.push(c),c}const ie=Nc;function Nc(e,t=null,n=null,r=0,s=null,i=!1){if((!e||e===Ai)&&(e=_e),Sn(e)){const l=tt(e,t,!0);return n&&Br(l,n),Kt>0&&!i&&Re&&(l.shapeFlag&6?Re[Re.indexOf(e)]=l:Re.push(l)),l.patchFlag|=-2,l}if(Wc(e)&&(e=e.__vccOpts),t){t=Fc(t);let{class:l,style:c}=t;l&&!Q(l)&&(t.class=Ln(l)),ee(c)&&(yi(c)&&!B(c)&&(c=oe({},c)),t.style=In(c))}const o=Q(e)?1:Ul(e)?128:Oc(e)?64:ee(e)?4:W(e)?2:0;return to(e,t,n,r,s,o,i,!0)}function Fc(e){return e?yi(e)||Vi(e)?oe({},e):e:null}function tt(e,t,n=!1,r=!1){const{props:s,ref:i,patchFlag:o,children:l,transition:c}=e,a=t?Hc(s||{},t):s,f={__v_isVNode:!0,__v_skip:!0,type:e.type,props:a,key:a&&eo(a),ref:t&&t.ref?n&&i?B(i)?i.concat(_n(t)):[i,_n(t)]:_n(t):i,scopeId:e.scopeId,slotScopeIds:e.slotScopeIds,children:l,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:c,component:e.component,suspense:e.suspense,ssContent:e.ssContent&&tt(e.ssContent),ssFallback:e.ssFallback&&tt(e.ssFallback),el:e.el,anchor:e.anchor,ctx:e.ctx,ce:e.ce};return c&&r&&(f.transition=c.clone(f)),f}function no(e=" ",t=0){return ie(Et,null,e,t)}function Ru(e,t){const n=ie(Dt,null,e);return n.staticCount=t,n}function Mu(e="",t=!1){return t?(Xi(),Qi(_e,null,e)):ie(_e,null,e)}function Ae(e){return e==null||typeof e=="boolean"?ie(_e):B(e)?ie(ge,null,e.slice()):typeof e=="object"?Je(e):ie(Et,null,String(e))}function Je(e){return e.el===null&&e.patchFlag!==-1||e.memo?e:tt(e)}function Br(e,t){let n=0;const{shapeFlag:r}=e;if(t==null)t=null;else if(B(t))n=16;else if(typeof t=="object")if(r&65){const s=t.default;s&&(s._c&&(s._d=!1),Br(e,s()),s._c&&(s._d=!0));return}else{n=32;const s=t._;!s&&!Vi(t)?t._ctx=ce:s===3&&ce&&(ce.slots._===1?t._=1:(t._=2,e.patchFlag|=1024))}else W(t)?(t={default:t,_ctx:ce},n=32):(t=String(t),r&64?(n=16,t=[no(t)]):n=8);e.children=t,e.shapeFlag|=n}function Hc(...e){const t={};for(let n=0;nue||ce;let An,yr;{const e=Er(),t=(n,r)=>{let s;return(s=e[n])||(s=e[n]=[]),s.push(r),i=>{s.length>1?s.forEach(o=>o(i)):s[0](i)}};An=t("__VUE_INSTANCE_SETTERS__",n=>ue=n),yr=t("__VUE_SSR_SETTERS__",n=>Zt=n)}const Xt=e=>{const t=ue;return An(e),e.scope.on(),()=>{e.scope.off(),An(t)}},Cs=()=>{ue&&ue.scope.off(),An(null)};function ro(e){return e.vnode.shapeFlag&4}let Zt=!1;function jc(e,t=!1){t&&yr(t);const{props:n,children:r}=e.vnode,s=ro(e);gc(e,n,s,t),yc(e,r);const i=s?Vc(e,t):void 0;return t&&yr(!1),i}function Vc(e,t){const n=e.type;e.accessCache=Object.create(null),e.proxy=new Proxy(e.ctx,ic);const{setup:r}=n;if(r){const s=e.setupContext=r.length>1?io(e):null,i=Xt(e);Ve();const o=je(r,e,0,[e.props,s]);if(Ue(),i(),ei(o)){if(o.then(Cs,Cs),t)return o.then(l=>{Es(e,l,t)}).catch(l=>{Gt(l,e,0)});e.asyncDep=o}else Es(e,o,t)}else so(e,t)}function Es(e,t,n){W(t)?e.type.__ssrInlineRender?e.ssrRender=t:e.render=t:ee(t)&&(e.setupState=wi(t)),so(e,n)}let Ts;function so(e,t,n){const r=e.type;if(!e.render){if(!t&&Ts&&!r.render){const s=r.template||jr(e).template;if(s){const{isCustomElement:i,compilerOptions:o}=e.appContext.config,{delimiters:l,compilerOptions:c}=r,a=oe(oe({isCustomElement:i,delimiters:l},o),c);r.render=Ts(s,a)}}e.render=r.render||we}{const s=Xt(e);Ve();try{lc(e)}finally{Ue(),s()}}}const Uc={get(e,t){return ye(e,"get",""),e[t]}};function io(e){const t=n=>{e.exposed=n||{}};return{attrs:new Proxy(e.attrs,Uc),slots:e.slots,emit:e.emit,expose:t}}function Kn(e){if(e.exposed)return e.exposeProxy||(e.exposeProxy=new Proxy(wi(mn(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}}))}const Bc=/(?:^|[-_])(\w)/g,Kc=e=>e.replace(Bc,t=>t.toUpperCase()).replace(/[-_]/g,"");function oo(e,t=!0){return W(e)?e.displayName||e.name:e.name||t&&e.__name}function lo(e,t,n=!1){let r=oo(t);if(!r&&t.__file){const s=t.__file.match(/([^/\\]+)\.\w+$/);s&&(r=s[1])}if(!r&&e&&e.parent){const s=i=>{for(const o in i)if(i[o]===t)return o};r=s(e.components||e.parent.type.components)||s(e.appContext.components)}return r?Kc(r):n?"App":"Anonymous"}function Wc(e){return W(e)&&"__vccOpts"in e}const se=(e,t)=>yl(e,t,Zt);function vr(e,t,n){const r=arguments.length;return r===2?ee(t)&&!B(t)?Sn(t)?ie(e,null,[t]):ie(e,t):ie(e,null,t):(r>3?n=Array.prototype.slice.call(arguments,2):r===3&&Sn(n)&&(n=[n]),ie(e,t,n))}const qc="3.4.26";/** +* @vue/runtime-dom v3.4.26 +* (c) 2018-present Yuxi (Evan) You and Vue contributors +* @license MIT +**/const zc="http://www.w3.org/2000/svg",Yc="http://www.w3.org/1998/Math/MathML",Xe=typeof document<"u"?document:null,Ss=Xe&&Xe.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==="svg"?Xe.createElementNS(zc,e):t==="mathml"?Xe.createElementNS(Yc,e):Xe.createElement(e,n?{is:n}:void 0);return e==="select"&&r&&r.multiple!=null&&s.setAttribute("multiple",r.multiple),s},createText:e=>Xe.createTextNode(e),createComment:e=>Xe.createComment(e),setText:(e,t)=>{e.nodeValue=t},setElementText:(e,t)=>{e.textContent=t},parentNode:e=>e.parentNode,nextSibling:e=>e.nextSibling,querySelector:e=>Xe.querySelector(e),setScopeId(e,t){e.setAttribute(t,"")},insertStaticContent(e,t,n,r,s,i){const o=n?n.previousSibling:t.lastChild;if(s&&(s===i||s.nextSibling))for(;t.insertBefore(s.cloneNode(!0),n),!(s===i||!(s=s.nextSibling)););else{Ss.innerHTML=r==="svg"?`${e}`:r==="mathml"?`${e}`:e;const l=Ss.content;if(r==="svg"||r==="mathml"){const c=l.firstChild;for(;c.firstChild;)l.appendChild(c.firstChild);l.removeChild(c)}t.insertBefore(l,n)}return[o?o.nextSibling:t.firstChild,n?n.previousSibling:t.lastChild]}},We="transition",Mt="animation",Wt=Symbol("_vtc"),co=(e,{slots:t})=>vr(Yl,Jc(e),t);co.displayName="Transition";const ao={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};co.props=oe({},Ii,ao);const it=(e,t=[])=>{B(e)?e.forEach(n=>n(...t)):e&&e(...t)},As=e=>e?B(e)?e.some(t=>t.length>1):e.length>1:!1;function Jc(e){const t={};for(const R in e)R in ao||(t[R]=e[R]);if(e.css===!1)return t;const{name:n="v",type:r,duration:s,enterFromClass:i=`${n}-enter-from`,enterActiveClass:o=`${n}-enter-active`,enterToClass:l=`${n}-enter-to`,appearFromClass:c=i,appearActiveClass:a=o,appearToClass:f=l,leaveFromClass:h=`${n}-leave-from`,leaveActiveClass:g=`${n}-leave-active`,leaveToClass:y=`${n}-leave-to`}=e,C=Xc(s),I=C&&C[0],k=C&&C[1],{onBeforeEnter:K,onEnter:j,onEnterCancelled:p,onLeave:_,onLeaveCancelled:L,onBeforeAppear:M=K,onAppear:N=j,onAppearCancelled:S=p}=t,F=(R,z,ne)=>{ot(R,z?f:l),ot(R,z?a:o),ne&&ne()},w=(R,z)=>{R._isLeaving=!1,ot(R,h),ot(R,y),ot(R,g),z&&z()},H=R=>(z,ne)=>{const de=R?N:j,V=()=>F(z,R,ne);it(de,[z,V]),Rs(()=>{ot(z,R?c:i),qe(z,R?f:l),As(de)||Ms(z,r,I,V)})};return oe(t,{onBeforeEnter(R){it(K,[R]),qe(R,i),qe(R,o)},onBeforeAppear(R){it(M,[R]),qe(R,c),qe(R,a)},onEnter:H(!1),onAppear:H(!0),onLeave(R,z){R._isLeaving=!0;const ne=()=>w(R,z);qe(R,h),qe(R,g),ea(),Rs(()=>{R._isLeaving&&(ot(R,h),qe(R,y),As(_)||Ms(R,r,k,ne))}),it(_,[R,ne])},onEnterCancelled(R){F(R,!1),it(p,[R])},onAppearCancelled(R){F(R,!0),it(S,[R])},onLeaveCancelled(R){w(R),it(L,[R])}})}function Xc(e){if(e==null)return null;if(ee(e))return[er(e.enter),er(e.leave)];{const t=er(e);return[t,t]}}function er(e){return $o(e)}function qe(e,t){t.split(/\s+/).forEach(n=>n&&e.classList.add(n)),(e[Wt]||(e[Wt]=new Set)).add(t)}function ot(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 Rs(e){requestAnimationFrame(()=>{requestAnimationFrame(e)})}let Zc=0;function Ms(e,t,n,r){const s=e._endId=++Zc,i=()=>{s===e._endId&&r()};if(n)return setTimeout(i,n);const{type:o,timeout:l,propCount:c}=Qc(e,t);if(!o)return r();const a=o+"end";let f=0;const h=()=>{e.removeEventListener(a,g),i()},g=y=>{y.target===e&&++f>=c&&h()};setTimeout(()=>{f(n[C]||"").split(", "),s=r(`${We}Delay`),i=r(`${We}Duration`),o=Os(s,i),l=r(`${Mt}Delay`),c=r(`${Mt}Duration`),a=Os(l,c);let f=null,h=0,g=0;t===We?o>0&&(f=We,h=o,g=i.length):t===Mt?a>0&&(f=Mt,h=a,g=c.length):(h=Math.max(o,a),f=h>0?o>a?We:Mt:null,g=f?f===We?i.length:c.length:0);const y=f===We&&/\b(transform|all)(,|$)/.test(r(`${We}Property`).toString());return{type:f,timeout:h,propCount:g,hasTransform:y}}function Os(e,t){for(;e.lengthIs(n)+Is(e[r])))}function Is(e){return e==="auto"?0:Number(e.slice(0,-1).replace(",","."))*1e3}function ea(){return document.body.offsetHeight}function ta(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 Ls=Symbol("_vod"),na=Symbol("_vsh"),ra=Symbol(""),sa=/(^|;)\s*display\s*:/;function ia(e,t,n){const r=e.style,s=Q(n);let i=!1;if(n&&!s){if(t)if(Q(t))for(const o of t.split(";")){const l=o.slice(0,o.indexOf(":")).trim();n[l]==null&&yn(r,l,"")}else for(const o in t)n[o]==null&&yn(r,o,"");for(const o in n)o==="display"&&(i=!0),yn(r,o,n[o])}else if(s){if(t!==n){const o=r[ra];o&&(n+=";"+o),r.cssText=n,i=sa.test(n)}}else t&&e.removeAttribute("style");Ls in e&&(e[Ls]=i?r.display:"",e[na]&&(r.display="none"))}const Ps=/\s*!important$/;function yn(e,t,n){if(B(n))n.forEach(r=>yn(e,t,r));else if(n==null&&(n=""),t.startsWith("--"))e.setProperty(t,n);else{const r=oa(e,t);Ps.test(n)?e.setProperty(nt(r),n.replace(Ps,""),"important"):e[r]=n}}const Ns=["Webkit","Moz","ms"],tr={};function oa(e,t){const n=tr[t];if(n)return n;let r=Fe(t);if(r!=="filter"&&r in e)return tr[t]=r;r=On(r);for(let s=0;snr||(da.then(()=>nr=0),nr=Date.now());function pa(e,t){const n=r=>{if(!r._vts)r._vts=Date.now();else if(r._vts<=n.attached)return;Te(ga(r,n.value),t,5,[r])};return n.value=e,n.attached=ha(),n}function ga(e,t){if(B(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 $s=e=>e.charCodeAt(0)===111&&e.charCodeAt(1)===110&&e.charCodeAt(2)>96&&e.charCodeAt(2)<123,ma=(e,t,n,r,s,i,o,l,c)=>{const a=s==="svg";t==="class"?ta(e,r,a):t==="style"?ia(e,n,r):Yt(t)?wr(t)||ua(e,t,n,r,o):(t[0]==="."?(t=t.slice(1),!0):t[0]==="^"?(t=t.slice(1),!1):_a(e,t,r,a))?ca(e,t,r,i,o,l,c):(t==="true-value"?e._trueValue=r:t==="false-value"&&(e._falseValue=r),la(e,t,r,a))};function _a(e,t,n,r){if(r)return!!(t==="innerHTML"||t==="textContent"||t in e&&$s(t)&&W(n));if(t==="spellcheck"||t==="draggable"||t==="translate"||t==="form"||t==="list"&&e.tagName==="INPUT"||t==="type"&&e.tagName==="TEXTAREA")return!1;if(t==="width"||t==="height"){const s=e.tagName;if(s==="IMG"||s==="VIDEO"||s==="CANVAS"||s==="SOURCE")return!1}return $s(t)&&Q(n)?!1:t in e}const ks=e=>{const t=e.props["onUpdate:modelValue"]||!1;return B(t)?n=>gn(t,n):t};function ya(e){e.target.composing=!0}function js(e){const t=e.target;t.composing&&(t.composing=!1,t.dispatchEvent(new Event("input")))}const rr=Symbol("_assign"),Ou={created(e,{modifiers:{lazy:t,trim:n,number:r}},s){e[rr]=ks(s);const i=r||s.props&&s.props.type==="number";_t(e,t?"change":"input",o=>{if(o.target.composing)return;let l=e.value;n&&(l=l.trim()),i&&(l=or(l)),e[rr](l)}),n&&_t(e,"change",()=>{e.value=e.value.trim()}),t||(_t(e,"compositionstart",ya),_t(e,"compositionend",js),_t(e,"change",js))},mounted(e,{value:t}){e.value=t??""},beforeUpdate(e,{value:t,modifiers:{lazy:n,trim:r,number:s}},i){if(e[rr]=ks(i),e.composing)return;const o=(s||e.type==="number")&&!/^0\d/.test(e.value)?or(e.value):e.value,l=t??"";o!==l&&(document.activeElement===e&&e.type!=="range"&&(n||r&&e.value.trim()===l)||(e.value=l))}},va=["ctrl","shift","alt","meta"],ba={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)=>va.some(n=>e[`${n}Key`]&&!t.includes(n))},Iu=(e,t)=>{const n=e._withMods||(e._withMods={}),r=t.join(".");return n[r]||(n[r]=(s,...i)=>{for(let o=0;o{const n=e._withKeys||(e._withKeys={}),r=t.join(".");return n[r]||(n[r]=s=>{if(!("key"in s))return;const i=nt(s.key);if(t.some(o=>o===i||wa[o]===i))return e(s)})},uo=oe({patchProp:ma},Gc);let kt,Vs=!1;function xa(){return kt||(kt=Ac(uo))}function Ca(){return kt=Vs?kt:Rc(uo),Vs=!0,kt}const Pu=(...e)=>{const t=xa().createApp(...e),{mount:n}=t;return t.mount=r=>{const s=ho(r);if(!s)return;const i=t._component;!W(i)&&!i.render&&!i.template&&(i.template=s.innerHTML),s.innerHTML="";const o=n(s,!1,fo(s));return s instanceof Element&&(s.removeAttribute("v-cloak"),s.setAttribute("data-v-app","")),o},t},Nu=(...e)=>{const t=Ca().createApp(...e),{mount:n}=t;return t.mount=r=>{const s=ho(r);if(s)return n(s,!0,fo(s))},t};function fo(e){if(e instanceof SVGElement)return"svg";if(typeof MathMLElement=="function"&&e instanceof MathMLElement)return"mathml"}function ho(e){return Q(e)?document.querySelector(e):e}const Fu=(e,t)=>{const n=e.__vccOpts||e;for(const[r,s]of t)n[r]=s;return n},Ea="modulepreload",Ta=function(e){return"/"+e},Us={},Hu=function(t,n,r){if(!n||n.length===0)return t();const s=document.getElementsByTagName("link");return Promise.all(n.map(i=>{if(i=Ta(i),i in Us)return;Us[i]=!0;const o=i.endsWith(".css"),l=o?'[rel="stylesheet"]':"";if(!!r)for(let f=s.length-1;f>=0;f--){const h=s[f];if(h.href===i&&(!o||h.rel==="stylesheet"))return}else if(document.querySelector(`link[href="${i}"]${l}`))return;const a=document.createElement("link");if(a.rel=o?"stylesheet":Ea,o||(a.as="script",a.crossOrigin=""),a.href=i,document.head.appendChild(a),o)return new Promise((f,h)=>{a.addEventListener("load",f),a.addEventListener("error",()=>h(new Error(`Unable to preload CSS for ${i}`)))})})).then(()=>t()).catch(i=>{const o=new Event("vite:preloadError",{cancelable:!0});if(o.payload=i,window.dispatchEvent(o),!o.defaultPrevented)throw i})},Sa=window.__VP_SITE_DATA__;function Kr(e){return oi()?(Jo(e),!0):!1}function Me(e){return typeof e=="function"?e():Fr(e)}function Du(e,t){const n=(t==null?void 0:t.computedGetter)===!1?Fr:Me;return function(...r){return se(()=>e.apply(this,r.map(s=>n(s))))}}const po=typeof window<"u"&&typeof document<"u";typeof WorkerGlobalScope<"u"&&globalThis instanceof WorkerGlobalScope;const Aa=Object.prototype.toString,Ra=e=>Aa.call(e)==="[object Object]",qt=()=>{},Bs=Ma();function Ma(){var e,t;return po&&((e=window==null?void 0:window.navigator)==null?void 0:e.userAgent)&&(/iP(ad|hone|od)/.test(window.navigator.userAgent)||((t=window==null?void 0:window.navigator)==null?void 0:t.maxTouchPoints)>2&&/iPad|Macintosh/.test(window==null?void 0:window.navigator.userAgent))}function Oa(e,t){function n(...r){return new Promise((s,i)=>{Promise.resolve(e(()=>t.apply(this,r),{fn:t,thisArg:this,args:r})).then(s).catch(i)})}return n}const go=e=>e();function Ia(e,t={}){let n,r,s=qt;const i=l=>{clearTimeout(l),s(),s=qt};return l=>{const c=Me(e),a=Me(t.maxWait);return n&&i(n),c<=0||a!==void 0&&a<=0?(r&&(i(r),r=null),Promise.resolve(l())):new Promise((f,h)=>{s=t.rejectOnCancel?h:f,a&&!r&&(r=setTimeout(()=>{n&&i(n),r=null,f(l())},a)),n=setTimeout(()=>{r&&i(r),r=null,f(l())},c)})}}function La(e=go){const t=le(!0);function n(){t.value=!1}function r(){t.value=!0}const s=(...i)=>{t.value&&e(...i)};return{isActive:Fn(t),pause:n,resume:r,eventFilter:s}}function Pa(e){return e||Bn()}function mo(...e){if(e.length!==1)return Tl(...e);const t=e[0];return typeof t=="function"?Fn(xl(()=>({get:t,set:qt}))):le(t)}function _o(e,t,n={}){const{eventFilter:r=go,...s}=n;return Qe(e,Oa(r,t),s)}function Na(e,t,n={}){const{eventFilter:r,...s}=n,{eventFilter:i,pause:o,resume:l,isActive:c}=La(r);return{stop:_o(e,t,{...s,eventFilter:i}),pause:o,resume:l,isActive:c}}function yo(e,t=!0,n){Pa()?St(e,n):t?e():Hn(e)}function $u(e,t,n={}){const{debounce:r=0,maxWait:s=void 0,...i}=n;return _o(e,t,{...i,eventFilter:Ia(r,{maxWait:s})})}function ku(e,t,n){let r;fe(n)?r={evaluating:n}:r=n||{};const{lazy:s=!1,evaluating:i=void 0,shallow:o=!0,onError:l=qt}=r,c=le(!s),a=o?Nr(t):le(t);let f=0;return $r(async h=>{if(!c.value)return;f++;const g=f;let y=!1;i&&Promise.resolve().then(()=>{i.value=!0});try{const C=await e(I=>{h(()=>{i&&(i.value=!1),y||I()})});g===f&&(a.value=C)}catch(C){l(C)}finally{i&&g===f&&(i.value=!1),y=!0}}),s?se(()=>(c.value=!0,a.value)):a}function vo(e){var t;const n=Me(e);return(t=n==null?void 0:n.$el)!=null?t:n}const He=po?window:void 0;function zt(...e){let t,n,r,s;if(typeof e[0]=="string"||Array.isArray(e[0])?([n,r,s]=e,t=He):[t,n,r,s]=e,!t)return qt;Array.isArray(n)||(n=[n]),Array.isArray(r)||(r=[r]);const i=[],o=()=>{i.forEach(f=>f()),i.length=0},l=(f,h,g,y)=>(f.addEventListener(h,g,y),()=>f.removeEventListener(h,g,y)),c=Qe(()=>[vo(t),Me(s)],([f,h])=>{if(o(),!f)return;const g=Ra(h)?{...h}:h;i.push(...n.flatMap(y=>r.map(C=>l(f,y,C,g))))},{immediate:!0,flush:"post"}),a=()=>{c(),o()};return Kr(a),a}function Fa(e){return typeof e=="function"?e:typeof e=="string"?t=>t.key===e:Array.isArray(e)?t=>e.includes(t.key):()=>!0}function ju(...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=He,eventName:i="keydown",passive:o=!1,dedupe:l=!1}=r,c=Fa(t);return zt(s,i,f=>{f.repeat&&Me(l)||c(f)&&n(f)},o)}function Ha(){const e=le(!1),t=Bn();return t&&St(()=>{e.value=!0},t),e}function Da(e){const t=Ha();return se(()=>(t.value,!!e()))}function $a(e,t={}){const{window:n=He}=t,r=Da(()=>n&&"matchMedia"in n&&typeof n.matchMedia=="function");let s;const i=le(!1),o=a=>{i.value=a.matches},l=()=>{s&&("removeEventListener"in s?s.removeEventListener("change",o):s.removeListener(o))},c=$r(()=>{r.value&&(l(),s=n.matchMedia(Me(e)),"addEventListener"in s?s.addEventListener("change",o):s.addListener(o),i.value=s.matches)});return Kr(()=>{c(),l(),s=void 0}),i}const un=typeof globalThis<"u"?globalThis:typeof window<"u"?window:typeof global<"u"?global:typeof self<"u"?self:{},fn="__vueuse_ssr_handlers__",ka=ja();function ja(){return fn in un||(un[fn]=un[fn]||{}),un[fn]}function bo(e,t){return ka[e]||t}function Va(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 Ua={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()}},Ks="vueuse-storage";function Wr(e,t,n,r={}){var s;const{flush:i="pre",deep:o=!0,listenToStorageChanges:l=!0,writeDefaults:c=!0,mergeDefaults:a=!1,shallow:f,window:h=He,eventFilter:g,onError:y=w=>{console.error(w)},initOnMounted:C}=r,I=(f?Nr:le)(typeof t=="function"?t():t);if(!n)try{n=bo("getDefaultStorage",()=>{var w;return(w=He)==null?void 0:w.localStorage})()}catch(w){y(w)}if(!n)return I;const k=Me(t),K=Va(k),j=(s=r.serializer)!=null?s:Ua[K],{pause:p,resume:_}=Na(I,()=>M(I.value),{flush:i,deep:o,eventFilter:g});h&&l&&yo(()=>{zt(h,"storage",S),zt(h,Ks,F),C&&S()}),C||S();function L(w,H){h&&h.dispatchEvent(new CustomEvent(Ks,{detail:{key:e,oldValue:w,newValue:H,storageArea:n}}))}function M(w){try{const H=n.getItem(e);if(w==null)L(H,null),n.removeItem(e);else{const R=j.write(w);H!==R&&(n.setItem(e,R),L(H,R))}}catch(H){y(H)}}function N(w){const H=w?w.newValue:n.getItem(e);if(H==null)return c&&k!=null&&n.setItem(e,j.write(k)),k;if(!w&&a){const R=j.read(H);return typeof a=="function"?a(R,k):K==="object"&&!Array.isArray(R)?{...k,...R}:R}else return typeof H!="string"?H:j.read(H)}function S(w){if(!(w&&w.storageArea!==n)){if(w&&w.key==null){I.value=k;return}if(!(w&&w.key!==e)){p();try{(w==null?void 0:w.newValue)!==j.write(I.value)&&(I.value=N(w))}catch(H){y(H)}finally{w?Hn(_):_()}}}}function F(w){S(w.detail)}return I}function wo(e){return $a("(prefers-color-scheme: dark)",e)}function Ba(e={}){const{selector:t="html",attribute:n="class",initialValue:r="auto",window:s=He,storage:i,storageKey:o="vueuse-color-scheme",listenToStorageChanges:l=!0,storageRef:c,emitAuto:a,disableTransition:f=!0}=e,h={auto:"",light:"light",dark:"dark",...e.modes||{}},g=wo({window:s}),y=se(()=>g.value?"dark":"light"),C=c||(o==null?mo(r):Wr(o,r,i,{window:s,listenToStorageChanges:l})),I=se(()=>C.value==="auto"?y.value:C.value),k=bo("updateHTMLAttrs",(_,L,M)=>{const N=typeof _=="string"?s==null?void 0:s.document.querySelector(_):vo(_);if(!N)return;let S;if(f){S=s.document.createElement("style");const F="*,*::before,*::after{-webkit-transition:none!important;-moz-transition:none!important;-o-transition:none!important;-ms-transition:none!important;transition:none!important}";S.appendChild(document.createTextNode(F)),s.document.head.appendChild(S)}if(L==="class"){const F=M.split(/\s/g);Object.values(h).flatMap(w=>(w||"").split(/\s/g)).filter(Boolean).forEach(w=>{F.includes(w)?N.classList.add(w):N.classList.remove(w)})}else N.setAttribute(L,M);f&&(s.getComputedStyle(S).opacity,document.head.removeChild(S))});function K(_){var L;k(t,n,(L=h[_])!=null?L:_)}function j(_){e.onChanged?e.onChanged(_,K):K(_)}Qe(I,j,{flush:"post",immediate:!0}),yo(()=>j(I.value));const p=se({get(){return a?C.value:I.value},set(_){C.value=_}});try{return Object.assign(p,{store:C,system:y,state:I})}catch{return p}}function Ka(e={}){const{valueDark:t="dark",valueLight:n="",window:r=He}=e,s=Ba({...e,onChanged:(l,c)=>{var a;e.onChanged?(a=e.onChanged)==null||a.call(e,l==="dark",c,l):c(l)},modes:{dark:t,light:n}}),i=se(()=>s.system?s.system.value:wo({window:r}).value?"dark":"light");return se({get(){return s.value==="dark"},set(l){const c=l?"dark":"light";i.value===c?s.value="auto":s.value=c}})}function sr(e){return typeof Window<"u"&&e instanceof Window?e.document.documentElement:typeof Document<"u"&&e instanceof Document?e.documentElement:e}function Vu(e,t,n={}){const{window:r=He}=n;return Wr(e,t,r==null?void 0:r.localStorage,n)}function xo(e){const t=window.getComputedStyle(e);if(t.overflowX==="scroll"||t.overflowY==="scroll"||t.overflowX==="auto"&&e.clientWidth1?!0:(t.preventDefault&&t.preventDefault(),!1)}const dn=new WeakMap;function Uu(e,t=!1){const n=le(t);let r=null;Qe(mo(e),o=>{const l=sr(Me(o));if(l){const c=l;dn.get(c)||dn.set(c,c.style.overflow),n.value&&(c.style.overflow="hidden")}},{immediate:!0});const s=()=>{const o=sr(Me(e));!o||n.value||(Bs&&(r=zt(o,"touchmove",l=>{Wa(l)},{passive:!1})),o.style.overflow="hidden",n.value=!0)},i=()=>{var o;const l=sr(Me(e));!l||!n.value||(Bs&&(r==null||r()),l.style.overflow=(o=dn.get(l))!=null?o:"",dn.delete(l),n.value=!1)};return Kr(i),se({get(){return n.value},set(o){o?s():i()}})}function Bu(e,t,n={}){const{window:r=He}=n;return Wr(e,t,r==null?void 0:r.sessionStorage,n)}function Ku(e={}){const{window:t=He,behavior:n="auto"}=e;if(!t)return{x:le(0),y:le(0)};const r=le(t.scrollX),s=le(t.scrollY),i=se({get(){return r.value},set(l){scrollTo({left:l,behavior:n})}}),o=se({get(){return s.value},set(l){scrollTo({top:l,behavior:n})}});return zt(t,"scroll",()=>{r.value=t.scrollX,s.value=t.scrollY},{capture:!1,passive:!0}),{x:i,y:o}}const Co=/^[a-z]+:/i,qa="vitepress-theme-appearance",Eo=/#.*$/,za=/(index)?\.(md|html)$/,Ee=typeof document<"u",To={relativePath:"",filePath:"",title:"404",description:"Not Found",headers:[],frontmatter:{sidebar:!1,layout:"page"},lastUpdated:0,isNotFound:!0};function Ya(e,t,n=!1){if(t===void 0)return!1;if(e=Ws(`/${e}`),n)return new RegExp(t).test(e);if(Ws(t)!==e)return!1;const r=t.match(Eo);return r?(Ee?location.hash:"")===r[0]:!0}function Ws(e){return decodeURI(e).replace(Eo,"").replace(za,"")}function Ga(e){return Co.test(e)}function Ja(e,t){var r,s,i,o,l,c,a;const n=Object.keys(e.locales).find(f=>f!=="root"&&!Ga(f)&&Ya(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:((i=e.locales[n])==null?void 0:i.title)??e.title,titleTemplate:((o=e.locales[n])==null?void 0:o.titleTemplate)??e.titleTemplate,description:((l=e.locales[n])==null?void 0:l.description)??e.description,head:Ao(e.head,((c=e.locales[n])==null?void 0:c.head)??[]),themeConfig:{...e.themeConfig,...(a=e.locales[n])==null?void 0:a.themeConfig}})}function So(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=Xa(e.title,r);return`${n}${s}`}function Xa(e,t){return t===!1?"":t===!0||t===void 0?` | ${e}`:e===t?"":` | ${t}`}function Za(e,t){const[n,r]=t;if(n!=="meta")return!1;const s=Object.entries(r)[0];return s==null?!1:e.some(([i,o])=>i===n&&o[s[0]]===s[1])}function Ao(e,t){return[...e.filter(n=>!Za(t,n)),...t]}const Qa=/[\u0000-\u001F"#$&*+,:;<=>?[\]^`{|}\u007F]/g,eu=/^[a-z]:/i;function qs(e){const t=eu.exec(e),n=t?t[0]:"";return n+e.slice(n.length).replace(Qa,"_").replace(/(^|\/)_+(?=[^/]*$)/,"$1")}const tu=Symbol(),dt=Nr(Sa);function Wu(e){const t=se(()=>Ja(dt.value,e.data.relativePath)),n=t.value.appearance,r=n==="force-dark"?le(!0):n?Ka({storageKey:qa,initialValue:()=>typeof n=="string"?n:"auto",...typeof n=="object"?n:{}}):le(!1);return{site:t,theme:se(()=>t.value.themeConfig),page:se(()=>e.data),frontmatter:se(()=>e.data.frontmatter),params:se(()=>e.data.params),lang:se(()=>t.value.lang),dir:se(()=>t.value.dir),localeIndex:se(()=>t.value.localeIndex||"root"),title:se(()=>So(t.value,e.data)),description:se(()=>e.data.description||t.value.description),isDark:r}}function nu(){const e=Ct(tu);if(!e)throw new Error("vitepress data not properly injected in app");return e}function ru(e,t){return`${e}${t}`.replace(/\/+/g,"/")}function zs(e){return Co.test(e)||!e.startsWith("/")?e:ru(dt.value.base,e)}function su(e){let t=e.replace(/\.html$/,"");if(t=decodeURIComponent(t),t=t.replace(/\/$/,"/index"),Ee){const n="/";t=qs(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=`./${qs(t.slice(1).replace(/\//g,"_"))}.md.js`;return t}let vn=[];function qu(e){vn.push(e),Un(()=>{vn=vn.filter(t=>t!==e)})}const iu=Symbol(),Ro="http://a.com",ou=()=>({path:"/",component:null,data:To});function zu(e,t){const n=Nn(ou()),r={route:n,go:s};async function s(l=Ee?location.href:"/"){var c,a;l=br(l),await((c=r.onBeforeRouteChange)==null?void 0:c.call(r,l))!==!1&&(Js(l),await o(l),await((a=r.onAfterRouteChanged)==null?void 0:a.call(r,l)))}let i=null;async function o(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,Ro),h=i=f.pathname;try{let y=await e(h);if(!y)throw new Error(`Page not found: ${h}`);if(i===h){i=null;const{default:C,__pageData:I}=y;if(!C)throw new Error(`Invalid route component: ${C}`);n.path=Ee?h:zs(h),n.component=mn(C),n.data=mn(I),Ee&&Hn(()=>{let k=dt.value.base+I.relativePath.replace(/(?:(^|\/)index)?\.md$/,"$1");if(!dt.value.cleanUrls&&!k.endsWith("/")&&(k+=".html"),k!==f.pathname&&(f.pathname=k,l=k+f.search+f.hash,history.replaceState(null,"",l)),f.hash&&!c){let K=null;try{K=document.getElementById(decodeURIComponent(f.hash).slice(1))}catch(j){console.warn(j)}if(K){Ys(K,f.hash);return}}window.scrollTo(0,c)})}}catch(y){if(!/fetch|Page not found/.test(y.message)&&!/^\/404(\.html|\/)?$/.test(l)&&console.error(y),!a)try{const C=await fetch(dt.value.base+"hashmap.json");window.__VP_HASH_MAP__=await C.json(),await o(l,c,!0);return}catch{}i===h&&(i=null,n.path=Ee?h:zs(h),n.component=t?mn(t):null,n.data=To)}}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:y,hash:C,search:I}=new URL(a.href instanceof SVGAnimatedString?a.href.animVal:a.href,a.baseURI),k=window.location,K=y.match(/\.\w+$/);!l.ctrlKey&&!l.shiftKey&&!l.altKey&&!l.metaKey&&!f&&g===k.origin&&!(K&&K[0]!==".html")&&(l.preventDefault(),y===k.pathname&&I===k.search?(C!==k.hash&&(history.pushState(null,"",C),window.dispatchEvent(new Event("hashchange"))),C?Ys(a,C,a.classList.contains("header-anchor")):(Js(h),window.scrollTo(0,0))):s(h))}},{capture:!0}),window.addEventListener("popstate",l=>{o(br(location.href),l.state&&l.state.scrollPosition||0)}),window.addEventListener("hashchange",l=>{l.preventDefault()})),r}function lu(){const e=Ct(iu);if(!e)throw new Error("useRouter() is called without provider.");return e}function Mo(){return lu().route}function Ys(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=dt.value.scrollOffset,i=0,o=24;if(typeof s=="object"&&"padding"in s&&(o=s.padding,s=s.selector),typeof s=="number")i=s;else if(typeof s=="string")i=Gs(s,o);else if(Array.isArray(s))for(const f of s){const h=Gs(f,o);if(h){i=h;break}}const l=parseInt(window.getComputedStyle(r).paddingTop,10),c=window.scrollY+r.getBoundingClientRect().top-i+l;requestAnimationFrame(a)}}function Gs(e,t){const n=document.querySelector(e);if(!n)return 0;const r=n.getBoundingClientRect().bottom;return r<0?0:r+t}function Js(e){Ee&&e!==br(location.href)&&(history.replaceState({scrollPosition:window.scrollY},document.title),history.pushState(null,"",e))}function br(e){const t=new URL(e,Ro);return t.pathname=t.pathname.replace(/(^|\/)index(\.html)?$/,"$1"),dt.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 Xs=()=>vn.forEach(e=>e()),Yu=kr({name:"VitePressContent",props:{as:{type:[Object,String],default:"div"}},setup(e){const t=Mo(),{site:n}=nu();return()=>vr(e.as,n.value.contentProps??{style:{position:"relative"}},[t.component?vr(t.component,{onVnodeMounted:Xs,onVnodeUpdated:Xs}):"404 Page Not Found"])}}),Gu=kr({setup(e,{slots:t}){const n=le(!1);return St(()=>{n.value=!0}),()=>n.value&&t.default?t.default():null}});function Ju(){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 i=r.querySelector(".blocks");if(!i)return;const o=Array.from(i.children).find(a=>a.classList.contains("active"));if(!o)return;const l=i.children[s];if(!l||o===l)return;o.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 Xu(){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,i=(r=n.nextElementSibling)==null?void 0:r.nextElementSibling;if(!s||!i)return;const o=/language-(shellscript|shell|bash|sh|zsh)/.test(s.className);let l="";i.querySelectorAll("span.line:not(.diff.remove)").forEach(c=>l+=(c.textContent||"")+` +`),l=l.slice(0,-1),o&&(l=l.replace(/^ *(\$|>) /gm,"").trim()),cu(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 cu(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 Zu(e,t){let n=[],r=!0;const s=i=>{if(r){r=!1;return}const o=i.map(Zs);n.forEach((l,c)=>{const a=o.findIndex(f=>f==null?void 0:f.isEqualNode(l??null));a!==-1?delete o[a]:(l==null||l.remove(),delete n[c])}),o.forEach(l=>l&&document.head.appendChild(l)),n=[...n,...o].filter(Boolean)};$r(()=>{const i=e.data,o=t.value,l=i&&i.description,c=i&&i.frontmatter.head||[],a=So(o,i);a!==document.title&&(document.title=a);const f=l||o.description;let h=document.querySelector("meta[name=description]");h?h.getAttribute("content")!==f&&h.setAttribute("content",f):Zs(["meta",{name:"description",content:f}]),s(Ao(o.head,uu(c)))})}function Zs([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 au(e){return e[0]==="meta"&&e[1]&&e[1].name==="description"}function uu(e){return e.filter(t=>!au(t))}const ir=new Set,Oo=()=>document.createElement("link"),fu=e=>{const t=Oo();t.rel="prefetch",t.href=e,document.head.appendChild(t)},du=e=>{const t=new XMLHttpRequest;t.open("GET",e,t.withCredentials=!0),t.send()};let hn;const hu=Ee&&(hn=Oo())&&hn.relList&&hn.relList.supports&&hn.relList.supports("prefetch")?fu:du;function Qu(){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(i=>{i.forEach(o=>{if(o.isIntersecting){const l=o.target;n.unobserve(l);const{pathname:c}=l;if(!ir.has(c)){ir.add(c);const a=su(c);a&&hu(a)}}})}),t(()=>{document.querySelectorAll("#app a").forEach(i=>{const{hostname:o,pathname:l}=new URL(i.href instanceof SVGAnimatedString?i.href.animVal:i.href,i.baseURI),c=l.match(/\.\w+$/);c&&c[0]!==".html"||i.target!=="_blank"&&o===location.hostname&&(l!==location.pathname?n.observe(i):ir.add(l))})})};St(r);const s=Mo();Qe(()=>s.path,r),Un(()=>{n&&n.disconnect()})}export{Eu as $,vu as A,ec as B,_u as C,xu as D,Nr as E,ge as F,qu as G,ie as H,yu as I,Co as J,Mo as K,Hc as L,Ct as M,In as N,Hn as O,Ku as P,Ru as Q,Fn as R,Du as S,co as T,Tl as U,ju as V,wu as W,Hu as X,Uu as Y,pc as Z,Fu as _,no as a,Lu as a0,Iu as a1,Tu as a2,vr as a3,Zu as a4,iu as a5,Wu as a6,tu as a7,Yu as a8,Gu as a9,dt as aa,Nu as ab,zu as ac,su as ad,Qu as ae,Xu as af,Ju as ag,vo as ah,Kr as ai,ku as aj,Bu as ak,Vu as al,$u as am,lu as an,zt as ao,Fi as ap,bu as aq,Ou as ar,fe as as,Su as at,mn as au,Pu as av,Qi as b,Au as c,kr as d,Mu as e,zs as f,se as g,le as h,Ga as i,St as j,to as k,Fr as l,mu as m,Ln as n,Xi as o,gu as p,Ya as q,Cu as r,Ee as s,pu as t,nu as u,$a as v,Dl as w,Qe as x,$r as y,Un as z}; diff --git a/assets/chunks/mainnet_versions.955428b6.js b/assets/chunks/mainnet_versions.955428b6.js new file mode 100644 index 00000000000..bd8c22359ec --- /dev/null +++ b/assets/chunks/mainnet_versions.955428b6.js @@ -0,0 +1 @@ +const e=Object.freeze({"app-latest-tag":"v2.1.2","app-latest-sha":"48173df3dc78f9348eedb3796f29ef9e9e5dc584","core-latest-tag":"v1.41.0-tm-v0.34.29","core-latest-sha":"aef322775c75783488b439cdf7ca33f38da08426","node-latest-tag":"v0.16.0","node-latest-sha":"6744f648649ebb5fee1b27faf7aca96ecf4519b2"});export{e as m}; diff --git a/assets/chunks/mocha_versions.7704b055.js b/assets/chunks/mocha_versions.7704b055.js new file mode 100644 index 00000000000..c73494bf47d --- /dev/null +++ b/assets/chunks/mocha_versions.7704b055.js @@ -0,0 +1 @@ +const e=Object.freeze({"app-latest-tag":"v2.1.2","app-latest-sha":"48173df3dc78f9348eedb3796f29ef9e9e5dc584","core-latest-tag":"v1.40.1-tm-v0.34.29-rc0","core-latest-sha":"0d2b63836d0f4587e162bfded58f53fba238e69c","node-latest-tag":"v0.16.0","node-latest-sha":"6744f648649ebb5fee1b27faf7aca96ecf4519b2"});export{e as m}; diff --git a/assets/chunks/theme.9c6b4fd0.js b/assets/chunks/theme.9c6b4fd0.js new file mode 100644 index 00000000000..6f498a2f630 --- /dev/null +++ b/assets/chunks/theme.9c6b4fd0.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,e as f,T as ce,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 K,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 Re,V as ke,W as qe,X as Ke,Y as Se,Z as Me,$ as We,a0 as Ye,a1 as Je,a2 as Xe}from"./framework.1a91c06a.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(K?location.hash:"");K&&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,R]=Vt(S,C,B);if(w){d(R);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"},Rt=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),qt=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),Kt=[Rt,qt];function Wt(s,e){return a(),i("svg",jt,Kt)}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"],Rs={key:0,class:"actions"},qs={key:0,class:"image"},Ks={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",Rs,[(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",qs,[c("div",Ks,[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 Rn(s,e){return a(),i("svg",En,jn)}const qn=m(Dn,[["render",Rn]]),Kn=["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(qn,{class:"menu-icon"}),c("span",Wn,L(l(e).sidebarMenuLabel||"Menu"),1)],8,Kn)):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(K){!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"},Ro=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 qo=m(Ro,[["__scopeId","data-v-e42ed9b3"]]),Ko=["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,Ko),c("div",Jo,[h(qo,{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 R=y==null?void 0:y[B];R&&(w=y=R);const W=V==null?void 0:V[B];W&&(w=V=W);const q=g==null?void 0:g[B];q&&(w=g=q),R||(y=w),W||(V=w),q||(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)(Re(()=>{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=qe(()=>Ke(()=>import("./VPLocalSearchBox.43aac247.js"),["assets/chunks/VPLocalSearchBox.43aac247.js","assets/chunks/framework.1a91c06a.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={},Ra={xmlns:"http://www.w3.org/2000/svg","aria-hidden":"true",focusable:"false",viewBox:"0 0 24 24"},qa=c("path",{d:"M0 0h24v24H0z",fill:"none"},null,-1),Ka=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=[qa,Ka];function Ya(s,e){return a(),i("svg",Ra,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(K?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(()=>{K&&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"],Rr=Ur(()=>c("div",{class:"indicator"},null,-1)),qr={key:1,class:"items"},Kr=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,R)=>{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},We(w.item.items?{click:C,keydown:C}:{},!0),{tabindex:w.item.items&&0}),[Rr,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:Ye(B,["enter"]),tabindex:"0"},[h(fe,{class:"caret-icon"})],32)):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,q=>(a(),$(W,{key:q.text,item:q,depth:w.depth+1},null,8,["item","depth"]))),128)):f("",!0)])):f("",!0)]),_:1},8,["class"])}}});const Wr=m(Kr,[["__scopeId","data-v-bd01e0d5"]]),He=s=>(H("data-v-ee2efba5"),s=s(),z(),s),Yr=He(()=>c("div",{class:"curtain"},null,-1)),Jr={class:"nav",id:"VPSidebarNav","aria-labelledby":"sidebar-aria-label",tabindex:"-1"},Xr=He(()=>c("span",{class:"visually-hidden",id:"sidebar-aria-label"}," Sidebar Navigation ",-1)),Zr=b({__name:"VPSidebar",props:{open:{type:Boolean}},setup(s){const{sidebarGroups:e,hasSidebar:t}=D(),n=s,o=M(null),r=Se(K?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"]))},[Yr,c("nav",Jr,[Xr,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(Wr,{item:_,depth:0},null,8,["item"])]))),128)),u(d.$slots,"sidebar-nav-after",{},void 0,!0)])],2)):f("",!0)}});const Qr=m(Zr,[["__scopeId","data-v-ee2efba5"]]),el=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 tl=m(el,[["__scopeId","data-v-c8291ffa"]]),sl=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(tl),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(Qr,{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 nl=m(sl,[["__scopeId","data-v-9d8abc1e"]]);const al={Layout:nl,enhanceApp:({app:s})=>{s.component("Badge",Qe)}};export{Na as c,al 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_coc.md.e7f86a12.js b/assets/community_coc.md.e7f86a12.js new file mode 100644 index 00000000000..12019d4563b --- /dev/null +++ b/assets/community_coc.md.e7f86a12.js @@ -0,0 +1 @@ +import{_ as e,o,c as a,Q as t}from"./chunks/framework.1a91c06a.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.e7f86a12.lean.js b/assets/community_coc.md.e7f86a12.lean.js new file mode 100644 index 00000000000..a2c196f702b --- /dev/null +++ b/assets/community_coc.md.e7f86a12.lean.js @@ -0,0 +1 @@ +import{_ as e,o,c as a,Q as t}from"./chunks/framework.1a91c06a.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.bc29b071.js b/assets/community_foundation-delegation-program.md.bc29b071.js new file mode 100644 index 00000000000..7c1d75af155 --- /dev/null +++ b/assets/community_foundation-delegation-program.md.bc29b071.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.1a91c06a.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":1726496322000}'),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 Beta validator or an active Mocha testnet validator for at least 1 month before application deadline
  • Run a bridge node (on Mainnet Beta if you are already an active Mainnet Beta 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
  • Maintain a fully archival (non pruned) bridge node for both Mainnet Beta and Mocha if selected for the program
  • Not running your infrastructure in Hetzner or OVH

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 Beta + 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 Beta
    • If you don't run an active Mainnet Beta 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 Beta 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.

  • Cohort 1: 50 Validator Seats
  • Cohort 2: 15 Validator Seats (Applications open June 1, 2024)
  • Cohort 3: 15 Validator Seats (Applications open October 1, 2024)
  • Cohort 4: 20 Validator Seats (Applications open February 1, 2025)

IMPORTANT: Each validator selected for the program has to maintain a fully archival (non pruned) bridge node for both Mainnet Beta and Mocha.

Feedback process

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

',39),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.bc29b071.lean.js b/assets/community_foundation-delegation-program.md.bc29b071.lean.js new file mode 100644 index 00000000000..7bf4da5d60d --- /dev/null +++ b/assets/community_foundation-delegation-program.md.bc29b071.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.1a91c06a.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":1726496322000}'),n={name:"community/foundation-delegation-program.md"},l=i("",39),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_modular-meetup-guide.md.d402e287.js b/assets/community_modular-meetup-guide.md.d402e287.js new file mode 100644 index 00000000000..3a3d4f4e284 --- /dev/null +++ b/assets/community_modular-meetup-guide.md.d402e287.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as i,Q as o}from"./chunks/framework.1a91c06a.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.d402e287.lean.js b/assets/community_modular-meetup-guide.md.d402e287.lean.js new file mode 100644 index 00000000000..e8bf615fdf5 --- /dev/null +++ b/assets/community_modular-meetup-guide.md.d402e287.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as i,Q as o}from"./chunks/framework.1a91c06a.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.8ebda042.js b/assets/community_modular-meetup-intro.md.8ebda042.js new file mode 100644 index 00000000000..ffd1b58a90b --- /dev/null +++ b/assets/community_modular-meetup-intro.md.8ebda042.js @@ -0,0 +1 @@ +import{_ as e,o,c as a,Q as t}from"./chunks/framework.1a91c06a.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.8ebda042.lean.js b/assets/community_modular-meetup-intro.md.8ebda042.lean.js new file mode 100644 index 00000000000..dc28f52e83d --- /dev/null +++ b/assets/community_modular-meetup-intro.md.8ebda042.lean.js @@ -0,0 +1 @@ +import{_ as e,o,c as a,Q as t}from"./chunks/framework.1a91c06a.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.6175a760.js b/assets/community_modular-meetup-toolkit.md.6175a760.js new file mode 100644 index 00000000000..49f228c92d3 --- /dev/null +++ b/assets/community_modular-meetup-toolkit.md.6175a760.js @@ -0,0 +1 @@ +import{_ as e,o,c as i,Q as t}from"./chunks/framework.1a91c06a.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.6175a760.lean.js b/assets/community_modular-meetup-toolkit.md.6175a760.lean.js new file mode 100644 index 00000000000..85463fe0a90 --- /dev/null +++ b/assets/community_modular-meetup-toolkit.md.6175a760.lean.js @@ -0,0 +1 @@ +import{_ as e,o,c as i,Q as t}from"./chunks/framework.1a91c06a.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_speaker-list.md.a4ac972d.js b/assets/community_speaker-list.md.a4ac972d.js new file mode 100644 index 00000000000..dda77ab70d6 --- /dev/null +++ b/assets/community_speaker-list.md.a4ac972d.js @@ -0,0 +1 @@ +import{_ as t,o as a,c as s,k as e,a as i}from"./chunks/framework.1a91c06a.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.a4ac972d.lean.js b/assets/community_speaker-list.md.a4ac972d.lean.js new file mode 100644 index 00000000000..dda77ab70d6 --- /dev/null +++ b/assets/community_speaker-list.md.a4ac972d.lean.js @@ -0,0 +1 @@ +import{_ as t,o as a,c as s,k as e,a as i}from"./chunks/framework.1a91c06a.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.79abe3b3.js b/assets/developers_arbitrum-bridge.md.79abe3b3.js new file mode 100644 index 00000000000..1a5204801d2 --- /dev/null +++ b/assets/developers_arbitrum-bridge.md.79abe3b3.js @@ -0,0 +1 @@ +import{_ as t,o as r,c as o,Q as i}from"./chunks/framework.1a91c06a.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.","next":{"text":"Intro to OP Stack integration","link":"/developers/intro-to-op-stack"},"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":1726491136000}'),k={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),A=[S];function O(C,B,P,q,T,I){return r(),o("div",null,A)}const L=t(k,[["render",O]]);export{D as __pageData,L as default}; diff --git a/assets/developers_arbitrum-bridge.md.79abe3b3.lean.js b/assets/developers_arbitrum-bridge.md.79abe3b3.lean.js new file mode 100644 index 00000000000..3e97610119b --- /dev/null +++ b/assets/developers_arbitrum-bridge.md.79abe3b3.lean.js @@ -0,0 +1 @@ +import{_ as t,o as r,c as o,Q as i}from"./chunks/framework.1a91c06a.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.","next":{"text":"Intro to OP Stack integration","link":"/developers/intro-to-op-stack"},"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":1726491136000}'),k={name:"developers/arbitrum-bridge.md"},S=i("",50),A=[S];function O(C,B,P,q,T,I){return r(),o("div",null,A)}const L=t(k,[["render",O]]);export{D as __pageData,L as default}; diff --git a/assets/developers_arbitrum-deploy.md.e3157085.js b/assets/developers_arbitrum-deploy.md.e3157085.js new file mode 100644 index 00000000000..9f3882c81c5 --- /dev/null +++ b/assets/developers_arbitrum-deploy.md.e3157085.js @@ -0,0 +1,111 @@ +import{_ as s,o as a,c as o,Q as n}from"./chunks/framework.1a91c06a.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.e3157085.lean.js b/assets/developers_arbitrum-deploy.md.e3157085.lean.js new file mode 100644 index 00000000000..7788fb457c6 --- /dev/null +++ b/assets/developers_arbitrum-deploy.md.e3157085.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as o,Q as n}from"./chunks/framework.1a91c06a.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.b72127e2.js b/assets/developers_arbitrum-full-node.md.b72127e2.js new file mode 100644 index 00000000000..a9504c199fd --- /dev/null +++ b/assets/developers_arbitrum-full-node.md.b72127e2.js @@ -0,0 +1,3 @@ +import{_ as n,o as e,c as o,Q as a}from"./chunks/framework.1a91c06a.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.b72127e2.lean.js b/assets/developers_arbitrum-full-node.md.b72127e2.lean.js new file mode 100644 index 00000000000..5898371c586 --- /dev/null +++ b/assets/developers_arbitrum-full-node.md.b72127e2.lean.js @@ -0,0 +1 @@ +import{_ as n,o as e,c as o,Q as a}from"./chunks/framework.1a91c06a.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.1e6a0883.js b/assets/developers_arbitrum-integration.md.1e6a0883.js new file mode 100644 index 00000000000..8467678e69e --- /dev/null +++ b/assets/developers_arbitrum-integration.md.1e6a0883.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.1a91c06a.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":1722891813000}'),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 node RPC endpoint, you can find a list of them for Mocha
  • 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),s=[n];function l(h,c,d,m,b,u){return t(),a("div",null,s)}const g=e(o,[["render",l]]);export{f as __pageData,g as default}; diff --git a/assets/developers_arbitrum-integration.md.1e6a0883.lean.js b/assets/developers_arbitrum-integration.md.1e6a0883.lean.js new file mode 100644 index 00000000000..0542357381b --- /dev/null +++ b/assets/developers_arbitrum-integration.md.1e6a0883.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.1a91c06a.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":1722891813000}'),o={name:"developers/arbitrum-integration.md"},n=i("",34),s=[n];function l(h,c,d,m,b,u){return t(),a("div",null,s)}const g=e(o,[["render",l]]);export{f as __pageData,g as default}; diff --git a/assets/developers_blobstream-contracts.md.6f2c6e34.js b/assets/developers_blobstream-contracts.md.6f2c6e34.js new file mode 100644 index 00000000000..1e6c98d6072 --- /dev/null +++ b/assets/developers_blobstream-contracts.md.6f2c6e34.js @@ -0,0 +1,87 @@ +import{_ as s,o as a,c as o,Q as e}from"./chunks/framework.1a91c06a.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","prev":{"text":"New SP1 Blobstream deployments","link":"/developers/sp1-blobstream-deploy"},"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":1725955964000}`),n={name:"developers/blobstream-contracts.md"},l=e(`

Integrate with Blobstream contracts

Getting started

Prerequisites

Make sure to have the following installed:

Installing Blobstream contracts

We will be using the IDAOracle interface to verify inclusion. So, we will install the Blobstream contracts repo as a dependency:

sh
forge install celestiaorg/blobstream-contracts --no-commit
forge install celestiaorg/blobstream-contracts --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;
+
+import "blobstream-contracts/IDAOracle.sol";
+import "blobstream-contracts/DataRootTuple.sol";
+import "blobstream-contracts/lib/tree/binary/BinaryMerkleProof.sol";
+
+contract MyRollup {
+    IDAOracle immutable blobstream;
+    bytes32[] public rollup_block_hashes;
+
+    constructor(IDAOracle _blobstream) {
+        blobstream = _blobstream;
+    }
+
+    function submitRollupBlock(
+        bytes32 _rollup_block_hash,
+        bytes calldata _zk_proof,
+        uint256 _blobstream_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(
+            blobstream.verifyAttestation(_blobstream_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;
+
+import "blobstream-contracts/IDAOracle.sol";
+import "blobstream-contracts/DataRootTuple.sol";
+import "blobstream-contracts/lib/tree/binary/BinaryMerkleProof.sol";
+
+contract MyRollup {
+    IDAOracle immutable blobstream;
+    bytes32[] public rollup_block_hashes;
+
+    constructor(IDAOracle _blobstream) {
+        blobstream = _blobstream;
+    }
+
+    function submitRollupBlock(
+        bytes32 _rollup_block_hash,
+        bytes calldata _zk_proof,
+        uint256 _blobstream_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(
+            blobstream.verifyAttestation(_blobstream_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.

`,27),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.6f2c6e34.lean.js b/assets/developers_blobstream-contracts.md.6f2c6e34.lean.js new file mode 100644 index 00000000000..817cf9781d7 --- /dev/null +++ b/assets/developers_blobstream-contracts.md.6f2c6e34.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as o,Q as e}from"./chunks/framework.1a91c06a.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","prev":{"text":"New SP1 Blobstream deployments","link":"/developers/sp1-blobstream-deploy"},"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":1725955964000}`),n={name:"developers/blobstream-contracts.md"},l=e("",27),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.a074cb83.js b/assets/developers_blobstream-offchain.md.a074cb83.js new file mode 100644 index 00000000000..467dce7fce9 --- /dev/null +++ b/assets/developers_blobstream-offchain.md.a074cb83.js @@ -0,0 +1,297 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.1a91c06a.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":1725955964000}`),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 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.a074cb83.lean.js b/assets/developers_blobstream-offchain.md.a074cb83.lean.js new file mode 100644 index 00000000000..f0ac9d4bbe7 --- /dev/null +++ b/assets/developers_blobstream-offchain.md.a074cb83.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.1a91c06a.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":1725955964000}`),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.34df4e6b.js b/assets/developers_blobstream-proof-queries.md.34df4e6b.js new file mode 100644 index 00000000000..003010471f2 --- /dev/null +++ b/assets/developers_blobstream-proof-queries.md.34df4e6b.js @@ -0,0 +1,1841 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.1a91c06a.js";const o="/img/blobstream/blobstream-square.png",p="/img/blobstream/blobstream-commitment-diagram.png",h=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":1726496322000}'),e={name:"developers/blobstream-proof-queries.md"},t=l(`

Blobstream proofs queries

Prerequisites

  • Access to a Celestia 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.

For golang snippets, the tendermint RPC client, referred to as trpc, will be used for the queries. It can be initialized using:

go
    trpc, err := http.New("<rpc_endpoint>", "/websocket")
+	if err != nil {
+		...
+	}
+	err = trpc.Start()
+	if err != nil {
+		return err
+	}
+	defer func(trpc *http.HTTP) {
+		err := trpc.Stop()
+		if err != nil {
+			...
+		}
+	}(trpc)
    trpc, err := http.New("<rpc_endpoint>", "/websocket")
+	if err != nil {
+		...
+	}
+	err = trpc.Start()
+	if err != nil {
+		return err
+	}
+	defer func(trpc *http.HTTP) {
+		err := trpc.Stop()
+		if err != nil {
+			...
+		}
+	}(trpc)

The <rpc_endpoint> can be retrieved from Mainnet Beta for and Mocha for the Mocha testnet.

In case the reader wants to interact with an on-chain contract that can be used to verify that data was posted to Celestia, the bindings of that contract are needed.

For Blobstream, the golang bindings can be found in the following links:

text
https://github.com/succinctlabs/blobstreamx/blob/main/bindings/BlobstreamX.go
https://github.com/succinctlabs/blobstreamx/blob/main/bindings/BlobstreamX.go
text
https://github.com/succinctlabs/sp1-blobstream/blob/main/bindings/SP1Blobstream.go
https://github.com/succinctlabs/sp1-blobstream/blob/main/bindings/SP1Blobstream.go

For other languages, the corresponding smart contract bindings should be generated. Refer to abigen for more information.

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. In a nutshell, Blobstream attests to the data posted to Celestia in the zk-Blobstream 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 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 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 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 contract.

HTTP query

Example HTTP request: <tendermint_rpc_endpoint>/data_root_inclusion_proof?height=15&start=10&end=20

Which queries the proof of the height 15 to the data commitment defined by the range [10, 20).

Example response:

json
{
+  "jsonrpc": "2.0",
+  "id": -1,
+  "result": {
+    "proof": {
+      "total": "10",
+      "index": "5",
+      "leaf_hash": "vkRaRg7FGtZ/ZhsJRh/Uhhb3U6dPaYJ1pJNEfrwq5HE=",
+      "aunts": [
+        "nmBWWwHpipHwagaI7MAqM/yhCDb4cz7z4lRxmVRq5f8=",
+        "nyzLbFJjnSKOfRZur8xvJiJLA+wBPtwm0KbYglILxLg=",
+        "GI/tJ9WSwcyHM0r0i8t+p3hPFtDieuYR9wSPVkL1r2s=",
+        "+SGf6MfzMmtDKz5MLlH+y7mPV9Moo2x5rLjLe3gbFQo="
+      ]
+    }
+  }
+}
{
+  "jsonrpc": "2.0",
+  "id": -1,
+  "result": {
+    "proof": {
+      "total": "10",
+      "index": "5",
+      "leaf_hash": "vkRaRg7FGtZ/ZhsJRh/Uhhb3U6dPaYJ1pJNEfrwq5HE=",
+      "aunts": [
+        "nmBWWwHpipHwagaI7MAqM/yhCDb4cz7z4lRxmVRq5f8=",
+        "nyzLbFJjnSKOfRZur8xvJiJLA+wBPtwm0KbYglILxLg=",
+        "GI/tJ9WSwcyHM0r0i8t+p3hPFtDieuYR9wSPVkL1r2s=",
+        "+SGf6MfzMmtDKz5MLlH+y7mPV9Moo2x5rLjLe3gbFQo="
+      ]
+    }
+  }
+}

NOTE: These values are base64 encoded. For these to be usable with the solidity smart contract, they need to be converted to bytes32. Check the next section for more information.

Golang client

The endpoint can also 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 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
+}
go
// Similar to Blobstream, except replace the BlobstreamX contract with SP1 Blobstream:
+import {
+  sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
+}
// Similar to Blobstream, except replace the BlobstreamX contract with SP1 Blobstream:
+import {
+  sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
+}

2. Transaction inclusion proof

To prove that a rollup transaction, the PFB transaction and not the blob containing the Rollup blocks data, 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.

Transaction inclusion proof using the transaction hash

Given a transaction hash, the transaction inclusion proof can be queried from the transaction query.

HTTP request

Example request: <tendermint_rpc_endpoint>/tx?hash=0xEF9F50BFB39F11B022A6CD7026574ECCDC6D596689BDCCC7B2C482A1B26B26B8&prove=true

Which queries the transaction whose hash is EF9F50BFB39F11B022A6CD7026574ECCDC6D596689BDCCC7B2C482A1B26B26B8 and sets the prove parameter as true to also get its inclusion proof.

Example response:

json
{
+  "jsonrpc": "2.0",
+  "id": -1,
+  "result": {
+    "hash": "EF9F50BFB39F11B022A6CD7026574ECCDC6D596689BDCCC7B2C482A1B26B26B8",
+    "height": "1350632",
+    "index": 4,
+    "tx_result": {
+      "code": 0,
+      "data": "EioKKC9jZWxlc3RpYS5ibG9iLnYxLk1zZ1BheUZvckJsb2JzUmVzcG9uc2U=",
+      "log": "[{\\"msg_index\\":0,\\"events\\":[{\\"type\\":\\"celestia.blob.v1.EventPayForBlobs\\",\\"attributes\\":[{\\"key\\":\\"blob_sizes\\",\\"value\\":\\"[120000]\\"},{\\"key\\":\\"namespaces\\",\\"value\\":\\"[\\\\\\"AAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUU=\\\\\\"]\\"},{\\"key\\":\\"signer\\",\\"value\\":\\"\\\\\\"celestia1vdjkcetnw35kzvtk8pjhxcm4xan82wtvwcurwwtt0f6n2at9va6k2atjw3cn2umhxe58xmfndejs40vqs9\\\\\\"\\"}]},{\\"type\\":\\"message\\",\\"attributes\\":[{\\"key\\":\\"action\\",\\"value\\":\\"/celestia.blob.v1.MsgPayForBlobs\\"}]}]}]",
+      "info": "",
+      "gas_wanted": "1095604",
+      "gas_used": "1080694",
+      "events": [
+        ...
+      ],
+      "codespace": ""
+    },
+    "tx": "CqEBCp4BCiAvY2VsZXN0aWEuYmxvYi52MS5Nc2dQYXlGb3JCbG9icxJ6Ci9jZWxlc3RpYTF2OGVzY3U3ZnU5bHY4NzlrenU1dWVndWV1cnRxNXN3NmhzbTNuZRIdAAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUUaA8CpByIgsVXWya9toI+AyTu3JJA2wkI5ZLkm72/gklCGFLCSrm9CAQASaApSCkYKHy9jb3Ntb3MuY3J5cHRvLnNlY3AyNTZrMS5QdWJLZXkSIwohAqdvBVUpglaNDGTlOcGSoHERBAFsFBB5l0WdvBJjEsEHEgQKAggBGJjCARISCgwKBHV0aWESBDIxOTIQtO9CGkCJRjcOYijj81bttfb2GUdG7o8AuAwf0bscBhW9PPD99xpQ1slpemfyq0y1joJ/aRFgE6QNxuiZ18VLGlGEwtW/",
+    "proof": {
+      "data": [
+        "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQBAAACugAAACbaAgrNAgqfAQqcAQogL2NlbGVzdGlhLmJsb2IudjEuTXNnUGF5Rm9yQmxvYnMSeAovY2VsZXN0aWExYWxwNGZwbHF5d21jNmN1aDl5MzVlY2xycHF5cWF4MjN2Z3ZrczUSHQAAAAAAAAAAAAAAAAAAAAAAAAAAAAgICAgICAgIGgFeIiAzlOEQsGxg3rOw7SR1rkQ7dVJYGp3aXkaqy4oG6HYc0EIBABJnClIKRgofL2Nvc21vcy5jcnlwdG8uc2VjcDI1NmsxLlB1YktleRIjCiECA4ief8FZEaBQLVc2wOceFs+LhAK0mDnmPnsxYLkqv7QSBAoCCAEY5PEBEhEKCwoEdXRpYRIDMTYwELTvBBpA7XBbSGYFrwTZcFHq3va1vHtbRiCzYd0ELkAJo6kLSDooEQCwoVaGuwTdP55V8Btf3WC7/FEK44BOESwEwecTExICgQIaBElORFjcAgrQAgqhAQqeAQogL2NlbGVzdGlhLmJsb2IudjEuTXNnUGF5Rm9yQmxvYnMSegovY2VsZXN0aWExdjhlc2N1N2Z1OWx2ODc5a3p1NXVlZ3VldXJ0cTVzdzZoc20zbmUSHQAAAAAAAAAAAAAAAAAAAAAAAAAAAAWEy1OISzk=",
+        "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQAAAAAAEUaA8CpByIgsVXWya9toI+AyTu3JJA2wkI5ZLkm72/gklCGFLCSrm9CAQASaApSCkYKHy9jb3Ntb3MuY3J5cHRvLnNlY3AyNTZrMS5QdWJLZXkSIwohAqdvBVUpglaNDGTlOcGSoHERBAFsFBB5l0WdvBJjEsEHEgQKAggBGJjCARISCgwKBHV0aWESBDIxOTIQtO9CGkCJRjcOYijj81bttfb2GUdG7o8AuAwf0bscBhW9PPD99xpQ1slpemfyq0y1joJ/aRFgE6QNxuiZ18VLGlGEwtW/EgEIGgRJTkRYAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA="
+      ],
+      "share_proofs": [
+        {
+          "start": 5,
+          "end": 7,
+          "nodes": [
+            "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQs98k+8wQ2iX2BdcTfoHjtRQbqybtPdB1BUFY/D7WRs",
+            "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAZz5Aj1MiJjrOWJdCifYJkr0pCrOIu2jigmd9BzuhZrO",
+            "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAP8AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA/y91DA3ZRyFzmc7L/ZXrxJ96/2ZDIr/JH0tyRmtrDtHA",
+            "AAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUUAAAAAAAAAAAAAAAAAAAAAAAAAAAAFhMtTiEs5RU6k2enIm7ThjQyCL82hSxpinyCELhed9QK+p9ZbNIDe",
+            "AAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUUAAAAAAAAAAAAAAAAAAAAAAAAAAAAFhMtTiEs5RT0T52LWq3L0FNM6KqQKA7NrxFNM8zC/kQKHkPJXMibY",
+            "/////////////////////////////////////////////////////////////////////////////76AU8rJ4VSYVAsPH5LGqQ2KG/oKPajw+kyhnQkq5Vch"
+          ]
+        }
+      ],
+      "namespace_id": "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABA==",
+      "row_proof": {
+        "row_roots": [
+          "00000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000584CB53884B394593052DF4039E58C7D51F0E45CACE7DD584125F62F9261B7900AC1EEEF5E82349"
+        ],
+        "proofs": [
+          {
+            "total": "128",
+            "index": "0",
+            "leaf_hash": "Qm/3wL9cWxS8rQbDDgUPU9p8jCfJ+Jc77zsSWdcF6PM=",
+            "aunts": [
+              "+mHKrh9boRgjN3oqt/Np5Wre24w+E79/hzFY/eouJY0=",
+              "8woyobNJs3MT1dKFRQID8VC75oJa2jNF3Wn/1USGfT4=",
+              "MTgbZqhrQoL61oKKZMhRfYq5bk6gOkLgWrVPArPYXvE=",
+              "plW1GXaBNavHWwurqsWB0xH25zv9xhiELqtVld0XQC4=",
+              "K/yH2ZDYNE9u/UT8sJtGuH+akiMNQTKUUu/uhlbGdgo=",
+              "J2pYcLT4KHpIQvh7b6Wp9KCdMgHLCT9eDfYDr7ZAQ9o=",
+              "grZxooejIhch93+g3MLdiBq6fF+nrOAKRBgupfu8mUo="
+            ]
+          }
+        ],
+        "start_row": 0,
+        "end_row": 0
+      },
+      "namespace_version": 0
+    }
+  }
+}
{
+  "jsonrpc": "2.0",
+  "id": -1,
+  "result": {
+    "hash": "EF9F50BFB39F11B022A6CD7026574ECCDC6D596689BDCCC7B2C482A1B26B26B8",
+    "height": "1350632",
+    "index": 4,
+    "tx_result": {
+      "code": 0,
+      "data": "EioKKC9jZWxlc3RpYS5ibG9iLnYxLk1zZ1BheUZvckJsb2JzUmVzcG9uc2U=",
+      "log": "[{\\"msg_index\\":0,\\"events\\":[{\\"type\\":\\"celestia.blob.v1.EventPayForBlobs\\",\\"attributes\\":[{\\"key\\":\\"blob_sizes\\",\\"value\\":\\"[120000]\\"},{\\"key\\":\\"namespaces\\",\\"value\\":\\"[\\\\\\"AAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUU=\\\\\\"]\\"},{\\"key\\":\\"signer\\",\\"value\\":\\"\\\\\\"celestia1vdjkcetnw35kzvtk8pjhxcm4xan82wtvwcurwwtt0f6n2at9va6k2atjw3cn2umhxe58xmfndejs40vqs9\\\\\\"\\"}]},{\\"type\\":\\"message\\",\\"attributes\\":[{\\"key\\":\\"action\\",\\"value\\":\\"/celestia.blob.v1.MsgPayForBlobs\\"}]}]}]",
+      "info": "",
+      "gas_wanted": "1095604",
+      "gas_used": "1080694",
+      "events": [
+        ...
+      ],
+      "codespace": ""
+    },
+    "tx": "CqEBCp4BCiAvY2VsZXN0aWEuYmxvYi52MS5Nc2dQYXlGb3JCbG9icxJ6Ci9jZWxlc3RpYTF2OGVzY3U3ZnU5bHY4NzlrenU1dWVndWV1cnRxNXN3NmhzbTNuZRIdAAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUUaA8CpByIgsVXWya9toI+AyTu3JJA2wkI5ZLkm72/gklCGFLCSrm9CAQASaApSCkYKHy9jb3Ntb3MuY3J5cHRvLnNlY3AyNTZrMS5QdWJLZXkSIwohAqdvBVUpglaNDGTlOcGSoHERBAFsFBB5l0WdvBJjEsEHEgQKAggBGJjCARISCgwKBHV0aWESBDIxOTIQtO9CGkCJRjcOYijj81bttfb2GUdG7o8AuAwf0bscBhW9PPD99xpQ1slpemfyq0y1joJ/aRFgE6QNxuiZ18VLGlGEwtW/",
+    "proof": {
+      "data": [
+        "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQBAAACugAAACbaAgrNAgqfAQqcAQogL2NlbGVzdGlhLmJsb2IudjEuTXNnUGF5Rm9yQmxvYnMSeAovY2VsZXN0aWExYWxwNGZwbHF5d21jNmN1aDl5MzVlY2xycHF5cWF4MjN2Z3ZrczUSHQAAAAAAAAAAAAAAAAAAAAAAAAAAAAgICAgICAgIGgFeIiAzlOEQsGxg3rOw7SR1rkQ7dVJYGp3aXkaqy4oG6HYc0EIBABJnClIKRgofL2Nvc21vcy5jcnlwdG8uc2VjcDI1NmsxLlB1YktleRIjCiECA4ief8FZEaBQLVc2wOceFs+LhAK0mDnmPnsxYLkqv7QSBAoCCAEY5PEBEhEKCwoEdXRpYRIDMTYwELTvBBpA7XBbSGYFrwTZcFHq3va1vHtbRiCzYd0ELkAJo6kLSDooEQCwoVaGuwTdP55V8Btf3WC7/FEK44BOESwEwecTExICgQIaBElORFjcAgrQAgqhAQqeAQogL2NlbGVzdGlhLmJsb2IudjEuTXNnUGF5Rm9yQmxvYnMSegovY2VsZXN0aWExdjhlc2N1N2Z1OWx2ODc5a3p1NXVlZ3VldXJ0cTVzdzZoc20zbmUSHQAAAAAAAAAAAAAAAAAAAAAAAAAAAAWEy1OISzk=",
+        "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQAAAAAAEUaA8CpByIgsVXWya9toI+AyTu3JJA2wkI5ZLkm72/gklCGFLCSrm9CAQASaApSCkYKHy9jb3Ntb3MuY3J5cHRvLnNlY3AyNTZrMS5QdWJLZXkSIwohAqdvBVUpglaNDGTlOcGSoHERBAFsFBB5l0WdvBJjEsEHEgQKAggBGJjCARISCgwKBHV0aWESBDIxOTIQtO9CGkCJRjcOYijj81bttfb2GUdG7o8AuAwf0bscBhW9PPD99xpQ1slpemfyq0y1joJ/aRFgE6QNxuiZ18VLGlGEwtW/EgEIGgRJTkRYAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA="
+      ],
+      "share_proofs": [
+        {
+          "start": 5,
+          "end": 7,
+          "nodes": [
+            "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQs98k+8wQ2iX2BdcTfoHjtRQbqybtPdB1BUFY/D7WRs",
+            "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAZz5Aj1MiJjrOWJdCifYJkr0pCrOIu2jigmd9BzuhZrO",
+            "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAP8AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA/y91DA3ZRyFzmc7L/ZXrxJ96/2ZDIr/JH0tyRmtrDtHA",
+            "AAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUUAAAAAAAAAAAAAAAAAAAAAAAAAAAAFhMtTiEs5RU6k2enIm7ThjQyCL82hSxpinyCELhed9QK+p9ZbNIDe",
+            "AAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUUAAAAAAAAAAAAAAAAAAAAAAAAAAAAFhMtTiEs5RT0T52LWq3L0FNM6KqQKA7NrxFNM8zC/kQKHkPJXMibY",
+            "/////////////////////////////////////////////////////////////////////////////76AU8rJ4VSYVAsPH5LGqQ2KG/oKPajw+kyhnQkq5Vch"
+          ]
+        }
+      ],
+      "namespace_id": "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABA==",
+      "row_proof": {
+        "row_roots": [
+          "00000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000584CB53884B394593052DF4039E58C7D51F0E45CACE7DD584125F62F9261B7900AC1EEEF5E82349"
+        ],
+        "proofs": [
+          {
+            "total": "128",
+            "index": "0",
+            "leaf_hash": "Qm/3wL9cWxS8rQbDDgUPU9p8jCfJ+Jc77zsSWdcF6PM=",
+            "aunts": [
+              "+mHKrh9boRgjN3oqt/Np5Wre24w+E79/hzFY/eouJY0=",
+              "8woyobNJs3MT1dKFRQID8VC75oJa2jNF3Wn/1USGfT4=",
+              "MTgbZqhrQoL61oKKZMhRfYq5bk6gOkLgWrVPArPYXvE=",
+              "plW1GXaBNavHWwurqsWB0xH25zv9xhiELqtVld0XQC4=",
+              "K/yH2ZDYNE9u/UT8sJtGuH+akiMNQTKUUu/uhlbGdgo=",
+              "J2pYcLT4KHpIQvh7b6Wp9KCdMgHLCT9eDfYDr7ZAQ9o=",
+              "grZxooejIhch93+g3MLdiBq6fF+nrOAKRBgupfu8mUo="
+            ]
+          }
+        ],
+        "start_row": 0,
+        "end_row": 0
+      },
+      "namespace_version": 0
+    }
+  }
+}

The proof field contains the transaction inclusion proof to the data root.

Also, the share range where this transaction spans is the end exclusive range defined by proof.share_proofs[0].start and proof.share_proofs[0].end.

NOTE: The values are base64 encoded. For these to be usable with the solidity smart contract, they need to be converted to bytes32. Check the next section for more information.

Golang client

Using the golang client:

go
    txHash, err := hex.DecodeString("<transaction_hash>")
+    if err != nil {
+		...
+	}
+    tx, err := trpc.Tx(cmd.Context(), txHash, true)
+    if err != nil {
+		...
+    }
    txHash, err := hex.DecodeString("<transaction_hash>")
+    if err != nil {
+		...
+	}
+    tx, err := trpc.Tx(cmd.Context(), txHash, true)
+    if err != nil {
+		...
+    }

Then, the proof is under tx.Proof.

Blob inclusion proof using the corresponding PFB transaction hash

Currently, querying the proof of a blob, which contains the Rollup block data, using its corresponding PFB transaction hash is possible only using the golang client. Otherwise, the corresponding share range is required so that the ProveShares endpoint can be used.

Golang client

Using the golang client:

go
import (
+	"context"
+	"encoding/hex"
+	"github.com/celestiaorg/celestia-app/v2/pkg/appconsts"
+	"github.com/celestiaorg/go-square/square"
+	"github.com/tendermint/tendermint/rpc/client/http"
+)
+
+func queryShareRange() error {
+	txHash, err := hex.DecodeString("<transaction_hash>")
+	if err != nil {
+		return err
+	}
+	tx, err := trpc.Tx(context.Background(), txHash, true)
+	if err != nil {
+		return err
+	}
+	
+	blockRes, err := trpc.Block(context.Background(), &tx.Height)
+	if err != nil {
+		return err
+	}
+
+	version := blockRes.Block.Header.Version.App
+	maxSquareSize := appconsts.SquareSizeUpperBound(version)
+	subtreeRootThreshold := appconsts.SubtreeRootThreshold(version)
+	blobShareRange, err := square.BlobShareRange(
+		blockRes.Block.Txs.ToSliceOfBytes(),
+		int(tx.Index),
+		<blob_index>,
+		maxSquareSize,
+		subtreeRootThreshold,
+	)
+	if err != nil {
+		return err
+	}
+}
import (
+	"context"
+	"encoding/hex"
+	"github.com/celestiaorg/celestia-app/v2/pkg/appconsts"
+	"github.com/celestiaorg/go-square/square"
+	"github.com/tendermint/tendermint/rpc/client/http"
+)
+
+func queryShareRange() error {
+	txHash, err := hex.DecodeString("<transaction_hash>")
+	if err != nil {
+		return err
+	}
+	tx, err := trpc.Tx(context.Background(), txHash, true)
+	if err != nil {
+		return err
+	}
+	
+	blockRes, err := trpc.Block(context.Background(), &tx.Height)
+	if err != nil {
+		return err
+	}
+
+	version := blockRes.Block.Header.Version.App
+	maxSquareSize := appconsts.SquareSizeUpperBound(version)
+	subtreeRootThreshold := appconsts.SubtreeRootThreshold(version)
+	blobShareRange, err := square.BlobShareRange(
+		blockRes.Block.Txs.ToSliceOfBytes(),
+		int(tx.Index),
+		<blob_index>,
+		maxSquareSize,
+		subtreeRootThreshold,
+	)
+	if err != nil {
+		return err
+	}
+}

With the <transaction_hash> being the transaction hash of the PFB containing the blob and, the <blob_index> being the index of the blob. In fact, PayForBlob transactions can contain multiple blobs. So, the <blob_index> is the index of the blob in the PFB.

Specific share range inclusion proof

To retrieve the inclusion proof of a set of shares for whom the share range is already known, the ProveShares query can be used to query it.

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 share to row root NMT proofs and multiple row root to data root binary proofs.

HTTP request

Example request: <tendermint_rpc_endpoint>/prove_shares?height=15&startShare=0&endShare=1

Which queries the proof of shares [0,1) in block 15.

Example response:

json
{
+  "jsonrpc": "2.0",
+  "id": -1,
+  "result": {
+    "data": [
+      "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQBAAABXAAAACbaAgrOAgqgAQqdAQogL2NlbGVzdGlhLmJsb2IudjEuTXNnUGF5Rm9yQmxvYnMSeQovY2VsZXN0aWExdWc1ZWt0MmNjN250dzRkdG1zZDlsN3N0cTBzN3Z5ZTd5bTJyZHISHQAAAAAAAAAAAAAAAAAAAAAAAAASExIyQkMkMoiZGgKXAiIgrfloW1M/Y33zlD2luveDELZzr9cF92+2eTaImIWhN9pCAQASZwpQCkYKHy9jb3Ntb3MuY3J5cHRvLnNlY3AyNTZrMS5QdWJLZXkSIwohA36hewmW/AXtrw6S+QsNUzFGfeg37Da6igoP2ZQcK+04EgQKAggBGAISEwoNCgR1dGlhEgUyMTAwMBDQ6AwaQClYLQPNrFoD6H8mgmwxjFeNhwhRu39EcrVKMFkNQ8+HHuodhdOQIG/8DXEmrBwrpwj6hi+3uEsZ+0p5vrf3v8sSAQEaBElORFgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA="
+    ],
+    "share_proofs": [
+      {
+        "end": 1,
+        "nodes": [
+          "AAAAAAAAAAAAAAAAAAAAAAAAABITEjJCQyQyiJkAAAAAAAAAAAAAAAAAAAAAAAAAEhMSMkJDJDKImbiwnpOdwIZBFr0UiFhPKwGy/XIIjL+gqm0fqxIw0z0o",
+          "/////////////////////////////////////////////////////////////////////////////3+fuhlzUfKJnZD8yg/JOtZla2V3g2Q7y+18iH5j0Uxk"
+        ]
+      }
+    ],
+    "namespace_id": "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABA==",
+    "row_proof": {
+      "row_roots": [
+        "000000000000000000000000000000000000000000000000000000000400000000000000000000000000000000000000121312324243243288993946154604701154F739F3D1B5475786DDD960F06D8708D4E870DA6501C51750"
+      ],
+      "proofs": [
+        {
+          "total": "8",
+          "index": "0",
+          "leaf_hash": "300xzO8TiLwPNuREY6OJcRKzTHQ4y6yy6qH0wAuMMrc=",
+          "aunts": [
+            "ugp0sV9YNEI5pOiYR7RdOdswwlfBh2o3XiRsmMNmbKs=",
+            "3dMFZFaWZMTZVXhphF5TxlCJ+CT3EvmMFOpiXFH+ID4=",
+            "srl59GiTSiwC9LqdYASzFC6TvusyY7njX8/XThp6Xws="
+          ]
+        }
+      ],
+      "start_row": 0,
+      "end_row": 0
+    },
+    "namespace_version": 0
+  }
+}
{
+  "jsonrpc": "2.0",
+  "id": -1,
+  "result": {
+    "data": [
+      "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQBAAABXAAAACbaAgrOAgqgAQqdAQogL2NlbGVzdGlhLmJsb2IudjEuTXNnUGF5Rm9yQmxvYnMSeQovY2VsZXN0aWExdWc1ZWt0MmNjN250dzRkdG1zZDlsN3N0cTBzN3Z5ZTd5bTJyZHISHQAAAAAAAAAAAAAAAAAAAAAAAAASExIyQkMkMoiZGgKXAiIgrfloW1M/Y33zlD2luveDELZzr9cF92+2eTaImIWhN9pCAQASZwpQCkYKHy9jb3Ntb3MuY3J5cHRvLnNlY3AyNTZrMS5QdWJLZXkSIwohA36hewmW/AXtrw6S+QsNUzFGfeg37Da6igoP2ZQcK+04EgQKAggBGAISEwoNCgR1dGlhEgUyMTAwMBDQ6AwaQClYLQPNrFoD6H8mgmwxjFeNhwhRu39EcrVKMFkNQ8+HHuodhdOQIG/8DXEmrBwrpwj6hi+3uEsZ+0p5vrf3v8sSAQEaBElORFgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA="
+    ],
+    "share_proofs": [
+      {
+        "end": 1,
+        "nodes": [
+          "AAAAAAAAAAAAAAAAAAAAAAAAABITEjJCQyQyiJkAAAAAAAAAAAAAAAAAAAAAAAAAEhMSMkJDJDKImbiwnpOdwIZBFr0UiFhPKwGy/XIIjL+gqm0fqxIw0z0o",
+          "/////////////////////////////////////////////////////////////////////////////3+fuhlzUfKJnZD8yg/JOtZla2V3g2Q7y+18iH5j0Uxk"
+        ]
+      }
+    ],
+    "namespace_id": "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABA==",
+    "row_proof": {
+      "row_roots": [
+        "000000000000000000000000000000000000000000000000000000000400000000000000000000000000000000000000121312324243243288993946154604701154F739F3D1B5475786DDD960F06D8708D4E870DA6501C51750"
+      ],
+      "proofs": [
+        {
+          "total": "8",
+          "index": "0",
+          "leaf_hash": "300xzO8TiLwPNuREY6OJcRKzTHQ4y6yy6qH0wAuMMrc=",
+          "aunts": [
+            "ugp0sV9YNEI5pOiYR7RdOdswwlfBh2o3XiRsmMNmbKs=",
+            "3dMFZFaWZMTZVXhphF5TxlCJ+CT3EvmMFOpiXFH+ID4=",
+            "srl59GiTSiwC9LqdYASzFC6TvusyY7njX8/XThp6Xws="
+          ]
+        }
+      ],
+      "start_row": 0,
+      "end_row": 0
+    },
+    "namespace_version": 0
+  }
+}

NOTE: The values are base64 encoded. For these to be usable with the solidity smart contract, they need to be converted to bytes32. Check the next section for more information.

WARNING

As of Celestia-app v1.10.0, the prove_shares endpoint is being deprecated in favor of prove_shares_v2. Please use the new endpoint for the queries as the old one will be removed in upcoming releases.

Golang client

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 {
+		...
+	}

WARNING

As of Celestia-app v1.10.0, the ProveShares method is being deprecated in favor of ProveSharesV2. Please use the new method for the queries as the old one will be removed in upcoming releases.

Converting the proofs to be usable in the DAVerifier library

Smart contracts that use the DAVerifier library take 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 Blobstream contract.
+    AttestationProof attestationProof;
+}
+
+/// @notice Contains the necessary parameters needed to verify that a data root tuple
+/// was committed to, by the Blobstream 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 Blobstream contract.
+    AttestationProof attestationProof;
+}
+
+/// @notice Contains the necessary parameters needed to verify that a data root tuple
+/// was committed to, by the Blobstream 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 adapt the queried response above 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.

If the data field is retrieved from an HTTP request, it should be converted to hex before using abi.encode(...).

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 in case of an HTTP request, 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 localized.

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 contract:

solidity
/// @notice Contains the necessary parameters needed to verify that a data root tuple
+/// was committed to, by the Blobstream 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 Blobstream 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 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 Blobstream 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 data commitment stored events from the Blobstream contract and looking for the nonce attesting to the corresponding data.

Querying the proof's tupleRootNonce

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")
+	}
go
// Similar to BlobstreamX, but instead of importing the BlobstreamX contract,
+// import the SP1 Blobstream contract:
+import {
+    sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
+} 
+// and use the \`BlobstreamDataCommitmentStored\` event instead.
// Similar to BlobstreamX, but instead of importing the BlobstreamX contract,
+// import the SP1 Blobstream contract:
+import {
+    sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
+} 
+// and use the \`BlobstreamDataCommitmentStored\` event instead.

Listening for new data commitments

For listening for new data commitment stored 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)
+	    }
+    }
go
// Similar to BlobstreamX, but instead of importing the BlobstreamX contract,
+// import the SP1 Blobstream contract:
+import {
+    sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
+} 
+// and use the \`BlobstreamDataCommitmentStored\` event instead.
// Similar to BlobstreamX, but instead of importing the BlobstreamX contract,
+// import the SP1 Blobstream contract:
+import {
+    sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
+} 
+// and use the \`BlobstreamDataCommitmentStored\` event instead.

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
+	// Note: if you're using Celestia-app v1.10.0 onwards, please switch
+	// to \`trpc.ProveSharesV2\` as \`trpc.ProveShars\` is deprecated.
+	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
+	// Note: if you're using Celestia-app v1.10.0 onwards, please switch
+	// to \`trpc.ProveSharesV2\` as \`trpc.ProveShars\` is deprecated.
+	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,
+	}
+}
go
// Similar to BlobstreamX, but instead of importing the BlobstreamX contract,
+// import the SP1 Blobstream contract:
+import {
+    sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
+} 
+// and use the \`BlobstreamDataCommitmentStored\` event instead.
// Similar to BlobstreamX, but instead of importing the BlobstreamX contract,
+// import the SP1 Blobstream contract:
+import {
+    sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
+} 
+// and use the \`BlobstreamDataCommitmentStored\` event instead.

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

For an example BlobstreamX project that uses the above proof queries, checkout the blobstreamx-example sample project.

Learn more on the Lightlink docs.

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.

`,159),c=[t];function r(E,y,i,A,F,d){return n(),a("div",null,c)}const m=s(e,[["render",r]]);export{h as __pageData,m as default}; diff --git a/assets/developers_blobstream-proof-queries.md.34df4e6b.lean.js b/assets/developers_blobstream-proof-queries.md.34df4e6b.lean.js new file mode 100644 index 00000000000..f6c1dcec72f --- /dev/null +++ b/assets/developers_blobstream-proof-queries.md.34df4e6b.lean.js @@ -0,0 +1 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.1a91c06a.js";const o="/img/blobstream/blobstream-square.png",p="/img/blobstream/blobstream-commitment-diagram.png",h=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":1726496322000}'),e={name:"developers/blobstream-proof-queries.md"},t=l("",159),c=[t];function r(E,y,i,A,F,d){return n(),a("div",null,c)}const m=s(e,[["render",r]]);export{h as __pageData,m as default}; diff --git a/assets/developers_blobstream-rollups.md.7d43d4fb.js b/assets/developers_blobstream-rollups.md.7d43d4fb.js new file mode 100644 index 00000000000..211d4f7188d --- /dev/null +++ b/assets/developers_blobstream-rollups.md.7d43d4fb.js @@ -0,0 +1,3 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.1a91c06a.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.","prev":{"text":"Ethereum fallback mechanism","link":"/developers/ethereum-fallback"},"next":{"text":"Submitting data blobs to Celestia","link":"/developers/submit-data"},"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":1721851048000}'),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

This section will go over two constructs that can be used in building Blobstream rollups. Each with its pros and cons and the rollup developer can choose which one suits their needs better.

Note: Only the sequence of spans method can be used currently to build Blobstream rollups. The blob share commitment way still requires some tooling that will be built in the upcoming months.

Blob share commitment

The blob 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.

Blob share commitment: Proof details

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

  1. share inclusion proof to the blob 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 blob share commitment
  2. blob share commitment inclusion proof to the data root tuple root: meaning four merkle proofs:
    1. subtree roots merkle proofs to the blob 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 blob 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 blob share commitment has been posted to Celestia.

NOTE

Generating/verifying blob 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.

Blob share commitment: Compact proofs

There is a way to have compact proofs, when using blob 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 blob 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.

Blob share commitment: Pros

The pros of referencing rollup data using a blob 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.

Blob 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 blob 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 blob share commitment and a sequence of spans is that when using a blob share commitment, an extra merkle proof is needed to prove inclusion of the share to the blob 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 blob 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 blob share commitments

Another way to build a rollup is to replace the sequence of spans with a height and a blob 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 blob 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 blob 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 blob 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 blob 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 blob share commitments: Pros

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

Optimistic rollups that use blob 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 blob 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 blob 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 blob share commitments

To use blob 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 blob 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 blob share commitment is valid using the proofs laid out in the proof details of blob share commitment section.
  3. The zk-proof verifier would take as argument the data root and the blob share commitment. Then, inside the circuit, the protobuf encoded PFB transaction will be deserialized and then verify the following:
  • The deserialized blob 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 blob share commitments: Pros

None

Zk-rollups that use blob 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 blob 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 blob share commitment is valid using the proofs laid out in the blob share commitment proof details section.
  3. The zk-proof verifier would take as argument the data root and the blob share commitment. Then, inside the circuit:
  • It will verify that the input blob share commitment corresponds to the rollup data.
  • Verify that the input data root commits to that blob share commitment. Check the blob 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.

FAQ

Should I use the Celestia transaction hash to reference the rollup data?

This is asked a lot since it's the most intuitive way of referencing data. However, in Celestia, referencing the data using the transaction hash is not recommended.

A transaction proof in Celestia goes back to providing an inclusion proof of the shares containing the transaction. This means if the transaction hash is used to reference data in a Celestia block, the rollup verification mechanism should do the following:

  • Verify an inclusion proof of the shares comprising the transaction up to the data root tuple root
  • Decode those shares and parse the transaction, then hash its components to generate the transaction hash
  • Verify that the generated transaction hash matches the one used to reference the data

At this level, the transaction hash is authenticated and the verification contract has the shares of the transaction. Then, the verification contract needs to take the share commitment from the parsed transaction and follow the steps outlined in the blob share commitment section.

As observed, using the transaction hash is expensive and doesn't yield any advantages over using the blob share commitment, which in turn is more expensive than using the sequence of spans.

So, unless there are more reasons to use the transaction hash to reference the rollup data, the sequence of spans approach remains better.

`,113),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.7d43d4fb.lean.js b/assets/developers_blobstream-rollups.md.7d43d4fb.lean.js new file mode 100644 index 00000000000..96b016b2aeb --- /dev/null +++ b/assets/developers_blobstream-rollups.md.7d43d4fb.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.1a91c06a.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.","prev":{"text":"Ethereum fallback mechanism","link":"/developers/ethereum-fallback"},"next":{"text":"Submitting data blobs to Celestia","link":"/developers/submit-data"},"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":1721851048000}'),s={name:"developers/blobstream-rollups.md"},r=a("",113),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.31be9369.js b/assets/developers_blobstream-x-deploy.md.31be9369.js new file mode 100644 index 00000000000..891a149a0be --- /dev/null +++ b/assets/developers_blobstream-x-deploy.md.31be9369.js @@ -0,0 +1,173 @@ +import{_ as s,o as a,c as n,Q as e}from"./chunks/framework.1a91c06a.js";const u=JSON.parse('{"title":"New Blobstream X deployments","description":"","frontmatter":{"next":{"text":"Celestia-node key","link":"/developers/celestia-node-key"},"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":1721850820000}'),o={name:"developers/blobstream-x-deploy.md"},l=e(`

New Blobstream X deployments

This document will go over the instructions to deploy BlobstreamX to a new chain.

Deploying the contracts

To deploy a Blobstream X to a new chain, where a Gateway contract does not exist yet, the following steps need to be followed.

If any of the components already exist in the target chain, feel free to skip the corresponding step.

Deploy a new SuccinctGateway

The SuccinctGateway is a contract that acts as a registry for onchain circuit verifiers and manages their access control. It is the entrypoint for proof verification and does the following:

  1. Receive a PLONK proof from the prover
  2. Fetch the address of the target function verifier
  3. Verify if the prover is whitelisted, if whitelisting is enabled
  4. Forward the proof to the function verifier to be verified
  5. If the proof is valid, it calls back the BlobstreamX contract to update its state

The BlobstreamX requires the update to be provided through the SuccinctGateway. Otherwise, the contract can't be updated.

To deploy a SuccinctGateway contract, you need to have foundry installed. If not, refer to foundry documentation.

Then, clone the succinctx repo:

shell
git clone https://github.com/succinctlabs/succinctx
+cd succinctx
git clone https://github.com/succinctlabs/succinctx
+cd succinctx

Next, build the contracts:

shell
cd contracts
+forge build
cd contracts
+forge build

Then, setup the .env containing the required information for deployment. An example .env.example is provided to be inspired from.

Assume we're deploying to a chain which chainID is 12345. The .env should look like:

shell
# The salt to be used for CREATE2 deployments. It's a 32-byte string in hex format.
+# Example: 0x0000000000000000000000000000000000000000000000000000000000000001
+CREATE2_SALT=
+# The 'owner' of the contracts, recommended to be an EOA
+GUARDIAN=
+# The default prover to fullfill requests for Functions that have not opted for a different prover
+PROVER=
+# RPC URLs for each chain you want to deploy to
+RPC_12345=
+# Etherscan API keys for each chain you want to deploy to
+ETHERSCAN_API_KEY_12345=
+# Wallet type is the type of the wallet.
+# If you are using a ledger to sign the transaction,
+# set \`WALLET_TYPE=LEDGER\` and specify the \`MNEMONIC_INDEX\`
+# for which signer you want to use.
+# If you are using a private key,
+# set \`WALLET_TYPE=PRIVATE_KEY\` and specify the \`PRIVATE_KEY\`.
+# In this example, we're using private key.
+WALLET_TYPE=PRIVATE_KEY
+# The private key of the deployer account
+PRIVATE_KEY=
+# The address of the succinct fee vault contract. Set it to an example address
+# if you don't want to use a vault like: 0x0000000000000000000000000000000000000001
+SUCCINCT_FEE_VAULT_12345=
# The salt to be used for CREATE2 deployments. It's a 32-byte string in hex format.
+# Example: 0x0000000000000000000000000000000000000000000000000000000000000001
+CREATE2_SALT=
+# The 'owner' of the contracts, recommended to be an EOA
+GUARDIAN=
+# The default prover to fullfill requests for Functions that have not opted for a different prover
+PROVER=
+# RPC URLs for each chain you want to deploy to
+RPC_12345=
+# Etherscan API keys for each chain you want to deploy to
+ETHERSCAN_API_KEY_12345=
+# Wallet type is the type of the wallet.
+# If you are using a ledger to sign the transaction,
+# set \`WALLET_TYPE=LEDGER\` and specify the \`MNEMONIC_INDEX\`
+# for which signer you want to use.
+# If you are using a private key,
+# set \`WALLET_TYPE=PRIVATE_KEY\` and specify the \`PRIVATE_KEY\`.
+# In this example, we're using private key.
+WALLET_TYPE=PRIVATE_KEY
+# The private key of the deployer account
+PRIVATE_KEY=
+# The address of the succinct fee vault contract. Set it to an example address
+# if you don't want to use a vault like: 0x0000000000000000000000000000000000000001
+SUCCINCT_FEE_VAULT_12345=
  • PRIVATE_KEY: the private key of the account used to send the transaction.
  • CREATE2_SALT: the salt used to generate the gateway address. Using the same salt between deployments in different chains will make the gateways be on the same address.
  • GUARDIAN: the owner of the SuccinctGateway contract. It should be set to an account that will have full access to the gateway.
  • PROVER: the address of the account used to submit the proofs to the gateway. It is enabled by default for all the registered function verifiers.
  • RPC_12345: the RPC endpoint for the EVM chain whose chain ID is 12345. If the chain ID is different, make sure to change it in the environment variable name as well.
  • ETHERSCAN_API_KEY_12345: the Etherscan API key corresponding to the chain whose chain ID is 12345. Similar to RPC_12345, make sure to change the chain ID in the environment variable name if it's a different chain.

Then, save the environment to a .env file and run the following:

shell
./script/deploy.sh "SuccinctGateway" "12345 1234"
./script/deploy.sh "SuccinctGateway" "12345 1234"

with 12345 and 1234 being chainID that we want to deploy to, given that the corresponding environment variables, as specified above, are setup correctly.

Now the SuccinctGateway address should be printed on the terminal.

Deploy the function verifiers

The function verifiers are the onchain ciruit verifiers that take PLONK proofs as an input and verify them onchain.

The function verifiers can be either downloaded from the succinct platform or regenerated.

To download the function verifiers, along with the circuits binaries, check the list in the Succinct documentation and download the binaries and function verifiers corresponding to the circuits you want to verify.

Alternatively, you can generate them locally as specified in the regenerating the downloaded artifacts section.

After getting the FunctionVerifier.sol, either via downloading it or generating it locally, you can choose your favourite way to deploy it.

A simple way would be to copy the contract in the BlobstreamX repo and deploy it:

shell
cd $HOME
+git clone https://github.com/succinctlabs/blobstreamx
+cd blobstreamx/contracts
+
+forge install
+
+# copy the function verifier contract to the blobstream X contracts
+cp <path_to_function_verifier>/FunctionVerifier.sol src
+
+forge build
+
+forge create FunctionVerifier --rpc-url <rpc_url> --private-key <private_key> --verify --verifier etherscan --etherscan-api-key <etherscan_api_key>
+# ^ will return the address of the function verifier
cd $HOME
+git clone https://github.com/succinctlabs/blobstreamx
+cd blobstreamx/contracts
+
+forge install
+
+# copy the function verifier contract to the blobstream X contracts
+cp <path_to_function_verifier>/FunctionVerifier.sol src
+
+forge build
+
+forge create FunctionVerifier --rpc-url <rpc_url> --private-key <private_key> --verify --verifier etherscan --etherscan-api-key <etherscan_api_key>
+# ^ will return the address of the function verifier

Register the function verifier in the deployed SuccinctGateway

After deploying a function verifier for your target circuit, we should register it in a SuccinctGateway contract, either deployed by you in a previous step or pre-existing on the chain, to get a functionID and be able to use it to verify circuits.

A simple way to do it is to use cast tool from the Foundry toolset:

First, we will get the expected functionID by calling cast call which simulates the transaction execution and returns the return value:

shell
cast call <succinct_gateway_contract_address> "registerFunction(address,address,bytes32)(bytes32)" --rpc-url <rpc_url> <owner_of_the_function_verifier> <address_of_the_function_verifier> <32_bytes_of_salt> --private-key <private_key>
cast call <succinct_gateway_contract_address> "registerFunction(address,address,bytes32)(bytes32)" --rpc-url <rpc_url> <owner_of_the_function_verifier> <address_of_the_function_verifier> <32_bytes_of_salt> --private-key <private_key>

This will return a bytes32 which is the function ID.

Now, we will execute the transaction to register that returned functionID in the gateway, using the same parameters:

shell
cast send <succinct_gateway_contract_address> "registerFunction(address,address,bytes32)(bytes32)" --rpc-url <rpc_url> <owner_of_the_function_verifier> <address_of_the_function_verifier> <32_bytes_of_salt> --private-key <private_key>
cast send <succinct_gateway_contract_address> "registerFunction(address,address,bytes32)(bytes32)" --rpc-url <rpc_url> <owner_of_the_function_verifier> <address_of_the_function_verifier> <32_bytes_of_salt> --private-key <private_key>

To verify that the registration was successful, run:

shell
cast call <succinct_gateway_contract_address> "verifiers(bytes32)(address)" --rpc-url <rpc_url> <returned_function_ID>
cast call <succinct_gateway_contract_address> "verifiers(bytes32)(address)" --rpc-url <rpc_url> <returned_function_ID>

which will return the address of the function verifier that was deployed in the previous section.

NOTE: For BlobstreamX, there are always two function verifiers, corresponding to the two circuits: header_range and next_header. Make sure to register both the function verifiers corresponding to those circuits as we will use those function IDs to deploy the BlobstreamX contract in the next section.

Enable prover whitelisting

Now that the function verifier's contract is deployed and registered in the succinct gateway, we can define whitelisting rules for the proof submission.

by default, the whitelist status is set to Default. This means that only the default verifier, which was setup when deploying the SuccinctGateway. And if you want to restrict the list of provers that can submit proofs to your registered function verifier, you can set the whitelisting status of the function verifier and then add a custom prover. Or even allow for permissionlss submission.

Set Whitelist Status

Set the whitelist status of a functionID to:

  • 0: Default status, which only allows the default prover to submit proofs
  • 1: Custom status, which only allows a custom list of provers to submit the proofs. The custom list can be setup by calling the addCustomProver(bytes32 _functionId, address _prover) function.
  • 2: Disabled status, which allow for permissionless proof submission, i.e., any relayer can submit proofs to be verified by that function verifier.
shell
cast calldata "setWhitelistStatus(bytes32,uint8)" <YOUR_FUNCTION_ID> <WHITELIST_STATUS>
cast calldata "setWhitelistStatus(bytes32,uint8)" <YOUR_FUNCTION_ID> <WHITELIST_STATUS>

Add Custom Prover

Add a custom prover for a specific functionID.

shell
cast calldata "addCustomProver(bytes32,address)" <FUNCTION_ID> <CUSTOM_PROVER_ADDRESS>
cast calldata "addCustomProver(bytes32,address)" <FUNCTION_ID> <CUSTOM_PROVER_ADDRESS>

Deploy the BlobstreamX contract

To deploy a BlobstreamX contract, you need to have foundry installed. If not, refer to foundry documentation.

Then, clone the repository:

shell
cd $HOME
+git clone https://github.com/succinctlabs/blobstreamx
+cd blobstreamx
cd $HOME
+git clone https://github.com/succinctlabs/blobstreamx
+cd blobstreamx

Deploying the BlobstreamX contract requires initializing it with a trusted header.

In case we want to run a local prover, as is documented in the run a local prover section, the trusted header can be any header in the last three weeks of blocks, aka the unbonding period. Using an older header is also possible, but in case of a malicious RPC provider, the malicious validators won't be slashed.

Otherwise, we can run a proof replayer that will read the proofs from an existing deployment and submit them into the new one. This is documented under the run a proof replayer from an existing deployment section. In this case, the trusted header should correspond to a height already submitted to the existing deployment, and was the beginning of a batch. These heights can be found via querying the DataCommitmentStored events, or navigating to the events tab in Etherscan for the existing deployment, and choosing a startBlock from a certain event.

Querying the trusted hash

Given that we have a trusted height, we can query its corresponding trusted hash using:

shell
TENDERMINT_RPC_URL=<celestia_rpc_address> cargo run --bin genesis -- --block <trusted_height>
TENDERMINT_RPC_URL=<celestia_rpc_address> cargo run --bin genesis -- --block <trusted_height>

Alternatively, the trusted hash can be obtained via querying the <celestia_rpc>/block?height=<trusted_height> and getting the result.block_id.hash field.

Deployment instructions

Once we have the trusted hash, we can deploy the BlobstreamX contract. To do so, create a .env file containing the necessary variables. An example one can be found under contracts/.env.example:

shell
cd contracts
+cp .env.example .env
+vim .env # and fill the necessary environment variables
cd contracts
+cp .env.example .env
+vim .env # and fill the necessary environment variables

The needed environment variables are:

  • PRIVATE_KEY: the EVM private key to use to deploy the contract.
  • RPC_URL: the RPC endpoint of the chain where we want to deploy the BlobstreamX contract.
  • ETHERSCAN_API_KEY: the Etherscan API for the target chain.
  • CREATE2_SALT: salt for the CREATE2 method. Example: 0xaa.
  • GUARDIAN_ADDRESS: the BlobstreamX contract's owner. It allows it to update the functionIDs of the deployment, freeze the contract, upgrade it, and others.
  • GATEWAY_ADDRESS: the address of the SuccinctGateway deployment that will update the contract's state. As stated in the section above, the SuccinctGateway is the entrypoint that routes the proof to the function verifier, and if the proof is valid, it updates the BlobstreamX contract.
  • NEXT_HEADER_FUNCTION_ID: the functionID of the next header circuit verifier. It is obtained after deploying a function verifier, corresponding to the next header circuit, and registering it in the SuccinctGateway. More information in the function verifier registration section.
  • HEADER_RANGE_FUNCTION_ID: similar to NEXT_HEADER_FUNCTION_ID but for the header range circuit.
  • GENESIS_HEIGHT: is the height of the trusted header that we queried its trusted hash in the querying the trusted hash section. If the value starts with 0x, it will be parsed as a hex value. Otherwise, it will be treated as a decimal number.
  • GENESIS_HEADER: the queried trusted hash in the querying the trusted hash section.
  • DEPLOY: set to true since we're deploying the contract.
  • UPGRADE: set to false since we're not upgrading an existing deployment.
  • UPDATE_GENESIS_STATE: set to false since we're not updating the genesis state.
  • UPDATE_FUNCTION_IDS: set to false since we're not updating the function IDs.
  • UPDATE_GATEWAY: set to false since we're not updating the address of the gateway.
  • CONTRACT_ADDRESS: set to an empty value since we're not upgrading the deployment.

Save the .env file then run:

shell
forge install
+
+source .env
+
+forge script script/Deploy.s.sol --rpc-url $RPC_URL --private-key $PRIVATE_KEY --broadcast --verify --verifier etherscan --etherscan-api-key $ETHERSCAN_API_KEY
forge install
+
+source .env
+
+forge script script/Deploy.s.sol --rpc-url $RPC_URL --private-key $PRIVATE_KEY --broadcast --verify --verifier etherscan --etherscan-api-key $ETHERSCAN_API_KEY

And you should see the address printed in the logs.

Run a local prover

Now that the BlobstreamX contract is deployed, we can either run a local prover or opt for proofs replaying.

For running a local prover, make sure to have a beefy machine. It is recommended to run the prover in at least a 32CPU 256GB RAM machine to be able to generate proofs in time for a 1hr proofs submission frequency.

To run the prover:

shell
cd $HOME
+# in case you still did not clone the repository:
+# git clone https://github.com/succinctlabs/blobstreamx
+cd blobstreamx
+# create the environment for the prover
+cp .env.example .env
+# edit the environment file
+vim .env
cd $HOME
+# in case you still did not clone the repository:
+# git clone https://github.com/succinctlabs/blobstreamx
+cd blobstreamx
+# create the environment for the prover
+cp .env.example .env
+# edit the environment file
+vim .env

The following is the required environment for the prover:

  • PRIVATE_KEY: the EVM private key to use to submit the proofs.
  • RPC_URL: the RPC endpoint of the chain where the BlobstreamX contract is deployed.
  • TENDERMINT_RPC_URL: the Celestia chain RPC endpoint. Accepts a comma-separated list of RPC URLs for fail over.
  • CHAIN_ID: the target EVM chain ID.
  • CONTRACT_ADDRESS: the target BlobstreamX contract address.
  • NEXT_HEADER_FUNCTION_ID: the function ID of the next header function registered in the succinct gateway.
  • HEADER_RANGE_FUNCTION_ID: the function ID of the header range function registered in the succinct gateway.
  • LOOP_DELAY_MINS: the time to wait before sending the proofs in minutes. For example, for having a proof every 1hr, set it to 60.
  • LOCAL_PROVE_MODE: set to true to enable local proving, which is what these instructions are about.
  • LOCAL_RELAY_MODE: set to true to enable submitting the proofs onchain, which is what these instructions are about.
  • GATEWAY_ADDRESS: the address of the succinct gateway contract in the target chain.

For the circuits binaries, you can either generate them as specified in the build the circuits section, or download them from the Succinct platform. And then set the following environment variables:

  • PROVE_BINARY_0xFILL_IN_NEXT_HEADER_FUNCTION_ID: the next header circuit binary path. Make sure to also change the 0xFILL_IN_NEXT_HEADER_FUNCTION_ID in the environment variable name to the functionID of the corresponding function verifier.
  • PROVE_BINARY_0xFILL_IN_HEADER_RANGE_FUNCTION_ID: the header range circuit binary path. Make sure to also change the 0xFILL_IN_HEADER_RANGE_FUNCTION_ID in the environment variable name to the functionID of the corresponding function verifier.

Similarly, the verifier build can be generated as specified in the regeneration of the verifier build section, or downloaded from the succinct platform. And then set the following environment variable:

  • WRAPPER_BINARY: the path the wrapper verifier

After setting the environment, run:

shell
source .env
+cargo run --bin blobstreamx --release
source .env
+cargo run --bin blobstreamx --release

And you should see the operator's logs.

Run a proof replayer from an existing deployment

A cheaper alternative to running the prover locally is to deploy the BlobstreamX contract at a height that is followed by an existing BlobstreamX contract. Then, keep replaying the proofs from the existing deployment to the new one.

Check the blobstream-ops repository for more documentation.

Optional: Regenerating the downloaded artifacts

In case you want to generate verifier-build, which is the template solidity contract used to verify the PLONK proofs onchain, or the circuits build, which are used to generate the proofs, follow this section.

Otherwise, use the downloaded ones as specified in the previous section.

Regenerate the verifier-build

The verifier-build is the gnark wrapper circuit used by the plonky2x circuits for cheap on-chain verification. It outputs a PLONK proof.

It can be built using the following instructions:

shell
cd $HOME
+git clone https://github.com/succinctlabs/succinctx
+cd succinctx
+cargo test test_wrapper
+# ^ This should generate a data folder under plonky2x/verifier/data
+
+# Compile circuit
+cd plonky2x/verifier
+mkdir verifier-build
+go run . -compile -data verifier-build -circuit data/dummy
+
+# Compile executable
+CGO_ENABLED=0 go build -o verifier-build/verifier -ldflags "-s -w" .
+
+
+ls verifier-build/
+# ^ should have: pk.bin  r1cs.bin  verifier  Verifier.sol  vk.bin
cd $HOME
+git clone https://github.com/succinctlabs/succinctx
+cd succinctx
+cargo test test_wrapper
+# ^ This should generate a data folder under plonky2x/verifier/data
+
+# Compile circuit
+cd plonky2x/verifier
+mkdir verifier-build
+go run . -compile -data verifier-build -circuit data/dummy
+
+# Compile executable
+CGO_ENABLED=0 go build -o verifier-build/verifier -ldflags "-s -w" .
+
+
+ls verifier-build/
+# ^ should have: pk.bin  r1cs.bin  verifier  Verifier.sol  vk.bin

This should generate the Verifier.sol contract which is a template contract used to generate the final function verifiers based on the plonky2x circuits.

In the generated contract, only the circuit digest is specific to each circuit; the rest of the logic is universal. This is why a random circuit was used to create the template, indicated by the flag -circuit data/dummy. When we build a specific circuit, the Verifier.sol contract is fed to the circuit builder, which then replaces the circuit digest with the one corresponding to the actual circuit being built.

Also, you will see the verifier binary which is the PLONK wrapper that takes a plonky2x proof and wraps it into a PLONK proof.

During the build, the CRS string gets downloaded:

bash
2024/04/15 19:35:00 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript04.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript04.dat
+2024/04/15 19:35:05 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript05.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript05.dat
+2024/04/15 19:35:11 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript06.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript06.dat
+2024/04/15 19:35:41 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript11.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript11.dat
+2024/04/15 19:35:47 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript12.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript12.dat
+2024/04/15 19:35:53 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript13.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript13.dat
+2024/04/15 19:35:58 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript14.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript14.dat
+2024/04/15 19:36:03 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript15.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript15.dat
2024/04/15 19:35:00 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript04.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript04.dat
+2024/04/15 19:35:05 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript05.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript05.dat
+2024/04/15 19:35:11 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript06.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript06.dat
+2024/04/15 19:35:41 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript11.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript11.dat
+2024/04/15 19:35:47 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript12.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript12.dat
+2024/04/15 19:35:53 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript13.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript13.dat
+2024/04/15 19:35:58 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript14.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript14.dat
+2024/04/15 19:36:03 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript15.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript15.dat

This randomness is taken from the Aztec trusted setup.

NOTE: In a scaleway instance of 64CPU and 504G RAM, the build takes ~20 minutes to complete. Make sure to run it on a beefy machine.

Build the circuits and function verifiers

To build the circuits:

bash
cd $HOME
+git clone https://github.com/succinctlabs/blobstreamx
+cd blobstreamx
+# if a specific version is needed: git checkout <commit>
+# this assumes the verifier-build was generated in the previous section.
+# if not, it can be downloaded as specified in the first section.
+cp -r ../succinctx/plonky2x/verifier/verifier-build .
+# ^ This copies the verifier build to the blobstreamx folder
+mkdir -p build && RUST_LOG=debug cargo run --bin header_range_1024 --release build --wrapper-path verifier-build/ && mv ./target/release/header_range_1024 ./build/header_range_1024
+ls build
+# ^ should return something like: 0x007d0b2a2e2b013612e8.circuit  0x9039e58b2089e5f9abbb.circuit  0xce1636cfaf2bd5497c11.circuit  FunctionVerifier.sol  main.circuit 0x8e1ede4ce0865b41d714.circuit  0xa2140c9bde000dc5e21e.circuit  0xf6759ff933786ddacb92.circuit  header_range_1024
cd $HOME
+git clone https://github.com/succinctlabs/blobstreamx
+cd blobstreamx
+# if a specific version is needed: git checkout <commit>
+# this assumes the verifier-build was generated in the previous section.
+# if not, it can be downloaded as specified in the first section.
+cp -r ../succinctx/plonky2x/verifier/verifier-build .
+# ^ This copies the verifier build to the blobstreamx folder
+mkdir -p build && RUST_LOG=debug cargo run --bin header_range_1024 --release build --wrapper-path verifier-build/ && mv ./target/release/header_range_1024 ./build/header_range_1024
+ls build
+# ^ should return something like: 0x007d0b2a2e2b013612e8.circuit  0x9039e58b2089e5f9abbb.circuit  0xce1636cfaf2bd5497c11.circuit  FunctionVerifier.sol  main.circuit 0x8e1ede4ce0865b41d714.circuit  0xa2140c9bde000dc5e21e.circuit  0xf6759ff933786ddacb92.circuit  header_range_1024

The header_range_1024 is a specific circuit. Other circuit names can be used there. The current circuits that we have for BlobstreamX:

  • header_range_1024: skip function circuit for batches that are <= 1024 block.
  • header_range_2048: skip function circuit for batches that are <= 2048 block.
  • next_header: step function circuit.

All the deployments currently rely on two circuits: a header range circuit, either the 1024 or the 2048, depending on the frequency of the batches; The 1024 is mainly for batches that are at a ~1hr frequency and the 2048 for batches that are at a ~3-4hr frequency; and a next header circuit. So, if you're re-building the circuits, make sure to build the correct two circuits for your target deployment.

Now, if you check the build folder, you will find a file called FunctionVerifier.sol, which is the function verifier contract for your circuit that you can deploy on-chain.

Also, you will find the header_range_1024 under build folder which is the binary used to generate the proofs. It is mainly used by the operator to generate the plonky2x proofs that will be PLONK wrapped later using the generated verifier-build.

At this level, you can deploy the FunctionVerifier.sol onchain, then register it in the SuccinctGateway, then use the generated header_range_1024 circuit and the verifier to generate the proofs and submit them onchain. These steps are detailed in the previous section.

NOTE: In a scaleway instance of 64CPU and 504G RAM, the build takes ~10 minutes to complete. Make sure to run it on a beefy machine.

`,112),t=[l];function p(c,r,i,y,d,E){return a(),n("div",null,t)}const F=s(o,[["render",p]]);export{u as __pageData,F as default}; diff --git a/assets/developers_blobstream-x-deploy.md.31be9369.lean.js b/assets/developers_blobstream-x-deploy.md.31be9369.lean.js new file mode 100644 index 00000000000..7bedbf13b44 --- /dev/null +++ b/assets/developers_blobstream-x-deploy.md.31be9369.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as e}from"./chunks/framework.1a91c06a.js";const u=JSON.parse('{"title":"New Blobstream X deployments","description":"","frontmatter":{"next":{"text":"Celestia-node key","link":"/developers/celestia-node-key"},"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":1721850820000}'),o={name:"developers/blobstream-x-deploy.md"},l=e("",112),t=[l];function p(c,r,i,y,d,E){return a(),n("div",null,t)}const F=s(o,[["render",p]]);export{u as __pageData,F as default}; diff --git a/assets/developers_blobstream-x-requesting-data-commitment-ranges.md.729e2aa5.js b/assets/developers_blobstream-x-requesting-data-commitment-ranges.md.729e2aa5.js new file mode 100644 index 00000000000..6518b01e76c --- /dev/null +++ b/assets/developers_blobstream-x-requesting-data-commitment-ranges.md.729e2aa5.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.1a91c06a.js";const f=JSON.parse('{"title":"Requesting data commitment ranges","description":"","frontmatter":{"prev":{"text":"Overview of Blobstream X","link":"/developers/blobstreamx"},"head":[["meta",{"name":"og:title","content":"Requesting data commitment ranges | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstream-x-requesting-data-commitment-ranges.md","filePath":"developers/blobstream-x-requesting-data-commitment-ranges.md","lastUpdated":1726496322000}'),r={name:"developers/blobstream-x-requesting-data-commitment-ranges.md"},n=a('

Requesting data commitment ranges

By default, the Blobstream X deployments on Ethereum will be updated 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:

NOTE: The requested proof ranges cannot include blocks that were already used in a previous batch. The ranges should start from the last proven block, aka, latest_block and they should end in a block already committed by Celestia. In other words, it's the end-inclusive range defined by [latest_block, target_block] with target_block <= Celestia tip.

Local proving

To run the Blobstream X operator with local proving, follow this guide.

Local proving allows self-generating the proofs and submitting them to an existing BlobstreamX contract. Alternatively, if a team needs a very specific cadence that starts at very specific heights, they can deploy their own BlobstreamX contract and submit proofs to it. Deployment instructions can be found in the BlobstreamX deploy documentation.

TIP

Requires a large cloud machine to run in a reasonable amount of time. EC2 r6a.16xlarge, i.e., 64CPU 512GB RAM, takes ~30 minutes to generate a header range proof.

Request proofs from the Succinct platform

NOTE: Requesting a proof from the succinct platform requires having a Succinct API key. It can be requested using this form.

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.

Request proofs 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.

',17),s=[n];function c(i,l,d,h,p,m){return t(),o("div",null,s)}const b=e(r,[["render",c]]);export{f as __pageData,b as default}; diff --git a/assets/developers_blobstream-x-requesting-data-commitment-ranges.md.729e2aa5.lean.js b/assets/developers_blobstream-x-requesting-data-commitment-ranges.md.729e2aa5.lean.js new file mode 100644 index 00000000000..b744189fc7c --- /dev/null +++ b/assets/developers_blobstream-x-requesting-data-commitment-ranges.md.729e2aa5.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.1a91c06a.js";const f=JSON.parse('{"title":"Requesting data commitment ranges","description":"","frontmatter":{"prev":{"text":"Overview of Blobstream X","link":"/developers/blobstreamx"},"head":[["meta",{"name":"og:title","content":"Requesting data commitment ranges | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstream-x-requesting-data-commitment-ranges.md","filePath":"developers/blobstream-x-requesting-data-commitment-ranges.md","lastUpdated":1726496322000}'),r={name:"developers/blobstream-x-requesting-data-commitment-ranges.md"},n=a("",17),s=[n];function c(i,l,d,h,p,m){return t(),o("div",null,s)}const b=e(r,[["render",c]]);export{f as __pageData,b as default}; diff --git a/assets/developers_blobstream.md.16782cf1.js b/assets/developers_blobstream.md.16782cf1.js new file mode 100644 index 00000000000..15b5e0a9f29 --- /dev/null +++ b/assets/developers_blobstream.md.16782cf1.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as r}from"./chunks/framework.1a91c06a.js";const o="/img/blobstream/blobstream_logo.png",f=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":1725955964000}'),s={name:"developers/blobstream.md"},l=r('

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.

The latest implementation of Blobstream X is SP1 Blobstream, which is written in Rust for the SP1 zkVM. SP1 Blobstream offers improved performance and efficiency while maintaining the security guarantees of the original Blobstream X.

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

Implementations of Blobstream

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 SP1 Blobstream?

SP1 Blobstream is the latest implementation of Blobstream in Rust using the SP1 zkVM.

SP1 Blobstream is the latest 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. SP1 Blobstream uses 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 SP1 Blobstream Ethereum smart contract to update it with the latest Celestia header.

The SP1 Blobstream 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.

If you're looking to deploy SP1 blobstream to a new chain, see new Sp1 Blobstream deployments.

Learn more at the sp1-blobstream repo.

NOTE

The current Blobstream deployments all use SP1 Blobstream.

Integrate with SP1 Blobstream

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

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

The first deployments of SP1 Blobstream 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 contract that will include a new data commitment range that covers a 1-hour block range from the latestBlock in the contract. On Ethereum Mainnet, the contract will be updated every 4 hours.

How to integrate with Blobstream

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

Blobstream rollups

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

Deployed contracts

You can interact with the SP1 Blobstream contracts today. The SP1 Blobstream Solidity smart contracts are currently deployed on the following chains:

ContractEVM networkContract addressAttested data on CelestiaLink to Celenium
SP1 BlobstreamEthereum Mainnet0x7Cf3876F681Dbb6EdA8f6FfC45D66B996Df08fAeMainnet BetaDeployment on Celenium
SP1 BlobstreamArbitrum One0xA83ca7775Bc2889825BcDeDfFa5b758cf69e8794Mainnet BetaDeployment on Celenium
SP1 BlobstreamBase0xA83ca7775Bc2889825BcDeDfFa5b758cf69e8794Mainnet BetaDeployment on Celenium
SP1 BlobstreamSepolia0xf0c6429ebab2e7dc6e05dafb61128be21f13cb1eMocha testnetDeployment on Celenium
SP1 BlobstreamArbitrum Sepolia0xc3e209eb245Fd59c8586777b499d6A665DF3ABD2Mocha testnetDeployment on Celenium
SP1 BlobstreamBase Sepolia0xc3e209eb245Fd59c8586777b499d6A665DF3ABD2Mocha testnetDeployment on Celenium
',39),i=[l];function n(h,c,m,b,d,p){return t(),a("div",null,i)}const g=e(s,[["render",n]]);export{f as __pageData,g as default}; diff --git a/assets/developers_blobstream.md.16782cf1.lean.js b/assets/developers_blobstream.md.16782cf1.lean.js new file mode 100644 index 00000000000..87e2ecac72d --- /dev/null +++ b/assets/developers_blobstream.md.16782cf1.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as r}from"./chunks/framework.1a91c06a.js";const o="/img/blobstream/blobstream_logo.png",f=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":1725955964000}'),s={name:"developers/blobstream.md"},l=r("",39),i=[l];function n(h,c,m,b,d,p){return t(),a("div",null,i)}const g=e(s,[["render",n]]);export{f as __pageData,g as default}; diff --git a/assets/developers_blobstreamx.md.fad0ad62.js b/assets/developers_blobstreamx.md.fad0ad62.js new file mode 100644 index 00000000000..84288a8f134 --- /dev/null +++ b/assets/developers_blobstreamx.md.fad0ad62.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.1a91c06a.js";const r="/img/blobstream/Celestia_Blobstream_X1b.png",s="/img/blobstream/Celestia_Blobstream_X2b.png",g=JSON.parse('{"title":"Blobstream X: the previous zk implementation of Blobstream","description":"What is BlobstreamX","frontmatter":{"description":"What is BlobstreamX","prev":{"text":"New SP1 Blobstream deployments","link":"/developers/sp1-blobstream-deploy"},"next":{"text":"Requesting data commitment ranges","link":"/developers/blobstream-x-requesting-data-commitment-ranges"},"head":[["meta",{"name":"og:title","content":"Blobstream X: the previous zk implementation of Blobstream | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstreamx.md","filePath":"developers/blobstreamx.md","lastUpdated":1726491136000}'),c={name:"developers/blobstreamx.md"},l=a('

Blobstream X: the previous zk implementation of Blobstream

blobstream x draft diagram

What is Blobstream X?

Blobstream X is the previous implementation of Blobstream. It uses plonky2x to create circuits that verify the Celestia consensus and generate the corresponding proofs.

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

NOTE

The Blobstream deployments below don't use the BlobstreamX circuits.

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

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.

Deploy Blobstream X

It is possible to deploy and maintain a Blobstream x instance and have the same security guarantees.

First, you will need to create a multisig that governs the Blobstream X contract and also the function identifiers. The function identifiers can be registered in the Succinct gateway.

Then, check the deployment documentation for how to deploy the contract.

Then, you will need to run a relayer, which will generate the proofs and relay them to your deployed Blobstream X contract. Check the local proving documentation for more information.

',18),n=[l];function i(m,b,d,h,p,u){return t(),o("div",null,n)}const B=e(c,[["render",i]]);export{g as __pageData,B as default}; diff --git a/assets/developers_blobstreamx.md.fad0ad62.lean.js b/assets/developers_blobstreamx.md.fad0ad62.lean.js new file mode 100644 index 00000000000..aa0f57e915b --- /dev/null +++ b/assets/developers_blobstreamx.md.fad0ad62.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.1a91c06a.js";const r="/img/blobstream/Celestia_Blobstream_X1b.png",s="/img/blobstream/Celestia_Blobstream_X2b.png",g=JSON.parse('{"title":"Blobstream X: the previous zk implementation of Blobstream","description":"What is BlobstreamX","frontmatter":{"description":"What is BlobstreamX","prev":{"text":"New SP1 Blobstream deployments","link":"/developers/sp1-blobstream-deploy"},"next":{"text":"Requesting data commitment ranges","link":"/developers/blobstream-x-requesting-data-commitment-ranges"},"head":[["meta",{"name":"og:title","content":"Blobstream X: the previous zk implementation of Blobstream | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/blobstreamx.md","filePath":"developers/blobstreamx.md","lastUpdated":1726491136000}'),c={name:"developers/blobstreamx.md"},l=a("",18),n=[l];function i(m,b,d,h,p,u){return t(),o("div",null,n)}const B=e(c,[["render",i]]);export{g as __pageData,B as default}; diff --git a/assets/developers_bubs-testnet.md.bc7bccdc.js b/assets/developers_bubs-testnet.md.bc7bccdc.js new file mode 100644 index 00000000000..ca2a8c5c297 --- /dev/null +++ b/assets/developers_bubs-testnet.md.bc7bccdc.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as s}from"./chunks/framework.1a91c06a.js";const r="/img/Celestia_Bubs_Testnet.jpg",m=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.","next":{"text":"Ethereum fallback mechanism","link":"/developers/ethereum-fallback"},"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":1726491136000}'),o={name:"developers/bubs-testnet.md"},i=s('

Bubs testnet

Bubs testnet

Bubs Testnet is a the first OP Stack testnet with Celestia underneath hosted by Caldera with support from Celestia Labs. Bubs 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. Bubs' data is posted to Celestia on the Mocha testnet. View the namespace for Bubs on Celestia's Mocha testnet.

Learn more about the setup of the integration in the introduction.

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.

',29),n=[i];function l(h,c,d,p,u,b){return t(),a("div",null,n)}const g=e(o,[["render",l]]);export{m as __pageData,g as default}; diff --git a/assets/developers_bubs-testnet.md.bc7bccdc.lean.js b/assets/developers_bubs-testnet.md.bc7bccdc.lean.js new file mode 100644 index 00000000000..c309ed88ddd --- /dev/null +++ b/assets/developers_bubs-testnet.md.bc7bccdc.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as s}from"./chunks/framework.1a91c06a.js";const r="/img/Celestia_Bubs_Testnet.jpg",m=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.","next":{"text":"Ethereum fallback mechanism","link":"/developers/ethereum-fallback"},"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":1726491136000}'),o={name:"developers/bubs-testnet.md"},i=s("",29),n=[i];function l(h,c,d,p,u,b){return t(),a("div",null,n)}const g=e(o,[["render",l]]);export{m as __pageData,g as default}; diff --git a/assets/developers_build-whatever.md.4b1651ac.js b/assets/developers_build-whatever.md.4b1651ac.js new file mode 100644 index 00000000000..3037b032d18 --- /dev/null +++ b/assets/developers_build-whatever.md.4b1651ac.js @@ -0,0 +1 @@ +import{_ as d,o as l,c as r,k as t,t as i,e as p,H as e,Q as s,a as n}from"./chunks/framework.1a91c06a.js";const h="/img/da-and-validity.png";const m={name:"UrlImageButton",props:{url:String,imageSrc:String,text:String,notes:{type:String,default:""}}},b=["href"],g={class:"url-image-button"},f=["src"],y={class:"button-text"},k={key:0,class:"notes-text"};function _(c,u,o,B,T,z){return l(),r("a",{href:o.url,target:"_blank",rel:"noopener noreferrer",class:"button-link"},[t("button",g,[t("img",{src:o.imageSrc,alt:"",class:"button-icon"},null,8,f),t("span",y,i(o.text),1),o.notes?(l(),r("span",k,i(o.notes),1)):p("",!0)])],8,b)}const a=d(m,[["render",_],["__scopeId","data-v-21baf7e3"]]),w=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"}},V=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":1726228958000}'),O={name:"developers/build-whatever.md"},I=Object.assign(O,{setup(c){return(u,o)=>(l(),r("div",null,[w,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:"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/",imageSrc:"/build/dymension.webp",text:"Dymension",alt:"Dymension logo","aria-label":"Dymension"}),e(a,{url:"https://docs.stf.xyz",imageSrc:"/build/stackr.webp",text:"Stackr",alt:"Stackr logo","aria-label":"Stackr"}),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",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://www.gelato.network/",imageSrc:"/build/gelato.webp",text:"Gelato",notes:"Orbit, OP Stack",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://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",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"})]),V]))}});export{R as __pageData,I as default}; diff --git a/assets/developers_build-whatever.md.4b1651ac.lean.js b/assets/developers_build-whatever.md.4b1651ac.lean.js new file mode 100644 index 00000000000..cb443ea16c4 --- /dev/null +++ b/assets/developers_build-whatever.md.4b1651ac.lean.js @@ -0,0 +1 @@ +import{_ as d,o as l,c as r,k as t,t as i,e as p,H as e,Q as s,a as n}from"./chunks/framework.1a91c06a.js";const h="/img/da-and-validity.png";const m={name:"UrlImageButton",props:{url:String,imageSrc:String,text:String,notes:{type:String,default:""}}},b=["href"],g={class:"url-image-button"},f=["src"],y={class:"button-text"},k={key:0,class:"notes-text"};function _(c,u,o,B,T,z){return l(),r("a",{href:o.url,target:"_blank",rel:"noopener noreferrer",class:"button-link"},[t("button",g,[t("img",{src:o.imageSrc,alt:"",class:"button-icon"},null,8,f),t("span",y,i(o.text),1),o.notes?(l(),r("span",k,i(o.notes),1)):p("",!0)])],8,b)}const a=d(m,[["render",_],["__scopeId","data-v-21baf7e3"]]),w=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"}},V=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":1726228958000}'),O={name:"developers/build-whatever.md"},I=Object.assign(O,{setup(c){return(u,o)=>(l(),r("div",null,[w,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:"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/",imageSrc:"/build/dymension.webp",text:"Dymension",alt:"Dymension logo","aria-label":"Dymension"}),e(a,{url:"https://docs.stf.xyz",imageSrc:"/build/stackr.webp",text:"Stackr",alt:"Stackr logo","aria-label":"Stackr"}),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",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://www.gelato.network/",imageSrc:"/build/gelato.webp",text:"Gelato",notes:"Orbit, OP Stack",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://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",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"})]),V]))}});export{R as __pageData,I as default}; diff --git a/assets/developers_celestia-node-key.md.23b7816e.js b/assets/developers_celestia-node-key.md.23b7816e.js new file mode 100644 index 00000000000..52b431390d0 --- /dev/null +++ b/assets/developers_celestia-node-key.md.23b7816e.js @@ -0,0 +1,63 @@ +import{m as o}from"./chunks/mocha_versions.7704b055.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.1a91c06a.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:

  • --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 default directory under /keys so you should make sure to point cel-key utility to the correct directory via the p2p.network or home flags if you have specified a custom directory or network other than Arabica, Mocha, or Mainnet Beta.

Also keep in mind that if you do not specify a network with --p2p.network, the default one will always be celestia (Mainnet Beta).

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.keyname 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.keyname 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),I=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":"New Blobstream X deployments","link":"/developers/blobstream-x-deploy"},"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":1724432671000}'),w={name:"developers/celestia-node-key.md"},S=Object.assign(w,{setup(A){return(f,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{I as __pageData,S as default}; diff --git a/assets/developers_celestia-node-key.md.23b7816e.lean.js b/assets/developers_celestia-node-key.md.23b7816e.lean.js new file mode 100644 index 00000000000..7c412fe9918 --- /dev/null +++ b/assets/developers_celestia-node-key.md.23b7816e.lean.js @@ -0,0 +1,5 @@ +import{m as o}from"./chunks/mocha_versions.7704b055.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.1a91c06a.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),I=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":"New Blobstream X deployments","link":"/developers/blobstream-x-deploy"},"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":1724432671000}'),w={name:"developers/celestia-node-key.md"},S=Object.assign(w,{setup(A){return(f,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{I as __pageData,S as default}; diff --git a/assets/developers_ethereum-fallback.md.4892edb4.js b/assets/developers_ethereum-fallback.md.4892edb4.js new file mode 100644 index 00000000000..5054fb8f4fa --- /dev/null +++ b/assets/developers_ethereum-fallback.md.4892edb4.js @@ -0,0 +1 @@ +import{_ as a,o as l,c as r,k as e,a as t}from"./chunks/framework.1a91c06a.js";const s="/img/Celestia_ethereum-fallback.jpg",y=JSON.parse('{"title":"Ethereum fallback","description":"The DA fallback mechanism to Ethereum for rollups.","frontmatter":{"description":"The DA fallback mechanism to Ethereum for rollups.","next":{"text":"Blobstream rollups","link":"/developers/blobstream-rollups"},"prev":{"text":"Run an OP Stack devnet posting Celestia","link":"/developers/optimism"},"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":1722440867000}'),n={name:"developers/ethereum-fallback.md"},o=e("h1",{id:"ethereum-fallback",tabindex:"-1"},[t("Ethereum fallback "),e("a",{class:"header-anchor",href:"#ethereum-fallback","aria-label":'Permalink to "Ethereum fallback"'},"​")],-1),i=e("p",null,"Ethereum fallback is a 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:",-1),c=e("ul",null,[e("li",null,[e("a",{href:"./arbitrum-integration#ethereum-fallback-mechanism-in-nitro"},"Arbitrum Nitro")])],-1),h=e("p",null,"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.",-1),m=e("p",null,[t("Ethereum fallback is triggered whenever the sequencer has an error sending the "),e("code",null,"PayForBlobs"),t(" 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.")],-1),u=e("p",null,[e("img",{src:s,alt:"Ethereum fallback"})],-1),d=[o,i,c,h,m,u];function p(b,f,k,_,g,v){return l(),r("div",null,d)}const x=a(n,[["render",p]]);export{y as __pageData,x as default}; diff --git a/assets/developers_ethereum-fallback.md.4892edb4.lean.js b/assets/developers_ethereum-fallback.md.4892edb4.lean.js new file mode 100644 index 00000000000..5054fb8f4fa --- /dev/null +++ b/assets/developers_ethereum-fallback.md.4892edb4.lean.js @@ -0,0 +1 @@ +import{_ as a,o as l,c as r,k as e,a as t}from"./chunks/framework.1a91c06a.js";const s="/img/Celestia_ethereum-fallback.jpg",y=JSON.parse('{"title":"Ethereum fallback","description":"The DA fallback mechanism to Ethereum for rollups.","frontmatter":{"description":"The DA fallback mechanism to Ethereum for rollups.","next":{"text":"Blobstream rollups","link":"/developers/blobstream-rollups"},"prev":{"text":"Run an OP Stack devnet posting Celestia","link":"/developers/optimism"},"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":1722440867000}'),n={name:"developers/ethereum-fallback.md"},o=e("h1",{id:"ethereum-fallback",tabindex:"-1"},[t("Ethereum fallback "),e("a",{class:"header-anchor",href:"#ethereum-fallback","aria-label":'Permalink to "Ethereum fallback"'},"​")],-1),i=e("p",null,"Ethereum fallback is a 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:",-1),c=e("ul",null,[e("li",null,[e("a",{href:"./arbitrum-integration#ethereum-fallback-mechanism-in-nitro"},"Arbitrum Nitro")])],-1),h=e("p",null,"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.",-1),m=e("p",null,[t("Ethereum fallback is triggered whenever the sequencer has an error sending the "),e("code",null,"PayForBlobs"),t(" 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.")],-1),u=e("p",null,[e("img",{src:s,alt:"Ethereum fallback"})],-1),d=[o,i,c,h,m,u];function p(b,f,k,_,g,v){return l(),r("div",null,d)}const x=a(n,[["render",p]]);export{y as __pageData,x as default}; diff --git a/assets/developers_feegrant-for-blobs.md.5c24e85c.js b/assets/developers_feegrant-for-blobs.md.5c24e85c.js new file mode 100644 index 00000000000..aaba3d73003 --- /dev/null +++ b/assets/developers_feegrant-for-blobs.md.5c24e85c.js @@ -0,0 +1,63 @@ +import{_ as s,o as a,c as n,Q as e}from"./chunks/framework.1a91c06a.js";const h=JSON.parse('{"title":"FeeGrant module for blobs submission","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"FeeGrant module for blobs submission | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/feegrant-for-blobs.md","filePath":"developers/feegrant-for-blobs.md","lastUpdated":1725871381000}'),o={name:"developers/feegrant-for-blobs.md"},l=e(`

FeeGrant module for blobs submission

Overview

This guide provides developers with the knowledge to use the FeeGrant module on the Celestia's Mocha testnet chain for granting a data availability node's account to submit blobs without constantly funding it, enabling a third-party account to cover the transaction fees.

Pre-requisites

  • celestia-node binary (celestia) installed
  • Access to a Mocha node (e.g., https://rpc.celestia-mocha.com:443)
  • Running DA Light node on Mocha testnet
  • One account with sufficient funds, the "granter"
  • One account with no funds, the "grantee"

Introduction

Each DA node contains a Celestia account that is used to pay for blobs submissions. To unify the fee payment process, the FeeGrant module allows a third-party account (granter) to pay for the fees incurred by a DA node's (grantee) account. You will need one account that will contain the funds, the granter, and another account that will be in the DA node you run to post blobs, the grantee. You will see the DA node's account once you initialize the node. Learn more about managing accounts with cel-key in create a wallet with celestia-node.

Granting fee allowances using celestia-node

To get started granting the fee allowance, you will need two separate keys to run the light node with. One to begin the FeeGrant as the granter and another to use the FeeGrant as the grantee.

Set some variables for your accounts for the remainder of the guide:

bash
export GRANTER_ADDRESS=<your-granter-account-address>
+export GRANTEE_ADDRESS=<your-grantee-account-address>
+export RPC_URL=rpc.celestia-mocha.com
export GRANTER_ADDRESS=<your-granter-account-address>
+export GRANTEE_ADDRESS=<your-grantee-account-address>
+export RPC_URL=rpc.celestia-mocha.com

FeeGrant module implementation in celestia-node

Using celestia-node, you can now easily give permission for other nodes to submit transactions on your behalf. It is also possible to revoke the grant.

The FeeGrant functionality can now be used during runtime without the need to restart the node.

Grant permission for an allowance as a granter

First, start your node:

bash
celestia light start --p2p.network mocha --core.ip $RPC_URL
celestia light start --p2p.network mocha --core.ip $RPC_URL

Then, grant the fee to the grantee:

bash
celestia state grant-fee $GRANTEE_ADDRESS --amount 2000
celestia state grant-fee $GRANTEE_ADDRESS --amount 2000

Note that the --amount flag specifies the spend limit (in utia) for the grantee. If not specified, the grantee does not have a spend limit.

Using a FeeGrant allowance as a grantee in celestia-node

Start your node:

bash
celestia light start --core.ip $RPC_URL --p2p.network=mocha
celestia light start --core.ip $RPC_URL --p2p.network=mocha

To check the balance of a light node, use the following command:

bash
celestia state balance
celestia state balance

Example response when the account balance does not exist:

json
{
+  "result": {
+    "denom": "utia",
+    "amount": "0"
+  }
+}
{
+  "result": {
+    "denom": "utia",
+    "amount": "0"
+  }
+}

This indicates that the light node currently does not have any funds.

Now submit a blob using the FeeGrant:

bash
celestia blob submit 0x42690c204d39600fddd3 'gm' --granter.address $GRANTER_ADDRESS
celestia blob submit 0x42690c204d39600fddd3 'gm' --granter.address $GRANTER_ADDRESS

You'll see the height and the commitment of your blob:

json
{
+  "result": {
+    "height": 1639397,
+    "commitments": [
+      "19L/C4iBEsqXGzC5ZxJ3vtuGBiAdQAMIEnbYjKEGcac="
+    ]
+  }
+}
{
+  "result": {
+    "height": 1639397,
+    "commitments": [
+      "19L/C4iBEsqXGzC5ZxJ3vtuGBiAdQAMIEnbYjKEGcac="
+    ]
+  }
+}

Checking account balances after submission

Light node account: After submitting a blob, you can check the light node account's balance to verify that the fees have been deducted:

bash
celestia state balance
celestia state balance

Example output showing fees are not deducted:

json
{
+  "result": {
+    "denom": "utia",
+    "amount": "0"
+  }
+}
{
+  "result": {
+    "denom": "utia",
+    "amount": "0"
+  }
+}

Optional: Revoke permission for a FeeGrant allowance as a granter

To revoke the feegrant, run:

bash
celestia state revoke-grant-fee $GRANTEE_ADDRESS
celestia state revoke-grant-fee $GRANTEE_ADDRESS

Optional: Submitting a blob from file input

To submit a blob from file input:

bash
celestia blob submit --input-file blob.json
celestia blob submit --input-file blob.json

Optional: Granting fee allowances using celestia-appd

To grant fee allowances, allowing a third-party (granter) account to pay for the fees incurred by a Celestia data availability node (grantee) account, use the following commands.

Set your account addresses for grantee and granter, and the RPC URL:

bash
export GRANTER_ADDRESS=<your-granter-account-address>
+export GRANTEE_ADDRESS=<your-grantee-account-address>
+export RPC_URL=https://rpc.celestia-mocha.com:443
export GRANTER_ADDRESS=<your-granter-account-address>
+export GRANTEE_ADDRESS=<your-grantee-account-address>
+export RPC_URL=https://rpc.celestia-mocha.com:443

Then, send the feegrant transaction:

bash
celestia-appd tx feegrant grant \\
+  $GRANTER_ADDRESS $GRANTEE_ADDRESS \\
+  --node $RPC_URL \\
+  --spend-limit 1000000utia \\
+  --allowed-messages "/cosmos.bank.v1beta1.MsgSend,/celestia.blob.v1.MsgPayForBlobs" \\
+  --chain-id mocha-4 \\
+  --keyring-backend test \\
+  --fees 20000utia \\
+  --broadcast-mode block \\
+  --yes
celestia-appd tx feegrant grant \\
+  $GRANTER_ADDRESS $GRANTEE_ADDRESS \\
+  --node $RPC_URL \\
+  --spend-limit 1000000utia \\
+  --allowed-messages "/cosmos.bank.v1beta1.MsgSend,/celestia.blob.v1.MsgPayForBlobs" \\
+  --chain-id mocha-4 \\
+  --keyring-backend test \\
+  --fees 20000utia \\
+  --broadcast-mode block \\
+  --yes

Example: FeeGrant transaction on Mocha

Optional: Checking the granter's account

To confirm that the fees have been deducted from the granter's account that granted the fee allowance, run:

bash
celestia-appd query bank balances $GRANTER_ADDRESS \\
+--node https://rpc.celestia-mocha.com:443 --denom utia
celestia-appd query bank balances $GRANTER_ADDRESS \\
+--node https://rpc.celestia-mocha.com:443 --denom utia

This output will show the remaining balance after fees have been deducted, confirming that the FeeGrant module is working as intended.

`,54),p=[l];function t(c,r,i,E,y,d){return a(),n("div",null,p)}const g=s(o,[["render",t]]);export{h as __pageData,g as default}; diff --git a/assets/developers_feegrant-for-blobs.md.5c24e85c.lean.js b/assets/developers_feegrant-for-blobs.md.5c24e85c.lean.js new file mode 100644 index 00000000000..6dd93125a6c --- /dev/null +++ b/assets/developers_feegrant-for-blobs.md.5c24e85c.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as e}from"./chunks/framework.1a91c06a.js";const h=JSON.parse('{"title":"FeeGrant module for blobs submission","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"FeeGrant module for blobs submission | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/feegrant-for-blobs.md","filePath":"developers/feegrant-for-blobs.md","lastUpdated":1725871381000}'),o={name:"developers/feegrant-for-blobs.md"},l=e("",54),p=[l];function t(c,r,i,E,y,d){return a(),n("div",null,p)}const g=s(o,[["render",t]]);export{h as __pageData,g as default}; diff --git a/assets/developers_golang-client-tutorial.md.6e7c47af.js b/assets/developers_golang-client-tutorial.md.6e7c47af.js new file mode 100644 index 00000000000..3cb4b3e18af --- /dev/null +++ b/assets/developers_golang-client-tutorial.md.6e7c47af.js @@ -0,0 +1,239 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.1a91c06a.js";const b=JSON.parse('{"title":"Golang client library tutorial","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Golang client library tutorial | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/golang-client-tutorial.md","filePath":"developers/golang-client-tutorial.md","lastUpdated":1724256441000}'),p={name:"developers/golang-client-tutorial.md"},o=l(`

Golang client library tutorial

This section tutorial will guide you through using the most common RPC endpoints with the golang client library.

Install dependencies and celestia-node if you have not already.

Project setup

To start, add celestia-openrpc as a dependency to your project:

bash
go get github.com/celestiaorg/celestia-openrpc
go get github.com/celestiaorg/celestia-openrpc

To use the following methods, you will need the node URL and your auth token. To get your auth token, see this guide. To run your node without an auth token, you can use the --rpc.skip-auth flag when starting your node. This allows you to pass an empty string as your auth token.

The default URL is http://localhost:26658. If you would like to use subscription methods, such as SubscribeHeaders below, you must use the ws protocol in place of http: ws://localhost:26658.

Submitting and retrieving blobs

The blob.Submit method takes a slice of blobs and a gas price, returning the height the blob was successfully posted at.

  • The namespace can be generated with share.NewBlobNamespaceV0.
  • The blobs can be generated with blob.NewBlobV0.
  • You can use blob.NewSubmitOptions(), which has celestia-node automatically determine an appropriate gas price. To set your own gas price, use blob.NewSubmitOptions().WithGasPrice(X). The available options are WithGasPrice, WithGas, WithKeyName, WithSignerAddress, and WithFeeGranterAddress.

The blob.GetAll method takes a height and slice of namespaces, returning the slice of blobs found in the given namespaces.

go
import (
+	"bytes"
+	"context"
+	"fmt"
+
+	client "github.com/celestiaorg/celestia-openrpc"
+	"github.com/celestiaorg/celestia-openrpc/types/blob"
+	"github.com/celestiaorg/celestia-openrpc/types/share"
+)
+
+// SubmitBlob submits a blob containing "Hello, World!" to the 0xDEADBEEF namespace. It uses the default signer on the running node.
+func SubmitBlob(ctx context.Context, url string, token string) error {
+	client, err := client.NewClient(ctx, url, token)
+	if err != nil {
+		return err
+	}
+
+	// let's post to 0xDEADBEEF namespace
+	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
+	if err != nil {
+		return err
+	}
+
+	// create a blob
+	helloWorldBlob, err := blob.NewBlobV0(namespace, []byte("Hello, World!"))
+	if err != nil {
+		return err
+	}
+
+	// submit the blob to the network
+	height, err := client.Blob.Submit(ctx, []*blob.Blob{helloWorldBlob}, blob.NewSubmitOptions())
+	if err != nil {
+		return err
+	}
+
+	fmt.Printf("Blob was included at height %d\\n", height)
+
+	// fetch the blob back from the network
+	retrievedBlobs, err := client.Blob.GetAll(ctx, height, []share.Namespace{namespace})
+	if err != nil {
+		return err
+	}
+
+	fmt.Printf("Blobs are equal? %v\\n", bytes.Equal(helloWorldBlob.Commitment, retrievedBlobs[0].Commitment))
+	return nil
+}
import (
+	"bytes"
+	"context"
+	"fmt"
+
+	client "github.com/celestiaorg/celestia-openrpc"
+	"github.com/celestiaorg/celestia-openrpc/types/blob"
+	"github.com/celestiaorg/celestia-openrpc/types/share"
+)
+
+// SubmitBlob submits a blob containing "Hello, World!" to the 0xDEADBEEF namespace. It uses the default signer on the running node.
+func SubmitBlob(ctx context.Context, url string, token string) error {
+	client, err := client.NewClient(ctx, url, token)
+	if err != nil {
+		return err
+	}
+
+	// let's post to 0xDEADBEEF namespace
+	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
+	if err != nil {
+		return err
+	}
+
+	// create a blob
+	helloWorldBlob, err := blob.NewBlobV0(namespace, []byte("Hello, World!"))
+	if err != nil {
+		return err
+	}
+
+	// submit the blob to the network
+	height, err := client.Blob.Submit(ctx, []*blob.Blob{helloWorldBlob}, blob.NewSubmitOptions())
+	if err != nil {
+		return err
+	}
+
+	fmt.Printf("Blob was included at height %d\\n", height)
+
+	// fetch the blob back from the network
+	retrievedBlobs, err := client.Blob.GetAll(ctx, height, []share.Namespace{namespace})
+	if err != nil {
+		return err
+	}
+
+	fmt.Printf("Blobs are equal? %v\\n", bytes.Equal(helloWorldBlob.Commitment, retrievedBlobs[0].Commitment))
+	return nil
+}

Subscribing to new blobs

You can subscribe to new blobs in a namespace using the blob.Subscribe method. This method returns a channel that will receive new blobs as they are produced. In this example, we will fetch all blobs in the 0xDEADBEEF namespace.

go
func SubscribeBlobs(ctx context.Context, url string, token string) error {
+	client, err := client.NewClient(ctx, url, token)
+	if err != nil {
+		return err
+	}
+
+	// create a namespace to filter blobs with
+	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
+	if err != nil {
+		return err
+	}
+
+	// subscribe to new blobs using a <-chan *blob.BlobResponse channel
+	blobChan, err := client.Blob.Subscribe(ctx)
+	if err != nil {
+		return err
+	}
+
+	for {
+		select {
+		case resp := <-blobChan:
+			fmt.Printf("Found %d blobs at height %d in 0xDEADBEEF namespace\\n", len(resp.Blobs()), resp.Height)
+		case <-ctx.Done():
+			return nil
+		}
+	}
+}
func SubscribeBlobs(ctx context.Context, url string, token string) error {
+	client, err := client.NewClient(ctx, url, token)
+	if err != nil {
+		return err
+	}
+
+	// create a namespace to filter blobs with
+	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
+	if err != nil {
+		return err
+	}
+
+	// subscribe to new blobs using a <-chan *blob.BlobResponse channel
+	blobChan, err := client.Blob.Subscribe(ctx)
+	if err != nil {
+		return err
+	}
+
+	for {
+		select {
+		case resp := <-blobChan:
+			fmt.Printf("Found %d blobs at height %d in 0xDEADBEEF namespace\\n", len(resp.Blobs()), resp.Height)
+		case <-ctx.Done():
+			return nil
+		}
+	}
+}

Subscribing to new headers

Alternatively, you can subscribe to new headers using the header.Subscribe method. This method returns a channel that will receive new headers as they are produced. In this example, we will fetch all blobs at the height of the new header in the 0xDEADBEEF namespace.

go
// SubscribeHeaders subscribes to new headers and fetches all blobs at the height of the new header in the 0xDEADBEEF namespace.
+func SubscribeHeaders(ctx context.Context, url string, token string) error {
+	client, err := client.NewClient(ctx, url, token)
+	if err != nil {
+		return err
+	}
+
+	// create a namespace to filter blobs with
+	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
+	if err != nil {
+		return err
+	}
+
+	// subscribe to new headers using a <-chan *header.ExtendedHeader channel
+	headerChan, err := client.Header.Subscribe(ctx)
+	if err != nil {
+		return err
+	}
+
+	for {
+		select {
+		case header := <-headerChan:
+			// fetch all blobs at the height of the new header
+			blobs, err := client.Blob.GetAll(context.TODO(), header.Height(), []share.Namespace{namespace})
+			if err != nil {
+				fmt.Printf("Error fetching blobs: %v\\n", err)
+			}
+
+			fmt.Printf("Found %d blobs at height %d in 0xDEADBEEF namespace\\n", len(blobs), header.Height())
+		case <-ctx.Done():
+			return nil
+		}
+	}
+}
// SubscribeHeaders subscribes to new headers and fetches all blobs at the height of the new header in the 0xDEADBEEF namespace.
+func SubscribeHeaders(ctx context.Context, url string, token string) error {
+	client, err := client.NewClient(ctx, url, token)
+	if err != nil {
+		return err
+	}
+
+	// create a namespace to filter blobs with
+	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
+	if err != nil {
+		return err
+	}
+
+	// subscribe to new headers using a <-chan *header.ExtendedHeader channel
+	headerChan, err := client.Header.Subscribe(ctx)
+	if err != nil {
+		return err
+	}
+
+	for {
+		select {
+		case header := <-headerChan:
+			// fetch all blobs at the height of the new header
+			blobs, err := client.Blob.GetAll(context.TODO(), header.Height(), []share.Namespace{namespace})
+			if err != nil {
+				fmt.Printf("Error fetching blobs: %v\\n", err)
+			}
+
+			fmt.Printf("Found %d blobs at height %d in 0xDEADBEEF namespace\\n", len(blobs), header.Height())
+		case <-ctx.Done():
+			return nil
+		}
+	}
+}

Fetching an Extended Data Square (EDS)

You can fetch an Extended Data Square (EDS) using the share.GetEDS method. This method takes a header and returns the EDS at the given height.

go
// GetEDS fetches the EDS at the given height.
+func GetEDS(ctx context.Context, url string, token string, height uint64) (*rsmt2d.ExtendedDataSquare, error) {
+	client, err := client.NewClient(ctx, url, token)
+	if err != nil {
+		return nil, err
+	}
+
+	// First get the header of the block you want to fetch the EDS from
+	header, err := client.Header.GetByHeight(ctx, height)
+	if err != nil {
+		return nil, err
+	}
+
+	// Fetch the EDS
+	return client.Share.GetEDS(ctx, header)
+}
// GetEDS fetches the EDS at the given height.
+func GetEDS(ctx context.Context, url string, token string, height uint64) (*rsmt2d.ExtendedDataSquare, error) {
+	client, err := client.NewClient(ctx, url, token)
+	if err != nil {
+		return nil, err
+	}
+
+	// First get the header of the block you want to fetch the EDS from
+	header, err := client.Header.GetByHeight(ctx, height)
+	if err != nil {
+		return nil, err
+	}
+
+	// Fetch the EDS
+	return client.Share.GetEDS(ctx, header)
+}

API documentation

To see the full list of available methods, see the API documentation.

`,24),e=[o];function t(r,c,E,y,i,F){return n(),a("div",null,e)}const u=s(p,[["render",t]]);export{b as __pageData,u as default}; diff --git a/assets/developers_golang-client-tutorial.md.6e7c47af.lean.js b/assets/developers_golang-client-tutorial.md.6e7c47af.lean.js new file mode 100644 index 00000000000..0dda3ee0ec6 --- /dev/null +++ b/assets/developers_golang-client-tutorial.md.6e7c47af.lean.js @@ -0,0 +1 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.1a91c06a.js";const b=JSON.parse('{"title":"Golang client library tutorial","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Golang client library tutorial | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/golang-client-tutorial.md","filePath":"developers/golang-client-tutorial.md","lastUpdated":1724256441000}'),p={name:"developers/golang-client-tutorial.md"},o=l("",24),e=[o];function t(r,c,E,y,i,F){return n(),a("div",null,e)}const u=s(p,[["render",t]]);export{b as __pageData,u as default}; diff --git a/assets/developers_integrate-celestia.md.6f9ae561.js b/assets/developers_integrate-celestia.md.6f9ae561.js new file mode 100644 index 00000000000..4f6fedc665f --- /dev/null +++ b/assets/developers_integrate-celestia.md.6f9ae561.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.1a91c06a.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":1718636946000}'),r={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),s=[n];function o(l,d,c,h,u,m){return t(),a("div",null,s)}const b=e(r,[["render",o]]);export{g as __pageData,b as default}; diff --git a/assets/developers_integrate-celestia.md.6f9ae561.lean.js b/assets/developers_integrate-celestia.md.6f9ae561.lean.js new file mode 100644 index 00000000000..5d343528540 --- /dev/null +++ b/assets/developers_integrate-celestia.md.6f9ae561.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as i}from"./chunks/framework.1a91c06a.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":1718636946000}'),r={name:"developers/integrate-celestia.md"},n=i("",21),s=[n];function o(l,d,c,h,u,m){return t(),a("div",null,s)}const b=e(r,[["render",o]]);export{g as __pageData,b as default}; diff --git a/assets/developers_intro-to-op-stack.md.60ea8345.js b/assets/developers_intro-to-op-stack.md.60ea8345.js new file mode 100644 index 00000000000..2aef94a08ee --- /dev/null +++ b/assets/developers_intro-to-op-stack.md.60ea8345.js @@ -0,0 +1 @@ +import{_ as t,o as e,c as a,Q as o}from"./chunks/framework.1a91c06a.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.","prev":{"text":"Bridging in and out of your Orbit rollup","link":"/developers/arbitrum-bridge"},"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":1726491136000}'),i={name:"developers/intro-to-op-stack.md"},r=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 availability (DA) network, built to enable anyone to easily deploy their own blockchain with minimal overhead.

Together, they allow developers to create rollups that post data to Celestia and settle on Ethereum.

About the integration

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

The integration of OP Stack with Celestia underneath for DA allows rollup operators to reduce overhead that is associated with posting data as calldata on Ethereum. Instead, op-batcher batches up rollup blocks and posts them to Celestia's DA network.

Data is managed 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. While previously op-node was reading from calldata on Ethereum, it now 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.

GitHub repository

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

WARNING

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

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.

',15),n=[r];function s(l,c,h,d,p,m){return e(),a("div",null,n)}const g=t(i,[["render",s]]);export{b as __pageData,g as default}; diff --git a/assets/developers_intro-to-op-stack.md.60ea8345.lean.js b/assets/developers_intro-to-op-stack.md.60ea8345.lean.js new file mode 100644 index 00000000000..392f3dc70a9 --- /dev/null +++ b/assets/developers_intro-to-op-stack.md.60ea8345.lean.js @@ -0,0 +1 @@ +import{_ as t,o as e,c as a,Q as o}from"./chunks/framework.1a91c06a.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.","prev":{"text":"Bridging in and out of your Orbit rollup","link":"/developers/arbitrum-bridge"},"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":1726491136000}'),i={name:"developers/intro-to-op-stack.md"},r=o("",15),n=[r];function s(l,c,h,d,p,m){return e(),a("div",null,n)}const g=t(i,[["render",s]]);export{b as __pageData,g as default}; diff --git a/assets/developers_multiaccounts.md.678a9ea9.js b/assets/developers_multiaccounts.md.678a9ea9.js new file mode 100644 index 00000000000..d73bbd19c99 --- /dev/null +++ b/assets/developers_multiaccounts.md.678a9ea9.js @@ -0,0 +1,5 @@ +import{_ as s,o as a,c as e,Q as n}from"./chunks/framework.1a91c06a.js";const b=JSON.parse('{"title":"MultiAccounts feature for blobs submission","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"MultiAccounts feature for blobs submission | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/multiaccounts.md","filePath":"developers/multiaccounts.md","lastUpdated":1725871430000}'),o={name:"developers/multiaccounts.md"},t=n(`

MultiAccounts feature for blobs submission

Overview

By default, a celestia-node creates a key named my_celes_key during initialization. This document explains how to run a node with a different default key name and how to submit blobs using different signers.

Running a node with a different default key name

To start a Celestia node with a different default key name, use the following command:

sh
celestia light start --core.ip=consensus.celestia-arabica-11.com \\
+    --p2p.network=arabica --keyring.keyname testKey
celestia light start --core.ip=consensus.celestia-arabica-11.com \\
+    --p2p.network=arabica --keyring.keyname testKey

In this example, testKey becomes the default node key, and the node's address will change accordingly.

Submitting blobs with a different signer/key name

Option 1: Submit passing key name

You can submit a blob by specifying a different key name:

sh
celestia blob submit 0x42690c204d39600fddd3 'gm' --key.name testKey2
celestia blob submit 0x42690c204d39600fddd3 'gm' --key.name testKey2

This transaction will be signed by the address associated with testKey2.

Option 2: Submit passing signer address

Alternatively, you can submit a blob by specifying the signer's address:

sh
celestia blob submit 0x42690c204d39600fddd3 'gm' --signer $SIGNER_ADDRESS
celestia blob submit 0x42690c204d39600fddd3 'gm' --signer $SIGNER_ADDRESS

Both options achieve the same result but use different inputs. The testKey2 points to SIGNER_ADDRESS in the KeyStore.

Key management

All keys and addresses must be added to the KeyStore. To create a new key, use the cel-key library:

Creating a new key

sh
./cel-key add testKey --keyring-backend test \\
+    --node.type light --p2p.network arabica
./cel-key add testKey --keyring-backend test \\
+    --node.type light --p2p.network arabica

Importing an existing key

sh
./cel-key import
./cel-key import

Learn more on the Create a wallet with celestia-node page.

Optional flags for write transactions

All other flags are now optional for all write transactions. This means you don't have to specify gas/fee parameters each time. The configuration can handle it for you automatically.

The default configuration applies to all write transactions, including those in the state module and blob.Submit. This simplifies the process of submitting transactions and reduces the need for manual input.

For reference, see the:

`,28),l=[t];function p(r,i,c,d,y,h){return a(),e("div",null,l)}const E=s(o,[["render",p]]);export{b as __pageData,E as default}; diff --git a/assets/developers_multiaccounts.md.678a9ea9.lean.js b/assets/developers_multiaccounts.md.678a9ea9.lean.js new file mode 100644 index 00000000000..dea32b69ba1 --- /dev/null +++ b/assets/developers_multiaccounts.md.678a9ea9.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as e,Q as n}from"./chunks/framework.1a91c06a.js";const b=JSON.parse('{"title":"MultiAccounts feature for blobs submission","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"MultiAccounts feature for blobs submission | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/multiaccounts.md","filePath":"developers/multiaccounts.md","lastUpdated":1725871430000}'),o={name:"developers/multiaccounts.md"},t=n("",28),l=[t];function p(r,i,c,d,y,h){return a(),e("div",null,l)}const E=s(o,[["render",p]]);export{b as __pageData,E as default}; diff --git a/assets/developers_node-api.md.bc539c55.js b/assets/developers_node-api.md.bc539c55.js new file mode 100644 index 00000000000..360c172f7fb --- /dev/null +++ b/assets/developers_node-api.md.bc539c55.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as r}from"./chunks/framework.1a91c06a.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.bc539c55.lean.js b/assets/developers_node-api.md.bc539c55.lean.js new file mode 100644 index 00000000000..9051bf55168 --- /dev/null +++ b/assets/developers_node-api.md.bc539c55.lean.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as r}from"./chunks/framework.1a91c06a.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.c48e1a3a.js b/assets/developers_node-tutorial.md.c48e1a3a.js new file mode 100644 index 00000000000..53b6e2146b8 --- /dev/null +++ b/assets/developers_node-tutorial.md.c48e1a3a.js @@ -0,0 +1,397 @@ +import{c as a}from"./chunks/constants.fa173a21.js";import{o as e,c as t,k as s,t as n,l as o,a as p,Q as l}from"./chunks/framework.1a91c06a.js";const c=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

Install dependencies and celestia-node if you have not already.

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 core endpoint

Let's now run the Celestia Light node with a gRPC connection to an example core endpoint. Connecting to a core endpoint provides the light node with access to state queries (reading balances, submitting transactions, and other state-related queries).

Note: You are also encouraged to find an RPC endpoint for Mainnet Beta, Mocha testnet, or Arabica devnet. If you are running a production application, use a production endpoint.

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.keyname <key-name>
celestia light start --core.ip <URI> --keyring.keyname <key-name>
bash
celestia light start --core.ip <URI> --keyring.keyname <key-name> \\
+  --p2p.network mocha
celestia light start --core.ip <URI> --keyring.keyname <key-name> \\
+  --p2p.network mocha
bash
celestia light start --core.ip <URI> --keyring.keyname <key-name> \\
+  --p2p.network arabica
celestia light start --core.ip <URI> --keyring.keyname <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:

:::note Previously, the node.store flag had to be specified manually for each request. This has changed in v0.14.0+ and you can read more about the implementation in celestia-node troubleshooting. :::

bash
celestia blob submit 0x42690c204d39600fddd3 'gm'
celestia blob submit 0x42690c204d39600fddd3 'gm'

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; however, none are necessary using default configurations.

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. This is only required if you are using a non-default node store path.

`,71),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={class:"line"},u=s("span",{style:{color:"#F97583"}},"export",-1),F=s("span",{style:{color:"#E1E4E8"}}," NODE_STORE",-1),h=s("span",{style:{color:"#F97583"}},"=",-1),C=s("span",{style:{color:"#E1E4E8"}},"$HOME",-1),g={style:{color:"#9ECBFF"}},b={class:"shiki github-light vp-code-light"},q={class:"line"},B=s("span",{style:{color:"#D73A49"}},"export",-1),m=s("span",{style:{color:"#24292E"}}," NODE_STORE",-1),v=s("span",{style:{color:"#D73A49"}},"=",-1),k=s("span",{style:{color:"#24292E"}},"$HOME",-1),A={style:{color:"#032F62"}},f=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 with a custom node store path, you will need to set the location of the node store in your auth command.

bash
--node.store $HOME/your-custom-path/.celestia-light-private
--node.store $HOME/your-custom-path/.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/your-custom-path/.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/your-custom-path/.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] [optional: node store | auth token]
celestia blob submit <hex-encoded namespace> <hex-encoded data> \\
+  [optional: fee] [optional: gasLimit] [optional: node store | auth token]

We run the following to submit a blob to the network in hexadecimal format:

bash
celestia blob submit 0x42690c204d39600fddd3 0x676d \\
celestia blob submit 0x42690c204d39600fddd3 0x676d \\

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'
celestia blob submit 0x42690c204d39600fddd3 'gm'

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),w=l(`
bash
celestia blob get 252614 0x42690c204d39600fddd3 IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E=
celestia blob get 252614 0x42690c204d39600fddd3 IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E=

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
celestia blob get 252614 0x42690c204d39600fddd3 IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E= \\
+  --base64=TRUE

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
celestia blob get-all 252614 0x42690c204d39600fddd3

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
celestia blob get-all 252614 0x42690c204d39600fddd3 \\
+  --base64=TRUE

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
celestia blob submit 0x42690c204d39600fddd3 'gm' --gas.price 0.004

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
celestia state balance

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>
celestia state balance-for-address <address>

Let's query our node for the balance of another address:

bash
celestia state balance-for-address celestia10rtd9lhel2cuh6c659l25yncl6atcyt37umard
celestia state balance-for-address celestia10rtd9lhel2cuh6c659l25yncl6atcyt37umard

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
celestia p2p info

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
celestia state account-address

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>
celestia header get-by-height <height>

Now, let's get the block header information.

Here we will get the header from Block 1:

bash
celestia header get-by-height 1
+\`\`\`
+
+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
+\`\`\`
+
+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 | jq '.result.dah' -r)" 0x42690c204d39600fddd3
celestia share get-by-namespace "$(celestia header get-by-height 147105 | jq '.result.dah' -r)" 0x42690c204d39600fddd3

Get data availability sampler stats

bash
celestia das sampling-stats
celestia das sampling-stats

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>
celestia state transfer $ADDRESS <amount in utia> <gas fee in utia> <gas fee in utia>

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
celestia state transfer $ADDRESS 100000 8000 80000

If you'd just like to return the transaction hash, you can use jq:

bash
celestia state transfer $ADDRESS 100000 8000 80000 | jq .result.txhash
celestia state transfer $ADDRESS 100000 8000 80000 | jq .result.txhash

API version

To query your node's API version, you can use the following command:

bash
celestia node info
celestia node info

Help

To get help and view the CLI menu, use the following command:

bash
celestia --help
celestia --help

To view the help menu for a specific method, use the following command:

bash
celestia <module> <method> --help
celestia <module> <method> --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 | jq '.result.dah' -r)" \\
+  0x42690c204d39600fddd3
celestia share get-by-namespace \\
+  "$(celestia header get-by-height 252614 | jq '.result.dah' -r)" \\
+  0x42690c204d39600fddd3

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.

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.

`,93),R=JSON.parse('{"title":"Celestia-node RPC CLI tutorial","description":"","frontmatter":{"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":1726693572000}'),D={name:"developers/node-tutorial.md"},S=Object.assign(D,{setup(_){return(x,T)=>(e(),t("div",null,[c,s("p",null,"To set a custom node store for a light node on "+n(o(a).mochaChainId)+", you can use the following command:",1),s("div",r,[y,i,s("pre",E,[s("code",null,[s("span",d,[u,F,h,C,s("span",g,"/your-custom-path/celestia-light-"+n(o(a).mochaChainId),1)])])]),s("pre",b,[s("code",null,[s("span",q,[B,m,v,k,s("span",A,"/your-custom-path/celestia-light-"+n(o(a).mochaChainId),1)])])])]),f,s("p",null,[p("Here is an example command to retrieve the data from above, on "),s("code",null,n(o(a).arabicaChainId),1),p(":")]),w]))}});export{R as __pageData,S as default}; diff --git a/assets/developers_node-tutorial.md.c48e1a3a.lean.js b/assets/developers_node-tutorial.md.c48e1a3a.lean.js new file mode 100644 index 00000000000..1bc710d54b6 --- /dev/null +++ b/assets/developers_node-tutorial.md.c48e1a3a.lean.js @@ -0,0 +1 @@ +import{c as a}from"./chunks/constants.fa173a21.js";import{o as e,c as t,k as s,t as n,l as o,a as p,Q as l}from"./chunks/framework.1a91c06a.js";const c=l("",71),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={class:"line"},u=s("span",{style:{color:"#F97583"}},"export",-1),F=s("span",{style:{color:"#E1E4E8"}}," NODE_STORE",-1),h=s("span",{style:{color:"#F97583"}},"=",-1),C=s("span",{style:{color:"#E1E4E8"}},"$HOME",-1),g={style:{color:"#9ECBFF"}},b={class:"shiki github-light vp-code-light"},q={class:"line"},B=s("span",{style:{color:"#D73A49"}},"export",-1),m=s("span",{style:{color:"#24292E"}}," NODE_STORE",-1),v=s("span",{style:{color:"#D73A49"}},"=",-1),k=s("span",{style:{color:"#24292E"}},"$HOME",-1),A={style:{color:"#032F62"}},f=l("",36),w=l("",93),R=JSON.parse('{"title":"Celestia-node RPC CLI tutorial","description":"","frontmatter":{"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":1726693572000}'),D={name:"developers/node-tutorial.md"},S=Object.assign(D,{setup(_){return(x,T)=>(e(),t("div",null,[c,s("p",null,"To set a custom node store for a light node on "+n(o(a).mochaChainId)+", you can use the following command:",1),s("div",r,[y,i,s("pre",E,[s("code",null,[s("span",d,[u,F,h,C,s("span",g,"/your-custom-path/celestia-light-"+n(o(a).mochaChainId),1)])])]),s("pre",b,[s("code",null,[s("span",q,[B,m,v,k,s("span",A,"/your-custom-path/celestia-light-"+n(o(a).mochaChainId),1)])])])]),f,s("p",null,[p("Here is an example command to retrieve the data from above, on "),s("code",null,n(o(a).arabicaChainId),1),p(":")]),w]))}});export{R as __pageData,S as default}; diff --git a/assets/developers_optimism-devnet.md.0a0a7e82.js b/assets/developers_optimism-devnet.md.0a0a7e82.js new file mode 100644 index 00000000000..c5ae452c40d --- /dev/null +++ b/assets/developers_optimism-devnet.md.0a0a7e82.js @@ -0,0 +1,75 @@ +import{_ as s,o as a,c as n,Q as e}from"./chunks/framework.1a91c06a.js";const f=JSON.parse('{"title":"Optimism devnet deep dive","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Optimism devnet deep dive | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/optimism-devnet.md","filePath":"developers/optimism-devnet.md","lastUpdated":1722440867000}'),l={name:"developers/optimism-devnet.md"},o=e(`

Optimism devnet deep dive

This page is for those interested in doing a deep dive on their pre-op-plasma-celestia @celestiaorg/optimism rollups.

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=3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f
+export ENCODED_INPUT=$(echo "$INPUT" | xxd -r -p | base64)
export INPUT=3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f
+export ENCODED_INPUT=$(echo "$INPUT" | xxd -r -p | base64)

TIP

Remember to remove the 0xce prefix!

Find the data on Celestia

bash
curl -s -X POST -H "Content-Type: application/json" -H "Authorization: Bearer $CELESTIA_NODE_AUTH_TOKEN" -d '{ "id": 1, "jsonrpc": "2.0", "method": "da.Get", "params": [["$ENCODED_INPUT"], "AAAAAAAAAAAAAAAAAAAAAAAAAAECAwQFBgcICRA="]}' http://127.0.0.1:26658
curl -s -X POST -H "Content-Type: application/json" -H "Authorization: Bearer $CELESTIA_NODE_AUTH_TOKEN" -d '{ "id": 1, "jsonrpc": "2.0", "method": "da.Get", "params": [["$ENCODED_INPUT"], "AAAAAAAAAAAAAAAAAAAAAAAAAAECAwQFBgcICRA="]}' http://127.0.0.1:26658

The params are []blobs, namespace, base64-encoded.

Your result will look similar to the below!

console
{"jsonrpc":"2.0","result":["SGVsbG8gd28ybGQh"],"id":1}
{"jsonrpc":"2.0","result":["SGVsbG8gd28ybGQh"],"id":1}

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.

`,26),p=[o];function c(t,r,d,i,y,b){return a(),n("div",null,p)}const h=s(l,[["render",c]]);export{f as __pageData,h as default}; diff --git a/assets/developers_optimism-devnet.md.0a0a7e82.lean.js b/assets/developers_optimism-devnet.md.0a0a7e82.lean.js new file mode 100644 index 00000000000..1e516ca486f --- /dev/null +++ b/assets/developers_optimism-devnet.md.0a0a7e82.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as e}from"./chunks/framework.1a91c06a.js";const f=JSON.parse('{"title":"Optimism devnet deep dive","description":"","frontmatter":{"head":[["meta",{"name":"og:title","content":"Optimism devnet deep dive | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/optimism-devnet.md","filePath":"developers/optimism-devnet.md","lastUpdated":1722440867000}'),l={name:"developers/optimism-devnet.md"},o=e("",26),p=[o];function c(t,r,d,i,y,b){return a(),n("div",null,p)}const h=s(l,[["render",c]]);export{f as __pageData,h as default}; diff --git a/assets/developers_optimism.md.a839aad4.js b/assets/developers_optimism.md.a839aad4.js new file mode 100644 index 00000000000..400e323395d --- /dev/null +++ b/assets/developers_optimism.md.a839aad4.js @@ -0,0 +1,45 @@ +import{c as a}from"./chunks/constants.fa173a21.js";import{o as t,c as l,k as s,t as e,l as n,Q as o}from"./chunks/framework.1a91c06a.js";const p=o('

Run an OP Stack rollup with Celestia underneath

This guide will show you how to run your own OP Stack devnet and testnet that posts data to Celestia's Mocha testnet using roll-op and op-plasma-celestia.

If you don't have devops experience and would like to use a Rollups as a Service (RaaS) provider, see the RaaS category in the menu.

Dependency setup

Setting up your light node

Sync and fund a Celestia light node. The light node must be fully synced and funded for you to be able to submit and retrieve PayForBlobs to Mocha Testnet. This allows your rollup to post and retrieve data without any errors.

In order to mount existing data, you must have a node store that is in the default directory:

',8),c={class:"vp-code-group vp-adaptive-theme"},r=o('
',1),i={class:"blocks"},d={class:"language-bash vp-adaptive-theme active"},u=s("button",{title:"Copy Code",class:"copy"},null,-1),h=s("span",{class:"lang"},"bash",-1),y={class:"shiki github-dark vp-code-dark"},_={class:"line"},E={style:{color:"#E1E4E8"}},g={class:"shiki github-light vp-code-light"},m={class:"line"},b={style:{color:"#24292E"}},v=o('
bash
$HOME/.celestia-light
$HOME/.celestia-light
',1),k={class:"language-bash vp-adaptive-theme"},C=s("button",{title:"Copy Code",class:"copy"},null,-1),q=s("span",{class:"lang"},"bash",-1),F={class:"shiki github-dark vp-code-dark"},f={class:"line"},A={style:{color:"#E1E4E8"}},S={class:"shiki github-light vp-code-light"},T={class:"line"},w={style:{color:"#24292E"}},D=o(`

By default, the node will run with the account named my_celes_key on Mocha. This is the account that needs to be funded.

TIP

Unless you changed your configuration, you won't have to change anything. 😎

Deploying a devnet to Mocha

See the Alt-DA x Celestia README for instructions on how to deploy a Devnet.

TIP for macOS users

If you are on macOS, you will need to run a venv before starting roll-op.

sh
cd $HOME/roll-op
+python3 -m venv ./venv
+source ./venv/bin/activate
cd $HOME/roll-op
+python3 -m venv ./venv
+source ./venv/bin/activate

Congrats! Your devnet is running on a mock EVM chain and Celestia Mocha.

Deploying a testnet to an L1 (or L2) and Mocha

See the Alt-DA x Celestia README for instructions on how to deploy a Testnet.

TIP

If you are using a public RPC for your EVM chain, you should to enable deploy_slowly = true in your config.toml. If you still have issues, we recommend running the integration with a high-availability, paid endpoint.

When you are deploying to a live EVM network, pay attention and modify the configuration to post to non-Sepolia EVM chains.

Here is an example:

toml
# Chain ID of your rollup
+l2_chain_id = 1117733 
+
+# Sepolia Ethereum
+l1_chain_id = 11155111
+l1_rpc_url = "https://ethereum-sepolia-rpc.publicnode.com"
+
+## Avoid issues with public RPC
+deploy_slowly = true
+
+## Keys
+contract_deployer_account = "0xaddress"
+contract_deployer_key = "privatekey"
+batcher_account = "0xaddress"
+batcher_key = "privatekey"
+proposer_account = "0xaddress"
+proposer_key = "privatekey"
+admin_account = "0xaddress"
+admin_key = "privatekey"
+p2p_sequencer_account = "0xaddress"
+p2p_sequencer_key = "privatekey"
# Chain ID of your rollup
+l2_chain_id = 1117733 
+
+# Sepolia Ethereum
+l1_chain_id = 11155111
+l1_rpc_url = "https://ethereum-sepolia-rpc.publicnode.com"
+
+## Avoid issues with public RPC
+deploy_slowly = true
+
+## Keys
+contract_deployer_account = "0xaddress"
+contract_deployer_key = "privatekey"
+batcher_account = "0xaddress"
+batcher_key = "privatekey"
+proposer_account = "0xaddress"
+proposer_key = "privatekey"
+admin_account = "0xaddress"
+admin_key = "privatekey"
+p2p_sequencer_account = "0xaddress"
+p2p_sequencer_key = "privatekey"

Your 0xaddress key must also be funded with testnet ETH. We recommend at least 10 SepoliaETH to get your chain started, but you will need more to keep it running longer.

Congratulations

Congrats! You now have an OP Stack rollup running with Celestia underneath.

You can learn more about Alt-DA in Optimism docs.

`,16),O=JSON.parse('{"title":"Run an OP Stack rollup with Celestia underneath","description":"Start your own rollup with op-plasma-celestia and roll-op.","frontmatter":{"description":"Start your own rollup with op-plasma-celestia and roll-op.","head":[["meta",{"name":"og:title","content":"Run an OP Stack rollup with Celestia underneath | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/optimism.md","filePath":"developers/optimism.md","lastUpdated":1726560482000}'),P={name:"developers/optimism.md"},V=Object.assign(P,{setup(B){return(M,x)=>(t(),l("div",null,[p,s("div",c,[r,s("div",i,[s("div",d,[u,h,s("pre",y,[s("code",null,[s("span",_,[s("span",E,"$HOME/.celestia-light-"+e(n(a).mochaChainId),1)])])]),s("pre",g,[s("code",null,[s("span",m,[s("span",b,"$HOME/.celestia-light-"+e(n(a).mochaChainId),1)])])])]),v,s("div",k,[C,q,s("pre",F,[s("code",null,[s("span",f,[s("span",A,"$HOME/.celestia-light-"+e(n(a).arabicaChainId),1)])])]),s("pre",S,[s("code",null,[s("span",T,[s("span",w,"$HOME/.celestia-light-"+e(n(a).arabicaChainId),1)])])])])])]),D]))}});export{O as __pageData,V as default}; diff --git a/assets/developers_optimism.md.a839aad4.lean.js b/assets/developers_optimism.md.a839aad4.lean.js new file mode 100644 index 00000000000..ad6e1d48317 --- /dev/null +++ b/assets/developers_optimism.md.a839aad4.lean.js @@ -0,0 +1 @@ +import{c as a}from"./chunks/constants.fa173a21.js";import{o as t,c as l,k as s,t as e,l as n,Q as o}from"./chunks/framework.1a91c06a.js";const p=o("",8),c={class:"vp-code-group vp-adaptive-theme"},r=o("",1),i={class:"blocks"},d={class:"language-bash vp-adaptive-theme active"},u=s("button",{title:"Copy Code",class:"copy"},null,-1),h=s("span",{class:"lang"},"bash",-1),y={class:"shiki github-dark vp-code-dark"},_={class:"line"},E={style:{color:"#E1E4E8"}},g={class:"shiki github-light vp-code-light"},m={class:"line"},b={style:{color:"#24292E"}},v=o("",1),k={class:"language-bash vp-adaptive-theme"},C=s("button",{title:"Copy Code",class:"copy"},null,-1),q=s("span",{class:"lang"},"bash",-1),F={class:"shiki github-dark vp-code-dark"},f={class:"line"},A={style:{color:"#E1E4E8"}},S={class:"shiki github-light vp-code-light"},T={class:"line"},w={style:{color:"#24292E"}},D=o("",16),O=JSON.parse('{"title":"Run an OP Stack rollup with Celestia underneath","description":"Start your own rollup with op-plasma-celestia and roll-op.","frontmatter":{"description":"Start your own rollup with op-plasma-celestia and roll-op.","head":[["meta",{"name":"og:title","content":"Run an OP Stack rollup with Celestia underneath | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/optimism.md","filePath":"developers/optimism.md","lastUpdated":1726560482000}'),P={name:"developers/optimism.md"},V=Object.assign(P,{setup(B){return(M,x)=>(t(),l("div",null,[p,s("div",c,[r,s("div",i,[s("div",d,[u,h,s("pre",y,[s("code",null,[s("span",_,[s("span",E,"$HOME/.celestia-light-"+e(n(a).mochaChainId),1)])])]),s("pre",g,[s("code",null,[s("span",m,[s("span",b,"$HOME/.celestia-light-"+e(n(a).mochaChainId),1)])])])]),v,s("div",k,[C,q,s("pre",F,[s("code",null,[s("span",f,[s("span",A,"$HOME/.celestia-light-"+e(n(a).arabicaChainId),1)])])]),s("pre",S,[s("code",null,[s("span",T,[s("span",w,"$HOME/.celestia-light-"+e(n(a).arabicaChainId),1)])])])])])]),D]))}});export{O as __pageData,V as default}; diff --git a/assets/developers_prompt-scavenger.md.b002a641.js b/assets/developers_prompt-scavenger.md.b002a641.js new file mode 100644 index 00000000000..661c2d20649 --- /dev/null +++ b/assets/developers_prompt-scavenger.md.b002a641.js @@ -0,0 +1,419 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.1a91c06a.js";const h=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":"Rust client tutorial","link":"/developers/rust-client-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":1718711052000}'),p={name:"developers/prompt-scavenger.md"},o=l(`

Prompt scavenger

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 start by initializing our light node and funding our account with some tokens. We will be using the Arabica testnet for this tutorial.

sh
celestia light init --p2p.network arabica
celestia light init --p2p.network arabica

You will see an output ending with something looking like this:

2024-05-22T14:15:49.554+0200	INFO	node	nodebuilder/init.go:211	NO KEY FOUND IN STORE, GENERATING NEW KEY...
+2024-05-22T14:15:49.564+0200	INFO	node	nodebuilder/init.go:216	NEW KEY GENERATED...
+
+NAME: my_celes_key
+ADDRESS: celestia1hn25k7gkfq0fy5a0vmphs6mjma2de74gsn36ef
+MNEMONIC (save this somewhere safe!!!):
+**** **** **** ****
2024-05-22T14:15:49.554+0200	INFO	node	nodebuilder/init.go:211	NO KEY FOUND IN STORE, GENERATING NEW KEY...
+2024-05-22T14:15:49.564+0200	INFO	node	nodebuilder/init.go:216	NEW KEY GENERATED...
+
+NAME: my_celes_key
+ADDRESS: celestia1hn25k7gkfq0fy5a0vmphs6mjma2de74gsn36ef
+MNEMONIC (save this somewhere safe!!!):
+**** **** **** ****

To fund your account, copy the address from the "ADDRESS" log and paste it in the Arabica Faucet to request tokens.

While waiting for our account to be funded, we can start our light node.

sh
celestia light start --core.ip validator-1.celestia-arabica-11.com --p2p.network arabica --rpc.skip-auth
celestia light start --core.ip validator-1.celestia-arabica-11.com --p2p.network arabica --rpc.skip-auth

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.

We can now check if your account has been successfully funded by running the following command. If your balance is still 0, wait a few seconds and try again.

sh
celestia state balance --node.store ~/.celestia-light-arabica-11
celestia state balance --node.store ~/.celestia-light-arabica-11

TIP

Make sure you run this command in a different terminal window because the node has to be running for it to work.

OpenAI key

Visit OpenAI to sign up for an account and generate an API key. order to sign up for an account and generate an OpenAI API key. The key will be needed to communicate with OpenAI.

Once you have created an API key, set it as an environment variable with the following command, pasting in your own key:

sh
export OPENAI_KEY=sk-proj-xxxxxxxxxxxxxxxxxxxxxxxx
export OPENAI_KEY=sk-proj-xxxxxxxxxxxxxxxxxxxxxxxx

Building the Prompt Scavenger

Initialize your Go project

To initialize your go project, run the following commands:

sh
mkdir test_scavenger
+cd test_scavenger
+go mod init prompt-scavenger
+go get github.com/celestiaorg/celestia-openrpc
+go get github.com/sashabaranov/go-openai
mkdir test_scavenger
+cd test_scavenger
+go mod init prompt-scavenger
+go get github.com/celestiaorg/celestia-openrpc
+go get github.com/sashabaranov/go-openai

This will set up a go project in a new directory and download the required modules.

Build your import statements

Inside the directory, create a main.go file and setup the import statements:

go
package main
+
+import (
+	"context"
+	"encoding/hex"
+	"fmt"
+	"log"
+	"os"
+
+	nodeclient "github.com/celestiaorg/celestia-openrpc"
+	"github.com/celestiaorg/celestia-openrpc/types/blob"
+	"github.com/celestiaorg/celestia-openrpc/types/share"
+	 openai "github.com/sashabaranov/go-openai"
+)
+
+func main() {
+  // TODO:
+	// - [ ] Load program arguments
+	// - [ ] Initialize the node API client
+	// - [ ] Create a namespace ID
+	// - [ ] Create and submit a blob
+	// - [ ] Retrieve the blob from the network
+	// - [ ] Prompt chatgpt with the retrieved blob data
+}
package main
+
+import (
+	"context"
+	"encoding/hex"
+	"fmt"
+	"log"
+	"os"
+
+	nodeclient "github.com/celestiaorg/celestia-openrpc"
+	"github.com/celestiaorg/celestia-openrpc/types/blob"
+	"github.com/celestiaorg/celestia-openrpc/types/share"
+	 openai "github.com/sashabaranov/go-openai"
+)
+
+func main() {
+  // TODO:
+	// - [ ] Load program arguments
+	// - [ ] Initialize the node API client
+	// - [ ] Create a namespace ID
+	// - [ ] Create and submit a blob
+	// - [ ] Retrieve the blob from the network
+	// - [ ] Prompt chatgpt with the retrieved blob data
+}

Here we set up all required libraries we need to use plus the main function that we will use for our program. function that we will use for our program.

TIP

Depending on your IDE, unused import statements may be removed every time you save the file. If this is the case, come back to this section and add them one by one as they come up in the code snippets.

Main function

Let's start populating our main function. To begin, we need to load the arguments we pass to the program. and do some sanity checks. We will then initialize the node API client.

go

+func main() {
+	ctx, cancel := context.WithCancel(context.Background())
+	defer cancel()
+
+	// Get IP, namespace, and prompt from program arguments
+	if len(os.Args) != 4 {
+		log.Fatal("Usage: go run main.go <nodeIP> <namespace> <prompt>")
+	}
+	nodeIP, namespaceHex, prompt := os.Args[1], os.Args[2], os.Args[3]
+
+	// We pass an empty string as the jwt token, since we
+	// disabled auth with the --rpc.skip-auth flag
+	client, err := nodeclient.NewClient(ctx, nodeIP, "")
+	if err != nil {
+		log.Fatalf("Failed to create client: %v", err)
+	}
+	defer client.Close()
+
+	// TODO:
+	// - [X] Load program arguments
+	// - [X] Initialize the node API client
+	// - [ ] Create a namespace ID
+	// - [ ] Create and submit a blob
+	// - [ ] Retrieve the blob from the network
+	// - [ ] Prompt chatgpt with the retrieved blob data
+}

+func main() {
+	ctx, cancel := context.WithCancel(context.Background())
+	defer cancel()
+
+	// Get IP, namespace, and prompt from program arguments
+	if len(os.Args) != 4 {
+		log.Fatal("Usage: go run main.go <nodeIP> <namespace> <prompt>")
+	}
+	nodeIP, namespaceHex, prompt := os.Args[1], os.Args[2], os.Args[3]
+
+	// We pass an empty string as the jwt token, since we
+	// disabled auth with the --rpc.skip-auth flag
+	client, err := nodeclient.NewClient(ctx, nodeIP, "")
+	if err != nil {
+		log.Fatalf("Failed to create client: %v", err)
+	}
+	defer client.Close()
+
+	// TODO:
+	// - [X] Load program arguments
+	// - [X] Initialize the node API client
+	// - [ ] Create a namespace ID
+	// - [ ] Create and submit a blob
+	// - [ ] Retrieve the blob from the network
+	// - [ ] Prompt chatgpt with the retrieved blob data
+}

Next, we need to create some utility functions that will help us with our next TODO items.

Utility functions

First, we need a function to convert a hex string to a NamespaceID type that is used for blob creation. This is needed because the namespace we pass in the program arguments will be in hexadecimal format.

go
// createNamespaceID converts a hex string to a NamespaceID
+func createNamespaceID(nIDString string) (share.Namespace, error) {
+	// First, we parse the passed hex string into a []byte slice
+	namespaceBytes, err := hex.DecodeString(nIDString)
+	if err != nil {
+		return nil, fmt.Errorf("error decoding hex string: %w", err)
+	}
+
+	// Next, we create a new NamespaceID using the parsed bytes
+	return share.NewBlobNamespaceV0(namespaceBytes)
+}
// createNamespaceID converts a hex string to a NamespaceID
+func createNamespaceID(nIDString string) (share.Namespace, error) {
+	// First, we parse the passed hex string into a []byte slice
+	namespaceBytes, err := hex.DecodeString(nIDString)
+	if err != nil {
+		return nil, fmt.Errorf("error decoding hex string: %w", err)
+	}
+
+	// Next, we create a new NamespaceID using the parsed bytes
+	return share.NewBlobNamespaceV0(namespaceBytes)
+}

Next, we need a utility that takes the namespace generated by createNamespaceID and constructs and submits a blob to the network.

If successful, it returns the created blob, the height at which it was posted, and an empty error. Otherwise, only the error field is populated.

go
// createAndSubmitBlob creates a new blob and submits it to the network.
+func createAndSubmitBlob(
+	ctx context.Context,
+	client *nodeclient.Client,
+	ns share.Namespace,
+	payload string,
+) (*blob.Blob, uint64, error) {
+	// First we can create the blob using the namespace and payload.
+	createdBlob, err := blob.NewBlobV0(ns, []byte(payload))
+	if err != nil {
+		return nil, 0, fmt.Errorf("Failed to create blob: %w", err)
+	}
+
+	// After we've created the blob, we can submit it to the network.
+	// Here we use the default gas price.
+	height, err := client.Blob.Submit(ctx, []*blob.Blob{createdBlob}, blob.DefaultGasPrice())
+	if err != nil {
+		return nil, 0, fmt.Errorf("Failed to submit blob: %v", err)
+	}
+
+	log.Printf("Blob submitted successfully at height: %d! \\n", height)
+	log.Printf("Explorer link: https://arabica.celenium.io/block/%d \\n", height)
+
+	return createdBlob, height, nil
+}
// createAndSubmitBlob creates a new blob and submits it to the network.
+func createAndSubmitBlob(
+	ctx context.Context,
+	client *nodeclient.Client,
+	ns share.Namespace,
+	payload string,
+) (*blob.Blob, uint64, error) {
+	// First we can create the blob using the namespace and payload.
+	createdBlob, err := blob.NewBlobV0(ns, []byte(payload))
+	if err != nil {
+		return nil, 0, fmt.Errorf("Failed to create blob: %w", err)
+	}
+
+	// After we've created the blob, we can submit it to the network.
+	// Here we use the default gas price.
+	height, err := client.Blob.Submit(ctx, []*blob.Blob{createdBlob}, blob.DefaultGasPrice())
+	if err != nil {
+		return nil, 0, fmt.Errorf("Failed to submit blob: %v", err)
+	}
+
+	log.Printf("Blob submitted successfully at height: %d! \\n", height)
+	log.Printf("Explorer link: https://arabica.celenium.io/block/%d \\n", height)
+
+	return createdBlob, height, nil
+}

With our updated main function, we can now call these utility functions to check off our next TODO items.

go
func main() {
+	ctx, cancel := context.WithCancel(context.Background())
+	defer cancel()
+
+	// Get IP, namespace, and prompt from program arguments
+	if len(os.Args) != 4 {
+		log.Fatal("Usage: go run main.go <nodeIP> <namespace> <prompt>")
+	}
+	nodeIP, namespaceHex, prompt := os.Args[1], os.Args[2], os.Args[3]
+
+	// We pass an empty string as the jwt token, since we
+	// disabled auth with the --rpc.skip-auth flag
+	client, err := nodeclient.NewClient(ctx, nodeIP, "")
+	if err != nil {
+		log.Fatalf("Failed to create client: %v", err)
+	}
+	defer client.Close()
+
+	// Next, we convert the namespace hex string to the
+	// concrete NamespaceID type
+	namespaceID, err := createNamespaceID(namespaceHex)
+	if err != nil {
+		log.Fatalf("Failed to decode namespace: %v", err)
+	}
+
+	// We can then create and submit a blob using the NamespaceID and our prompt.
+	createdBlob, height, err := createAndSubmitBlob(ctx, client, namespaceID, prompt)
+	if err != nil {
+		log.Fatal(err)
+	}
+
+	// Now we will fetch the blob back from the network, using the height, namespace, and blob commitment.
+	fetchedBlob, err := client.Blob.Get(ctx, height, namespaceID, createdBlob.Commitment)
+	if err != nil {
+		log.Fatalf("Failed to fetch blob: %v", err)
+	}
+
+	log.Printf("Fetched blob: %s\\n", string(fetchedBlob.Data))
+
+	// TODO:
+	// - [X] Load program arguments
+	// - [X] Initialize the node API client
+	// - [X] Create a namespace ID
+	// - [X] Create and submit a blob
+	// - [X] Retrieve the blob from the network
+	// - [ ] Prompt chatgpt with the retrieved blob data
+}
func main() {
+	ctx, cancel := context.WithCancel(context.Background())
+	defer cancel()
+
+	// Get IP, namespace, and prompt from program arguments
+	if len(os.Args) != 4 {
+		log.Fatal("Usage: go run main.go <nodeIP> <namespace> <prompt>")
+	}
+	nodeIP, namespaceHex, prompt := os.Args[1], os.Args[2], os.Args[3]
+
+	// We pass an empty string as the jwt token, since we
+	// disabled auth with the --rpc.skip-auth flag
+	client, err := nodeclient.NewClient(ctx, nodeIP, "")
+	if err != nil {
+		log.Fatalf("Failed to create client: %v", err)
+	}
+	defer client.Close()
+
+	// Next, we convert the namespace hex string to the
+	// concrete NamespaceID type
+	namespaceID, err := createNamespaceID(namespaceHex)
+	if err != nil {
+		log.Fatalf("Failed to decode namespace: %v", err)
+	}
+
+	// We can then create and submit a blob using the NamespaceID and our prompt.
+	createdBlob, height, err := createAndSubmitBlob(ctx, client, namespaceID, prompt)
+	if err != nil {
+		log.Fatal(err)
+	}
+
+	// Now we will fetch the blob back from the network, using the height, namespace, and blob commitment.
+	fetchedBlob, err := client.Blob.Get(ctx, height, namespaceID, createdBlob.Commitment)
+	if err != nil {
+		log.Fatalf("Failed to fetch blob: %v", err)
+	}
+
+	log.Printf("Fetched blob: %s\\n", string(fetchedBlob.Data))
+
+	// TODO:
+	// - [X] Load program arguments
+	// - [X] Initialize the node API client
+	// - [X] Create a namespace ID
+	// - [X] Create and submit a blob
+	// - [X] Retrieve the blob from the network
+	// - [ ] Prompt chatgpt with the retrieved blob data
+}

TIP

Alternatively to client.Blob.Get, you could also use client.Blob.GetAll(ctx, height, []share.Namespace{namespaceID}) which fetches all blobs in the namespace at the given height.

Now our program is able to create the namespace and blob, then submit and fetch it from the arabica network. The next step is to prompt ChatGPT with the fetched blob data.

Prompting ChatGPT

First, we need one more utility function to help us prompt GPT-3.5. It reads the OPENAI_KEY environment variable and uses it to create a new GPT-3 client, which it uses to prompt and retrieve the answer.

go
// gpt3 processes a given message using GPT-3 and returns the response.
+func gpt3(ctx context.Context, msg string) (string, error) {
+    // Set the authentication header
+    openAIKey := os.Getenv("OPENAI_KEY")
+   	if openAIKey == "" {
+    	return "", fmt.Errorf("OPENAI_KEY environment variable not set")
+    }
+    client := openai.NewClient(openAIKey)
+    resp, err := client.CreateChatCompletion(
+		ctx,
+        openai.ChatCompletionRequest{
+            Model: openai.GPT3Dot5Turbo,
+            Messages: []openai.ChatCompletionMessage{
+                {
+                    Role:    openai.ChatMessageRoleUser,
+                    Content: msg,
+                },
+            },
+        },
+    )
+
+    if err != nil {
+       	return "", fmt.Errorf("ChatCompletion error: %w", err)
+    }
+
+    return resp.Choices[0].Message.Content, nil
+}
// gpt3 processes a given message using GPT-3 and returns the response.
+func gpt3(ctx context.Context, msg string) (string, error) {
+    // Set the authentication header
+    openAIKey := os.Getenv("OPENAI_KEY")
+   	if openAIKey == "" {
+    	return "", fmt.Errorf("OPENAI_KEY environment variable not set")
+    }
+    client := openai.NewClient(openAIKey)
+    resp, err := client.CreateChatCompletion(
+		ctx,
+        openai.ChatCompletionRequest{
+            Model: openai.GPT3Dot5Turbo,
+            Messages: []openai.ChatCompletionMessage{
+                {
+                    Role:    openai.ChatMessageRoleUser,
+                    Content: msg,
+                },
+            },
+        },
+    )
+
+    if err != nil {
+       	return "", fmt.Errorf("ChatCompletion error: %w", err)
+    }
+
+    return resp.Choices[0].Message.Content, nil
+}

Wrapping things up

Now, we will update our main function to finish our last TODO item: prompting CHATGPT with the fetched blob data.

go
func main() {
+	ctx, cancel := context.WithCancel(context.Background())
+	defer cancel()
+
+	// Get IP, namespace, and prompt from program arguments
+	if len(os.Args) != 4 {
+		log.Fatal("Usage: go run main.go <nodeIP> <namespace> <prompt>")
+	}
+	nodeIP, namespaceHex, prompt := os.Args[1], os.Args[2], os.Args[3]
+
+	// We pass an empty string as the jwt token, since we
+	// disabled auth with the --rpc.skip-auth flag
+	client, err := nodeclient.NewClient(ctx, nodeIP, "")
+	if err != nil {
+		log.Fatalf("Failed to create client: %v", err)
+	}
+	defer client.Close()
+
+	// Next, we convert the namespace hex string to the
+	// concrete NamespaceID type
+	namespaceID, err := createNamespaceID(namespaceHex)
+	if err != nil {
+		log.Fatalf("Failed to decode namespace: %v", err)
+	}
+
+	// We can then create and submit a blob using the NamespaceID and our prompt.
+	createdBlob, height, err := createAndSubmitBlob(ctx, client, namespaceID, prompt)
+	if err != nil {
+		log.Fatal(err)
+	}
+
+	// Now we will fetch the blob back from the network.
+	fetchedBlob, err := client.Blob.Get(ctx, height, namespaceID, createdBlob.Commitment)
+	if err != nil {
+		log.Fatalf("Failed to fetch blob: %v", err)
+	}
+
+	log.Printf("Fetched blob: %s\\n", string(fetchedBlob.Data))
+	promptAnswer, err := gpt3(ctx, string(fetchedBlob.Data))
+	if err != nil {
+		log.Fatalf("Failed to process message with GPT-3: %v", err)
+	}
+
+	log.Printf("GPT-3 response: %s\\n", promptAnswer)
+}
func main() {
+	ctx, cancel := context.WithCancel(context.Background())
+	defer cancel()
+
+	// Get IP, namespace, and prompt from program arguments
+	if len(os.Args) != 4 {
+		log.Fatal("Usage: go run main.go <nodeIP> <namespace> <prompt>")
+	}
+	nodeIP, namespaceHex, prompt := os.Args[1], os.Args[2], os.Args[3]
+
+	// We pass an empty string as the jwt token, since we
+	// disabled auth with the --rpc.skip-auth flag
+	client, err := nodeclient.NewClient(ctx, nodeIP, "")
+	if err != nil {
+		log.Fatalf("Failed to create client: %v", err)
+	}
+	defer client.Close()
+
+	// Next, we convert the namespace hex string to the
+	// concrete NamespaceID type
+	namespaceID, err := createNamespaceID(namespaceHex)
+	if err != nil {
+		log.Fatalf("Failed to decode namespace: %v", err)
+	}
+
+	// We can then create and submit a blob using the NamespaceID and our prompt.
+	createdBlob, height, err := createAndSubmitBlob(ctx, client, namespaceID, prompt)
+	if err != nil {
+		log.Fatal(err)
+	}
+
+	// Now we will fetch the blob back from the network.
+	fetchedBlob, err := client.Blob.Get(ctx, height, namespaceID, createdBlob.Commitment)
+	if err != nil {
+		log.Fatalf("Failed to fetch blob: %v", err)
+	}
+
+	log.Printf("Fetched blob: %s\\n", string(fetchedBlob.Data))
+	promptAnswer, err := gpt3(ctx, string(fetchedBlob.Data))
+	if err != nil {
+		log.Fatalf("Failed to process message with GPT-3: %v", err)
+	}
+
+	log.Printf("GPT-3 response: %s\\n", promptAnswer)
+}

And now you have the final version of the prompt scavenger!

Run the golang script with the following command:

sh
go run main.go <nodeIP> <namespace> <prompt>
go run main.go <nodeIP> <namespace> <prompt>

For example, you could run:

sh
go run main.go ws://localhost:26658 ce1e5714 'What is a modular blockchain?'
go run main.go ws://localhost:26658 ce1e5714 'What is a modular blockchain?'

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 blob, submit it to Celestia, get it back from Celestia, decode its contents, then for added bonus, submit the message to GPT-3.5.

If you're up for a challenge, you can refer to the Node API client guide and try to implement more advanced features, such as:

  • Subscribing to new prompts inside the ce1e5714 namespace, submitting each one to GPT-3.5
  • Posting the responses back to Celestia under a different namespace.
`,64),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/developers_prompt-scavenger.md.b002a641.lean.js b/assets/developers_prompt-scavenger.md.b002a641.lean.js new file mode 100644 index 00000000000..e09b07e3c4d --- /dev/null +++ b/assets/developers_prompt-scavenger.md.b002a641.lean.js @@ -0,0 +1 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.1a91c06a.js";const h=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":"Rust client tutorial","link":"/developers/rust-client-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":1718711052000}'),p={name:"developers/prompt-scavenger.md"},o=l("",64),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/developers_rust-client-tutorial.md.78994657.js b/assets/developers_rust-client-tutorial.md.78994657.js new file mode 100644 index 00000000000..ca03ebb6e90 --- /dev/null +++ b/assets/developers_rust-client-tutorial.md.78994657.js @@ -0,0 +1,171 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.1a91c06a.js";const h=JSON.parse('{"title":"Rust client library tutorial","description":"","frontmatter":{"next":{"text":"Prompt Scavenger","link":"/developers/prompt-scavenger"},"head":[["meta",{"name":"og:title","content":"Rust client library tutorial | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/rust-client-tutorial.md","filePath":"developers/rust-client-tutorial.md","lastUpdated":1718751588000}'),p={name:"developers/rust-client-tutorial.md"},o=l(`

Rust client library tutorial

This section tutorial will guide you through using the most common RPC endpoints with Lumina's rust client library.

Install dependencies and celestia-node if you have not already.

Project setup

To start, add celestia_rpc and celestia_types as a dependency to your project:

bash
cargo add celestia_rpc celestia_types
cargo add celestia_rpc celestia_types

To use the following methods, you will need the node URL and your auth token. To get your auth token, see this guide. To run your node without an auth token, you can use the --rpc.skip-auth flag when starting your node. This allows you to pass an empty string as your auth token.

The default URL is http://localhost:26658. If you would like to use subscription methods, such as SubscribeHeaders below, you must use the ws protocol in place of http: ws://localhost:26658.

Submitting and retrieving blobs

The blob.Submit method takes an array of blobs and a gas price, returning the height the blob was successfully posted at.

  • The namespace can be generated with Namespace::new_v0.
  • The blobs can be generated with Blob::new.
  • You can set GasPrice::default() as the gas price to have celestia-node automatically determine an appropriate gas price.

The blob.GetAll method takes a height and array of namespaces, returning the array of blobs found in the given namespaces.

rust
use celestia_rpc::{BlobClient, Client, HeaderClient, ShareClient};
+use celestia_types::blob::GasPrice;
+use celestia_types::{nmt::Namespace, Blob, ExtendedDataSquare};
+
+async fn submit_blob(url: &str, token: &str) {
+    let client = Client::new(url, Some(token))
+        .await
+        .expect("Failed creating rpc client");
+
+    // let's use the DEADBEEF namespace
+    let namespace = Namespace::new_v0(&[0xDE, 0xAD, 0xBE, 0xEF]).expect("Invalid namespace");
+
+    // create a blob
+    let blob = Blob::new(namespace, b"Hello, World!".to_vec()).expect("Blob creation failed");
+
+    // submit the blob to the network
+    let height = client
+        .blob_submit(&[blob.clone()], GasPrice::default())
+        .await
+        .expect("Failed submitting blob");
+
+    println!("Blob was included at height {}", height);
+
+    // fetch the blob back from the network
+    let retrieved_blobs = client
+        .blob_get_all(height, &[namespace])
+        .await
+        .expect("Failed to retrieve blobs");
+
+    assert_eq!(retrieved_blobs.len(), 1);
+    assert_eq!(retrieved_blobs[0].data, b"Hello, World!");
+    assert_eq!(retrieved_blobs[0].commitment, blob.commitment);
+}
use celestia_rpc::{BlobClient, Client, HeaderClient, ShareClient};
+use celestia_types::blob::GasPrice;
+use celestia_types::{nmt::Namespace, Blob, ExtendedDataSquare};
+
+async fn submit_blob(url: &str, token: &str) {
+    let client = Client::new(url, Some(token))
+        .await
+        .expect("Failed creating rpc client");
+
+    // let's use the DEADBEEF namespace
+    let namespace = Namespace::new_v0(&[0xDE, 0xAD, 0xBE, 0xEF]).expect("Invalid namespace");
+
+    // create a blob
+    let blob = Blob::new(namespace, b"Hello, World!".to_vec()).expect("Blob creation failed");
+
+    // submit the blob to the network
+    let height = client
+        .blob_submit(&[blob.clone()], GasPrice::default())
+        .await
+        .expect("Failed submitting blob");
+
+    println!("Blob was included at height {}", height);
+
+    // fetch the blob back from the network
+    let retrieved_blobs = client
+        .blob_get_all(height, &[namespace])
+        .await
+        .expect("Failed to retrieve blobs");
+
+    assert_eq!(retrieved_blobs.len(), 1);
+    assert_eq!(retrieved_blobs[0].data, b"Hello, World!");
+    assert_eq!(retrieved_blobs[0].commitment, blob.commitment);
+}

Subscribing to new headers

You can subscribe to new headers using the header.Subscribe method. This method returns a Subscription that will receive new headers as they are produced. In this example, we will fetch all blobs at the height of the new header in the 0xDEADBEEF namespace.

rust
async fn subscribe_headers(url: &str, token: &str) {
+    let client = Client::new(url, Some(token))
+        .await
+        .expect("Failed creating rpc client");
+
+    let mut header_sub = client
+        .header_subscribe()
+        .await
+        .expect("Failed subscribing to incoming headers");
+
+    // setup the namespace we will filter blobs by
+    let namespace = Namespace::new_v0(&[0xDE, 0xAD, 0xBE, 0xEF]).expect("Invalid namespace");
+
+    while let Some(extended_header) = header_sub.next().await {
+        match extended_header {
+            Ok(header) => {
+                let height = header.header.height.value();
+                // fetch all blobs at the height of the new header
+
+                let blobs = match client.blob_get_all(height, &[namespace]).await {
+                    Ok(blobs) => blobs,
+                    Err(e) => {
+                        eprintln!("Error fetching blobs: {}", e);
+                        continue;
+                    }
+                };
+
+                println!(
+                    "Found {} blobs at height {} in the 0xDEADBEEF namespace",
+                    blobs.len(),
+                    height
+                );
+            }
+            Err(e) => {
+                eprintln!("Error receiving header: {}", e);
+            }
+        }
+    }
+}
async fn subscribe_headers(url: &str, token: &str) {
+    let client = Client::new(url, Some(token))
+        .await
+        .expect("Failed creating rpc client");
+
+    let mut header_sub = client
+        .header_subscribe()
+        .await
+        .expect("Failed subscribing to incoming headers");
+
+    // setup the namespace we will filter blobs by
+    let namespace = Namespace::new_v0(&[0xDE, 0xAD, 0xBE, 0xEF]).expect("Invalid namespace");
+
+    while let Some(extended_header) = header_sub.next().await {
+        match extended_header {
+            Ok(header) => {
+                let height = header.header.height.value();
+                // fetch all blobs at the height of the new header
+
+                let blobs = match client.blob_get_all(height, &[namespace]).await {
+                    Ok(blobs) => blobs,
+                    Err(e) => {
+                        eprintln!("Error fetching blobs: {}", e);
+                        continue;
+                    }
+                };
+
+                println!(
+                    "Found {} blobs at height {} in the 0xDEADBEEF namespace",
+                    blobs.len(),
+                    height
+                );
+            }
+            Err(e) => {
+                eprintln!("Error receiving header: {}", e);
+            }
+        }
+    }
+}

Fetching an Extended Data Square (EDS)

You can fetch an Extended Data Square (EDS) using the share.GetEDS method. This method takes a header and returns the EDS at the given height.

rust
async fn get_eds(url: &str, token: &str) -> ExtendedDataSquare {
+    let client = Client::new(url, Some(token))
+        .await
+        .expect("Failed creating rpc client");
+
+    // first get the header of the block you want to fetch the EDS from
+    let latest_header = client
+        .header_local_head()
+        .await
+        .expect("Failed fetching header");
+
+    client
+        .share_get_eds(&latest_header)
+        .await
+        .expect("Failed to get EDS from latest header")
+}
async fn get_eds(url: &str, token: &str) -> ExtendedDataSquare {
+    let client = Client::new(url, Some(token))
+        .await
+        .expect("Failed creating rpc client");
+
+    // first get the header of the block you want to fetch the EDS from
+    let latest_header = client
+        .header_local_head()
+        .await
+        .expect("Failed fetching header");
+
+    client
+        .share_get_eds(&latest_header)
+        .await
+        .expect("Failed to get EDS from latest header")
+}

API documentation

To see the full list of available methods, see the API documentation.

`,21),e=[o];function t(c,r,E,y,i,F){return a(),n("div",null,e)}const u=s(p,[["render",t]]);export{h as __pageData,u as default}; diff --git a/assets/developers_rust-client-tutorial.md.78994657.lean.js b/assets/developers_rust-client-tutorial.md.78994657.lean.js new file mode 100644 index 00000000000..4939f427d9c --- /dev/null +++ b/assets/developers_rust-client-tutorial.md.78994657.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.1a91c06a.js";const h=JSON.parse('{"title":"Rust client library tutorial","description":"","frontmatter":{"next":{"text":"Prompt Scavenger","link":"/developers/prompt-scavenger"},"head":[["meta",{"name":"og:title","content":"Rust client library tutorial | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/rust-client-tutorial.md","filePath":"developers/rust-client-tutorial.md","lastUpdated":1718751588000}'),p={name:"developers/rust-client-tutorial.md"},o=l("",21),e=[o];function t(c,r,E,y,i,F){return a(),n("div",null,e)}const u=s(p,[["render",t]]);export{h as __pageData,u as default}; diff --git a/assets/developers_sp1-blobstream-deploy.md.38c380ed.js b/assets/developers_sp1-blobstream-deploy.md.38c380ed.js new file mode 100644 index 00000000000..71454bc2711 --- /dev/null +++ b/assets/developers_sp1-blobstream-deploy.md.38c380ed.js @@ -0,0 +1,3 @@ +import{_ as e,o,c as t,Q as s}from"./chunks/framework.1a91c06a.js";const y=JSON.parse('{"title":"New SP1 Blobstream deployments","description":"","frontmatter":{"next":{"text":"Overview of Blobstream X","link":"/developers/blobstreamx"},"head":[["meta",{"name":"og:title","content":"New SP1 Blobstream deployments | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/sp1-blobstream-deploy.md","filePath":"developers/sp1-blobstream-deploy.md","lastUpdated":1726491136000}'),a={name:"developers/sp1-blobstream-deploy.md"},l=s(`

New SP1 Blobstream deployments

This document provides instructions for deploying SP1 Blobstream to a new chain.

SP1 Blobstream is the latest implementation of Blobstream in Rust using the SP1 zkVM.

Deploying the contracts

To deploy SP1 Blobstream to a new chain, follow these steps:

  1. Clone the sp1-blobstream repository:
shell
git clone https://github.com/succinctlabs/sp1-blobstream
+cd sp1-blobstream
git clone https://github.com/succinctlabs/sp1-blobstream
+cd sp1-blobstream
  1. Follow the deployment instructions in the sp1-blobstream README.

  2. If you're deploying on a chain where there isn't a canonical verifier listed in the SP1 contract addresses, you'll need to:

    a. Deploy your own SP1 Verifier from the sp1-contracts matching your sp1-sdk version. b. Set the SP1_VERIFIER_ADDRESS in your .env file to the address of your deployed verifier.

  3. To run the prover:

    • For local proving, set SP1_PROVER=local in your environment.
    • To use the Succinct Proving Network for remote proving, set SP1_PROVER=network.
    • We recommend an instance with 64 vCPU and 128GB of RAM for local proving.

Note: Any whitelisting for custom provers would need to be implemented in the application's smart contracts (e.g., by using an approvedProvers mapping).

`,9),n=[l];function r(p,c,i,d,b,h){return o(),t("div",null,n)}const u=e(a,[["render",r]]);export{y as __pageData,u as default}; diff --git a/assets/developers_sp1-blobstream-deploy.md.38c380ed.lean.js b/assets/developers_sp1-blobstream-deploy.md.38c380ed.lean.js new file mode 100644 index 00000000000..a1f0aa298a6 --- /dev/null +++ b/assets/developers_sp1-blobstream-deploy.md.38c380ed.lean.js @@ -0,0 +1 @@ +import{_ as e,o,c as t,Q as s}from"./chunks/framework.1a91c06a.js";const y=JSON.parse('{"title":"New SP1 Blobstream deployments","description":"","frontmatter":{"next":{"text":"Overview of Blobstream X","link":"/developers/blobstreamx"},"head":[["meta",{"name":"og:title","content":"New SP1 Blobstream deployments | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"developers/sp1-blobstream-deploy.md","filePath":"developers/sp1-blobstream-deploy.md","lastUpdated":1726491136000}'),a={name:"developers/sp1-blobstream-deploy.md"},l=s("",9),n=[l];function r(p,c,i,d,b,h){return o(),t("div",null,n)}const u=e(a,[["render",r]]);export{y as __pageData,u as default}; diff --git a/assets/developers_submit-data.md.e151ce10.js b/assets/developers_submit-data.md.e151ce10.js new file mode 100644 index 00000000000..580b5d8553d --- /dev/null +++ b/assets/developers_submit-data.md.e151ce10.js @@ -0,0 +1,299 @@ +import{_ as t,o as a,c as n,k as s,a as l,Q as e}from"./chunks/framework.1a91c06a.js";const ts=JSON.parse('{"title":"Submitting data blobs to Celestia","description":"","frontmatter":{"prev":{"text":"Blobstream rollups","link":"/developers/blobstream-rollups"},"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":1726479327000}'),o={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.

Maximum blob size

The maximum total blob size in a transaction is just under 2 MiB (1,973,786 bytes), based on a 64x64 share grid (4096 shares). With one share for the PFB transaction, 4095 shares remain: 1 at 478 bytes and 4094 at 482 bytes each.

This is subject to change based on governance parameters. Learn more on the Mainnet Beta page under "Maximum bytes".

It is advisable to submit transactions where the total blob size is significantly smaller than 1.8 MiB (e.g. 500 KiB) in order for your transaction to get included in a block quickly. If a tx contains blobs approaching 1.8 MiB then there will be no room for any other transactions. This means that your transaction will only be included in a block if it has a higher gas price than every other transaction in the mempool.

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:

',21),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"},i=e('',1),Q=[i],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),y=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),E={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},d={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"},h=e('',1),m=[h],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"},f=e('',1),H=[f],C=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),w={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},x={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),B=[k],D=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),V={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},L={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"},A=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),_=[A],v=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),M={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},q={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"},Z=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),P=[Z],S=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),I={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},N={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"},j=e('',1),G=[j],O=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),z={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},R={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"},W=e('',1),J=[W],K=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),U=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 six 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.

The celestia-node API golang client

For more celestia-node API golang examples, refer to the golang client tutorial.

go
import (
+	"bytes"
+	"context"
+	"fmt"
+
+	client "github.com/celestiaorg/celestia-openrpc"
+	"github.com/celestiaorg/celestia-openrpc/types/blob"
+	"github.com/celestiaorg/celestia-openrpc/types/share"
+)
+
+// SubmitBlob submits a blob containing "Hello, World!" to the 0xDEADBEEF namespace. It uses the default signer on the running node.
+func SubmitBlob(ctx context.Context, url string, token string) error {
+	client, err := client.NewClient(ctx, url, token)
+	if err != nil {
+		return err
+	}
+
+	// let's post to 0xDEADBEEF namespace
+	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
+	if err != nil {
+		return err
+	}
+
+	// create a blob
+	helloWorldBlob, err := blob.NewBlobV0(namespace, []byte("Hello, World!"))
+	if err != nil {
+		return err
+	}
+
+	// submit the blob to the network
+	height, err := client.Blob.Submit(ctx, []*blob.Blob{helloWorldBlob}, blob.DefaultGasPrice())
+	if err != nil {
+		return err
+	}
+
+	fmt.Printf("Blob was included at height %d\\n", height)
+
+	// bonus: fetch the blob back from the network
+	retrievedBlobs, err := client.Blob.GetAll(ctx, height, []share.Namespace{namespace})
+	if err != nil {
+		return err
+	}
+
+	fmt.Printf("Blobs are equal? %v\\n", bytes.Equal(helloWorldBlob.Commitment, retrievedBlobs[0].Commitment))
+	return nil
+}}
import (
+	"bytes"
+	"context"
+	"fmt"
+
+	client "github.com/celestiaorg/celestia-openrpc"
+	"github.com/celestiaorg/celestia-openrpc/types/blob"
+	"github.com/celestiaorg/celestia-openrpc/types/share"
+)
+
+// SubmitBlob submits a blob containing "Hello, World!" to the 0xDEADBEEF namespace. It uses the default signer on the running node.
+func SubmitBlob(ctx context.Context, url string, token string) error {
+	client, err := client.NewClient(ctx, url, token)
+	if err != nil {
+		return err
+	}
+
+	// let's post to 0xDEADBEEF namespace
+	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
+	if err != nil {
+		return err
+	}
+
+	// create a blob
+	helloWorldBlob, err := blob.NewBlobV0(namespace, []byte("Hello, World!"))
+	if err != nil {
+		return err
+	}
+
+	// submit the blob to the network
+	height, err := client.Blob.Submit(ctx, []*blob.Blob{helloWorldBlob}, blob.DefaultGasPrice())
+	if err != nil {
+		return err
+	}
+
+	fmt.Printf("Blob was included at height %d\\n", height)
+
+	// bonus: fetch the blob back from the network
+	retrievedBlobs, err := client.Blob.GetAll(ctx, height, []share.Namespace{namespace})
+	if err != nil {
+		return err
+	}
+
+	fmt.Printf("Blobs are equal? %v\\n", bytes.Equal(helloWorldBlob.Commitment, retrievedBlobs[0].Commitment))
+	return nil
+}}

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.

Post a blob directly from Celenium

Celenium provides a user-friendly interface to view and interact with data on Celestia, and allows for submitting blobs directly from the explorer interface.

To submit a blob from Celenium, follow these steps:

  1. Navigate to the Celenium explorer and connect your wallet.
  2. Click on the terminal button in the top right corner of the screen and select the "Submit data blob" option.
  3. Next, ensure the file you are submitting is in a supported format and upload it.
  4. In the "Namespace" field, input the namespace you want to use for the blob in hex format.
  5. Finally, click on the "Continue" button to submit your blob, then approve the transaction in your wallet.

Once the blob is submitted, you will see its hash on the screen. You can also use Celenium’s search bar to search for the blob's hash and view its details.

`,33);function X($,Y,ss,as,ns,es){return a(),n("div",null,[p,s("p",null,[s("mjx-container",r,[(a(),n("svg",c,Q)),T])]),y,s("p",null,[s("mjx-container",E,[(a(),n("svg",d,m)),u])]),g,s("ul",null,[s("li",null,[s("mjx-container",b,[(a(),n("svg",F,H)),C]),l(" = Fixed Cost, is a static value (65,000 gas)")]),s("li",null,[s("mjx-container",w,[(a(),n("svg",x,B)),D]),l(" = SparseSharesNeeded for the "),s("mjx-container",V,[(a(),n("svg",L,_)),v]),l("th Blob, is the number of shares needed for the "),s("mjx-container",M,[(a(),n("svg",q,P)),S]),l("th blob in the transaction")]),s("li",null,[s("mjx-container",I,[(a(),n("svg",N,G)),O]),l(" = Share Size, is the size of each share")]),s("li",null,[s("mjx-container",z,[(a(),n("svg",R,J)),K]),l(" = Gas Cost Per Blob Byte, is a parameter that could potentially be adjusted through the system's governance mechanisms.")])]),U])}const os=t(o,[["render",X]]);export{ts as __pageData,os as default}; diff --git a/assets/developers_submit-data.md.e151ce10.lean.js b/assets/developers_submit-data.md.e151ce10.lean.js new file mode 100644 index 00000000000..dd5ee12b30c --- /dev/null +++ b/assets/developers_submit-data.md.e151ce10.lean.js @@ -0,0 +1 @@ +import{_ as t,o as a,c as n,k as s,a as l,Q as e}from"./chunks/framework.1a91c06a.js";const ts=JSON.parse('{"title":"Submitting data blobs to Celestia","description":"","frontmatter":{"prev":{"text":"Blobstream rollups","link":"/developers/blobstream-rollups"},"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":1726479327000}'),o={name:"developers/submit-data.md"},p=e("",21),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"},i=e("",1),Q=[i],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),y=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),E={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},d={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"},h=e("",1),m=[h],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"},f=e("",1),H=[f],C=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),w={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},x={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),B=[k],D=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),V={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},L={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"},A=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),_=[A],v=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),M={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},q={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"},Z=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),P=[Z],S=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),I={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},N={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"},j=e("",1),G=[j],O=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),z={class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},R={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"},W=e("",1),J=[W],K=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),U=e("",33);function X($,Y,ss,as,ns,es){return a(),n("div",null,[p,s("p",null,[s("mjx-container",r,[(a(),n("svg",c,Q)),T])]),y,s("p",null,[s("mjx-container",E,[(a(),n("svg",d,m)),u])]),g,s("ul",null,[s("li",null,[s("mjx-container",b,[(a(),n("svg",F,H)),C]),l(" = Fixed Cost, is a static value (65,000 gas)")]),s("li",null,[s("mjx-container",w,[(a(),n("svg",x,B)),D]),l(" = SparseSharesNeeded for the "),s("mjx-container",V,[(a(),n("svg",L,_)),v]),l("th Blob, is the number of shares needed for the "),s("mjx-container",M,[(a(),n("svg",q,P)),S]),l("th blob in the transaction")]),s("li",null,[s("mjx-container",I,[(a(),n("svg",N,G)),O]),l(" = Share Size, is the size of each share")]),s("li",null,[s("mjx-container",z,[(a(),n("svg",R,J)),K]),l(" = Gas Cost Per Blob Byte, is a parameter that could potentially be adjusted through the system's governance mechanisms.")])]),U])}const os=t(o,[["render",X]]);export{ts as __pageData,os as default}; diff --git a/assets/developers_transaction-resubmission.md.f1e4eb76.js b/assets/developers_transaction-resubmission.md.f1e4eb76.js new file mode 100644 index 00000000000..0744718faad --- /dev/null +++ b/assets/developers_transaction-resubmission.md.f1e4eb76.js @@ -0,0 +1 @@ +import{_ as e,o as s,c as n,Q as i}from"./chunks/framework.1a91c06a.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":1726496322000}'),o={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.

If you are running a production application on Celestia, it is recommended to use a high-availability, production RPC provider for Mainnet Beta,Mocha, or Arabica.

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.
',9),t=[a];function r(c,d,l,p,u,h){return s(),n("div",null,t)}const f=e(o,[["render",r]]);export{b as __pageData,f as default}; diff --git a/assets/developers_transaction-resubmission.md.f1e4eb76.lean.js b/assets/developers_transaction-resubmission.md.f1e4eb76.lean.js new file mode 100644 index 00000000000..cb0aa7e1902 --- /dev/null +++ b/assets/developers_transaction-resubmission.md.f1e4eb76.lean.js @@ -0,0 +1 @@ +import{_ as e,o as s,c as n,Q as i}from"./chunks/framework.1a91c06a.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":1726496322000}'),o={name:"developers/transaction-resubmission.md"},a=i("",9),t=[a];function r(c,d,l,p,u,h){return s(),n("div",null,t)}const f=e(o,[["render",r]]);export{b as __pageData,f as default}; diff --git a/assets/developers_wallets.md.a3e6d0a6.js b/assets/developers_wallets.md.a3e6d0a6.js new file mode 100644 index 00000000000..73854e5b9ca --- /dev/null +++ b/assets/developers_wallets.md.a3e6d0a6.js @@ -0,0 +1,203 @@ +import{c as p}from"./chunks/constants.fa173a21.js";import{_ as e,o as t,c,k as s,a as n,t as l,Q as o}from"./chunks/framework.1a91c06a.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"}},b=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}}," chainName: 'Celestia',")],-1),q={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),P=s("span",{class:"line"},null,-1),I=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),N=s("span",{class:"line"},null,-1),f=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),G=s("span",{class:"line"},null,-1),Y=o('{<AddNetworkKeplr params={MAINNET_PARAMS}/>}',1),H={class:"language-js vp-adaptive-theme"},O=s("button",{title:"Copy Code",class:"copy"},null,-1),W=s("span",{class:"lang"},"js",-1),Z={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),J=o('export const MOCHA_PARAMS = {`{',1),X={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"},bs=s("button",{title:"Copy Code",class:"copy"},null,-1),qs=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),Ps={class:"line"},Is={style:{color:"#9ECBFF"}},Ss=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}}," chainName: 'Arabica devnet',")],-1),vs={class:"line"},Ns={style:{color:"#9ECBFF"}},fs={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),Gs={class:"line"},Ys={style:{color:"#032F62"}},Hs=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}}," chainName: 'Arabica devnet',")],-1),Os={class:"line"},Ws={style:{color:"#032F62"}},Zs={class:"line"},Qs={style:{color:"#032F62"}},zs=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}},"}`"),s("span",{style:{color:"#24292E"}},"}")],-1),Js=s("span",{class:"line"},null,-1),Xs=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(` +`),b,n(` +`),s("span",q,[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(` +`),P,n(` +`),I])]),s("pre",S,[s("code",null,[v,n(` +`),N,n(` +`),f,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(` +`),G,n(` +`),Y])])]),s("div",H,[O,W,s("pre",Z,[s("code",null,[Q,n(` +`),z,n(` +`),J,n(` +`),s("span",X,[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,[bs,qs,s("pre",gs,[s("code",null,[Ds,n(` +`),ks,n(` +`),Bs,n(` +`),s("span",Ps,[s("span",Is," chainId: '"+l(a.constants.arabicaChainId)+"',",1)]),n(` +`),Ss,n(` +`),s("span",vs,[s("span",Ns," rpc: '"+l(a.constants.arabicaRpcUrl)+"',",1)]),n(` +`),s("span",fs,[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",Gs,[s("span",Ys," chainId: '"+l(a.constants.arabicaChainId)+"',",1)]),n(` +`),Hs,n(` +`),s("span",Os,[s("span",Ws," rpc: '"+l(a.constants.arabicaRpcUrl)+"',",1)]),n(` +`),s("span",Zs,[s("span",Qs," rest: '"+l(a.constants.arabicaRestUrl)+"'",1)]),n(` +`),zs,n(` +`),Js,n(` +`),Xs])])])])]),$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.a3e6d0a6.lean.js b/assets/developers_wallets.md.a3e6d0a6.lean.js new file mode 100644 index 00000000000..bfb77db2268 --- /dev/null +++ b/assets/developers_wallets.md.a3e6d0a6.lean.js @@ -0,0 +1,55 @@ +import{c as p}from"./chunks/constants.fa173a21.js";import{_ as e,o as t,c,k as s,a as n,t as l,Q as o}from"./chunks/framework.1a91c06a.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"}},b=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}}," chainName: 'Celestia',")],-1),q={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),P=s("span",{class:"line"},null,-1),I=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),N=s("span",{class:"line"},null,-1),f=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),G=s("span",{class:"line"},null,-1),Y=o("",1),H={class:"language-js vp-adaptive-theme"},O=s("button",{title:"Copy Code",class:"copy"},null,-1),W=s("span",{class:"lang"},"js",-1),Z={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),J=o("",1),X={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"},bs=s("button",{title:"Copy Code",class:"copy"},null,-1),qs=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),Ps={class:"line"},Is={style:{color:"#9ECBFF"}},Ss=s("span",{class:"line"},[s("span",{style:{color:"#9ECBFF"}}," chainName: 'Arabica devnet',")],-1),vs={class:"line"},Ns={style:{color:"#9ECBFF"}},fs={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),Gs={class:"line"},Ys={style:{color:"#032F62"}},Hs=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}}," chainName: 'Arabica devnet',")],-1),Os={class:"line"},Ws={style:{color:"#032F62"}},Zs={class:"line"},Qs={style:{color:"#032F62"}},zs=s("span",{class:"line"},[s("span",{style:{color:"#032F62"}},"}`"),s("span",{style:{color:"#24292E"}},"}")],-1),Js=s("span",{class:"line"},null,-1),Xs=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(` +`),b,n(` +`),s("span",q,[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(` +`),P,n(` +`),I])]),s("pre",S,[s("code",null,[v,n(` +`),N,n(` +`),f,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(` +`),G,n(` +`),Y])])]),s("div",H,[O,W,s("pre",Z,[s("code",null,[Q,n(` +`),z,n(` +`),J,n(` +`),s("span",X,[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,[bs,qs,s("pre",gs,[s("code",null,[Ds,n(` +`),ks,n(` +`),Bs,n(` +`),s("span",Ps,[s("span",Is," chainId: '"+l(a.constants.arabicaChainId)+"',",1)]),n(` +`),Ss,n(` +`),s("span",vs,[s("span",Ns," rpc: '"+l(a.constants.arabicaRpcUrl)+"',",1)]),n(` +`),s("span",fs,[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",Gs,[s("span",Ys," chainId: '"+l(a.constants.arabicaChainId)+"',",1)]),n(` +`),Hs,n(` +`),s("span",Os,[s("span",Ws," rpc: '"+l(a.constants.arabicaRpcUrl)+"',",1)]),n(` +`),s("span",Zs,[s("span",Qs," rest: '"+l(a.constants.arabicaRestUrl)+"'",1)]),n(` +`),zs,n(` +`),Js,n(` +`),Xs])])])])]),$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.74361e6f.js b/assets/index.md.74361e6f.js new file mode 100644 index 00000000000..5073f63e1e9 --- /dev/null +++ b/assets/index.md.74361e6f.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a}from"./chunks/framework.1a91c06a.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 discord to connect, collaborate, and contribute to the future of modular blockchains.","link":"https://discord.gg/8affx48xyb","icon":"🏰"}],"head":[["meta",{"name":"og:title","content":"Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"index.md","filePath":"index.md","lastUpdated":1726693572000}'),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.74361e6f.lean.js b/assets/index.md.74361e6f.lean.js new file mode 100644 index 00000000000..5073f63e1e9 --- /dev/null +++ b/assets/index.md.74361e6f.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a}from"./chunks/framework.1a91c06a.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 discord to connect, collaborate, and contribute to the future of modular blockchains.","link":"https://discord.gg/8affx48xyb","icon":"🏰"}],"head":[["meta",{"name":"og:title","content":"Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"index.md","filePath":"index.md","lastUpdated":1726693572000}'),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.55daaf96.js b/assets/learn_how-celestia-works_data-availability-faq.md.55daaf96.js new file mode 100644 index 00000000000..e9cf1f390ae --- /dev/null +++ b/assets/learn_how-celestia-works_data-availability-faq.md.55daaf96.js @@ -0,0 +1 @@ +import{_ as a,o as t,c as e,Q as i}from"./chunks/framework.1a91c06a.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.55daaf96.lean.js b/assets/learn_how-celestia-works_data-availability-faq.md.55daaf96.lean.js new file mode 100644 index 00000000000..f3d66a8db1e --- /dev/null +++ b/assets/learn_how-celestia-works_data-availability-faq.md.55daaf96.lean.js @@ -0,0 +1 @@ +import{_ as a,o as t,c as e,Q as i}from"./chunks/framework.1a91c06a.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.648c5274.js b/assets/learn_how-celestia-works_data-availability-layer.md.648c5274.js new file mode 100644 index 00000000000..dcf73516b40 --- /dev/null +++ b/assets/learn_how-celestia-works_data-availability-layer.md.648c5274.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.1a91c06a.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.648c5274.lean.js b/assets/learn_how-celestia-works_data-availability-layer.md.648c5274.lean.js new file mode 100644 index 00000000000..35172f05fa4 --- /dev/null +++ b/assets/learn_how-celestia-works_data-availability-layer.md.648c5274.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.1a91c06a.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.80df84fc.js b/assets/learn_how-celestia-works_monolithic-vs-modular.md.80df84fc.js new file mode 100644 index 00000000000..a3e6da8f852 --- /dev/null +++ b/assets/learn_how-celestia-works_monolithic-vs-modular.md.80df84fc.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.1a91c06a.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.80df84fc.lean.js b/assets/learn_how-celestia-works_monolithic-vs-modular.md.80df84fc.lean.js new file mode 100644 index 00000000000..ff905081829 --- /dev/null +++ b/assets/learn_how-celestia-works_monolithic-vs-modular.md.80df84fc.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.1a91c06a.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.2b53d98a.js b/assets/learn_how-celestia-works_overview.md.2b53d98a.js new file mode 100644 index 00000000000..6bcf721c506 --- /dev/null +++ b/assets/learn_how-celestia-works_overview.md.2b53d98a.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as r}from"./chunks/framework.1a91c06a.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.2b53d98a.lean.js b/assets/learn_how-celestia-works_overview.md.2b53d98a.lean.js new file mode 100644 index 00000000000..df87b36dc82 --- /dev/null +++ b/assets/learn_how-celestia-works_overview.md.2b53d98a.lean.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as r}from"./chunks/framework.1a91c06a.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.56692f20.js b/assets/learn_how-celestia-works_transaction-lifecycle.md.56692f20.js new file mode 100644 index 00000000000..c5ad3f6c96c --- /dev/null +++ b/assets/learn_how-celestia-works_transaction-lifecycle.md.56692f20.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.1a91c06a.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.56692f20.lean.js b/assets/learn_how-celestia-works_transaction-lifecycle.md.56692f20.lean.js new file mode 100644 index 00000000000..95f46ae66d7 --- /dev/null +++ b/assets/learn_how-celestia-works_transaction-lifecycle.md.56692f20.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.1a91c06a.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.53e17214.js b/assets/learn_how-to-stake-tia.md.53e17214.js new file mode 100644 index 00000000000..6398c1ef0c9 --- /dev/null +++ b/assets/learn_how-to-stake-tia.md.53e17214.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as o}from"./chunks/framework.1a91c06a.js";const r="/img/keplr.png",l="/img/leap.png",i="/img/gem.png",n="/img/keplr/keplr1.gif",s="/img/keplr/keplr2.gif",p="/img/keplr/keplr3.gif",c="/img/keplr/keplr4.gif",h="/img/leap/leap1.gif",d="/img/leap/leap2.gif",m="/img/leap/leap3.gif",k="/img/gem/gem1.gif",w="/img/gem/gem2.gif",u="/img/gem/gem3.gif",f="/img/gem/gem4.gif",C=JSON.parse('{"title":"How to stake TIA","description":"This tutorial covers how to stake TIA with Keplr, Leap, or Gem wallet","frontmatter":{"sidebar_label":"How to stake TIA","description":"This tutorial covers how to stake TIA with Keplr, Leap, or Gem 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":1718652884000}'),g={name:"learn/how-to-stake-tia.md"},_=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, Leap, and Gem wallets.

    Select your preferred wallet

    Keplr
    Leap
    Gem Wallet

    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

    Stake TIA with Gem wallet

    1️⃣ Open your Gem Wallet app

    Navigate to Celestia and select Stake.

    Gem1

    2️⃣ Choose the amount of Celestia and search for a validator.

    Select the amount of Celestia tokens and choose a validator from the list.

    Gem2

    3️⃣ Stake your TIA tokens

    Review the network terms and commission, then press Confirm to proceed.

    Gem3

    4️⃣ Manage your TIA

    After your transaction is confirmed, you will have access to a control panel where you can claim rewards, unstake, redelegate, or stake additional tokens.

    Gem4

    ',48),b=[_];function y(v,T,A,I,S,q){return a(),t("div",null,b)}const K=e(g,[["render",y]]);export{C as __pageData,K as default}; diff --git a/assets/learn_how-to-stake-tia.md.53e17214.lean.js b/assets/learn_how-to-stake-tia.md.53e17214.lean.js new file mode 100644 index 00000000000..353d495d206 --- /dev/null +++ b/assets/learn_how-to-stake-tia.md.53e17214.lean.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as o}from"./chunks/framework.1a91c06a.js";const r="/img/keplr.png",l="/img/leap.png",i="/img/gem.png",n="/img/keplr/keplr1.gif",s="/img/keplr/keplr2.gif",p="/img/keplr/keplr3.gif",c="/img/keplr/keplr4.gif",h="/img/leap/leap1.gif",d="/img/leap/leap2.gif",m="/img/leap/leap3.gif",k="/img/gem/gem1.gif",w="/img/gem/gem2.gif",u="/img/gem/gem3.gif",f="/img/gem/gem4.gif",C=JSON.parse('{"title":"How to stake TIA","description":"This tutorial covers how to stake TIA with Keplr, Leap, or Gem wallet","frontmatter":{"sidebar_label":"How to stake TIA","description":"This tutorial covers how to stake TIA with Keplr, Leap, or Gem 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":1718652884000}'),g={name:"learn/how-to-stake-tia.md"},_=o("",48),b=[_];function y(v,T,A,I,S,q){return a(),t("div",null,b)}const K=e(g,[["render",y]]);export{C as __pageData,K as default}; diff --git a/assets/learn_paying-for-blobspace.md.7953d033.js b/assets/learn_paying-for-blobspace.md.7953d033.js new file mode 100644 index 00000000000..81f0126b763 --- /dev/null +++ b/assets/learn_paying-for-blobspace.md.7953d033.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.1a91c06a.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.7953d033.lean.js b/assets/learn_paying-for-blobspace.md.7953d033.lean.js new file mode 100644 index 00000000000..de4be0cffb8 --- /dev/null +++ b/assets/learn_paying-for-blobspace.md.7953d033.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.1a91c06a.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.1218f7e5.js b/assets/learn_retrievability.md.1218f7e5.js new file mode 100644 index 00000000000..d9dc0ba0d88 --- /dev/null +++ b/assets/learn_retrievability.md.1218f7e5.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as i}from"./chunks/framework.1a91c06a.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.1218f7e5.lean.js b/assets/learn_retrievability.md.1218f7e5.lean.js new file mode 100644 index 00000000000..c4d1787a17b --- /dev/null +++ b/assets/learn_retrievability.md.1218f7e5.lean.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as i}from"./chunks/framework.1a91c06a.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.acda4d48.js b/assets/learn_staking-governance-supply.md.acda4d48.js new file mode 100644 index 00000000000..286df2200f9 --- /dev/null +++ b/assets/learn_staking-governance-supply.md.acda4d48.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as o}from"./chunks/framework.1a91c06a.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":1725971347000}'),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.41%
    Future initiatives: 12.59%
    20.00%
    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.79%
    Early Backers: Series A&BEarly supporters of Celestia19.67%
    Early Backers: SeedEarly supporters of Celestia15.90%
    Initial Core ContributorsMembers of Celestia Labs, the first core contributor to Celestia17.64%

    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.00% unlocked at launch.
    Remaining 75.00% unlocks continuously from year 1 to year 4.
    Initial Core Contributors33.33% unlocked at year 1.
    Remaining 66.67% unlocks continuously from year 1 to year 3.
    Early Backers: Seed33.33% unlocked at year 1.
    Remaining 66.67% unlocks continuously from year 1 to year 2.
    Early Backers: Series A&B33.33% unlocked at year 1.
    Remaining 66.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.acda4d48.lean.js b/assets/learn_staking-governance-supply.md.acda4d48.lean.js new file mode 100644 index 00000000000..33f05758024 --- /dev/null +++ b/assets/learn_staking-governance-supply.md.acda4d48.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as o}from"./chunks/framework.1a91c06a.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":1725971347000}'),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.39685d72.js b/assets/learn_staking.md.39685d72.js new file mode 100644 index 00000000000..53d237dd073 --- /dev/null +++ b/assets/learn_staking.md.39685d72.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as n}from"./chunks/framework.1a91c06a.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":1726491136000}'),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.39685d72.lean.js b/assets/learn_staking.md.39685d72.lean.js new file mode 100644 index 00000000000..f5ec6ff67fc --- /dev/null +++ b/assets/learn_staking.md.39685d72.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as n}from"./chunks/framework.1a91c06a.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":1726491136000}'),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.a88f9d95.js b/assets/learn_tia.md.a88f9d95.js new file mode 100644 index 00000000000..74805fde930 --- /dev/null +++ b/assets/learn_tia.md.a88f9d95.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.1a91c06a.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.a88f9d95.lean.js b/assets/learn_tia.md.a88f9d95.lean.js new file mode 100644 index 00000000000..8cf054f259b --- /dev/null +++ b/assets/learn_tia.md.a88f9d95.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.1a91c06a.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.05307ae8.js b/assets/nodes_arabica-devnet.md.05307ae8.js new file mode 100644 index 00000000000..292c4176f41 --- /dev/null +++ b/assets/nodes_arabica-devnet.md.05307ae8.js @@ -0,0 +1,3 @@ +import{A as d}from"./chunks/ArabicaVersionTags.f0aaf832.js";import{c as p}from"./chunks/constants.fa173a21.js";import{_ as h,o,c as r,k as e,t as a,H as s,Q as n,a as u}from"./chunks/framework.1a91c06a.js";import"./chunks/arabica_versions.1930378b.js";const b="/img/arabica-devnet.png",g={name:"ArabicaDevnetDetails",data(){return{constants:p}}},y=e("tr",null,[e("th",null,"Detail"),e("th",null,"Value")],-1),m=e("td",null,"Chain ID",-1),f=e("tr",null,[e("td",null,"Genesis hash"),e("td",null,[e("code",null,"27122593765E07329BC348E8D16E92DCB4C75B34CCCB35C640FD7A4484D4C711")])],-1),E=e("td",null,"Genesis file",-1),v=["href"],k=e("td",null,"Peers file",-1),w=["href"],_=e("tr",null,[e("td",null,"Validators"),e("td",null," 4 ")],-1);function C(l,i,c,q,t,x){return o(),r("table",null,[y,e("tr",null,[m,e("td",null,[e("code",null,a(t.constants.arabicaChainId),1)])]),f,e("tr",null,[E,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,[k,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,w)])]),_])}const A=h(g,[["render",C]]),D=n('

    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 network upgrades upgrades in coordination with validators.

    Network details

    ',8),F=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),P=n(`

    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.

    Production RPC endpoints

    These RPC providers are meant to be used in production environments.

    ProviderURL
    NewMetrichttps://app.newmetric.xyz/start
    NumiaFor RPC access: https://docs.numia.xyz/overview/rpc-api-access
    NumiaFor data warehouse access: https://docs.numia.xyz/overview/sql-access/chains/celestia
    Grovehttps://www.grove.city/

    Community RPC endpoints

    WARNING

    Do not rely on the free community endpoints listed below for production deployments. Production deployments should rely on service providers with SLAs or your own node.

    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

    Data availability (DA) RPC endpoints for bridge node sync

    These RPC endpoints allow bridge nodes to sync blocks from the Celestia network. For users, they will need to provide a –core.ip string from a consensus node’s URL or IP that populates a default RPC port at 26657 to their respective DA node.

    Data availability (DA) gRPC endpoints for state access

    These gRPC endpoints for DA nodes 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 a default gRPC port at 9090 to their respective DA node.

    EXAMPLE

    bash
    celestia <da_type> start –core.ip <url> -–core.grpc.port <port>
    celestia <da_type> start –core.ip <url> -–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 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.

    See the network upgrade process page to learn more about specific upgrades like the Lemongrass network upgrade.

    `,35),B=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":1726494281000}'),T={name:"nodes/arabica-devnet.md"},L=Object.assign(T,{setup(l){return(i,c)=>(o(),r("div",null,[D,s(A),F,s(d),P]))}});export{B as __pageData,L as default}; diff --git a/assets/nodes_arabica-devnet.md.05307ae8.lean.js b/assets/nodes_arabica-devnet.md.05307ae8.lean.js new file mode 100644 index 00000000000..7492aee3b29 --- /dev/null +++ b/assets/nodes_arabica-devnet.md.05307ae8.lean.js @@ -0,0 +1 @@ +import{A as d}from"./chunks/ArabicaVersionTags.f0aaf832.js";import{c as p}from"./chunks/constants.fa173a21.js";import{_ as h,o,c as r,k as e,t as a,H as s,Q as n,a as u}from"./chunks/framework.1a91c06a.js";import"./chunks/arabica_versions.1930378b.js";const b="/img/arabica-devnet.png",g={name:"ArabicaDevnetDetails",data(){return{constants:p}}},y=e("tr",null,[e("th",null,"Detail"),e("th",null,"Value")],-1),m=e("td",null,"Chain ID",-1),f=e("tr",null,[e("td",null,"Genesis hash"),e("td",null,[e("code",null,"27122593765E07329BC348E8D16E92DCB4C75B34CCCB35C640FD7A4484D4C711")])],-1),E=e("td",null,"Genesis file",-1),v=["href"],k=e("td",null,"Peers file",-1),w=["href"],_=e("tr",null,[e("td",null,"Validators"),e("td",null," 4 ")],-1);function C(l,i,c,q,t,x){return o(),r("table",null,[y,e("tr",null,[m,e("td",null,[e("code",null,a(t.constants.arabicaChainId),1)])]),f,e("tr",null,[E,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,[k,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,w)])]),_])}const A=h(g,[["render",C]]),D=n("",8),F=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),P=n("",35),B=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":1726494281000}'),T={name:"nodes/arabica-devnet.md"},L=Object.assign(T,{setup(l){return(i,c)=>(o(),r("div",null,[D,s(A),F,s(d),P]))}});export{B as __pageData,L as default}; diff --git a/assets/nodes_bridge-node.md.1354e885.js b/assets/nodes_bridge-node.md.1354e885.js new file mode 100644 index 00000000000..1fcbdffd22e --- /dev/null +++ b/assets/nodes_bridge-node.md.1354e885.js @@ -0,0 +1,7 @@ +import{_ as s,o as e,c as a,Q as o}from"./chunks/framework.1a91c06a.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":1724873086000}'),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: 16 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.

    Connecting to a core endpoint with --core.ip string provides the light node with access to state queries (reading balances, submitting transactions, and other state-related queries).

    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.keyname <name-of-custom-key>
    celestia bridge start --core.ip <URI> --keyring.keyname <name-of-custom-key>
    sh
    celestia bridge start --core.ip <URI> --keyring.keyname <name-of-custom-key> \\
    +  --p2p.network mocha
    celestia bridge start --core.ip <URI> --keyring.keyname <name-of-custom-key> \\
    +  --p2p.network mocha
    sh
    celestia bridge start --core.ip <URI> --keyring.keyname <name-of-custom-key> \\
    +  --p2p.network arabica
    celestia bridge start --core.ip <URI> --keyring.keyname <name-of-custom-key> \\
    +  --p2p.network arabica

    Optional: Migrate node id to another server

    To migrate a bridge node ID:

    1. You need to back up two files located in the celestia-bridge node directory at the correct path (default: ~/.celestia-bridge/keys).
    2. Upload the files to the new server and start the node.

    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.

    `,53),p=[t];function r(c,i,d,y,E,h){return e(),a("div",null,p)}const b=s(l,[["render",r]]);export{u as __pageData,b as default}; diff --git a/assets/nodes_bridge-node.md.1354e885.lean.js b/assets/nodes_bridge-node.md.1354e885.lean.js new file mode 100644 index 00000000000..117697ffae0 --- /dev/null +++ b/assets/nodes_bridge-node.md.1354e885.lean.js @@ -0,0 +1 @@ +import{_ as s,o as e,c as a,Q as o}from"./chunks/framework.1a91c06a.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":1724873086000}'),l={name:"nodes/bridge-node.md"},t=o("",53),p=[t];function r(c,i,d,y,E,h){return e(),a("div",null,p)}const b=s(l,[["render",r]]);export{u as __pageData,b as default}; diff --git a/assets/nodes_celestia-app-commands.md.3c444cb8.js b/assets/nodes_celestia-app-commands.md.3c444cb8.js new file mode 100644 index 00000000000..0e46f72dd83 --- /dev/null +++ b/assets/nodes_celestia-app-commands.md.3c444cb8.js @@ -0,0 +1,213 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.1a91c06a.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.","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":1726491136000}'),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:443
    celestia-appd q bank balances celestia1czpgn3hdh9sodm06d5qk23xzgpq2uyc8ggdqgw \\
    +    --node https://rpc-mocha.pops.one:443

    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:443 --chain-id mocha
    celestia-appd tx bank send <FROM_ADDRESS> <TO_ADDRESS> \\
    +    19000000utia --node https://rpc-mocha.pops.one:443 --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.3c444cb8.lean.js b/assets/nodes_celestia-app-commands.md.3c444cb8.lean.js new file mode 100644 index 00000000000..65be34892a7 --- /dev/null +++ b/assets/nodes_celestia-app-commands.md.3c444cb8.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.1a91c06a.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.","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":1726491136000}'),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.97be623c.js b/assets/nodes_celestia-app-metrics.md.97be623c.js new file mode 100644 index 00000000000..8ddbf0c1228 --- /dev/null +++ b/assets/nodes_celestia-app-metrics.md.97be623c.js @@ -0,0 +1,83 @@ +import{_ as s,o as a,c as n,Q as e}from"./chunks/framework.1a91c06a.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.97be623c.lean.js b/assets/nodes_celestia-app-metrics.md.97be623c.lean.js new file mode 100644 index 00000000000..4041b49173c --- /dev/null +++ b/assets/nodes_celestia-app-metrics.md.97be623c.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as e}from"./chunks/framework.1a91c06a.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.6c014039.js b/assets/nodes_celestia-app-multisig.md.6c014039.js new file mode 100644 index 00000000000..7e9d46f0a7a --- /dev/null +++ b/assets/nodes_celestia-app-multisig.md.6c014039.js @@ -0,0 +1,103 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.1a91c06a.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.6c014039.lean.js b/assets/nodes_celestia-app-multisig.md.6c014039.lean.js new file mode 100644 index 00000000000..a29b8c2c4e6 --- /dev/null +++ b/assets/nodes_celestia-app-multisig.md.6c014039.lean.js @@ -0,0 +1 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.1a91c06a.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.d2c1a178.js b/assets/nodes_celestia-app-slashing.md.d2c1a178.js new file mode 100644 index 00000000000..bc5bc0e7338 --- /dev/null +++ b/assets/nodes_celestia-app-slashing.md.d2c1a178.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as i}from"./chunks/framework.1a91c06a.js";const f=JSON.parse('{"title":"Jailing and slashing on Celestia","description":"This section covers the jailing and slashing mechanics for validators in Celestia.","frontmatter":{"description":"This section covers the jailing and slashing mechanics for validators in Celestia.","head":[["meta",{"name":"og:title","content":"Jailing and slashing on Celestia | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-slashing.md","filePath":"nodes/celestia-app-slashing.md","lastUpdated":1726491136000}'),s={name:"nodes/celestia-app-slashing.md"},o=i('

    Jailing and 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 jailed or 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 getting slashed. 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 jailing and slashing parameters, refer to the celestia-app specifications page.

    ',7),n=[o];function l(r,d,h,c,p,g){return a(),t("div",null,n)}const u=e(s,[["render",l]]);export{f as __pageData,u as default}; diff --git a/assets/nodes_celestia-app-slashing.md.d2c1a178.lean.js b/assets/nodes_celestia-app-slashing.md.d2c1a178.lean.js new file mode 100644 index 00000000000..e17d538dc1c --- /dev/null +++ b/assets/nodes_celestia-app-slashing.md.d2c1a178.lean.js @@ -0,0 +1 @@ +import{_ as e,o as a,c as t,Q as i}from"./chunks/framework.1a91c06a.js";const f=JSON.parse('{"title":"Jailing and slashing on Celestia","description":"This section covers the jailing and slashing mechanics for validators in Celestia.","frontmatter":{"description":"This section covers the jailing and slashing mechanics for validators in Celestia.","head":[["meta",{"name":"og:title","content":"Jailing and slashing on Celestia | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-app-slashing.md","filePath":"nodes/celestia-app-slashing.md","lastUpdated":1726491136000}'),s={name:"nodes/celestia-app-slashing.md"},o=i("",7),n=[o];function l(r,d,h,c,p,g){return a(),t("div",null,n)}const u=e(s,[["render",l]]);export{f as __pageData,u as default}; diff --git a/assets/nodes_celestia-app-upgrade-monitor.md.9a019c51.js b/assets/nodes_celestia-app-upgrade-monitor.md.9a019c51.js new file mode 100644 index 00000000000..a17aa697162 --- /dev/null +++ b/assets/nodes_celestia-app-upgrade-monitor.md.9a019c51.js @@ -0,0 +1 @@ +import{_ as o,o as a,c as r,k as e,a as t}from"./chunks/framework.1a91c06a.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.9a019c51.lean.js b/assets/nodes_celestia-app-upgrade-monitor.md.9a019c51.lean.js new file mode 100644 index 00000000000..a17aa697162 --- /dev/null +++ b/assets/nodes_celestia-app-upgrade-monitor.md.9a019c51.lean.js @@ -0,0 +1 @@ +import{_ as o,o as a,c as r,k as e,a as t}from"./chunks/framework.1a91c06a.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.1d8992c8.js b/assets/nodes_celestia-app-vesting.md.1d8992c8.js new file mode 100644 index 00000000000..4d749ca4a4e --- /dev/null +++ b/assets/nodes_celestia-app-vesting.md.1d8992c8.js @@ -0,0 +1,215 @@ +import{_ as s,o as a,c as n,Q as o}from"./chunks/framework.1a91c06a.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":1726496322000}'),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 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.

    If you are running a production application, use a production endpoint.

    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 consensus node or validator

    Running a consensus node or validator will prevent you from needing to use an RPC.

    You can set up a validator or 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!

    `,99),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.1d8992c8.lean.js b/assets/nodes_celestia-app-vesting.md.1d8992c8.lean.js new file mode 100644 index 00000000000..b19b0f3eabe --- /dev/null +++ b/assets/nodes_celestia-app-vesting.md.1d8992c8.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as o}from"./chunks/framework.1a91c06a.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":1726496322000}'),l={name:"nodes/celestia-app-vesting.md"},p=o("",99),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.9beb14f0.js b/assets/nodes_celestia-app-wallet.md.9beb14f0.js new file mode 100644 index 00000000000..ec89086c088 --- /dev/null +++ b/assets/nodes_celestia-app-wallet.md.9beb14f0.js @@ -0,0 +1,3 @@ +import{_ as a,o as e,c as s,Q as t}from"./chunks/framework.1a91c06a.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":1726693572000}'),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.9beb14f0.lean.js b/assets/nodes_celestia-app-wallet.md.9beb14f0.lean.js new file mode 100644 index 00000000000..8ba4217cebe --- /dev/null +++ b/assets/nodes_celestia-app-wallet.md.9beb14f0.lean.js @@ -0,0 +1 @@ +import{_ as a,o as e,c as s,Q as t}from"./chunks/framework.1a91c06a.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":1726693572000}'),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.2a20c575.js b/assets/nodes_celestia-app.md.2a20c575.js new file mode 100644 index 00000000000..333e3887874 --- /dev/null +++ b/assets/nodes_celestia-app.md.2a20c575.js @@ -0,0 +1,7 @@ +import{a as o}from"./chunks/arabica_versions.1930378b.js";import{m as l}from"./chunks/mocha_versions.7704b055.js";import{m as n}from"./chunks/mainnet_versions.955428b6.js";import{o as p,c,k as s,t as e,l as t,Q as a}from"./chunks/framework.1a91c06a.js";const i=a('

    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=a(`
  • 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"},_=a('
    ',1),u={class:"blocks"},y={class:"language-bash vp-adaptive-theme active"},b=s("button",{title:"Copy Code",class:"copy"},null,-1),g=s("span",{class:"lang"},"bash",-1),m={class:"shiki github-dark vp-code-dark"},E={class:"line"},F=s("span",{style:{color:"#B392F0"}},"git",-1),C=s("span",{style:{color:"#E1E4E8"}}," ",-1),f=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),v=s("span",{style:{color:"#E1E4E8"}}," ",-1),k={style:{color:"#9ECBFF"}},T={class:"shiki github-light vp-code-light"},w={class:"line"},B=s("span",{style:{color:"#6F42C1"}},"git",-1),P=s("span",{style:{color:"#24292E"}}," ",-1),A=s("span",{style:{color:"#032F62"}},"checkout",-1),I=s("span",{style:{color:"#24292E"}}," ",-1),S={style:{color:"#032F62"}},q={class:"language-bash vp-adaptive-theme"},V=s("button",{title:"Copy Code",class:"copy"},null,-1),D=s("span",{class:"lang"},"bash",-1),N={class:"shiki github-dark vp-code-dark"},x={class:"line"},L=s("span",{style:{color:"#B392F0"}},"git",-1),M=s("span",{style:{color:"#E1E4E8"}}," ",-1),O=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),R=s("span",{style:{color:"#E1E4E8"}}," ",-1),$={style:{color:"#9ECBFF"}},j={class:"shiki github-light vp-code-light"},H={class:"line"},G=s("span",{style:{color:"#6F42C1"}},"git",-1),Q=s("span",{style:{color:"#24292E"}}," ",-1),Z=s("span",{style:{color:"#032F62"}},"checkout",-1),U=s("span",{style:{color:"#24292E"}}," ",-1),X={style:{color:"#032F62"}},Y={class:"language-bash vp-adaptive-theme"},J=s("button",{title:"Copy Code",class:"copy"},null,-1),W=s("span",{class:"lang"},"bash",-1),z={class:"shiki github-dark vp-code-dark"},K={class:"line"},ss=s("span",{style:{color:"#B392F0"}},"git",-1),es=s("span",{style:{color:"#E1E4E8"}}," ",-1),ts=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),as=s("span",{style:{color:"#E1E4E8"}}," ",-1),os={style:{color:"#9ECBFF"}},ls={class:"shiki github-light vp-code-light"},ns={class:"line"},ps=s("span",{style:{color:"#6F42C1"}},"git",-1),cs=s("span",{style:{color:"#24292E"}}," ",-1),is=s("span",{style:{color:"#032F62"}},"checkout",-1),rs=s("span",{style:{color:"#24292E"}}," ",-1),ds={style:{color:"#032F62"}},hs=a('
  • 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),_s=a('

    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),fs=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":1725369955000}'),us={name:"nodes/celestia-app.md"},vs=Object.assign(us,{setup(ys){return(bs,gs)=>(p(),c("div",null,[i,s("ol",null,[r,s("li",null,[d,s("div",h,[_,s("div",u,[s("div",y,[b,g,s("pre",m,[s("code",null,[s("span",E,[F,C,f,v,s("span",k,"tags/"+e(t(n)["app-latest-tag"]),1)])])]),s("pre",T,[s("code",null,[s("span",w,[B,P,A,I,s("span",S,"tags/"+e(t(n)["app-latest-tag"]),1)])])])]),s("div",q,[V,D,s("pre",N,[s("code",null,[s("span",x,[L,M,O,R,s("span",$,"tags/"+e(t(l)["app-latest-tag"]),1)])])]),s("pre",j,[s("code",null,[s("span",H,[G,Q,Z,U,s("span",X,"tags/"+e(t(l)["app-latest-tag"]),1)])])])]),s("div",Y,[J,W,s("pre",z,[s("code",null,[s("span",K,[ss,es,ts,as,s("span",os,"tags/"+e(t(o)["app-latest-tag"]),1)])])]),s("pre",ls,[s("code",null,[s("span",ns,[ps,cs,is,rs,s("span",ds,"tags/"+e(t(o)["app-latest-tag"]),1)])])])])])])]),hs]),_s]))}});export{fs as __pageData,vs as default}; diff --git a/assets/nodes_celestia-app.md.2a20c575.lean.js b/assets/nodes_celestia-app.md.2a20c575.lean.js new file mode 100644 index 00000000000..c52efef2f5f --- /dev/null +++ b/assets/nodes_celestia-app.md.2a20c575.lean.js @@ -0,0 +1 @@ +import{a as o}from"./chunks/arabica_versions.1930378b.js";import{m as l}from"./chunks/mocha_versions.7704b055.js";import{m as n}from"./chunks/mainnet_versions.955428b6.js";import{o as p,c,k as s,t as e,l as t,Q as a}from"./chunks/framework.1a91c06a.js";const i=a("",5),r=a("",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"},_=a("",1),u={class:"blocks"},y={class:"language-bash vp-adaptive-theme active"},b=s("button",{title:"Copy Code",class:"copy"},null,-1),g=s("span",{class:"lang"},"bash",-1),m={class:"shiki github-dark vp-code-dark"},E={class:"line"},F=s("span",{style:{color:"#B392F0"}},"git",-1),C=s("span",{style:{color:"#E1E4E8"}}," ",-1),f=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),v=s("span",{style:{color:"#E1E4E8"}}," ",-1),k={style:{color:"#9ECBFF"}},T={class:"shiki github-light vp-code-light"},w={class:"line"},B=s("span",{style:{color:"#6F42C1"}},"git",-1),P=s("span",{style:{color:"#24292E"}}," ",-1),A=s("span",{style:{color:"#032F62"}},"checkout",-1),I=s("span",{style:{color:"#24292E"}}," ",-1),S={style:{color:"#032F62"}},q={class:"language-bash vp-adaptive-theme"},V=s("button",{title:"Copy Code",class:"copy"},null,-1),D=s("span",{class:"lang"},"bash",-1),N={class:"shiki github-dark vp-code-dark"},x={class:"line"},L=s("span",{style:{color:"#B392F0"}},"git",-1),M=s("span",{style:{color:"#E1E4E8"}}," ",-1),O=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),R=s("span",{style:{color:"#E1E4E8"}}," ",-1),$={style:{color:"#9ECBFF"}},j={class:"shiki github-light vp-code-light"},H={class:"line"},G=s("span",{style:{color:"#6F42C1"}},"git",-1),Q=s("span",{style:{color:"#24292E"}}," ",-1),Z=s("span",{style:{color:"#032F62"}},"checkout",-1),U=s("span",{style:{color:"#24292E"}}," ",-1),X={style:{color:"#032F62"}},Y={class:"language-bash vp-adaptive-theme"},J=s("button",{title:"Copy Code",class:"copy"},null,-1),W=s("span",{class:"lang"},"bash",-1),z={class:"shiki github-dark vp-code-dark"},K={class:"line"},ss=s("span",{style:{color:"#B392F0"}},"git",-1),es=s("span",{style:{color:"#E1E4E8"}}," ",-1),ts=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),as=s("span",{style:{color:"#E1E4E8"}}," ",-1),os={style:{color:"#9ECBFF"}},ls={class:"shiki github-light vp-code-light"},ns={class:"line"},ps=s("span",{style:{color:"#6F42C1"}},"git",-1),cs=s("span",{style:{color:"#24292E"}}," ",-1),is=s("span",{style:{color:"#032F62"}},"checkout",-1),rs=s("span",{style:{color:"#24292E"}}," ",-1),ds={style:{color:"#032F62"}},hs=a("",2),_s=a("",18),fs=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":1725369955000}'),us={name:"nodes/celestia-app.md"},vs=Object.assign(us,{setup(ys){return(bs,gs)=>(p(),c("div",null,[i,s("ol",null,[r,s("li",null,[d,s("div",h,[_,s("div",u,[s("div",y,[b,g,s("pre",m,[s("code",null,[s("span",E,[F,C,f,v,s("span",k,"tags/"+e(t(n)["app-latest-tag"]),1)])])]),s("pre",T,[s("code",null,[s("span",w,[B,P,A,I,s("span",S,"tags/"+e(t(n)["app-latest-tag"]),1)])])])]),s("div",q,[V,D,s("pre",N,[s("code",null,[s("span",x,[L,M,O,R,s("span",$,"tags/"+e(t(l)["app-latest-tag"]),1)])])]),s("pre",j,[s("code",null,[s("span",H,[G,Q,Z,U,s("span",X,"tags/"+e(t(l)["app-latest-tag"]),1)])])])]),s("div",Y,[J,W,s("pre",z,[s("code",null,[s("span",K,[ss,es,ts,as,s("span",os,"tags/"+e(t(o)["app-latest-tag"]),1)])])]),s("pre",ls,[s("code",null,[s("span",ns,[ps,cs,is,rs,s("span",ds,"tags/"+e(t(o)["app-latest-tag"]),1)])])])])])])]),hs]),_s]))}});export{fs as __pageData,vs as default}; diff --git a/assets/nodes_celestia-node-custom-networks.md.c55d089b.js b/assets/nodes_celestia-node-custom-networks.md.c55d089b.js new file mode 100644 index 00000000000..7ec0614d570 --- /dev/null +++ b/assets/nodes_celestia-node-custom-networks.md.c55d089b.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as o,Q as n}from"./chunks/framework.1a91c06a.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.c55d089b.lean.js b/assets/nodes_celestia-node-custom-networks.md.c55d089b.lean.js new file mode 100644 index 00000000000..ca5e0c51c51 --- /dev/null +++ b/assets/nodes_celestia-node-custom-networks.md.c55d089b.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as o,Q as n}from"./chunks/framework.1a91c06a.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.b99ab7c2.js b/assets/nodes_celestia-node-metrics.md.b99ab7c2.js new file mode 100644 index 00000000000..d7865755d57 --- /dev/null +++ b/assets/nodes_celestia-node-metrics.md.b99ab7c2.js @@ -0,0 +1,29 @@ +import{_ as s,o as n,c as a,Q as o}from"./chunks/framework.1a91c06a.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.b99ab7c2.lean.js b/assets/nodes_celestia-node-metrics.md.b99ab7c2.lean.js new file mode 100644 index 00000000000..a1472b4e45b --- /dev/null +++ b/assets/nodes_celestia-node-metrics.md.b99ab7c2.lean.js @@ -0,0 +1 @@ +import{_ as s,o as n,c as a,Q as o}from"./chunks/framework.1a91c06a.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.6777331f.js b/assets/nodes_celestia-node-troubleshooting.md.6777331f.js new file mode 100644 index 00000000000..b8b5815e415 --- /dev/null +++ b/assets/nodes_celestia-node-troubleshooting.md.6777331f.js @@ -0,0 +1,61 @@ +import{c as n}from"./chunks/constants.fa173a21.js";import{o as p,c as t,k as s,a,t as o,l,Q as e}from"./chunks/framework.1a91c06a.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),u=s("td",null,"Arabica",-1),g=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

    Background

    An enhancement has been made in v0.14.0+ to automate the detection of the running node, eliminating the need to manually specify the --node.store flag for each RPC request.

    Assumptions:

    • The presence of a lock signifies a running node.
    • Networks are ordered as Mainnet Beta, Mocha, Arabica, private, custom.
    • Node types are ordered as bridge, full, and light.
    • Each network has only one running node type.
    • Multiple nodes of the same network and type are prohibited (resulting in an Error: node: store is in use).

    Key Points:

    • Authentication token and other flags maintain their previous behavior and take precedence.
    • Address and port details are fetched from the configuration.
    • skipAuth allows bypassing authentication for trusted setups and follows Unix daemon conventions.
    • Non-default node store and cel-key configurations still require specific flags in the configuration settings.

    Demonstration

    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/

    Examples

    Mainnet Beta full and Mocha light

    This example uses a Mainnet Beta full node and a Mocha light node. When making the request:

    bash
     celestia blob get 1318129 0x42690c204d39600fddd3 0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=
    +{
    + "result": "RPC client error: sendRequest failed: http status 401 Unauthorized unmarshaling response: EOF"
    +}
     celestia blob get 1318129 0x42690c204d39600fddd3 0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=
    +{
    + "result": "RPC client error: sendRequest failed: http status 401 Unauthorized unmarshaling response: EOF"
    +}

    The request will go to the Mainnet Beta node, and a 401 will show in this node's logs. Note that a 401 is expected because this blob was posted to Mocha and neither the namespace nor the blob exist on Mainnet Beta.

    Mocha full and Arabica light

    This example uses a Mocha full node and an Arabica light node. When making the request:

    bash
     celestia blob get 1318129 0x42690c204d39600fddd3 0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=
    +{
    +  "result": {
    +    "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +    "data": "0x676d",
    +    "share_version": 0,
    +    "commitment": "0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=",
    +    "index": 23
    +  }
    +}
     celestia blob get 1318129 0x42690c204d39600fddd3 0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=
    +{
    +  "result": {
    +    "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +    "data": "0x676d",
    +    "share_version": 0,
    +    "commitment": "0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=",
    +    "index": 23
    +  }
    +}

    The request will go to the Mocha full node, and result shown as expected.

    Using a custom rpc.config address

    When using a custom RPC config address 0.0.0.1 and port 25231, the CLI accurately routes to the custom address and port, where no node is running. It fails as expected:

    bash
     celestia blob get 1318129 0x42690c204d39600fddd3 0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=
    +{
    +  "result": "RPC client error: sendRequest failed: Post \\"http://0.0.0.1:25231\\": dial tcp 0.0.0.1:25231: connect: no route to host"
    +}
     celestia blob get 1318129 0x42690c204d39600fddd3 0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=
    +{
    +  "result": "RPC client error: sendRequest failed: Post \\"http://0.0.0.1:25231\\": dial tcp 0.0.0.1:25231: connect: no route to host"
    +}

    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
    `,70),A=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":1726496322000}'),b={name:"nodes/celestia-node-troubleshooting.md"},w=Object.assign(b,{setup(B){return(m,k)=>(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 "+o(l(n).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,o(l(n).mainnetChainId),1),d]),s("tr",null,[h,s("td",null,o(l(n).mochaChainId),1),F]),s("tr",null,[u,s("td",null,o(l(n).arabicaChainId),1),g])])]),C]))}});export{A as __pageData,w as default}; diff --git a/assets/nodes_celestia-node-troubleshooting.md.6777331f.lean.js b/assets/nodes_celestia-node-troubleshooting.md.6777331f.lean.js new file mode 100644 index 00000000000..a35e34f1fac --- /dev/null +++ b/assets/nodes_celestia-node-troubleshooting.md.6777331f.lean.js @@ -0,0 +1 @@ +import{c as n}from"./chunks/constants.fa173a21.js";import{o as p,c as t,k as s,a,t as o,l,Q as e}from"./chunks/framework.1a91c06a.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),u=s("td",null,"Arabica",-1),g=s("td",null,[s("code",null,"--p2p.network arabica")],-1),C=e("",70),A=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":1726496322000}'),b={name:"nodes/celestia-node-troubleshooting.md"},w=Object.assign(b,{setup(B){return(m,k)=>(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 "+o(l(n).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,o(l(n).mainnetChainId),1),d]),s("tr",null,[h,s("td",null,o(l(n).mochaChainId),1),F]),s("tr",null,[u,s("td",null,o(l(n).arabicaChainId),1),g])])]),C]))}});export{A as __pageData,w as default}; diff --git a/assets/nodes_celestia-node-trusted-hash.md.9ca1f0ed.js b/assets/nodes_celestia-node-trusted-hash.md.9ca1f0ed.js new file mode 100644 index 00000000000..cc216459a88 --- /dev/null +++ b/assets/nodes_celestia-node-trusted-hash.md.9ca1f0ed.js @@ -0,0 +1,3 @@ +import{_ as s,o as a,c as e,Q as o}from"./chunks/framework.1a91c06a.js";const g=JSON.parse('{"title":"Syncing a light node from a trusted hash","description":"How to sync a light node from a trusted hash.","frontmatter":{"description":"How to sync a light node from a trusted hash.","head":[["meta",{"name":"og:title","content":"Syncing a light node from a trusted hash | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-node-trusted-hash.md","filePath":"nodes/celestia-node-trusted-hash.md","lastUpdated":1723038028000}'),t={name:"nodes/celestia-node-trusted-hash.md"},n=o(`

    Syncing a light node from a trusted hash

    This guide goes over how to sync a DA light node from a trusted hash. The example is with Mainnet Beta. You will need to adjust the commands accordingly for Mocha, Arabica, or a custom network.

    WARNING

    Syncing to a trusted hash means that you will not sample the entire chain. This adds a trust assumption that you trust the history of the chain up to that point and that you trust the entity where you get the hash from. In this example, the trusted entity is a consensus endpoint or Celenium

    1. Get trusted height & hash from a consensus endpoint or Celenium.

    2. Initialize the node store

      sh
      celestia light init --p2p.network <network>
      celestia light init --p2p.network <network>
    3. Set the trusted height & hash

      1. Open your config.toml at .celestia-light/config.toml (or .celestia-light-<other-network>/config.toml)
      2. Set DASer.SampleFrom to the trusted height (e.g. SampleFrom = 123456)
    4. Run the node with the hash and flag:

    sh
    celestia light start --headers.trusted-hash <hash_of_block_n> \\
    +    --p2p.network <network> --core.ip <consensus-node-rpc>
    celestia light start --headers.trusted-hash <hash_of_block_n> \\
    +    --p2p.network <network> --core.ip <consensus-node-rpc>

    For service operators

    If you're using multiple light nodes for similar services like tracking the same rollup, it is recommended to use the same hash and height for them all services using the same starting height.

    `,7),l=[n];function p(r,c,i,h,d,y){return a(),e("div",null,l)}const u=s(t,[["render",p]]);export{g as __pageData,u as default}; diff --git a/assets/nodes_celestia-node-trusted-hash.md.9ca1f0ed.lean.js b/assets/nodes_celestia-node-trusted-hash.md.9ca1f0ed.lean.js new file mode 100644 index 00000000000..c6961b8328b --- /dev/null +++ b/assets/nodes_celestia-node-trusted-hash.md.9ca1f0ed.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as e,Q as o}from"./chunks/framework.1a91c06a.js";const g=JSON.parse('{"title":"Syncing a light node from a trusted hash","description":"How to sync a light node from a trusted hash.","frontmatter":{"description":"How to sync a light node from a trusted hash.","head":[["meta",{"name":"og:title","content":"Syncing a light node from a trusted hash | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/celestia-node-trusted-hash.md","filePath":"nodes/celestia-node-trusted-hash.md","lastUpdated":1723038028000}'),t={name:"nodes/celestia-node-trusted-hash.md"},n=o("",7),l=[n];function p(r,c,i,h,d,y){return a(),e("div",null,l)}const u=s(t,[["render",p]]);export{g as __pageData,u as default}; diff --git a/assets/nodes_celestia-node.md.727b45ab.js b/assets/nodes_celestia-node.md.727b45ab.js new file mode 100644 index 00000000000..2a5564edebf --- /dev/null +++ b/assets/nodes_celestia-node.md.727b45ab.js @@ -0,0 +1,7 @@ +import{a as t}from"./chunks/arabica_versions.1930378b.js";import{m as l}from"./chunks/mocha_versions.7704b055.js";import{m as n}from"./chunks/mainnet_versions.955428b6.js";import{o as i,c,k as s,t as e,l as a,Q as o}from"./chunks/framework.1a91c06a.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),P=s("span",{style:{color:"#032F62"}},"checkout",-1),S=s("span",{style:{color:"#24292E"}}," ",-1),A={style:{color:"#032F62"}},x={class:"language-bash vp-adaptive-theme"},N=s("button",{title:"Copy Code",class:"copy"},null,-1),V=s("span",{class:"lang"},"bash",-1),q={class:"shiki github-dark vp-code-dark"},D={class:"line"},O=s("span",{style:{color:"#B392F0"}},"git",-1),R=s("span",{style:{color:"#E1E4E8"}}," ",-1),M=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),L=s("span",{style:{color:"#E1E4E8"}}," ",-1),j={style:{color:"#9ECBFF"}},G={class:"shiki github-light vp-code-light"},$={class:"line"},H=s("span",{style:{color:"#6F42C1"}},"git",-1),U=s("span",{style:{color:"#24292E"}}," ",-1),Z=s("span",{style:{color:"#032F62"}},"checkout",-1),J=s("span",{style:{color:"#24292E"}}," ",-1),Q={style:{color:"#032F62"}},Y={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,P,S,s("span",A,"tags/"+e(a(n)["node-latest-tag"]),1)])])])]),s("div",x,[N,V,s("pre",q,[s("code",null,[s("span",D,[O,R,M,L,s("span",j,"tags/"+e(a(l)["node-latest-tag"]),1)])])]),s("pre",G,[s("code",null,[s("span",$,[H,U,Z,J,s("span",Q,"tags/"+e(a(l)["node-latest-tag"]),1)])])])]),s("div",Y,[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.727b45ab.lean.js b/assets/nodes_celestia-node.md.727b45ab.lean.js new file mode 100644 index 00000000000..5931aa80004 --- /dev/null +++ b/assets/nodes_celestia-node.md.727b45ab.lean.js @@ -0,0 +1 @@ +import{a as t}from"./chunks/arabica_versions.1930378b.js";import{m as l}from"./chunks/mocha_versions.7704b055.js";import{m as n}from"./chunks/mainnet_versions.955428b6.js";import{o as i,c,k as s,t as e,l as a,Q as o}from"./chunks/framework.1a91c06a.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),P=s("span",{style:{color:"#032F62"}},"checkout",-1),S=s("span",{style:{color:"#24292E"}}," ",-1),A={style:{color:"#032F62"}},x={class:"language-bash vp-adaptive-theme"},N=s("button",{title:"Copy Code",class:"copy"},null,-1),V=s("span",{class:"lang"},"bash",-1),q={class:"shiki github-dark vp-code-dark"},D={class:"line"},O=s("span",{style:{color:"#B392F0"}},"git",-1),R=s("span",{style:{color:"#E1E4E8"}}," ",-1),M=s("span",{style:{color:"#9ECBFF"}},"checkout",-1),L=s("span",{style:{color:"#E1E4E8"}}," ",-1),j={style:{color:"#9ECBFF"}},G={class:"shiki github-light vp-code-light"},$={class:"line"},H=s("span",{style:{color:"#6F42C1"}},"git",-1),U=s("span",{style:{color:"#24292E"}}," ",-1),Z=s("span",{style:{color:"#032F62"}},"checkout",-1),J=s("span",{style:{color:"#24292E"}}," ",-1),Q={style:{color:"#032F62"}},Y={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,P,S,s("span",A,"tags/"+e(a(n)["node-latest-tag"]),1)])])])]),s("div",x,[N,V,s("pre",q,[s("code",null,[s("span",D,[O,R,M,L,s("span",j,"tags/"+e(a(l)["node-latest-tag"]),1)])])]),s("pre",G,[s("code",null,[s("span",$,[H,U,Z,J,s("span",Q,"tags/"+e(a(l)["node-latest-tag"]),1)])])])]),s("div",Y,[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.b1d84050.js b/assets/nodes_config-toml.md.b1d84050.js new file mode 100644 index 00000000000..c0798b8db4b --- /dev/null +++ b/assets/nodes_config-toml.md.b1d84050.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.1a91c06a.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.b1d84050.lean.js b/assets/nodes_config-toml.md.b1d84050.lean.js new file mode 100644 index 00000000000..0f9776d6d28 --- /dev/null +++ b/assets/nodes_config-toml.md.b1d84050.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as o,Q as a}from"./chunks/framework.1a91c06a.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_consensus-node.md.b07b70a7.js b/assets/nodes_consensus-node.md.b07b70a7.js new file mode 100644 index 00000000000..d1d1685211a --- /dev/null +++ b/assets/nodes_consensus-node.md.b07b70a7.js @@ -0,0 +1,77 @@ +import{c as e}from"./chunks/constants.fa173a21.js";import{o as l,c as p,k as s,t as n,l as t,a,Q as o}from"./chunks/framework.1a91c06a.js";const c="/img/nodes/consensus-node.jpg",r=o('

    Consensus node

    This guide covers how to set up a consensus node on Celestia. Consensus nodes allow you to sync the entire blockchain history in the Celestia consensus layer.

    consensus node

    Minimum hardware requirements

    The following minimum hardware requirements are recommended for running a consensus node:

    • Memory: 16 GB RAM
    • CPU: Quad-Core
    • Disk: 2 TB SSD Storage
    • Bandwidth: 1 Gbps for Download/1 Gbps for Upload

    Set up a 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

    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.

    ',14),i={class:"vp-code-group vp-adaptive-theme"},d=o('
    ',1),_={class:"blocks"},y={class:"language-bash vp-adaptive-theme active"},h=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"},m=o('celestia-appd init "node-name" --chain-id ',8),S={style:{color:"#032F62"}},v={class:"language-bash vp-adaptive-theme"},k=s("button",{title:"Copy Code",class:"copy"},null,-1),B=s("span",{class:"lang"},"bash",-1),A={class:"shiki github-dark vp-code-dark"},f={class:"line"},P=o('celestia-appd init "node-name" --chain-id ',8),q={style:{color:"#9ECBFF"}},I={class:"shiki github-light vp-code-light"},x={class:"line"},V=o('celestia-appd init "node-name" --chain-id ',8),R={style:{color:"#032F62"}},D={class:"language-bash vp-adaptive-theme"},w=s("button",{title:"Copy Code",class:"copy"},null,-1),N=s("span",{class:"lang"},"bash",-1),$={class:"shiki github-dark vp-code-dark"},O={class:"line"},M=o('celestia-appd init "node-name" --chain-id ',8),H={style:{color:"#9ECBFF"}},L={class:"shiki github-light vp-code-light"},J={class:"line"},Q=o('celestia-appd init "node-name" --chain-id ',8),j={style:{color:"#032F62"}},U=s("p",null,[a("Download the "),s("code",null,"genesis.json"),a(" file:")],-1),Y={class:"vp-code-group vp-adaptive-theme"},Z=o('
    ',1),z={class:"blocks"},G={class:"language-bash vp-adaptive-theme active"},X=s("button",{title:"Copy Code",class:"copy"},null,-1),K=s("span",{class:"lang"},"bash",-1),W={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),es=s("span",{style:{color:"#9ECBFF"}},"download-genesis",-1),ns=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),rs=s("span",{style:{color:"#24292E"}}," ",-1),is=s("span",{style:{color:"#032F62"}},"download-genesis",-1),ds=s("span",{style:{color:"#24292E"}}," ",-1),_s={style:{color:"#032F62"}},ys={class:"language-bash vp-adaptive-theme"},hs=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),ms={style:{color:"#9ECBFF"}},Ss={class:"shiki github-light vp-code-light"},vs={class:"line"},ks=s("span",{style:{color:"#6F42C1"}},"celestia-appd",-1),Bs=s("span",{style:{color:"#24292E"}}," ",-1),As=s("span",{style:{color:"#032F62"}},"download-genesis",-1),fs=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),xs=s("span",{class:"lang"},"bash",-1),Vs={class:"shiki github-dark vp-code-dark"},Rs={class:"line"},Ds=s("span",{style:{color:"#B392F0"}},"celestia-appd",-1),ws=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ns=s("span",{style:{color:"#9ECBFF"}},"download-genesis",-1),$s=s("span",{style:{color:"#E1E4E8"}}," ",-1),Os={style:{color:"#9ECBFF"}},Ms={class:"shiki github-light vp-code-light"},Hs={class:"line"},Ls=s("span",{style:{color:"#6F42C1"}},"celestia-appd",-1),Js=s("span",{style:{color:"#24292E"}}," ",-1),Qs=s("span",{style:{color:"#032F62"}},"download-genesis",-1),js=s("span",{style:{color:"#24292E"}}," ",-1),Us={style:{color:"#032F62"}},Ys=s("p",null,[a("Set seeds in the "),s("code",null,"$HOME/.celestia-app/config/config.toml"),a(" file:")],-1),Zs={class:"vp-code-group vp-adaptive-theme"},zs=o('
    ',1),Gs={class:"blocks"},Xs={class:"language-bash vp-adaptive-theme active"},Ks=s("button",{title:"Copy Code",class:"copy"},null,-1),Ws=s("span",{class:"lang"},"bash",-1),so={class:"shiki github-dark vp-code-dark"},oo={class:"line"},ao=o('SEEDS=$(curl -sL',6),eo={style:{color:"#9ECBFF"}},no=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),ro=o('sed -i.bak -e "s/^seeds *=.*/seeds = \\"$SEEDS\\"/" $HOME/.celestia-app/config/config.toml',1),io={class:"shiki github-light vp-code-light"},_o={class:"line"},yo=o('SEEDS=$(curl -sL',6),ho={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"},mo=s("button",{title:"Copy Code",class:"copy"},null,-1),So=s("span",{class:"lang"},"bash",-1),vo={class:"shiki github-dark vp-code-dark"},ko={class:"line"},Bo=o('SEEDS=$(curl -sL',6),Ao={style:{color:"#9ECBFF"}},fo=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),xo=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $SEEDS")],-1),Vo=o('sed -i.bak -e "s/^seeds *=.*/seeds = \\"$SEEDS\\"/" $HOME/.celestia-app/config/config.toml',1),Ro={class:"shiki github-light vp-code-light"},Do={class:"line"},wo=o('SEEDS=$(curl -sL',6),No={style:{color:"#032F62"}},$o=s("span",{style:{color:"#D73A49"}},"|",-1),Oo=s("span",{style:{color:"#032F62"}}," ",-1),Mo=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),Jo=o('sed -i.bak -e "s/^seeds *=.*/seeds = \\"$SEEDS\\"/" $HOME/.celestia-app/config/config.toml',1),Qo=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),jo=o('Optional: Set persistent peers

    Optionally, you can set persistent peers in your config.toml file. If you set persistent peers, your node will always try to connect to these peers. This is useful when running a local devnet, for example, when you would always want to connect to the same local nodes in your devnet. In production, setting persistent peers is advised only if you are running a sentry node.

    You can get the persistent peers from the @cosmos/chain-registry repository (for Mainnet Beta) or @celestiaorg/networks repository repo (for Mocha and Arabica) with the following commands:

    ',3),Uo={class:"vp-code-group vp-adaptive-theme"},Yo=o('
    ',1),Zo={class:"blocks"},zo={class:"language-bash vp-adaptive-theme active"},Go=s("button",{title:"Copy Code",class:"copy"},null,-1),Xo=s("span",{class:"lang"},"bash",-1),Ko={class:"shiki github-dark vp-code-dark"},Wo={class:"line"},sa=o('PERSISTENT_PEERS=$(curl -s',6),oa={style:{color:"#9ECBFF"}},aa=o('| jq -r '.peers.persistent_peers[].address' | tr '\\n' ',' | sed 's/,$/\\n/')',14),ea=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $PERSISTENT_PEERS")],-1),na=o('sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \\"$PERSISTENT_PEERS\\"/" $HOME/.celestia-app/config/config.toml',1),ta={class:"shiki github-light vp-code-light"},la={class:"line"},pa=o('PERSISTENT_PEERS=$(curl -s',6),ca={style:{color:"#032F62"}},ra=o('| jq -r '.peers.persistent_peers[].address' | tr '\\n' ',' | sed 's/,$/\\n/')',14),ia=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $PERSISTENT_PEERS")],-1),da=o('sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \\"$PERSISTENT_PEERS\\"/" $HOME/.celestia-app/config/config.toml',1),_a={class:"language-bash vp-adaptive-theme"},ya=s("button",{title:"Copy Code",class:"copy"},null,-1),ha=s("span",{class:"lang"},"bash",-1),Ea={class:"shiki github-dark vp-code-dark"},ua={class:"line"},Fa=o('PERSISTENT_PEERS=$(curl -sL',6),Ca={style:{color:"#9ECBFF"}},ga=s("span",{style:{color:"#F97583"}},"|",-1),Ta=s("span",{style:{color:"#9ECBFF"}}," ",-1),ba=s("span",{style:{color:"#B392F0"}},"tr",-1),ma=s("span",{style:{color:"#9ECBFF"}}," '\\n' ',')",-1),Sa=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $PERSISTENT_PEERS")],-1),va=o('sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \\"$PERSISTENT_PEERS\\"/" $HOME/.celestia-app/config/config.toml',1),ka={class:"shiki github-light vp-code-light"},Ba={class:"line"},Aa=o('PERSISTENT_PEERS=$(curl -sL',6),fa={style:{color:"#032F62"}},Pa=s("span",{style:{color:"#D73A49"}},"|",-1),qa=s("span",{style:{color:"#032F62"}}," ",-1),Ia=s("span",{style:{color:"#6F42C1"}},"tr",-1),xa=s("span",{style:{color:"#032F62"}}," '\\n' ',')",-1),Va=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $PERSISTENT_PEERS")],-1),Ra=o('sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \\"$PERSISTENT_PEERS\\"/" $HOME/.celestia-app/config/config.toml',1),Da={class:"language-bash vp-adaptive-theme"},wa=s("button",{title:"Copy Code",class:"copy"},null,-1),Na=s("span",{class:"lang"},"bash",-1),$a={class:"shiki github-dark vp-code-dark"},Oa={class:"line"},Ma=o('PERSISTENT_PEERS=$(curl -sL',6),Ha={style:{color:"#9ECBFF"}},La=s("span",{style:{color:"#F97583"}},"|",-1),Ja=s("span",{style:{color:"#9ECBFF"}}," ",-1),Qa=s("span",{style:{color:"#B392F0"}},"tr",-1),ja=s("span",{style:{color:"#9ECBFF"}}," '\\n' ',')",-1),Ua=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $PERSISTENT_PEERS")],-1),Ya=o('sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \\"$PERSISTENT_PEERS\\"/" $HOME/.celestia-app/config/config.toml',1),Za={class:"shiki github-light vp-code-light"},za={class:"line"},Ga=o('PERSISTENT_PEERS=$(curl -sL',6),Xa={style:{color:"#032F62"}},Ka=s("span",{style:{color:"#D73A49"}},"|",-1),Wa=s("span",{style:{color:"#032F62"}}," ",-1),se=s("span",{style:{color:"#6F42C1"}},"tr",-1),oe=s("span",{style:{color:"#032F62"}}," '\\n' ',')",-1),ae=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $PERSISTENT_PEERS")],-1),ee=o('sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \\"$PERSISTENT_PEERS\\"/" $HOME/.celestia-app/config/config.toml',1),ne=o(`

    Storage and pruning configurations

    Optional: Connect 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

    Query 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"

    Optional: Access 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"

    Save 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

    Sync types

    Sync modeTimeNotes
    Block sync~3 weeksDownloads and executes all blocks from genesis to the tip
    State sync~1 hourDownloads a snapshot of the state then downloads and executes all blocks after that snapshot to the tip.
    Quick sync~5 hoursDownloads the data directory from a node. Time depends on your download speed because the data being downloaded can exceed 1 TB for mainnet.

    Option 1: Block sync

    By default, a consensus node will sync using block sync; which will 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 weeks depending on the height of the blockchain).

    There are two alternatives for quicker syncing.

    Option 2: 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 = ""

    And also set statesync to true:

    toml
    #######################################################
    +###         State Sync Configuration Options        ###
    +#######################################################
    +[statesync]
    +enable = true
    #######################################################
    +###         State Sync Configuration Options        ###
    +#######################################################
    +[statesync]
    +enable = true

    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.

    Option 3: 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:

    `,33),te={class:"vp-code-group vp-adaptive-theme"},le=o('
    ',1),pe={class:"blocks"},ce={class:"language-bash vp-adaptive-theme active"},re=s("button",{title:"Copy Code",class:"copy"},null,-1),ie=s("span",{class:"lang"},"bash",-1),de={class:"shiki github-dark vp-code-dark"},_e=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"cd"),s("span",{style:{color:"#E1E4E8"}}," $HOME")],-1),ye=o('rm -rf ~/.celestia-app/data',1),he=o('mkdir -p ~/.celestia-app/data',1),Ee=o('SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \\',1),ue={class:"line"},Fe=s("span",{style:{color:"#9ECBFF"}}," ",-1),Ce=s("span",{style:{color:"#B392F0"}},"egrep",-1),ge=s("span",{style:{color:"#9ECBFF"}}," ",-1),Te=s("span",{style:{color:"#79B8FF"}},"-o",-1),be={style:{color:"#9ECBFF"}},me=o('| tr -d ">")',6),Se=o('aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"',1),ve=o('tar xf celestia-snap.tar -C ~/.celestia-app/data/',1),ke={class:"shiki github-light vp-code-light"},Be=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"cd"),s("span",{style:{color:"#24292E"}}," $HOME")],-1),Ae=o('rm -rf ~/.celestia-app/data',1),fe=o('mkdir -p ~/.celestia-app/data',1),Pe=o('SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \\',1),qe={class:"line"},Ie=s("span",{style:{color:"#032F62"}}," ",-1),xe=s("span",{style:{color:"#6F42C1"}},"egrep",-1),Ve=s("span",{style:{color:"#032F62"}}," ",-1),Re=s("span",{style:{color:"#005CC5"}},"-o",-1),De={style:{color:"#032F62"}},we=o('| tr -d ">")',6),Ne=o('aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"',1),$e=o('tar xf celestia-snap.tar -C ~/.celestia-app/data/',1),Oe={class:"language-bash vp-adaptive-theme"},Me=s("button",{title:"Copy Code",class:"copy"},null,-1),He=s("span",{class:"lang"},"bash",-1),Le={class:"shiki github-dark vp-code-dark"},Je=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"cd"),s("span",{style:{color:"#E1E4E8"}}," $HOME")],-1),Qe=o('rm -rf ~/.celestia-app/data',1),je=o('mkdir -p ~/.celestia-app/data',1),Ue=o('SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \\',1),Ye={class:"line"},Ze=s("span",{style:{color:"#9ECBFF"}}," ",-1),ze=s("span",{style:{color:"#B392F0"}},"egrep",-1),Ge=s("span",{style:{color:"#9ECBFF"}}," ",-1),Xe=s("span",{style:{color:"#79B8FF"}},"-o",-1),Ke={style:{color:"#9ECBFF"}},We=o('| tr -d ">")',6),sn=o('aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"',1),on=o('tar xf celestia-snap.tar -C ~/.celestia-app/data/',1),an={class:"shiki github-light vp-code-light"},en=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),tn=o('mkdir -p ~/.celestia-app/data',1),ln=o('SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \\',1),pn={class:"line"},cn=s("span",{style:{color:"#032F62"}}," ",-1),rn=s("span",{style:{color:"#6F42C1"}},"egrep",-1),dn=s("span",{style:{color:"#032F62"}}," ",-1),_n=s("span",{style:{color:"#005CC5"}},"-o",-1),yn={style:{color:"#032F62"}},hn=o('| tr -d ">")',6),En=o('aria2c -x 16 -s 16 -o celestia-snap.tar "https://snaps.qubelabs.io/celestia/${SNAP_NAME}"',1),un=o('tar xf celestia-snap.tar -C ~/.celestia-app/data/',1),Fn={class:"language-bash vp-adaptive-theme"},Cn=s("button",{title:"Copy Code",class:"copy"},null,-1),gn=s("span",{class:"lang"},"bash",-1),Tn={class:"shiki github-dark vp-code-dark"},bn=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"cd"),s("span",{style:{color:"#E1E4E8"}}," $HOME")],-1),mn=o('rm -rf ~/.celestia-app/data',1),Sn=o('mkdir -p ~/.celestia-app/data',1),vn=o('SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \\',1),kn={class:"line"},Bn=s("span",{style:{color:"#9ECBFF"}}," ",-1),An=s("span",{style:{color:"#B392F0"}},"egrep",-1),fn=s("span",{style:{color:"#9ECBFF"}}," ",-1),Pn=s("span",{style:{color:"#79B8FF"}},"-o",-1),qn={style:{color:"#9ECBFF"}},In=o('| tr -d ">")',6),xn=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),wn=o('rm -rf ~/.celestia-app/data',1),Nn=o('mkdir -p ~/.celestia-app/data',1),$n=o('SNAP_NAME=$(curl -s https://snaps.qubelabs.io/celestia/ | \\',1),On={class:"line"},Mn=s("span",{style:{color:"#032F62"}}," ",-1),Hn=s("span",{style:{color:"#6F42C1"}},"egrep",-1),Ln=s("span",{style:{color:"#032F62"}}," ",-1),Jn=s("span",{style:{color:"#005CC5"}},"-o",-1),Qn={style:{color:"#032F62"}},jn=o('| tr -d ">")',6),Un=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),Zn=o('

    Start the consensus node

    If you are running celestia-app v1.x.x:

    sh
    celestia-appd start
    celestia-appd start

    If you are running celestia-app >= v2.0.0: then you'll want to start the node with a --v2-upgrade-height that is dependent on the network. The --v2-upgrade-height flag is only needed during the v2 upgrade height so after your node has executed the upgrade (e.g. you see the log upgraded from app version 1 to 2), you don't need to provide this flag for future celestia-appd start invocations.

    sh
    celestia-appd start --v2-upgrade-height 2371495
    celestia-appd start --v2-upgrade-height 2371495
    sh
    celestia-appd start --v2-upgrade-height 2585031
    celestia-appd start --v2-upgrade-height 2585031
    sh
    celestia-appd start --v2-upgrade-height 1751707
    celestia-appd start --v2-upgrade-height 1751707

    Optional: If you would like celestia-app to run as a background process, you can follow the SystemD tutorial.

    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: Configure an RPC endpoint

    You can configure your 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)\nsed -i.bak -e "s/^external-address = ""/external-address = "$EXTERNAL-ADDRESS:26656"/" \\\n    $HOME/.celestia-app/config/config.toml
    EXTERNAL-ADDRESS=$(wget -qO- eth0.me)\nsed -i.bak -e "s/^external-address = ""/external-address = "$EXTERNAL-ADDRESS:26656"/" \\\n    $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.

    FAQ

    +2/3 committed an invalid block: wrong Block.Header.Version

    If you encounter an error like:

    bash
    2024-04-25 14:48:24 6:48PM ERR CONSENSUS FAILURE!!! err="+2/3 committed an invalid block: wrong Block.Header.Version. Expected {11 1}, got {11 2}" module=consensus stack="goroutine 214 [running]:\\nruntime/debug.Stack()\\n\\t/usr/local/go/src/runtime/debug/stack.go:24 +0x64\\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine.func2()\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:746 +0x44\\npanic({0x1b91180?, 0x400153b240?})\\n\\t/usr/local/go/src/runtime/panic.go:770 +0x124\\ngithub.com/tendermint/tendermint/consensus.(*State).finalizeCommit(0x400065ea88, 0x3)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1637 +0xd30\\ngithub.com/tendermint/tendermint/consensus.(*State).tryFinalizeCommit(0x400065ea88, 0x3)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1606 +0x26c\\ngithub.com/tendermint/tendermint/consensus.(*State).handleCompleteProposal(0x400065ea88, 0x3)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:2001 +0x2d8\\ngithub.com/tendermint/tendermint/consensus.(*State).handleMsg(0x400065ea88, {{0x2b30a00, 0x400143e048}, {0x40002a61b0, 0x28}})\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:856 +0x1c8\\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine(0x400065ea88, 0x0)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:782 +0x2c4\\ncreated by github.com/tendermint/tendermint/consensus.(*State).OnStart in goroutine 169\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:391 +0x110\\n"
    2024-04-25 14:48:24 6:48PM ERR CONSENSUS FAILURE!!! err="+2/3 committed an invalid block: wrong Block.Header.Version. Expected {11 1}, got {11 2}" module=consensus stack="goroutine 214 [running]:\\nruntime/debug.Stack()\\n\\t/usr/local/go/src/runtime/debug/stack.go:24 +0x64\\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine.func2()\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:746 +0x44\\npanic({0x1b91180?, 0x400153b240?})\\n\\t/usr/local/go/src/runtime/panic.go:770 +0x124\\ngithub.com/tendermint/tendermint/consensus.(*State).finalizeCommit(0x400065ea88, 0x3)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1637 +0xd30\\ngithub.com/tendermint/tendermint/consensus.(*State).tryFinalizeCommit(0x400065ea88, 0x3)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1606 +0x26c\\ngithub.com/tendermint/tendermint/consensus.(*State).handleCompleteProposal(0x400065ea88, 0x3)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:2001 +0x2d8\\ngithub.com/tendermint/tendermint/consensus.(*State).handleMsg(0x400065ea88, {{0x2b30a00, 0x400143e048}, {0x40002a61b0, 0x28}})\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:856 +0x1c8\\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine(0x400065ea88, 0x0)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:782 +0x2c4\\ncreated by github.com/tendermint/tendermint/consensus.(*State).OnStart in goroutine 169\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:391 +0x110\\n"

    then it is likely that the network has upgraded to a new app version but your consensus node was not prepared for the upgrade. To fix this, you'll need to:

    1. Remove DBs from your CELESTIA_HOME directory via: celestia-appd tendermint reset-state.
    2. Remove the data/application.db inside your CELESTIA_HOME directory.
    3. Download the latest binary for your network.
    4. Restart your consensus node with the relevant --v2-upgrade-height for the network you're running on.
    ',42),ot=JSON.parse('{"title":"Consensus node","description":"Learn how to set up a Celestia consensus node.","frontmatter":{"description":"Learn how to set up a Celestia consensus node.","outline":"deep","head":[["meta",{"name":"og:title","content":"Consensus node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/consensus-node.md","filePath":"nodes/consensus-node.md","lastUpdated":1726697682000}'),zn={name:"nodes/consensus-node.md"},at=Object.assign(zn,{setup(Gn){return(Xn,Kn)=>(l(),p("div",null,[r,s("div",i,[d,s("div",_,[s("div",y,[h,E,s("pre",u,[s("code",null,[s("span",F,[C,s("span",g,n(t(e).mainnetChainId),1)])])]),s("pre",T,[s("code",null,[s("span",b,[m,s("span",S,n(t(e).mainnetChainId),1)])])])]),s("div",v,[k,B,s("pre",A,[s("code",null,[s("span",f,[P,s("span",q,n(t(e).mochaChainId),1)])])]),s("pre",I,[s("code",null,[s("span",x,[V,s("span",R,n(t(e).mochaChainId),1)])])])]),s("div",D,[w,N,s("pre",$,[s("code",null,[s("span",O,[M,s("span",H,n(t(e).arabicaChainId),1)])])]),s("pre",L,[s("code",null,[s("span",J,[Q,s("span",j,n(t(e).arabicaChainId),1)])])])])])]),U,s("div",Y,[Z,s("div",z,[s("div",G,[X,K,s("pre",W,[s("code",null,[s("span",ss,[os,as,es,ns,s("span",ts,n(t(e).mainnetChainId),1)])])]),s("pre",ls,[s("code",null,[s("span",ps,[cs,rs,is,ds,s("span",_s,n(t(e).mainnetChainId),1)])])])]),s("div",ys,[hs,Es,s("pre",us,[s("code",null,[s("span",Fs,[Cs,gs,Ts,bs,s("span",ms,n(t(e).mochaChainId),1)])])]),s("pre",Ss,[s("code",null,[s("span",vs,[ks,Bs,As,fs,s("span",Ps,n(t(e).mochaChainId),1)])])])]),s("div",qs,[Is,xs,s("pre",Vs,[s("code",null,[s("span",Rs,[Ds,ws,Ns,$s,s("span",Os,n(t(e).arabicaChainId),1)])])]),s("pre",Ms,[s("code",null,[s("span",Hs,[Ls,Js,Qs,js,s("span",Us,n(t(e).arabicaChainId),1)])])])])])]),Ys,s("div",Zs,[zs,s("div",Gs,[s("div",Xs,[Ks,Ws,s("pre",so,[s("code",null,[s("span",oo,[ao,s("span",eo," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).mainnetChainId)+"/seeds.txt ",1),no,to,lo,po]),a(` +`),co,a(` +`),ro])]),s("pre",io,[s("code",null,[s("span",_o,[yo,s("span",ho," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).mainnetChainId)+"/seeds.txt ",1),Eo,uo,Fo,Co]),a(` +`),go,a(` +`),To])])]),s("div",bo,[mo,So,s("pre",vo,[s("code",null,[s("span",ko,[Bo,s("span",Ao," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).mochaChainId)+"/seeds.txt ",1),fo,Po,qo,Io]),a(` +`),xo,a(` +`),Vo])]),s("pre",Ro,[s("code",null,[s("span",Do,[wo,s("span",No," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).mochaChainId)+"/seeds.txt ",1),$o,Oo,Mo,Ho]),a(` +`),Lo,a(` +`),Jo])])]),Qo])]),s("details",null,[jo,s("div",Uo,[Yo,s("div",Zo,[s("div",zo,[Go,Xo,s("pre",Ko,[s("code",null,[s("span",Wo,[sa,s("span",oa," https://raw.githubusercontent.com/cosmos/chain-registry/master/"+n(t(e).mainnetChainId)+"/chain.json ",1),aa]),a(` +`),ea,a(` +`),na])]),s("pre",ta,[s("code",null,[s("span",la,[pa,s("span",ca," https://raw.githubusercontent.com/cosmos/chain-registry/master/"+n(t(e).mainnetChainId)+"/chain.json ",1),ra]),a(` +`),ia,a(` +`),da])])]),s("div",_a,[ya,ha,s("pre",Ea,[s("code",null,[s("span",ua,[Fa,s("span",Ca," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).mochaChainId)+"/peers.txt ",1),ga,Ta,ba,ma]),a(` +`),Sa,a(` +`),va])]),s("pre",ka,[s("code",null,[s("span",Ba,[Aa,s("span",fa," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).mochaChainId)+"/peers.txt ",1),Pa,qa,Ia,xa]),a(` +`),Va,a(` +`),Ra])])]),s("div",Da,[wa,Na,s("pre",$a,[s("code",null,[s("span",Oa,[Ma,s("span",Ha," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).arabicaChainId)+"/peers.txt ",1),La,Ja,Qa,ja]),a(` +`),Ua,a(` +`),Ya])]),s("pre",Za,[s("code",null,[s("span",za,[Ga,s("span",Xa," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).arabicaChainId)+"/peers.txt ",1),Ka,Wa,se,oe]),a(` +`),ae,a(` +`),ee])])])])])]),ne,s("div",te,[le,s("div",pe,[s("div",ce,[re,ie,s("pre",de,[s("code",null,[_e,a(` +`),ye,a(` +`),he,a(` +`),Ee,a(` +`),s("span",ue,[Fe,Ce,ge,Te,s("span",be,' ">'+n(t(e).mainnetChainId)+'.*tar" ',1),me]),a(` +`),Se,a(` +`),ve])]),s("pre",ke,[s("code",null,[Be,a(` +`),Ae,a(` +`),fe,a(` +`),Pe,a(` +`),s("span",qe,[Ie,xe,Ve,Re,s("span",De,' ">'+n(t(e).mainnetChainId)+'.*tar" ',1),we]),a(` +`),Ne,a(` +`),$e])])]),s("div",Oe,[Me,He,s("pre",Le,[s("code",null,[Je,a(` +`),Qe,a(` +`),je,a(` +`),Ue,a(` +`),s("span",Ye,[Ze,ze,Ge,Xe,s("span",Ke,' ">'+n(t(e).mochaChainId)+'.*tar" ',1),We]),a(` +`),sn,a(` +`),on])]),s("pre",an,[s("code",null,[en,a(` +`),nn,a(` +`),tn,a(` +`),ln,a(` +`),s("span",pn,[cn,rn,dn,_n,s("span",yn,' ">'+n(t(e).mochaChainId)+'.*tar" ',1),hn]),a(` +`),En,a(` +`),un])])]),s("div",Fn,[Cn,gn,s("pre",Tn,[s("code",null,[bn,a(` +`),mn,a(` +`),Sn,a(` +`),vn,a(` +`),s("span",kn,[Bn,An,fn,Pn,s("span",qn,' ">'+n(t(e).arabicaChainId)+'.*tar" ',1),In]),a(` +`),xn,a(` +`),Vn])]),s("pre",Rn,[s("code",null,[Dn,a(` +`),wn,a(` +`),Nn,a(` +`),$n,a(` +`),s("span",On,[Mn,Hn,Ln,Jn,s("span",Qn,' ">'+n(t(e).arabicaChainId)+'.*tar" ',1),jn]),a(` +`),Un,a(` +`),Yn])])])])]),Zn]))}});export{ot as __pageData,at as default}; diff --git a/assets/nodes_consensus-node.md.b07b70a7.lean.js b/assets/nodes_consensus-node.md.b07b70a7.lean.js new file mode 100644 index 00000000000..6e32080b0e4 --- /dev/null +++ b/assets/nodes_consensus-node.md.b07b70a7.lean.js @@ -0,0 +1,57 @@ +import{c as e}from"./chunks/constants.fa173a21.js";import{o as l,c as p,k as s,t as n,l as t,a,Q as o}from"./chunks/framework.1a91c06a.js";const c="/img/nodes/consensus-node.jpg",r=o("",14),i={class:"vp-code-group vp-adaptive-theme"},d=o("",1),_={class:"blocks"},y={class:"language-bash vp-adaptive-theme active"},h=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"},m=o("",8),S={style:{color:"#032F62"}},v={class:"language-bash vp-adaptive-theme"},k=s("button",{title:"Copy Code",class:"copy"},null,-1),B=s("span",{class:"lang"},"bash",-1),A={class:"shiki github-dark vp-code-dark"},f={class:"line"},P=o("",8),q={style:{color:"#9ECBFF"}},I={class:"shiki github-light vp-code-light"},x={class:"line"},V=o("",8),R={style:{color:"#032F62"}},D={class:"language-bash vp-adaptive-theme"},w=s("button",{title:"Copy Code",class:"copy"},null,-1),N=s("span",{class:"lang"},"bash",-1),$={class:"shiki github-dark vp-code-dark"},O={class:"line"},M=o("",8),H={style:{color:"#9ECBFF"}},L={class:"shiki github-light vp-code-light"},J={class:"line"},Q=o("",8),j={style:{color:"#032F62"}},U=s("p",null,[a("Download the "),s("code",null,"genesis.json"),a(" file:")],-1),Y={class:"vp-code-group vp-adaptive-theme"},Z=o("",1),z={class:"blocks"},G={class:"language-bash vp-adaptive-theme active"},X=s("button",{title:"Copy Code",class:"copy"},null,-1),K=s("span",{class:"lang"},"bash",-1),W={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),es=s("span",{style:{color:"#9ECBFF"}},"download-genesis",-1),ns=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),rs=s("span",{style:{color:"#24292E"}}," ",-1),is=s("span",{style:{color:"#032F62"}},"download-genesis",-1),ds=s("span",{style:{color:"#24292E"}}," ",-1),_s={style:{color:"#032F62"}},ys={class:"language-bash vp-adaptive-theme"},hs=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),ms={style:{color:"#9ECBFF"}},Ss={class:"shiki github-light vp-code-light"},vs={class:"line"},ks=s("span",{style:{color:"#6F42C1"}},"celestia-appd",-1),Bs=s("span",{style:{color:"#24292E"}}," ",-1),As=s("span",{style:{color:"#032F62"}},"download-genesis",-1),fs=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),xs=s("span",{class:"lang"},"bash",-1),Vs={class:"shiki github-dark vp-code-dark"},Rs={class:"line"},Ds=s("span",{style:{color:"#B392F0"}},"celestia-appd",-1),ws=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ns=s("span",{style:{color:"#9ECBFF"}},"download-genesis",-1),$s=s("span",{style:{color:"#E1E4E8"}}," ",-1),Os={style:{color:"#9ECBFF"}},Ms={class:"shiki github-light vp-code-light"},Hs={class:"line"},Ls=s("span",{style:{color:"#6F42C1"}},"celestia-appd",-1),Js=s("span",{style:{color:"#24292E"}}," ",-1),Qs=s("span",{style:{color:"#032F62"}},"download-genesis",-1),js=s("span",{style:{color:"#24292E"}}," ",-1),Us={style:{color:"#032F62"}},Ys=s("p",null,[a("Set seeds in the "),s("code",null,"$HOME/.celestia-app/config/config.toml"),a(" file:")],-1),Zs={class:"vp-code-group vp-adaptive-theme"},zs=o("",1),Gs={class:"blocks"},Xs={class:"language-bash vp-adaptive-theme active"},Ks=s("button",{title:"Copy Code",class:"copy"},null,-1),Ws=s("span",{class:"lang"},"bash",-1),so={class:"shiki github-dark vp-code-dark"},oo={class:"line"},ao=o("",6),eo={style:{color:"#9ECBFF"}},no=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),ro=o("",1),io={class:"shiki github-light vp-code-light"},_o={class:"line"},yo=o("",6),ho={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"},mo=s("button",{title:"Copy Code",class:"copy"},null,-1),So=s("span",{class:"lang"},"bash",-1),vo={class:"shiki github-dark vp-code-dark"},ko={class:"line"},Bo=o("",6),Ao={style:{color:"#9ECBFF"}},fo=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),xo=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $SEEDS")],-1),Vo=o("",1),Ro={class:"shiki github-light vp-code-light"},Do={class:"line"},wo=o("",6),No={style:{color:"#032F62"}},$o=s("span",{style:{color:"#D73A49"}},"|",-1),Oo=s("span",{style:{color:"#032F62"}}," ",-1),Mo=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),Jo=o("",1),Qo=o("",1),jo=o("",3),Uo={class:"vp-code-group vp-adaptive-theme"},Yo=o("",1),Zo={class:"blocks"},zo={class:"language-bash vp-adaptive-theme active"},Go=s("button",{title:"Copy Code",class:"copy"},null,-1),Xo=s("span",{class:"lang"},"bash",-1),Ko={class:"shiki github-dark vp-code-dark"},Wo={class:"line"},sa=o("",6),oa={style:{color:"#9ECBFF"}},aa=o("",14),ea=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $PERSISTENT_PEERS")],-1),na=o("",1),ta={class:"shiki github-light vp-code-light"},la={class:"line"},pa=o("",6),ca={style:{color:"#032F62"}},ra=o("",14),ia=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $PERSISTENT_PEERS")],-1),da=o("",1),_a={class:"language-bash vp-adaptive-theme"},ya=s("button",{title:"Copy Code",class:"copy"},null,-1),ha=s("span",{class:"lang"},"bash",-1),Ea={class:"shiki github-dark vp-code-dark"},ua={class:"line"},Fa=o("",6),Ca={style:{color:"#9ECBFF"}},ga=s("span",{style:{color:"#F97583"}},"|",-1),Ta=s("span",{style:{color:"#9ECBFF"}}," ",-1),ba=s("span",{style:{color:"#B392F0"}},"tr",-1),ma=s("span",{style:{color:"#9ECBFF"}}," '\\n' ',')",-1),Sa=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $PERSISTENT_PEERS")],-1),va=o("",1),ka={class:"shiki github-light vp-code-light"},Ba={class:"line"},Aa=o("",6),fa={style:{color:"#032F62"}},Pa=s("span",{style:{color:"#D73A49"}},"|",-1),qa=s("span",{style:{color:"#032F62"}}," ",-1),Ia=s("span",{style:{color:"#6F42C1"}},"tr",-1),xa=s("span",{style:{color:"#032F62"}}," '\\n' ',')",-1),Va=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $PERSISTENT_PEERS")],-1),Ra=o("",1),Da={class:"language-bash vp-adaptive-theme"},wa=s("button",{title:"Copy Code",class:"copy"},null,-1),Na=s("span",{class:"lang"},"bash",-1),$a={class:"shiki github-dark vp-code-dark"},Oa={class:"line"},Ma=o("",6),Ha={style:{color:"#9ECBFF"}},La=s("span",{style:{color:"#F97583"}},"|",-1),Ja=s("span",{style:{color:"#9ECBFF"}}," ",-1),Qa=s("span",{style:{color:"#B392F0"}},"tr",-1),ja=s("span",{style:{color:"#9ECBFF"}}," '\\n' ',')",-1),Ua=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"echo"),s("span",{style:{color:"#E1E4E8"}}," $PERSISTENT_PEERS")],-1),Ya=o("",1),Za={class:"shiki github-light vp-code-light"},za={class:"line"},Ga=o("",6),Xa={style:{color:"#032F62"}},Ka=s("span",{style:{color:"#D73A49"}},"|",-1),Wa=s("span",{style:{color:"#032F62"}}," ",-1),se=s("span",{style:{color:"#6F42C1"}},"tr",-1),oe=s("span",{style:{color:"#032F62"}}," '\\n' ',')",-1),ae=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"echo"),s("span",{style:{color:"#24292E"}}," $PERSISTENT_PEERS")],-1),ee=o("",1),ne=o("",33),te={class:"vp-code-group vp-adaptive-theme"},le=o("",1),pe={class:"blocks"},ce={class:"language-bash vp-adaptive-theme active"},re=s("button",{title:"Copy Code",class:"copy"},null,-1),ie=s("span",{class:"lang"},"bash",-1),de={class:"shiki github-dark vp-code-dark"},_e=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"cd"),s("span",{style:{color:"#E1E4E8"}}," $HOME")],-1),ye=o("",1),he=o("",1),Ee=o("",1),ue={class:"line"},Fe=s("span",{style:{color:"#9ECBFF"}}," ",-1),Ce=s("span",{style:{color:"#B392F0"}},"egrep",-1),ge=s("span",{style:{color:"#9ECBFF"}}," ",-1),Te=s("span",{style:{color:"#79B8FF"}},"-o",-1),be={style:{color:"#9ECBFF"}},me=o("",6),Se=o("",1),ve=o("",1),ke={class:"shiki github-light vp-code-light"},Be=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"cd"),s("span",{style:{color:"#24292E"}}," $HOME")],-1),Ae=o("",1),fe=o("",1),Pe=o("",1),qe={class:"line"},Ie=s("span",{style:{color:"#032F62"}}," ",-1),xe=s("span",{style:{color:"#6F42C1"}},"egrep",-1),Ve=s("span",{style:{color:"#032F62"}}," ",-1),Re=s("span",{style:{color:"#005CC5"}},"-o",-1),De={style:{color:"#032F62"}},we=o("",6),Ne=o("",1),$e=o("",1),Oe={class:"language-bash vp-adaptive-theme"},Me=s("button",{title:"Copy Code",class:"copy"},null,-1),He=s("span",{class:"lang"},"bash",-1),Le={class:"shiki github-dark vp-code-dark"},Je=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"cd"),s("span",{style:{color:"#E1E4E8"}}," $HOME")],-1),Qe=o("",1),je=o("",1),Ue=o("",1),Ye={class:"line"},Ze=s("span",{style:{color:"#9ECBFF"}}," ",-1),ze=s("span",{style:{color:"#B392F0"}},"egrep",-1),Ge=s("span",{style:{color:"#9ECBFF"}}," ",-1),Xe=s("span",{style:{color:"#79B8FF"}},"-o",-1),Ke={style:{color:"#9ECBFF"}},We=o("",6),sn=o("",1),on=o("",1),an={class:"shiki github-light vp-code-light"},en=s("span",{class:"line"},[s("span",{style:{color:"#005CC5"}},"cd"),s("span",{style:{color:"#24292E"}}," $HOME")],-1),nn=o("",1),tn=o("",1),ln=o("",1),pn={class:"line"},cn=s("span",{style:{color:"#032F62"}}," ",-1),rn=s("span",{style:{color:"#6F42C1"}},"egrep",-1),dn=s("span",{style:{color:"#032F62"}}," ",-1),_n=s("span",{style:{color:"#005CC5"}},"-o",-1),yn={style:{color:"#032F62"}},hn=o("",6),En=o("",1),un=o("",1),Fn={class:"language-bash vp-adaptive-theme"},Cn=s("button",{title:"Copy Code",class:"copy"},null,-1),gn=s("span",{class:"lang"},"bash",-1),Tn={class:"shiki github-dark vp-code-dark"},bn=s("span",{class:"line"},[s("span",{style:{color:"#79B8FF"}},"cd"),s("span",{style:{color:"#E1E4E8"}}," $HOME")],-1),mn=o("",1),Sn=o("",1),vn=o("",1),kn={class:"line"},Bn=s("span",{style:{color:"#9ECBFF"}}," ",-1),An=s("span",{style:{color:"#B392F0"}},"egrep",-1),fn=s("span",{style:{color:"#9ECBFF"}}," ",-1),Pn=s("span",{style:{color:"#79B8FF"}},"-o",-1),qn={style:{color:"#9ECBFF"}},In=o("",6),xn=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),wn=o("",1),Nn=o("",1),$n=o("",1),On={class:"line"},Mn=s("span",{style:{color:"#032F62"}}," ",-1),Hn=s("span",{style:{color:"#6F42C1"}},"egrep",-1),Ln=s("span",{style:{color:"#032F62"}}," ",-1),Jn=s("span",{style:{color:"#005CC5"}},"-o",-1),Qn={style:{color:"#032F62"}},jn=o("",6),Un=o("",1),Yn=o("",1),Zn=o("",42),ot=JSON.parse('{"title":"Consensus node","description":"Learn how to set up a Celestia consensus node.","frontmatter":{"description":"Learn how to set up a Celestia consensus node.","outline":"deep","head":[["meta",{"name":"og:title","content":"Consensus node | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/consensus-node.md","filePath":"nodes/consensus-node.md","lastUpdated":1726697682000}'),zn={name:"nodes/consensus-node.md"},at=Object.assign(zn,{setup(Gn){return(Xn,Kn)=>(l(),p("div",null,[r,s("div",i,[d,s("div",_,[s("div",y,[h,E,s("pre",u,[s("code",null,[s("span",F,[C,s("span",g,n(t(e).mainnetChainId),1)])])]),s("pre",T,[s("code",null,[s("span",b,[m,s("span",S,n(t(e).mainnetChainId),1)])])])]),s("div",v,[k,B,s("pre",A,[s("code",null,[s("span",f,[P,s("span",q,n(t(e).mochaChainId),1)])])]),s("pre",I,[s("code",null,[s("span",x,[V,s("span",R,n(t(e).mochaChainId),1)])])])]),s("div",D,[w,N,s("pre",$,[s("code",null,[s("span",O,[M,s("span",H,n(t(e).arabicaChainId),1)])])]),s("pre",L,[s("code",null,[s("span",J,[Q,s("span",j,n(t(e).arabicaChainId),1)])])])])])]),U,s("div",Y,[Z,s("div",z,[s("div",G,[X,K,s("pre",W,[s("code",null,[s("span",ss,[os,as,es,ns,s("span",ts,n(t(e).mainnetChainId),1)])])]),s("pre",ls,[s("code",null,[s("span",ps,[cs,rs,is,ds,s("span",_s,n(t(e).mainnetChainId),1)])])])]),s("div",ys,[hs,Es,s("pre",us,[s("code",null,[s("span",Fs,[Cs,gs,Ts,bs,s("span",ms,n(t(e).mochaChainId),1)])])]),s("pre",Ss,[s("code",null,[s("span",vs,[ks,Bs,As,fs,s("span",Ps,n(t(e).mochaChainId),1)])])])]),s("div",qs,[Is,xs,s("pre",Vs,[s("code",null,[s("span",Rs,[Ds,ws,Ns,$s,s("span",Os,n(t(e).arabicaChainId),1)])])]),s("pre",Ms,[s("code",null,[s("span",Hs,[Ls,Js,Qs,js,s("span",Us,n(t(e).arabicaChainId),1)])])])])])]),Ys,s("div",Zs,[zs,s("div",Gs,[s("div",Xs,[Ks,Ws,s("pre",so,[s("code",null,[s("span",oo,[ao,s("span",eo," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).mainnetChainId)+"/seeds.txt ",1),no,to,lo,po]),a(` +`),co,a(` +`),ro])]),s("pre",io,[s("code",null,[s("span",_o,[yo,s("span",ho," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).mainnetChainId)+"/seeds.txt ",1),Eo,uo,Fo,Co]),a(` +`),go,a(` +`),To])])]),s("div",bo,[mo,So,s("pre",vo,[s("code",null,[s("span",ko,[Bo,s("span",Ao," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).mochaChainId)+"/seeds.txt ",1),fo,Po,qo,Io]),a(` +`),xo,a(` +`),Vo])]),s("pre",Ro,[s("code",null,[s("span",Do,[wo,s("span",No," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).mochaChainId)+"/seeds.txt ",1),$o,Oo,Mo,Ho]),a(` +`),Lo,a(` +`),Jo])])]),Qo])]),s("details",null,[jo,s("div",Uo,[Yo,s("div",Zo,[s("div",zo,[Go,Xo,s("pre",Ko,[s("code",null,[s("span",Wo,[sa,s("span",oa," https://raw.githubusercontent.com/cosmos/chain-registry/master/"+n(t(e).mainnetChainId)+"/chain.json ",1),aa]),a(` +`),ea,a(` +`),na])]),s("pre",ta,[s("code",null,[s("span",la,[pa,s("span",ca," https://raw.githubusercontent.com/cosmos/chain-registry/master/"+n(t(e).mainnetChainId)+"/chain.json ",1),ra]),a(` +`),ia,a(` +`),da])])]),s("div",_a,[ya,ha,s("pre",Ea,[s("code",null,[s("span",ua,[Fa,s("span",Ca," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).mochaChainId)+"/peers.txt ",1),ga,Ta,ba,ma]),a(` +`),Sa,a(` +`),va])]),s("pre",ka,[s("code",null,[s("span",Ba,[Aa,s("span",fa," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).mochaChainId)+"/peers.txt ",1),Pa,qa,Ia,xa]),a(` +`),Va,a(` +`),Ra])])]),s("div",Da,[wa,Na,s("pre",$a,[s("code",null,[s("span",Oa,[Ma,s("span",Ha," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).arabicaChainId)+"/peers.txt ",1),La,Ja,Qa,ja]),a(` +`),Ua,a(` +`),Ya])]),s("pre",Za,[s("code",null,[s("span",za,[Ga,s("span",Xa," https://raw.githubusercontent.com/celestiaorg/networks/master/"+n(t(e).arabicaChainId)+"/peers.txt ",1),Ka,Wa,se,oe]),a(` +`),ae,a(` +`),ee])])])])])]),ne,s("div",te,[le,s("div",pe,[s("div",ce,[re,ie,s("pre",de,[s("code",null,[_e,a(` +`),ye,a(` +`),he,a(` +`),Ee,a(` +`),s("span",ue,[Fe,Ce,ge,Te,s("span",be,' ">'+n(t(e).mainnetChainId)+'.*tar" ',1),me]),a(` +`),Se,a(` +`),ve])]),s("pre",ke,[s("code",null,[Be,a(` +`),Ae,a(` +`),fe,a(` +`),Pe,a(` +`),s("span",qe,[Ie,xe,Ve,Re,s("span",De,' ">'+n(t(e).mainnetChainId)+'.*tar" ',1),we]),a(` +`),Ne,a(` +`),$e])])]),s("div",Oe,[Me,He,s("pre",Le,[s("code",null,[Je,a(` +`),Qe,a(` +`),je,a(` +`),Ue,a(` +`),s("span",Ye,[Ze,ze,Ge,Xe,s("span",Ke,' ">'+n(t(e).mochaChainId)+'.*tar" ',1),We]),a(` +`),sn,a(` +`),on])]),s("pre",an,[s("code",null,[en,a(` +`),nn,a(` +`),tn,a(` +`),ln,a(` +`),s("span",pn,[cn,rn,dn,_n,s("span",yn,' ">'+n(t(e).mochaChainId)+'.*tar" ',1),hn]),a(` +`),En,a(` +`),un])])]),s("div",Fn,[Cn,gn,s("pre",Tn,[s("code",null,[bn,a(` +`),mn,a(` +`),Sn,a(` +`),vn,a(` +`),s("span",kn,[Bn,An,fn,Pn,s("span",qn,' ">'+n(t(e).arabicaChainId)+'.*tar" ',1),In]),a(` +`),xn,a(` +`),Vn])]),s("pre",Rn,[s("code",null,[Dn,a(` +`),wn,a(` +`),Nn,a(` +`),$n,a(` +`),s("span",On,[Mn,Hn,Ln,Jn,s("span",Qn,' ">'+n(t(e).arabicaChainId)+'.*tar" ',1),jn]),a(` +`),Un,a(` +`),Yn])])])])]),Zn]))}});export{ot as __pageData,at as default}; diff --git a/assets/nodes_decide-node.md.0a555064.js b/assets/nodes_decide-node.md.0a555064.js new file mode 100644 index 00000000000..f3d81571c92 --- /dev/null +++ b/assets/nodes_decide-node.md.0a555064.js @@ -0,0 +1 @@ +import{_ as e,o,c as n,Q as a}from"./chunks/framework.1a91c06a.js";const f=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":1726491136000}'),t={name:"nodes/decide-node.md"},d=a('

    Deciding which node to run

    Now that you have installed the basic dependencies, you can start exploring which nodes to run!

    Light node

    It is highly recommended to get started with running a data availability light node.

    You can also play around with the Data Availability API in this tutorial for posting and retrieving data with a light node.

    Other DA nodes

    Depending on your use case, you also may want to run a bridge node or a full DA node.

    Consensus node

    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.

    ',10),i=[d];function r(s,h,c,l,u,p){return o(),n("div",null,i)}const _=e(t,[["render",r]]);export{f as __pageData,_ as default}; diff --git a/assets/nodes_decide-node.md.0a555064.lean.js b/assets/nodes_decide-node.md.0a555064.lean.js new file mode 100644 index 00000000000..88221b7979b --- /dev/null +++ b/assets/nodes_decide-node.md.0a555064.lean.js @@ -0,0 +1 @@ +import{_ as e,o,c as n,Q as a}from"./chunks/framework.1a91c06a.js";const f=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":1726491136000}'),t={name:"nodes/decide-node.md"},d=a("",10),i=[d];function r(s,h,c,l,u,p){return o(),n("div",null,i)}const _=e(t,[["render",r]]);export{f as __pageData,_ as default}; diff --git a/assets/nodes_docker-images.md.97bd64bd.js b/assets/nodes_docker-images.md.97bd64bd.js new file mode 100644 index 00000000000..ea5b8f0c01d --- /dev/null +++ b/assets/nodes_docker-images.md.97bd64bd.js @@ -0,0 +1,59 @@ +import{a as l}from"./chunks/arabica_versions.1930378b.js";import{m as t}from"./chunks/mocha_versions.7704b055.js";import{m as p}from"./chunks/mainnet_versions.955428b6.js";import{o as c,c as r,k as s,a,t as n,l as e,Q as o}from"./chunks/framework.1a91c06a.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),B={class:"shiki github-light vp-code-light"},S=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),N={class:"line"},R=s("span",{style:{color:"#24292E"}}," ",-1),D={style:{color:"#032F62"}},O=s("span",{style:{color:"#24292E"}}," ",-1),V=s("span",{style:{color:"#005CC5"}},"\\",-1),I=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),W={class:"shiki github-dark vp-code-dark"},K=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),Y={class:"line"},M=s("span",{style:{color:"#E1E4E8"}}," ",-1),x={style:{color:"#9ECBFF"}},q=s("span",{style:{color:"#E1E4E8"}}," ",-1),H=s("span",{style:{color:"#79B8FF"}},"\\",-1),U=o(' celestia $NODE_TYPE start --core.ip $RPC_URL --p2p.network $NETWORK',1),L={class:"shiki github-light vp-code-light"},z=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),J={class:"line"},Q=s("span",{style:{color:"#24292E"}}," ",-1),G={style:{color:"#032F62"}},j=s("span",{style:{color:"#24292E"}}," ",-1),Z=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"},Bs=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),Ss=o(' -v $HOME/my-node-store:/home/celestia \\',1),Ns={class:"line"},Rs=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ds={style:{color:"#9ECBFF"}},Os=s("span",{style:{color:"#E1E4E8"}}," ",-1),Vs=s("span",{style:{color:"#79B8FF"}},"\\",-1),Is=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),Ws={class:"line"},Ks=s("span",{style:{color:"#24292E"}}," ",-1),Ys={style:{color:"#032F62"}},Ms=s("span",{style:{color:"#24292E"}}," ",-1),xs=s("span",{style:{color:"#005CC5"}},"\\",-1),qs=o(' celestia light init --p2p.network $NETWORK',1),Hs={class:"language-bash vp-adaptive-theme"},Us=s("button",{title:"Copy Code",class:"copy"},null,-1),Ls=s("span",{class:"lang"},"bash",-1),zs={class:"shiki github-dark vp-code-dark"},Js=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),Qs=o(' -v $HOME/my-node-store:/home/celestia \\',1),Gs={class:"line"},js=s("span",{style:{color:"#E1E4E8"}}," ",-1),Zs={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),Bo={class:"line"},So=s("span",{style:{color:"#24292E"}}," ",-1),No={style:{color:"#032F62"}},Ro=s("span",{style:{color:"#24292E"}}," ",-1),Do=s("span",{style:{color:"#005CC5"}},"\\",-1),Oo=o(' celestia light init --p2p.network $NETWORK',1),Vo=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),Io={class:"vp-code-group vp-adaptive-theme"},fo=o('
    ',1),$o={class:"blocks"},wo={class:"language-bash vp-adaptive-theme active"},Wo=s("button",{title:"Copy Code",class:"copy"},null,-1),Ko=s("span",{class:"lang"},"bash",-1),Yo={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),qo={class:"line"},Ho=s("span",{style:{color:"#E1E4E8"}}," ",-1),Uo={style:{color:"#9ECBFF"}},Lo=s("span",{style:{color:"#E1E4E8"}}," ",-1),zo=s("span",{style:{color:"#79B8FF"}},"\\",-1),Jo=o(' celestia light start --core.ip $RPC_URL',1),Qo={class:"shiki github-light vp-code-light"},Go=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),jo=o(' -v $HOME/my-node-store:/home/celestia \\',1),Zo={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),Ba=s("span",{class:"lang"},"bash",-1),Sa={class:"shiki github-dark vp-code-dark"},Na=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),Ra=o(' -v $HOME/my-node-store:/home/celestia \\',1),Da={class:"line"},Oa=s("span",{style:{color:"#E1E4E8"}}," ",-1),Va={style:{color:"#9ECBFF"}},Ia=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"},Wa=o('docker run -e NODE_TYPE=$NODE_TYPE -e P2P_NETWORK=$NETWORK \\',1),Ka=o(' -v $HOME/my-node-store:/home/celestia \\',1),Ya={class:"line"},Ma=s("span",{style:{color:"#24292E"}}," ",-1),xa={style:{color:"#032F62"}},qa=s("span",{style:{color:"#24292E"}}," ",-1),Ha=s("span",{style:{color:"#005CC5"}},"\\",-1),Ua=o(' celestia light start --core.ip $RPC_URL',1),La=s("p",null,"Congratulations! You now have a node running with persistent storage.",-1),za=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),Qa=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),Ga=s("div",{class:"youtube-wrapper"},[s("iframe",{class:"youtube-video",title:"Running a Celestia light node",src:"https://youtube.com/embed/ROZv871Q7RM",allowfullscreen:""})],-1),ja=s("h2",{id:"troubleshooting",tabindex:"-1"},[a("Troubleshooting "),s("a",{class:"header-anchor",href:"#troubleshooting","aria-label":'Permalink to "Troubleshooting"'},"​")],-1),Za=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":1726496322000}'),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",B,[s("code",null,[S,a(` +`),s("span",N,[R,s("span",D,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),O,V]),a(` +`),I])])]),s("div",f,[$,w,s("pre",W,[s("code",null,[K,a(` +`),s("span",Y,[M,s("span",x,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),q,H]),a(` +`),U])]),s("pre",L,[s("code",null,[z,a(` +`),s("span",J,[Q,s("span",G,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),j,Z]),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,[Bs,a(` +`),Ss,a(` +`),s("span",Ns,[Rs,s("span",Ds,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),Os,Vs]),a(` +`),Is])]),s("pre",fs,[s("code",null,[$s,a(` +`),ws,a(` +`),s("span",Ws,[Ks,s("span",Ys,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),Ms,xs]),a(` +`),qs])])]),s("div",Hs,[Us,Ls,s("pre",zs,[s("code",null,[Js,a(` +`),Qs,a(` +`),s("span",Gs,[js,s("span",Zs,"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",Bo,[So,s("span",No,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),Ro,Do]),a(` +`),Oo])])])])]),Vo,s("div",Io,[fo,s("div",$o,[s("div",wo,[Wo,Ko,s("pre",Yo,[s("code",null,[Mo,a(` +`),xo,a(` +`),s("span",qo,[Ho,s("span",Uo,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),Lo,zo]),a(` +`),Jo])]),s("pre",Qo,[s("code",null,[Go,a(` +`),jo,a(` +`),s("span",Zo,[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,Ba,s("pre",Sa,[s("code",null,[Na,a(` +`),Ra,a(` +`),s("span",Da,[Oa,s("span",Va,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),Ia,fa]),a(` +`),$a])]),s("pre",wa,[s("code",null,[Wa,a(` +`),Ka,a(` +`),s("span",Ya,[Ma,s("span",xa,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),qa,Ha]),a(` +`),Ua])])])])]),La,za,Ja,Qa,Ga,ja,Za]))}});export{pn as __pageData,cn as default}; diff --git a/assets/nodes_docker-images.md.97bd64bd.lean.js b/assets/nodes_docker-images.md.97bd64bd.lean.js new file mode 100644 index 00000000000..ffcb83b87ba --- /dev/null +++ b/assets/nodes_docker-images.md.97bd64bd.lean.js @@ -0,0 +1,49 @@ +import{a as l}from"./chunks/arabica_versions.1930378b.js";import{m as t}from"./chunks/mocha_versions.7704b055.js";import{m as p}from"./chunks/mainnet_versions.955428b6.js";import{o as c,c as r,k as s,a,t as n,l as e,Q as o}from"./chunks/framework.1a91c06a.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),B={class:"shiki github-light vp-code-light"},S=o("",1),N={class:"line"},R=s("span",{style:{color:"#24292E"}}," ",-1),D={style:{color:"#032F62"}},O=s("span",{style:{color:"#24292E"}}," ",-1),V=s("span",{style:{color:"#005CC5"}},"\\",-1),I=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),W={class:"shiki github-dark vp-code-dark"},K=o("",1),Y={class:"line"},M=s("span",{style:{color:"#E1E4E8"}}," ",-1),x={style:{color:"#9ECBFF"}},q=s("span",{style:{color:"#E1E4E8"}}," ",-1),H=s("span",{style:{color:"#79B8FF"}},"\\",-1),U=o("",1),L={class:"shiki github-light vp-code-light"},z=o("",1),J={class:"line"},Q=s("span",{style:{color:"#24292E"}}," ",-1),G={style:{color:"#032F62"}},j=s("span",{style:{color:"#24292E"}}," ",-1),Z=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"},Bs=o("",1),Ss=o("",1),Ns={class:"line"},Rs=s("span",{style:{color:"#E1E4E8"}}," ",-1),Ds={style:{color:"#9ECBFF"}},Os=s("span",{style:{color:"#E1E4E8"}}," ",-1),Vs=s("span",{style:{color:"#79B8FF"}},"\\",-1),Is=o("",1),fs={class:"shiki github-light vp-code-light"},$s=o("",1),ws=o("",1),Ws={class:"line"},Ks=s("span",{style:{color:"#24292E"}}," ",-1),Ys={style:{color:"#032F62"}},Ms=s("span",{style:{color:"#24292E"}}," ",-1),xs=s("span",{style:{color:"#005CC5"}},"\\",-1),qs=o("",1),Hs={class:"language-bash vp-adaptive-theme"},Us=s("button",{title:"Copy Code",class:"copy"},null,-1),Ls=s("span",{class:"lang"},"bash",-1),zs={class:"shiki github-dark vp-code-dark"},Js=o("",1),Qs=o("",1),Gs={class:"line"},js=s("span",{style:{color:"#E1E4E8"}}," ",-1),Zs={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),Bo={class:"line"},So=s("span",{style:{color:"#24292E"}}," ",-1),No={style:{color:"#032F62"}},Ro=s("span",{style:{color:"#24292E"}}," ",-1),Do=s("span",{style:{color:"#005CC5"}},"\\",-1),Oo=o("",1),Vo=o("",4),Io={class:"vp-code-group vp-adaptive-theme"},fo=o("",1),$o={class:"blocks"},wo={class:"language-bash vp-adaptive-theme active"},Wo=s("button",{title:"Copy Code",class:"copy"},null,-1),Ko=s("span",{class:"lang"},"bash",-1),Yo={class:"shiki github-dark vp-code-dark"},Mo=o("",1),xo=o("",1),qo={class:"line"},Ho=s("span",{style:{color:"#E1E4E8"}}," ",-1),Uo={style:{color:"#9ECBFF"}},Lo=s("span",{style:{color:"#E1E4E8"}}," ",-1),zo=s("span",{style:{color:"#79B8FF"}},"\\",-1),Jo=o("",1),Qo={class:"shiki github-light vp-code-light"},Go=o("",1),jo=o("",1),Zo={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),Ba=s("span",{class:"lang"},"bash",-1),Sa={class:"shiki github-dark vp-code-dark"},Na=o("",1),Ra=o("",1),Da={class:"line"},Oa=s("span",{style:{color:"#E1E4E8"}}," ",-1),Va={style:{color:"#9ECBFF"}},Ia=s("span",{style:{color:"#E1E4E8"}}," ",-1),fa=s("span",{style:{color:"#79B8FF"}},"\\",-1),$a=o("",1),wa={class:"shiki github-light vp-code-light"},Wa=o("",1),Ka=o("",1),Ya={class:"line"},Ma=s("span",{style:{color:"#24292E"}}," ",-1),xa={style:{color:"#032F62"}},qa=s("span",{style:{color:"#24292E"}}," ",-1),Ha=s("span",{style:{color:"#005CC5"}},"\\",-1),Ua=o("",1),La=s("p",null,"Congratulations! You now have a node running with persistent storage.",-1),za=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),Qa=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),Ga=s("div",{class:"youtube-wrapper"},[s("iframe",{class:"youtube-video",title:"Running a Celestia light node",src:"https://youtube.com/embed/ROZv871Q7RM",allowfullscreen:""})],-1),ja=s("h2",{id:"troubleshooting",tabindex:"-1"},[a("Troubleshooting "),s("a",{class:"header-anchor",href:"#troubleshooting","aria-label":'Permalink to "Troubleshooting"'},"​")],-1),Za=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":1726496322000}'),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",B,[s("code",null,[S,a(` +`),s("span",N,[R,s("span",D,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),O,V]),a(` +`),I])])]),s("div",f,[$,w,s("pre",W,[s("code",null,[K,a(` +`),s("span",Y,[M,s("span",x,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),q,H]),a(` +`),U])]),s("pre",L,[s("code",null,[z,a(` +`),s("span",J,[Q,s("span",G,"ghcr.io/celestiaorg/celestia-node:"+n(e(t)["node-latest-tag"]),1),j,Z]),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,[Bs,a(` +`),Ss,a(` +`),s("span",Ns,[Rs,s("span",Ds,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),Os,Vs]),a(` +`),Is])]),s("pre",fs,[s("code",null,[$s,a(` +`),ws,a(` +`),s("span",Ws,[Ks,s("span",Ys,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),Ms,xs]),a(` +`),qs])])]),s("div",Hs,[Us,Ls,s("pre",zs,[s("code",null,[Js,a(` +`),Qs,a(` +`),s("span",Gs,[js,s("span",Zs,"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",Bo,[So,s("span",No,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),Ro,Do]),a(` +`),Oo])])])])]),Vo,s("div",Io,[fo,s("div",$o,[s("div",wo,[Wo,Ko,s("pre",Yo,[s("code",null,[Mo,a(` +`),xo,a(` +`),s("span",qo,[Ho,s("span",Uo,"ghcr.io/celestiaorg/celestia-node:"+n(e(p)["node-latest-tag"]),1),Lo,zo]),a(` +`),Jo])]),s("pre",Qo,[s("code",null,[Go,a(` +`),jo,a(` +`),s("span",Zo,[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,Ba,s("pre",Sa,[s("code",null,[Na,a(` +`),Ra,a(` +`),s("span",Da,[Oa,s("span",Va,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),Ia,fa]),a(` +`),$a])]),s("pre",wa,[s("code",null,[Wa,a(` +`),Ka,a(` +`),s("span",Ya,[Ma,s("span",xa,"ghcr.io/celestiaorg/celestia-node:"+n(e(l)["node-latest-tag"]),1),qa,Ha]),a(` +`),Ua])])])])]),La,za,Ja,Qa,Ga,ja,Za]))}});export{pn as __pageData,cn as default}; diff --git a/assets/nodes_environment.md.4596b94b.js b/assets/nodes_environment.md.4596b94b.js new file mode 100644 index 00000000000..968b559fbf6 --- /dev/null +++ b/assets/nodes_environment.md.4596b94b.js @@ -0,0 +1,51 @@ +import{c as a}from"./chunks/constants.fa173a21.js";import{o as p,c as e,k as s,t as n,l as o,Q as l}from"./chunks/framework.1a91c06a.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),M=s("span",{style:{color:"#F97583"}},"=",-1),H={style:{color:"#9ECBFF"}},T={class:"shiki github-light vp-code-light"},x={class:"line"},P=s("span",{style:{color:"#24292E"}},"ver",-1),D=s("span",{style:{color:"#D73A49"}},"=",-1),O={style:{color:"#032F62"}},I={class:"language-bash vp-adaptive-theme"},S=s("button",{title:"Copy Code",class:"copy"},null,-1),N=s("span",{class:"lang"},"bash",-1),j={class:"shiki github-dark vp-code-dark"},V={class:"line"},G=s("span",{style:{color:"#E1E4E8"}},"ver",-1),L=s("span",{style:{color:"#F97583"}},"=",-1),K={style:{color:"#9ECBFF"}},X={class:"shiki github-light vp-code-light"},U={class:"line"},Q=s("span",{style:{color:"#24292E"}},"ver",-1),R=s("span",{style:{color:"#D73A49"}},"=",-1),J={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

    TIP

    Use echo $SHELL to figure out what type of shell you are using!

  • 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":1723131620000}'),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,M,s("span",H,'"'+n(o(a).golangNodeMocha)+'"',1)])])]),s("pre",T,[s("code",null,[s("span",x,[P,D,s("span",O,'"'+n(o(a).golangNodeMocha)+'"',1)])])])]),s("div",I,[S,N,s("pre",j,[s("code",null,[s("span",V,[G,L,s("span",K,'"'+n(o(a).golangNodeArabica)+'"',1)])])]),s("pre",X,[s("code",null,[s("span",U,[Q,R,s("span",J,'"'+n(o(a).golangNodeArabica)+'"',1)])])])])])])]),Y]),Z]))}});export{ps as __pageData,es as default}; diff --git a/assets/nodes_environment.md.4596b94b.lean.js b/assets/nodes_environment.md.4596b94b.lean.js new file mode 100644 index 00000000000..5c94d305695 --- /dev/null +++ b/assets/nodes_environment.md.4596b94b.lean.js @@ -0,0 +1 @@ +import{c as a}from"./chunks/constants.fa173a21.js";import{o as p,c as e,k as s,t as n,l as o,Q as l}from"./chunks/framework.1a91c06a.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),M=s("span",{style:{color:"#F97583"}},"=",-1),H={style:{color:"#9ECBFF"}},T={class:"shiki github-light vp-code-light"},x={class:"line"},P=s("span",{style:{color:"#24292E"}},"ver",-1),D=s("span",{style:{color:"#D73A49"}},"=",-1),O={style:{color:"#032F62"}},I={class:"language-bash vp-adaptive-theme"},S=s("button",{title:"Copy Code",class:"copy"},null,-1),N=s("span",{class:"lang"},"bash",-1),j={class:"shiki github-dark vp-code-dark"},V={class:"line"},G=s("span",{style:{color:"#E1E4E8"}},"ver",-1),L=s("span",{style:{color:"#F97583"}},"=",-1),K={style:{color:"#9ECBFF"}},X={class:"shiki github-light vp-code-light"},U={class:"line"},Q=s("span",{style:{color:"#24292E"}},"ver",-1),R=s("span",{style:{color:"#D73A49"}},"=",-1),J={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":1723131620000}'),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,M,s("span",H,'"'+n(o(a).golangNodeMocha)+'"',1)])])]),s("pre",T,[s("code",null,[s("span",x,[P,D,s("span",O,'"'+n(o(a).golangNodeMocha)+'"',1)])])])]),s("div",I,[S,N,s("pre",j,[s("code",null,[s("span",V,[G,L,s("span",K,'"'+n(o(a).golangNodeArabica)+'"',1)])])]),s("pre",X,[s("code",null,[s("span",U,[Q,R,s("span",J,'"'+n(o(a).golangNodeArabica)+'"',1)])])])])])])]),Y]),Z]))}});export{ps as __pageData,es as default}; diff --git a/assets/nodes_full-storage-node.md.16ca408c.js b/assets/nodes_full-storage-node.md.16ca408c.js new file mode 100644 index 00000000000..fe1206d894c --- /dev/null +++ b/assets/nodes_full-storage-node.md.16ca408c.js @@ -0,0 +1,11 @@ +import{_ as s,o as a,c as e,Q as o}from"./chunks/framework.1a91c06a.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":1726496322000}'),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 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: 16 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 Mainnet Beta, Mocha testnet or Arabica devnet, start your node.

    Connecting to a core endpoint with --core.ip string provides the light node with access to state queries (reading balances, submitting transactions, and other state-related queries).

    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.keyname <name-of-custom-key> \\
    celestia full start --core.ip <URI> \\
    +  --keyring.keyname <name-of-custom-key> \\
    sh
    celestia full start --core.ip <URI> \\
    +  --keyring.keyname <name-of-custom-key> \\
    +  --p2p.network mocha
    celestia full start --core.ip <URI> \\
    +  --keyring.keyname <name-of-custom-key> \\
    +  --p2p.network mocha
    sh
    celestia full start --core.ip <URI> \\
    +  --keyring.keyname <name-of-custom-key> \\
    +  --p2p.network arabica
    celestia full start --core.ip <URI> \\
    +  --keyring.keyname <name-of-custom-key> \\
    +  --p2p.network arabica

    Optional: Migrate node id to another server

    To migrate a full storage node ID:

    1. You need to back up two files located in the celestia-full node directory at the correct path (default: ~/.celestia-full/keys).
    2. Upload the files to the new server and start the node.

    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.

    `,44),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.16ca408c.lean.js b/assets/nodes_full-storage-node.md.16ca408c.lean.js new file mode 100644 index 00000000000..d778ff8f664 --- /dev/null +++ b/assets/nodes_full-storage-node.md.16ca408c.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as e,Q as o}from"./chunks/framework.1a91c06a.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":1726496322000}'),n={name:"nodes/full-storage-node.md"},t=o("",44),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_ibc-relayer.md.6041bb8b.js b/assets/nodes_ibc-relayer.md.6041bb8b.js new file mode 100644 index 00000000000..88a3b0bacd1 --- /dev/null +++ b/assets/nodes_ibc-relayer.md.6041bb8b.js @@ -0,0 +1,455 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.1a91c06a.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":1726693572000}'),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.6041bb8b.lean.js b/assets/nodes_ibc-relayer.md.6041bb8b.lean.js new file mode 100644 index 00000000000..9856172b925 --- /dev/null +++ b/assets/nodes_ibc-relayer.md.6041bb8b.lean.js @@ -0,0 +1 @@ +import{_ as s,o as n,c as a,Q as l}from"./chunks/framework.1a91c06a.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":1726693572000}'),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.b0829848.js b/assets/nodes_instantiate-testnet.md.b0829848.js new file mode 100644 index 00000000000..3204092df69 --- /dev/null +++ b/assets/nodes_instantiate-testnet.md.b0829848.js @@ -0,0 +1,21 @@ +import{_ as e,o as s,c as a,Q as n}from"./chunks/framework.1a91c06a.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.b0829848.lean.js b/assets/nodes_instantiate-testnet.md.b0829848.lean.js new file mode 100644 index 00000000000..61a45d34d3b --- /dev/null +++ b/assets/nodes_instantiate-testnet.md.b0829848.lean.js @@ -0,0 +1 @@ +import{_ as e,o as s,c as a,Q as n}from"./chunks/framework.1a91c06a.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.768fa4a4.js b/assets/nodes_light-node.md.768fa4a4.js new file mode 100644 index 00000000000..1447ee5bcfd --- /dev/null +++ b/assets/nodes_light-node.md.768fa4a4.js @@ -0,0 +1,23 @@ +import{_ as s,o as a,c as e,Q as n}from"./chunks/framework.1a91c06a.js";const o="/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":1726496322000}'),l={name:"nodes/light-node.md"},t=n('

    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: 100 GB SSD Storage
    • Bandwidth: 56 Kbps for Download/56 Kbps for Upload

    Quickstart: Run a light node in your browser

    The easiest way to run a Celestia light node is with Lumina.rs in your browser.

    Lumina.rs in browser

    You can also run Lumina on the first decentralized block explorer, Celenium.

    Celenium running a light node with Lumina.rs

    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.

    sh
    celestia light start --core.ip rpc.celestia.pops.one --p2p.network celestia
    celestia light start --core.ip rpc.celestia.pops.one --p2p.network celestia
    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
    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

    Tip: you can replace the core.ip with a consensus node RPC endpoint from Mainnet Beta, Mocha testnet, or Arabica devnet.

    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.keyname my_celes_key \\
    +    --core.ip consensus.lunaroasis.net
    celestia light start --keyring.keyname my_celes_key \\
    +    --core.ip consensus.lunaroasis.net
    sh
    celestia light start --keyring.keyname my_celes_key \\
    +    --core.ip rpc-mocha.pops.one --p2p.network mocha
    celestia light start --keyring.keyname my_celes_key \\
    +    --core.ip rpc-mocha.pops.one --p2p.network mocha
    sh
    celestia light start --keyring.keyname my_celes_key \\
    +    --core.ip validator-1.celestia-arabica-11.com \\
    +    --p2p.network arabica
    celestia light start --keyring.keyname 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.keyname <name-of-custom-key> \\
    celestia light start --core.ip <URI> \\
    +    --keyring.keyname <name-of-custom-key> \\
    sh
    celestia light start --core.ip <URI> \\
    +    --keyring.keyname <name-of-custom-key> \\
    +    --p2p.network arabica
    celestia light start --core.ip <URI> \\
    +    --keyring.keyname <name-of-custom-key> \\
    +    --p2p.network arabica
    sh
    celestia light start --core.ip <URI> \\
    +    --keyring.keyname <name-of-custom-key> \\
    +    --p2p.network mocha
    celestia light start --core.ip <URI> \\
    +    --keyring.keyname <name-of-custom-key> \\
    +    --p2p.network mocha

    Optional: Migrate node id to another server

    To migrate a light node ID:

    1. You need to back up two files located in the celestia-light node directory at the correct path (default: ~/.celestia-light/keys).
    2. Upload the files to the new server and start the node.

    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.

    `,55),p=[t];function r(c,i,y,d,E,h){return a(),e("div",null,p)}const u=s(l,[["render",r]]);export{F as __pageData,u as default}; diff --git a/assets/nodes_light-node.md.768fa4a4.lean.js b/assets/nodes_light-node.md.768fa4a4.lean.js new file mode 100644 index 00000000000..26faed77f9d --- /dev/null +++ b/assets/nodes_light-node.md.768fa4a4.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as e,Q as n}from"./chunks/framework.1a91c06a.js";const o="/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":1726496322000}'),l={name:"nodes/light-node.md"},t=n("",55),p=[t];function r(c,i,y,d,E,h){return a(),e("div",null,p)}const u=s(l,[["render",r]]);export{F as __pageData,u as default}; diff --git a/assets/nodes_mainnet.md.37db891f.js b/assets/nodes_mainnet.md.37db891f.js new file mode 100644 index 00000000000..55d2d30a1f8 --- /dev/null +++ b/assets/nodes_mainnet.md.37db891f.js @@ -0,0 +1 @@ +import{M as p}from"./chunks/MainnetVersionTags.4fd38854.js";import{c as h}from"./chunks/constants.fa173a21.js";import{_ as m,o as s,c as r,k as e,t as o,C as n,H as a,w as c,Q as i,a as u}from"./chunks/framework.1a91c06a.js";import"./chunks/mainnet_versions.955428b6.js";const f="/img/Mainnet-Beta.png",b="/grove/grove-sandbox.png",y={name:"MainnetBetaDetails",data(){return{constants:h}}},g=e("tr",null,[e("th",null,"Detail"),e("th",null,"Value")],-1),H=e("td",null,"Chain ID",-1),k=e("tr",null,[e("td",null,"Genesis hash"),e("td",null,[e("code",null,"6BE39EFD10BA412A9DB5288488303F5DD32CF386707A5BEF33617F4C43301872")])],-1),V=e("td",null,"Genesis file",-1),w=["href"],L=e("td",null,"Peers file",-1),v=["href"],_=e("tr",null,[e("td",null,"Validators"),e("td",null," 100 ")],-1);function x(d,T,Q,l,t,F){return s(),r("table",null,[g,e("tr",null,[H,e("td",null,[e("code",null,o(t.constants.mainnetChainId),1)])]),k,e("tr",null,[V,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/"+o(t.constants.mainnetChainId)+"/genesis.json ",9,w)])]),e("tr",null,[L,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/"+o(t.constants.mainnetChainId)+"/peers.txt ",9,v)])]),_])}const M=m(y,[["render",x]]),C=i('

    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 Beta 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),P=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),Z=i('

    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),E={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"},q=e("g",{stroke:"currentColor",fill:"currentColor","stroke-width":"0",transform:"scale(1,-1)"},[e("g",{"data-mml-node":"math"},[e("g",{"data-mml-node":"mtext"},[e("path",{"data-c":"54",d:"M36 443Q37 448 46 558T55 671V677H666V671Q667 666 676 556T685 443V437H645V443Q645 445 642 478T631 544T610 593Q593 614 555 625Q534 630 478 630H451H443Q417 630 414 618Q413 616 413 339V63Q420 53 439 50T528 46H558V0H545L361 3Q186 1 177 0H164V46H194Q264 46 283 49T309 63V339V550Q309 620 304 625T271 630H244H224Q154 630 119 601Q101 585 93 554T81 486T76 443V437H36V443Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"6F",d:"M28 214Q28 309 93 378T250 448Q340 448 405 380T471 215Q471 120 407 55T250 -10Q153 -10 91 57T28 214ZM250 30Q372 30 372 193V225V250Q372 272 371 288T364 326T348 362T317 390T268 410Q263 411 252 411Q222 411 195 399Q152 377 139 338T126 246V226Q126 130 145 91Q177 30 250 30Z",transform:"translate(722,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"74",d:"M27 422Q80 426 109 478T141 600V615H181V431H316V385H181V241Q182 116 182 100T189 68Q203 29 238 29Q282 29 292 100Q293 108 293 146V181H333V146V134Q333 57 291 17Q264 -10 221 -10Q187 -10 162 2T124 33T105 68T98 100Q97 107 97 248V385H18V422H27Z",transform:"translate(1222,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"61",d:"M137 305T115 305T78 320T63 359Q63 394 97 421T218 448Q291 448 336 416T396 340Q401 326 401 309T402 194V124Q402 76 407 58T428 40Q443 40 448 56T453 109V145H493V106Q492 66 490 59Q481 29 455 12T400 -6T353 12T329 54V58L327 55Q325 52 322 49T314 40T302 29T287 17T269 6T247 -2T221 -8T190 -11Q130 -11 82 20T34 107Q34 128 41 147T68 188T116 225T194 253T304 268H318V290Q318 324 312 340Q290 411 215 411Q197 411 181 410T156 406T148 403Q170 388 170 359Q170 334 154 320ZM126 106Q126 75 150 51T209 26Q247 26 276 49T315 109Q317 116 318 175Q318 233 317 233Q309 233 296 232T251 223T193 203T147 166T126 106Z",transform:"translate(1611,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"6C",d:"M42 46H56Q95 46 103 60V68Q103 77 103 91T103 124T104 167T104 217T104 272T104 329Q104 366 104 407T104 482T104 542T103 586T103 603Q100 622 89 628T44 637H26V660Q26 683 28 683L38 684Q48 685 67 686T104 688Q121 689 141 690T171 693T182 694H185V379Q185 62 186 60Q190 52 198 49Q219 46 247 46H263V0H255L232 1Q209 2 183 2T145 3T107 3T57 1L34 0H26V46H42Z",transform:"translate(2111,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"20",d:"",transform:"translate(2389,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"42",d:"M131 622Q124 629 120 631T104 634T61 637H28V683H229H267H346Q423 683 459 678T531 651Q574 627 599 590T624 512Q624 461 583 419T476 360L466 357Q539 348 595 302T651 187Q651 119 600 67T469 3Q456 1 242 0H28V46H61Q103 47 112 49T131 61V622ZM511 513Q511 560 485 594T416 636Q415 636 403 636T371 636T333 637Q266 637 251 636T232 628Q229 624 229 499V374H312L396 375L406 377Q410 378 417 380T442 393T474 417T499 456T511 513ZM537 188Q537 239 509 282T430 336L329 337H229V200V116Q229 57 234 52Q240 47 334 47H383Q425 47 443 53Q486 67 511 104T537 188Z",transform:"translate(2639,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"79",d:"M69 -66Q91 -66 104 -80T118 -116Q118 -134 109 -145T91 -160Q84 -163 97 -166Q104 -168 111 -168Q131 -168 148 -159T175 -138T197 -106T213 -75T225 -43L242 0L170 183Q150 233 125 297Q101 358 96 368T80 381Q79 382 78 382Q66 385 34 385H19V431H26L46 430Q65 430 88 429T122 428Q129 428 142 428T171 429T200 430T224 430L233 431H241V385H232Q183 385 185 366L286 112Q286 113 332 227L376 341V350Q376 365 366 373T348 383T334 385H331V431H337H344Q351 431 361 431T382 430T405 429T422 429Q477 429 503 431H508V385H497Q441 380 422 345Q420 343 378 235T289 9T227 -131Q180 -204 113 -204Q69 -204 44 -177T19 -116Q19 -89 35 -78T69 -66Z",transform:"translate(3347,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"74",d:"M27 422Q80 426 109 478T141 600V615H181V431H316V385H181V241Q182 116 182 100T189 68Q203 29 238 29Q282 29 292 100Q293 108 293 146V181H333V146V134Q333 57 291 17Q264 -10 221 -10Q187 -10 162 2T124 33T105 68T98 100Q97 107 97 248V385H18V422H27Z",transform:"translate(3875,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"65",d:"M28 218Q28 273 48 318T98 391T163 433T229 448Q282 448 320 430T378 380T406 316T415 245Q415 238 408 231H126V216Q126 68 226 36Q246 30 270 30Q312 30 342 62Q359 79 369 104L379 128Q382 131 395 131H398Q415 131 415 121Q415 117 412 108Q393 53 349 21T250 -11Q155 -11 92 58T28 218ZM333 275Q322 403 238 411H236Q228 411 220 410T195 402T166 381T143 340T127 274V267H333V275Z",transform:"translate(4264,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"73",d:"M295 316Q295 356 268 385T190 414Q154 414 128 401Q98 382 98 349Q97 344 98 336T114 312T157 287Q175 282 201 278T245 269T277 256Q294 248 310 236T342 195T359 133Q359 71 321 31T198 -10H190Q138 -10 94 26L86 19L77 10Q71 4 65 -1L54 -11H46H42Q39 -11 33 -5V74V132Q33 153 35 157T45 162H54Q66 162 70 158T75 146T82 119T101 77Q136 26 198 26Q295 26 295 104Q295 133 277 151Q257 175 194 187T111 210Q75 227 54 256T33 318Q33 357 50 384T93 424T143 442T187 447H198Q238 447 268 432L283 424L292 431Q302 440 314 448H322H326Q329 448 335 442V310L329 304H301Q295 310 295 316Z",transform:"translate(4708,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(5379.8,0)"},[e("path",{"data-c":"3D",d:"M56 347Q56 360 70 367H707Q722 359 722 347Q722 336 708 328L390 327H72Q56 332 56 347ZM56 153Q56 168 72 173H708Q722 163 722 153Q722 140 707 133H70Q56 140 56 153Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(6435.6,0)"},[e("path",{"data-c":"28",d:"M94 250Q94 319 104 381T127 488T164 576T202 643T244 695T277 729T302 750H315H319Q333 750 333 741Q333 738 316 720T275 667T226 581T184 443T167 250T184 58T225 -81T274 -167T316 -220T333 -241Q333 -250 318 -250H315H302L274 -226Q180 -141 137 -14T94 250Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mn",transform:"translate(6824.6,0)"},[e("path",{"data-c":"31",d:"M213 578L200 573Q186 568 160 563T102 556H83V602H102Q149 604 189 617T245 641T273 663Q275 666 285 666Q294 666 302 660V361L303 61Q310 54 315 52T339 48T401 46H427V0H416Q395 3 257 3Q121 3 100 0H88V46H114Q136 46 152 46T177 47T193 50T201 52T207 57T213 61V578Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(7546.8,0)"},[e("path",{"data-c":"D7",d:"M630 29Q630 9 609 9Q604 9 587 25T493 118L389 222L284 117Q178 13 175 11Q171 9 168 9Q160 9 154 15T147 29Q147 36 161 51T255 146L359 250L255 354Q174 435 161 449T147 471Q147 480 153 485T168 490Q173 490 175 489Q178 487 284 383L389 278L493 382Q570 459 587 475T609 491Q630 491 630 471Q630 464 620 453T522 355L418 250L522 145Q606 61 618 48T630 29Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mn",transform:"translate(8547,0)"},[e("path",{"data-c":"34",d:"M462 0Q444 3 333 3Q217 3 199 0H190V46H221Q241 46 248 46T265 48T279 53T286 61Q287 63 287 115V165H28V211L179 442Q332 674 334 675Q336 677 355 677H373L379 671V211H471V165H379V114Q379 73 379 66T385 54Q393 47 442 46H471V0H462ZM293 211V545L74 212L183 211H293Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"37",d:"M55 458Q56 460 72 567L88 674Q88 676 108 676H128V672Q128 662 143 655T195 646T364 644H485V605L417 512Q408 500 387 472T360 435T339 403T319 367T305 330T292 284T284 230T278 162T275 80Q275 66 275 52T274 28V19Q270 2 255 -10T221 -22Q210 -22 200 -19T179 0T168 40Q168 198 265 368Q285 400 349 489L395 552H302Q128 552 119 546Q113 543 108 522T98 479L95 458V455H55V458Z",transform:"translate(500,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"38",d:"M70 417T70 494T124 618T248 666Q319 666 374 624T429 515Q429 485 418 459T392 417T361 389T335 371T324 363L338 354Q352 344 366 334T382 323Q457 264 457 174Q457 95 399 37T249 -22Q159 -22 101 29T43 155Q43 263 172 335L154 348Q133 361 127 368Q70 417 70 494ZM286 386L292 390Q298 394 301 396T311 403T323 413T334 425T345 438T355 454T364 471T369 491T371 513Q371 556 342 586T275 624Q268 625 242 625Q201 625 165 599T128 534Q128 511 141 492T167 463T217 431Q224 426 228 424L286 386ZM250 21Q308 21 350 55T392 137Q392 154 387 169T375 194T353 216T330 234T301 253T274 270Q260 279 244 289T218 306L210 311Q204 311 181 294T133 239T107 157Q107 98 150 60T250 21Z",transform:"translate(1000,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mstyle",transform:"translate(10047,0)"},[e("g",{"data-mml-node":"mspace"})]),e("g",{"data-mml-node":"mtext",transform:"translate(10214,0)"},[e("path",{"data-c":"62",d:"M307 -11Q234 -11 168 55L158 37Q156 34 153 28T147 17T143 10L138 1L118 0H98V298Q98 599 97 603Q94 622 83 628T38 637H20V660Q20 683 22 683L32 684Q42 685 61 686T98 688Q115 689 135 690T165 693T176 694H179V543Q179 391 180 391L183 394Q186 397 192 401T207 411T228 421T254 431T286 439T323 442Q401 442 461 379T522 216Q522 115 458 52T307 -11ZM182 98Q182 97 187 90T196 79T206 67T218 55T233 44T250 35T271 29T295 26Q330 26 363 46T412 113Q424 148 424 212Q424 287 412 323Q385 405 300 405Q270 405 239 390T188 347L182 339V98Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"79",d:"M69 -66Q91 -66 104 -80T118 -116Q118 -134 109 -145T91 -160Q84 -163 97 -166Q104 -168 111 -168Q131 -168 148 -159T175 -138T197 -106T213 -75T225 -43L242 0L170 183Q150 233 125 297Q101 358 96 368T80 381Q79 382 78 382Q66 385 34 385H19V431H26L46 430Q65 430 88 429T122 428Q129 428 142 428T171 429T200 430T224 430L233 431H241V385H232Q183 385 185 366L286 112Q286 113 332 227L376 341V350Q376 365 366 373T348 383T334 385H331V431H337H344Q351 431 361 431T382 430T405 429T422 429Q477 429 503 431H508V385H497Q441 380 422 345Q420 343 378 235T289 9T227 -131Q180 -204 113 -204Q69 -204 44 -177T19 -116Q19 -89 35 -78T69 -66Z",transform:"translate(556,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"74",d:"M27 422Q80 426 109 478T141 600V615H181V431H316V385H181V241Q182 116 182 100T189 68Q203 29 238 29Q282 29 292 100Q293 108 293 146V181H333V146V134Q333 57 291 17Q264 -10 221 -10Q187 -10 162 2T124 33T105 68T98 100Q97 107 97 248V385H18V422H27Z",transform:"translate(1084,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"65",d:"M28 218Q28 273 48 318T98 391T163 433T229 448Q282 448 320 430T378 380T406 316T415 245Q415 238 408 231H126V216Q126 68 226 36Q246 30 270 30Q312 30 342 62Q359 79 369 104L379 128Q382 131 395 131H398Q415 131 415 121Q415 117 412 108Q393 53 349 21T250 -11Q155 -11 92 58T28 218ZM333 275Q322 403 238 411H236Q228 411 220 410T195 402T166 381T143 340T127 274V267H333V275Z",transform:"translate(1473,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"73",d:"M295 316Q295 356 268 385T190 414Q154 414 128 401Q98 382 98 349Q97 344 98 336T114 312T157 287Q175 282 201 278T245 269T277 256Q294 248 310 236T342 195T359 133Q359 71 321 31T198 -10H190Q138 -10 94 26L86 19L77 10Q71 4 65 -1L54 -11H46H42Q39 -11 33 -5V74V132Q33 153 35 157T45 162H54Q66 162 70 158T75 146T82 119T101 77Q136 26 198 26Q295 26 295 104Q295 133 277 151Q257 175 194 187T111 210Q75 227 54 256T33 318Q33 357 50 384T93 424T143 442T187 447H198Q238 447 268 432L283 424L292 431Q302 440 314 448H322H326Q329 448 335 442V310L329 304H301Q295 310 295 316Z",transform:"translate(1917,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(12525,0)"},[e("path",{"data-c":"29",d:"M60 749L64 750Q69 750 74 750H86L114 726Q208 641 251 514T294 250Q294 182 284 119T261 12T224 -76T186 -143T145 -194T113 -227T90 -246Q87 -249 86 -250H74Q66 -250 63 -250T58 -247T55 -238Q56 -237 66 -225Q221 -64 221 250T66 725Q56 737 55 738Q55 746 60 749Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(13136.2,0)"},[e("path",{"data-c":"2B",d:"M56 237T56 250T70 270H369V420L370 570Q380 583 389 583Q402 583 409 568V270H707Q722 262 722 250T707 230H409V-68Q401 -82 391 -82H389H387Q375 -82 369 -68V230H70Q56 237 56 250Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(14136.4,0)"},[e("path",{"data-c":"28",d:"M94 250Q94 319 104 381T127 488T164 576T202 643T244 695T277 729T302 750H315H319Q333 750 333 741Q333 738 316 720T275 667T226 581T184 443T167 250T184 58T225 -81T274 -167T316 -220T333 -241Q333 -250 318 -250H315H302L274 -226Q180 -141 137 -14T94 250Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mn",transform:"translate(14525.4,0)"},[e("path",{"data-c":"34",d:"M462 0Q444 3 333 3Q217 3 199 0H190V46H221Q241 46 248 46T265 48T279 53T286 61Q287 63 287 115V165H28V211L179 442Q332 674 334 675Q336 677 355 677H373L379 671V211H471V165H379V114Q379 73 379 66T385 54Q393 47 442 46H471V0H462ZM293 211V545L74 212L183 211H293Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"30",d:"M96 585Q152 666 249 666Q297 666 345 640T423 548Q460 465 460 320Q460 165 417 83Q397 41 362 16T301 -15T250 -22Q224 -22 198 -16T137 16T82 83Q39 165 39 320Q39 494 96 585ZM321 597Q291 629 250 629Q208 629 178 597Q153 571 145 525T137 333Q137 175 145 125T181 46Q209 16 250 16Q290 16 318 46Q347 76 354 130T362 333Q362 478 354 524T321 597Z",transform:"translate(500,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"39",d:"M352 287Q304 211 232 211Q154 211 104 270T44 396Q42 412 42 436V444Q42 537 111 606Q171 666 243 666Q245 666 249 666T257 665H261Q273 665 286 663T323 651T370 619T413 560Q456 472 456 334Q456 194 396 97Q361 41 312 10T208 -22Q147 -22 108 7T68 93T121 149Q143 149 158 135T173 96Q173 78 164 65T148 49T135 44L131 43Q131 41 138 37T164 27T206 22H212Q272 22 313 86Q352 142 352 280V287ZM244 248Q292 248 321 297T351 430Q351 508 343 542Q341 552 337 562T323 588T293 615T246 625Q208 625 181 598Q160 576 154 546T147 441Q147 358 152 329T172 282Q197 248 244 248Z",transform:"translate(1000,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"34",d:"M462 0Q444 3 333 3Q217 3 199 0H190V46H221Q241 46 248 46T265 48T279 53T286 61Q287 63 287 115V165H28V211L179 442Q332 674 334 675Q336 677 355 677H373L379 671V211H471V165H379V114Q379 73 379 66T385 54Q393 47 442 46H471V0H462ZM293 211V545L74 212L183 211H293Z",transform:"translate(1500,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(16747.7,0)"},[e("path",{"data-c":"D7",d:"M630 29Q630 9 609 9Q604 9 587 25T493 118L389 222L284 117Q178 13 175 11Q171 9 168 9Q160 9 154 15T147 29Q147 36 161 51T255 146L359 250L255 354Q174 435 161 449T147 471Q147 480 153 485T168 490Q173 490 175 489Q178 487 284 383L389 278L493 382Q570 459 587 475T609 491Q630 491 630 471Q630 464 620 453T522 355L418 250L522 145Q606 61 618 48T630 29Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mn",transform:"translate(17747.9,0)"},[e("path",{"data-c":"34",d:"M462 0Q444 3 333 3Q217 3 199 0H190V46H221Q241 46 248 46T265 48T279 53T286 61Q287 63 287 115V165H28V211L179 442Q332 674 334 675Q336 677 355 677H373L379 671V211H471V165H379V114Q379 73 379 66T385 54Q393 47 442 46H471V0H462ZM293 211V545L74 212L183 211H293Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"38",d:"M70 417T70 494T124 618T248 666Q319 666 374 624T429 515Q429 485 418 459T392 417T361 389T335 371T324 363L338 354Q352 344 366 334T382 323Q457 264 457 174Q457 95 399 37T249 -22Q159 -22 101 29T43 155Q43 263 172 335L154 348Q133 361 127 368Q70 417 70 494ZM286 386L292 390Q298 394 301 396T311 403T323 413T334 425T345 438T355 454T364 471T369 491T371 513Q371 556 342 586T275 624Q268 625 242 625Q201 625 165 599T128 534Q128 511 141 492T167 463T217 431Q224 426 228 424L286 386ZM250 21Q308 21 350 55T392 137Q392 154 387 169T375 194T353 216T330 234T301 253T274 270Q260 279 244 289T218 306L210 311Q204 311 181 294T133 239T107 157Q107 98 150 60T250 21Z",transform:"translate(500,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"32",d:"M109 429Q82 429 66 447T50 491Q50 562 103 614T235 666Q326 666 387 610T449 465Q449 422 429 383T381 315T301 241Q265 210 201 149L142 93L218 92Q375 92 385 97Q392 99 409 186V189H449V186Q448 183 436 95T421 3V0H50V19V31Q50 38 56 46T86 81Q115 113 136 137Q145 147 170 174T204 211T233 244T261 278T284 308T305 340T320 369T333 401T340 431T343 464Q343 527 309 573T212 619Q179 619 154 602T119 569T109 550Q109 549 114 549Q132 549 151 535T170 489Q170 464 154 447T109 429Z",transform:"translate(1000,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mstyle",transform:"translate(19247.9,0)"},[e("g",{"data-mml-node":"mspace"})]),e("g",{"data-mml-node":"mtext",transform:"translate(19414.9,0)"},[e("path",{"data-c":"62",d:"M307 -11Q234 -11 168 55L158 37Q156 34 153 28T147 17T143 10L138 1L118 0H98V298Q98 599 97 603Q94 622 83 628T38 637H20V660Q20 683 22 683L32 684Q42 685 61 686T98 688Q115 689 135 690T165 693T176 694H179V543Q179 391 180 391L183 394Q186 397 192 401T207 411T228 421T254 431T286 439T323 442Q401 442 461 379T522 216Q522 115 458 52T307 -11ZM182 98Q182 97 187 90T196 79T206 67T218 55T233 44T250 35T271 29T295 26Q330 26 363 46T412 113Q424 148 424 212Q424 287 412 323Q385 405 300 405Q270 405 239 390T188 347L182 339V98Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"79",d:"M69 -66Q91 -66 104 -80T118 -116Q118 -134 109 -145T91 -160Q84 -163 97 -166Q104 -168 111 -168Q131 -168 148 -159T175 -138T197 -106T213 -75T225 -43L242 0L170 183Q150 233 125 297Q101 358 96 368T80 381Q79 382 78 382Q66 385 34 385H19V431H26L46 430Q65 430 88 429T122 428Q129 428 142 428T171 429T200 430T224 430L233 431H241V385H232Q183 385 185 366L286 112Q286 113 332 227L376 341V350Q376 365 366 373T348 383T334 385H331V431H337H344Q351 431 361 431T382 430T405 429T422 429Q477 429 503 431H508V385H497Q441 380 422 345Q420 343 378 235T289 9T227 -131Q180 -204 113 -204Q69 -204 44 -177T19 -116Q19 -89 35 -78T69 -66Z",transform:"translate(556,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"74",d:"M27 422Q80 426 109 478T141 600V615H181V431H316V385H181V241Q182 116 182 100T189 68Q203 29 238 29Q282 29 292 100Q293 108 293 146V181H333V146V134Q333 57 291 17Q264 -10 221 -10Q187 -10 162 2T124 33T105 68T98 100Q97 107 97 248V385H18V422H27Z",transform:"translate(1084,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"65",d:"M28 218Q28 273 48 318T98 391T163 433T229 448Q282 448 320 430T378 380T406 316T415 245Q415 238 408 231H126V216Q126 68 226 36Q246 30 270 30Q312 30 342 62Q359 79 369 104L379 128Q382 131 395 131H398Q415 131 415 121Q415 117 412 108Q393 53 349 21T250 -11Q155 -11 92 58T28 218ZM333 275Q322 403 238 411H236Q228 411 220 410T195 402T166 381T143 340T127 274V267H333V275Z",transform:"translate(1473,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"73",d:"M295 316Q295 356 268 385T190 414Q154 414 128 401Q98 382 98 349Q97 344 98 336T114 312T157 287Q175 282 201 278T245 269T277 256Q294 248 310 236T342 195T359 133Q359 71 321 31T198 -10H190Q138 -10 94 26L86 19L77 10Q71 4 65 -1L54 -11H46H42Q39 -11 33 -5V74V132Q33 153 35 157T45 162H54Q66 162 70 158T75 146T82 119T101 77Q136 26 198 26Q295 26 295 104Q295 133 277 151Q257 175 194 187T111 210Q75 227 54 256T33 318Q33 357 50 384T93 424T143 442T187 447H198Q238 447 268 432L283 424L292 431Q302 440 314 448H322H326Q329 448 335 442V310L329 304H301Q295 310 295 316Z",transform:"translate(1917,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(21725.9,0)"},[e("path",{"data-c":"29",d:"M60 749L64 750Q69 750 74 750H86L114 726Q208 641 251 514T294 250Q294 182 284 119T261 12T224 -76T186 -143T145 -194T113 -227T90 -246Q87 -249 86 -250H74Q66 -250 63 -250T58 -247T55 -238Q56 -237 66 -225Q221 -64 221 250T66 725Q56 737 55 738Q55 746 60 749Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(22392.7,0)"},[e("path",{"data-c":"3D",d:"M56 347Q56 360 70 367H707Q722 359 722 347Q722 336 708 328L390 327H72Q56 332 56 347ZM56 153Q56 168 72 173H708Q722 163 722 153Q722 140 707 133H70Q56 140 56 153Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mn",transform:"translate(23448.4,0)"},[e("path",{"data-c":"31",d:"M213 578L200 573Q186 568 160 563T102 556H83V602H102Q149 604 189 617T245 641T273 663Q275 666 285 666Q294 666 302 660V361L303 61Q310 54 315 52T339 48T401 46H427V0H416Q395 3 257 3Q121 3 100 0H88V46H114Q136 46 152 46T177 47T193 50T201 52T207 57T213 61V578Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(23948.4,0)"},[e("path",{"data-c":"2C",d:"M78 35T78 60T94 103T137 121Q165 121 187 96T210 8Q210 -27 201 -60T180 -117T154 -158T130 -185T117 -194Q113 -194 104 -185T95 -172Q95 -168 106 -156T131 -126T157 -76T173 -3V9L172 8Q170 7 167 6T161 3T152 1T140 0Q113 0 96 17Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mn",transform:"translate(24393.1,0)"},[e("path",{"data-c":"39",d:"M352 287Q304 211 232 211Q154 211 104 270T44 396Q42 412 42 436V444Q42 537 111 606Q171 666 243 666Q245 666 249 666T257 665H261Q273 665 286 663T323 651T370 619T413 560Q456 472 456 334Q456 194 396 97Q361 41 312 10T208 -22Q147 -22 108 7T68 93T121 149Q143 149 158 135T173 96Q173 78 164 65T148 49T135 44L131 43Q131 41 138 37T164 27T206 22H212Q272 22 313 86Q352 142 352 280V287ZM244 248Q292 248 321 297T351 430Q351 508 343 542Q341 552 337 562T323 588T293 615T246 625Q208 625 181 598Q160 576 154 546T147 441Q147 358 152 329T172 282Q197 248 244 248Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"37",d:"M55 458Q56 460 72 567L88 674Q88 676 108 676H128V672Q128 662 143 655T195 646T364 644H485V605L417 512Q408 500 387 472T360 435T339 403T319 367T305 330T292 284T284 230T278 162T275 80Q275 66 275 52T274 28V19Q270 2 255 -10T221 -22Q210 -22 200 -19T179 0T168 40Q168 198 265 368Q285 400 349 489L395 552H302Q128 552 119 546Q113 543 108 522T98 479L95 458V455H55V458Z",transform:"translate(500,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"33",d:"M127 463Q100 463 85 480T69 524Q69 579 117 622T233 665Q268 665 277 664Q351 652 390 611T430 522Q430 470 396 421T302 350L299 348Q299 347 308 345T337 336T375 315Q457 262 457 175Q457 96 395 37T238 -22Q158 -22 100 21T42 130Q42 158 60 175T105 193Q133 193 151 175T169 130Q169 119 166 110T159 94T148 82T136 74T126 70T118 67L114 66Q165 21 238 21Q293 21 321 74Q338 107 338 175V195Q338 290 274 322Q259 328 213 329L171 330L168 332Q166 335 166 348Q166 366 174 366Q202 366 232 371Q266 376 294 413T322 525V533Q322 590 287 612Q265 626 240 626Q208 626 181 615T143 592T132 580H135Q138 579 143 578T153 573T165 566T175 555T183 540T186 520Q186 498 172 481T127 463Z",transform:"translate(1000,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(25893.1,0)"},[e("path",{"data-c":"2C",d:"M78 35T78 60T94 103T137 121Q165 121 187 96T210 8Q210 -27 201 -60T180 -117T154 -158T130 -185T117 -194Q113 -194 104 -185T95 -172Q95 -168 106 -156T131 -126T157 -76T173 -3V9L172 8Q170 7 167 6T161 3T152 1T140 0Q113 0 96 17Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mn",transform:"translate(26337.8,0)"},[e("path",{"data-c":"37",d:"M55 458Q56 460 72 567L88 674Q88 676 108 676H128V672Q128 662 143 655T195 646T364 644H485V605L417 512Q408 500 387 472T360 435T339 403T319 367T305 330T292 284T284 230T278 162T275 80Q275 66 275 52T274 28V19Q270 2 255 -10T221 -22Q210 -22 200 -19T179 0T168 40Q168 198 265 368Q285 400 349 489L395 552H302Q128 552 119 546Q113 543 108 522T98 479L95 458V455H55V458Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"38",d:"M70 417T70 494T124 618T248 666Q319 666 374 624T429 515Q429 485 418 459T392 417T361 389T335 371T324 363L338 354Q352 344 366 334T382 323Q457 264 457 174Q457 95 399 37T249 -22Q159 -22 101 29T43 155Q43 263 172 335L154 348Q133 361 127 368Q70 417 70 494ZM286 386L292 390Q298 394 301 396T311 403T323 413T334 425T345 438T355 454T364 471T369 491T371 513Q371 556 342 586T275 624Q268 625 242 625Q201 625 165 599T128 534Q128 511 141 492T167 463T217 431Q224 426 228 424L286 386ZM250 21Q308 21 350 55T392 137Q392 154 387 169T375 194T353 216T330 234T301 253T274 270Q260 279 244 289T218 306L210 311Q204 311 181 294T133 239T107 157Q107 98 150 60T250 21Z",transform:"translate(500,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"36",d:"M42 313Q42 476 123 571T303 666Q372 666 402 630T432 550Q432 525 418 510T379 495Q356 495 341 509T326 548Q326 592 373 601Q351 623 311 626Q240 626 194 566Q147 500 147 364L148 360Q153 366 156 373Q197 433 263 433H267Q313 433 348 414Q372 400 396 374T435 317Q456 268 456 210V192Q456 169 451 149Q440 90 387 34T253 -22Q225 -22 199 -14T143 16T92 75T56 172T42 313ZM257 397Q227 397 205 380T171 335T154 278T148 216Q148 133 160 97T198 39Q222 21 251 21Q302 21 329 59Q342 77 347 104T352 209Q352 289 347 316T329 361Q302 397 257 397Z",transform:"translate(1000,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mstyle",transform:"translate(27837.8,0)"},[e("g",{"data-mml-node":"mspace"})]),e("g",{"data-mml-node":"mtext",transform:"translate(28004.8,0)"},[e("path",{"data-c":"62",d:"M307 -11Q234 -11 168 55L158 37Q156 34 153 28T147 17T143 10L138 1L118 0H98V298Q98 599 97 603Q94 622 83 628T38 637H20V660Q20 683 22 683L32 684Q42 685 61 686T98 688Q115 689 135 690T165 693T176 694H179V543Q179 391 180 391L183 394Q186 397 192 401T207 411T228 421T254 431T286 439T323 442Q401 442 461 379T522 216Q522 115 458 52T307 -11ZM182 98Q182 97 187 90T196 79T206 67T218 55T233 44T250 35T271 29T295 26Q330 26 363 46T412 113Q424 148 424 212Q424 287 412 323Q385 405 300 405Q270 405 239 390T188 347L182 339V98Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"79",d:"M69 -66Q91 -66 104 -80T118 -116Q118 -134 109 -145T91 -160Q84 -163 97 -166Q104 -168 111 -168Q131 -168 148 -159T175 -138T197 -106T213 -75T225 -43L242 0L170 183Q150 233 125 297Q101 358 96 368T80 381Q79 382 78 382Q66 385 34 385H19V431H26L46 430Q65 430 88 429T122 428Q129 428 142 428T171 429T200 430T224 430L233 431H241V385H232Q183 385 185 366L286 112Q286 113 332 227L376 341V350Q376 365 366 373T348 383T334 385H331V431H337H344Q351 431 361 431T382 430T405 429T422 429Q477 429 503 431H508V385H497Q441 380 422 345Q420 343 378 235T289 9T227 -131Q180 -204 113 -204Q69 -204 44 -177T19 -116Q19 -89 35 -78T69 -66Z",transform:"translate(556,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"74",d:"M27 422Q80 426 109 478T141 600V615H181V431H316V385H181V241Q182 116 182 100T189 68Q203 29 238 29Q282 29 292 100Q293 108 293 146V181H333V146V134Q333 57 291 17Q264 -10 221 -10Q187 -10 162 2T124 33T105 68T98 100Q97 107 97 248V385H18V422H27Z",transform:"translate(1084,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"65",d:"M28 218Q28 273 48 318T98 391T163 433T229 448Q282 448 320 430T378 380T406 316T415 245Q415 238 408 231H126V216Q126 68 226 36Q246 30 270 30Q312 30 342 62Q359 79 369 104L379 128Q382 131 395 131H398Q415 131 415 121Q415 117 412 108Q393 53 349 21T250 -11Q155 -11 92 58T28 218ZM333 275Q322 403 238 411H236Q228 411 220 410T195 402T166 381T143 340T127 274V267H333V275Z",transform:"translate(1473,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"73",d:"M295 316Q295 356 268 385T190 414Q154 414 128 401Q98 382 98 349Q97 344 98 336T114 312T157 287Q175 282 201 278T245 269T277 256Q294 248 310 236T342 195T359 133Q359 71 321 31T198 -10H190Q138 -10 94 26L86 19L77 10Q71 4 65 -1L54 -11H46H42Q39 -11 33 -5V74V132Q33 153 35 157T45 162H54Q66 162 70 158T75 146T82 119T101 77Q136 26 198 26Q295 26 295 104Q295 133 277 151Q257 175 194 187T111 210Q75 227 54 256T33 318Q33 357 50 384T93 424T143 442T187 447H198Q238 447 268 432L283 424L292 431Q302 440 314 448H322H326Q329 448 335 442V310L329 304H301Q295 310 295 316Z",transform:"translate(1917,0)",style:{"stroke-width":"3"}})])])],-1),D=[q],A=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),R=i('

    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.

    See the code in celestia-app and celestia-node.

    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:

    Production RPC endpoints

    These RPC providers are meant to be used in production environments.

    ProviderURL
    NewMetrichttps://app.newmetric.xyz/start
    NumiaFor RPC access: https://docs.numia.xyz/overview/rpc-api-access
    NumiaFor data warehouse access: https://docs.numia.xyz/overview/sql-access/chains/celestia
    Grovehttps://www.grove.city/

    WARNING

    Do not rely on the free community endpoints listed below for production deployments. Production deployments should rely on service providers with SLAs or your own node.

    Consensus nodes

    Community consensus RPC endpoints

    WARNING

    Do not rely on the free community endpoints listed below for production deployments. Production deployments should rely on service providers with SLAs.

    • 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
    • rpc-celestia.staker.space
    • celestia-rpc.noders.services
    • celestia.moonli.me
    • celestia-mainnet-rpc.itrocket.net:443
    • rpc.celestia.mainnet.dteam.tech:443

    Community 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
    • celestia-api.chainode.tech
    • api-celestia.staker.space
    • celestia-api.noders.services
    • celestia.moonli.me/api
    • celestia-mainnet-api.itrocket.net:443
    • api.celestia.mainnet.dteam.tech:443

    Community 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
    • grpc-celestia.staker.space
    • celestia-grpc.noders.services:11090
    • celestia-mainnet-grpc.itrocket.net:443
    • grpc.celestia.mainnet.dteam.tech:28090

    Community WebSocket endpoints

    • wss://celestia-ws.chainode.tech:33373/websocket
    • wss://celestia-mainnet-ws.itrocket.net:443/websocket
    • wss://rpc.celestia.mainnet.dteam.tech:443/websocket

    Data availability nodes

    Community Data availability (DA) RPC endpoints for bridge node sync

    These RPC endpoints allow bridge nodes to sync blocks from the Celestia network. For users, they will need to provide a –core.ip string from a consensus node’s URL or IP that populates a default RPC port at 26657 to their respective DA node.

    Community Data availability (DA) gRPC endpoints for state access

    These gRPC endpoints for DA nodes 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 a default gRPC port at 9090 to their respective DA node.

    TIP

    bash
    celestia <da_type> start --core.ip <url> -–core.grpc.port <port>
    celestia <da_type> start --core.ip <url> -–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 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
    • celestia-mainnet-consensus.itrocket.net
      • gRPC: port 9090
      • RPC: port 26657
    • rpc.celestia.mainnet.dteam.tech
      • gRPC: port 28090
      • RPC: 28657
    • celestia-consensus-mainnet.noders.services
      • gRPC: port 9080
      • RPC: port 26557

    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

    There are a few ways to stay informed about network upgrades on Mainnet Beta:

    See the network upgrade process page to learn more about specific upgrades like the Lemongrass network upgrade.

    ',50),j=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":1726496322000}'),B={name:"nodes/mainnet.md"},G=Object.assign(B,{setup(d){return(T,Q)=>{const l=n("mjx-assistive-mml"),t=n("mjx-container");return s(),r("div",null,[C,a(M),P,a(p),Z,e("p",null,[a(t,{class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},{default:c(()=>[(s(),r("svg",E,D)),a(l,{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"}},{default:c(()=>[A]),_:1})]),_:1})]),R])}}});export{j as __pageData,G as default}; diff --git a/assets/nodes_mainnet.md.37db891f.lean.js b/assets/nodes_mainnet.md.37db891f.lean.js new file mode 100644 index 00000000000..c6c195f5d80 --- /dev/null +++ b/assets/nodes_mainnet.md.37db891f.lean.js @@ -0,0 +1 @@ +import{M as p}from"./chunks/MainnetVersionTags.4fd38854.js";import{c as h}from"./chunks/constants.fa173a21.js";import{_ as m,o as s,c as r,k as e,t as o,C as n,H as a,w as c,Q as i,a as u}from"./chunks/framework.1a91c06a.js";import"./chunks/mainnet_versions.955428b6.js";const f="/img/Mainnet-Beta.png",b="/grove/grove-sandbox.png",y={name:"MainnetBetaDetails",data(){return{constants:h}}},g=e("tr",null,[e("th",null,"Detail"),e("th",null,"Value")],-1),H=e("td",null,"Chain ID",-1),k=e("tr",null,[e("td",null,"Genesis hash"),e("td",null,[e("code",null,"6BE39EFD10BA412A9DB5288488303F5DD32CF386707A5BEF33617F4C43301872")])],-1),V=e("td",null,"Genesis file",-1),w=["href"],L=e("td",null,"Peers file",-1),v=["href"],_=e("tr",null,[e("td",null,"Validators"),e("td",null," 100 ")],-1);function x(d,T,Q,l,t,F){return s(),r("table",null,[g,e("tr",null,[H,e("td",null,[e("code",null,o(t.constants.mainnetChainId),1)])]),k,e("tr",null,[V,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/"+o(t.constants.mainnetChainId)+"/genesis.json ",9,w)])]),e("tr",null,[L,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/"+o(t.constants.mainnetChainId)+"/peers.txt ",9,v)])]),_])}const M=m(y,[["render",x]]),C=i("",8),P=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),Z=i("",8),E={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"},q=e("g",{stroke:"currentColor",fill:"currentColor","stroke-width":"0",transform:"scale(1,-1)"},[e("g",{"data-mml-node":"math"},[e("g",{"data-mml-node":"mtext"},[e("path",{"data-c":"54",d:"M36 443Q37 448 46 558T55 671V677H666V671Q667 666 676 556T685 443V437H645V443Q645 445 642 478T631 544T610 593Q593 614 555 625Q534 630 478 630H451H443Q417 630 414 618Q413 616 413 339V63Q420 53 439 50T528 46H558V0H545L361 3Q186 1 177 0H164V46H194Q264 46 283 49T309 63V339V550Q309 620 304 625T271 630H244H224Q154 630 119 601Q101 585 93 554T81 486T76 443V437H36V443Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"6F",d:"M28 214Q28 309 93 378T250 448Q340 448 405 380T471 215Q471 120 407 55T250 -10Q153 -10 91 57T28 214ZM250 30Q372 30 372 193V225V250Q372 272 371 288T364 326T348 362T317 390T268 410Q263 411 252 411Q222 411 195 399Q152 377 139 338T126 246V226Q126 130 145 91Q177 30 250 30Z",transform:"translate(722,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"74",d:"M27 422Q80 426 109 478T141 600V615H181V431H316V385H181V241Q182 116 182 100T189 68Q203 29 238 29Q282 29 292 100Q293 108 293 146V181H333V146V134Q333 57 291 17Q264 -10 221 -10Q187 -10 162 2T124 33T105 68T98 100Q97 107 97 248V385H18V422H27Z",transform:"translate(1222,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"61",d:"M137 305T115 305T78 320T63 359Q63 394 97 421T218 448Q291 448 336 416T396 340Q401 326 401 309T402 194V124Q402 76 407 58T428 40Q443 40 448 56T453 109V145H493V106Q492 66 490 59Q481 29 455 12T400 -6T353 12T329 54V58L327 55Q325 52 322 49T314 40T302 29T287 17T269 6T247 -2T221 -8T190 -11Q130 -11 82 20T34 107Q34 128 41 147T68 188T116 225T194 253T304 268H318V290Q318 324 312 340Q290 411 215 411Q197 411 181 410T156 406T148 403Q170 388 170 359Q170 334 154 320ZM126 106Q126 75 150 51T209 26Q247 26 276 49T315 109Q317 116 318 175Q318 233 317 233Q309 233 296 232T251 223T193 203T147 166T126 106Z",transform:"translate(1611,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"6C",d:"M42 46H56Q95 46 103 60V68Q103 77 103 91T103 124T104 167T104 217T104 272T104 329Q104 366 104 407T104 482T104 542T103 586T103 603Q100 622 89 628T44 637H26V660Q26 683 28 683L38 684Q48 685 67 686T104 688Q121 689 141 690T171 693T182 694H185V379Q185 62 186 60Q190 52 198 49Q219 46 247 46H263V0H255L232 1Q209 2 183 2T145 3T107 3T57 1L34 0H26V46H42Z",transform:"translate(2111,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"20",d:"",transform:"translate(2389,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"42",d:"M131 622Q124 629 120 631T104 634T61 637H28V683H229H267H346Q423 683 459 678T531 651Q574 627 599 590T624 512Q624 461 583 419T476 360L466 357Q539 348 595 302T651 187Q651 119 600 67T469 3Q456 1 242 0H28V46H61Q103 47 112 49T131 61V622ZM511 513Q511 560 485 594T416 636Q415 636 403 636T371 636T333 637Q266 637 251 636T232 628Q229 624 229 499V374H312L396 375L406 377Q410 378 417 380T442 393T474 417T499 456T511 513ZM537 188Q537 239 509 282T430 336L329 337H229V200V116Q229 57 234 52Q240 47 334 47H383Q425 47 443 53Q486 67 511 104T537 188Z",transform:"translate(2639,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"79",d:"M69 -66Q91 -66 104 -80T118 -116Q118 -134 109 -145T91 -160Q84 -163 97 -166Q104 -168 111 -168Q131 -168 148 -159T175 -138T197 -106T213 -75T225 -43L242 0L170 183Q150 233 125 297Q101 358 96 368T80 381Q79 382 78 382Q66 385 34 385H19V431H26L46 430Q65 430 88 429T122 428Q129 428 142 428T171 429T200 430T224 430L233 431H241V385H232Q183 385 185 366L286 112Q286 113 332 227L376 341V350Q376 365 366 373T348 383T334 385H331V431H337H344Q351 431 361 431T382 430T405 429T422 429Q477 429 503 431H508V385H497Q441 380 422 345Q420 343 378 235T289 9T227 -131Q180 -204 113 -204Q69 -204 44 -177T19 -116Q19 -89 35 -78T69 -66Z",transform:"translate(3347,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"74",d:"M27 422Q80 426 109 478T141 600V615H181V431H316V385H181V241Q182 116 182 100T189 68Q203 29 238 29Q282 29 292 100Q293 108 293 146V181H333V146V134Q333 57 291 17Q264 -10 221 -10Q187 -10 162 2T124 33T105 68T98 100Q97 107 97 248V385H18V422H27Z",transform:"translate(3875,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"65",d:"M28 218Q28 273 48 318T98 391T163 433T229 448Q282 448 320 430T378 380T406 316T415 245Q415 238 408 231H126V216Q126 68 226 36Q246 30 270 30Q312 30 342 62Q359 79 369 104L379 128Q382 131 395 131H398Q415 131 415 121Q415 117 412 108Q393 53 349 21T250 -11Q155 -11 92 58T28 218ZM333 275Q322 403 238 411H236Q228 411 220 410T195 402T166 381T143 340T127 274V267H333V275Z",transform:"translate(4264,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"73",d:"M295 316Q295 356 268 385T190 414Q154 414 128 401Q98 382 98 349Q97 344 98 336T114 312T157 287Q175 282 201 278T245 269T277 256Q294 248 310 236T342 195T359 133Q359 71 321 31T198 -10H190Q138 -10 94 26L86 19L77 10Q71 4 65 -1L54 -11H46H42Q39 -11 33 -5V74V132Q33 153 35 157T45 162H54Q66 162 70 158T75 146T82 119T101 77Q136 26 198 26Q295 26 295 104Q295 133 277 151Q257 175 194 187T111 210Q75 227 54 256T33 318Q33 357 50 384T93 424T143 442T187 447H198Q238 447 268 432L283 424L292 431Q302 440 314 448H322H326Q329 448 335 442V310L329 304H301Q295 310 295 316Z",transform:"translate(4708,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(5379.8,0)"},[e("path",{"data-c":"3D",d:"M56 347Q56 360 70 367H707Q722 359 722 347Q722 336 708 328L390 327H72Q56 332 56 347ZM56 153Q56 168 72 173H708Q722 163 722 153Q722 140 707 133H70Q56 140 56 153Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(6435.6,0)"},[e("path",{"data-c":"28",d:"M94 250Q94 319 104 381T127 488T164 576T202 643T244 695T277 729T302 750H315H319Q333 750 333 741Q333 738 316 720T275 667T226 581T184 443T167 250T184 58T225 -81T274 -167T316 -220T333 -241Q333 -250 318 -250H315H302L274 -226Q180 -141 137 -14T94 250Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mn",transform:"translate(6824.6,0)"},[e("path",{"data-c":"31",d:"M213 578L200 573Q186 568 160 563T102 556H83V602H102Q149 604 189 617T245 641T273 663Q275 666 285 666Q294 666 302 660V361L303 61Q310 54 315 52T339 48T401 46H427V0H416Q395 3 257 3Q121 3 100 0H88V46H114Q136 46 152 46T177 47T193 50T201 52T207 57T213 61V578Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(7546.8,0)"},[e("path",{"data-c":"D7",d:"M630 29Q630 9 609 9Q604 9 587 25T493 118L389 222L284 117Q178 13 175 11Q171 9 168 9Q160 9 154 15T147 29Q147 36 161 51T255 146L359 250L255 354Q174 435 161 449T147 471Q147 480 153 485T168 490Q173 490 175 489Q178 487 284 383L389 278L493 382Q570 459 587 475T609 491Q630 491 630 471Q630 464 620 453T522 355L418 250L522 145Q606 61 618 48T630 29Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mn",transform:"translate(8547,0)"},[e("path",{"data-c":"34",d:"M462 0Q444 3 333 3Q217 3 199 0H190V46H221Q241 46 248 46T265 48T279 53T286 61Q287 63 287 115V165H28V211L179 442Q332 674 334 675Q336 677 355 677H373L379 671V211H471V165H379V114Q379 73 379 66T385 54Q393 47 442 46H471V0H462ZM293 211V545L74 212L183 211H293Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"37",d:"M55 458Q56 460 72 567L88 674Q88 676 108 676H128V672Q128 662 143 655T195 646T364 644H485V605L417 512Q408 500 387 472T360 435T339 403T319 367T305 330T292 284T284 230T278 162T275 80Q275 66 275 52T274 28V19Q270 2 255 -10T221 -22Q210 -22 200 -19T179 0T168 40Q168 198 265 368Q285 400 349 489L395 552H302Q128 552 119 546Q113 543 108 522T98 479L95 458V455H55V458Z",transform:"translate(500,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"38",d:"M70 417T70 494T124 618T248 666Q319 666 374 624T429 515Q429 485 418 459T392 417T361 389T335 371T324 363L338 354Q352 344 366 334T382 323Q457 264 457 174Q457 95 399 37T249 -22Q159 -22 101 29T43 155Q43 263 172 335L154 348Q133 361 127 368Q70 417 70 494ZM286 386L292 390Q298 394 301 396T311 403T323 413T334 425T345 438T355 454T364 471T369 491T371 513Q371 556 342 586T275 624Q268 625 242 625Q201 625 165 599T128 534Q128 511 141 492T167 463T217 431Q224 426 228 424L286 386ZM250 21Q308 21 350 55T392 137Q392 154 387 169T375 194T353 216T330 234T301 253T274 270Q260 279 244 289T218 306L210 311Q204 311 181 294T133 239T107 157Q107 98 150 60T250 21Z",transform:"translate(1000,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mstyle",transform:"translate(10047,0)"},[e("g",{"data-mml-node":"mspace"})]),e("g",{"data-mml-node":"mtext",transform:"translate(10214,0)"},[e("path",{"data-c":"62",d:"M307 -11Q234 -11 168 55L158 37Q156 34 153 28T147 17T143 10L138 1L118 0H98V298Q98 599 97 603Q94 622 83 628T38 637H20V660Q20 683 22 683L32 684Q42 685 61 686T98 688Q115 689 135 690T165 693T176 694H179V543Q179 391 180 391L183 394Q186 397 192 401T207 411T228 421T254 431T286 439T323 442Q401 442 461 379T522 216Q522 115 458 52T307 -11ZM182 98Q182 97 187 90T196 79T206 67T218 55T233 44T250 35T271 29T295 26Q330 26 363 46T412 113Q424 148 424 212Q424 287 412 323Q385 405 300 405Q270 405 239 390T188 347L182 339V98Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"79",d:"M69 -66Q91 -66 104 -80T118 -116Q118 -134 109 -145T91 -160Q84 -163 97 -166Q104 -168 111 -168Q131 -168 148 -159T175 -138T197 -106T213 -75T225 -43L242 0L170 183Q150 233 125 297Q101 358 96 368T80 381Q79 382 78 382Q66 385 34 385H19V431H26L46 430Q65 430 88 429T122 428Q129 428 142 428T171 429T200 430T224 430L233 431H241V385H232Q183 385 185 366L286 112Q286 113 332 227L376 341V350Q376 365 366 373T348 383T334 385H331V431H337H344Q351 431 361 431T382 430T405 429T422 429Q477 429 503 431H508V385H497Q441 380 422 345Q420 343 378 235T289 9T227 -131Q180 -204 113 -204Q69 -204 44 -177T19 -116Q19 -89 35 -78T69 -66Z",transform:"translate(556,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"74",d:"M27 422Q80 426 109 478T141 600V615H181V431H316V385H181V241Q182 116 182 100T189 68Q203 29 238 29Q282 29 292 100Q293 108 293 146V181H333V146V134Q333 57 291 17Q264 -10 221 -10Q187 -10 162 2T124 33T105 68T98 100Q97 107 97 248V385H18V422H27Z",transform:"translate(1084,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"65",d:"M28 218Q28 273 48 318T98 391T163 433T229 448Q282 448 320 430T378 380T406 316T415 245Q415 238 408 231H126V216Q126 68 226 36Q246 30 270 30Q312 30 342 62Q359 79 369 104L379 128Q382 131 395 131H398Q415 131 415 121Q415 117 412 108Q393 53 349 21T250 -11Q155 -11 92 58T28 218ZM333 275Q322 403 238 411H236Q228 411 220 410T195 402T166 381T143 340T127 274V267H333V275Z",transform:"translate(1473,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"73",d:"M295 316Q295 356 268 385T190 414Q154 414 128 401Q98 382 98 349Q97 344 98 336T114 312T157 287Q175 282 201 278T245 269T277 256Q294 248 310 236T342 195T359 133Q359 71 321 31T198 -10H190Q138 -10 94 26L86 19L77 10Q71 4 65 -1L54 -11H46H42Q39 -11 33 -5V74V132Q33 153 35 157T45 162H54Q66 162 70 158T75 146T82 119T101 77Q136 26 198 26Q295 26 295 104Q295 133 277 151Q257 175 194 187T111 210Q75 227 54 256T33 318Q33 357 50 384T93 424T143 442T187 447H198Q238 447 268 432L283 424L292 431Q302 440 314 448H322H326Q329 448 335 442V310L329 304H301Q295 310 295 316Z",transform:"translate(1917,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(12525,0)"},[e("path",{"data-c":"29",d:"M60 749L64 750Q69 750 74 750H86L114 726Q208 641 251 514T294 250Q294 182 284 119T261 12T224 -76T186 -143T145 -194T113 -227T90 -246Q87 -249 86 -250H74Q66 -250 63 -250T58 -247T55 -238Q56 -237 66 -225Q221 -64 221 250T66 725Q56 737 55 738Q55 746 60 749Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(13136.2,0)"},[e("path",{"data-c":"2B",d:"M56 237T56 250T70 270H369V420L370 570Q380 583 389 583Q402 583 409 568V270H707Q722 262 722 250T707 230H409V-68Q401 -82 391 -82H389H387Q375 -82 369 -68V230H70Q56 237 56 250Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(14136.4,0)"},[e("path",{"data-c":"28",d:"M94 250Q94 319 104 381T127 488T164 576T202 643T244 695T277 729T302 750H315H319Q333 750 333 741Q333 738 316 720T275 667T226 581T184 443T167 250T184 58T225 -81T274 -167T316 -220T333 -241Q333 -250 318 -250H315H302L274 -226Q180 -141 137 -14T94 250Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mn",transform:"translate(14525.4,0)"},[e("path",{"data-c":"34",d:"M462 0Q444 3 333 3Q217 3 199 0H190V46H221Q241 46 248 46T265 48T279 53T286 61Q287 63 287 115V165H28V211L179 442Q332 674 334 675Q336 677 355 677H373L379 671V211H471V165H379V114Q379 73 379 66T385 54Q393 47 442 46H471V0H462ZM293 211V545L74 212L183 211H293Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"30",d:"M96 585Q152 666 249 666Q297 666 345 640T423 548Q460 465 460 320Q460 165 417 83Q397 41 362 16T301 -15T250 -22Q224 -22 198 -16T137 16T82 83Q39 165 39 320Q39 494 96 585ZM321 597Q291 629 250 629Q208 629 178 597Q153 571 145 525T137 333Q137 175 145 125T181 46Q209 16 250 16Q290 16 318 46Q347 76 354 130T362 333Q362 478 354 524T321 597Z",transform:"translate(500,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"39",d:"M352 287Q304 211 232 211Q154 211 104 270T44 396Q42 412 42 436V444Q42 537 111 606Q171 666 243 666Q245 666 249 666T257 665H261Q273 665 286 663T323 651T370 619T413 560Q456 472 456 334Q456 194 396 97Q361 41 312 10T208 -22Q147 -22 108 7T68 93T121 149Q143 149 158 135T173 96Q173 78 164 65T148 49T135 44L131 43Q131 41 138 37T164 27T206 22H212Q272 22 313 86Q352 142 352 280V287ZM244 248Q292 248 321 297T351 430Q351 508 343 542Q341 552 337 562T323 588T293 615T246 625Q208 625 181 598Q160 576 154 546T147 441Q147 358 152 329T172 282Q197 248 244 248Z",transform:"translate(1000,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"34",d:"M462 0Q444 3 333 3Q217 3 199 0H190V46H221Q241 46 248 46T265 48T279 53T286 61Q287 63 287 115V165H28V211L179 442Q332 674 334 675Q336 677 355 677H373L379 671V211H471V165H379V114Q379 73 379 66T385 54Q393 47 442 46H471V0H462ZM293 211V545L74 212L183 211H293Z",transform:"translate(1500,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(16747.7,0)"},[e("path",{"data-c":"D7",d:"M630 29Q630 9 609 9Q604 9 587 25T493 118L389 222L284 117Q178 13 175 11Q171 9 168 9Q160 9 154 15T147 29Q147 36 161 51T255 146L359 250L255 354Q174 435 161 449T147 471Q147 480 153 485T168 490Q173 490 175 489Q178 487 284 383L389 278L493 382Q570 459 587 475T609 491Q630 491 630 471Q630 464 620 453T522 355L418 250L522 145Q606 61 618 48T630 29Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mn",transform:"translate(17747.9,0)"},[e("path",{"data-c":"34",d:"M462 0Q444 3 333 3Q217 3 199 0H190V46H221Q241 46 248 46T265 48T279 53T286 61Q287 63 287 115V165H28V211L179 442Q332 674 334 675Q336 677 355 677H373L379 671V211H471V165H379V114Q379 73 379 66T385 54Q393 47 442 46H471V0H462ZM293 211V545L74 212L183 211H293Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"38",d:"M70 417T70 494T124 618T248 666Q319 666 374 624T429 515Q429 485 418 459T392 417T361 389T335 371T324 363L338 354Q352 344 366 334T382 323Q457 264 457 174Q457 95 399 37T249 -22Q159 -22 101 29T43 155Q43 263 172 335L154 348Q133 361 127 368Q70 417 70 494ZM286 386L292 390Q298 394 301 396T311 403T323 413T334 425T345 438T355 454T364 471T369 491T371 513Q371 556 342 586T275 624Q268 625 242 625Q201 625 165 599T128 534Q128 511 141 492T167 463T217 431Q224 426 228 424L286 386ZM250 21Q308 21 350 55T392 137Q392 154 387 169T375 194T353 216T330 234T301 253T274 270Q260 279 244 289T218 306L210 311Q204 311 181 294T133 239T107 157Q107 98 150 60T250 21Z",transform:"translate(500,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"32",d:"M109 429Q82 429 66 447T50 491Q50 562 103 614T235 666Q326 666 387 610T449 465Q449 422 429 383T381 315T301 241Q265 210 201 149L142 93L218 92Q375 92 385 97Q392 99 409 186V189H449V186Q448 183 436 95T421 3V0H50V19V31Q50 38 56 46T86 81Q115 113 136 137Q145 147 170 174T204 211T233 244T261 278T284 308T305 340T320 369T333 401T340 431T343 464Q343 527 309 573T212 619Q179 619 154 602T119 569T109 550Q109 549 114 549Q132 549 151 535T170 489Q170 464 154 447T109 429Z",transform:"translate(1000,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mstyle",transform:"translate(19247.9,0)"},[e("g",{"data-mml-node":"mspace"})]),e("g",{"data-mml-node":"mtext",transform:"translate(19414.9,0)"},[e("path",{"data-c":"62",d:"M307 -11Q234 -11 168 55L158 37Q156 34 153 28T147 17T143 10L138 1L118 0H98V298Q98 599 97 603Q94 622 83 628T38 637H20V660Q20 683 22 683L32 684Q42 685 61 686T98 688Q115 689 135 690T165 693T176 694H179V543Q179 391 180 391L183 394Q186 397 192 401T207 411T228 421T254 431T286 439T323 442Q401 442 461 379T522 216Q522 115 458 52T307 -11ZM182 98Q182 97 187 90T196 79T206 67T218 55T233 44T250 35T271 29T295 26Q330 26 363 46T412 113Q424 148 424 212Q424 287 412 323Q385 405 300 405Q270 405 239 390T188 347L182 339V98Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"79",d:"M69 -66Q91 -66 104 -80T118 -116Q118 -134 109 -145T91 -160Q84 -163 97 -166Q104 -168 111 -168Q131 -168 148 -159T175 -138T197 -106T213 -75T225 -43L242 0L170 183Q150 233 125 297Q101 358 96 368T80 381Q79 382 78 382Q66 385 34 385H19V431H26L46 430Q65 430 88 429T122 428Q129 428 142 428T171 429T200 430T224 430L233 431H241V385H232Q183 385 185 366L286 112Q286 113 332 227L376 341V350Q376 365 366 373T348 383T334 385H331V431H337H344Q351 431 361 431T382 430T405 429T422 429Q477 429 503 431H508V385H497Q441 380 422 345Q420 343 378 235T289 9T227 -131Q180 -204 113 -204Q69 -204 44 -177T19 -116Q19 -89 35 -78T69 -66Z",transform:"translate(556,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"74",d:"M27 422Q80 426 109 478T141 600V615H181V431H316V385H181V241Q182 116 182 100T189 68Q203 29 238 29Q282 29 292 100Q293 108 293 146V181H333V146V134Q333 57 291 17Q264 -10 221 -10Q187 -10 162 2T124 33T105 68T98 100Q97 107 97 248V385H18V422H27Z",transform:"translate(1084,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"65",d:"M28 218Q28 273 48 318T98 391T163 433T229 448Q282 448 320 430T378 380T406 316T415 245Q415 238 408 231H126V216Q126 68 226 36Q246 30 270 30Q312 30 342 62Q359 79 369 104L379 128Q382 131 395 131H398Q415 131 415 121Q415 117 412 108Q393 53 349 21T250 -11Q155 -11 92 58T28 218ZM333 275Q322 403 238 411H236Q228 411 220 410T195 402T166 381T143 340T127 274V267H333V275Z",transform:"translate(1473,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"73",d:"M295 316Q295 356 268 385T190 414Q154 414 128 401Q98 382 98 349Q97 344 98 336T114 312T157 287Q175 282 201 278T245 269T277 256Q294 248 310 236T342 195T359 133Q359 71 321 31T198 -10H190Q138 -10 94 26L86 19L77 10Q71 4 65 -1L54 -11H46H42Q39 -11 33 -5V74V132Q33 153 35 157T45 162H54Q66 162 70 158T75 146T82 119T101 77Q136 26 198 26Q295 26 295 104Q295 133 277 151Q257 175 194 187T111 210Q75 227 54 256T33 318Q33 357 50 384T93 424T143 442T187 447H198Q238 447 268 432L283 424L292 431Q302 440 314 448H322H326Q329 448 335 442V310L329 304H301Q295 310 295 316Z",transform:"translate(1917,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(21725.9,0)"},[e("path",{"data-c":"29",d:"M60 749L64 750Q69 750 74 750H86L114 726Q208 641 251 514T294 250Q294 182 284 119T261 12T224 -76T186 -143T145 -194T113 -227T90 -246Q87 -249 86 -250H74Q66 -250 63 -250T58 -247T55 -238Q56 -237 66 -225Q221 -64 221 250T66 725Q56 737 55 738Q55 746 60 749Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(22392.7,0)"},[e("path",{"data-c":"3D",d:"M56 347Q56 360 70 367H707Q722 359 722 347Q722 336 708 328L390 327H72Q56 332 56 347ZM56 153Q56 168 72 173H708Q722 163 722 153Q722 140 707 133H70Q56 140 56 153Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mn",transform:"translate(23448.4,0)"},[e("path",{"data-c":"31",d:"M213 578L200 573Q186 568 160 563T102 556H83V602H102Q149 604 189 617T245 641T273 663Q275 666 285 666Q294 666 302 660V361L303 61Q310 54 315 52T339 48T401 46H427V0H416Q395 3 257 3Q121 3 100 0H88V46H114Q136 46 152 46T177 47T193 50T201 52T207 57T213 61V578Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(23948.4,0)"},[e("path",{"data-c":"2C",d:"M78 35T78 60T94 103T137 121Q165 121 187 96T210 8Q210 -27 201 -60T180 -117T154 -158T130 -185T117 -194Q113 -194 104 -185T95 -172Q95 -168 106 -156T131 -126T157 -76T173 -3V9L172 8Q170 7 167 6T161 3T152 1T140 0Q113 0 96 17Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mn",transform:"translate(24393.1,0)"},[e("path",{"data-c":"39",d:"M352 287Q304 211 232 211Q154 211 104 270T44 396Q42 412 42 436V444Q42 537 111 606Q171 666 243 666Q245 666 249 666T257 665H261Q273 665 286 663T323 651T370 619T413 560Q456 472 456 334Q456 194 396 97Q361 41 312 10T208 -22Q147 -22 108 7T68 93T121 149Q143 149 158 135T173 96Q173 78 164 65T148 49T135 44L131 43Q131 41 138 37T164 27T206 22H212Q272 22 313 86Q352 142 352 280V287ZM244 248Q292 248 321 297T351 430Q351 508 343 542Q341 552 337 562T323 588T293 615T246 625Q208 625 181 598Q160 576 154 546T147 441Q147 358 152 329T172 282Q197 248 244 248Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"37",d:"M55 458Q56 460 72 567L88 674Q88 676 108 676H128V672Q128 662 143 655T195 646T364 644H485V605L417 512Q408 500 387 472T360 435T339 403T319 367T305 330T292 284T284 230T278 162T275 80Q275 66 275 52T274 28V19Q270 2 255 -10T221 -22Q210 -22 200 -19T179 0T168 40Q168 198 265 368Q285 400 349 489L395 552H302Q128 552 119 546Q113 543 108 522T98 479L95 458V455H55V458Z",transform:"translate(500,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"33",d:"M127 463Q100 463 85 480T69 524Q69 579 117 622T233 665Q268 665 277 664Q351 652 390 611T430 522Q430 470 396 421T302 350L299 348Q299 347 308 345T337 336T375 315Q457 262 457 175Q457 96 395 37T238 -22Q158 -22 100 21T42 130Q42 158 60 175T105 193Q133 193 151 175T169 130Q169 119 166 110T159 94T148 82T136 74T126 70T118 67L114 66Q165 21 238 21Q293 21 321 74Q338 107 338 175V195Q338 290 274 322Q259 328 213 329L171 330L168 332Q166 335 166 348Q166 366 174 366Q202 366 232 371Q266 376 294 413T322 525V533Q322 590 287 612Q265 626 240 626Q208 626 181 615T143 592T132 580H135Q138 579 143 578T153 573T165 566T175 555T183 540T186 520Q186 498 172 481T127 463Z",transform:"translate(1000,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mo",transform:"translate(25893.1,0)"},[e("path",{"data-c":"2C",d:"M78 35T78 60T94 103T137 121Q165 121 187 96T210 8Q210 -27 201 -60T180 -117T154 -158T130 -185T117 -194Q113 -194 104 -185T95 -172Q95 -168 106 -156T131 -126T157 -76T173 -3V9L172 8Q170 7 167 6T161 3T152 1T140 0Q113 0 96 17Z",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mn",transform:"translate(26337.8,0)"},[e("path",{"data-c":"37",d:"M55 458Q56 460 72 567L88 674Q88 676 108 676H128V672Q128 662 143 655T195 646T364 644H485V605L417 512Q408 500 387 472T360 435T339 403T319 367T305 330T292 284T284 230T278 162T275 80Q275 66 275 52T274 28V19Q270 2 255 -10T221 -22Q210 -22 200 -19T179 0T168 40Q168 198 265 368Q285 400 349 489L395 552H302Q128 552 119 546Q113 543 108 522T98 479L95 458V455H55V458Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"38",d:"M70 417T70 494T124 618T248 666Q319 666 374 624T429 515Q429 485 418 459T392 417T361 389T335 371T324 363L338 354Q352 344 366 334T382 323Q457 264 457 174Q457 95 399 37T249 -22Q159 -22 101 29T43 155Q43 263 172 335L154 348Q133 361 127 368Q70 417 70 494ZM286 386L292 390Q298 394 301 396T311 403T323 413T334 425T345 438T355 454T364 471T369 491T371 513Q371 556 342 586T275 624Q268 625 242 625Q201 625 165 599T128 534Q128 511 141 492T167 463T217 431Q224 426 228 424L286 386ZM250 21Q308 21 350 55T392 137Q392 154 387 169T375 194T353 216T330 234T301 253T274 270Q260 279 244 289T218 306L210 311Q204 311 181 294T133 239T107 157Q107 98 150 60T250 21Z",transform:"translate(500,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"36",d:"M42 313Q42 476 123 571T303 666Q372 666 402 630T432 550Q432 525 418 510T379 495Q356 495 341 509T326 548Q326 592 373 601Q351 623 311 626Q240 626 194 566Q147 500 147 364L148 360Q153 366 156 373Q197 433 263 433H267Q313 433 348 414Q372 400 396 374T435 317Q456 268 456 210V192Q456 169 451 149Q440 90 387 34T253 -22Q225 -22 199 -14T143 16T92 75T56 172T42 313ZM257 397Q227 397 205 380T171 335T154 278T148 216Q148 133 160 97T198 39Q222 21 251 21Q302 21 329 59Q342 77 347 104T352 209Q352 289 347 316T329 361Q302 397 257 397Z",transform:"translate(1000,0)",style:{"stroke-width":"3"}})]),e("g",{"data-mml-node":"mstyle",transform:"translate(27837.8,0)"},[e("g",{"data-mml-node":"mspace"})]),e("g",{"data-mml-node":"mtext",transform:"translate(28004.8,0)"},[e("path",{"data-c":"62",d:"M307 -11Q234 -11 168 55L158 37Q156 34 153 28T147 17T143 10L138 1L118 0H98V298Q98 599 97 603Q94 622 83 628T38 637H20V660Q20 683 22 683L32 684Q42 685 61 686T98 688Q115 689 135 690T165 693T176 694H179V543Q179 391 180 391L183 394Q186 397 192 401T207 411T228 421T254 431T286 439T323 442Q401 442 461 379T522 216Q522 115 458 52T307 -11ZM182 98Q182 97 187 90T196 79T206 67T218 55T233 44T250 35T271 29T295 26Q330 26 363 46T412 113Q424 148 424 212Q424 287 412 323Q385 405 300 405Q270 405 239 390T188 347L182 339V98Z",style:{"stroke-width":"3"}}),e("path",{"data-c":"79",d:"M69 -66Q91 -66 104 -80T118 -116Q118 -134 109 -145T91 -160Q84 -163 97 -166Q104 -168 111 -168Q131 -168 148 -159T175 -138T197 -106T213 -75T225 -43L242 0L170 183Q150 233 125 297Q101 358 96 368T80 381Q79 382 78 382Q66 385 34 385H19V431H26L46 430Q65 430 88 429T122 428Q129 428 142 428T171 429T200 430T224 430L233 431H241V385H232Q183 385 185 366L286 112Q286 113 332 227L376 341V350Q376 365 366 373T348 383T334 385H331V431H337H344Q351 431 361 431T382 430T405 429T422 429Q477 429 503 431H508V385H497Q441 380 422 345Q420 343 378 235T289 9T227 -131Q180 -204 113 -204Q69 -204 44 -177T19 -116Q19 -89 35 -78T69 -66Z",transform:"translate(556,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"74",d:"M27 422Q80 426 109 478T141 600V615H181V431H316V385H181V241Q182 116 182 100T189 68Q203 29 238 29Q282 29 292 100Q293 108 293 146V181H333V146V134Q333 57 291 17Q264 -10 221 -10Q187 -10 162 2T124 33T105 68T98 100Q97 107 97 248V385H18V422H27Z",transform:"translate(1084,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"65",d:"M28 218Q28 273 48 318T98 391T163 433T229 448Q282 448 320 430T378 380T406 316T415 245Q415 238 408 231H126V216Q126 68 226 36Q246 30 270 30Q312 30 342 62Q359 79 369 104L379 128Q382 131 395 131H398Q415 131 415 121Q415 117 412 108Q393 53 349 21T250 -11Q155 -11 92 58T28 218ZM333 275Q322 403 238 411H236Q228 411 220 410T195 402T166 381T143 340T127 274V267H333V275Z",transform:"translate(1473,0)",style:{"stroke-width":"3"}}),e("path",{"data-c":"73",d:"M295 316Q295 356 268 385T190 414Q154 414 128 401Q98 382 98 349Q97 344 98 336T114 312T157 287Q175 282 201 278T245 269T277 256Q294 248 310 236T342 195T359 133Q359 71 321 31T198 -10H190Q138 -10 94 26L86 19L77 10Q71 4 65 -1L54 -11H46H42Q39 -11 33 -5V74V132Q33 153 35 157T45 162H54Q66 162 70 158T75 146T82 119T101 77Q136 26 198 26Q295 26 295 104Q295 133 277 151Q257 175 194 187T111 210Q75 227 54 256T33 318Q33 357 50 384T93 424T143 442T187 447H198Q238 447 268 432L283 424L292 431Q302 440 314 448H322H326Q329 448 335 442V310L329 304H301Q295 310 295 316Z",transform:"translate(1917,0)",style:{"stroke-width":"3"}})])])],-1),D=[q],A=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),R=i("",50),j=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":1726496322000}'),B={name:"nodes/mainnet.md"},G=Object.assign(B,{setup(d){return(T,Q)=>{const l=n("mjx-assistive-mml"),t=n("mjx-container");return s(),r("div",null,[C,a(M),P,a(p),Z,e("p",null,[a(t,{class:"MathJax",jax:"SVG",style:{direction:"ltr",position:"relative"}},{default:c(()=>[(s(),r("svg",E,D)),a(l,{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"}},{default:c(()=>[A]),_:1})]),_:1})]),R])}}});export{j as __pageData,G as default}; diff --git a/assets/nodes_mocha-testnet.md.a45f13e7.js b/assets/nodes_mocha-testnet.md.a45f13e7.js new file mode 100644 index 00000000000..bee0be0a821 --- /dev/null +++ b/assets/nodes_mocha-testnet.md.a45f13e7.js @@ -0,0 +1 @@ +import{M as d}from"./chunks/MochaVersionTags.290e3e99.js";import{c as h}from"./chunks/constants.fa173a21.js";import{_ as p,o as s,c as i,k as e,t as o,H as a,Q as r,a as u}from"./chunks/framework.1a91c06a.js";import"./chunks/mocha_versions.7704b055.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),k=e("tr",null,[e("td",null,"Genesis hash"),e("td",null,[e("code",null,"B93BBE20A0FBFDF955811B6420F8433904664D45DB4BF51022BE4200C1A1680D")])],-1),y=e("td",null,"Genesis file",-1),_=["href"],w=e("td",null,"Peers file",-1),v=["href"],C=e("tr",null,[e("td",null,"Validators"),e("td",null," 100 ")],-1);function P(l,n,c,S,t,E){return s(),i("table",null,[g,e("tr",null,[b,e("td",null,[e("code",null,o(t.constants.mochaChainId),1)])]),k,e("tr",null,[y,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,_)])]),e("tr",null,[w,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 T=p(f,[["render",P]]),R=r('

    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),x=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=r('

    RPC for DA bridge, full, and light nodes

    Production RPC endpoints

    These RPC providers are meant to be used in production environments.

    ProviderURL
    NewMetrichttps://app.newmetric.xyz/start
    NumiaFor RPC access: https://docs.numia.xyz/overview/rpc-api-access
    NumiaFor data warehouse access: https://docs.numia.xyz/overview/sql-access/chains/celestia
    Grovehttps://www.grove.city/

    WARNING

    Do not rely on the free community endpoints listed below for production deployments. Production deployments should rely on service providers with SLAs or your own node.

    Community Data availability (DA) RPC endpoints for bridge node sync

    These RPC endpoints allow bridge nodes to sync blocks from the Celestia network. For users, they will need to provide a –core.ip string from a consensus node’s URL or IP that populates a default RPC port at 26657 to their respective DA node.

    Community Data availability (DA) gRPC endpoints for state access

    These gRPC endpoints for DA nodes 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 a default gRPC port at 9090 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 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
    • celestia-testnet-consensus.itrocket.net
      • RPC port: 26657
      • gRPC port: 9090
    • rpc-celestia-testnet.cryptech.com.ua
      • gRPC: grpc-celestia-testnet.cryptech.com.ua:443
    • rpc.celestia.testnet.dteam.tech:443
      • gRPC: grpc.celestia.testnet.dteam.tech:27090
    • celestia-consensus-testnet.noders.services
      • RPC port: 26357
      • gRPC port: 9070

    Community 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
    • 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/
    • rpc-1.testnet.celestia.nodes.guru
    • rpc-2.testnet.celestia.nodes.guru
    • celestia-testnet-rpc.itrocket.net:443
    • rpc-celestia-testnet.cryptech.com.ua:443
    • rpc.celestia.testnet.dteam.tech:443

    Community 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.

    Community 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
    • 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
    • grpc-1.testnet.celestia.nodes.guru:10790
    • grpc-2.testnet.celestia.nodes.guru:10790
    • celestia-testnet-grpc.itrocket.net:443
    • grpc-celestia-testnet.cryptech.com.ua:443
    • grpc.celestia.testnet.dteam.tech:27090

    Community 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.

    Analytics

    The following websites provide analytics for Mocha Testnet:

    Explorers

    There are several explorers you can use for Mocha:

    Network upgrades

    There are a few ways to stay informed about network upgrades on Mocha testnet:

    See the network upgrade process page to learn more about specific upgrades like the Lemongrass network upgrade.

    ',46),F=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":1726494281000}'),D={name:"nodes/mocha-testnet.md"},B=Object.assign(D,{setup(l){return(n,c)=>(s(),i("div",null,[R,a(T),x,a(d),A]))}});export{F as __pageData,B as default}; diff --git a/assets/nodes_mocha-testnet.md.a45f13e7.lean.js b/assets/nodes_mocha-testnet.md.a45f13e7.lean.js new file mode 100644 index 00000000000..07fe3d2b367 --- /dev/null +++ b/assets/nodes_mocha-testnet.md.a45f13e7.lean.js @@ -0,0 +1 @@ +import{M as d}from"./chunks/MochaVersionTags.290e3e99.js";import{c as h}from"./chunks/constants.fa173a21.js";import{_ as p,o as s,c as i,k as e,t as o,H as a,Q as r,a as u}from"./chunks/framework.1a91c06a.js";import"./chunks/mocha_versions.7704b055.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),k=e("tr",null,[e("td",null,"Genesis hash"),e("td",null,[e("code",null,"B93BBE20A0FBFDF955811B6420F8433904664D45DB4BF51022BE4200C1A1680D")])],-1),y=e("td",null,"Genesis file",-1),_=["href"],w=e("td",null,"Peers file",-1),v=["href"],C=e("tr",null,[e("td",null,"Validators"),e("td",null," 100 ")],-1);function P(l,n,c,S,t,E){return s(),i("table",null,[g,e("tr",null,[b,e("td",null,[e("code",null,o(t.constants.mochaChainId),1)])]),k,e("tr",null,[y,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,_)])]),e("tr",null,[w,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 T=p(f,[["render",P]]),R=r("",11),x=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=r("",46),F=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":1726494281000}'),D={name:"nodes/mocha-testnet.md"},B=Object.assign(D,{setup(l){return(n,c)=>(s(),i("div",null,[R,a(T),x,a(d),A]))}});export{F as __pageData,B as default}; diff --git a/assets/nodes_network-upgrade-process.md.a106e947.js b/assets/nodes_network-upgrade-process.md.a106e947.js new file mode 100644 index 00000000000..7ecdc8cc328 --- /dev/null +++ b/assets/nodes_network-upgrade-process.md.a106e947.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as r}from"./chunks/framework.1a91c06a.js";const w=JSON.parse('{"title":"Celestia network upgrade process","description":"Overview of the Celestia network upgrade process.","frontmatter":{"description":"Overview of the Celestia network upgrade process.","head":[["meta",{"name":"og:title","content":"Celestia network upgrade process | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/network-upgrade-process.md","filePath":"nodes/network-upgrade-process.md","lastUpdated":1726751131000}'),o={name:"nodes/network-upgrade-process.md"},n=r('

    Celestia network upgrade 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 network upgrade, which can be breaking or non-breaking. During planned network upgrades, the Celestia Labs team will coordinate with the validators to prepare for the upcoming network upgrade.

    Breaking network upgrades 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.

    Non-breaking network upgrades are backward-compatible and require less coordination.

    General process

    The general process can be broken down into several components:

    • Network upgrade 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 Labs team in order for validators to upgrade their nodes to the new binary.
    • A block number for when the breaking network upgrade. Even if validators upgrade their binary to be network upgrade 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, which happens on testnets first prior to activating on Mainnet Beta in order to ensure the network can upgrade securely.

    The two testnets where network upgrades are deployed are:

    Lemongrass network upgrade

    The Lemongrass network upgrade is the first consensus layer breaking change since Celestia's Mainnet Beta genesis block. The Lemongrass network upgrade includes all of the CIPs listed in CIP-17. The Lemongrass network upgrade will be executed on Arabica, then Mocha, then Mainnet Beta. The network upgrade will take place at an "upgrade height" that will be coordinated offline on a per-network basis. The upgrade heights will be announced in advance (see network upgrades channels) to give node operators time to download and start a compatible binary prior to the upgrade height.

    • If you are a consensus node or validator operator: you will need to download and run a celestia-app binary >= v2.0.0 prior to the --v2-upgrade-height to remain on the canonical chain.
    • If you are a DA node operator, you will need to download and run a compatible celestia-node binary >= v0.16.0-rc0 prior to the upgrade height.
    NetworkChain IDDate and approximate time--v2-upgrade-height
    Arabicaarabica-112024/08/19 @ 14:00 UTC1751707
    Mochamocha-42024/08/28 @ 14:00 UTC2585031
    Mainnet Betacelestia2024/09/18 @ 14:00 UTC2371495

    WARNING

    You do not need to use a tool like cosmovisor to upgrade the binary at the upgrade height. Please upgrade your binary several blocks before the upgrade height.

    ',15),i=[n];function s(d,l,p,c,h,u){return t(),a("div",null,i)}const k=e(o,[["render",s]]);export{w as __pageData,k as default}; diff --git a/assets/nodes_network-upgrade-process.md.a106e947.lean.js b/assets/nodes_network-upgrade-process.md.a106e947.lean.js new file mode 100644 index 00000000000..bcfba07344a --- /dev/null +++ b/assets/nodes_network-upgrade-process.md.a106e947.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as r}from"./chunks/framework.1a91c06a.js";const w=JSON.parse('{"title":"Celestia network upgrade process","description":"Overview of the Celestia network upgrade process.","frontmatter":{"description":"Overview of the Celestia network upgrade process.","head":[["meta",{"name":"og:title","content":"Celestia network upgrade process | Celestia Docs"},{"name":"og:description","content":false}]]},"headers":[],"relativePath":"nodes/network-upgrade-process.md","filePath":"nodes/network-upgrade-process.md","lastUpdated":1726751131000}'),o={name:"nodes/network-upgrade-process.md"},n=r("",15),i=[n];function s(d,l,p,c,h,u){return t(),a("div",null,i)}const k=e(o,[["render",s]]);export{w as __pageData,k as default}; diff --git a/assets/nodes_overview.md.84db2210.js b/assets/nodes_overview.md.84db2210.js new file mode 100644 index 00000000000..c31c8e9ceac --- /dev/null +++ b/assets/nodes_overview.md.84db2210.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as o}from"./chunks/framework.1a91c06a.js";const v=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":1724431383000}'),n={name:"nodes/overview.md"},d=o('

    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.
    • 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.

    Data availability nodes

    Node typeMemoryCPUDiskBandwidth
    Light node500 MB RAMSingle core100 GB SSD56 Kbps
    Bridge node16 GB RAM6 cores10 TB SSD1 Gbps
    Full storage node16 GB RAMQuad-core10 TB SSD1 Gbps

    Consensus nodes

    Node typeMemoryCPUDiskBandwidth
    Validator16 GB RAM8 cores2 TB SSD1 Gbps
    Consensus node16 GB RAMQuad-core2 TB SSD1 Gbps

    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!

    ',14),i=[d];function s(r,l,h,c,u,p){return t(),a("div",null,i)}const m=e(n,[["render",s]]);export{v as __pageData,m as default}; diff --git a/assets/nodes_overview.md.84db2210.lean.js b/assets/nodes_overview.md.84db2210.lean.js new file mode 100644 index 00000000000..e0d77a38531 --- /dev/null +++ b/assets/nodes_overview.md.84db2210.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as o}from"./chunks/framework.1a91c06a.js";const v=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":1724431383000}'),n={name:"nodes/overview.md"},d=o("",14),i=[d];function s(r,l,h,c,u,p){return t(),a("div",null,i)}const m=e(n,[["render",s]]);export{v as __pageData,m as default}; diff --git a/assets/nodes_participate.md.e6ceada6.js b/assets/nodes_participate.md.e6ceada6.js new file mode 100644 index 00000000000..6aed19a6c40 --- /dev/null +++ b/assets/nodes_participate.md.e6ceada6.js @@ -0,0 +1 @@ +import{A as t}from"./chunks/ArabicaVersionTags.f0aaf832.js";import{M as r}from"./chunks/MochaVersionTags.290e3e99.js";import{M as o}from"./chunks/MainnetVersionTags.4fd38854.js";import{o as n,c as i,H as a,Q as e}from"./chunks/framework.1a91c06a.js";import"./chunks/arabica_versions.1930378b.js";import"./chunks/constants.fa173a21.js";import"./chunks/mocha_versions.7704b055.js";import"./chunks/mainnet_versions.955428b6.js";const s=e('

    Participate in the Celestia networks

    Mainnet Beta

    Celestia’s Mainnet Beta is the production network for deploying Mainnet Beta 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),c=e('

    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),l=e('

    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 breaking network upgrades in coordination with the validator community on Mocha.

    Compatible software versions for Mocha testnet

    ',3),d=e('

    Network upgrades

    There are a few ways to stay informed about network upgrades:

    See the network upgrade process page to learn more about specific upgrades like the Lemongrass network upgrade.

    ',4),A=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":1726496322000}'),h={name:"nodes/participate.md"},P=Object.assign(h,{setup(p){return(m,_)=>(n(),i("div",null,[s,a(o),c,a(t),l,a(r),d]))}});export{A as __pageData,P as default}; diff --git a/assets/nodes_participate.md.e6ceada6.lean.js b/assets/nodes_participate.md.e6ceada6.lean.js new file mode 100644 index 00000000000..05dd033bf7e --- /dev/null +++ b/assets/nodes_participate.md.e6ceada6.lean.js @@ -0,0 +1 @@ +import{A as t}from"./chunks/ArabicaVersionTags.f0aaf832.js";import{M as r}from"./chunks/MochaVersionTags.290e3e99.js";import{M as o}from"./chunks/MainnetVersionTags.4fd38854.js";import{o as n,c as i,H as a,Q as e}from"./chunks/framework.1a91c06a.js";import"./chunks/arabica_versions.1930378b.js";import"./chunks/constants.fa173a21.js";import"./chunks/mocha_versions.7704b055.js";import"./chunks/mainnet_versions.955428b6.js";const s=e("",4),c=e("",5),l=e("",3),d=e("",4),A=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":1726496322000}'),h={name:"nodes/participate.md"},P=Object.assign(h,{setup(p){return(m,_)=>(n(),i("div",null,[s,a(o),c,a(t),l,a(r),d]))}});export{A as __pageData,P as default}; diff --git a/assets/nodes_quick-start.md.60c923e3.js b/assets/nodes_quick-start.md.60c923e3.js new file mode 100644 index 00000000000..a4887989202 --- /dev/null +++ b/assets/nodes_quick-start.md.60c923e3.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as o}from"./chunks/framework.1a91c06a.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.60c923e3.lean.js b/assets/nodes_quick-start.md.60c923e3.lean.js new file mode 100644 index 00000000000..083e2a26404 --- /dev/null +++ b/assets/nodes_quick-start.md.60c923e3.lean.js @@ -0,0 +1 @@ +import{_ as e,o as t,c as a,Q as o}from"./chunks/framework.1a91c06a.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.e4f637ee.js b/assets/nodes_systemd.md.e4f637ee.js new file mode 100644 index 00000000000..fdb82563e2a --- /dev/null +++ b/assets/nodes_systemd.md.e4f637ee.js @@ -0,0 +1,125 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.1a91c06a.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":1722891813000}'),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 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.e4f637ee.lean.js b/assets/nodes_systemd.md.e4f637ee.lean.js new file mode 100644 index 00000000000..04df009cab3 --- /dev/null +++ b/assets/nodes_systemd.md.e4f637ee.lean.js @@ -0,0 +1 @@ +import{_ as s,o as a,c as n,Q as l}from"./chunks/framework.1a91c06a.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":1722891813000}'),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.df92fc20.js b/assets/nodes_validator-node.md.df92fc20.js new file mode 100644 index 00000000000..694b0a4f39a --- /dev/null +++ b/assets/nodes_validator-node.md.df92fc20.js @@ -0,0 +1,83 @@ +import{c as e}from"./chunks/constants.fa173a21.js";import{o as l,c as p,k as s,a,t as o,l as t,Q as n}from"./chunks/framework.1a91c06a.js";const c="/img/nodes/validator.png",r=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: 16 GB RAM
    • CPU: 8 cores
    • Disk: 2 TB 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 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),i={class:"language-bash vp-adaptive-theme"},d=s("button",{title:"Copy Code",class:"copy"},null,-1),h=s("span",{class:"lang"},"bash",-1),u={class:"shiki github-dark vp-code-dark"},_=n('celestia-appd tx staking delegate \\',1),y=n('celestiavaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u4q4gx4p 1000000utia \\',1),g={class:"line"},E=s("span",{style:{color:"#E1E4E8"}},"--from=$VALIDATOR_WALLET ",-1),C={style:{color:"#79B8FF"}},m=s("span",{style:{color:"#E1E4E8"}}," ",-1),F=s("span",{style:{color:"#79B8FF"}},"\\",-1),v=s("span",{class:"line"},[s("span",{style:{color:"#E1E4E8"}},"--fees=21000utia")],-1),b={class:"shiki github-light vp-code-light"},T=n('celestia-appd tx staking delegate \\',1),k=n('celestiavaloper1q3v5cugc8cdpud87u4zwy0a74uxkk6u4q4gx4p 1000000utia \\',1),f={class:"line"},A=s("span",{style:{color:"#24292E"}},"--from=$VALIDATOR_WALLET ",-1),S={style:{color:"#005CC5"}},B=s("span",{style:{color:"#24292E"}}," ",-1),x=s("span",{style:{color:"#005CC5"}},"\\",-1),q=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.

    Using an RPC of your own, or one from Mainnet Beta, Mocha testnet or Arabica devnet, initialize your node.

    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

    If you are running celestia-app v1.x.x:

    sh
    celestia-appd start
    celestia-appd start

    If you are running celestia-app >= v2.0.0: then you'll want to start the node with a --v2-upgrade-height that is dependent on the network. The --v2-upgrade-height flag is only needed during the v2 upgrade height so after your node has executed the upgrade (e.g. you see the log upgraded from app version 1 to 2), you don't need to provide this flag for future celestia-appd start invocations.

    sh
    celestia-appd start --v2-upgrade-height 2371495
    celestia-appd start --v2-upgrade-height 2371495
    sh
    celestia-appd start --v2-upgrade-height 2585031
    celestia-appd start --v2-upgrade-height 2585031
    sh
    celestia-appd start --v2-upgrade-height 1751707
    celestia-appd start --v2-upgrade-height 1751707

    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:

    `,30),I={class:"language-bash vp-adaptive-theme"},P=s("button",{title:"Copy Code",class:"copy"},null,-1),V=s("span",{class:"lang"},"bash",-1),R={class:"shiki github-dark vp-code-dark"},D=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),M=n(' --amount=1000000utia \\',1),U=n(' --pubkey=$(celestia-appd tendermint show-validator) \\',1),W=n(' --moniker=$MONIKER \\',1),z={class:"line"},$=s("span",{style:{color:"#E1E4E8"}}," ",-1),H={style:{color:"#79B8FF"}},Y=s("span",{style:{color:"#E1E4E8"}}," ",-1),K=s("span",{style:{color:"#79B8FF"}},"\\",-1),Q=n(' --commission-rate=0.1 \\',1),G=n(' --commission-max-rate=0.2 \\',1),X=n(' --commission-max-change-rate=0.01 \\',1),j=n(' --min-self-delegation=1000000 \\',1),J=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"},es=s("span",{class:"line"},[s("span",{style:{color:"#24292E"}},"MONIKER"),s("span",{style:{color:"#D73A49"}},"="),s("span",{style:{color:"#032F62"}},'"your_moniker"')],-1),os=s("span",{class:"line"},[s("span",{style:{color:"#24292E"}},"VALIDATOR_WALLET"),s("span",{style:{color:"#D73A49"}},"="),s("span",{style:{color:"#032F62"}},'"validator"')],-1),ts=s("span",{class:"line"},null,-1),ls=n('celestia-appd tx staking create-validator \\',1),ps=n(' --amount=1000000utia \\',1),cs=n(' --pubkey=$(celestia-appd tendermint show-validator) \\',1),rs=n(' --moniker=$MONIKER \\',1),is={class:"line"},ds=s("span",{style:{color:"#24292E"}}," ",-1),hs={style:{color:"#005CC5"}},us=s("span",{style:{color:"#24292E"}}," ",-1),_s=s("span",{style:{color:"#005CC5"}},"\\",-1),ys=n(' --commission-rate=0.1 \\',1),gs=n(' --commission-max-rate=0.2 \\',1),Es=n(' --commission-max-change-rate=0.01 \\',1),Cs=n(' --min-self-delegation=1000000 \\',1),ms=n(' --from=$VALIDATOR_WALLET \\',1),Fs=n(' --keyring-backend=test \\',1),vs=n(' --fees=21000utia \\',1),bs=s("span",{class:"line"},[s("span",{style:{color:"#24292E"}}," "),s("span",{style:{color:"#005CC5"}},"--gas=220000")],-1),Ts=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 consensus node page.

    FAQ

    +2/3 committed an invalid block: wrong Block.Header.Version

    If you encounter an error like:

    bash
    2024-04-25 14:48:24 6:48PM ERR CONSENSUS FAILURE!!! err="+2/3 committed an invalid block: wrong Block.Header.Version. Expected {11 1}, got {11 2}" module=consensus stack="goroutine 214 [running]:\\nruntime/debug.Stack()\\n\\t/usr/local/go/src/runtime/debug/stack.go:24 +0x64\\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine.func2()\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:746 +0x44\\npanic({0x1b91180?, 0x400153b240?})\\n\\t/usr/local/go/src/runtime/panic.go:770 +0x124\\ngithub.com/tendermint/tendermint/consensus.(*State).finalizeCommit(0x400065ea88, 0x3)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1637 +0xd30\\ngithub.com/tendermint/tendermint/consensus.(*State).tryFinalizeCommit(0x400065ea88, 0x3)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1606 +0x26c\\ngithub.com/tendermint/tendermint/consensus.(*State).handleCompleteProposal(0x400065ea88, 0x3)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:2001 +0x2d8\\ngithub.com/tendermint/tendermint/consensus.(*State).handleMsg(0x400065ea88, {{0x2b30a00, 0x400143e048}, {0x40002a61b0, 0x28}})\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:856 +0x1c8\\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine(0x400065ea88, 0x0)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:782 +0x2c4\\ncreated by github.com/tendermint/tendermint/consensus.(*State).OnStart in goroutine 169\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:391 +0x110\\n"
    2024-04-25 14:48:24 6:48PM ERR CONSENSUS FAILURE!!! err="+2/3 committed an invalid block: wrong Block.Header.Version. Expected {11 1}, got {11 2}" module=consensus stack="goroutine 214 [running]:\\nruntime/debug.Stack()\\n\\t/usr/local/go/src/runtime/debug/stack.go:24 +0x64\\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine.func2()\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:746 +0x44\\npanic({0x1b91180?, 0x400153b240?})\\n\\t/usr/local/go/src/runtime/panic.go:770 +0x124\\ngithub.com/tendermint/tendermint/consensus.(*State).finalizeCommit(0x400065ea88, 0x3)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1637 +0xd30\\ngithub.com/tendermint/tendermint/consensus.(*State).tryFinalizeCommit(0x400065ea88, 0x3)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1606 +0x26c\\ngithub.com/tendermint/tendermint/consensus.(*State).handleCompleteProposal(0x400065ea88, 0x3)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:2001 +0x2d8\\ngithub.com/tendermint/tendermint/consensus.(*State).handleMsg(0x400065ea88, {{0x2b30a00, 0x400143e048}, {0x40002a61b0, 0x28}})\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:856 +0x1c8\\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine(0x400065ea88, 0x0)\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:782 +0x2c4\\ncreated by github.com/tendermint/tendermint/consensus.(*State).OnStart in goroutine 169\\n\\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:391 +0x110\\n"

    then it is likely that the network has upgraded to a new app version but your consensus node was not prepared for the upgrade. To fix this, you'll need to update your binary to the latest version and restart your node with the relevant --v2-upgrade-height for the network you're running on. If your node still can't sync to the tip of the chain after the above steps, consider a celestia-appd tendermint reset-state to reset your node and start syncing from the genesis block.

    1. [Optional] Back up your validator keys.
    2. [Optional] Back up the data/priv_validator_state.json inside your CELESTIA_HOME directory.
    3. Remove DBs from your CELESTIA_HOME directory via: celestia-appd tendermint reset-state.
    4. Remove the data/application.db inside your CELESTIA_HOME directory.
    5. Download the latest binary for your network.
    6. Restart your consensus node with the relevant --v2-upgrade-height for the network you're running on.
    `,17),qs=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":1726697682000}'),ks={name:"nodes/validator-node.md"},ws=Object.assign(ks,{setup(fs){return(As,Ss)=>(l(),p("div",null,[r,s("div",i,[d,h,s("pre",u,[s("code",null,[_,a(` +`),y,a(` +`),s("span",g,[E,s("span",C,"--chain-id="+o(t(e).mochaChainId),1),m,F]),a(` +`),v])]),s("pre",b,[s("code",null,[T,a(` +`),k,a(` +`),s("span",f,[A,s("span",S,"--chain-id="+o(t(e).mochaChainId),1),B,x]),a(` +`),q])])]),w,s("div",I,[P,V,s("pre",R,[s("code",null,[D,a(` +`),N,a(` +`),L,a(` +`),O,a(` +`),M,a(` +`),U,a(` +`),W,a(` +`),s("span",z,[$,s("span",H,"--chain-id="+o(t(e).mochaChainId),1),Y,K]),a(` +`),Q,a(` +`),G,a(` +`),X,a(` +`),j,a(` +`),J,a(` +`),Z,a(` +`),ss,a(` +`),as])]),s("pre",ns,[s("code",null,[es,a(` +`),os,a(` +`),ts,a(` +`),ls,a(` +`),ps,a(` +`),cs,a(` +`),rs,a(` +`),s("span",is,[ds,s("span",hs,"--chain-id="+o(t(e).mochaChainId),1),us,_s]),a(` +`),ys,a(` +`),gs,a(` +`),Es,a(` +`),Cs,a(` +`),ms,a(` +`),Fs,a(` +`),vs,a(` +`),bs])])]),Ts]))}});export{qs as __pageData,ws as default}; diff --git a/assets/nodes_validator-node.md.df92fc20.lean.js b/assets/nodes_validator-node.md.df92fc20.lean.js new file mode 100644 index 00000000000..ab6d1b9550b --- /dev/null +++ b/assets/nodes_validator-node.md.df92fc20.lean.js @@ -0,0 +1,37 @@ +import{c as e}from"./chunks/constants.fa173a21.js";import{o as l,c as p,k as s,a,t as o,l as t,Q as n}from"./chunks/framework.1a91c06a.js";const c="/img/nodes/validator.png",r=n("",19),i={class:"language-bash vp-adaptive-theme"},d=s("button",{title:"Copy Code",class:"copy"},null,-1),h=s("span",{class:"lang"},"bash",-1),u={class:"shiki github-dark vp-code-dark"},_=n("",1),y=n("",1),g={class:"line"},E=s("span",{style:{color:"#E1E4E8"}},"--from=$VALIDATOR_WALLET ",-1),C={style:{color:"#79B8FF"}},m=s("span",{style:{color:"#E1E4E8"}}," ",-1),F=s("span",{style:{color:"#79B8FF"}},"\\",-1),v=s("span",{class:"line"},[s("span",{style:{color:"#E1E4E8"}},"--fees=21000utia")],-1),b={class:"shiki github-light vp-code-light"},T=n("",1),k=n("",1),f={class:"line"},A=s("span",{style:{color:"#24292E"}},"--from=$VALIDATOR_WALLET ",-1),S={style:{color:"#005CC5"}},B=s("span",{style:{color:"#24292E"}}," ",-1),x=s("span",{style:{color:"#005CC5"}},"\\",-1),q=s("span",{class:"line"},[s("span",{style:{color:"#24292E"}},"--fees=21000utia")],-1),w=n("",30),I={class:"language-bash vp-adaptive-theme"},P=s("button",{title:"Copy Code",class:"copy"},null,-1),V=s("span",{class:"lang"},"bash",-1),R={class:"shiki github-dark vp-code-dark"},D=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),M=n("",1),U=n("",1),W=n("",1),z={class:"line"},$=s("span",{style:{color:"#E1E4E8"}}," ",-1),H={style:{color:"#79B8FF"}},Y=s("span",{style:{color:"#E1E4E8"}}," ",-1),K=s("span",{style:{color:"#79B8FF"}},"\\",-1),Q=n("",1),G=n("",1),X=n("",1),j=n("",1),J=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"},es=s("span",{class:"line"},[s("span",{style:{color:"#24292E"}},"MONIKER"),s("span",{style:{color:"#D73A49"}},"="),s("span",{style:{color:"#032F62"}},'"your_moniker"')],-1),os=s("span",{class:"line"},[s("span",{style:{color:"#24292E"}},"VALIDATOR_WALLET"),s("span",{style:{color:"#D73A49"}},"="),s("span",{style:{color:"#032F62"}},'"validator"')],-1),ts=s("span",{class:"line"},null,-1),ls=n("",1),ps=n("",1),cs=n("",1),rs=n("",1),is={class:"line"},ds=s("span",{style:{color:"#24292E"}}," ",-1),hs={style:{color:"#005CC5"}},us=s("span",{style:{color:"#24292E"}}," ",-1),_s=s("span",{style:{color:"#005CC5"}},"\\",-1),ys=n("",1),gs=n("",1),Es=n("",1),Cs=n("",1),ms=n("",1),Fs=n("",1),vs=n("",1),bs=s("span",{class:"line"},[s("span",{style:{color:"#24292E"}}," "),s("span",{style:{color:"#005CC5"}},"--gas=220000")],-1),Ts=n("",17),qs=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":1726697682000}'),ks={name:"nodes/validator-node.md"},ws=Object.assign(ks,{setup(fs){return(As,Ss)=>(l(),p("div",null,[r,s("div",i,[d,h,s("pre",u,[s("code",null,[_,a(` +`),y,a(` +`),s("span",g,[E,s("span",C,"--chain-id="+o(t(e).mochaChainId),1),m,F]),a(` +`),v])]),s("pre",b,[s("code",null,[T,a(` +`),k,a(` +`),s("span",f,[A,s("span",S,"--chain-id="+o(t(e).mochaChainId),1),B,x]),a(` +`),q])])]),w,s("div",I,[P,V,s("pre",R,[s("code",null,[D,a(` +`),N,a(` +`),L,a(` +`),O,a(` +`),M,a(` +`),U,a(` +`),W,a(` +`),s("span",z,[$,s("span",H,"--chain-id="+o(t(e).mochaChainId),1),Y,K]),a(` +`),Q,a(` +`),G,a(` +`),X,a(` +`),j,a(` +`),J,a(` +`),Z,a(` +`),ss,a(` +`),as])]),s("pre",ns,[s("code",null,[es,a(` +`),os,a(` +`),ts,a(` +`),ls,a(` +`),ps,a(` +`),cs,a(` +`),rs,a(` +`),s("span",is,[ds,s("span",hs,"--chain-id="+o(t(e).mochaChainId),1),us,_s]),a(` +`),ys,a(` +`),gs,a(` +`),Es,a(` +`),Cs,a(` +`),ms,a(` +`),Fs,a(` +`),vs,a(` +`),bs])])]),Ts]))}});export{qs as __pageData,ws as default}; diff --git a/assets/style.b22e62bb.css b/assets/style.b22e62bb.css new file mode 100644 index 00000000000..1013a6c0ba9 --- /dev/null +++ b/assets/style.b22e62bb.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.6.0 | 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-key-pressed-shadow:inset 0 -2px 0 0 #cdcde6,inset 0 0 1px 1px #fff,0 1px 1px 0 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-key-pressed-shadow:inset 0 -2px 0 0 #282d55,inset 0 0 1px 1px #51577d,0 1px 1px 0 rgba(3,4,9,.30196078431372547);--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}.DocSearch-Button-Key--pressed{transform:translate3d(0,1px,0);box-shadow:var(--docsearch-key-pressed-shadow)}@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}.DocSearch-VisuallyHiddenForAccessibility{clip:rect(0 0 0 0);clip-path:inset(50%);height:1px;overflow:hidden;position:absolute;white-space:nowrap;width:1px}@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/audits/SP1_Blobstream_Ottersec_Audit.pdf b/audits/SP1_Blobstream_Ottersec_Audit.pdf new file mode 100644 index 00000000000..9b7a36ed952 Binary files /dev/null and b/audits/SP1_Blobstream_Ottersec_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/stackr.webp b/build/stackr.webp new file mode 100644 index 00000000000..6e483565361 Binary files /dev/null and b/build/stackr.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/coc.html b/community/coc.html new file mode 100644 index 00000000000..52d277de82a --- /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..10c23c8c723 --- /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 Beta validator or an active Mocha testnet validator for at least 1 month before application deadline
    • Run a bridge node (on Mainnet Beta if you are already an active Mainnet Beta 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
    • Maintain a fully archival (non pruned) bridge node for both Mainnet Beta and Mocha if selected for the program
    • Not running your infrastructure in Hetzner or OVH

    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 Beta + 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 Beta
      • If you don't run an active Mainnet Beta 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 Beta 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.

    • Cohort 1: 50 Validator Seats
    • Cohort 2: 15 Validator Seats (Applications open June 1, 2024)
    • Cohort 3: 15 Validator Seats (Applications open October 1, 2024)
    • Cohort 4: 20 Validator Seats (Applications open February 1, 2025)

    IMPORTANT: Each validator selected for the program has to maintain a fully archival (non pruned) bridge node for both Mainnet Beta and Mocha.

    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/modular-meetup-guide.html b/community/modular-meetup-guide.html new file mode 100644 index 00000000000..7deedd0599d --- /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..0e1d48b3d4a --- /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..f2b1de574f6 --- /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/speaker-list.html b/community/speaker-list.html new file mode 100644 index 00000000000..2a10db3c594 --- /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..5f8fc9545cb --- /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-deploy.html b/developers/arbitrum-deploy.html new file mode 100644 index 00000000000..f630cac59ed --- /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..0be46a8a64f --- /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..05223df0eea --- /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 node RPC endpoint, you can find a list of them for Mocha
    • 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/blobstream-contracts.html b/developers/blobstream-contracts.html new file mode 100644 index 00000000000..70bd219fa9d --- /dev/null +++ b/developers/blobstream-contracts.html @@ -0,0 +1,131 @@ + + + + + + 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 contracts

    We will be using the IDAOracle interface to verify inclusion. So, we will install the Blobstream contracts repo as a dependency:

    sh
    forge install celestiaorg/blobstream-contracts --no-commit
    forge install celestiaorg/blobstream-contracts --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;
    +
    +import "blobstream-contracts/IDAOracle.sol";
    +import "blobstream-contracts/DataRootTuple.sol";
    +import "blobstream-contracts/lib/tree/binary/BinaryMerkleProof.sol";
    +
    +contract MyRollup {
    +    IDAOracle immutable blobstream;
    +    bytes32[] public rollup_block_hashes;
    +
    +    constructor(IDAOracle _blobstream) {
    +        blobstream = _blobstream;
    +    }
    +
    +    function submitRollupBlock(
    +        bytes32 _rollup_block_hash,
    +        bytes calldata _zk_proof,
    +        uint256 _blobstream_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(
    +            blobstream.verifyAttestation(_blobstream_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;
    +
    +import "blobstream-contracts/IDAOracle.sol";
    +import "blobstream-contracts/DataRootTuple.sol";
    +import "blobstream-contracts/lib/tree/binary/BinaryMerkleProof.sol";
    +
    +contract MyRollup {
    +    IDAOracle immutable blobstream;
    +    bytes32[] public rollup_block_hashes;
    +
    +    constructor(IDAOracle _blobstream) {
    +        blobstream = _blobstream;
    +    }
    +
    +    function submitRollupBlock(
    +        bytes32 _rollup_block_hash,
    +        bytes calldata _zk_proof,
    +        uint256 _blobstream_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(
    +            blobstream.verifyAttestation(_blobstream_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..13430c33406 --- /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 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..488a5db6050 --- /dev/null +++ b/developers/blobstream-proof-queries.html @@ -0,0 +1,1885 @@ + + + + + + Blobstream proofs queries | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Blobstream proofs queries

    Prerequisites

    • Access to a Celestia 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.

    For golang snippets, the tendermint RPC client, referred to as trpc, will be used for the queries. It can be initialized using:

    go
        trpc, err := http.New("<rpc_endpoint>", "/websocket")
    +	if err != nil {
    +		...
    +	}
    +	err = trpc.Start()
    +	if err != nil {
    +		return err
    +	}
    +	defer func(trpc *http.HTTP) {
    +		err := trpc.Stop()
    +		if err != nil {
    +			...
    +		}
    +	}(trpc)
        trpc, err := http.New("<rpc_endpoint>", "/websocket")
    +	if err != nil {
    +		...
    +	}
    +	err = trpc.Start()
    +	if err != nil {
    +		return err
    +	}
    +	defer func(trpc *http.HTTP) {
    +		err := trpc.Stop()
    +		if err != nil {
    +			...
    +		}
    +	}(trpc)

    The <rpc_endpoint> can be retrieved from Mainnet Beta for and Mocha for the Mocha testnet.

    In case the reader wants to interact with an on-chain contract that can be used to verify that data was posted to Celestia, the bindings of that contract are needed.

    For Blobstream, the golang bindings can be found in the following links:

    text
    https://github.com/succinctlabs/blobstreamx/blob/main/bindings/BlobstreamX.go
    https://github.com/succinctlabs/blobstreamx/blob/main/bindings/BlobstreamX.go
    text
    https://github.com/succinctlabs/sp1-blobstream/blob/main/bindings/SP1Blobstream.go
    https://github.com/succinctlabs/sp1-blobstream/blob/main/bindings/SP1Blobstream.go

    For other languages, the corresponding smart contract bindings should be generated. Refer to abigen for more information.

    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. In a nutshell, Blobstream attests to the data posted to Celestia in the zk-Blobstream 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 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 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 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 contract.

    HTTP query

    Example HTTP request: <tendermint_rpc_endpoint>/data_root_inclusion_proof?height=15&start=10&end=20

    Which queries the proof of the height 15 to the data commitment defined by the range [10, 20).

    Example response:

    json
    {
    +  "jsonrpc": "2.0",
    +  "id": -1,
    +  "result": {
    +    "proof": {
    +      "total": "10",
    +      "index": "5",
    +      "leaf_hash": "vkRaRg7FGtZ/ZhsJRh/Uhhb3U6dPaYJ1pJNEfrwq5HE=",
    +      "aunts": [
    +        "nmBWWwHpipHwagaI7MAqM/yhCDb4cz7z4lRxmVRq5f8=",
    +        "nyzLbFJjnSKOfRZur8xvJiJLA+wBPtwm0KbYglILxLg=",
    +        "GI/tJ9WSwcyHM0r0i8t+p3hPFtDieuYR9wSPVkL1r2s=",
    +        "+SGf6MfzMmtDKz5MLlH+y7mPV9Moo2x5rLjLe3gbFQo="
    +      ]
    +    }
    +  }
    +}
    {
    +  "jsonrpc": "2.0",
    +  "id": -1,
    +  "result": {
    +    "proof": {
    +      "total": "10",
    +      "index": "5",
    +      "leaf_hash": "vkRaRg7FGtZ/ZhsJRh/Uhhb3U6dPaYJ1pJNEfrwq5HE=",
    +      "aunts": [
    +        "nmBWWwHpipHwagaI7MAqM/yhCDb4cz7z4lRxmVRq5f8=",
    +        "nyzLbFJjnSKOfRZur8xvJiJLA+wBPtwm0KbYglILxLg=",
    +        "GI/tJ9WSwcyHM0r0i8t+p3hPFtDieuYR9wSPVkL1r2s=",
    +        "+SGf6MfzMmtDKz5MLlH+y7mPV9Moo2x5rLjLe3gbFQo="
    +      ]
    +    }
    +  }
    +}

    NOTE: These values are base64 encoded. For these to be usable with the solidity smart contract, they need to be converted to bytes32. Check the next section for more information.

    Golang client

    The endpoint can also 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 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
    +}
    go
    // Similar to Blobstream, except replace the BlobstreamX contract with SP1 Blobstream:
    +import {
    +  sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
    +}
    // Similar to Blobstream, except replace the BlobstreamX contract with SP1 Blobstream:
    +import {
    +  sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
    +}

    2. Transaction inclusion proof

    To prove that a rollup transaction, the PFB transaction and not the blob containing the Rollup blocks data, 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.

    Transaction inclusion proof using the transaction hash

    Given a transaction hash, the transaction inclusion proof can be queried from the transaction query.

    HTTP request

    Example request: <tendermint_rpc_endpoint>/tx?hash=0xEF9F50BFB39F11B022A6CD7026574ECCDC6D596689BDCCC7B2C482A1B26B26B8&prove=true

    Which queries the transaction whose hash is EF9F50BFB39F11B022A6CD7026574ECCDC6D596689BDCCC7B2C482A1B26B26B8 and sets the prove parameter as true to also get its inclusion proof.

    Example response:

    json
    {
    +  "jsonrpc": "2.0",
    +  "id": -1,
    +  "result": {
    +    "hash": "EF9F50BFB39F11B022A6CD7026574ECCDC6D596689BDCCC7B2C482A1B26B26B8",
    +    "height": "1350632",
    +    "index": 4,
    +    "tx_result": {
    +      "code": 0,
    +      "data": "EioKKC9jZWxlc3RpYS5ibG9iLnYxLk1zZ1BheUZvckJsb2JzUmVzcG9uc2U=",
    +      "log": "[{\"msg_index\":0,\"events\":[{\"type\":\"celestia.blob.v1.EventPayForBlobs\",\"attributes\":[{\"key\":\"blob_sizes\",\"value\":\"[120000]\"},{\"key\":\"namespaces\",\"value\":\"[\\\"AAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUU=\\\"]\"},{\"key\":\"signer\",\"value\":\"\\\"celestia1vdjkcetnw35kzvtk8pjhxcm4xan82wtvwcurwwtt0f6n2at9va6k2atjw3cn2umhxe58xmfndejs40vqs9\\\"\"}]},{\"type\":\"message\",\"attributes\":[{\"key\":\"action\",\"value\":\"/celestia.blob.v1.MsgPayForBlobs\"}]}]}]",
    +      "info": "",
    +      "gas_wanted": "1095604",
    +      "gas_used": "1080694",
    +      "events": [
    +        ...
    +      ],
    +      "codespace": ""
    +    },
    +    "tx": "CqEBCp4BCiAvY2VsZXN0aWEuYmxvYi52MS5Nc2dQYXlGb3JCbG9icxJ6Ci9jZWxlc3RpYTF2OGVzY3U3ZnU5bHY4NzlrenU1dWVndWV1cnRxNXN3NmhzbTNuZRIdAAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUUaA8CpByIgsVXWya9toI+AyTu3JJA2wkI5ZLkm72/gklCGFLCSrm9CAQASaApSCkYKHy9jb3Ntb3MuY3J5cHRvLnNlY3AyNTZrMS5QdWJLZXkSIwohAqdvBVUpglaNDGTlOcGSoHERBAFsFBB5l0WdvBJjEsEHEgQKAggBGJjCARISCgwKBHV0aWESBDIxOTIQtO9CGkCJRjcOYijj81bttfb2GUdG7o8AuAwf0bscBhW9PPD99xpQ1slpemfyq0y1joJ/aRFgE6QNxuiZ18VLGlGEwtW/",
    +    "proof": {
    +      "data": [
    +        "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQBAAACugAAACbaAgrNAgqfAQqcAQogL2NlbGVzdGlhLmJsb2IudjEuTXNnUGF5Rm9yQmxvYnMSeAovY2VsZXN0aWExYWxwNGZwbHF5d21jNmN1aDl5MzVlY2xycHF5cWF4MjN2Z3ZrczUSHQAAAAAAAAAAAAAAAAAAAAAAAAAAAAgICAgICAgIGgFeIiAzlOEQsGxg3rOw7SR1rkQ7dVJYGp3aXkaqy4oG6HYc0EIBABJnClIKRgofL2Nvc21vcy5jcnlwdG8uc2VjcDI1NmsxLlB1YktleRIjCiECA4ief8FZEaBQLVc2wOceFs+LhAK0mDnmPnsxYLkqv7QSBAoCCAEY5PEBEhEKCwoEdXRpYRIDMTYwELTvBBpA7XBbSGYFrwTZcFHq3va1vHtbRiCzYd0ELkAJo6kLSDooEQCwoVaGuwTdP55V8Btf3WC7/FEK44BOESwEwecTExICgQIaBElORFjcAgrQAgqhAQqeAQogL2NlbGVzdGlhLmJsb2IudjEuTXNnUGF5Rm9yQmxvYnMSegovY2VsZXN0aWExdjhlc2N1N2Z1OWx2ODc5a3p1NXVlZ3VldXJ0cTVzdzZoc20zbmUSHQAAAAAAAAAAAAAAAAAAAAAAAAAAAAWEy1OISzk=",
    +        "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQAAAAAAEUaA8CpByIgsVXWya9toI+AyTu3JJA2wkI5ZLkm72/gklCGFLCSrm9CAQASaApSCkYKHy9jb3Ntb3MuY3J5cHRvLnNlY3AyNTZrMS5QdWJLZXkSIwohAqdvBVUpglaNDGTlOcGSoHERBAFsFBB5l0WdvBJjEsEHEgQKAggBGJjCARISCgwKBHV0aWESBDIxOTIQtO9CGkCJRjcOYijj81bttfb2GUdG7o8AuAwf0bscBhW9PPD99xpQ1slpemfyq0y1joJ/aRFgE6QNxuiZ18VLGlGEwtW/EgEIGgRJTkRYAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA="
    +      ],
    +      "share_proofs": [
    +        {
    +          "start": 5,
    +          "end": 7,
    +          "nodes": [
    +            "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQs98k+8wQ2iX2BdcTfoHjtRQbqybtPdB1BUFY/D7WRs",
    +            "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAZz5Aj1MiJjrOWJdCifYJkr0pCrOIu2jigmd9BzuhZrO",
    +            "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAP8AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA/y91DA3ZRyFzmc7L/ZXrxJ96/2ZDIr/JH0tyRmtrDtHA",
    +            "AAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUUAAAAAAAAAAAAAAAAAAAAAAAAAAAAFhMtTiEs5RU6k2enIm7ThjQyCL82hSxpinyCELhed9QK+p9ZbNIDe",
    +            "AAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUUAAAAAAAAAAAAAAAAAAAAAAAAAAAAFhMtTiEs5RT0T52LWq3L0FNM6KqQKA7NrxFNM8zC/kQKHkPJXMibY",
    +            "/////////////////////////////////////////////////////////////////////////////76AU8rJ4VSYVAsPH5LGqQ2KG/oKPajw+kyhnQkq5Vch"
    +          ]
    +        }
    +      ],
    +      "namespace_id": "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABA==",
    +      "row_proof": {
    +        "row_roots": [
    +          "00000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000584CB53884B394593052DF4039E58C7D51F0E45CACE7DD584125F62F9261B7900AC1EEEF5E82349"
    +        ],
    +        "proofs": [
    +          {
    +            "total": "128",
    +            "index": "0",
    +            "leaf_hash": "Qm/3wL9cWxS8rQbDDgUPU9p8jCfJ+Jc77zsSWdcF6PM=",
    +            "aunts": [
    +              "+mHKrh9boRgjN3oqt/Np5Wre24w+E79/hzFY/eouJY0=",
    +              "8woyobNJs3MT1dKFRQID8VC75oJa2jNF3Wn/1USGfT4=",
    +              "MTgbZqhrQoL61oKKZMhRfYq5bk6gOkLgWrVPArPYXvE=",
    +              "plW1GXaBNavHWwurqsWB0xH25zv9xhiELqtVld0XQC4=",
    +              "K/yH2ZDYNE9u/UT8sJtGuH+akiMNQTKUUu/uhlbGdgo=",
    +              "J2pYcLT4KHpIQvh7b6Wp9KCdMgHLCT9eDfYDr7ZAQ9o=",
    +              "grZxooejIhch93+g3MLdiBq6fF+nrOAKRBgupfu8mUo="
    +            ]
    +          }
    +        ],
    +        "start_row": 0,
    +        "end_row": 0
    +      },
    +      "namespace_version": 0
    +    }
    +  }
    +}
    {
    +  "jsonrpc": "2.0",
    +  "id": -1,
    +  "result": {
    +    "hash": "EF9F50BFB39F11B022A6CD7026574ECCDC6D596689BDCCC7B2C482A1B26B26B8",
    +    "height": "1350632",
    +    "index": 4,
    +    "tx_result": {
    +      "code": 0,
    +      "data": "EioKKC9jZWxlc3RpYS5ibG9iLnYxLk1zZ1BheUZvckJsb2JzUmVzcG9uc2U=",
    +      "log": "[{\"msg_index\":0,\"events\":[{\"type\":\"celestia.blob.v1.EventPayForBlobs\",\"attributes\":[{\"key\":\"blob_sizes\",\"value\":\"[120000]\"},{\"key\":\"namespaces\",\"value\":\"[\\\"AAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUU=\\\"]\"},{\"key\":\"signer\",\"value\":\"\\\"celestia1vdjkcetnw35kzvtk8pjhxcm4xan82wtvwcurwwtt0f6n2at9va6k2atjw3cn2umhxe58xmfndejs40vqs9\\\"\"}]},{\"type\":\"message\",\"attributes\":[{\"key\":\"action\",\"value\":\"/celestia.blob.v1.MsgPayForBlobs\"}]}]}]",
    +      "info": "",
    +      "gas_wanted": "1095604",
    +      "gas_used": "1080694",
    +      "events": [
    +        ...
    +      ],
    +      "codespace": ""
    +    },
    +    "tx": "CqEBCp4BCiAvY2VsZXN0aWEuYmxvYi52MS5Nc2dQYXlGb3JCbG9icxJ6Ci9jZWxlc3RpYTF2OGVzY3U3ZnU5bHY4NzlrenU1dWVndWV1cnRxNXN3NmhzbTNuZRIdAAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUUaA8CpByIgsVXWya9toI+AyTu3JJA2wkI5ZLkm72/gklCGFLCSrm9CAQASaApSCkYKHy9jb3Ntb3MuY3J5cHRvLnNlY3AyNTZrMS5QdWJLZXkSIwohAqdvBVUpglaNDGTlOcGSoHERBAFsFBB5l0WdvBJjEsEHEgQKAggBGJjCARISCgwKBHV0aWESBDIxOTIQtO9CGkCJRjcOYijj81bttfb2GUdG7o8AuAwf0bscBhW9PPD99xpQ1slpemfyq0y1joJ/aRFgE6QNxuiZ18VLGlGEwtW/",
    +    "proof": {
    +      "data": [
    +        "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQBAAACugAAACbaAgrNAgqfAQqcAQogL2NlbGVzdGlhLmJsb2IudjEuTXNnUGF5Rm9yQmxvYnMSeAovY2VsZXN0aWExYWxwNGZwbHF5d21jNmN1aDl5MzVlY2xycHF5cWF4MjN2Z3ZrczUSHQAAAAAAAAAAAAAAAAAAAAAAAAAAAAgICAgICAgIGgFeIiAzlOEQsGxg3rOw7SR1rkQ7dVJYGp3aXkaqy4oG6HYc0EIBABJnClIKRgofL2Nvc21vcy5jcnlwdG8uc2VjcDI1NmsxLlB1YktleRIjCiECA4ief8FZEaBQLVc2wOceFs+LhAK0mDnmPnsxYLkqv7QSBAoCCAEY5PEBEhEKCwoEdXRpYRIDMTYwELTvBBpA7XBbSGYFrwTZcFHq3va1vHtbRiCzYd0ELkAJo6kLSDooEQCwoVaGuwTdP55V8Btf3WC7/FEK44BOESwEwecTExICgQIaBElORFjcAgrQAgqhAQqeAQogL2NlbGVzdGlhLmJsb2IudjEuTXNnUGF5Rm9yQmxvYnMSegovY2VsZXN0aWExdjhlc2N1N2Z1OWx2ODc5a3p1NXVlZ3VldXJ0cTVzdzZoc20zbmUSHQAAAAAAAAAAAAAAAAAAAAAAAAAAAAWEy1OISzk=",
    +        "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQAAAAAAEUaA8CpByIgsVXWya9toI+AyTu3JJA2wkI5ZLkm72/gklCGFLCSrm9CAQASaApSCkYKHy9jb3Ntb3MuY3J5cHRvLnNlY3AyNTZrMS5QdWJLZXkSIwohAqdvBVUpglaNDGTlOcGSoHERBAFsFBB5l0WdvBJjEsEHEgQKAggBGJjCARISCgwKBHV0aWESBDIxOTIQtO9CGkCJRjcOYijj81bttfb2GUdG7o8AuAwf0bscBhW9PPD99xpQ1slpemfyq0y1joJ/aRFgE6QNxuiZ18VLGlGEwtW/EgEIGgRJTkRYAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA="
    +      ],
    +      "share_proofs": [
    +        {
    +          "start": 5,
    +          "end": 7,
    +          "nodes": [
    +            "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQs98k+8wQ2iX2BdcTfoHjtRQbqybtPdB1BUFY/D7WRs",
    +            "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAZz5Aj1MiJjrOWJdCifYJkr0pCrOIu2jigmd9BzuhZrO",
    +            "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAP8AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA/y91DA3ZRyFzmc7L/ZXrxJ96/2ZDIr/JH0tyRmtrDtHA",
    +            "AAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUUAAAAAAAAAAAAAAAAAAAAAAAAAAAAFhMtTiEs5RU6k2enIm7ThjQyCL82hSxpinyCELhed9QK+p9ZbNIDe",
    +            "AAAAAAAAAAAAAAAAAAAAAAAAAAAABYTLU4hLOUUAAAAAAAAAAAAAAAAAAAAAAAAAAAAFhMtTiEs5RT0T52LWq3L0FNM6KqQKA7NrxFNM8zC/kQKHkPJXMibY",
    +            "/////////////////////////////////////////////////////////////////////////////76AU8rJ4VSYVAsPH5LGqQ2KG/oKPajw+kyhnQkq5Vch"
    +          ]
    +        }
    +      ],
    +      "namespace_id": "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABA==",
    +      "row_proof": {
    +        "row_roots": [
    +          "00000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000584CB53884B394593052DF4039E58C7D51F0E45CACE7DD584125F62F9261B7900AC1EEEF5E82349"
    +        ],
    +        "proofs": [
    +          {
    +            "total": "128",
    +            "index": "0",
    +            "leaf_hash": "Qm/3wL9cWxS8rQbDDgUPU9p8jCfJ+Jc77zsSWdcF6PM=",
    +            "aunts": [
    +              "+mHKrh9boRgjN3oqt/Np5Wre24w+E79/hzFY/eouJY0=",
    +              "8woyobNJs3MT1dKFRQID8VC75oJa2jNF3Wn/1USGfT4=",
    +              "MTgbZqhrQoL61oKKZMhRfYq5bk6gOkLgWrVPArPYXvE=",
    +              "plW1GXaBNavHWwurqsWB0xH25zv9xhiELqtVld0XQC4=",
    +              "K/yH2ZDYNE9u/UT8sJtGuH+akiMNQTKUUu/uhlbGdgo=",
    +              "J2pYcLT4KHpIQvh7b6Wp9KCdMgHLCT9eDfYDr7ZAQ9o=",
    +              "grZxooejIhch93+g3MLdiBq6fF+nrOAKRBgupfu8mUo="
    +            ]
    +          }
    +        ],
    +        "start_row": 0,
    +        "end_row": 0
    +      },
    +      "namespace_version": 0
    +    }
    +  }
    +}

    The proof field contains the transaction inclusion proof to the data root.

    Also, the share range where this transaction spans is the end exclusive range defined by proof.share_proofs[0].start and proof.share_proofs[0].end.

    NOTE: The values are base64 encoded. For these to be usable with the solidity smart contract, they need to be converted to bytes32. Check the next section for more information.

    Golang client

    Using the golang client:

    go
        txHash, err := hex.DecodeString("<transaction_hash>")
    +    if err != nil {
    +		...
    +	}
    +    tx, err := trpc.Tx(cmd.Context(), txHash, true)
    +    if err != nil {
    +		...
    +    }
        txHash, err := hex.DecodeString("<transaction_hash>")
    +    if err != nil {
    +		...
    +	}
    +    tx, err := trpc.Tx(cmd.Context(), txHash, true)
    +    if err != nil {
    +		...
    +    }

    Then, the proof is under tx.Proof.

    Blob inclusion proof using the corresponding PFB transaction hash

    Currently, querying the proof of a blob, which contains the Rollup block data, using its corresponding PFB transaction hash is possible only using the golang client. Otherwise, the corresponding share range is required so that the ProveShares endpoint can be used.

    Golang client

    Using the golang client:

    go
    import (
    +	"context"
    +	"encoding/hex"
    +	"github.com/celestiaorg/celestia-app/v2/pkg/appconsts"
    +	"github.com/celestiaorg/go-square/square"
    +	"github.com/tendermint/tendermint/rpc/client/http"
    +)
    +
    +func queryShareRange() error {
    +	txHash, err := hex.DecodeString("<transaction_hash>")
    +	if err != nil {
    +		return err
    +	}
    +	tx, err := trpc.Tx(context.Background(), txHash, true)
    +	if err != nil {
    +		return err
    +	}
    +	
    +	blockRes, err := trpc.Block(context.Background(), &tx.Height)
    +	if err != nil {
    +		return err
    +	}
    +
    +	version := blockRes.Block.Header.Version.App
    +	maxSquareSize := appconsts.SquareSizeUpperBound(version)
    +	subtreeRootThreshold := appconsts.SubtreeRootThreshold(version)
    +	blobShareRange, err := square.BlobShareRange(
    +		blockRes.Block.Txs.ToSliceOfBytes(),
    +		int(tx.Index),
    +		<blob_index>,
    +		maxSquareSize,
    +		subtreeRootThreshold,
    +	)
    +	if err != nil {
    +		return err
    +	}
    +}
    import (
    +	"context"
    +	"encoding/hex"
    +	"github.com/celestiaorg/celestia-app/v2/pkg/appconsts"
    +	"github.com/celestiaorg/go-square/square"
    +	"github.com/tendermint/tendermint/rpc/client/http"
    +)
    +
    +func queryShareRange() error {
    +	txHash, err := hex.DecodeString("<transaction_hash>")
    +	if err != nil {
    +		return err
    +	}
    +	tx, err := trpc.Tx(context.Background(), txHash, true)
    +	if err != nil {
    +		return err
    +	}
    +	
    +	blockRes, err := trpc.Block(context.Background(), &tx.Height)
    +	if err != nil {
    +		return err
    +	}
    +
    +	version := blockRes.Block.Header.Version.App
    +	maxSquareSize := appconsts.SquareSizeUpperBound(version)
    +	subtreeRootThreshold := appconsts.SubtreeRootThreshold(version)
    +	blobShareRange, err := square.BlobShareRange(
    +		blockRes.Block.Txs.ToSliceOfBytes(),
    +		int(tx.Index),
    +		<blob_index>,
    +		maxSquareSize,
    +		subtreeRootThreshold,
    +	)
    +	if err != nil {
    +		return err
    +	}
    +}

    With the <transaction_hash> being the transaction hash of the PFB containing the blob and, the <blob_index> being the index of the blob. In fact, PayForBlob transactions can contain multiple blobs. So, the <blob_index> is the index of the blob in the PFB.

    Specific share range inclusion proof

    To retrieve the inclusion proof of a set of shares for whom the share range is already known, the ProveShares query can be used to query it.

    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 share to row root NMT proofs and multiple row root to data root binary proofs.

    HTTP request

    Example request: <tendermint_rpc_endpoint>/prove_shares?height=15&startShare=0&endShare=1

    Which queries the proof of shares [0,1) in block 15.

    Example response:

    json
    {
    +  "jsonrpc": "2.0",
    +  "id": -1,
    +  "result": {
    +    "data": [
    +      "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQBAAABXAAAACbaAgrOAgqgAQqdAQogL2NlbGVzdGlhLmJsb2IudjEuTXNnUGF5Rm9yQmxvYnMSeQovY2VsZXN0aWExdWc1ZWt0MmNjN250dzRkdG1zZDlsN3N0cTBzN3Z5ZTd5bTJyZHISHQAAAAAAAAAAAAAAAAAAAAAAAAASExIyQkMkMoiZGgKXAiIgrfloW1M/Y33zlD2luveDELZzr9cF92+2eTaImIWhN9pCAQASZwpQCkYKHy9jb3Ntb3MuY3J5cHRvLnNlY3AyNTZrMS5QdWJLZXkSIwohA36hewmW/AXtrw6S+QsNUzFGfeg37Da6igoP2ZQcK+04EgQKAggBGAISEwoNCgR1dGlhEgUyMTAwMBDQ6AwaQClYLQPNrFoD6H8mgmwxjFeNhwhRu39EcrVKMFkNQ8+HHuodhdOQIG/8DXEmrBwrpwj6hi+3uEsZ+0p5vrf3v8sSAQEaBElORFgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA="
    +    ],
    +    "share_proofs": [
    +      {
    +        "end": 1,
    +        "nodes": [
    +          "AAAAAAAAAAAAAAAAAAAAAAAAABITEjJCQyQyiJkAAAAAAAAAAAAAAAAAAAAAAAAAEhMSMkJDJDKImbiwnpOdwIZBFr0UiFhPKwGy/XIIjL+gqm0fqxIw0z0o",
    +          "/////////////////////////////////////////////////////////////////////////////3+fuhlzUfKJnZD8yg/JOtZla2V3g2Q7y+18iH5j0Uxk"
    +        ]
    +      }
    +    ],
    +    "namespace_id": "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABA==",
    +    "row_proof": {
    +      "row_roots": [
    +        "000000000000000000000000000000000000000000000000000000000400000000000000000000000000000000000000121312324243243288993946154604701154F739F3D1B5475786DDD960F06D8708D4E870DA6501C51750"
    +      ],
    +      "proofs": [
    +        {
    +          "total": "8",
    +          "index": "0",
    +          "leaf_hash": "300xzO8TiLwPNuREY6OJcRKzTHQ4y6yy6qH0wAuMMrc=",
    +          "aunts": [
    +            "ugp0sV9YNEI5pOiYR7RdOdswwlfBh2o3XiRsmMNmbKs=",
    +            "3dMFZFaWZMTZVXhphF5TxlCJ+CT3EvmMFOpiXFH+ID4=",
    +            "srl59GiTSiwC9LqdYASzFC6TvusyY7njX8/XThp6Xws="
    +          ]
    +        }
    +      ],
    +      "start_row": 0,
    +      "end_row": 0
    +    },
    +    "namespace_version": 0
    +  }
    +}
    {
    +  "jsonrpc": "2.0",
    +  "id": -1,
    +  "result": {
    +    "data": [
    +      "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAQBAAABXAAAACbaAgrOAgqgAQqdAQogL2NlbGVzdGlhLmJsb2IudjEuTXNnUGF5Rm9yQmxvYnMSeQovY2VsZXN0aWExdWc1ZWt0MmNjN250dzRkdG1zZDlsN3N0cTBzN3Z5ZTd5bTJyZHISHQAAAAAAAAAAAAAAAAAAAAAAAAASExIyQkMkMoiZGgKXAiIgrfloW1M/Y33zlD2luveDELZzr9cF92+2eTaImIWhN9pCAQASZwpQCkYKHy9jb3Ntb3MuY3J5cHRvLnNlY3AyNTZrMS5QdWJLZXkSIwohA36hewmW/AXtrw6S+QsNUzFGfeg37Da6igoP2ZQcK+04EgQKAggBGAISEwoNCgR1dGlhEgUyMTAwMBDQ6AwaQClYLQPNrFoD6H8mgmwxjFeNhwhRu39EcrVKMFkNQ8+HHuodhdOQIG/8DXEmrBwrpwj6hi+3uEsZ+0p5vrf3v8sSAQEaBElORFgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA="
    +    ],
    +    "share_proofs": [
    +      {
    +        "end": 1,
    +        "nodes": [
    +          "AAAAAAAAAAAAAAAAAAAAAAAAABITEjJCQyQyiJkAAAAAAAAAAAAAAAAAAAAAAAAAEhMSMkJDJDKImbiwnpOdwIZBFr0UiFhPKwGy/XIIjL+gqm0fqxIw0z0o",
    +          "/////////////////////////////////////////////////////////////////////////////3+fuhlzUfKJnZD8yg/JOtZla2V3g2Q7y+18iH5j0Uxk"
    +        ]
    +      }
    +    ],
    +    "namespace_id": "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABA==",
    +    "row_proof": {
    +      "row_roots": [
    +        "000000000000000000000000000000000000000000000000000000000400000000000000000000000000000000000000121312324243243288993946154604701154F739F3D1B5475786DDD960F06D8708D4E870DA6501C51750"
    +      ],
    +      "proofs": [
    +        {
    +          "total": "8",
    +          "index": "0",
    +          "leaf_hash": "300xzO8TiLwPNuREY6OJcRKzTHQ4y6yy6qH0wAuMMrc=",
    +          "aunts": [
    +            "ugp0sV9YNEI5pOiYR7RdOdswwlfBh2o3XiRsmMNmbKs=",
    +            "3dMFZFaWZMTZVXhphF5TxlCJ+CT3EvmMFOpiXFH+ID4=",
    +            "srl59GiTSiwC9LqdYASzFC6TvusyY7njX8/XThp6Xws="
    +          ]
    +        }
    +      ],
    +      "start_row": 0,
    +      "end_row": 0
    +    },
    +    "namespace_version": 0
    +  }
    +}

    NOTE: The values are base64 encoded. For these to be usable with the solidity smart contract, they need to be converted to bytes32. Check the next section for more information.

    WARNING

    As of Celestia-app v1.10.0, the prove_shares endpoint is being deprecated in favor of prove_shares_v2. Please use the new endpoint for the queries as the old one will be removed in upcoming releases.

    Golang client

    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 {
    +		...
    +	}

    WARNING

    As of Celestia-app v1.10.0, the ProveShares method is being deprecated in favor of ProveSharesV2. Please use the new method for the queries as the old one will be removed in upcoming releases.

    Converting the proofs to be usable in the DAVerifier library

    Smart contracts that use the DAVerifier library take 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 Blobstream contract.
    +    AttestationProof attestationProof;
    +}
    +
    +/// @notice Contains the necessary parameters needed to verify that a data root tuple
    +/// was committed to, by the Blobstream 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 Blobstream contract.
    +    AttestationProof attestationProof;
    +}
    +
    +/// @notice Contains the necessary parameters needed to verify that a data root tuple
    +/// was committed to, by the Blobstream 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 adapt the queried response above 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.

    If the data field is retrieved from an HTTP request, it should be converted to hex before using abi.encode(...).

    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 in case of an HTTP request, 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 localized.

    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 contract:

    solidity
    /// @notice Contains the necessary parameters needed to verify that a data root tuple
    +/// was committed to, by the Blobstream 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 Blobstream 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 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 Blobstream 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 data commitment stored events from the Blobstream contract and looking for the nonce attesting to the corresponding data.

    Querying the proof's tupleRootNonce

    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")
    +	}
    go
    // Similar to BlobstreamX, but instead of importing the BlobstreamX contract,
    +// import the SP1 Blobstream contract:
    +import {
    +    sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
    +} 
    +// and use the `BlobstreamDataCommitmentStored` event instead.
    // Similar to BlobstreamX, but instead of importing the BlobstreamX contract,
    +// import the SP1 Blobstream contract:
    +import {
    +    sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
    +} 
    +// and use the `BlobstreamDataCommitmentStored` event instead.

    Listening for new data commitments

    For listening for new data commitment stored 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)
    +	    }
    +    }
    go
    // Similar to BlobstreamX, but instead of importing the BlobstreamX contract,
    +// import the SP1 Blobstream contract:
    +import {
    +    sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
    +} 
    +// and use the `BlobstreamDataCommitmentStored` event instead.
    // Similar to BlobstreamX, but instead of importing the BlobstreamX contract,
    +// import the SP1 Blobstream contract:
    +import {
    +    sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
    +} 
    +// and use the `BlobstreamDataCommitmentStored` event instead.

    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
    +	// Note: if you're using Celestia-app v1.10.0 onwards, please switch
    +	// to `trpc.ProveSharesV2` as `trpc.ProveShars` is deprecated.
    +	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
    +	// Note: if you're using Celestia-app v1.10.0 onwards, please switch
    +	// to `trpc.ProveSharesV2` as `trpc.ProveShars` is deprecated.
    +	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,
    +	}
    +}
    go
    // Similar to BlobstreamX, but instead of importing the BlobstreamX contract,
    +// import the SP1 Blobstream contract:
    +import {
    +    sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
    +} 
    +// and use the `BlobstreamDataCommitmentStored` event instead.
    // Similar to BlobstreamX, but instead of importing the BlobstreamX contract,
    +// import the SP1 Blobstream contract:
    +import {
    +    sp1blobstreamwrapper "github.com/succinctlabs/sp1-blobstream/bindings"
    +} 
    +// and use the `BlobstreamDataCommitmentStored` event instead.

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

    For an example BlobstreamX project that uses the above proof queries, checkout the blobstreamx-example sample project.

    Learn more on the Lightlink docs.

    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..68c0e33e369 --- /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

    This section will go over two constructs that can be used in building Blobstream rollups. Each with its pros and cons and the rollup developer can choose which one suits their needs better.

    Note: Only the sequence of spans method can be used currently to build Blobstream rollups. The blob share commitment way still requires some tooling that will be built in the upcoming months.

    Blob share commitment

    The blob 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.

    Blob share commitment: Proof details

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

    1. share inclusion proof to the blob 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 blob share commitment
    2. blob share commitment inclusion proof to the data root tuple root: meaning four merkle proofs:
      1. subtree roots merkle proofs to the blob 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 blob 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 blob share commitment has been posted to Celestia.

    NOTE

    Generating/verifying blob 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.

    Blob share commitment: Compact proofs

    There is a way to have compact proofs, when using blob 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 blob 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.

    Blob share commitment: Pros

    The pros of referencing rollup data using a blob 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.

    Blob 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 blob 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 blob share commitment and a sequence of spans is that when using a blob share commitment, an extra merkle proof is needed to prove inclusion of the share to the blob 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 blob 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 blob share commitments

    Another way to build a rollup is to replace the sequence of spans with a height and a blob 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 blob 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 blob 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 blob 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 blob 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 blob share commitments: Pros

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

    Optimistic rollups that use blob 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 blob 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 blob 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 blob share commitments

    To use blob 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 blob 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 blob share commitment is valid using the proofs laid out in the proof details of blob share commitment section.
    3. The zk-proof verifier would take as argument the data root and the blob share commitment. Then, inside the circuit, the protobuf encoded PFB transaction will be deserialized and then verify the following:
    • The deserialized blob 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 blob share commitments: Pros

    None

    Zk-rollups that use blob 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 blob 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 blob share commitment is valid using the proofs laid out in the blob share commitment proof details section.
    3. The zk-proof verifier would take as argument the data root and the blob share commitment. Then, inside the circuit:
    • It will verify that the input blob share commitment corresponds to the rollup data.
    • Verify that the input data root commits to that blob share commitment. Check the blob 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.

    FAQ

    Should I use the Celestia transaction hash to reference the rollup data?

    This is asked a lot since it's the most intuitive way of referencing data. However, in Celestia, referencing the data using the transaction hash is not recommended.

    A transaction proof in Celestia goes back to providing an inclusion proof of the shares containing the transaction. This means if the transaction hash is used to reference data in a Celestia block, the rollup verification mechanism should do the following:

    • Verify an inclusion proof of the shares comprising the transaction up to the data root tuple root
    • Decode those shares and parse the transaction, then hash its components to generate the transaction hash
    • Verify that the generated transaction hash matches the one used to reference the data

    At this level, the transaction hash is authenticated and the verification contract has the shares of the transaction. Then, the verification contract needs to take the share commitment from the parsed transaction and follow the steps outlined in the blob share commitment section.

    As observed, using the transaction hash is expensive and doesn't yield any advantages over using the blob share commitment, which in turn is more expensive than using the sequence of spans.

    So, unless there are more reasons to use the transaction hash to reference the rollup data, the sequence of spans approach remains better.

    + + + + \ 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..9792b64da86 --- /dev/null +++ b/developers/blobstream-x-deploy.html @@ -0,0 +1,217 @@ + + + + + + New Blobstream X deployments | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    New Blobstream X deployments

    This document will go over the instructions to deploy BlobstreamX to a new chain.

    Deploying the contracts

    To deploy a Blobstream X to a new chain, where a Gateway contract does not exist yet, the following steps need to be followed.

    If any of the components already exist in the target chain, feel free to skip the corresponding step.

    Deploy a new SuccinctGateway

    The SuccinctGateway is a contract that acts as a registry for onchain circuit verifiers and manages their access control. It is the entrypoint for proof verification and does the following:

    1. Receive a PLONK proof from the prover
    2. Fetch the address of the target function verifier
    3. Verify if the prover is whitelisted, if whitelisting is enabled
    4. Forward the proof to the function verifier to be verified
    5. If the proof is valid, it calls back the BlobstreamX contract to update its state

    The BlobstreamX requires the update to be provided through the SuccinctGateway. Otherwise, the contract can't be updated.

    To deploy a SuccinctGateway contract, you need to have foundry installed. If not, refer to foundry documentation.

    Then, clone the succinctx repo:

    shell
    git clone https://github.com/succinctlabs/succinctx
    +cd succinctx
    git clone https://github.com/succinctlabs/succinctx
    +cd succinctx

    Next, build the contracts:

    shell
    cd contracts
    +forge build
    cd contracts
    +forge build

    Then, setup the .env containing the required information for deployment. An example .env.example is provided to be inspired from.

    Assume we're deploying to a chain which chainID is 12345. The .env should look like:

    shell
    # The salt to be used for CREATE2 deployments. It's a 32-byte string in hex format.
    +# Example: 0x0000000000000000000000000000000000000000000000000000000000000001
    +CREATE2_SALT=
    +# The 'owner' of the contracts, recommended to be an EOA
    +GUARDIAN=
    +# The default prover to fullfill requests for Functions that have not opted for a different prover
    +PROVER=
    +# RPC URLs for each chain you want to deploy to
    +RPC_12345=
    +# Etherscan API keys for each chain you want to deploy to
    +ETHERSCAN_API_KEY_12345=
    +# Wallet type is the type of the wallet.
    +# If you are using a ledger to sign the transaction,
    +# set `WALLET_TYPE=LEDGER` and specify the `MNEMONIC_INDEX`
    +# for which signer you want to use.
    +# If you are using a private key,
    +# set `WALLET_TYPE=PRIVATE_KEY` and specify the `PRIVATE_KEY`.
    +# In this example, we're using private key.
    +WALLET_TYPE=PRIVATE_KEY
    +# The private key of the deployer account
    +PRIVATE_KEY=
    +# The address of the succinct fee vault contract. Set it to an example address
    +# if you don't want to use a vault like: 0x0000000000000000000000000000000000000001
    +SUCCINCT_FEE_VAULT_12345=
    # The salt to be used for CREATE2 deployments. It's a 32-byte string in hex format.
    +# Example: 0x0000000000000000000000000000000000000000000000000000000000000001
    +CREATE2_SALT=
    +# The 'owner' of the contracts, recommended to be an EOA
    +GUARDIAN=
    +# The default prover to fullfill requests for Functions that have not opted for a different prover
    +PROVER=
    +# RPC URLs for each chain you want to deploy to
    +RPC_12345=
    +# Etherscan API keys for each chain you want to deploy to
    +ETHERSCAN_API_KEY_12345=
    +# Wallet type is the type of the wallet.
    +# If you are using a ledger to sign the transaction,
    +# set `WALLET_TYPE=LEDGER` and specify the `MNEMONIC_INDEX`
    +# for which signer you want to use.
    +# If you are using a private key,
    +# set `WALLET_TYPE=PRIVATE_KEY` and specify the `PRIVATE_KEY`.
    +# In this example, we're using private key.
    +WALLET_TYPE=PRIVATE_KEY
    +# The private key of the deployer account
    +PRIVATE_KEY=
    +# The address of the succinct fee vault contract. Set it to an example address
    +# if you don't want to use a vault like: 0x0000000000000000000000000000000000000001
    +SUCCINCT_FEE_VAULT_12345=
    • PRIVATE_KEY: the private key of the account used to send the transaction.
    • CREATE2_SALT: the salt used to generate the gateway address. Using the same salt between deployments in different chains will make the gateways be on the same address.
    • GUARDIAN: the owner of the SuccinctGateway contract. It should be set to an account that will have full access to the gateway.
    • PROVER: the address of the account used to submit the proofs to the gateway. It is enabled by default for all the registered function verifiers.
    • RPC_12345: the RPC endpoint for the EVM chain whose chain ID is 12345. If the chain ID is different, make sure to change it in the environment variable name as well.
    • ETHERSCAN_API_KEY_12345: the Etherscan API key corresponding to the chain whose chain ID is 12345. Similar to RPC_12345, make sure to change the chain ID in the environment variable name if it's a different chain.

    Then, save the environment to a .env file and run the following:

    shell
    ./script/deploy.sh "SuccinctGateway" "12345 1234"
    ./script/deploy.sh "SuccinctGateway" "12345 1234"

    with 12345 and 1234 being chainID that we want to deploy to, given that the corresponding environment variables, as specified above, are setup correctly.

    Now the SuccinctGateway address should be printed on the terminal.

    Deploy the function verifiers

    The function verifiers are the onchain ciruit verifiers that take PLONK proofs as an input and verify them onchain.

    The function verifiers can be either downloaded from the succinct platform or regenerated.

    To download the function verifiers, along with the circuits binaries, check the list in the Succinct documentation and download the binaries and function verifiers corresponding to the circuits you want to verify.

    Alternatively, you can generate them locally as specified in the regenerating the downloaded artifacts section.

    After getting the FunctionVerifier.sol, either via downloading it or generating it locally, you can choose your favourite way to deploy it.

    A simple way would be to copy the contract in the BlobstreamX repo and deploy it:

    shell
    cd $HOME
    +git clone https://github.com/succinctlabs/blobstreamx
    +cd blobstreamx/contracts
    +
    +forge install
    +
    +# copy the function verifier contract to the blobstream X contracts
    +cp <path_to_function_verifier>/FunctionVerifier.sol src
    +
    +forge build
    +
    +forge create FunctionVerifier --rpc-url <rpc_url> --private-key <private_key> --verify --verifier etherscan --etherscan-api-key <etherscan_api_key>
    +# ^ will return the address of the function verifier
    cd $HOME
    +git clone https://github.com/succinctlabs/blobstreamx
    +cd blobstreamx/contracts
    +
    +forge install
    +
    +# copy the function verifier contract to the blobstream X contracts
    +cp <path_to_function_verifier>/FunctionVerifier.sol src
    +
    +forge build
    +
    +forge create FunctionVerifier --rpc-url <rpc_url> --private-key <private_key> --verify --verifier etherscan --etherscan-api-key <etherscan_api_key>
    +# ^ will return the address of the function verifier

    Register the function verifier in the deployed SuccinctGateway

    After deploying a function verifier for your target circuit, we should register it in a SuccinctGateway contract, either deployed by you in a previous step or pre-existing on the chain, to get a functionID and be able to use it to verify circuits.

    A simple way to do it is to use cast tool from the Foundry toolset:

    First, we will get the expected functionID by calling cast call which simulates the transaction execution and returns the return value:

    shell
    cast call <succinct_gateway_contract_address> "registerFunction(address,address,bytes32)(bytes32)" --rpc-url <rpc_url> <owner_of_the_function_verifier> <address_of_the_function_verifier> <32_bytes_of_salt> --private-key <private_key>
    cast call <succinct_gateway_contract_address> "registerFunction(address,address,bytes32)(bytes32)" --rpc-url <rpc_url> <owner_of_the_function_verifier> <address_of_the_function_verifier> <32_bytes_of_salt> --private-key <private_key>

    This will return a bytes32 which is the function ID.

    Now, we will execute the transaction to register that returned functionID in the gateway, using the same parameters:

    shell
    cast send <succinct_gateway_contract_address> "registerFunction(address,address,bytes32)(bytes32)" --rpc-url <rpc_url> <owner_of_the_function_verifier> <address_of_the_function_verifier> <32_bytes_of_salt> --private-key <private_key>
    cast send <succinct_gateway_contract_address> "registerFunction(address,address,bytes32)(bytes32)" --rpc-url <rpc_url> <owner_of_the_function_verifier> <address_of_the_function_verifier> <32_bytes_of_salt> --private-key <private_key>

    To verify that the registration was successful, run:

    shell
    cast call <succinct_gateway_contract_address> "verifiers(bytes32)(address)" --rpc-url <rpc_url> <returned_function_ID>
    cast call <succinct_gateway_contract_address> "verifiers(bytes32)(address)" --rpc-url <rpc_url> <returned_function_ID>

    which will return the address of the function verifier that was deployed in the previous section.

    NOTE: For BlobstreamX, there are always two function verifiers, corresponding to the two circuits: header_range and next_header. Make sure to register both the function verifiers corresponding to those circuits as we will use those function IDs to deploy the BlobstreamX contract in the next section.

    Enable prover whitelisting

    Now that the function verifier's contract is deployed and registered in the succinct gateway, we can define whitelisting rules for the proof submission.

    by default, the whitelist status is set to Default. This means that only the default verifier, which was setup when deploying the SuccinctGateway. And if you want to restrict the list of provers that can submit proofs to your registered function verifier, you can set the whitelisting status of the function verifier and then add a custom prover. Or even allow for permissionlss submission.

    Set Whitelist Status

    Set the whitelist status of a functionID to:

    • 0: Default status, which only allows the default prover to submit proofs
    • 1: Custom status, which only allows a custom list of provers to submit the proofs. The custom list can be setup by calling the addCustomProver(bytes32 _functionId, address _prover) function.
    • 2: Disabled status, which allow for permissionless proof submission, i.e., any relayer can submit proofs to be verified by that function verifier.
    shell
    cast calldata "setWhitelistStatus(bytes32,uint8)" <YOUR_FUNCTION_ID> <WHITELIST_STATUS>
    cast calldata "setWhitelistStatus(bytes32,uint8)" <YOUR_FUNCTION_ID> <WHITELIST_STATUS>

    Add Custom Prover

    Add a custom prover for a specific functionID.

    shell
    cast calldata "addCustomProver(bytes32,address)" <FUNCTION_ID> <CUSTOM_PROVER_ADDRESS>
    cast calldata "addCustomProver(bytes32,address)" <FUNCTION_ID> <CUSTOM_PROVER_ADDRESS>

    Deploy the BlobstreamX contract

    To deploy a BlobstreamX contract, you need to have foundry installed. If not, refer to foundry documentation.

    Then, clone the repository:

    shell
    cd $HOME
    +git clone https://github.com/succinctlabs/blobstreamx
    +cd blobstreamx
    cd $HOME
    +git clone https://github.com/succinctlabs/blobstreamx
    +cd blobstreamx

    Deploying the BlobstreamX contract requires initializing it with a trusted header.

    In case we want to run a local prover, as is documented in the run a local prover section, the trusted header can be any header in the last three weeks of blocks, aka the unbonding period. Using an older header is also possible, but in case of a malicious RPC provider, the malicious validators won't be slashed.

    Otherwise, we can run a proof replayer that will read the proofs from an existing deployment and submit them into the new one. This is documented under the run a proof replayer from an existing deployment section. In this case, the trusted header should correspond to a height already submitted to the existing deployment, and was the beginning of a batch. These heights can be found via querying the DataCommitmentStored events, or navigating to the events tab in Etherscan for the existing deployment, and choosing a startBlock from a certain event.

    Querying the trusted hash

    Given that we have a trusted height, we can query its corresponding trusted hash using:

    shell
    TENDERMINT_RPC_URL=<celestia_rpc_address> cargo run --bin genesis -- --block <trusted_height>
    TENDERMINT_RPC_URL=<celestia_rpc_address> cargo run --bin genesis -- --block <trusted_height>

    Alternatively, the trusted hash can be obtained via querying the <celestia_rpc>/block?height=<trusted_height> and getting the result.block_id.hash field.

    Deployment instructions

    Once we have the trusted hash, we can deploy the BlobstreamX contract. To do so, create a .env file containing the necessary variables. An example one can be found under contracts/.env.example:

    shell
    cd contracts
    +cp .env.example .env
    +vim .env # and fill the necessary environment variables
    cd contracts
    +cp .env.example .env
    +vim .env # and fill the necessary environment variables

    The needed environment variables are:

    • PRIVATE_KEY: the EVM private key to use to deploy the contract.
    • RPC_URL: the RPC endpoint of the chain where we want to deploy the BlobstreamX contract.
    • ETHERSCAN_API_KEY: the Etherscan API for the target chain.
    • CREATE2_SALT: salt for the CREATE2 method. Example: 0xaa.
    • GUARDIAN_ADDRESS: the BlobstreamX contract's owner. It allows it to update the functionIDs of the deployment, freeze the contract, upgrade it, and others.
    • GATEWAY_ADDRESS: the address of the SuccinctGateway deployment that will update the contract's state. As stated in the section above, the SuccinctGateway is the entrypoint that routes the proof to the function verifier, and if the proof is valid, it updates the BlobstreamX contract.
    • NEXT_HEADER_FUNCTION_ID: the functionID of the next header circuit verifier. It is obtained after deploying a function verifier, corresponding to the next header circuit, and registering it in the SuccinctGateway. More information in the function verifier registration section.
    • HEADER_RANGE_FUNCTION_ID: similar to NEXT_HEADER_FUNCTION_ID but for the header range circuit.
    • GENESIS_HEIGHT: is the height of the trusted header that we queried its trusted hash in the querying the trusted hash section. If the value starts with 0x, it will be parsed as a hex value. Otherwise, it will be treated as a decimal number.
    • GENESIS_HEADER: the queried trusted hash in the querying the trusted hash section.
    • DEPLOY: set to true since we're deploying the contract.
    • UPGRADE: set to false since we're not upgrading an existing deployment.
    • UPDATE_GENESIS_STATE: set to false since we're not updating the genesis state.
    • UPDATE_FUNCTION_IDS: set to false since we're not updating the function IDs.
    • UPDATE_GATEWAY: set to false since we're not updating the address of the gateway.
    • CONTRACT_ADDRESS: set to an empty value since we're not upgrading the deployment.

    Save the .env file then run:

    shell
    forge install
    +
    +source .env
    +
    +forge script script/Deploy.s.sol --rpc-url $RPC_URL --private-key $PRIVATE_KEY --broadcast --verify --verifier etherscan --etherscan-api-key $ETHERSCAN_API_KEY
    forge install
    +
    +source .env
    +
    +forge script script/Deploy.s.sol --rpc-url $RPC_URL --private-key $PRIVATE_KEY --broadcast --verify --verifier etherscan --etherscan-api-key $ETHERSCAN_API_KEY

    And you should see the address printed in the logs.

    Run a local prover

    Now that the BlobstreamX contract is deployed, we can either run a local prover or opt for proofs replaying.

    For running a local prover, make sure to have a beefy machine. It is recommended to run the prover in at least a 32CPU 256GB RAM machine to be able to generate proofs in time for a 1hr proofs submission frequency.

    To run the prover:

    shell
    cd $HOME
    +# in case you still did not clone the repository:
    +# git clone https://github.com/succinctlabs/blobstreamx
    +cd blobstreamx
    +# create the environment for the prover
    +cp .env.example .env
    +# edit the environment file
    +vim .env
    cd $HOME
    +# in case you still did not clone the repository:
    +# git clone https://github.com/succinctlabs/blobstreamx
    +cd blobstreamx
    +# create the environment for the prover
    +cp .env.example .env
    +# edit the environment file
    +vim .env

    The following is the required environment for the prover:

    • PRIVATE_KEY: the EVM private key to use to submit the proofs.
    • RPC_URL: the RPC endpoint of the chain where the BlobstreamX contract is deployed.
    • TENDERMINT_RPC_URL: the Celestia chain RPC endpoint. Accepts a comma-separated list of RPC URLs for fail over.
    • CHAIN_ID: the target EVM chain ID.
    • CONTRACT_ADDRESS: the target BlobstreamX contract address.
    • NEXT_HEADER_FUNCTION_ID: the function ID of the next header function registered in the succinct gateway.
    • HEADER_RANGE_FUNCTION_ID: the function ID of the header range function registered in the succinct gateway.
    • LOOP_DELAY_MINS: the time to wait before sending the proofs in minutes. For example, for having a proof every 1hr, set it to 60.
    • LOCAL_PROVE_MODE: set to true to enable local proving, which is what these instructions are about.
    • LOCAL_RELAY_MODE: set to true to enable submitting the proofs onchain, which is what these instructions are about.
    • GATEWAY_ADDRESS: the address of the succinct gateway contract in the target chain.

    For the circuits binaries, you can either generate them as specified in the build the circuits section, or download them from the Succinct platform. And then set the following environment variables:

    • PROVE_BINARY_0xFILL_IN_NEXT_HEADER_FUNCTION_ID: the next header circuit binary path. Make sure to also change the 0xFILL_IN_NEXT_HEADER_FUNCTION_ID in the environment variable name to the functionID of the corresponding function verifier.
    • PROVE_BINARY_0xFILL_IN_HEADER_RANGE_FUNCTION_ID: the header range circuit binary path. Make sure to also change the 0xFILL_IN_HEADER_RANGE_FUNCTION_ID in the environment variable name to the functionID of the corresponding function verifier.

    Similarly, the verifier build can be generated as specified in the regeneration of the verifier build section, or downloaded from the succinct platform. And then set the following environment variable:

    • WRAPPER_BINARY: the path the wrapper verifier

    After setting the environment, run:

    shell
    source .env
    +cargo run --bin blobstreamx --release
    source .env
    +cargo run --bin blobstreamx --release

    And you should see the operator's logs.

    Run a proof replayer from an existing deployment

    A cheaper alternative to running the prover locally is to deploy the BlobstreamX contract at a height that is followed by an existing BlobstreamX contract. Then, keep replaying the proofs from the existing deployment to the new one.

    Check the blobstream-ops repository for more documentation.

    Optional: Regenerating the downloaded artifacts

    In case you want to generate verifier-build, which is the template solidity contract used to verify the PLONK proofs onchain, or the circuits build, which are used to generate the proofs, follow this section.

    Otherwise, use the downloaded ones as specified in the previous section.

    Regenerate the verifier-build

    The verifier-build is the gnark wrapper circuit used by the plonky2x circuits for cheap on-chain verification. It outputs a PLONK proof.

    It can be built using the following instructions:

    shell
    cd $HOME
    +git clone https://github.com/succinctlabs/succinctx
    +cd succinctx
    +cargo test test_wrapper
    +# ^ This should generate a data folder under plonky2x/verifier/data
    +
    +# Compile circuit
    +cd plonky2x/verifier
    +mkdir verifier-build
    +go run . -compile -data verifier-build -circuit data/dummy
    +
    +# Compile executable
    +CGO_ENABLED=0 go build -o verifier-build/verifier -ldflags "-s -w" .
    +
    +
    +ls verifier-build/
    +# ^ should have: pk.bin  r1cs.bin  verifier  Verifier.sol  vk.bin
    cd $HOME
    +git clone https://github.com/succinctlabs/succinctx
    +cd succinctx
    +cargo test test_wrapper
    +# ^ This should generate a data folder under plonky2x/verifier/data
    +
    +# Compile circuit
    +cd plonky2x/verifier
    +mkdir verifier-build
    +go run . -compile -data verifier-build -circuit data/dummy
    +
    +# Compile executable
    +CGO_ENABLED=0 go build -o verifier-build/verifier -ldflags "-s -w" .
    +
    +
    +ls verifier-build/
    +# ^ should have: pk.bin  r1cs.bin  verifier  Verifier.sol  vk.bin

    This should generate the Verifier.sol contract which is a template contract used to generate the final function verifiers based on the plonky2x circuits.

    In the generated contract, only the circuit digest is specific to each circuit; the rest of the logic is universal. This is why a random circuit was used to create the template, indicated by the flag -circuit data/dummy. When we build a specific circuit, the Verifier.sol contract is fed to the circuit builder, which then replaces the circuit digest with the one corresponding to the actual circuit being built.

    Also, you will see the verifier binary which is the PLONK wrapper that takes a plonky2x proof and wraps it into a PLONK proof.

    During the build, the CRS string gets downloaded:

    bash
    2024/04/15 19:35:00 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript04.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript04.dat
    +2024/04/15 19:35:05 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript05.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript05.dat
    +2024/04/15 19:35:11 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript06.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript06.dat
    +2024/04/15 19:35:41 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript11.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript11.dat
    +2024/04/15 19:35:47 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript12.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript12.dat
    +2024/04/15 19:35:53 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript13.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript13.dat
    +2024/04/15 19:35:58 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript14.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript14.dat
    +2024/04/15 19:36:03 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript15.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript15.dat
    2024/04/15 19:35:00 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript04.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript04.dat
    +2024/04/15 19:35:05 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript05.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript05.dat
    +2024/04/15 19:35:11 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript06.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript06.dat
    +2024/04/15 19:35:41 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript11.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript11.dat
    +2024/04/15 19:35:47 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript12.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript12.dat
    +2024/04/15 19:35:53 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript13.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript13.dat
    +2024/04/15 19:35:58 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript14.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript14.dat
    +2024/04/15 19:36:03 download https://aztec-ignition.s3.amazonaws.com/MAIN%20IGNITION/196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript15.dat 196_0x9c3f372a2aacc0d7272f56a4664311b7647b0031/transcript15.dat

    This randomness is taken from the Aztec trusted setup.

    NOTE: In a scaleway instance of 64CPU and 504G RAM, the build takes ~20 minutes to complete. Make sure to run it on a beefy machine.

    Build the circuits and function verifiers

    To build the circuits:

    bash
    cd $HOME
    +git clone https://github.com/succinctlabs/blobstreamx
    +cd blobstreamx
    +# if a specific version is needed: git checkout <commit>
    +# this assumes the verifier-build was generated in the previous section.
    +# if not, it can be downloaded as specified in the first section.
    +cp -r ../succinctx/plonky2x/verifier/verifier-build .
    +# ^ This copies the verifier build to the blobstreamx folder
    +mkdir -p build && RUST_LOG=debug cargo run --bin header_range_1024 --release build --wrapper-path verifier-build/ && mv ./target/release/header_range_1024 ./build/header_range_1024
    +ls build
    +# ^ should return something like: 0x007d0b2a2e2b013612e8.circuit  0x9039e58b2089e5f9abbb.circuit  0xce1636cfaf2bd5497c11.circuit  FunctionVerifier.sol  main.circuit 0x8e1ede4ce0865b41d714.circuit  0xa2140c9bde000dc5e21e.circuit  0xf6759ff933786ddacb92.circuit  header_range_1024
    cd $HOME
    +git clone https://github.com/succinctlabs/blobstreamx
    +cd blobstreamx
    +# if a specific version is needed: git checkout <commit>
    +# this assumes the verifier-build was generated in the previous section.
    +# if not, it can be downloaded as specified in the first section.
    +cp -r ../succinctx/plonky2x/verifier/verifier-build .
    +# ^ This copies the verifier build to the blobstreamx folder
    +mkdir -p build && RUST_LOG=debug cargo run --bin header_range_1024 --release build --wrapper-path verifier-build/ && mv ./target/release/header_range_1024 ./build/header_range_1024
    +ls build
    +# ^ should return something like: 0x007d0b2a2e2b013612e8.circuit  0x9039e58b2089e5f9abbb.circuit  0xce1636cfaf2bd5497c11.circuit  FunctionVerifier.sol  main.circuit 0x8e1ede4ce0865b41d714.circuit  0xa2140c9bde000dc5e21e.circuit  0xf6759ff933786ddacb92.circuit  header_range_1024

    The header_range_1024 is a specific circuit. Other circuit names can be used there. The current circuits that we have for BlobstreamX:

    • header_range_1024: skip function circuit for batches that are <= 1024 block.
    • header_range_2048: skip function circuit for batches that are <= 2048 block.
    • next_header: step function circuit.

    All the deployments currently rely on two circuits: a header range circuit, either the 1024 or the 2048, depending on the frequency of the batches; The 1024 is mainly for batches that are at a ~1hr frequency and the 2048 for batches that are at a ~3-4hr frequency; and a next header circuit. So, if you're re-building the circuits, make sure to build the correct two circuits for your target deployment.

    Now, if you check the build folder, you will find a file called FunctionVerifier.sol, which is the function verifier contract for your circuit that you can deploy on-chain.

    Also, you will find the header_range_1024 under build folder which is the binary used to generate the proofs. It is mainly used by the operator to generate the plonky2x proofs that will be PLONK wrapped later using the generated verifier-build.

    At this level, you can deploy the FunctionVerifier.sol onchain, then register it in the SuccinctGateway, then use the generated header_range_1024 circuit and the verifier to generate the proofs and submit them onchain. These steps are detailed in the previous section.

    NOTE: In a scaleway instance of 64CPU and 504G RAM, the build takes ~10 minutes to complete. Make sure to run it on a beefy machine.

    + + + + \ No newline at end of file diff --git a/developers/blobstream-x-requesting-data-commitment-ranges.html b/developers/blobstream-x-requesting-data-commitment-ranges.html new file mode 100644 index 00000000000..7c0c9929562 --- /dev/null +++ b/developers/blobstream-x-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 updated 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:

    NOTE: The requested proof ranges cannot include blocks that were already used in a previous batch. The ranges should start from the last proven block, aka, latest_block and they should end in a block already committed by Celestia. In other words, it's the end-inclusive range defined by [latest_block, target_block] with target_block <= Celestia tip.

    Local proving

    To run the Blobstream X operator with local proving, follow this guide.

    Local proving allows self-generating the proofs and submitting them to an existing BlobstreamX contract. Alternatively, if a team needs a very specific cadence that starts at very specific heights, they can deploy their own BlobstreamX contract and submit proofs to it. Deployment instructions can be found in the BlobstreamX deploy documentation.

    TIP

    Requires a large cloud machine to run in a reasonable amount of time. EC2 r6a.16xlarge, i.e., 64CPU 512GB RAM, takes ~30 minutes to generate a header range proof.

    Request proofs from the Succinct platform

    NOTE: Requesting a proof from the succinct platform requires having a Succinct API key. It can be requested using this form.

    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.

    Request proofs 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/blobstream.html b/developers/blobstream.html new file mode 100644 index 00000000000..f2b308c116f --- /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.

    The latest implementation of Blobstream X is SP1 Blobstream, which is written in Rust for the SP1 zkVM. SP1 Blobstream offers improved performance and efficiency while maintaining the security guarantees of the original Blobstream X.

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

    Implementations of Blobstream

    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 SP1 Blobstream?

    SP1 Blobstream is the latest implementation of Blobstream in Rust using the SP1 zkVM.

    SP1 Blobstream is the latest 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. SP1 Blobstream uses 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 SP1 Blobstream Ethereum smart contract to update it with the latest Celestia header.

    The SP1 Blobstream 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.

    If you're looking to deploy SP1 blobstream to a new chain, see new Sp1 Blobstream deployments.

    Learn more at the sp1-blobstream repo.

    NOTE

    The current Blobstream deployments all use SP1 Blobstream.

    Integrate with SP1 Blobstream

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

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

    The first deployments of SP1 Blobstream 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 contract that will include a new data commitment range that covers a 1-hour block range from the latestBlock in the contract. On Ethereum Mainnet, the contract will be updated every 4 hours.

    How to integrate with Blobstream

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

    Blobstream rollups

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

    Deployed contracts

    You can interact with the SP1 Blobstream contracts today. The SP1 Blobstream Solidity smart contracts are currently deployed on the following chains:

    ContractEVM networkContract addressAttested data on CelestiaLink to Celenium
    SP1 BlobstreamEthereum Mainnet0x7Cf3876F681Dbb6EdA8f6FfC45D66B996Df08fAeMainnet BetaDeployment on Celenium
    SP1 BlobstreamArbitrum One0xA83ca7775Bc2889825BcDeDfFa5b758cf69e8794Mainnet BetaDeployment on Celenium
    SP1 BlobstreamBase0xA83ca7775Bc2889825BcDeDfFa5b758cf69e8794Mainnet BetaDeployment on Celenium
    SP1 BlobstreamSepolia0xf0c6429ebab2e7dc6e05dafb61128be21f13cb1eMocha testnetDeployment on Celenium
    SP1 BlobstreamArbitrum Sepolia0xc3e209eb245Fd59c8586777b499d6A665DF3ABD2Mocha testnetDeployment on Celenium
    SP1 BlobstreamBase Sepolia0xc3e209eb245Fd59c8586777b499d6A665DF3ABD2Mocha testnetDeployment on Celenium
    + + + + \ No newline at end of file diff --git a/developers/blobstreamx.html b/developers/blobstreamx.html new file mode 100644 index 00000000000..e9165039525 --- /dev/null +++ b/developers/blobstreamx.html @@ -0,0 +1,45 @@ + + + + + + Blobstream X: the previous zk implementation of Blobstream | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Blobstream X: the previous zk implementation of Blobstream

    blobstream x draft diagram

    What is Blobstream X?

    Blobstream X is the previous implementation of Blobstream. It uses plonky2x to create circuits that verify the Celestia consensus and generate the corresponding proofs.

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

    NOTE

    The Blobstream deployments below don't use the BlobstreamX circuits.

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

    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.

    Deploy Blobstream X

    It is possible to deploy and maintain a Blobstream x instance and have the same security guarantees.

    First, you will need to create a multisig that governs the Blobstream X contract and also the function identifiers. The function identifiers can be registered in the Succinct gateway.

    Then, check the deployment documentation for how to deploy the contract.

    Then, you will need to run a relayer, which will generate the proofs and relay them to your deployed Blobstream X contract. Check the local proving documentation for more information.

    + + + + \ No newline at end of file diff --git a/developers/bubs-testnet.html b/developers/bubs-testnet.html new file mode 100644 index 00000000000..fe55d4d5b33 --- /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 the first OP Stack testnet with Celestia underneath hosted by Caldera with support from Celestia Labs. Bubs 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. Bubs' data is posted to Celestia on the Mocha testnet. View the namespace for Bubs on Celestia's Mocha testnet.

    Learn more about the setup of the integration in the introduction.

    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..2e998cfee54 --- /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..a7ccde453b7 --- /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:

    • --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 default directory under /keys so you should make sure to point cel-key utility to the correct directory via the p2p.network or home flags if you have specified a custom directory or network other than Arabica, Mocha, or Mainnet Beta.

    Also keep in mind that if you do not specify a network with --p2p.network, the default one will always be celestia (Mainnet Beta).

    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.16.0 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.16.0 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.keyname 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.keyname 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/ethereum-fallback.html b/developers/ethereum-fallback.html new file mode 100644 index 00000000000..65b51628892 --- /dev/null +++ b/developers/ethereum-fallback.html @@ -0,0 +1,45 @@ + + + + + + Ethereum fallback | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Ethereum fallback

    Ethereum fallback is a 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/feegrant-for-blobs.html b/developers/feegrant-for-blobs.html new file mode 100644 index 00000000000..5f4f5bca4b0 --- /dev/null +++ b/developers/feegrant-for-blobs.html @@ -0,0 +1,107 @@ + + + + + + FeeGrant module for blobs submission | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    FeeGrant module for blobs submission

    Overview

    This guide provides developers with the knowledge to use the FeeGrant module on the Celestia's Mocha testnet chain for granting a data availability node's account to submit blobs without constantly funding it, enabling a third-party account to cover the transaction fees.

    Pre-requisites

    • celestia-node binary (celestia) installed
    • Access to a Mocha node (e.g., https://rpc.celestia-mocha.com:443)
    • Running DA Light node on Mocha testnet
    • One account with sufficient funds, the "granter"
    • One account with no funds, the "grantee"

    Introduction

    Each DA node contains a Celestia account that is used to pay for blobs submissions. To unify the fee payment process, the FeeGrant module allows a third-party account (granter) to pay for the fees incurred by a DA node's (grantee) account. You will need one account that will contain the funds, the granter, and another account that will be in the DA node you run to post blobs, the grantee. You will see the DA node's account once you initialize the node. Learn more about managing accounts with cel-key in create a wallet with celestia-node.

    Granting fee allowances using celestia-node

    To get started granting the fee allowance, you will need two separate keys to run the light node with. One to begin the FeeGrant as the granter and another to use the FeeGrant as the grantee.

    Set some variables for your accounts for the remainder of the guide:

    bash
    export GRANTER_ADDRESS=<your-granter-account-address>
    +export GRANTEE_ADDRESS=<your-grantee-account-address>
    +export RPC_URL=rpc.celestia-mocha.com
    export GRANTER_ADDRESS=<your-granter-account-address>
    +export GRANTEE_ADDRESS=<your-grantee-account-address>
    +export RPC_URL=rpc.celestia-mocha.com

    FeeGrant module implementation in celestia-node

    Using celestia-node, you can now easily give permission for other nodes to submit transactions on your behalf. It is also possible to revoke the grant.

    The FeeGrant functionality can now be used during runtime without the need to restart the node.

    Grant permission for an allowance as a granter

    First, start your node:

    bash
    celestia light start --p2p.network mocha --core.ip $RPC_URL
    celestia light start --p2p.network mocha --core.ip $RPC_URL

    Then, grant the fee to the grantee:

    bash
    celestia state grant-fee $GRANTEE_ADDRESS --amount 2000
    celestia state grant-fee $GRANTEE_ADDRESS --amount 2000

    Note that the --amount flag specifies the spend limit (in utia) for the grantee. If not specified, the grantee does not have a spend limit.

    Using a FeeGrant allowance as a grantee in celestia-node

    Start your node:

    bash
    celestia light start --core.ip $RPC_URL --p2p.network=mocha
    celestia light start --core.ip $RPC_URL --p2p.network=mocha

    To check the balance of a light node, use the following command:

    bash
    celestia state balance
    celestia state balance

    Example response when the account balance does not exist:

    json
    {
    +  "result": {
    +    "denom": "utia",
    +    "amount": "0"
    +  }
    +}
    {
    +  "result": {
    +    "denom": "utia",
    +    "amount": "0"
    +  }
    +}

    This indicates that the light node currently does not have any funds.

    Now submit a blob using the FeeGrant:

    bash
    celestia blob submit 0x42690c204d39600fddd3 'gm' --granter.address $GRANTER_ADDRESS
    celestia blob submit 0x42690c204d39600fddd3 'gm' --granter.address $GRANTER_ADDRESS

    You'll see the height and the commitment of your blob:

    json
    {
    +  "result": {
    +    "height": 1639397,
    +    "commitments": [
    +      "19L/C4iBEsqXGzC5ZxJ3vtuGBiAdQAMIEnbYjKEGcac="
    +    ]
    +  }
    +}
    {
    +  "result": {
    +    "height": 1639397,
    +    "commitments": [
    +      "19L/C4iBEsqXGzC5ZxJ3vtuGBiAdQAMIEnbYjKEGcac="
    +    ]
    +  }
    +}

    Checking account balances after submission

    Light node account: After submitting a blob, you can check the light node account's balance to verify that the fees have been deducted:

    bash
    celestia state balance
    celestia state balance

    Example output showing fees are not deducted:

    json
    {
    +  "result": {
    +    "denom": "utia",
    +    "amount": "0"
    +  }
    +}
    {
    +  "result": {
    +    "denom": "utia",
    +    "amount": "0"
    +  }
    +}

    Optional: Revoke permission for a FeeGrant allowance as a granter

    To revoke the feegrant, run:

    bash
    celestia state revoke-grant-fee $GRANTEE_ADDRESS
    celestia state revoke-grant-fee $GRANTEE_ADDRESS

    Optional: Submitting a blob from file input

    To submit a blob from file input:

    bash
    celestia blob submit --input-file blob.json
    celestia blob submit --input-file blob.json

    Optional: Granting fee allowances using celestia-appd

    To grant fee allowances, allowing a third-party (granter) account to pay for the fees incurred by a Celestia data availability node (grantee) account, use the following commands.

    Set your account addresses for grantee and granter, and the RPC URL:

    bash
    export GRANTER_ADDRESS=<your-granter-account-address>
    +export GRANTEE_ADDRESS=<your-grantee-account-address>
    +export RPC_URL=https://rpc.celestia-mocha.com:443
    export GRANTER_ADDRESS=<your-granter-account-address>
    +export GRANTEE_ADDRESS=<your-grantee-account-address>
    +export RPC_URL=https://rpc.celestia-mocha.com:443

    Then, send the feegrant transaction:

    bash
    celestia-appd tx feegrant grant \
    +  $GRANTER_ADDRESS $GRANTEE_ADDRESS \
    +  --node $RPC_URL \
    +  --spend-limit 1000000utia \
    +  --allowed-messages "/cosmos.bank.v1beta1.MsgSend,/celestia.blob.v1.MsgPayForBlobs" \
    +  --chain-id mocha-4 \
    +  --keyring-backend test \
    +  --fees 20000utia \
    +  --broadcast-mode block \
    +  --yes
    celestia-appd tx feegrant grant \
    +  $GRANTER_ADDRESS $GRANTEE_ADDRESS \
    +  --node $RPC_URL \
    +  --spend-limit 1000000utia \
    +  --allowed-messages "/cosmos.bank.v1beta1.MsgSend,/celestia.blob.v1.MsgPayForBlobs" \
    +  --chain-id mocha-4 \
    +  --keyring-backend test \
    +  --fees 20000utia \
    +  --broadcast-mode block \
    +  --yes

    Example: FeeGrant transaction on Mocha

    Optional: Checking the granter's account

    To confirm that the fees have been deducted from the granter's account that granted the fee allowance, run:

    bash
    celestia-appd query bank balances $GRANTER_ADDRESS \
    +--node https://rpc.celestia-mocha.com:443 --denom utia
    celestia-appd query bank balances $GRANTER_ADDRESS \
    +--node https://rpc.celestia-mocha.com:443 --denom utia

    This output will show the remaining balance after fees have been deducted, confirming that the FeeGrant module is working as intended.

    + + + + \ No newline at end of file diff --git a/developers/golang-client-tutorial.html b/developers/golang-client-tutorial.html new file mode 100644 index 00000000000..c65efaa8a7c --- /dev/null +++ b/developers/golang-client-tutorial.html @@ -0,0 +1,283 @@ + + + + + + Golang client library tutorial | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Golang client library tutorial

    This section tutorial will guide you through using the most common RPC endpoints with the golang client library.

    Install dependencies and celestia-node if you have not already.

    Project setup

    To start, add celestia-openrpc as a dependency to your project:

    bash
    go get github.com/celestiaorg/celestia-openrpc
    go get github.com/celestiaorg/celestia-openrpc

    To use the following methods, you will need the node URL and your auth token. To get your auth token, see this guide. To run your node without an auth token, you can use the --rpc.skip-auth flag when starting your node. This allows you to pass an empty string as your auth token.

    The default URL is http://localhost:26658. If you would like to use subscription methods, such as SubscribeHeaders below, you must use the ws protocol in place of http: ws://localhost:26658.

    Submitting and retrieving blobs

    The blob.Submit method takes a slice of blobs and a gas price, returning the height the blob was successfully posted at.

    • The namespace can be generated with share.NewBlobNamespaceV0.
    • The blobs can be generated with blob.NewBlobV0.
    • You can use blob.NewSubmitOptions(), which has celestia-node automatically determine an appropriate gas price. To set your own gas price, use blob.NewSubmitOptions().WithGasPrice(X). The available options are WithGasPrice, WithGas, WithKeyName, WithSignerAddress, and WithFeeGranterAddress.

    The blob.GetAll method takes a height and slice of namespaces, returning the slice of blobs found in the given namespaces.

    go
    import (
    +	"bytes"
    +	"context"
    +	"fmt"
    +
    +	client "github.com/celestiaorg/celestia-openrpc"
    +	"github.com/celestiaorg/celestia-openrpc/types/blob"
    +	"github.com/celestiaorg/celestia-openrpc/types/share"
    +)
    +
    +// SubmitBlob submits a blob containing "Hello, World!" to the 0xDEADBEEF namespace. It uses the default signer on the running node.
    +func SubmitBlob(ctx context.Context, url string, token string) error {
    +	client, err := client.NewClient(ctx, url, token)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// let's post to 0xDEADBEEF namespace
    +	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
    +	if err != nil {
    +		return err
    +	}
    +
    +	// create a blob
    +	helloWorldBlob, err := blob.NewBlobV0(namespace, []byte("Hello, World!"))
    +	if err != nil {
    +		return err
    +	}
    +
    +	// submit the blob to the network
    +	height, err := client.Blob.Submit(ctx, []*blob.Blob{helloWorldBlob}, blob.NewSubmitOptions())
    +	if err != nil {
    +		return err
    +	}
    +
    +	fmt.Printf("Blob was included at height %d\n", height)
    +
    +	// fetch the blob back from the network
    +	retrievedBlobs, err := client.Blob.GetAll(ctx, height, []share.Namespace{namespace})
    +	if err != nil {
    +		return err
    +	}
    +
    +	fmt.Printf("Blobs are equal? %v\n", bytes.Equal(helloWorldBlob.Commitment, retrievedBlobs[0].Commitment))
    +	return nil
    +}
    import (
    +	"bytes"
    +	"context"
    +	"fmt"
    +
    +	client "github.com/celestiaorg/celestia-openrpc"
    +	"github.com/celestiaorg/celestia-openrpc/types/blob"
    +	"github.com/celestiaorg/celestia-openrpc/types/share"
    +)
    +
    +// SubmitBlob submits a blob containing "Hello, World!" to the 0xDEADBEEF namespace. It uses the default signer on the running node.
    +func SubmitBlob(ctx context.Context, url string, token string) error {
    +	client, err := client.NewClient(ctx, url, token)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// let's post to 0xDEADBEEF namespace
    +	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
    +	if err != nil {
    +		return err
    +	}
    +
    +	// create a blob
    +	helloWorldBlob, err := blob.NewBlobV0(namespace, []byte("Hello, World!"))
    +	if err != nil {
    +		return err
    +	}
    +
    +	// submit the blob to the network
    +	height, err := client.Blob.Submit(ctx, []*blob.Blob{helloWorldBlob}, blob.NewSubmitOptions())
    +	if err != nil {
    +		return err
    +	}
    +
    +	fmt.Printf("Blob was included at height %d\n", height)
    +
    +	// fetch the blob back from the network
    +	retrievedBlobs, err := client.Blob.GetAll(ctx, height, []share.Namespace{namespace})
    +	if err != nil {
    +		return err
    +	}
    +
    +	fmt.Printf("Blobs are equal? %v\n", bytes.Equal(helloWorldBlob.Commitment, retrievedBlobs[0].Commitment))
    +	return nil
    +}

    Subscribing to new blobs

    You can subscribe to new blobs in a namespace using the blob.Subscribe method. This method returns a channel that will receive new blobs as they are produced. In this example, we will fetch all blobs in the 0xDEADBEEF namespace.

    go
    func SubscribeBlobs(ctx context.Context, url string, token string) error {
    +	client, err := client.NewClient(ctx, url, token)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// create a namespace to filter blobs with
    +	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
    +	if err != nil {
    +		return err
    +	}
    +
    +	// subscribe to new blobs using a <-chan *blob.BlobResponse channel
    +	blobChan, err := client.Blob.Subscribe(ctx)
    +	if err != nil {
    +		return err
    +	}
    +
    +	for {
    +		select {
    +		case resp := <-blobChan:
    +			fmt.Printf("Found %d blobs at height %d in 0xDEADBEEF namespace\n", len(resp.Blobs()), resp.Height)
    +		case <-ctx.Done():
    +			return nil
    +		}
    +	}
    +}
    func SubscribeBlobs(ctx context.Context, url string, token string) error {
    +	client, err := client.NewClient(ctx, url, token)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// create a namespace to filter blobs with
    +	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
    +	if err != nil {
    +		return err
    +	}
    +
    +	// subscribe to new blobs using a <-chan *blob.BlobResponse channel
    +	blobChan, err := client.Blob.Subscribe(ctx)
    +	if err != nil {
    +		return err
    +	}
    +
    +	for {
    +		select {
    +		case resp := <-blobChan:
    +			fmt.Printf("Found %d blobs at height %d in 0xDEADBEEF namespace\n", len(resp.Blobs()), resp.Height)
    +		case <-ctx.Done():
    +			return nil
    +		}
    +	}
    +}

    Subscribing to new headers

    Alternatively, you can subscribe to new headers using the header.Subscribe method. This method returns a channel that will receive new headers as they are produced. In this example, we will fetch all blobs at the height of the new header in the 0xDEADBEEF namespace.

    go
    // SubscribeHeaders subscribes to new headers and fetches all blobs at the height of the new header in the 0xDEADBEEF namespace.
    +func SubscribeHeaders(ctx context.Context, url string, token string) error {
    +	client, err := client.NewClient(ctx, url, token)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// create a namespace to filter blobs with
    +	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
    +	if err != nil {
    +		return err
    +	}
    +
    +	// subscribe to new headers using a <-chan *header.ExtendedHeader channel
    +	headerChan, err := client.Header.Subscribe(ctx)
    +	if err != nil {
    +		return err
    +	}
    +
    +	for {
    +		select {
    +		case header := <-headerChan:
    +			// fetch all blobs at the height of the new header
    +			blobs, err := client.Blob.GetAll(context.TODO(), header.Height(), []share.Namespace{namespace})
    +			if err != nil {
    +				fmt.Printf("Error fetching blobs: %v\n", err)
    +			}
    +
    +			fmt.Printf("Found %d blobs at height %d in 0xDEADBEEF namespace\n", len(blobs), header.Height())
    +		case <-ctx.Done():
    +			return nil
    +		}
    +	}
    +}
    // SubscribeHeaders subscribes to new headers and fetches all blobs at the height of the new header in the 0xDEADBEEF namespace.
    +func SubscribeHeaders(ctx context.Context, url string, token string) error {
    +	client, err := client.NewClient(ctx, url, token)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// create a namespace to filter blobs with
    +	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
    +	if err != nil {
    +		return err
    +	}
    +
    +	// subscribe to new headers using a <-chan *header.ExtendedHeader channel
    +	headerChan, err := client.Header.Subscribe(ctx)
    +	if err != nil {
    +		return err
    +	}
    +
    +	for {
    +		select {
    +		case header := <-headerChan:
    +			// fetch all blobs at the height of the new header
    +			blobs, err := client.Blob.GetAll(context.TODO(), header.Height(), []share.Namespace{namespace})
    +			if err != nil {
    +				fmt.Printf("Error fetching blobs: %v\n", err)
    +			}
    +
    +			fmt.Printf("Found %d blobs at height %d in 0xDEADBEEF namespace\n", len(blobs), header.Height())
    +		case <-ctx.Done():
    +			return nil
    +		}
    +	}
    +}

    Fetching an Extended Data Square (EDS)

    You can fetch an Extended Data Square (EDS) using the share.GetEDS method. This method takes a header and returns the EDS at the given height.

    go
    // GetEDS fetches the EDS at the given height.
    +func GetEDS(ctx context.Context, url string, token string, height uint64) (*rsmt2d.ExtendedDataSquare, error) {
    +	client, err := client.NewClient(ctx, url, token)
    +	if err != nil {
    +		return nil, err
    +	}
    +
    +	// First get the header of the block you want to fetch the EDS from
    +	header, err := client.Header.GetByHeight(ctx, height)
    +	if err != nil {
    +		return nil, err
    +	}
    +
    +	// Fetch the EDS
    +	return client.Share.GetEDS(ctx, header)
    +}
    // GetEDS fetches the EDS at the given height.
    +func GetEDS(ctx context.Context, url string, token string, height uint64) (*rsmt2d.ExtendedDataSquare, error) {
    +	client, err := client.NewClient(ctx, url, token)
    +	if err != nil {
    +		return nil, err
    +	}
    +
    +	// First get the header of the block you want to fetch the EDS from
    +	header, err := client.Header.GetByHeight(ctx, height)
    +	if err != nil {
    +		return nil, err
    +	}
    +
    +	// Fetch the EDS
    +	return client.Share.GetEDS(ctx, header)
    +}

    API documentation

    To see the full list of available methods, see the API documentation.

    + + + + \ No newline at end of file diff --git a/developers/integrate-celestia.html b/developers/integrate-celestia.html new file mode 100644 index 00000000000..9f67beac1c1 --- /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..b10e126bb59 --- /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 availability (DA) network, built to enable anyone to easily deploy their own blockchain with minimal overhead.

    Together, they allow developers to create rollups that post data to Celestia and settle on Ethereum.

    About the integration

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

    The integration of OP Stack with Celestia underneath for DA allows rollup operators to reduce overhead that is associated with posting data as calldata on Ethereum. Instead, op-batcher batches up rollup blocks and posts them to Celestia's DA network.

    Data is managed 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. While previously op-node was reading from calldata on Ethereum, it now 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.

    GitHub repository

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

    WARNING

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

    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/multiaccounts.html b/developers/multiaccounts.html new file mode 100644 index 00000000000..0a5707c446a --- /dev/null +++ b/developers/multiaccounts.html @@ -0,0 +1,49 @@ + + + + + + MultiAccounts feature for blobs submission | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    MultiAccounts feature for blobs submission

    Overview

    By default, a celestia-node creates a key named my_celes_key during initialization. This document explains how to run a node with a different default key name and how to submit blobs using different signers.

    Running a node with a different default key name

    To start a Celestia node with a different default key name, use the following command:

    sh
    celestia light start --core.ip=consensus.celestia-arabica-11.com \
    +    --p2p.network=arabica --keyring.keyname testKey
    celestia light start --core.ip=consensus.celestia-arabica-11.com \
    +    --p2p.network=arabica --keyring.keyname testKey

    In this example, testKey becomes the default node key, and the node's address will change accordingly.

    Submitting blobs with a different signer/key name

    Option 1: Submit passing key name

    You can submit a blob by specifying a different key name:

    sh
    celestia blob submit 0x42690c204d39600fddd3 'gm' --key.name testKey2
    celestia blob submit 0x42690c204d39600fddd3 'gm' --key.name testKey2

    This transaction will be signed by the address associated with testKey2.

    Option 2: Submit passing signer address

    Alternatively, you can submit a blob by specifying the signer's address:

    sh
    celestia blob submit 0x42690c204d39600fddd3 'gm' --signer $SIGNER_ADDRESS
    celestia blob submit 0x42690c204d39600fddd3 'gm' --signer $SIGNER_ADDRESS

    Both options achieve the same result but use different inputs. The testKey2 points to SIGNER_ADDRESS in the KeyStore.

    Key management

    All keys and addresses must be added to the KeyStore. To create a new key, use the cel-key library:

    Creating a new key

    sh
    ./cel-key add testKey --keyring-backend test \
    +    --node.type light --p2p.network arabica
    ./cel-key add testKey --keyring-backend test \
    +    --node.type light --p2p.network arabica

    Importing an existing key

    sh
    ./cel-key import
    ./cel-key import

    Learn more on the Create a wallet with celestia-node page.

    Optional flags for write transactions

    All other flags are now optional for all write transactions. This means you don't have to specify gas/fee parameters each time. The configuration can handle it for you automatically.

    The default configuration applies to all write transactions, including those in the state module and blob.Submit. This simplifies the process of submitting transactions and reduces the need for manual input.

    For reference, see the:

    + + + + \ No newline at end of file diff --git a/developers/node-api.html b/developers/node-api.html new file mode 100644 index 00000000000..468502a58f8 --- /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..c1403a914cb --- /dev/null +++ b/developers/node-tutorial.html @@ -0,0 +1,442 @@ + + + + + + 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

    Install dependencies and celestia-node if you have not already.

    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 core endpoint

    Let's now run the Celestia Light node with a gRPC connection to an example core endpoint. Connecting to a core endpoint provides the light node with access to state queries (reading balances, submitting transactions, and other state-related queries).

    Note: You are also encouraged to find an RPC endpoint for Mainnet Beta, Mocha testnet, or Arabica devnet. If you are running a production application, use a production endpoint.

    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.keyname <key-name>
    celestia light start --core.ip <URI> --keyring.keyname <key-name>
    bash
    celestia light start --core.ip <URI> --keyring.keyname <key-name> \
    +  --p2p.network mocha
    celestia light start --core.ip <URI> --keyring.keyname <key-name> \
    +  --p2p.network mocha
    bash
    celestia light start --core.ip <URI> --keyring.keyname <key-name> \
    +  --p2p.network arabica
    celestia light start --core.ip <URI> --keyring.keyname <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:

    :::note Previously, the node.store flag had to be specified manually for each request. This has changed in v0.14.0+ and you can read more about the implementation in celestia-node troubleshooting. :::

    bash
    celestia blob submit 0x42690c204d39600fddd3 'gm'
    celestia blob submit 0x42690c204d39600fddd3 'gm'

    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; however, none are necessary using default configurations.

    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. This is only required if you are using a non-default node store path.

    To set a custom node store for a light node on mocha-4, you can use the following command:

    bash
    export NODE_STORE=$HOME/your-custom-path/celestia-light-mocha-4
    export NODE_STORE=$HOME/your-custom-path/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 with a custom node store path, you will need to set the location of the node store in your auth command.

    bash
    --node.store $HOME/your-custom-path/.celestia-light-private
    --node.store $HOME/your-custom-path/.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/your-custom-path/.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/your-custom-path/.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] [optional: node store | auth token]
    celestia blob submit <hex-encoded namespace> <hex-encoded data> \
    +  [optional: fee] [optional: gasLimit] [optional: node store | auth token]

    We run the following to submit a blob to the network in hexadecimal format:

    bash
    celestia blob submit 0x42690c204d39600fddd3 0x676d \
    celestia blob submit 0x42690c204d39600fddd3 0x676d \

    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'
    celestia blob submit 0x42690c204d39600fddd3 'gm'

    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=
    celestia blob get 252614 0x42690c204d39600fddd3 IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E=

    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
    celestia blob get 252614 0x42690c204d39600fddd3 IXg+08HV5RsPF3Lle8PH+B2TUGsGUsBiseflxh6wB5E= \
    +  --base64=TRUE

    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
    celestia blob get-all 252614 0x42690c204d39600fddd3

    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
    celestia blob get-all 252614 0x42690c204d39600fddd3 \
    +  --base64=TRUE

    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
    celestia blob submit 0x42690c204d39600fddd3 'gm' --gas.price 0.004

    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
    celestia state balance

    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>
    celestia state balance-for-address <address>

    Let's query our node for the balance of another address:

    bash
    celestia state balance-for-address celestia10rtd9lhel2cuh6c659l25yncl6atcyt37umard
    celestia state balance-for-address celestia10rtd9lhel2cuh6c659l25yncl6atcyt37umard

    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
    celestia p2p info

    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
    celestia state account-address

    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>
    celestia header get-by-height <height>

    Now, let's get the block header information.

    Here we will get the header from Block 1:

    bash
    celestia header get-by-height 1
    +```
    +
    +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
    +```
    +
    +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 | jq '.result.dah' -r)" 0x42690c204d39600fddd3
    celestia share get-by-namespace "$(celestia header get-by-height 147105 | jq '.result.dah' -r)" 0x42690c204d39600fddd3

    Get data availability sampler stats

    bash
    celestia das sampling-stats
    celestia das sampling-stats

    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>
    celestia state transfer $ADDRESS <amount in utia> <gas fee in utia> <gas fee in utia>

    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
    celestia state transfer $ADDRESS 100000 8000 80000

    If you'd just like to return the transaction hash, you can use jq:

    bash
    celestia state transfer $ADDRESS 100000 8000 80000 | jq .result.txhash
    celestia state transfer $ADDRESS 100000 8000 80000 | jq .result.txhash

    API version

    To query your node's API version, you can use the following command:

    bash
    celestia node info
    celestia node info

    Help

    To get help and view the CLI menu, use the following command:

    bash
    celestia --help
    celestia --help

    To view the help menu for a specific method, use the following command:

    bash
    celestia <module> <method> --help
    celestia <module> <method> --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 | jq '.result.dah' -r)" \
    +  0x42690c204d39600fddd3
    celestia share get-by-namespace \
    +  "$(celestia header get-by-height 252614 | jq '.result.dah' -r)" \
    +  0x42690c204d39600fddd3

    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.

    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..f93377d686e --- /dev/null +++ b/developers/optimism-devnet.html @@ -0,0 +1,119 @@ + + + + + + Optimism devnet deep dive | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Optimism devnet deep dive

    This page is for those interested in doing a deep dive on their pre-op-plasma-celestia @celestiaorg/optimism rollups.

    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=3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f
    +export ENCODED_INPUT=$(echo "$INPUT" | xxd -r -p | base64)
    export INPUT=3500000000000000769074a923011bdda721eacc34c8a77c69c10f2b6c8e659f987e82f217a5340f
    +export ENCODED_INPUT=$(echo "$INPUT" | xxd -r -p | base64)

    TIP

    Remember to remove the 0xce prefix!

    Find the data on Celestia

    bash
    curl -s -X POST -H "Content-Type: application/json" -H "Authorization: Bearer $CELESTIA_NODE_AUTH_TOKEN" -d '{ "id": 1, "jsonrpc": "2.0", "method": "da.Get", "params": [["$ENCODED_INPUT"], "AAAAAAAAAAAAAAAAAAAAAAAAAAECAwQFBgcICRA="]}' http://127.0.0.1:26658
    curl -s -X POST -H "Content-Type: application/json" -H "Authorization: Bearer $CELESTIA_NODE_AUTH_TOKEN" -d '{ "id": 1, "jsonrpc": "2.0", "method": "da.Get", "params": [["$ENCODED_INPUT"], "AAAAAAAAAAAAAAAAAAAAAAAAAAECAwQFBgcICRA="]}' http://127.0.0.1:26658

    The params are []blobs, namespace, base64-encoded.

    Your result will look similar to the below!

    console
    {"jsonrpc":"2.0","result":["SGVsbG8gd28ybGQh"],"id":1}
    {"jsonrpc":"2.0","result":["SGVsbG8gd28ybGQh"],"id":1}

    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..3a147987801 --- /dev/null +++ b/developers/optimism.html @@ -0,0 +1,90 @@ + + + + + + Run an OP Stack rollup with Celestia underneath | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Run an OP Stack rollup with Celestia underneath

    This guide will show you how to run your own OP Stack devnet and testnet that posts data to Celestia's Mocha testnet using roll-op and op-plasma-celestia.

    If you don't have devops experience and would like to use a Rollups as a Service (RaaS) provider, see the RaaS category in the menu.

    Dependency setup

    Setting up your light node

    Sync and fund a Celestia light node. The light node must be fully synced and funded for you to be able to submit and retrieve PayForBlobs to Mocha Testnet. This allows your rollup to post and retrieve data without any errors.

    In order to mount existing data, you must have a node store that is in the default directory:

    bash
    $HOME/.celestia-light-mocha-4
    $HOME/.celestia-light-mocha-4
    bash
    $HOME/.celestia-light
    $HOME/.celestia-light
    bash
    $HOME/.celestia-light-arabica-11
    $HOME/.celestia-light-arabica-11

    By default, the node will run with the account named my_celes_key on Mocha. This is the account that needs to be funded.

    TIP

    Unless you changed your configuration, you won't have to change anything. 😎

    Deploying a devnet to Mocha

    See the Alt-DA x Celestia README for instructions on how to deploy a Devnet.

    TIP for macOS users

    If you are on macOS, you will need to run a venv before starting roll-op.

    sh
    cd $HOME/roll-op
    +python3 -m venv ./venv
    +source ./venv/bin/activate
    cd $HOME/roll-op
    +python3 -m venv ./venv
    +source ./venv/bin/activate

    Congrats! Your devnet is running on a mock EVM chain and Celestia Mocha.

    Deploying a testnet to an L1 (or L2) and Mocha

    See the Alt-DA x Celestia README for instructions on how to deploy a Testnet.

    TIP

    If you are using a public RPC for your EVM chain, you should to enable deploy_slowly = true in your config.toml. If you still have issues, we recommend running the integration with a high-availability, paid endpoint.

    When you are deploying to a live EVM network, pay attention and modify the configuration to post to non-Sepolia EVM chains.

    Here is an example:

    toml
    # Chain ID of your rollup
    +l2_chain_id = 1117733 
    +
    +# Sepolia Ethereum
    +l1_chain_id = 11155111
    +l1_rpc_url = "https://ethereum-sepolia-rpc.publicnode.com"
    +
    +## Avoid issues with public RPC
    +deploy_slowly = true
    +
    +## Keys
    +contract_deployer_account = "0xaddress"
    +contract_deployer_key = "privatekey"
    +batcher_account = "0xaddress"
    +batcher_key = "privatekey"
    +proposer_account = "0xaddress"
    +proposer_key = "privatekey"
    +admin_account = "0xaddress"
    +admin_key = "privatekey"
    +p2p_sequencer_account = "0xaddress"
    +p2p_sequencer_key = "privatekey"
    # Chain ID of your rollup
    +l2_chain_id = 1117733 
    +
    +# Sepolia Ethereum
    +l1_chain_id = 11155111
    +l1_rpc_url = "https://ethereum-sepolia-rpc.publicnode.com"
    +
    +## Avoid issues with public RPC
    +deploy_slowly = true
    +
    +## Keys
    +contract_deployer_account = "0xaddress"
    +contract_deployer_key = "privatekey"
    +batcher_account = "0xaddress"
    +batcher_key = "privatekey"
    +proposer_account = "0xaddress"
    +proposer_key = "privatekey"
    +admin_account = "0xaddress"
    +admin_key = "privatekey"
    +p2p_sequencer_account = "0xaddress"
    +p2p_sequencer_key = "privatekey"

    Your 0xaddress key must also be funded with testnet ETH. We recommend at least 10 SepoliaETH to get your chain started, but you will need more to keep it running longer.

    Congratulations

    Congrats! You now have an OP Stack rollup running with Celestia underneath.

    You can learn more about Alt-DA in Optimism docs.

    + + + + \ No newline at end of file diff --git a/developers/prompt-scavenger.html b/developers/prompt-scavenger.html new file mode 100644 index 00000000000..e4ba01d8e81 --- /dev/null +++ b/developers/prompt-scavenger.html @@ -0,0 +1,463 @@ + + + + + + Prompt scavenger | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Prompt scavenger

    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 start by initializing our light node and funding our account with some tokens. We will be using the Arabica testnet for this tutorial.

    sh
    celestia light init --p2p.network arabica
    celestia light init --p2p.network arabica

    You will see an output ending with something looking like this:

    2024-05-22T14:15:49.554+0200	INFO	node	nodebuilder/init.go:211	NO KEY FOUND IN STORE, GENERATING NEW KEY...
    +2024-05-22T14:15:49.564+0200	INFO	node	nodebuilder/init.go:216	NEW KEY GENERATED...
    +
    +NAME: my_celes_key
    +ADDRESS: celestia1hn25k7gkfq0fy5a0vmphs6mjma2de74gsn36ef
    +MNEMONIC (save this somewhere safe!!!):
    +**** **** **** ****
    2024-05-22T14:15:49.554+0200	INFO	node	nodebuilder/init.go:211	NO KEY FOUND IN STORE, GENERATING NEW KEY...
    +2024-05-22T14:15:49.564+0200	INFO	node	nodebuilder/init.go:216	NEW KEY GENERATED...
    +
    +NAME: my_celes_key
    +ADDRESS: celestia1hn25k7gkfq0fy5a0vmphs6mjma2de74gsn36ef
    +MNEMONIC (save this somewhere safe!!!):
    +**** **** **** ****

    To fund your account, copy the address from the "ADDRESS" log and paste it in the Arabica Faucet to request tokens.

    While waiting for our account to be funded, we can start our light node.

    sh
    celestia light start --core.ip validator-1.celestia-arabica-11.com --p2p.network arabica --rpc.skip-auth
    celestia light start --core.ip validator-1.celestia-arabica-11.com --p2p.network arabica --rpc.skip-auth

    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.

    We can now check if your account has been successfully funded by running the following command. If your balance is still 0, wait a few seconds and try again.

    sh
    celestia state balance --node.store ~/.celestia-light-arabica-11
    celestia state balance --node.store ~/.celestia-light-arabica-11

    TIP

    Make sure you run this command in a different terminal window because the node has to be running for it to work.

    OpenAI key

    Visit OpenAI to sign up for an account and generate an API key. order to sign up for an account and generate an OpenAI API key. The key will be needed to communicate with OpenAI.

    Once you have created an API key, set it as an environment variable with the following command, pasting in your own key:

    sh
    export OPENAI_KEY=sk-proj-xxxxxxxxxxxxxxxxxxxxxxxx
    export OPENAI_KEY=sk-proj-xxxxxxxxxxxxxxxxxxxxxxxx

    Building the Prompt Scavenger

    Initialize your Go project

    To initialize your go project, run the following commands:

    sh
    mkdir test_scavenger
    +cd test_scavenger
    +go mod init prompt-scavenger
    +go get github.com/celestiaorg/celestia-openrpc
    +go get github.com/sashabaranov/go-openai
    mkdir test_scavenger
    +cd test_scavenger
    +go mod init prompt-scavenger
    +go get github.com/celestiaorg/celestia-openrpc
    +go get github.com/sashabaranov/go-openai

    This will set up a go project in a new directory and download the required modules.

    Build your import statements

    Inside the directory, create a main.go file and setup the import statements:

    go
    package main
    +
    +import (
    +	"context"
    +	"encoding/hex"
    +	"fmt"
    +	"log"
    +	"os"
    +
    +	nodeclient "github.com/celestiaorg/celestia-openrpc"
    +	"github.com/celestiaorg/celestia-openrpc/types/blob"
    +	"github.com/celestiaorg/celestia-openrpc/types/share"
    +	 openai "github.com/sashabaranov/go-openai"
    +)
    +
    +func main() {
    +  // TODO:
    +	// - [ ] Load program arguments
    +	// - [ ] Initialize the node API client
    +	// - [ ] Create a namespace ID
    +	// - [ ] Create and submit a blob
    +	// - [ ] Retrieve the blob from the network
    +	// - [ ] Prompt chatgpt with the retrieved blob data
    +}
    package main
    +
    +import (
    +	"context"
    +	"encoding/hex"
    +	"fmt"
    +	"log"
    +	"os"
    +
    +	nodeclient "github.com/celestiaorg/celestia-openrpc"
    +	"github.com/celestiaorg/celestia-openrpc/types/blob"
    +	"github.com/celestiaorg/celestia-openrpc/types/share"
    +	 openai "github.com/sashabaranov/go-openai"
    +)
    +
    +func main() {
    +  // TODO:
    +	// - [ ] Load program arguments
    +	// - [ ] Initialize the node API client
    +	// - [ ] Create a namespace ID
    +	// - [ ] Create and submit a blob
    +	// - [ ] Retrieve the blob from the network
    +	// - [ ] Prompt chatgpt with the retrieved blob data
    +}

    Here we set up all required libraries we need to use plus the main function that we will use for our program. function that we will use for our program.

    TIP

    Depending on your IDE, unused import statements may be removed every time you save the file. If this is the case, come back to this section and add them one by one as they come up in the code snippets.

    Main function

    Let's start populating our main function. To begin, we need to load the arguments we pass to the program. and do some sanity checks. We will then initialize the node API client.

    go
    
    +func main() {
    +	ctx, cancel := context.WithCancel(context.Background())
    +	defer cancel()
    +
    +	// Get IP, namespace, and prompt from program arguments
    +	if len(os.Args) != 4 {
    +		log.Fatal("Usage: go run main.go <nodeIP> <namespace> <prompt>")
    +	}
    +	nodeIP, namespaceHex, prompt := os.Args[1], os.Args[2], os.Args[3]
    +
    +	// We pass an empty string as the jwt token, since we
    +	// disabled auth with the --rpc.skip-auth flag
    +	client, err := nodeclient.NewClient(ctx, nodeIP, "")
    +	if err != nil {
    +		log.Fatalf("Failed to create client: %v", err)
    +	}
    +	defer client.Close()
    +
    +	// TODO:
    +	// - [X] Load program arguments
    +	// - [X] Initialize the node API client
    +	// - [ ] Create a namespace ID
    +	// - [ ] Create and submit a blob
    +	// - [ ] Retrieve the blob from the network
    +	// - [ ] Prompt chatgpt with the retrieved blob data
    +}
    
    +func main() {
    +	ctx, cancel := context.WithCancel(context.Background())
    +	defer cancel()
    +
    +	// Get IP, namespace, and prompt from program arguments
    +	if len(os.Args) != 4 {
    +		log.Fatal("Usage: go run main.go <nodeIP> <namespace> <prompt>")
    +	}
    +	nodeIP, namespaceHex, prompt := os.Args[1], os.Args[2], os.Args[3]
    +
    +	// We pass an empty string as the jwt token, since we
    +	// disabled auth with the --rpc.skip-auth flag
    +	client, err := nodeclient.NewClient(ctx, nodeIP, "")
    +	if err != nil {
    +		log.Fatalf("Failed to create client: %v", err)
    +	}
    +	defer client.Close()
    +
    +	// TODO:
    +	// - [X] Load program arguments
    +	// - [X] Initialize the node API client
    +	// - [ ] Create a namespace ID
    +	// - [ ] Create and submit a blob
    +	// - [ ] Retrieve the blob from the network
    +	// - [ ] Prompt chatgpt with the retrieved blob data
    +}

    Next, we need to create some utility functions that will help us with our next TODO items.

    Utility functions

    First, we need a function to convert a hex string to a NamespaceID type that is used for blob creation. This is needed because the namespace we pass in the program arguments will be in hexadecimal format.

    go
    // createNamespaceID converts a hex string to a NamespaceID
    +func createNamespaceID(nIDString string) (share.Namespace, error) {
    +	// First, we parse the passed hex string into a []byte slice
    +	namespaceBytes, err := hex.DecodeString(nIDString)
    +	if err != nil {
    +		return nil, fmt.Errorf("error decoding hex string: %w", err)
    +	}
    +
    +	// Next, we create a new NamespaceID using the parsed bytes
    +	return share.NewBlobNamespaceV0(namespaceBytes)
    +}
    // createNamespaceID converts a hex string to a NamespaceID
    +func createNamespaceID(nIDString string) (share.Namespace, error) {
    +	// First, we parse the passed hex string into a []byte slice
    +	namespaceBytes, err := hex.DecodeString(nIDString)
    +	if err != nil {
    +		return nil, fmt.Errorf("error decoding hex string: %w", err)
    +	}
    +
    +	// Next, we create a new NamespaceID using the parsed bytes
    +	return share.NewBlobNamespaceV0(namespaceBytes)
    +}

    Next, we need a utility that takes the namespace generated by createNamespaceID and constructs and submits a blob to the network.

    If successful, it returns the created blob, the height at which it was posted, and an empty error. Otherwise, only the error field is populated.

    go
    // createAndSubmitBlob creates a new blob and submits it to the network.
    +func createAndSubmitBlob(
    +	ctx context.Context,
    +	client *nodeclient.Client,
    +	ns share.Namespace,
    +	payload string,
    +) (*blob.Blob, uint64, error) {
    +	// First we can create the blob using the namespace and payload.
    +	createdBlob, err := blob.NewBlobV0(ns, []byte(payload))
    +	if err != nil {
    +		return nil, 0, fmt.Errorf("Failed to create blob: %w", err)
    +	}
    +
    +	// After we've created the blob, we can submit it to the network.
    +	// Here we use the default gas price.
    +	height, err := client.Blob.Submit(ctx, []*blob.Blob{createdBlob}, blob.DefaultGasPrice())
    +	if err != nil {
    +		return nil, 0, fmt.Errorf("Failed to submit blob: %v", err)
    +	}
    +
    +	log.Printf("Blob submitted successfully at height: %d! \n", height)
    +	log.Printf("Explorer link: https://arabica.celenium.io/block/%d \n", height)
    +
    +	return createdBlob, height, nil
    +}
    // createAndSubmitBlob creates a new blob and submits it to the network.
    +func createAndSubmitBlob(
    +	ctx context.Context,
    +	client *nodeclient.Client,
    +	ns share.Namespace,
    +	payload string,
    +) (*blob.Blob, uint64, error) {
    +	// First we can create the blob using the namespace and payload.
    +	createdBlob, err := blob.NewBlobV0(ns, []byte(payload))
    +	if err != nil {
    +		return nil, 0, fmt.Errorf("Failed to create blob: %w", err)
    +	}
    +
    +	// After we've created the blob, we can submit it to the network.
    +	// Here we use the default gas price.
    +	height, err := client.Blob.Submit(ctx, []*blob.Blob{createdBlob}, blob.DefaultGasPrice())
    +	if err != nil {
    +		return nil, 0, fmt.Errorf("Failed to submit blob: %v", err)
    +	}
    +
    +	log.Printf("Blob submitted successfully at height: %d! \n", height)
    +	log.Printf("Explorer link: https://arabica.celenium.io/block/%d \n", height)
    +
    +	return createdBlob, height, nil
    +}

    With our updated main function, we can now call these utility functions to check off our next TODO items.

    go
    func main() {
    +	ctx, cancel := context.WithCancel(context.Background())
    +	defer cancel()
    +
    +	// Get IP, namespace, and prompt from program arguments
    +	if len(os.Args) != 4 {
    +		log.Fatal("Usage: go run main.go <nodeIP> <namespace> <prompt>")
    +	}
    +	nodeIP, namespaceHex, prompt := os.Args[1], os.Args[2], os.Args[3]
    +
    +	// We pass an empty string as the jwt token, since we
    +	// disabled auth with the --rpc.skip-auth flag
    +	client, err := nodeclient.NewClient(ctx, nodeIP, "")
    +	if err != nil {
    +		log.Fatalf("Failed to create client: %v", err)
    +	}
    +	defer client.Close()
    +
    +	// Next, we convert the namespace hex string to the
    +	// concrete NamespaceID type
    +	namespaceID, err := createNamespaceID(namespaceHex)
    +	if err != nil {
    +		log.Fatalf("Failed to decode namespace: %v", err)
    +	}
    +
    +	// We can then create and submit a blob using the NamespaceID and our prompt.
    +	createdBlob, height, err := createAndSubmitBlob(ctx, client, namespaceID, prompt)
    +	if err != nil {
    +		log.Fatal(err)
    +	}
    +
    +	// Now we will fetch the blob back from the network, using the height, namespace, and blob commitment.
    +	fetchedBlob, err := client.Blob.Get(ctx, height, namespaceID, createdBlob.Commitment)
    +	if err != nil {
    +		log.Fatalf("Failed to fetch blob: %v", err)
    +	}
    +
    +	log.Printf("Fetched blob: %s\n", string(fetchedBlob.Data))
    +
    +	// TODO:
    +	// - [X] Load program arguments
    +	// - [X] Initialize the node API client
    +	// - [X] Create a namespace ID
    +	// - [X] Create and submit a blob
    +	// - [X] Retrieve the blob from the network
    +	// - [ ] Prompt chatgpt with the retrieved blob data
    +}
    func main() {
    +	ctx, cancel := context.WithCancel(context.Background())
    +	defer cancel()
    +
    +	// Get IP, namespace, and prompt from program arguments
    +	if len(os.Args) != 4 {
    +		log.Fatal("Usage: go run main.go <nodeIP> <namespace> <prompt>")
    +	}
    +	nodeIP, namespaceHex, prompt := os.Args[1], os.Args[2], os.Args[3]
    +
    +	// We pass an empty string as the jwt token, since we
    +	// disabled auth with the --rpc.skip-auth flag
    +	client, err := nodeclient.NewClient(ctx, nodeIP, "")
    +	if err != nil {
    +		log.Fatalf("Failed to create client: %v", err)
    +	}
    +	defer client.Close()
    +
    +	// Next, we convert the namespace hex string to the
    +	// concrete NamespaceID type
    +	namespaceID, err := createNamespaceID(namespaceHex)
    +	if err != nil {
    +		log.Fatalf("Failed to decode namespace: %v", err)
    +	}
    +
    +	// We can then create and submit a blob using the NamespaceID and our prompt.
    +	createdBlob, height, err := createAndSubmitBlob(ctx, client, namespaceID, prompt)
    +	if err != nil {
    +		log.Fatal(err)
    +	}
    +
    +	// Now we will fetch the blob back from the network, using the height, namespace, and blob commitment.
    +	fetchedBlob, err := client.Blob.Get(ctx, height, namespaceID, createdBlob.Commitment)
    +	if err != nil {
    +		log.Fatalf("Failed to fetch blob: %v", err)
    +	}
    +
    +	log.Printf("Fetched blob: %s\n", string(fetchedBlob.Data))
    +
    +	// TODO:
    +	// - [X] Load program arguments
    +	// - [X] Initialize the node API client
    +	// - [X] Create a namespace ID
    +	// - [X] Create and submit a blob
    +	// - [X] Retrieve the blob from the network
    +	// - [ ] Prompt chatgpt with the retrieved blob data
    +}

    TIP

    Alternatively to client.Blob.Get, you could also use client.Blob.GetAll(ctx, height, []share.Namespace{namespaceID}) which fetches all blobs in the namespace at the given height.

    Now our program is able to create the namespace and blob, then submit and fetch it from the arabica network. The next step is to prompt ChatGPT with the fetched blob data.

    Prompting ChatGPT

    First, we need one more utility function to help us prompt GPT-3.5. It reads the OPENAI_KEY environment variable and uses it to create a new GPT-3 client, which it uses to prompt and retrieve the answer.

    go
    // gpt3 processes a given message using GPT-3 and returns the response.
    +func gpt3(ctx context.Context, msg string) (string, error) {
    +    // Set the authentication header
    +    openAIKey := os.Getenv("OPENAI_KEY")
    +   	if openAIKey == "" {
    +    	return "", fmt.Errorf("OPENAI_KEY environment variable not set")
    +    }
    +    client := openai.NewClient(openAIKey)
    +    resp, err := client.CreateChatCompletion(
    +		ctx,
    +        openai.ChatCompletionRequest{
    +            Model: openai.GPT3Dot5Turbo,
    +            Messages: []openai.ChatCompletionMessage{
    +                {
    +                    Role:    openai.ChatMessageRoleUser,
    +                    Content: msg,
    +                },
    +            },
    +        },
    +    )
    +
    +    if err != nil {
    +       	return "", fmt.Errorf("ChatCompletion error: %w", err)
    +    }
    +
    +    return resp.Choices[0].Message.Content, nil
    +}
    // gpt3 processes a given message using GPT-3 and returns the response.
    +func gpt3(ctx context.Context, msg string) (string, error) {
    +    // Set the authentication header
    +    openAIKey := os.Getenv("OPENAI_KEY")
    +   	if openAIKey == "" {
    +    	return "", fmt.Errorf("OPENAI_KEY environment variable not set")
    +    }
    +    client := openai.NewClient(openAIKey)
    +    resp, err := client.CreateChatCompletion(
    +		ctx,
    +        openai.ChatCompletionRequest{
    +            Model: openai.GPT3Dot5Turbo,
    +            Messages: []openai.ChatCompletionMessage{
    +                {
    +                    Role:    openai.ChatMessageRoleUser,
    +                    Content: msg,
    +                },
    +            },
    +        },
    +    )
    +
    +    if err != nil {
    +       	return "", fmt.Errorf("ChatCompletion error: %w", err)
    +    }
    +
    +    return resp.Choices[0].Message.Content, nil
    +}

    Wrapping things up

    Now, we will update our main function to finish our last TODO item: prompting CHATGPT with the fetched blob data.

    go
    func main() {
    +	ctx, cancel := context.WithCancel(context.Background())
    +	defer cancel()
    +
    +	// Get IP, namespace, and prompt from program arguments
    +	if len(os.Args) != 4 {
    +		log.Fatal("Usage: go run main.go <nodeIP> <namespace> <prompt>")
    +	}
    +	nodeIP, namespaceHex, prompt := os.Args[1], os.Args[2], os.Args[3]
    +
    +	// We pass an empty string as the jwt token, since we
    +	// disabled auth with the --rpc.skip-auth flag
    +	client, err := nodeclient.NewClient(ctx, nodeIP, "")
    +	if err != nil {
    +		log.Fatalf("Failed to create client: %v", err)
    +	}
    +	defer client.Close()
    +
    +	// Next, we convert the namespace hex string to the
    +	// concrete NamespaceID type
    +	namespaceID, err := createNamespaceID(namespaceHex)
    +	if err != nil {
    +		log.Fatalf("Failed to decode namespace: %v", err)
    +	}
    +
    +	// We can then create and submit a blob using the NamespaceID and our prompt.
    +	createdBlob, height, err := createAndSubmitBlob(ctx, client, namespaceID, prompt)
    +	if err != nil {
    +		log.Fatal(err)
    +	}
    +
    +	// Now we will fetch the blob back from the network.
    +	fetchedBlob, err := client.Blob.Get(ctx, height, namespaceID, createdBlob.Commitment)
    +	if err != nil {
    +		log.Fatalf("Failed to fetch blob: %v", err)
    +	}
    +
    +	log.Printf("Fetched blob: %s\n", string(fetchedBlob.Data))
    +	promptAnswer, err := gpt3(ctx, string(fetchedBlob.Data))
    +	if err != nil {
    +		log.Fatalf("Failed to process message with GPT-3: %v", err)
    +	}
    +
    +	log.Printf("GPT-3 response: %s\n", promptAnswer)
    +}
    func main() {
    +	ctx, cancel := context.WithCancel(context.Background())
    +	defer cancel()
    +
    +	// Get IP, namespace, and prompt from program arguments
    +	if len(os.Args) != 4 {
    +		log.Fatal("Usage: go run main.go <nodeIP> <namespace> <prompt>")
    +	}
    +	nodeIP, namespaceHex, prompt := os.Args[1], os.Args[2], os.Args[3]
    +
    +	// We pass an empty string as the jwt token, since we
    +	// disabled auth with the --rpc.skip-auth flag
    +	client, err := nodeclient.NewClient(ctx, nodeIP, "")
    +	if err != nil {
    +		log.Fatalf("Failed to create client: %v", err)
    +	}
    +	defer client.Close()
    +
    +	// Next, we convert the namespace hex string to the
    +	// concrete NamespaceID type
    +	namespaceID, err := createNamespaceID(namespaceHex)
    +	if err != nil {
    +		log.Fatalf("Failed to decode namespace: %v", err)
    +	}
    +
    +	// We can then create and submit a blob using the NamespaceID and our prompt.
    +	createdBlob, height, err := createAndSubmitBlob(ctx, client, namespaceID, prompt)
    +	if err != nil {
    +		log.Fatal(err)
    +	}
    +
    +	// Now we will fetch the blob back from the network.
    +	fetchedBlob, err := client.Blob.Get(ctx, height, namespaceID, createdBlob.Commitment)
    +	if err != nil {
    +		log.Fatalf("Failed to fetch blob: %v", err)
    +	}
    +
    +	log.Printf("Fetched blob: %s\n", string(fetchedBlob.Data))
    +	promptAnswer, err := gpt3(ctx, string(fetchedBlob.Data))
    +	if err != nil {
    +		log.Fatalf("Failed to process message with GPT-3: %v", err)
    +	}
    +
    +	log.Printf("GPT-3 response: %s\n", promptAnswer)
    +}

    And now you have the final version of the prompt scavenger!

    Run the golang script with the following command:

    sh
    go run main.go <nodeIP> <namespace> <prompt>
    go run main.go <nodeIP> <namespace> <prompt>

    For example, you could run:

    sh
    go run main.go ws://localhost:26658 ce1e5714 'What is a modular blockchain?'
    go run main.go ws://localhost:26658 ce1e5714 'What is a modular blockchain?'

    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 blob, submit it to Celestia, get it back from Celestia, decode its contents, then for added bonus, submit the message to GPT-3.5.

    If you're up for a challenge, you can refer to the Node API client guide and try to implement more advanced features, such as:

    • Subscribing to new prompts inside the ce1e5714 namespace, submitting each one to GPT-3.5
    • Posting the responses back to Celestia under a different namespace.
    + + + + \ No newline at end of file diff --git a/developers/rust-client-tutorial.html b/developers/rust-client-tutorial.html new file mode 100644 index 00000000000..5bfe297c2cf --- /dev/null +++ b/developers/rust-client-tutorial.html @@ -0,0 +1,215 @@ + + + + + + Rust client library tutorial | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Rust client library tutorial

    This section tutorial will guide you through using the most common RPC endpoints with Lumina's rust client library.

    Install dependencies and celestia-node if you have not already.

    Project setup

    To start, add celestia_rpc and celestia_types as a dependency to your project:

    bash
    cargo add celestia_rpc celestia_types
    cargo add celestia_rpc celestia_types

    To use the following methods, you will need the node URL and your auth token. To get your auth token, see this guide. To run your node without an auth token, you can use the --rpc.skip-auth flag when starting your node. This allows you to pass an empty string as your auth token.

    The default URL is http://localhost:26658. If you would like to use subscription methods, such as SubscribeHeaders below, you must use the ws protocol in place of http: ws://localhost:26658.

    Submitting and retrieving blobs

    The blob.Submit method takes an array of blobs and a gas price, returning the height the blob was successfully posted at.

    • The namespace can be generated with Namespace::new_v0.
    • The blobs can be generated with Blob::new.
    • You can set GasPrice::default() as the gas price to have celestia-node automatically determine an appropriate gas price.

    The blob.GetAll method takes a height and array of namespaces, returning the array of blobs found in the given namespaces.

    rust
    use celestia_rpc::{BlobClient, Client, HeaderClient, ShareClient};
    +use celestia_types::blob::GasPrice;
    +use celestia_types::{nmt::Namespace, Blob, ExtendedDataSquare};
    +
    +async fn submit_blob(url: &str, token: &str) {
    +    let client = Client::new(url, Some(token))
    +        .await
    +        .expect("Failed creating rpc client");
    +
    +    // let's use the DEADBEEF namespace
    +    let namespace = Namespace::new_v0(&[0xDE, 0xAD, 0xBE, 0xEF]).expect("Invalid namespace");
    +
    +    // create a blob
    +    let blob = Blob::new(namespace, b"Hello, World!".to_vec()).expect("Blob creation failed");
    +
    +    // submit the blob to the network
    +    let height = client
    +        .blob_submit(&[blob.clone()], GasPrice::default())
    +        .await
    +        .expect("Failed submitting blob");
    +
    +    println!("Blob was included at height {}", height);
    +
    +    // fetch the blob back from the network
    +    let retrieved_blobs = client
    +        .blob_get_all(height, &[namespace])
    +        .await
    +        .expect("Failed to retrieve blobs");
    +
    +    assert_eq!(retrieved_blobs.len(), 1);
    +    assert_eq!(retrieved_blobs[0].data, b"Hello, World!");
    +    assert_eq!(retrieved_blobs[0].commitment, blob.commitment);
    +}
    use celestia_rpc::{BlobClient, Client, HeaderClient, ShareClient};
    +use celestia_types::blob::GasPrice;
    +use celestia_types::{nmt::Namespace, Blob, ExtendedDataSquare};
    +
    +async fn submit_blob(url: &str, token: &str) {
    +    let client = Client::new(url, Some(token))
    +        .await
    +        .expect("Failed creating rpc client");
    +
    +    // let's use the DEADBEEF namespace
    +    let namespace = Namespace::new_v0(&[0xDE, 0xAD, 0xBE, 0xEF]).expect("Invalid namespace");
    +
    +    // create a blob
    +    let blob = Blob::new(namespace, b"Hello, World!".to_vec()).expect("Blob creation failed");
    +
    +    // submit the blob to the network
    +    let height = client
    +        .blob_submit(&[blob.clone()], GasPrice::default())
    +        .await
    +        .expect("Failed submitting blob");
    +
    +    println!("Blob was included at height {}", height);
    +
    +    // fetch the blob back from the network
    +    let retrieved_blobs = client
    +        .blob_get_all(height, &[namespace])
    +        .await
    +        .expect("Failed to retrieve blobs");
    +
    +    assert_eq!(retrieved_blobs.len(), 1);
    +    assert_eq!(retrieved_blobs[0].data, b"Hello, World!");
    +    assert_eq!(retrieved_blobs[0].commitment, blob.commitment);
    +}

    Subscribing to new headers

    You can subscribe to new headers using the header.Subscribe method. This method returns a Subscription that will receive new headers as they are produced. In this example, we will fetch all blobs at the height of the new header in the 0xDEADBEEF namespace.

    rust
    async fn subscribe_headers(url: &str, token: &str) {
    +    let client = Client::new(url, Some(token))
    +        .await
    +        .expect("Failed creating rpc client");
    +
    +    let mut header_sub = client
    +        .header_subscribe()
    +        .await
    +        .expect("Failed subscribing to incoming headers");
    +
    +    // setup the namespace we will filter blobs by
    +    let namespace = Namespace::new_v0(&[0xDE, 0xAD, 0xBE, 0xEF]).expect("Invalid namespace");
    +
    +    while let Some(extended_header) = header_sub.next().await {
    +        match extended_header {
    +            Ok(header) => {
    +                let height = header.header.height.value();
    +                // fetch all blobs at the height of the new header
    +
    +                let blobs = match client.blob_get_all(height, &[namespace]).await {
    +                    Ok(blobs) => blobs,
    +                    Err(e) => {
    +                        eprintln!("Error fetching blobs: {}", e);
    +                        continue;
    +                    }
    +                };
    +
    +                println!(
    +                    "Found {} blobs at height {} in the 0xDEADBEEF namespace",
    +                    blobs.len(),
    +                    height
    +                );
    +            }
    +            Err(e) => {
    +                eprintln!("Error receiving header: {}", e);
    +            }
    +        }
    +    }
    +}
    async fn subscribe_headers(url: &str, token: &str) {
    +    let client = Client::new(url, Some(token))
    +        .await
    +        .expect("Failed creating rpc client");
    +
    +    let mut header_sub = client
    +        .header_subscribe()
    +        .await
    +        .expect("Failed subscribing to incoming headers");
    +
    +    // setup the namespace we will filter blobs by
    +    let namespace = Namespace::new_v0(&[0xDE, 0xAD, 0xBE, 0xEF]).expect("Invalid namespace");
    +
    +    while let Some(extended_header) = header_sub.next().await {
    +        match extended_header {
    +            Ok(header) => {
    +                let height = header.header.height.value();
    +                // fetch all blobs at the height of the new header
    +
    +                let blobs = match client.blob_get_all(height, &[namespace]).await {
    +                    Ok(blobs) => blobs,
    +                    Err(e) => {
    +                        eprintln!("Error fetching blobs: {}", e);
    +                        continue;
    +                    }
    +                };
    +
    +                println!(
    +                    "Found {} blobs at height {} in the 0xDEADBEEF namespace",
    +                    blobs.len(),
    +                    height
    +                );
    +            }
    +            Err(e) => {
    +                eprintln!("Error receiving header: {}", e);
    +            }
    +        }
    +    }
    +}

    Fetching an Extended Data Square (EDS)

    You can fetch an Extended Data Square (EDS) using the share.GetEDS method. This method takes a header and returns the EDS at the given height.

    rust
    async fn get_eds(url: &str, token: &str) -> ExtendedDataSquare {
    +    let client = Client::new(url, Some(token))
    +        .await
    +        .expect("Failed creating rpc client");
    +
    +    // first get the header of the block you want to fetch the EDS from
    +    let latest_header = client
    +        .header_local_head()
    +        .await
    +        .expect("Failed fetching header");
    +
    +    client
    +        .share_get_eds(&latest_header)
    +        .await
    +        .expect("Failed to get EDS from latest header")
    +}
    async fn get_eds(url: &str, token: &str) -> ExtendedDataSquare {
    +    let client = Client::new(url, Some(token))
    +        .await
    +        .expect("Failed creating rpc client");
    +
    +    // first get the header of the block you want to fetch the EDS from
    +    let latest_header = client
    +        .header_local_head()
    +        .await
    +        .expect("Failed fetching header");
    +
    +    client
    +        .share_get_eds(&latest_header)
    +        .await
    +        .expect("Failed to get EDS from latest header")
    +}

    API documentation

    To see the full list of available methods, see the API documentation.

    + + + + \ No newline at end of file diff --git a/developers/sp1-blobstream-deploy.html b/developers/sp1-blobstream-deploy.html new file mode 100644 index 00000000000..087fe5416a3 --- /dev/null +++ b/developers/sp1-blobstream-deploy.html @@ -0,0 +1,47 @@ + + + + + + New SP1 Blobstream deployments | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    New SP1 Blobstream deployments

    This document provides instructions for deploying SP1 Blobstream to a new chain.

    SP1 Blobstream is the latest implementation of Blobstream in Rust using the SP1 zkVM.

    Deploying the contracts

    To deploy SP1 Blobstream to a new chain, follow these steps:

    1. Clone the sp1-blobstream repository:
    shell
    git clone https://github.com/succinctlabs/sp1-blobstream
    +cd sp1-blobstream
    git clone https://github.com/succinctlabs/sp1-blobstream
    +cd sp1-blobstream
    1. Follow the deployment instructions in the sp1-blobstream README.

    2. If you're deploying on a chain where there isn't a canonical verifier listed in the SP1 contract addresses, you'll need to:

      a. Deploy your own SP1 Verifier from the sp1-contracts matching your sp1-sdk version. b. Set the SP1_VERIFIER_ADDRESS in your .env file to the address of your deployed verifier.

    3. To run the prover:

      • For local proving, set SP1_PROVER=local in your environment.
      • To use the Succinct Proving Network for remote proving, set SP1_PROVER=network.
      • We recommend an instance with 64 vCPU and 128GB of RAM for local proving.

    Note: Any whitelisting for custom provers would need to be implemented in the application's smart contracts (e.g., by using an approvedProvers mapping).

    + + + + \ No newline at end of file diff --git a/developers/submit-data.html b/developers/submit-data.html new file mode 100644 index 00000000000..1e5a67a5806 --- /dev/null +++ b/developers/submit-data.html @@ -0,0 +1,343 @@ + + + + + + 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.

    Maximum blob size

    The maximum total blob size in a transaction is just under 2 MiB (1,973,786 bytes), based on a 64x64 share grid (4096 shares). With one share for the PFB transaction, 4095 shares remain: 1 at 478 bytes and 4094 at 482 bytes each.

    This is subject to change based on governance parameters. Learn more on the Mainnet Beta page under "Maximum bytes".

    It is advisable to submit transactions where the total blob size is significantly smaller than 1.8 MiB (e.g. 500 KiB) in order for your transaction to get included in a block quickly. If a tx contains blobs approaching 1.8 MiB then there will be no room for any other transactions. This means that your transaction will only be included in a block if it has a higher gas price than every other transaction in the mempool.

    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 six 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.

    The celestia-node API golang client

    For more celestia-node API golang examples, refer to the golang client tutorial.

    go
    import (
    +	"bytes"
    +	"context"
    +	"fmt"
    +
    +	client "github.com/celestiaorg/celestia-openrpc"
    +	"github.com/celestiaorg/celestia-openrpc/types/blob"
    +	"github.com/celestiaorg/celestia-openrpc/types/share"
    +)
    +
    +// SubmitBlob submits a blob containing "Hello, World!" to the 0xDEADBEEF namespace. It uses the default signer on the running node.
    +func SubmitBlob(ctx context.Context, url string, token string) error {
    +	client, err := client.NewClient(ctx, url, token)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// let's post to 0xDEADBEEF namespace
    +	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
    +	if err != nil {
    +		return err
    +	}
    +
    +	// create a blob
    +	helloWorldBlob, err := blob.NewBlobV0(namespace, []byte("Hello, World!"))
    +	if err != nil {
    +		return err
    +	}
    +
    +	// submit the blob to the network
    +	height, err := client.Blob.Submit(ctx, []*blob.Blob{helloWorldBlob}, blob.DefaultGasPrice())
    +	if err != nil {
    +		return err
    +	}
    +
    +	fmt.Printf("Blob was included at height %d\n", height)
    +
    +	// bonus: fetch the blob back from the network
    +	retrievedBlobs, err := client.Blob.GetAll(ctx, height, []share.Namespace{namespace})
    +	if err != nil {
    +		return err
    +	}
    +
    +	fmt.Printf("Blobs are equal? %v\n", bytes.Equal(helloWorldBlob.Commitment, retrievedBlobs[0].Commitment))
    +	return nil
    +}}
    import (
    +	"bytes"
    +	"context"
    +	"fmt"
    +
    +	client "github.com/celestiaorg/celestia-openrpc"
    +	"github.com/celestiaorg/celestia-openrpc/types/blob"
    +	"github.com/celestiaorg/celestia-openrpc/types/share"
    +)
    +
    +// SubmitBlob submits a blob containing "Hello, World!" to the 0xDEADBEEF namespace. It uses the default signer on the running node.
    +func SubmitBlob(ctx context.Context, url string, token string) error {
    +	client, err := client.NewClient(ctx, url, token)
    +	if err != nil {
    +		return err
    +	}
    +
    +	// let's post to 0xDEADBEEF namespace
    +	namespace, err := share.NewBlobNamespaceV0([]byte{0xDE, 0xAD, 0xBE, 0xEF})
    +	if err != nil {
    +		return err
    +	}
    +
    +	// create a blob
    +	helloWorldBlob, err := blob.NewBlobV0(namespace, []byte("Hello, World!"))
    +	if err != nil {
    +		return err
    +	}
    +
    +	// submit the blob to the network
    +	height, err := client.Blob.Submit(ctx, []*blob.Blob{helloWorldBlob}, blob.DefaultGasPrice())
    +	if err != nil {
    +		return err
    +	}
    +
    +	fmt.Printf("Blob was included at height %d\n", height)
    +
    +	// bonus: fetch the blob back from the network
    +	retrievedBlobs, err := client.Blob.GetAll(ctx, height, []share.Namespace{namespace})
    +	if err != nil {
    +		return err
    +	}
    +
    +	fmt.Printf("Blobs are equal? %v\n", bytes.Equal(helloWorldBlob.Commitment, retrievedBlobs[0].Commitment))
    +	return nil
    +}}

    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.

    Post a blob directly from Celenium

    Celenium provides a user-friendly interface to view and interact with data on Celestia, and allows for submitting blobs directly from the explorer interface.

    To submit a blob from Celenium, follow these steps:

    1. Navigate to the Celenium explorer and connect your wallet.
    2. Click on the terminal button in the top right corner of the screen and select the "Submit data blob" option.
    3. Next, ensure the file you are submitting is in a supported format and upload it.
    4. In the "Namespace" field, input the namespace you want to use for the blob in hex format.
    5. Finally, click on the "Continue" button to submit your blob, then approve the transaction in your wallet.

    Once the blob is submitted, you will see its hash on the screen. You can also use Celenium’s search bar to search for the blob's hash and view its details.

    + + + + \ No newline at end of file diff --git a/developers/transaction-resubmission.html b/developers/transaction-resubmission.html new file mode 100644 index 00000000000..f55ee05652d --- /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.

    If you are running a production application on Celestia, it is recommended to use a high-availability, production RPC provider for Mainnet Beta,Mocha, or Arabica.

    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..d87a0afd9ef --- /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..0d20cdc143e --- /dev/null +++ b/hashmap.json @@ -0,0 +1 @@ +{"community_foundation-delegation-program.md":"bc29b071","developers_bubs-testnet.md":"bc7bccdc","developers_build-whatever.md":"4b1651ac","developers_feegrant-for-blobs.md":"5c24e85c","community_modular-meetup-toolkit.md":"6175a760","developers_multiaccounts.md":"678a9ea9","readme.md":"03785d82","community_modular-meetup-intro.md":"8ebda042","community_modular-meetup-guide.md":"d402e287","developers_arbitrum-deploy.md":"e3157085","developers_node-api.md":"bc539c55","developers_arbitrum-full-node.md":"b72127e2","developers_blobstream.md":"16782cf1","developers_arbitrum-integration.md":"1e6a0883","developers_integrate-celestia.md":"6f9ae561","developers_celestia-node-key.md":"23b7816e","developers_golang-client-tutorial.md":"6e7c47af","developers_blobstream-x-requesting-data-commitment-ranges.md":"729e2aa5","developers_blobstreamx.md":"fad0ad62","developers_blobstream-rollups.md":"7d43d4fb","developers_blobstream-contracts.md":"6f2c6e34","developers_intro-to-op-stack.md":"60ea8345","developers_ethereum-fallback.md":"4892edb4","community_speaker-list.md":"a4ac972d","developers_blobstream-x-deploy.md":"31be9369","developers_rust-client-tutorial.md":"78994657","developers_sp1-blobstream-deploy.md":"38c380ed","developers_prompt-scavenger.md":"b002a641","community_coc.md":"e7f86a12","index.md":"74361e6f","developers_blobstream-proof-queries.md":"34df4e6b","developers_blobstream-offchain.md":"a074cb83","developers_arbitrum-bridge.md":"79abe3b3","developers_submit-data.md":"e151ce10","learn_how-to-stake-tia.md":"53e17214","learn_paying-for-blobspace.md":"7953d033","learn_staking-governance-supply.md":"acda4d48","learn_staking.md":"39685d72","developers_optimism-devnet.md":"0a0a7e82","nodes_arabica-devnet.md":"05307ae8","learn_tia.md":"a88f9d95","developers_optimism.md":"a839aad4","learn_how-celestia-works_data-availability-faq.md":"55daaf96","learn_retrievability.md":"1218f7e5","developers_node-tutorial.md":"c48e1a3a","nodes_celestia-app-metrics.md":"97be623c","nodes_celestia-node-trusted-hash.md":"9ca1f0ed","nodes_bridge-node.md":"1354e885","nodes_celestia-node-troubleshooting.md":"6777331f","nodes_config-toml.md":"b1d84050","nodes_celestia-node.md":"727b45ab","developers_transaction-resubmission.md":"f1e4eb76","learn_how-celestia-works_monolithic-vs-modular.md":"80df84fc","developers_wallets.md":"a3e6d0a6","learn_how-celestia-works_overview.md":"2b53d98a","learn_how-celestia-works_transaction-lifecycle.md":"56692f20","nodes_celestia-app-slashing.md":"d2c1a178","nodes_celestia-node-custom-networks.md":"c55d089b","nodes_celestia-app-multisig.md":"6c014039","nodes_celestia-app-upgrade-monitor.md":"9a019c51","learn_how-celestia-works_data-availability-layer.md":"648c5274","nodes_docker-images.md":"97bd64bd","nodes_environment.md":"4596b94b","nodes_full-storage-node.md":"16ca408c","nodes_instantiate-testnet.md":"b0829848","nodes_consensus-node.md":"b07b70a7","nodes_celestia-app.md":"2a20c575","nodes_light-node.md":"768fa4a4","nodes_validator-node.md":"df92fc20","nodes_celestia-app-wallet.md":"9beb14f0","nodes_mainnet.md":"37db891f","nodes_participate.md":"e6ceada6","nodes_ibc-relayer.md":"6041bb8b","nodes_decide-node.md":"0a555064","nodes_celestia-node-metrics.md":"b99ab7c2","nodes_systemd.md":"e4f637ee","nodes_quick-start.md":"60c923e3","nodes_overview.md":"84db2210","nodes_celestia-app-commands.md":"3c444cb8","nodes_mocha-testnet.md":"a45f13e7","nodes_celestia-app-vesting.md":"1d8992c8","nodes_network-upgrade-process.md":"a106e947"} 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/gem.png b/img/gem.png new file mode 100644 index 00000000000..be859ff0236 Binary files /dev/null and b/img/gem.png differ diff --git a/img/gem/gem1.gif b/img/gem/gem1.gif new file mode 100644 index 00000000000..5566c037df2 Binary files /dev/null and b/img/gem/gem1.gif differ diff --git a/img/gem/gem2.gif b/img/gem/gem2.gif new file mode 100644 index 00000000000..fbd23c1ab27 Binary files /dev/null and b/img/gem/gem2.gif differ diff --git a/img/gem/gem3.gif b/img/gem/gem3.gif new file mode 100644 index 00000000000..90e98b7a3a8 Binary files /dev/null and b/img/gem/gem3.gif differ diff --git a/img/gem/gem4.gif b/img/gem/gem4.gif new file mode 100644 index 00000000000..cb0ee0b8d4a Binary files /dev/null and b/img/gem/gem4.gif 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..b2acf3c74e4 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/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/consensus-node.jpg b/img/nodes/consensus-node.jpg new file mode 100644 index 00000000000..5a117bd5591 Binary files /dev/null and b/img/nodes/consensus-node.jpg 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..2749adc931d --- /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..3102815d603 --- /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..916c4747803 --- /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..856b85e24ea --- /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..ee4c510fcad --- /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..2f1929c8257 --- /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..3e862f34266 --- /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, Leap, and Gem wallets.

    Select your preferred wallet

    Keplr
    Leap
    Gem Wallet

    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

    Stake TIA with Gem wallet

    1️⃣ Open your Gem Wallet app

    Navigate to Celestia and select Stake.

    Gem1

    2️⃣ Choose the amount of Celestia and search for a validator.

    Select the amount of Celestia tokens and choose a validator from the list.

    Gem2

    3️⃣ Stake your TIA tokens

    Review the network terms and commission, then press Confirm to proceed.

    Gem3

    4️⃣ Manage your TIA

    After your transaction is confirmed, you will have access to a control panel where you can claim rewards, unstake, redelegate, or stake additional tokens.

    Gem4

    + + + + \ 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..a6c3572f164 --- /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..2a4c2f9834a --- /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..4e1265f4dc3 --- /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.41%
    Future initiatives: 12.59%
    20.00%
    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.79%
    Early Backers: Series A&BEarly supporters of Celestia19.67%
    Early Backers: SeedEarly supporters of Celestia15.90%
    Initial Core ContributorsMembers of Celestia Labs, the first core contributor to Celestia17.64%

    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.00% unlocked at launch.
    Remaining 75.00% unlocks continuously from year 1 to year 4.
    Initial Core Contributors33.33% unlocked at year 1.
    Remaining 66.67% unlocks continuously from year 1 to year 3.
    Early Backers: Seed33.33% unlocked at year 1.
    Remaining 66.67% unlocks continuously from year 1 to year 2.
    Early Backers: Series A&B33.33% unlocked at year 1.
    Remaining 66.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..3452714faee --- /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..b2476147192 --- /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..413751490fe --- /dev/null +++ b/nodes/arabica-devnet.html @@ -0,0 +1,50 @@ + + + + + + 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 network upgrades 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.16.0
    celestia-appv2.1.2

    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.

    Production RPC endpoints

    These RPC providers are meant to be used in production environments.

    ProviderURL
    NewMetrichttps://app.newmetric.xyz/start
    NumiaFor RPC access: https://docs.numia.xyz/overview/rpc-api-access
    NumiaFor data warehouse access: https://docs.numia.xyz/overview/sql-access/chains/celestia
    Grovehttps://www.grove.city/

    Community RPC endpoints

    WARNING

    Do not rely on the free community endpoints listed below for production deployments. Production deployments should rely on service providers with SLAs or your own node.

    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

    Data availability (DA) RPC endpoints for bridge node sync

    These RPC endpoints allow bridge nodes to sync blocks from the Celestia network. For users, they will need to provide a –core.ip string from a consensus node’s URL or IP that populates a default RPC port at 26657 to their respective DA node.

    Data availability (DA) gRPC endpoints for state access

    These gRPC endpoints for DA nodes 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 a default gRPC port at 9090 to their respective DA node.

    EXAMPLE

    bash
    celestia <da_type> start –core.ip <url> -–core.grpc.port <port>
    celestia <da_type> start –core.ip <url> -–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 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.

    See the network upgrade process page to learn more about specific upgrades like the Lemongrass network upgrade.

    + + + + \ No newline at end of file diff --git a/nodes/bridge-node.html b/nodes/bridge-node.html new file mode 100644 index 00000000000..c344f0371ea --- /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: 16 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.

    Connecting to a core endpoint with --core.ip string provides the light node with access to state queries (reading balances, submitting transactions, and other state-related queries).

    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.keyname <name-of-custom-key>
    celestia bridge start --core.ip <URI> --keyring.keyname <name-of-custom-key>
    sh
    celestia bridge start --core.ip <URI> --keyring.keyname <name-of-custom-key> \
    +  --p2p.network mocha
    celestia bridge start --core.ip <URI> --keyring.keyname <name-of-custom-key> \
    +  --p2p.network mocha
    sh
    celestia bridge start --core.ip <URI> --keyring.keyname <name-of-custom-key> \
    +  --p2p.network arabica
    celestia bridge start --core.ip <URI> --keyring.keyname <name-of-custom-key> \
    +  --p2p.network arabica

    Optional: Migrate node id to another server

    To migrate a bridge node ID:

    1. You need to back up two files located in the celestia-bridge node directory at the correct path (default: ~/.celestia-bridge/keys).
    2. Upload the files to the new server and start the node.

    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..f54efdbced2 --- /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:443
    celestia-appd q bank balances celestia1czpgn3hdh9sodm06d5qk23xzgpq2uyc8ggdqgw \
    +    --node https://rpc-mocha.pops.one:443

    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:443 --chain-id mocha
    celestia-appd tx bank send <FROM_ADDRESS> <TO_ADDRESS> \
    +    19000000utia --node https://rpc-mocha.pops.one:443 --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..df113174265 --- /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..07898f74512 --- /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..a053dd86235 --- /dev/null +++ b/nodes/celestia-app-slashing.html @@ -0,0 +1,45 @@ + + + + + + Jailing and slashing on Celestia | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Jailing and 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 jailed or 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 getting slashed. 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 jailing and 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..0a6d09f1f21 --- /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..4eb7d0d657d --- /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 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.

    If you are running a production application, use a production endpoint.

    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 consensus node or validator

    Running a consensus node or validator will prevent you from needing to use an RPC.

    You can set up a validator or 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..33e2ce5a2b5 --- /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..c073fe9075e --- /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/v2.1.2
      git checkout tags/v2.1.2
      bash
      git checkout tags/v2.1.2
      git checkout tags/v2.1.2
      bash
      git checkout tags/v2.1.2
      git checkout tags/v2.1.2
    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..4eeff15f2d0 --- /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..94b29131c6e --- /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..a96213426de --- /dev/null +++ b/nodes/celestia-node-troubleshooting.html @@ -0,0 +1,106 @@ + + + + + + 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

    Background

    An enhancement has been made in v0.14.0+ to automate the detection of the running node, eliminating the need to manually specify the --node.store flag for each RPC request.

    Assumptions:

    • The presence of a lock signifies a running node.
    • Networks are ordered as Mainnet Beta, Mocha, Arabica, private, custom.
    • Node types are ordered as bridge, full, and light.
    • Each network has only one running node type.
    • Multiple nodes of the same network and type are prohibited (resulting in an Error: node: store is in use).

    Key Points:

    • Authentication token and other flags maintain their previous behavior and take precedence.
    • Address and port details are fetched from the configuration.
    • skipAuth allows bypassing authentication for trusted setups and follows Unix daemon conventions.
    • Non-default node store and cel-key configurations still require specific flags in the configuration settings.

    Demonstration

    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/

    Examples

    Mainnet Beta full and Mocha light

    This example uses a Mainnet Beta full node and a Mocha light node. When making the request:

    bash
     celestia blob get 1318129 0x42690c204d39600fddd3 0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=
    +{
    + "result": "RPC client error: sendRequest failed: http status 401 Unauthorized unmarshaling response: EOF"
    +}
     celestia blob get 1318129 0x42690c204d39600fddd3 0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=
    +{
    + "result": "RPC client error: sendRequest failed: http status 401 Unauthorized unmarshaling response: EOF"
    +}

    The request will go to the Mainnet Beta node, and a 401 will show in this node's logs. Note that a 401 is expected because this blob was posted to Mocha and neither the namespace nor the blob exist on Mainnet Beta.

    Mocha full and Arabica light

    This example uses a Mocha full node and an Arabica light node. When making the request:

    bash
     celestia blob get 1318129 0x42690c204d39600fddd3 0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=
    +{
    +  "result": {
    +    "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +    "data": "0x676d",
    +    "share_version": 0,
    +    "commitment": "0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=",
    +    "index": 23
    +  }
    +}
     celestia blob get 1318129 0x42690c204d39600fddd3 0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=
    +{
    +  "result": {
    +    "namespace": "AAAAAAAAAAAAAAAAAAAAAAAAAEJpDCBNOWAP3dM=",
    +    "data": "0x676d",
    +    "share_version": 0,
    +    "commitment": "0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=",
    +    "index": 23
    +  }
    +}

    The request will go to the Mocha full node, and result shown as expected.

    Using a custom rpc.config address

    When using a custom RPC config address 0.0.0.1 and port 25231, the CLI accurately routes to the custom address and port, where no node is running. It fails as expected:

    bash
     celestia blob get 1318129 0x42690c204d39600fddd3 0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=
    +{
    +  "result": "RPC client error: sendRequest failed: Post \"http://0.0.0.1:25231\": dial tcp 0.0.0.1:25231: connect: no route to host"
    +}
     celestia blob get 1318129 0x42690c204d39600fddd3 0MFhYKQUi2BU+U1jxPzG7QY2BVV1lb3kiU+zAK7nUiY=
    +{
    +  "result": "RPC client error: sendRequest failed: Post \"http://0.0.0.1:25231\": dial tcp 0.0.0.1:25231: connect: no route to host"
    +}

    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-trusted-hash.html b/nodes/celestia-node-trusted-hash.html new file mode 100644 index 00000000000..2437a870c20 --- /dev/null +++ b/nodes/celestia-node-trusted-hash.html @@ -0,0 +1,47 @@ + + + + + + Syncing a light node from a trusted hash | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Syncing a light node from a trusted hash

    This guide goes over how to sync a DA light node from a trusted hash. The example is with Mainnet Beta. You will need to adjust the commands accordingly for Mocha, Arabica, or a custom network.

    WARNING

    Syncing to a trusted hash means that you will not sample the entire chain. This adds a trust assumption that you trust the history of the chain up to that point and that you trust the entity where you get the hash from. In this example, the trusted entity is a consensus endpoint or Celenium

    1. Get trusted height & hash from a consensus endpoint or Celenium.

    2. Initialize the node store

      sh
      celestia light init --p2p.network <network>
      celestia light init --p2p.network <network>
    3. Set the trusted height & hash

      1. Open your config.toml at .celestia-light/config.toml (or .celestia-light-<other-network>/config.toml)
      2. Set DASer.SampleFrom to the trusted height (e.g. SampleFrom = 123456)
    4. Run the node with the hash and flag:

    sh
    celestia light start --headers.trusted-hash <hash_of_block_n> \
    +    --p2p.network <network> --core.ip <consensus-node-rpc>
    celestia light start --headers.trusted-hash <hash_of_block_n> \
    +    --p2p.network <network> --core.ip <consensus-node-rpc>

    For service operators

    If you're using multiple light nodes for similar services like tracking the same rollup, it is recommended to use the same hash and height for them all services using the same starting height.

    + + + + \ No newline at end of file diff --git a/nodes/celestia-node.html b/nodes/celestia-node.html new file mode 100644 index 00000000000..c4458036757 --- /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.16.0
      git checkout tags/v0.16.0
      bash
      git checkout tags/v0.16.0
      git checkout tags/v0.16.0
      bash
      git checkout tags/v0.16.0
      git checkout tags/v0.16.0
    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..599b7b8537c --- /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/consensus-node.html b/nodes/consensus-node.html new file mode 100644 index 00000000000..bd19ec56b8a --- /dev/null +++ b/nodes/consensus-node.html @@ -0,0 +1,126 @@ + + + + + + Consensus node | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Consensus node

    This guide covers how to set up a consensus node on Celestia. Consensus nodes allow you to sync the entire blockchain history in the Celestia consensus layer.

    consensus node

    Minimum hardware requirements

    The following minimum hardware requirements are recommended for running a consensus node:

    • Memory: 16 GB RAM
    • CPU: Quad-Core
    • Disk: 2 TB SSD Storage
    • Bandwidth: 1 Gbps for Download/1 Gbps for Upload

    Set up a 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

    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 = ""
    Optional: Set persistent peers

    Optionally, you can set persistent peers in your config.toml file. If you set persistent peers, your node will always try to connect to these peers. This is useful when running a local devnet, for example, when you would always want to connect to the same local nodes in your devnet. In production, setting persistent peers is advised only if you are running a sentry node.

    You can get the persistent peers from the @cosmos/chain-registry repository (for Mainnet Beta) or @celestiaorg/networks repository repo (for Mocha and Arabica) with the following commands:

    bash
    PERSISTENT_PEERS=$(curl -s https://raw.githubusercontent.com/cosmos/chain-registry/master/celestia/chain.json | jq -r '.peers.persistent_peers[].address' | tr '\n' ',' | sed 's/,$/\n/')
    +echo $PERSISTENT_PEERS
    +sed -i.bak -e "s/^persistent_peers *=.*/persistent_peers = \"$PERSISTENT_PEERS\"/" $HOME/.celestia-app/config/config.toml
    PERSISTENT_PEERS=$(curl -s https://raw.githubusercontent.com/cosmos/chain-registry/master/celestia/chain.json | jq -r '.peers.persistent_peers[].address' | tr '\n' ',' | sed 's/,$/\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

    Optional: Connect 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

    Query 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"

    Optional: Access 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"

    Save 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

    Sync types

    Sync modeTimeNotes
    Block sync~3 weeksDownloads and executes all blocks from genesis to the tip
    State sync~1 hourDownloads a snapshot of the state then downloads and executes all blocks after that snapshot to the tip.
    Quick sync~5 hoursDownloads the data directory from a node. Time depends on your download speed because the data being downloaded can exceed 1 TB for mainnet.

    Option 1: Block sync

    By default, a consensus node will sync using block sync; which will 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 weeks depending on the height of the blockchain).

    There are two alternatives for quicker syncing.

    Option 2: 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 = ""

    And also set statesync to true:

    toml
    #######################################################
    +###         State Sync Configuration Options        ###
    +#######################################################
    +[statesync]
    +enable = true
    #######################################################
    +###         State Sync Configuration Options        ###
    +#######################################################
    +[statesync]
    +enable = true

    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.

    Option 3: 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

    If you are running celestia-app v1.x.x:

    sh
    celestia-appd start
    celestia-appd start

    If you are running celestia-app >= v2.0.0: then you'll want to start the node with a --v2-upgrade-height that is dependent on the network. The --v2-upgrade-height flag is only needed during the v2 upgrade height so after your node has executed the upgrade (e.g. you see the log upgraded from app version 1 to 2), you don't need to provide this flag for future celestia-appd start invocations.

    sh
    celestia-appd start --v2-upgrade-height 2371495
    celestia-appd start --v2-upgrade-height 2371495
    sh
    celestia-appd start --v2-upgrade-height 2585031
    celestia-appd start --v2-upgrade-height 2585031
    sh
    celestia-appd start --v2-upgrade-height 1751707
    celestia-appd start --v2-upgrade-height 1751707

    Optional: If you would like celestia-app to run as a background process, you can follow the SystemD tutorial.

    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: Configure an RPC endpoint

    You can configure your 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.

    FAQ

    +2/3 committed an invalid block: wrong Block.Header.Version

    If you encounter an error like:

    bash
    2024-04-25 14:48:24 6:48PM ERR CONSENSUS FAILURE!!! err="+2/3 committed an invalid block: wrong Block.Header.Version. Expected {11 1}, got {11 2}" module=consensus stack="goroutine 214 [running]:\nruntime/debug.Stack()\n\t/usr/local/go/src/runtime/debug/stack.go:24 +0x64\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine.func2()\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:746 +0x44\npanic({0x1b91180?, 0x400153b240?})\n\t/usr/local/go/src/runtime/panic.go:770 +0x124\ngithub.com/tendermint/tendermint/consensus.(*State).finalizeCommit(0x400065ea88, 0x3)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1637 +0xd30\ngithub.com/tendermint/tendermint/consensus.(*State).tryFinalizeCommit(0x400065ea88, 0x3)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1606 +0x26c\ngithub.com/tendermint/tendermint/consensus.(*State).handleCompleteProposal(0x400065ea88, 0x3)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:2001 +0x2d8\ngithub.com/tendermint/tendermint/consensus.(*State).handleMsg(0x400065ea88, {{0x2b30a00, 0x400143e048}, {0x40002a61b0, 0x28}})\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:856 +0x1c8\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine(0x400065ea88, 0x0)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:782 +0x2c4\ncreated by github.com/tendermint/tendermint/consensus.(*State).OnStart in goroutine 169\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:391 +0x110\n"
    2024-04-25 14:48:24 6:48PM ERR CONSENSUS FAILURE!!! err="+2/3 committed an invalid block: wrong Block.Header.Version. Expected {11 1}, got {11 2}" module=consensus stack="goroutine 214 [running]:\nruntime/debug.Stack()\n\t/usr/local/go/src/runtime/debug/stack.go:24 +0x64\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine.func2()\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:746 +0x44\npanic({0x1b91180?, 0x400153b240?})\n\t/usr/local/go/src/runtime/panic.go:770 +0x124\ngithub.com/tendermint/tendermint/consensus.(*State).finalizeCommit(0x400065ea88, 0x3)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1637 +0xd30\ngithub.com/tendermint/tendermint/consensus.(*State).tryFinalizeCommit(0x400065ea88, 0x3)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1606 +0x26c\ngithub.com/tendermint/tendermint/consensus.(*State).handleCompleteProposal(0x400065ea88, 0x3)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:2001 +0x2d8\ngithub.com/tendermint/tendermint/consensus.(*State).handleMsg(0x400065ea88, {{0x2b30a00, 0x400143e048}, {0x40002a61b0, 0x28}})\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:856 +0x1c8\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine(0x400065ea88, 0x0)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:782 +0x2c4\ncreated by github.com/tendermint/tendermint/consensus.(*State).OnStart in goroutine 169\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:391 +0x110\n"

    then it is likely that the network has upgraded to a new app version but your consensus node was not prepared for the upgrade. To fix this, you'll need to:

    1. Remove DBs from your CELESTIA_HOME directory via: celestia-appd tendermint reset-state.
    2. Remove the data/application.db inside your CELESTIA_HOME directory.
    3. Download the latest binary for your network.
    4. Restart your consensus node with the relevant --v2-upgrade-height for the network you're running on.
    + + + + \ No newline at end of file diff --git a/nodes/decide-node.html b/nodes/decide-node.html new file mode 100644 index 00000000000..0ec0632ecdf --- /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!

    Light node

    It is highly recommended to get started with running a data availability light node.

    You can also play around with the Data Availability API in this tutorial for posting and retrieving data with a light node.

    Other DA nodes

    Depending on your use case, you also may want to run a bridge node or a full DA node.

    Consensus node

    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..73bd6594973 --- /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.16.0 \
      +    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.16.0 \
      +    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.16.0 \
      +    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.16.0 \
      +    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.16.0 \
      +    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.16.0 \
      +    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.16.0 \
    +    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.16.0 \
    +    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.16.0 \
    +    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.16.0 \
    +    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.16.0 \
    +    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.16.0 \
    +    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.16.0 \
    +    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.16.0 \
    +    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.16.0 \
    +    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.16.0 \
    +    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.16.0 \
    +    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.16.0 \
    +    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..9873c02741b --- /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.23.0"
      ver="1.23.0"
      bash
      ver="1.23.0"
      ver="1.23.0"
      bash
      ver="1.23.0"
      ver="1.23.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

      TIP

      Use echo $SHELL to figure out what type of shell you are using!

    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-storage-node.html b/nodes/full-storage-node.html new file mode 100644 index 00000000000..e0a5995ca6f --- /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 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: 16 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 Mainnet Beta, Mocha testnet or Arabica devnet, start your node.

    Connecting to a core endpoint with --core.ip string provides the light node with access to state queries (reading balances, submitting transactions, and other state-related queries).

    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.keyname <name-of-custom-key> \
    celestia full start --core.ip <URI> \
    +  --keyring.keyname <name-of-custom-key> \
    sh
    celestia full start --core.ip <URI> \
    +  --keyring.keyname <name-of-custom-key> \
    +  --p2p.network mocha
    celestia full start --core.ip <URI> \
    +  --keyring.keyname <name-of-custom-key> \
    +  --p2p.network mocha
    sh
    celestia full start --core.ip <URI> \
    +  --keyring.keyname <name-of-custom-key> \
    +  --p2p.network arabica
    celestia full start --core.ip <URI> \
    +  --keyring.keyname <name-of-custom-key> \
    +  --p2p.network arabica

    Optional: Migrate node id to another server

    To migrate a full storage node ID:

    1. You need to back up two files located in the celestia-full node directory at the correct path (default: ~/.celestia-full/keys).
    2. Upload the files to the new server and start the node.

    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/ibc-relayer.html b/nodes/ibc-relayer.html new file mode 100644 index 00000000000..84eb6e57f7b --- /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..d4d1d5dd4ad --- /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..1cbf1992b96 --- /dev/null +++ b/nodes/light-node.html @@ -0,0 +1,67 @@ + + + + + + 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: 100 GB SSD Storage
    • Bandwidth: 56 Kbps for Download/56 Kbps for Upload

    Quickstart: Run a light node in your browser

    The easiest way to run a Celestia light node is with Lumina.rs in your browser.

    Lumina.rs in browser

    You can also run Lumina on the first decentralized block explorer, Celenium.

    Celenium running a light node with Lumina.rs

    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.

    sh
    celestia light start --core.ip rpc.celestia.pops.one --p2p.network celestia
    celestia light start --core.ip rpc.celestia.pops.one --p2p.network celestia
    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
    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

    Tip: you can replace the core.ip with a consensus node RPC endpoint from Mainnet Beta, Mocha testnet, or Arabica devnet.

    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.keyname my_celes_key \
    +    --core.ip consensus.lunaroasis.net
    celestia light start --keyring.keyname my_celes_key \
    +    --core.ip consensus.lunaroasis.net
    sh
    celestia light start --keyring.keyname my_celes_key \
    +    --core.ip rpc-mocha.pops.one --p2p.network mocha
    celestia light start --keyring.keyname my_celes_key \
    +    --core.ip rpc-mocha.pops.one --p2p.network mocha
    sh
    celestia light start --keyring.keyname my_celes_key \
    +    --core.ip validator-1.celestia-arabica-11.com \
    +    --p2p.network arabica
    celestia light start --keyring.keyname 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.keyname <name-of-custom-key> \
    celestia light start --core.ip <URI> \
    +    --keyring.keyname <name-of-custom-key> \
    sh
    celestia light start --core.ip <URI> \
    +    --keyring.keyname <name-of-custom-key> \
    +    --p2p.network arabica
    celestia light start --core.ip <URI> \
    +    --keyring.keyname <name-of-custom-key> \
    +    --p2p.network arabica
    sh
    celestia light start --core.ip <URI> \
    +    --keyring.keyname <name-of-custom-key> \
    +    --p2p.network mocha
    celestia light start --core.ip <URI> \
    +    --keyring.keyname <name-of-custom-key> \
    +    --p2p.network mocha

    Optional: Migrate node id to another server

    To migrate a light node ID:

    1. You need to back up two files located in the celestia-light node directory at the correct path (default: ~/.celestia-light/keys).
    2. Upload the files to the new server and start the node.

    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..c6b486cbf7c --- /dev/null +++ b/nodes/mainnet.html @@ -0,0 +1,48 @@ + + + + + + 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 Beta 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.16.0
    celestia-appv2.1.2

    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.

    See the code in celestia-app and celestia-node.

    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:

    Production RPC endpoints

    These RPC providers are meant to be used in production environments.

    ProviderURL
    NewMetrichttps://app.newmetric.xyz/start
    NumiaFor RPC access: https://docs.numia.xyz/overview/rpc-api-access
    NumiaFor data warehouse access: https://docs.numia.xyz/overview/sql-access/chains/celestia
    Grovehttps://www.grove.city/

    WARNING

    Do not rely on the free community endpoints listed below for production deployments. Production deployments should rely on service providers with SLAs or your own node.

    Consensus nodes

    Community consensus RPC endpoints

    WARNING

    Do not rely on the free community endpoints listed below for production deployments. Production deployments should rely on service providers with SLAs.

    • 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
    • rpc-celestia.staker.space
    • celestia-rpc.noders.services
    • celestia.moonli.me
    • celestia-mainnet-rpc.itrocket.net:443
    • rpc.celestia.mainnet.dteam.tech:443

    Community 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
    • celestia-api.chainode.tech
    • api-celestia.staker.space
    • celestia-api.noders.services
    • celestia.moonli.me/api
    • celestia-mainnet-api.itrocket.net:443
    • api.celestia.mainnet.dteam.tech:443

    Community 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
    • grpc-celestia.staker.space
    • celestia-grpc.noders.services:11090
    • celestia-mainnet-grpc.itrocket.net:443
    • grpc.celestia.mainnet.dteam.tech:28090

    Community WebSocket endpoints

    • wss://celestia-ws.chainode.tech:33373/websocket
    • wss://celestia-mainnet-ws.itrocket.net:443/websocket
    • wss://rpc.celestia.mainnet.dteam.tech:443/websocket

    Data availability nodes

    Community Data availability (DA) RPC endpoints for bridge node sync

    These RPC endpoints allow bridge nodes to sync blocks from the Celestia network. For users, they will need to provide a –core.ip string from a consensus node’s URL or IP that populates a default RPC port at 26657 to their respective DA node.

    Community Data availability (DA) gRPC endpoints for state access

    These gRPC endpoints for DA nodes 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 a default gRPC port at 9090 to their respective DA node.

    TIP

    bash
    celestia <da_type> start --core.ip <url> -–core.grpc.port <port>
    celestia <da_type> start --core.ip <url> -–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 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
    • celestia-mainnet-consensus.itrocket.net
      • gRPC: port 9090
      • RPC: port 26657
    • rpc.celestia.mainnet.dteam.tech
      • gRPC: port 28090
      • RPC: 28657
    • celestia-consensus-mainnet.noders.services
      • gRPC: port 9080
      • RPC: port 26557

    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

    There are a few ways to stay informed about network upgrades on Mainnet Beta:

    See the network upgrade process page to learn more about specific upgrades like the Lemongrass network upgrade.

    + + + + \ No newline at end of file diff --git a/nodes/mocha-testnet.html b/nodes/mocha-testnet.html new file mode 100644 index 00000000000..c6f4f11a6f7 --- /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.16.0
    celestia-appv2.1.2

    RPC for DA bridge, full, and light nodes

    Production RPC endpoints

    These RPC providers are meant to be used in production environments.

    ProviderURL
    NewMetrichttps://app.newmetric.xyz/start
    NumiaFor RPC access: https://docs.numia.xyz/overview/rpc-api-access
    NumiaFor data warehouse access: https://docs.numia.xyz/overview/sql-access/chains/celestia
    Grovehttps://www.grove.city/

    WARNING

    Do not rely on the free community endpoints listed below for production deployments. Production deployments should rely on service providers with SLAs or your own node.

    Community Data availability (DA) RPC endpoints for bridge node sync

    These RPC endpoints allow bridge nodes to sync blocks from the Celestia network. For users, they will need to provide a –core.ip string from a consensus node’s URL or IP that populates a default RPC port at 26657 to their respective DA node.

    Community Data availability (DA) gRPC endpoints for state access

    These gRPC endpoints for DA nodes 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 a default gRPC port at 9090 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 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
    • celestia-testnet-consensus.itrocket.net
      • RPC port: 26657
      • gRPC port: 9090
    • rpc-celestia-testnet.cryptech.com.ua
      • gRPC: grpc-celestia-testnet.cryptech.com.ua:443
    • rpc.celestia.testnet.dteam.tech:443
      • gRPC: grpc.celestia.testnet.dteam.tech:27090
    • celestia-consensus-testnet.noders.services
      • RPC port: 26357
      • gRPC port: 9070

    Community 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
    • 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/
    • rpc-1.testnet.celestia.nodes.guru
    • rpc-2.testnet.celestia.nodes.guru
    • celestia-testnet-rpc.itrocket.net:443
    • rpc-celestia-testnet.cryptech.com.ua:443
    • rpc.celestia.testnet.dteam.tech:443

    Community 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.

    Community 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
    • 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
    • grpc-1.testnet.celestia.nodes.guru:10790
    • grpc-2.testnet.celestia.nodes.guru:10790
    • celestia-testnet-grpc.itrocket.net:443
    • grpc-celestia-testnet.cryptech.com.ua:443
    • grpc.celestia.testnet.dteam.tech:27090

    Community 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.

    Analytics

    The following websites provide analytics for Mocha Testnet:

    Explorers

    There are several explorers you can use for Mocha:

    Network upgrades

    There are a few ways to stay informed about network upgrades on Mocha testnet:

    See the network upgrade process page to learn more about specific upgrades like the Lemongrass network upgrade.

    + + + + \ No newline at end of file diff --git a/nodes/network-upgrade-process.html b/nodes/network-upgrade-process.html new file mode 100644 index 00000000000..9dd79d68186 --- /dev/null +++ b/nodes/network-upgrade-process.html @@ -0,0 +1,45 @@ + + + + + + Celestia network upgrade process | Celestia Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    Skip to content

    Celestia network upgrade 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 network upgrade, which can be breaking or non-breaking. During planned network upgrades, the Celestia Labs team will coordinate with the validators to prepare for the upcoming network upgrade.

    Breaking network upgrades 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.

    Non-breaking network upgrades are backward-compatible and require less coordination.

    General process

    The general process can be broken down into several components:

    • Network upgrade 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 Labs team in order for validators to upgrade their nodes to the new binary.
    • A block number for when the breaking network upgrade. Even if validators upgrade their binary to be network upgrade 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, which happens on testnets first prior to activating on Mainnet Beta in order to ensure the network can upgrade securely.

    The two testnets where network upgrades are deployed are:

    Lemongrass network upgrade

    The Lemongrass network upgrade is the first consensus layer breaking change since Celestia's Mainnet Beta genesis block. The Lemongrass network upgrade includes all of the CIPs listed in CIP-17. The Lemongrass network upgrade will be executed on Arabica, then Mocha, then Mainnet Beta. The network upgrade will take place at an "upgrade height" that will be coordinated offline on a per-network basis. The upgrade heights will be announced in advance (see network upgrades channels) to give node operators time to download and start a compatible binary prior to the upgrade height.

    • If you are a consensus node or validator operator: you will need to download and run a celestia-app binary >= v2.0.0 prior to the --v2-upgrade-height to remain on the canonical chain.
    • If you are a DA node operator, you will need to download and run a compatible celestia-node binary >= v0.16.0-rc0 prior to the upgrade height.
    NetworkChain IDDate and approximate time--v2-upgrade-height
    Arabicaarabica-112024/08/19 @ 14:00 UTC1751707
    Mochamocha-42024/08/28 @ 14:00 UTC2585031
    Mainnet Betacelestia2024/09/18 @ 14:00 UTC2371495

    WARNING

    You do not need to use a tool like cosmovisor to upgrade the binary at the upgrade height. Please upgrade your binary several blocks before the upgrade height.

    + + + + \ No newline at end of file diff --git a/nodes/overview.html b/nodes/overview.html new file mode 100644 index 00000000000..ca59fb06f0a --- /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.
    • 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.

    Data availability nodes

    Node typeMemoryCPUDiskBandwidth
    Light node500 MB RAMSingle core100 GB SSD56 Kbps
    Bridge node16 GB RAM6 cores10 TB SSD1 Gbps
    Full storage node16 GB RAMQuad-core10 TB SSD1 Gbps

    Consensus nodes

    Node typeMemoryCPUDiskBandwidth
    Validator16 GB RAM8 cores2 TB SSD1 Gbps
    Consensus node16 GB RAMQuad-core2 TB SSD1 Gbps

    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..e80489b770a --- /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 Beta 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.16.0
    celestia-appv2.1.2

    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.16.0
    celestia-appv2.1.2

    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 breaking network upgrades in coordination with the validator community on Mocha.

    Compatible software versions for Mocha testnet

    SoftwareVersion
    celestia-nodev0.16.0
    celestia-appv2.1.2

    Network upgrades

    There are a few ways to stay informed about network upgrades:

    See the network upgrade process page to learn more about specific upgrades like the Lemongrass network upgrade.

    + + + + \ No newline at end of file diff --git a/nodes/quick-start.html b/nodes/quick-start.html new file mode 100644 index 00000000000..30eb1fa84e5 --- /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..3e00f4fdc72 --- /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 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..f887948d41c --- /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: 16 GB RAM
    • CPU: 8 cores
    • Disk: 2 TB 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 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.

    Using an RPC of your own, or one from Mainnet Beta, Mocha testnet or Arabica devnet, initialize your node.

    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

    If you are running celestia-app v1.x.x:

    sh
    celestia-appd start
    celestia-appd start

    If you are running celestia-app >= v2.0.0: then you'll want to start the node with a --v2-upgrade-height that is dependent on the network. The --v2-upgrade-height flag is only needed during the v2 upgrade height so after your node has executed the upgrade (e.g. you see the log upgraded from app version 1 to 2), you don't need to provide this flag for future celestia-appd start invocations.

    sh
    celestia-appd start --v2-upgrade-height 2371495
    celestia-appd start --v2-upgrade-height 2371495
    sh
    celestia-appd start --v2-upgrade-height 2585031
    celestia-appd start --v2-upgrade-height 2585031
    sh
    celestia-appd start --v2-upgrade-height 1751707
    celestia-appd start --v2-upgrade-height 1751707

    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 consensus node page.

    FAQ

    +2/3 committed an invalid block: wrong Block.Header.Version

    If you encounter an error like:

    bash
    2024-04-25 14:48:24 6:48PM ERR CONSENSUS FAILURE!!! err="+2/3 committed an invalid block: wrong Block.Header.Version. Expected {11 1}, got {11 2}" module=consensus stack="goroutine 214 [running]:\nruntime/debug.Stack()\n\t/usr/local/go/src/runtime/debug/stack.go:24 +0x64\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine.func2()\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:746 +0x44\npanic({0x1b91180?, 0x400153b240?})\n\t/usr/local/go/src/runtime/panic.go:770 +0x124\ngithub.com/tendermint/tendermint/consensus.(*State).finalizeCommit(0x400065ea88, 0x3)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1637 +0xd30\ngithub.com/tendermint/tendermint/consensus.(*State).tryFinalizeCommit(0x400065ea88, 0x3)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1606 +0x26c\ngithub.com/tendermint/tendermint/consensus.(*State).handleCompleteProposal(0x400065ea88, 0x3)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:2001 +0x2d8\ngithub.com/tendermint/tendermint/consensus.(*State).handleMsg(0x400065ea88, {{0x2b30a00, 0x400143e048}, {0x40002a61b0, 0x28}})\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:856 +0x1c8\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine(0x400065ea88, 0x0)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:782 +0x2c4\ncreated by github.com/tendermint/tendermint/consensus.(*State).OnStart in goroutine 169\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:391 +0x110\n"
    2024-04-25 14:48:24 6:48PM ERR CONSENSUS FAILURE!!! err="+2/3 committed an invalid block: wrong Block.Header.Version. Expected {11 1}, got {11 2}" module=consensus stack="goroutine 214 [running]:\nruntime/debug.Stack()\n\t/usr/local/go/src/runtime/debug/stack.go:24 +0x64\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine.func2()\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:746 +0x44\npanic({0x1b91180?, 0x400153b240?})\n\t/usr/local/go/src/runtime/panic.go:770 +0x124\ngithub.com/tendermint/tendermint/consensus.(*State).finalizeCommit(0x400065ea88, 0x3)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1637 +0xd30\ngithub.com/tendermint/tendermint/consensus.(*State).tryFinalizeCommit(0x400065ea88, 0x3)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:1606 +0x26c\ngithub.com/tendermint/tendermint/consensus.(*State).handleCompleteProposal(0x400065ea88, 0x3)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:2001 +0x2d8\ngithub.com/tendermint/tendermint/consensus.(*State).handleMsg(0x400065ea88, {{0x2b30a00, 0x400143e048}, {0x40002a61b0, 0x28}})\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:856 +0x1c8\ngithub.com/tendermint/tendermint/consensus.(*State).receiveRoutine(0x400065ea88, 0x0)\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:782 +0x2c4\ncreated by github.com/tendermint/tendermint/consensus.(*State).OnStart in goroutine 169\n\t/go/pkg/mod/github.com/celestiaorg/celestia-core@v1.35.0-tm-v0.34.29/consensus/state.go:391 +0x110\n"

    then it is likely that the network has upgraded to a new app version but your consensus node was not prepared for the upgrade. To fix this, you'll need to update your binary to the latest version and restart your node with the relevant --v2-upgrade-height for the network you're running on. If your node still can't sync to the tip of the chain after the above steps, consider a celestia-appd tendermint reset-state to reset your node and start syncing from the genesis block.

    1. [Optional] Back up your validator keys.
    2. [Optional] Back up the data/priv_validator_state.json inside your CELESTIA_HOME directory.
    3. Remove DBs from your CELESTIA_HOME directory via: celestia-appd tendermint reset-state.
    4. Remove the data/application.db inside your CELESTIA_HOME directory.
    5. Download the latest binary for your network.
    6. Restart your consensus node with the relevant --v2-upgrade-height for the network you're running on.
    + + + + \ No newline at end of file diff --git a/sitemap.xml b/sitemap.xml new file mode 100644 index 00000000000..9ca22d3fd26 --- /dev/null +++ b/sitemap.xml @@ -0,0 +1 @@ +https://docs.celestia.org/README2024-09-16T14:18:42.000Zhttps://docs.celestia.org/community/coc2023-10-26T17:10:53.000Zhttps://docs.celestia.org/community/foundation-delegation-program2024-09-16T14:18:42.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/speaker-list2023-10-23T20:05:00.000Zhttps://docs.celestia.org/developers/arbitrum-bridge2024-09-16T12:52:16.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-08-05T21:03:33.000Zhttps://docs.celestia.org/developers/blobstream-contracts2024-09-10T08:12:44.000Zhttps://docs.celestia.org/developers/blobstream-offchain2024-09-10T08:12:44.000Zhttps://docs.celestia.org/developers/blobstream-proof-queries2024-09-16T14:18:42.000Zhttps://docs.celestia.org/developers/blobstream-rollups2024-07-24T19:57:28.000Zhttps://docs.celestia.org/developers/blobstream-x-deploy2024-07-24T19:53:40.000Zhttps://docs.celestia.org/developers/blobstream-x-requesting-data-commitment-ranges2024-09-16T14:18:42.000Zhttps://docs.celestia.org/developers/blobstream2024-09-10T08:12:44.000Zhttps://docs.celestia.org/developers/blobstreamx2024-09-16T12:52:16.000Zhttps://docs.celestia.org/developers/bubs-testnet2024-09-16T12:52:16.000Zhttps://docs.celestia.org/developers/build-whatever2024-09-13T12:02:38.000Zhttps://docs.celestia.org/developers/celestia-node-key2024-08-23T17:04:31.000Zhttps://docs.celestia.org/developers/ethereum-fallback2024-07-31T15:47:47.000Zhttps://docs.celestia.org/developers/feegrant-for-blobs2024-09-09T08:43:01.000Zhttps://docs.celestia.org/developers/golang-client-tutorial2024-08-21T16:07:21.000Zhttps://docs.celestia.org/developers/integrate-celestia2024-06-17T15:09:06.000Zhttps://docs.celestia.org/developers/intro-to-op-stack2024-09-16T12:52:16.000Zhttps://docs.celestia.org/developers/multiaccounts2024-09-09T08:43:50.000Zhttps://docs.celestia.org/developers/node-api2023-11-06T16:30:46.000Zhttps://docs.celestia.org/developers/node-tutorial2024-09-18T21:06:12.000Zhttps://docs.celestia.org/developers/optimism-devnet2024-07-31T15:47:47.000Zhttps://docs.celestia.org/developers/optimism2024-09-17T08:08:02.000Zhttps://docs.celestia.org/developers/prompt-scavenger2024-06-18T11:44:12.000Zhttps://docs.celestia.org/developers/rust-client-tutorial2024-06-18T22:59:48.000Zhttps://docs.celestia.org/developers/sp1-blobstream-deploy2024-09-16T12:52:16.000Zhttps://docs.celestia.org/developers/submit-data2024-09-16T09:35:27.000Zhttps://docs.celestia.org/developers/transaction-resubmission2024-09-16T14:18:42.000Zhttps://docs.celestia.org/developers/wallets2024-04-15T14:35:09.000Zhttps://docs.celestia.org/2024-09-18T21:06:12.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-06-17T19:34:44.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-09-10T12:29:07.000Zhttps://docs.celestia.org/learn/staking2024-09-16T12:52:16.000Zhttps://docs.celestia.org/learn/tia2023-12-11T18:21:14.000Zhttps://docs.celestia.org/nodes/arabica-devnet2024-09-16T13:44:41.000Zhttps://docs.celestia.org/nodes/bridge-node2024-08-28T19:24:46.000Zhttps://docs.celestia.org/nodes/celestia-app-commands2024-09-16T12:52:16.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-09-16T12:52:16.000Zhttps://docs.celestia.org/nodes/celestia-app-upgrade-monitor2023-12-14T20:10:16.000Zhttps://docs.celestia.org/nodes/celestia-app-vesting2024-09-16T14:18:42.000Zhttps://docs.celestia.org/nodes/celestia-app-wallet2024-09-18T21:06:12.000Zhttps://docs.celestia.org/nodes/celestia-app2024-09-03T13:25:55.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-09-16T14:18:42.000Zhttps://docs.celestia.org/nodes/celestia-node-trusted-hash2024-08-07T13:40:28.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/consensus-node2024-09-18T22:14:42.000Zhttps://docs.celestia.org/nodes/decide-node2024-09-16T12:52:16.000Zhttps://docs.celestia.org/nodes/docker-images2024-09-16T14:18:42.000Zhttps://docs.celestia.org/nodes/environment2024-08-08T15:40:20.000Zhttps://docs.celestia.org/nodes/full-storage-node2024-09-16T14:18:42.000Zhttps://docs.celestia.org/nodes/ibc-relayer2024-09-18T21:06:12.000Zhttps://docs.celestia.org/nodes/instantiate-testnet2024-03-29T20:40:58.000Zhttps://docs.celestia.org/nodes/light-node2024-09-16T14:18:42.000Zhttps://docs.celestia.org/nodes/mainnet2024-09-16T14:18:42.000Zhttps://docs.celestia.org/nodes/mocha-testnet2024-09-16T13:44:41.000Zhttps://docs.celestia.org/nodes/network-upgrade-process2024-09-19T13:05:31.000Zhttps://docs.celestia.org/nodes/overview2024-08-23T16:43:03.000Zhttps://docs.celestia.org/nodes/participate2024-09-16T14:18:42.000Zhttps://docs.celestia.org/nodes/quick-start2023-10-27T14:33:58.000Zhttps://docs.celestia.org/nodes/systemd2024-08-05T21:03:33.000Zhttps://docs.celestia.org/nodes/validator-node2024-09-18T22:14:42.000Z \ No newline at end of file