Compare commits
	
		
			58 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 293bb63ed8 | ||
|  | 8f8fb720af | ||
|  | 19f414d843 | ||
|  | eaca1c3170 | ||
|  | 9fc75925f9 | ||
|  | b5098ac87c | ||
|  | e23aec9728 | ||
|  | 57d490c84f | ||
|  | aa8c9f6a98 | ||
|  | 57772dc73d | ||
|  | 21706108e8 | ||
|  | 50d0caf00f | ||
|  | 2739332306 | ||
|  | c85c4699aa | ||
|  | 81add4d6bf | ||
|  | 8e31eeb696 | ||
|  | e1ce8b37ff | ||
|  | 3f831c05f5 | ||
|  | f0d7ce6bb6 | ||
|  | 6ba95b2545 | ||
|  | 6818e68542 | ||
|  | 43659d7deb | ||
|  | f24d8e7d2d | ||
|  | e10fe5e125 | ||
|  | 0f8c9ab1c4 | ||
|  | 8a9a9cb991 | ||
|  | 44208ffa67 | ||
|  | 5df0478fa3 | ||
|  | d52567933e | ||
|  | a32cdb9f4d | ||
|  | eacd8d986c | ||
|  | 1d32603b49 | ||
|  | 8c6f7ee5a4 | ||
|  | be482eed3f | ||
|  | 6e1c53cb0f | ||
|  | af92f205cf | ||
|  | 87047b08c8 | ||
|  | e282161872 | ||
|  | 01b1e287ed | ||
|  | d7fd1fa467 | ||
|  | bfa34cd494 | ||
|  | 915835e224 | ||
|  | 659332e7ac | ||
|  | 272986c6ac | ||
|  | 4d8ab45c56 | ||
|  | 932ae86d47 | ||
|  | 756e6334b0 | ||
|  | 4e6eca9748 | ||
|  | d9e37eb30c | ||
|  | 04d1b35926 | ||
|  | d13d609050 | ||
|  | 20426cf251 | ||
|  | 4a220d7c8e | ||
|  | 436eab41ca | ||
|  | c8472d0a96 | ||
|  | 1a7db9c17e | ||
|  | b468d9f17c | ||
|  | 41cf1d7d23 | 
							
								
								
									
										42
									
								
								.appveyor.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								.appveyor.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | |||||||
|  | version: '{build}' | ||||||
|  |  | ||||||
|  | branches: | ||||||
|  |   only: | ||||||
|  |   - master | ||||||
|  |   - /^v[0-9.]+\.[0-9.]+/ | ||||||
|  |  | ||||||
|  | cache: | ||||||
|  |   - '%USERPROFILE%\.cargo' | ||||||
|  |   - '%APPVEYOR_BUILD_FOLDER%\target' | ||||||
|  |  | ||||||
|  | clone_folder: d:\projects\solana | ||||||
|  |  | ||||||
|  | build_script: | ||||||
|  |   - bash ci/publish-tarball.sh | ||||||
|  |  | ||||||
|  | notifications: | ||||||
|  |   - provider: Slack | ||||||
|  |     incoming_webhook: | ||||||
|  |       secure: GJsBey+F5apAtUm86MHVJ68Uqa6WN1SImcuIc4TsTZrDhA8K1QWUNw9FFQPybUWDyOcS5dly3kubnUqlGt9ux6Ad2efsfRIQYWv0tOVXKeY= | ||||||
|  |     channel: ci-status | ||||||
|  |     on_build_success: false | ||||||
|  |     on_build_failure: true | ||||||
|  |     on_build_status_changed: true | ||||||
|  |  | ||||||
|  | deploy: | ||||||
|  |   - provider: S3 | ||||||
|  |     access_key_id: | ||||||
|  |       secure: fTbJl6JpFebR40J7cOWZ2mXBa3kIvEiXgzxAj6L3N7A= | ||||||
|  |     secret_access_key: | ||||||
|  |       secure: vItsBXb2rEFLvkWtVn/Rcxu5a5+2EwC+b7GsA0waJy9hXh6XuBAD0lnHd9re3g/4 | ||||||
|  |     bucket: release.solana.com | ||||||
|  |     region: us-west-1 | ||||||
|  |     set_public: true | ||||||
|  |  | ||||||
|  |   - provider: GitHub | ||||||
|  |     auth_token: | ||||||
|  |       secure: 81fEmPZ0cV1wLtNuUrcmtgxKF6ROQF1+/ft5m+fHX21z6PoeCbaNo8cTyLioWBj7 | ||||||
|  |     draft: false | ||||||
|  |     prerelease: false | ||||||
|  |     on: | ||||||
|  |       appveyor_repo_tag: true | ||||||
							
								
								
									
										5
									
								
								.buildkite/env/secrets.ejson
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.buildkite/env/secrets.ejson
									
									
									
									
										vendored
									
									
								
							| @@ -7,6 +7,9 @@ | |||||||
|       "GITHUB_TOKEN": "EJ[1:yGpTmjdbyjW2kjgIHkFoJv7Ue7EbUvUbqHyw6anGgWg=:Vq2dkGTOzfEpRht0BAGHFp/hDogMvXJe:tFXHg1epVt2mq9hkuc5sRHe+KAnVREi/p8S+IZu67XRyzdiA/nGak1k860FXYuuzuaE0QWekaEc=]", |       "GITHUB_TOKEN": "EJ[1:yGpTmjdbyjW2kjgIHkFoJv7Ue7EbUvUbqHyw6anGgWg=:Vq2dkGTOzfEpRht0BAGHFp/hDogMvXJe:tFXHg1epVt2mq9hkuc5sRHe+KAnVREi/p8S+IZu67XRyzdiA/nGak1k860FXYuuzuaE0QWekaEc=]", | ||||||
|       "INFLUX_DATABASE": "EJ[1:yGpTmjdbyjW2kjgIHkFoJv7Ue7EbUvUbqHyw6anGgWg=:5KI9WBkXx3R/W4m256mU5MJOE7N8aAT9:Cb8QFELZ9I60t5zhJ9h55Kcs]", |       "INFLUX_DATABASE": "EJ[1:yGpTmjdbyjW2kjgIHkFoJv7Ue7EbUvUbqHyw6anGgWg=:5KI9WBkXx3R/W4m256mU5MJOE7N8aAT9:Cb8QFELZ9I60t5zhJ9h55Kcs]", | ||||||
|       "INFLUX_PASSWORD": "EJ[1:yGpTmjdbyjW2kjgIHkFoJv7Ue7EbUvUbqHyw6anGgWg=:hQRMpLCrav+OYkNphkeM4hagdVoZv5Iw:AUO76rr6+gF1OLJA8ZLSG8wHKXgYCPNk6gRCV8rBhZBJ4KwDaxpvOhMl7bxxXG6jol7v4aRa/Lk=]", |       "INFLUX_PASSWORD": "EJ[1:yGpTmjdbyjW2kjgIHkFoJv7Ue7EbUvUbqHyw6anGgWg=:hQRMpLCrav+OYkNphkeM4hagdVoZv5Iw:AUO76rr6+gF1OLJA8ZLSG8wHKXgYCPNk6gRCV8rBhZBJ4KwDaxpvOhMl7bxxXG6jol7v4aRa/Lk=]", | ||||||
|       "INFLUX_USERNAME": "EJ[1:yGpTmjdbyjW2kjgIHkFoJv7Ue7EbUvUbqHyw6anGgWg=:R7BNmQjfeqoGDAFTJu9bYTGHol2NgnYN:Q2tOT/EBcFvhFk+DKLKmVU7tLCpVC3Ui]" |       "INFLUX_USERNAME": "EJ[1:yGpTmjdbyjW2kjgIHkFoJv7Ue7EbUvUbqHyw6anGgWg=:R7BNmQjfeqoGDAFTJu9bYTGHol2NgnYN:Q2tOT/EBcFvhFk+DKLKmVU7tLCpVC3Ui]", | ||||||
|  |       "SOLANA_INSTALL_UPDATE_MANIFEST_KEYPAIR_x86_64_unknown_linux_gnu": "EJ[1:yGpTmjdbyjW2kjgIHkFoJv7Ue7EbUvUbqHyw6anGgWg=:Egc2dMrHDU0NcZ71LwGv/V66shUhwYUE:04VoIb8CKy7KYhQ5W4cEW9SDKZltxWBL5Hob106lMBbUOD/yUvKYcG3Ep8JfTMwO3K8zowW5HpU/IdGoilX0XWLiJJ6t+p05WWK0TA16nOEtwrEG+UK8wm3sN+xCO20i4jDhpNpgg3FYFHT5rKTHW8+zaBTNUX/SFxkN67Lm+92IM28CXYE43SU1WV6H99hGFFVpTK5JVM3JuYU1ex/dHRE+xCzTr4MYUB/F+nGoNFW8HUDV/y0e1jxT9to3x0SmnytEEuk+5RUzFuEt9cKNFeNml3fOCi4qL+sfj/Y5pjH9xDiUxsvH/8NL35jbLP244aFHgWcp]", | ||||||
|  |       "SOLANA_INSTALL_UPDATE_MANIFEST_KEYPAIR_x86_64_apple_darwin": "EJ[1:yGpTmjdbyjW2kjgIHkFoJv7Ue7EbUvUbqHyw6anGgWg=:NeOxSoWCvXB9AL4H6OK26l/7bmsKd/oz:Ijfoxtvk2CHlN1ZXHup3Gg/914kbbAkEGWJfvozA8UIe+aUzUObMyTrKkVOeNAH8Q8YH9tNzk7RRnrTcpnzeCCBLlWcVEeruMxHox3mPRzmSeDLxtbzCl9VePlRO3T7jg90K5hW+ZAkd5J/WJNzpAcmr93ts/of3MbvGHSujId/efCTzJEcP6JInnBb8Vrj7TlgKbzUlnqpq1+NjYPSXN3maKa9pKeo2JWxZlGBMoy6QWUUY5GbYEylw9smwh1LJcHZjlaZNMuOl4gNKtaSr38IXQkAXaRUJDPAmPras00YObKzXU8RkTrP4EoP/jx5LPR7f]", | ||||||
|  |       "SOLANA_INSTALL_UPDATE_MANIFEST_KEYPAIR_x86_64_pc_windows_msvc": "EJ[1:yGpTmjdbyjW2kjgIHkFoJv7Ue7EbUvUbqHyw6anGgWg=:7t+56twjW+jR7fpFNNeRFLPd7E4lbmyN:JuviDpkQrfVcNUGRGsa2e/UhvH6tTYyk1s4cHHE5xZH1NByL7Kpqx36VG/+o1AUGEeSQdsBnKgzYdMoFYbO8o50DoRPc86QIEVXCupD6J9avxLFtQgOWgJp+/mCdUVXlqXiFs/vQgS/L4psrcKdF6WHd77BeUr6ll8DjH+9m5FC9Rcai2pXno6VbPpunHQ0oUdYzhFR64+LiRacBaefQ9igZ+nSEWDLqbaZSyfm9viWkijoVFTq8gAgdXXEh7g0QdxVE5T6bPristJhT6jWBhWunPUCDNFFErWIsbRGctepl4pbCWqh2hNTw9btSgVfeY6uGCOsdy9E=]" | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,18 +1,5 @@ | |||||||
| root: ./docs/src | root: ./book/src | ||||||
|  |  | ||||||
| structure: | structure: | ||||||
|     readme: introduction.md |     readme: introduction.md | ||||||
|     summary: SUMMARY.md |     summary: SUMMARY.md | ||||||
|  |  | ||||||
| redirects: |  | ||||||
|     wallet: ./wallet-guide/README.md |  | ||||||
|     wallet/app-wallets: ./wallet-guide/apps.md |  | ||||||
|     wallet/app-wallets/trust-wallet: ./wallet-guide/trust-wallet.md |  | ||||||
|     wallet/app-wallets/ledger-live:  ./wallet-guide/ledger-live.md |  | ||||||
|     wallet/cli-wallets:  ./wallet-guide/cli.md |  | ||||||
|     wallet/cli-wallets/paper-wallet:  ./paper-wallet/README.md |  | ||||||
|     wallet/cli-wallets/paper-wallet/paper-wallet-usage: ./paper-wallet/paper-wallet-usage.md |  | ||||||
|     wallet/cli-wallets/remote-wallet: ./hardware-wallets/README.md |  | ||||||
|     wallet/cli-wallets/remote-wallet/ledger: ./hardware-wallets/ledger.md |  | ||||||
|     wallet/cli-wallets/file-system-wallet: ./file-system-wallet/README.md |  | ||||||
|     wallet/support: ./wallet-guide/support.md |  | ||||||
|   | |||||||
							
								
								
									
										7
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,7 +1,6 @@ | |||||||
| /docs/html/ | /book/html/ | ||||||
| /docs/src/tests.ok | /book/src/tests.ok | ||||||
| /docs/src/cli/usage.md | /book/src/.gitbook/assets/*.svg | ||||||
| /docs/src/.gitbook/assets/*.svg |  | ||||||
| /farf/ | /farf/ | ||||||
| /solana-release/ | /solana-release/ | ||||||
| /solana-release.tar.bz2 | /solana-release.tar.bz2 | ||||||
|   | |||||||
							
								
								
									
										21
									
								
								.mergify.yml
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								.mergify.yml
									
									
									
									
									
								
							| @@ -19,30 +19,27 @@ pull_request_rules: | |||||||
|       label: |       label: | ||||||
|         add: |         add: | ||||||
|           - automerge |           - automerge | ||||||
|   - name: v1.0 backport |   - name: v0.22 backport | ||||||
|     conditions: |     conditions: | ||||||
|       - base=master |       - base=master | ||||||
|       - label=v1.0 |       - label=v0.22 | ||||||
|     actions: |     actions: | ||||||
|       backport: |       backport: | ||||||
|         ignore_conflicts: true |  | ||||||
|         branches: |         branches: | ||||||
|           - v1.0 |           - v0.22 | ||||||
|   - name: v1.1 backport |   - name: v0.23 backport | ||||||
|     conditions: |     conditions: | ||||||
|       - base=master |       - base=master | ||||||
|       - label=v1.1 |       - label=v0.23 | ||||||
|     actions: |     actions: | ||||||
|       backport: |       backport: | ||||||
|         ignore_conflicts: true |  | ||||||
|         branches: |         branches: | ||||||
|           - v1.1 |           - v0.23 | ||||||
|   - name: v1.2 backport |   - name: v0.24 backport | ||||||
|     conditions: |     conditions: | ||||||
|       - base=master |       - base=master | ||||||
|       - label=v1.2 |       - label=v0.24 | ||||||
|     actions: |     actions: | ||||||
|       backport: |       backport: | ||||||
|         ignore_conflicts: true |  | ||||||
|         branches: |         branches: | ||||||
|           - v1.2 |           - v0.24 | ||||||
|   | |||||||
| @@ -1,6 +1,5 @@ | |||||||
| os: | os: | ||||||
|   - osx |   - osx | ||||||
|   - windows |  | ||||||
|  |  | ||||||
| language: rust | language: rust | ||||||
| rust: | rust: | ||||||
|   | |||||||
| @@ -224,20 +224,21 @@ Inventing new terms is allowed, but should only be done when the term is widely | |||||||
| used and understood. Avoid introducing new 3-letter terms, which can be | used and understood. Avoid introducing new 3-letter terms, which can be | ||||||
| confused with 3-letter acronyms. | confused with 3-letter acronyms. | ||||||
|  |  | ||||||
| [Terms currently in use](docs/src/terminology.md) | [Terms currently in use](book/src/terminology.md) | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Design Proposals | ## Design Proposals | ||||||
|  |  | ||||||
| Solana's architecture is described by docs generated from markdown files in | Solana's architecture is described by a book generated from markdown files in | ||||||
| the `docs/src/` directory, maintained by an *editor* (currently @garious). To | the `book/src/` directory, maintained by an *editor* (currently @garious). To | ||||||
| add a design proposal, you'll need to include it in the | add a design proposal, you'll need to at least propose a change the content | ||||||
| [Accepted Design Proposals](https://docs.solana.com/proposals) | under the [Accepted Design | ||||||
| section of the Solana docs.  Here's the full process: | Proposals](https://docs.solana.com/book/v/master/proposals) chapter.  Here's | ||||||
|  | the full process: | ||||||
|  |  | ||||||
| 1. Propose a design by creating a PR that adds a markdown document to the | 1. Propose a design by creating a PR that adds a markdown document to the | ||||||
|    `docs/src/proposals` directory and references it from the [table of |    directory `book/src/` and references it from the [table of | ||||||
|    contents](docs/src/SUMMARY.md). Add any relevant *maintainers* to the PR |    contents](book/src/SUMMARY.md). Add any relevant *maintainers* to the PR | ||||||
|    review. |    review. | ||||||
| 2. The PR being merged indicates your proposed change was accepted and that the | 2. The PR being merged indicates your proposed change was accepted and that the | ||||||
|    maintainers support your plan of attack. |    maintainers support your plan of attack. | ||||||
|   | |||||||
							
								
								
									
										6853
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										6853
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -3,7 +3,6 @@ members = [ | |||||||
|     "bench-exchange", |     "bench-exchange", | ||||||
|     "bench-streamer", |     "bench-streamer", | ||||||
|     "bench-tps", |     "bench-tps", | ||||||
|     "accounts-bench", |  | ||||||
|     "banking-bench", |     "banking-bench", | ||||||
|     "chacha", |     "chacha", | ||||||
|     "chacha-cuda", |     "chacha-cuda", | ||||||
| @@ -11,8 +10,6 @@ members = [ | |||||||
|     "cli-config", |     "cli-config", | ||||||
|     "client", |     "client", | ||||||
|     "core", |     "core", | ||||||
|     "dos", |  | ||||||
|     "download-utils", |  | ||||||
|     "faucet", |     "faucet", | ||||||
|     "perf", |     "perf", | ||||||
|     "validator", |     "validator", | ||||||
| @@ -27,7 +24,6 @@ members = [ | |||||||
|     "logger", |     "logger", | ||||||
|     "log-analyzer", |     "log-analyzer", | ||||||
|     "merkle-tree", |     "merkle-tree", | ||||||
|     "streamer", |  | ||||||
|     "measure", |     "measure", | ||||||
|     "metrics", |     "metrics", | ||||||
|     "net-shaper", |     "net-shaper", | ||||||
| @@ -47,18 +43,13 @@ members = [ | |||||||
|     "archiver", |     "archiver", | ||||||
|     "archiver-lib", |     "archiver-lib", | ||||||
|     "archiver-utils", |     "archiver-utils", | ||||||
|     "remote-wallet", |  | ||||||
|     "runtime", |     "runtime", | ||||||
|     "sdk", |     "sdk", | ||||||
|     "sdk-c", |     "sdk-c", | ||||||
|     "scripts", |     "scripts", | ||||||
|     "stake-accounts", |  | ||||||
|     "stake-monitor", |  | ||||||
|     "sys-tuner", |     "sys-tuner", | ||||||
|     "transaction-status", |  | ||||||
|     "upload-perf", |     "upload-perf", | ||||||
|     "net-utils", |     "net-utils", | ||||||
|     "version", |  | ||||||
|     "vote-signer", |     "vote-signer", | ||||||
|     "cli", |     "cli", | ||||||
|     "rayon-threadlimit", |     "rayon-threadlimit", | ||||||
|   | |||||||
							
								
								
									
										70
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										70
									
								
								README.md
									
									
									
									
									
								
							| @@ -9,7 +9,60 @@ Blockchain Rebuilt for Scale | |||||||
| Solana™ is a new blockchain architecture built from the ground up for scale. The architecture supports | Solana™ is a new blockchain architecture built from the ground up for scale. The architecture supports | ||||||
| up to 710 thousand transactions per second on a gigabit network. | up to 710 thousand transactions per second on a gigabit network. | ||||||
|  |  | ||||||
| Read all about it at [Solana: Blockchain Rebuilt for Scale](https://docs.solana.com/v/master). | Disclaimer | ||||||
|  | === | ||||||
|  |  | ||||||
|  | All claims, content, designs, algorithms, estimates, roadmaps, specifications, and performance measurements described in this project are done with the author's best effort.  It is up to the reader to check and validate their accuracy and truthfulness.  Furthermore nothing in this project constitutes a solicitation for investment. | ||||||
|  |  | ||||||
|  | Introduction | ||||||
|  | === | ||||||
|  |  | ||||||
|  | It's possible for a centralized database to process 710,000 transactions per second on a standard gigabit network if the transactions are, on average, no more than 176 bytes. A centralized database can also replicate itself and maintain high availability without significantly compromising that transaction rate using the distributed system technique known as Optimistic Concurrency Control [\[H.T.Kung, J.T.Robinson (1981)\]](http://citeseerx.ist.psu.edu/viewdoc/summary?doi=10.1.1.65.4735). At Solana, we're demonstrating that these same theoretical limits apply just as well to blockchain on an adversarial network. The key ingredient? Finding a way to share time when nodes can't trust one-another. Once nodes can trust time, suddenly ~40 years of distributed systems research becomes applicable to blockchain! | ||||||
|  |  | ||||||
|  | > Perhaps the most striking difference between algorithms obtained by our method and ones based upon timeout is that using timeout produces a traditional distributed algorithm in which the processes operate asynchronously, while our method produces a globally synchronous one in which every process does the same thing at (approximately) the same time. Our method seems to contradict the whole purpose of distributed processing, which is to permit different processes to operate independently and perform different functions. However, if a distributed system is really a single system, then the processes must be synchronized in some way. Conceptually, the easiest way to synchronize processes is to get them all to do the same thing at the same time. Therefore, our method is used to implement a kernel that performs the necessary synchronization--for example, making sure that two different processes do not try to modify a file at the same time. Processes might spend only a small fraction of their time executing the synchronizing kernel; the rest of the time, they can operate independently--e.g., accessing different files. This is an approach we have advocated even when fault-tolerance is not required. The method's basic simplicity makes it easier to understand the precise properties of a system, which is crucial if one is to know just how fault-tolerant the system is. [\[L.Lamport (1984)\]](http://citeseerx.ist.psu.edu/viewdoc/summary?doi=10.1.1.71.1078) | ||||||
|  |  | ||||||
|  | Furthermore, and much to our surprise, it can be implemented using a mechanism that has existed in Bitcoin since day one. The Bitcoin feature is called nLocktime and it can be used to postdate transactions using block height instead of a timestamp. As a Bitcoin client, you'd use block height instead of a timestamp if you don't trust the network. Block height turns out to be an instance of what's being called a Verifiable Delay Function in cryptography circles. It's a cryptographically secure way to say time has passed. In Solana, we use a far more granular verifiable delay function, a SHA 256 hash chain, to checkpoint the ledger and coordinate consensus. With it, we implement Optimistic Concurrency Control and are now well en route towards that theoretical limit of 710,000 transactions per second. | ||||||
|  |  | ||||||
|  | Architecture | ||||||
|  | === | ||||||
|  |  | ||||||
|  | Before you jump into the code, review the online book [Solana: Blockchain Rebuilt for Scale](https://docs.solana.com/book/). | ||||||
|  |  | ||||||
|  | (The _latest_ development version of the online book is also [available here](https://docs.solana.com/book/v/master/).) | ||||||
|  |  | ||||||
|  | Release Binaries | ||||||
|  | === | ||||||
|  | Official release binaries are available at [Github Releases](https://github.com/solana-labs/solana/releases). | ||||||
|  |  | ||||||
|  | Additionally we provide pre-release binaries for the latest code on the edge and | ||||||
|  | beta channels.  Note that these pre-release binaries may be less stable than an | ||||||
|  | official release. | ||||||
|  |  | ||||||
|  | ### Edge channel | ||||||
|  | #### Linux (x86_64-unknown-linux-gnu) | ||||||
|  | * [solana.tar.bz2](http://release.solana.com/edge/solana-release-x86_64-unknown-linux-gnu.tar.bz2) | ||||||
|  | * [solana-install-init](http://release.solana.com/edge/solana-install-init-x86_64-unknown-linux-gnu) as a stand-alone executable | ||||||
|  | #### mac OS (x86_64-apple-darwin) | ||||||
|  | * [solana.tar.bz2](http://release.solana.com/edge/solana-release-x86_64-apple-darwin.tar.bz2) | ||||||
|  | * [solana-install-init](http://release.solana.com/edge/solana-install-init-x86_64-apple-darwin) as a stand-alone executable | ||||||
|  | #### Windows (x86_64-pc-windows-msvc) | ||||||
|  | * [solana.tar.bz2](http://release.solana.com/edge/solana-release-x86_64-pc-windows-msvc.tar.bz2) | ||||||
|  | * [solana-install-init.exe](http://release.solana.com/edge/solana-install-init-x86_64-pc-windows-msvc.exe) as a stand-alone executable | ||||||
|  | #### All platforms | ||||||
|  | * [solana-metrics.tar.bz2](http://release.solana.com.s3.amazonaws.com/edge/solana-metrics.tar.bz2) | ||||||
|  |  | ||||||
|  | ### Beta channel | ||||||
|  | #### Linux (x86_64-unknown-linux-gnu) | ||||||
|  | * [solana.tar.bz2](http://release.solana.com/beta/solana-release-x86_64-unknown-linux-gnu.tar.bz2) | ||||||
|  | * [solana-install-init](http://release.solana.com/beta/solana-install-init-x86_64-unknown-linux-gnu) as a stand-alone executable | ||||||
|  | #### mac OS (x86_64-apple-darwin) | ||||||
|  | * [solana.tar.bz2](http://release.solana.com/beta/solana-release-x86_64-apple-darwin.tar.bz2) | ||||||
|  | * [solana-install-init](http://release.solana.com/beta/solana-install-init-x86_64-apple-darwin) as a stand-alone executable | ||||||
|  | #### Windows (x86_64-pc-windows-msvc) | ||||||
|  | * [solana.tar.bz2](http://release.solana.com/beta/solana-release-x86_64-pc-windows-msvc.tar.bz2) | ||||||
|  | * [solana-install-init.exe](http://release.solana.com/beta/solana-install-init-x86_64-pc-windows-msvc.exe) as a stand-alone executable | ||||||
|  | #### All platforms | ||||||
|  | * [solana-metrics.tar.bz2](http://release.solana.com.s3.amazonaws.com/beta/solana-metrics.tar.bz2) | ||||||
|  |  | ||||||
| Developing | Developing | ||||||
| === | === | ||||||
| @@ -34,8 +87,7 @@ $ rustup update | |||||||
| On Linux systems you may need to install libssl-dev, pkg-config, zlib1g-dev, etc.  On Ubuntu: | On Linux systems you may need to install libssl-dev, pkg-config, zlib1g-dev, etc.  On Ubuntu: | ||||||
|  |  | ||||||
| ```bash | ```bash | ||||||
| $ sudo apt-get update | $ sudo apt-get install libssl-dev pkg-config zlib1g-dev llvm clang | ||||||
| $ sudo apt-get install libssl-dev libudev-dev pkg-config zlib1g-dev llvm clang |  | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| Download the source code: | Download the source code: | ||||||
| @@ -68,13 +120,16 @@ $ cargo test | |||||||
| Local Testnet | Local Testnet | ||||||
| --- | --- | ||||||
|  |  | ||||||
| Start your own testnet locally, instructions are in the online docs [Solana: Blockchain Rebuild for Scale: Getting Started](https://docs.solana.com/building-from-source). | Start your own testnet locally, instructions are in the book [Solana: Blockchain Rebuild for Scale: Getting Started](https://docs.solana.com/book/getting-started). | ||||||
|  |  | ||||||
| Remote Testnets | Remote Testnets | ||||||
| --- | --- | ||||||
|  |  | ||||||
| * `testnet` - public stable testnet accessible via devnet.solana.com. Runs 24/7 | We maintain several testnets: | ||||||
|  |  | ||||||
|  | * `testnet` - public stable testnet accessible via testnet.solana.com. Runs 24/7 | ||||||
|  | * `testnet-beta` - public beta channel testnet accessible via beta.testnet.solana.com. Runs 24/7 | ||||||
|  | * `testnet-edge` - public edge channel testnet accessible via edge.testnet.solana.com. Runs 24/7 | ||||||
|  |  | ||||||
| ## Deploy process | ## Deploy process | ||||||
|  |  | ||||||
| @@ -185,8 +240,3 @@ problem is solved by this code?" On the other hand, if a test does fail and you | |||||||
| better way to solve the same problem, a Pull Request with your solution would most certainly be | better way to solve the same problem, a Pull Request with your solution would most certainly be | ||||||
| welcome! Likewise, if rewriting a test can better communicate what code it's protecting, please | welcome! Likewise, if rewriting a test can better communicate what code it's protecting, please | ||||||
| send us that patch! | send us that patch! | ||||||
|  |  | ||||||
| Disclaimer |  | ||||||
| === |  | ||||||
|  |  | ||||||
| All claims, content, designs, algorithms, estimates, roadmaps, specifications, and performance measurements described in this project are done with the author's best effort.  It is up to the reader to check and validate their accuracy and truthfulness.  Furthermore nothing in this project constitutes a solicitation for investment. |  | ||||||
|   | |||||||
| @@ -138,11 +138,11 @@ There are three release channels that map to branches as follows: | |||||||
| ### Update documentation | ### Update documentation | ||||||
| TODO: Documentation update procedure is WIP as we move to gitbook | TODO: Documentation update procedure is WIP as we move to gitbook | ||||||
|  |  | ||||||
| Document the new recommended version by updating `docs/src/running-archiver.md` and `docs/src/validator-testnet.md` on the release (beta) branch to point at the `solana-install` for the upcoming release version. | Document the new recommended version by updating `book/src/running-archiver.md` and `book/src/validator-testnet.md` on the release (beta) branch to point at the `solana-install` for the upcoming release version. | ||||||
|  |  | ||||||
| ### Update software on devnet.solana.com | ### Update software on testnet.solana.com | ||||||
|  |  | ||||||
| The testnet running on devnet.solana.com is set to use a fixed release tag | The testnet running on testnet.solana.com is set to use a fixed release tag | ||||||
| which is set in the Buildkite testnet-management pipeline. | which is set in the Buildkite testnet-management pipeline. | ||||||
| This tag needs to be updated and the testnet restarted after a new release | This tag needs to be updated and the testnet restarted after a new release | ||||||
| tag is created. | tag is created. | ||||||
| @@ -182,4 +182,4 @@ TESTNET_OP=create-and-start | |||||||
| ### Alert the community | ### Alert the community | ||||||
|  |  | ||||||
| Notify Discord users on #validator-support that a new release for | Notify Discord users on #validator-support that a new release for | ||||||
| devnet.solana.com is available | testnet.solana.com is available | ||||||
|   | |||||||
| @@ -1,22 +0,0 @@ | |||||||
| [package] |  | ||||||
| authors = ["Solana Maintainers <maintainers@solana.com>"] |  | ||||||
| edition = "2018" |  | ||||||
| name = "solana-accounts-bench" |  | ||||||
| version = "1.1.12" |  | ||||||
| repository = "https://github.com/solana-labs/solana" |  | ||||||
| license = "Apache-2.0" |  | ||||||
| homepage = "https://solana.com/" |  | ||||||
|  |  | ||||||
| [dependencies] |  | ||||||
| log = "0.4.6" |  | ||||||
| rayon = "1.3.0" |  | ||||||
| solana-logger = { path = "../logger", version = "1.1.12" } |  | ||||||
| solana-runtime = { path = "../runtime", version = "1.1.12" } |  | ||||||
| solana-measure = { path = "../measure", version = "1.1.12" } |  | ||||||
| solana-sdk = { path = "../sdk", version = "1.1.12" } |  | ||||||
| rand = "0.7.0" |  | ||||||
| clap = "2.33.0" |  | ||||||
| crossbeam-channel = "0.4" |  | ||||||
|  |  | ||||||
| [package.metadata.docs.rs] |  | ||||||
| targets = ["x86_64-unknown-linux-gnu"] |  | ||||||
| @@ -1,103 +0,0 @@ | |||||||
| use clap::{value_t, App, Arg}; |  | ||||||
| use rayon::prelude::*; |  | ||||||
| use solana_measure::measure::Measure; |  | ||||||
| use solana_runtime::accounts::{create_test_accounts, update_accounts, Accounts}; |  | ||||||
| use solana_sdk::pubkey::Pubkey; |  | ||||||
| use std::collections::HashMap; |  | ||||||
| use std::fs; |  | ||||||
| use std::path::PathBuf; |  | ||||||
|  |  | ||||||
| fn main() { |  | ||||||
|     solana_logger::setup(); |  | ||||||
|  |  | ||||||
|     let matches = App::new("crate") |  | ||||||
|         .about("about") |  | ||||||
|         .version("version") |  | ||||||
|         .arg( |  | ||||||
|             Arg::with_name("num_slots") |  | ||||||
|                 .long("num_slots") |  | ||||||
|                 .takes_value(true) |  | ||||||
|                 .value_name("SLOTS") |  | ||||||
|                 .help("Number of slots to store to."), |  | ||||||
|         ) |  | ||||||
|         .arg( |  | ||||||
|             Arg::with_name("num_accounts") |  | ||||||
|                 .long("num_accounts") |  | ||||||
|                 .takes_value(true) |  | ||||||
|                 .value_name("NUM_ACCOUNTS") |  | ||||||
|                 .help("Total number of accounts"), |  | ||||||
|         ) |  | ||||||
|         .arg( |  | ||||||
|             Arg::with_name("iterations") |  | ||||||
|                 .long("iterations") |  | ||||||
|                 .takes_value(true) |  | ||||||
|                 .value_name("ITERATIONS") |  | ||||||
|                 .help("Number of bench iterations"), |  | ||||||
|         ) |  | ||||||
|         .arg( |  | ||||||
|             Arg::with_name("clean") |  | ||||||
|                 .long("clean") |  | ||||||
|                 .takes_value(false) |  | ||||||
|                 .help("Run clean"), |  | ||||||
|         ) |  | ||||||
|         .get_matches(); |  | ||||||
|  |  | ||||||
|     let num_slots = value_t!(matches, "num_slots", usize).unwrap_or(4); |  | ||||||
|     let num_accounts = value_t!(matches, "num_accounts", usize).unwrap_or(10_000); |  | ||||||
|     let iterations = value_t!(matches, "iterations", usize).unwrap_or(20); |  | ||||||
|     let clean = matches.is_present("clean"); |  | ||||||
|     println!("clean: {:?}", clean); |  | ||||||
|  |  | ||||||
|     let path = PathBuf::from("farf/accounts-bench"); |  | ||||||
|     if fs::remove_dir_all(path.clone()).is_err() { |  | ||||||
|         println!("Warning: Couldn't remove {:?}", path); |  | ||||||
|     } |  | ||||||
|     let accounts = Accounts::new(vec![path]); |  | ||||||
|     println!("Creating {} accounts", num_accounts); |  | ||||||
|     let mut create_time = Measure::start("create accounts"); |  | ||||||
|     let pubkeys: Vec<_> = (0..num_slots) |  | ||||||
|         .into_par_iter() |  | ||||||
|         .map(|slot| { |  | ||||||
|             let mut pubkeys: Vec<Pubkey> = vec![]; |  | ||||||
|             create_test_accounts( |  | ||||||
|                 &accounts, |  | ||||||
|                 &mut pubkeys, |  | ||||||
|                 num_accounts / num_slots, |  | ||||||
|                 slot as u64, |  | ||||||
|             ); |  | ||||||
|             pubkeys |  | ||||||
|         }) |  | ||||||
|         .collect(); |  | ||||||
|     let pubkeys: Vec<_> = pubkeys.into_iter().flatten().collect(); |  | ||||||
|     create_time.stop(); |  | ||||||
|     println!( |  | ||||||
|         "created {} accounts in {} slots {}", |  | ||||||
|         (num_accounts / num_slots) * num_slots, |  | ||||||
|         num_slots, |  | ||||||
|         create_time |  | ||||||
|     ); |  | ||||||
|     let mut ancestors: HashMap<u64, usize> = vec![(0, 0)].into_iter().collect(); |  | ||||||
|     for i in 1..num_slots { |  | ||||||
|         ancestors.insert(i as u64, i - 1); |  | ||||||
|         accounts.add_root(i as u64); |  | ||||||
|     } |  | ||||||
|     for x in 0..iterations { |  | ||||||
|         if clean { |  | ||||||
|             let mut time = Measure::start("clean"); |  | ||||||
|             accounts.accounts_db.clean_accounts(); |  | ||||||
|             time.stop(); |  | ||||||
|             println!("{}", time); |  | ||||||
|             for slot in 0..num_slots { |  | ||||||
|                 update_accounts(&accounts, &pubkeys, ((x + 1) * num_slots + slot) as u64); |  | ||||||
|                 accounts.add_root((x * num_slots + slot) as u64); |  | ||||||
|             } |  | ||||||
|         } else { |  | ||||||
|             let mut pubkeys: Vec<Pubkey> = vec![]; |  | ||||||
|             let mut time = Measure::start("hash"); |  | ||||||
|             let hash = accounts.accounts_db.update_accounts_hash(0, &ancestors); |  | ||||||
|             time.stop(); |  | ||||||
|             println!("hash: {} {}", hash, time); |  | ||||||
|             create_test_accounts(&accounts, &mut pubkeys, 1, 0); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| [package] | [package] | ||||||
| name = "solana-archiver-lib" | name = "solana-archiver-lib" | ||||||
| version = "1.1.12" | version = "0.23.1" | ||||||
| description = "Solana Archiver Library" | description = "Solana Archiver Library" | ||||||
| authors = ["Solana Maintainers <maintainers@solana.com>"] | authors = ["Solana Maintainers <maintainers@solana.com>"] | ||||||
| repository = "https://github.com/solana-labs/solana" | repository = "https://github.com/solana-labs/solana" | ||||||
| @@ -10,34 +10,30 @@ edition = "2018" | |||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
| bincode = "1.2.1" | bincode = "1.2.1" | ||||||
| crossbeam-channel = "0.4" | crossbeam-channel = "0.3" | ||||||
| ed25519-dalek = "=1.0.0-pre.3" | ed25519-dalek = "=1.0.0-pre.1" | ||||||
| log = "0.4.8" | log = "0.4.8" | ||||||
| rand = "0.7.0" | rand = "0.6.5" | ||||||
| rand_chacha = "0.2.2" | rand_chacha = "0.1.1" | ||||||
| solana-client = { path = "../client", version = "1.1.12" } | solana-client = { path = "../client", version = "0.23.1" } | ||||||
| solana-storage-program = { path = "../programs/storage", version = "1.1.12" } | solana-storage-program = { path = "../programs/storage", version = "0.23.1" } | ||||||
| thiserror = "1.0" | thiserror = "1.0" | ||||||
| serde = "1.0.105" | serde = "1.0.104" | ||||||
| serde_json = "1.0.48" | serde_json = "1.0.44" | ||||||
| serde_derive = "1.0.103" | serde_derive = "1.0.103" | ||||||
| solana-net-utils = { path = "../net-utils", version = "1.1.12" } | solana-net-utils = { path = "../net-utils", version = "0.23.1" } | ||||||
| solana-chacha = { path = "../chacha", version = "1.1.12" } | solana-chacha = { path = "../chacha", version = "0.23.1" } | ||||||
| solana-chacha-sys = { path = "../chacha-sys", version = "1.1.12" } | solana-chacha-sys = { path = "../chacha-sys", version = "0.23.1" } | ||||||
| solana-ledger = { path = "../ledger", version = "1.1.12" } | solana-ledger = { path = "../ledger", version = "0.23.1" } | ||||||
| solana-logger = { path = "../logger", version = "1.1.12" } | solana-logger = { path = "../logger", version = "0.23.1" } | ||||||
| solana-perf = { path = "../perf", version = "1.1.12" } | solana-perf = { path = "../perf", version = "0.23.1" } | ||||||
| solana-sdk = { path = "../sdk", version = "1.1.12" } | solana-sdk = { path = "../sdk", version = "0.23.1" } | ||||||
| solana-core = { path = "../core", version = "1.1.12" } | solana-core = { path = "../core", version = "0.23.1" } | ||||||
| solana-streamer = { path = "../streamer", version = "1.1.12" } | solana-archiver-utils = { path = "../archiver-utils", version = "0.23.1" } | ||||||
| solana-archiver-utils = { path = "../archiver-utils", version = "1.1.12" } | solana-metrics = { path = "../metrics", version = "0.23.1" } | ||||||
| solana-metrics = { path = "../metrics", version = "1.1.12" } |  | ||||||
|  |  | ||||||
| [dev-dependencies] | [dev-dependencies] | ||||||
| hex = "0.4.2" | hex = "0.4.0" | ||||||
|  |  | ||||||
| [lib] | [lib] | ||||||
| name = "solana_archiver_lib" | name = "solana_archiver_lib" | ||||||
|  |  | ||||||
| [package.metadata.docs.rs] |  | ||||||
| targets = ["x86_64-unknown-linux-gnu"] |  | ||||||
|   | |||||||
| @@ -1,7 +1,8 @@ | |||||||
| use crate::result::ArchiverError; | use crate::result::ArchiverError; | ||||||
| use crossbeam_channel::unbounded; | use crossbeam_channel::unbounded; | ||||||
| use rand::{thread_rng, Rng}; | use ed25519_dalek; | ||||||
| use rand_chacha::{rand_core::SeedableRng, ChaChaRng}; | use rand::{thread_rng, Rng, SeedableRng}; | ||||||
|  | use rand_chacha::ChaChaRng; | ||||||
| use solana_archiver_utils::sample_file; | use solana_archiver_utils::sample_file; | ||||||
| use solana_chacha::chacha::{chacha_cbc_encrypt_ledger, CHACHA_BLOCK_SIZE}; | use solana_chacha::chacha::{chacha_cbc_encrypt_ledger, CHACHA_BLOCK_SIZE}; | ||||||
| use solana_client::{ | use solana_client::{ | ||||||
| @@ -10,15 +11,15 @@ use solana_client::{ | |||||||
| }; | }; | ||||||
| use solana_core::{ | use solana_core::{ | ||||||
|     cluster_info::{ClusterInfo, Node, VALIDATOR_PORT_RANGE}, |     cluster_info::{ClusterInfo, Node, VALIDATOR_PORT_RANGE}, | ||||||
|     cluster_slots::ClusterSlots, |  | ||||||
|     contact_info::ContactInfo, |     contact_info::ContactInfo, | ||||||
|     gossip_service::GossipService, |     gossip_service::GossipService, | ||||||
|  |     packet::{limited_deserialize, PACKET_DATA_SIZE}, | ||||||
|     repair_service, |     repair_service, | ||||||
|     repair_service::{RepairService, RepairSlotRange, RepairStats, RepairStrategy}, |     repair_service::{RepairService, RepairSlotRange, RepairStrategy}, | ||||||
|     serve_repair::ServeRepair, |  | ||||||
|     shred_fetch_stage::ShredFetchStage, |     shred_fetch_stage::ShredFetchStage, | ||||||
|     sigverify_stage::{DisabledSigVerifier, SigVerifyStage}, |     sigverify_stage::{DisabledSigVerifier, SigVerifyStage}, | ||||||
|     storage_stage::NUM_STORAGE_SAMPLES, |     storage_stage::NUM_STORAGE_SAMPLES, | ||||||
|  |     streamer::{receiver, responder, PacketReceiver}, | ||||||
|     window_service::WindowService, |     window_service::WindowService, | ||||||
| }; | }; | ||||||
| use solana_ledger::{ | use solana_ledger::{ | ||||||
| @@ -26,7 +27,6 @@ use solana_ledger::{ | |||||||
| }; | }; | ||||||
| use solana_net_utils::bind_in_range; | use solana_net_utils::bind_in_range; | ||||||
| use solana_perf::packet::Packets; | use solana_perf::packet::Packets; | ||||||
| use solana_perf::packet::{limited_deserialize, PACKET_DATA_SIZE}; |  | ||||||
| use solana_perf::recycler::Recycler; | use solana_perf::recycler::Recycler; | ||||||
| use solana_sdk::packet::Packet; | use solana_sdk::packet::Packet; | ||||||
| use solana_sdk::{ | use solana_sdk::{ | ||||||
| @@ -36,7 +36,7 @@ use solana_sdk::{ | |||||||
|     commitment_config::CommitmentConfig, |     commitment_config::CommitmentConfig, | ||||||
|     hash::Hash, |     hash::Hash, | ||||||
|     message::Message, |     message::Message, | ||||||
|     signature::{Keypair, Signature, Signer}, |     signature::{Keypair, KeypairUtil, Signature}, | ||||||
|     timing::timestamp, |     timing::timestamp, | ||||||
|     transaction::Transaction, |     transaction::Transaction, | ||||||
|     transport::TransportError, |     transport::TransportError, | ||||||
| @@ -45,15 +45,14 @@ use solana_storage_program::{ | |||||||
|     storage_contract::StorageContract, |     storage_contract::StorageContract, | ||||||
|     storage_instruction::{self, StorageAccountType}, |     storage_instruction::{self, StorageAccountType}, | ||||||
| }; | }; | ||||||
| use solana_streamer::streamer::{receiver, responder, PacketReceiver}; |  | ||||||
| use std::{ | use std::{ | ||||||
|     io::{self, ErrorKind}, |     io::{self, ErrorKind}, | ||||||
|     net::{IpAddr, Ipv4Addr, SocketAddr, UdpSocket}, |     net::{SocketAddr, UdpSocket}, | ||||||
|     path::{Path, PathBuf}, |     path::{Path, PathBuf}, | ||||||
|     result, |     result, | ||||||
|     sync::atomic::{AtomicBool, Ordering}, |     sync::atomic::{AtomicBool, Ordering}, | ||||||
|     sync::mpsc::{channel, Receiver, Sender}, |     sync::mpsc::{channel, Receiver, Sender}, | ||||||
|     sync::Arc, |     sync::{Arc, RwLock}, | ||||||
|     thread::{sleep, spawn, JoinHandle}, |     thread::{sleep, spawn, JoinHandle}, | ||||||
|     time::Duration, |     time::Duration, | ||||||
| }; | }; | ||||||
| @@ -88,11 +87,11 @@ struct ArchiverMeta { | |||||||
| } | } | ||||||
|  |  | ||||||
| fn get_slot_from_signature( | fn get_slot_from_signature( | ||||||
|     signature: &Signature, |     signature: &ed25519_dalek::Signature, | ||||||
|     storage_turn: u64, |     storage_turn: u64, | ||||||
|     slots_per_segment: u64, |     slots_per_segment: u64, | ||||||
| ) -> u64 { | ) -> u64 { | ||||||
|     let signature_vec = signature.as_ref(); |     let signature_vec = signature.to_bytes(); | ||||||
|     let mut segment_index = u64::from(signature_vec[0]) |     let mut segment_index = u64::from(signature_vec[0]) | ||||||
|         | (u64::from(signature_vec[1]) << 8) |         | (u64::from(signature_vec[1]) << 8) | ||||||
|         | (u64::from(signature_vec[1]) << 16) |         | (u64::from(signature_vec[1]) << 16) | ||||||
| @@ -185,10 +184,10 @@ impl Archiver { | |||||||
|  |  | ||||||
|         info!("Archiver: id: {}", keypair.pubkey()); |         info!("Archiver: id: {}", keypair.pubkey()); | ||||||
|         info!("Creating cluster info...."); |         info!("Creating cluster info...."); | ||||||
|         let cluster_info = ClusterInfo::new(node.info.clone(), keypair.clone()); |         let mut cluster_info = ClusterInfo::new(node.info.clone(), keypair.clone()); | ||||||
|         cluster_info.set_entrypoint(cluster_entrypoint.clone()); |         cluster_info.set_entrypoint(cluster_entrypoint.clone()); | ||||||
|         let cluster_info = Arc::new(cluster_info); |         let cluster_info = Arc::new(RwLock::new(cluster_info)); | ||||||
|         let cluster_slots = Arc::new(ClusterSlots::default()); |  | ||||||
|         // Note for now, this ledger will not contain any of the existing entries |         // Note for now, this ledger will not contain any of the existing entries | ||||||
|         // in the ledger located at ledger_path, and will only append on newly received |         // in the ledger located at ledger_path, and will only append on newly received | ||||||
|         // entries after being passed to window_service |         // entries after being passed to window_service | ||||||
| @@ -196,11 +195,17 @@ impl Archiver { | |||||||
|             Blockstore::open(ledger_path).expect("Expected to be able to open database ledger"), |             Blockstore::open(ledger_path).expect("Expected to be able to open database ledger"), | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|         let gossip_service = GossipService::new(&cluster_info, None, node.sockets.gossip, &exit); |         let gossip_service = GossipService::new( | ||||||
|  |             &cluster_info, | ||||||
|  |             Some(blockstore.clone()), | ||||||
|  |             None, | ||||||
|  |             node.sockets.gossip, | ||||||
|  |             &exit, | ||||||
|  |         ); | ||||||
|  |  | ||||||
|         info!("Connecting to the cluster via {:?}", cluster_entrypoint); |         info!("Connecting to the cluster via {:?}", cluster_entrypoint); | ||||||
|         let (nodes, _) = |         let (nodes, _) = | ||||||
|             match solana_core::gossip_service::discover_cluster(&cluster_entrypoint.gossip, 2) { |             match solana_core::gossip_service::discover_cluster(&cluster_entrypoint.gossip, 1) { | ||||||
|                 Ok(nodes_and_archivers) => nodes_and_archivers, |                 Ok(nodes_and_archivers) => nodes_and_archivers, | ||||||
|                 Err(e) => { |                 Err(e) => { | ||||||
|                     //shutdown services before exiting |                     //shutdown services before exiting | ||||||
| @@ -212,9 +217,12 @@ impl Archiver { | |||||||
|         let client = solana_core::gossip_service::get_client(&nodes); |         let client = solana_core::gossip_service::get_client(&nodes); | ||||||
|  |  | ||||||
|         info!("Setting up mining account..."); |         info!("Setting up mining account..."); | ||||||
|         if let Err(e) = |         if let Err(e) = Self::setup_mining_account( | ||||||
|             Self::setup_mining_account(&client, &keypair, &storage_keypair, client_commitment) |             &client, | ||||||
|         { |             &keypair, | ||||||
|  |             &storage_keypair, | ||||||
|  |             client_commitment.clone(), | ||||||
|  |         ) { | ||||||
|             //shutdown services before exiting |             //shutdown services before exiting | ||||||
|             exit.store(true, Ordering::Relaxed); |             exit.store(true, Ordering::Relaxed); | ||||||
|             gossip_service.join()?; |             gossip_service.join()?; | ||||||
| @@ -236,7 +244,6 @@ impl Archiver { | |||||||
|             shred_forward_sockets, |             shred_forward_sockets, | ||||||
|             repair_socket.clone(), |             repair_socket.clone(), | ||||||
|             &shred_fetch_sender, |             &shred_fetch_sender, | ||||||
|             None, |  | ||||||
|             &exit, |             &exit, | ||||||
|         ); |         ); | ||||||
|         let (slot_sender, slot_receiver) = channel(); |         let (slot_sender, slot_receiver) = channel(); | ||||||
| @@ -263,7 +270,6 @@ impl Archiver { | |||||||
|                     repair_socket, |                     repair_socket, | ||||||
|                     shred_fetch_receiver, |                     shred_fetch_receiver, | ||||||
|                     slot_sender, |                     slot_sender, | ||||||
|                     cluster_slots, |  | ||||||
|                 ) { |                 ) { | ||||||
|                     Ok(window_service) => window_service, |                     Ok(window_service) => window_service, | ||||||
|                     Err(e) => { |                     Err(e) => { | ||||||
| @@ -308,7 +314,7 @@ impl Archiver { | |||||||
|     fn run( |     fn run( | ||||||
|         meta: &mut ArchiverMeta, |         meta: &mut ArchiverMeta, | ||||||
|         blockstore: &Arc<Blockstore>, |         blockstore: &Arc<Blockstore>, | ||||||
|         cluster_info: Arc<ClusterInfo>, |         cluster_info: Arc<RwLock<ClusterInfo>>, | ||||||
|         archiver_keypair: &Arc<Keypair>, |         archiver_keypair: &Arc<Keypair>, | ||||||
|         storage_keypair: &Arc<Keypair>, |         storage_keypair: &Arc<Keypair>, | ||||||
|         exit: &Arc<AtomicBool>, |         exit: &Arc<AtomicBool>, | ||||||
| @@ -358,23 +364,23 @@ impl Archiver { | |||||||
|                 &cluster_info, |                 &cluster_info, | ||||||
|                 archiver_keypair, |                 archiver_keypair, | ||||||
|                 storage_keypair, |                 storage_keypair, | ||||||
|                 meta.client_commitment, |                 meta.client_commitment.clone(), | ||||||
|             ); |             ); | ||||||
|         } |         } | ||||||
|         exit.store(true, Ordering::Relaxed); |         exit.store(true, Ordering::Relaxed); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fn redeem_rewards( |     fn redeem_rewards( | ||||||
|         cluster_info: &ClusterInfo, |         cluster_info: &Arc<RwLock<ClusterInfo>>, | ||||||
|         archiver_keypair: &Arc<Keypair>, |         archiver_keypair: &Arc<Keypair>, | ||||||
|         storage_keypair: &Arc<Keypair>, |         storage_keypair: &Arc<Keypair>, | ||||||
|         client_commitment: CommitmentConfig, |         client_commitment: CommitmentConfig, | ||||||
|     ) { |     ) { | ||||||
|         let nodes = cluster_info.tvu_peers(); |         let nodes = cluster_info.read().unwrap().tvu_peers(); | ||||||
|         let client = solana_core::gossip_service::get_client(&nodes); |         let client = solana_core::gossip_service::get_client(&nodes); | ||||||
|  |  | ||||||
|         if let Ok(Some(account)) = |         if let Ok(Some(account)) = | ||||||
|             client.get_account_with_commitment(&storage_keypair.pubkey(), client_commitment) |             client.get_account_with_commitment(&storage_keypair.pubkey(), client_commitment.clone()) | ||||||
|         { |         { | ||||||
|             if let Ok(StorageContract::ArchiverStorage { validations, .. }) = account.state() { |             if let Ok(StorageContract::ArchiverStorage { validations, .. }) = account.state() { | ||||||
|                 if !validations.is_empty() { |                 if !validations.is_empty() { | ||||||
| @@ -382,8 +388,9 @@ impl Archiver { | |||||||
|                         &archiver_keypair.pubkey(), |                         &archiver_keypair.pubkey(), | ||||||
|                         &storage_keypair.pubkey(), |                         &storage_keypair.pubkey(), | ||||||
|                     ); |                     ); | ||||||
|                     let message = Message::new_with_payer(&[ix], Some(&archiver_keypair.pubkey())); |                     let message = | ||||||
|                     if let Err(e) = client.send_message(&[archiver_keypair.as_ref()], message) { |                         Message::new_with_payer(vec![ix], Some(&archiver_keypair.pubkey())); | ||||||
|  |                     if let Err(e) = client.send_message(&[&archiver_keypair], message) { | ||||||
|                         error!("unable to redeem reward, tx failed: {:?}", e); |                         error!("unable to redeem reward, tx failed: {:?}", e); | ||||||
|                     } else { |                     } else { | ||||||
|                         info!( |                         info!( | ||||||
| @@ -402,10 +409,9 @@ impl Archiver { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     // Find a segment to replicate and download it. |     // Find a segment to replicate and download it. | ||||||
|     #[allow(clippy::too_many_arguments)] |  | ||||||
|     fn setup( |     fn setup( | ||||||
|         meta: &mut ArchiverMeta, |         meta: &mut ArchiverMeta, | ||||||
|         cluster_info: Arc<ClusterInfo>, |         cluster_info: Arc<RwLock<ClusterInfo>>, | ||||||
|         blockstore: &Arc<Blockstore>, |         blockstore: &Arc<Blockstore>, | ||||||
|         exit: &Arc<AtomicBool>, |         exit: &Arc<AtomicBool>, | ||||||
|         node_info: &ContactInfo, |         node_info: &ContactInfo, | ||||||
| @@ -413,10 +419,9 @@ impl Archiver { | |||||||
|         repair_socket: Arc<UdpSocket>, |         repair_socket: Arc<UdpSocket>, | ||||||
|         shred_fetch_receiver: PacketReceiver, |         shred_fetch_receiver: PacketReceiver, | ||||||
|         slot_sender: Sender<u64>, |         slot_sender: Sender<u64>, | ||||||
|         cluster_slots: Arc<ClusterSlots>, |  | ||||||
|     ) -> Result<WindowService> { |     ) -> Result<WindowService> { | ||||||
|         let slots_per_segment = |         let slots_per_segment = | ||||||
|             match Self::get_segment_config(&cluster_info, meta.client_commitment) { |             match Self::get_segment_config(&cluster_info, meta.client_commitment.clone()) { | ||||||
|                 Ok(slots_per_segment) => slots_per_segment, |                 Ok(slots_per_segment) => slots_per_segment, | ||||||
|                 Err(e) => { |                 Err(e) => { | ||||||
|                     error!("unable to get segment size configuration, exiting..."); |                     error!("unable to get segment size configuration, exiting..."); | ||||||
| @@ -438,13 +443,13 @@ impl Archiver { | |||||||
|                 return Err(e); |                 return Err(e); | ||||||
|             } |             } | ||||||
|         }; |         }; | ||||||
|         let signature = storage_keypair.sign_message(segment_blockhash.as_ref()); |         let signature = storage_keypair.sign(segment_blockhash.as_ref()); | ||||||
|         let slot = get_slot_from_signature(&signature, segment_slot, slots_per_segment); |         let slot = get_slot_from_signature(&signature, segment_slot, slots_per_segment); | ||||||
|         info!("replicating slot: {}", slot); |         info!("replicating slot: {}", slot); | ||||||
|         slot_sender.send(slot)?; |         slot_sender.send(slot)?; | ||||||
|         meta.slot = slot; |         meta.slot = slot; | ||||||
|         meta.slots_per_segment = slots_per_segment; |         meta.slots_per_segment = slots_per_segment; | ||||||
|         meta.signature = signature; |         meta.signature = Signature::new(&signature.to_bytes()); | ||||||
|         meta.blockhash = segment_blockhash; |         meta.blockhash = segment_blockhash; | ||||||
|  |  | ||||||
|         let mut repair_slot_range = RepairSlotRange::default(); |         let mut repair_slot_range = RepairSlotRange::default(); | ||||||
| @@ -471,7 +476,6 @@ impl Archiver { | |||||||
|             RepairStrategy::RepairRange(repair_slot_range), |             RepairStrategy::RepairRange(repair_slot_range), | ||||||
|             &Arc::new(LeaderScheduleCache::default()), |             &Arc::new(LeaderScheduleCache::default()), | ||||||
|             |_, _, _, _| true, |             |_, _, _, _| true, | ||||||
|             cluster_slots, |  | ||||||
|         ); |         ); | ||||||
|         info!("waiting for ledger download"); |         info!("waiting for ledger download"); | ||||||
|         Self::wait_for_segment_download( |         Self::wait_for_segment_download( | ||||||
| @@ -491,7 +495,7 @@ impl Archiver { | |||||||
|         blockstore: &Arc<Blockstore>, |         blockstore: &Arc<Blockstore>, | ||||||
|         exit: &Arc<AtomicBool>, |         exit: &Arc<AtomicBool>, | ||||||
|         node_info: &ContactInfo, |         node_info: &ContactInfo, | ||||||
|         cluster_info: Arc<ClusterInfo>, |         cluster_info: Arc<RwLock<ClusterInfo>>, | ||||||
|     ) { |     ) { | ||||||
|         info!( |         info!( | ||||||
|             "window created, waiting for ledger download starting at slot {:?}", |             "window created, waiting for ledger download starting at slot {:?}", | ||||||
| @@ -519,8 +523,11 @@ impl Archiver { | |||||||
|         contact_info.tvu = "0.0.0.0:0".parse().unwrap(); |         contact_info.tvu = "0.0.0.0:0".parse().unwrap(); | ||||||
|         contact_info.wallclock = timestamp(); |         contact_info.wallclock = timestamp(); | ||||||
|         // copy over the adopted shred_version from the entrypoint |         // copy over the adopted shred_version from the entrypoint | ||||||
|         contact_info.shred_version = cluster_info.my_shred_version(); |         contact_info.shred_version = cluster_info.read().unwrap().my_data().shred_version; | ||||||
|         cluster_info.update_contact_info(|current| *current = contact_info); |         { | ||||||
|  |             let mut cluster_info_w = cluster_info.write().unwrap(); | ||||||
|  |             cluster_info_w.insert_self(contact_info); | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fn encrypt_ledger(meta: &mut ArchiverMeta, blockstore: &Arc<Blockstore>) -> Result<()> { |     fn encrypt_ledger(meta: &mut ArchiverMeta, blockstore: &Arc<Blockstore>) -> Result<()> { | ||||||
| @@ -580,7 +587,7 @@ impl Archiver { | |||||||
|             &keypair.pubkey(), |             &keypair.pubkey(), | ||||||
|             &Duration::from_millis(100), |             &Duration::from_millis(100), | ||||||
|             &Duration::from_secs(5), |             &Duration::from_secs(5), | ||||||
|             client_commitment, |             client_commitment.clone(), | ||||||
|         )? == 0 |         )? == 0 | ||||||
|         { |         { | ||||||
|             return Err(ArchiverError::EmptyStorageAccountBalance); |             return Err(ArchiverError::EmptyStorageAccountBalance); | ||||||
| @@ -588,15 +595,16 @@ impl Archiver { | |||||||
|  |  | ||||||
|         info!("checking storage account keypair..."); |         info!("checking storage account keypair..."); | ||||||
|         // check if the storage account exists |         // check if the storage account exists | ||||||
|         let balance = |         let balance = client | ||||||
|             client.poll_get_balance_with_commitment(&storage_keypair.pubkey(), client_commitment); |             .poll_get_balance_with_commitment(&storage_keypair.pubkey(), client_commitment.clone()); | ||||||
|         if balance.is_err() || balance.unwrap() == 0 { |         if balance.is_err() || balance.unwrap() == 0 { | ||||||
|             let blockhash = match client.get_recent_blockhash_with_commitment(client_commitment) { |             let blockhash = | ||||||
|                 Ok((blockhash, _)) => blockhash, |                 match client.get_recent_blockhash_with_commitment(client_commitment.clone()) { | ||||||
|                 Err(e) => { |                     Ok((blockhash, _)) => blockhash, | ||||||
|                     return Err(ArchiverError::TransportError(e)); |                     Err(e) => { | ||||||
|                 } |                         return Err(ArchiverError::TransportError(e)); | ||||||
|             }; |                     } | ||||||
|  |                 }; | ||||||
|  |  | ||||||
|             let ix = storage_instruction::create_storage_account( |             let ix = storage_instruction::create_storage_account( | ||||||
|                 &keypair.pubkey(), |                 &keypair.pubkey(), | ||||||
| @@ -615,7 +623,6 @@ impl Archiver { | |||||||
|                         ErrorKind::Other, |                         ErrorKind::Other, | ||||||
|                         "setup_mining_account: signature not found", |                         "setup_mining_account: signature not found", | ||||||
|                     ), |                     ), | ||||||
|                     TransportError::Custom(e) => io::Error::new(ErrorKind::Other, e), |  | ||||||
|                 })?; |                 })?; | ||||||
|         } |         } | ||||||
|         Ok(()) |         Ok(()) | ||||||
| @@ -623,34 +630,39 @@ impl Archiver { | |||||||
|  |  | ||||||
|     fn submit_mining_proof( |     fn submit_mining_proof( | ||||||
|         meta: &ArchiverMeta, |         meta: &ArchiverMeta, | ||||||
|         cluster_info: &ClusterInfo, |         cluster_info: &Arc<RwLock<ClusterInfo>>, | ||||||
|         archiver_keypair: &Arc<Keypair>, |         archiver_keypair: &Arc<Keypair>, | ||||||
|         storage_keypair: &Arc<Keypair>, |         storage_keypair: &Arc<Keypair>, | ||||||
|     ) { |     ) { | ||||||
|         // No point if we've got no storage account... |         // No point if we've got no storage account... | ||||||
|         let nodes = cluster_info.tvu_peers(); |         let nodes = cluster_info.read().unwrap().tvu_peers(); | ||||||
|         let client = solana_core::gossip_service::get_client(&nodes); |         let client = solana_core::gossip_service::get_client(&nodes); | ||||||
|         let storage_balance = client |         let storage_balance = client.poll_get_balance_with_commitment( | ||||||
|             .poll_get_balance_with_commitment(&storage_keypair.pubkey(), meta.client_commitment); |             &storage_keypair.pubkey(), | ||||||
|  |             meta.client_commitment.clone(), | ||||||
|  |         ); | ||||||
|         if storage_balance.is_err() || storage_balance.unwrap() == 0 { |         if storage_balance.is_err() || storage_balance.unwrap() == 0 { | ||||||
|             error!("Unable to submit mining proof, no storage account"); |             error!("Unable to submit mining proof, no storage account"); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         // ...or no lamports for fees |         // ...or no lamports for fees | ||||||
|         let balance = client |         let balance = client.poll_get_balance_with_commitment( | ||||||
|             .poll_get_balance_with_commitment(&archiver_keypair.pubkey(), meta.client_commitment); |             &archiver_keypair.pubkey(), | ||||||
|  |             meta.client_commitment.clone(), | ||||||
|  |         ); | ||||||
|         if balance.is_err() || balance.unwrap() == 0 { |         if balance.is_err() || balance.unwrap() == 0 { | ||||||
|             error!("Unable to submit mining proof, insufficient Archiver Account balance"); |             error!("Unable to submit mining proof, insufficient Archiver Account balance"); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         let blockhash = match client.get_recent_blockhash_with_commitment(meta.client_commitment) { |         let blockhash = | ||||||
|             Ok((blockhash, _)) => blockhash, |             match client.get_recent_blockhash_with_commitment(meta.client_commitment.clone()) { | ||||||
|             Err(_) => { |                 Ok((blockhash, _)) => blockhash, | ||||||
|                 error!("unable to get recent blockhash, can't submit proof"); |                 Err(_) => { | ||||||
|                 return; |                     error!("unable to get recent blockhash, can't submit proof"); | ||||||
|             } |                     return; | ||||||
|         }; |                 } | ||||||
|  |             }; | ||||||
|         let instruction = storage_instruction::mining_proof( |         let instruction = storage_instruction::mining_proof( | ||||||
|             &storage_keypair.pubkey(), |             &storage_keypair.pubkey(), | ||||||
|             meta.sha_state, |             meta.sha_state, | ||||||
| @@ -658,14 +670,14 @@ impl Archiver { | |||||||
|             Signature::new(&meta.signature.as_ref()), |             Signature::new(&meta.signature.as_ref()), | ||||||
|             meta.blockhash, |             meta.blockhash, | ||||||
|         ); |         ); | ||||||
|         let message = Message::new_with_payer(&[instruction], Some(&archiver_keypair.pubkey())); |         let message = Message::new_with_payer(vec![instruction], Some(&archiver_keypair.pubkey())); | ||||||
|         let mut transaction = Transaction::new( |         let mut transaction = Transaction::new( | ||||||
|             &[archiver_keypair.as_ref(), storage_keypair.as_ref()], |             &[archiver_keypair.as_ref(), storage_keypair.as_ref()], | ||||||
|             message, |             message, | ||||||
|             blockhash, |             blockhash, | ||||||
|         ); |         ); | ||||||
|         if let Err(err) = client.send_and_confirm_transaction( |         if let Err(err) = client.send_and_confirm_transaction( | ||||||
|             &[archiver_keypair.as_ref(), storage_keypair.as_ref()], |             &[&archiver_keypair, &storage_keypair], | ||||||
|             &mut transaction, |             &mut transaction, | ||||||
|             10, |             10, | ||||||
|             0, |             0, | ||||||
| @@ -686,10 +698,13 @@ impl Archiver { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     fn get_segment_config( |     fn get_segment_config( | ||||||
|         cluster_info: &ClusterInfo, |         cluster_info: &Arc<RwLock<ClusterInfo>>, | ||||||
|         client_commitment: CommitmentConfig, |         client_commitment: CommitmentConfig, | ||||||
|     ) -> Result<u64> { |     ) -> Result<u64> { | ||||||
|         let rpc_peers = cluster_info.all_rpc_peers(); |         let rpc_peers = { | ||||||
|  |             let cluster_info = cluster_info.read().unwrap(); | ||||||
|  |             cluster_info.all_rpc_peers() | ||||||
|  |         }; | ||||||
|         debug!("rpc peers: {:?}", rpc_peers); |         debug!("rpc peers: {:?}", rpc_peers); | ||||||
|         if !rpc_peers.is_empty() { |         if !rpc_peers.is_empty() { | ||||||
|             let rpc_client = { |             let rpc_client = { | ||||||
| @@ -697,11 +712,16 @@ impl Archiver { | |||||||
|                 RpcClient::new_socket(rpc_peers[node_index].rpc) |                 RpcClient::new_socket(rpc_peers[node_index].rpc) | ||||||
|             }; |             }; | ||||||
|             Ok(rpc_client |             Ok(rpc_client | ||||||
|                 .send::<u64>( |                 .send( | ||||||
|                     RpcRequest::GetSlotsPerSegment, |                     &RpcRequest::GetSlotsPerSegment, | ||||||
|                     serde_json::json!([client_commitment]), |                     serde_json::json!([client_commitment]), | ||||||
|                     0, |                     0, | ||||||
|                 ) |                 ) | ||||||
|  |                 .map_err(|err| { | ||||||
|  |                     warn!("Error while making rpc request {:?}", err); | ||||||
|  |                     ArchiverError::ClientError(err) | ||||||
|  |                 })? | ||||||
|  |                 .as_u64() | ||||||
|                 .unwrap()) |                 .unwrap()) | ||||||
|         } else { |         } else { | ||||||
|             Err(ArchiverError::NoRpcPeers) |             Err(ArchiverError::NoRpcPeers) | ||||||
| @@ -710,7 +730,7 @@ impl Archiver { | |||||||
|  |  | ||||||
|     /// Waits until the first segment is ready, and returns the current segment |     /// Waits until the first segment is ready, and returns the current segment | ||||||
|     fn poll_for_segment( |     fn poll_for_segment( | ||||||
|         cluster_info: &ClusterInfo, |         cluster_info: &Arc<RwLock<ClusterInfo>>, | ||||||
|         slots_per_segment: u64, |         slots_per_segment: u64, | ||||||
|         previous_blockhash: &Hash, |         previous_blockhash: &Hash, | ||||||
|         exit: &Arc<AtomicBool>, |         exit: &Arc<AtomicBool>, | ||||||
| @@ -730,28 +750,38 @@ impl Archiver { | |||||||
|  |  | ||||||
|     /// Poll for a different blockhash and associated max_slot than `previous_blockhash` |     /// Poll for a different blockhash and associated max_slot than `previous_blockhash` | ||||||
|     fn poll_for_blockhash_and_slot( |     fn poll_for_blockhash_and_slot( | ||||||
|         cluster_info: &ClusterInfo, |         cluster_info: &Arc<RwLock<ClusterInfo>>, | ||||||
|         slots_per_segment: u64, |         slots_per_segment: u64, | ||||||
|         previous_blockhash: &Hash, |         previous_blockhash: &Hash, | ||||||
|         exit: &Arc<AtomicBool>, |         exit: &Arc<AtomicBool>, | ||||||
|     ) -> Result<(Hash, u64)> { |     ) -> Result<(Hash, u64)> { | ||||||
|         info!("waiting for the next turn..."); |         info!("waiting for the next turn..."); | ||||||
|         loop { |         loop { | ||||||
|             let rpc_peers = cluster_info.all_rpc_peers(); |             let rpc_peers = { | ||||||
|  |                 let cluster_info = cluster_info.read().unwrap(); | ||||||
|  |                 cluster_info.all_rpc_peers() | ||||||
|  |             }; | ||||||
|             debug!("rpc peers: {:?}", rpc_peers); |             debug!("rpc peers: {:?}", rpc_peers); | ||||||
|             if !rpc_peers.is_empty() { |             if !rpc_peers.is_empty() { | ||||||
|                 let rpc_client = { |                 let rpc_client = { | ||||||
|                     let node_index = thread_rng().gen_range(0, rpc_peers.len()); |                     let node_index = thread_rng().gen_range(0, rpc_peers.len()); | ||||||
|                     RpcClient::new_socket(rpc_peers[node_index].rpc) |                     RpcClient::new_socket(rpc_peers[node_index].rpc) | ||||||
|                 }; |                 }; | ||||||
|  |                 let response = rpc_client | ||||||
|  |                     .send( | ||||||
|  |                         &RpcRequest::GetStorageTurn, | ||||||
|  |                         serde_json::value::Value::Null, | ||||||
|  |                         0, | ||||||
|  |                     ) | ||||||
|  |                     .map_err(|err| { | ||||||
|  |                         warn!("Error while making rpc request {:?}", err); | ||||||
|  |                         ArchiverError::ClientError(err) | ||||||
|  |                     })?; | ||||||
|                 let RpcStorageTurn { |                 let RpcStorageTurn { | ||||||
|                     blockhash: storage_blockhash, |                     blockhash: storage_blockhash, | ||||||
|                     slot: turn_slot, |                     slot: turn_slot, | ||||||
|                 } = rpc_client.send( |                 } = serde_json::from_value::<RpcStorageTurn>(response) | ||||||
|                     RpcRequest::GetStorageTurn, |                     .map_err(ArchiverError::JsonError)?; | ||||||
|                     serde_json::value::Value::Null, |  | ||||||
|                     0, |  | ||||||
|                 )?; |  | ||||||
|                 let turn_blockhash = storage_blockhash.parse().map_err(|err| { |                 let turn_blockhash = storage_blockhash.parse().map_err(|err| { | ||||||
|                     io::Error::new( |                     io::Error::new( | ||||||
|                         io::ErrorKind::Other, |                         io::ErrorKind::Other, | ||||||
| @@ -784,20 +814,19 @@ impl Archiver { | |||||||
|     /// It is recommended to use a temporary blockstore for this since the download will not verify |     /// It is recommended to use a temporary blockstore for this since the download will not verify | ||||||
|     /// shreds received and might impact the chaining of shreds across slots |     /// shreds received and might impact the chaining of shreds across slots | ||||||
|     pub fn download_from_archiver( |     pub fn download_from_archiver( | ||||||
|         serve_repair: &ServeRepair, |         cluster_info: &Arc<RwLock<ClusterInfo>>, | ||||||
|         archiver_info: &ContactInfo, |         archiver_info: &ContactInfo, | ||||||
|         blockstore: &Arc<Blockstore>, |         blockstore: &Arc<Blockstore>, | ||||||
|         slots_per_segment: u64, |         slots_per_segment: u64, | ||||||
|     ) -> Result<u64> { |     ) -> Result<u64> { | ||||||
|         let ip_addr = IpAddr::V4(Ipv4Addr::new(0, 0, 0, 0)); |  | ||||||
|         // Create a client which downloads from the archiver and see that it |         // Create a client which downloads from the archiver and see that it | ||||||
|         // can respond with shreds. |         // can respond with shreds. | ||||||
|         let start_slot = Self::get_archiver_segment_slot(ip_addr, archiver_info.storage_addr); |         let start_slot = Self::get_archiver_segment_slot(archiver_info.storage_addr); | ||||||
|         info!("Archiver download: start at {}", start_slot); |         info!("Archiver download: start at {}", start_slot); | ||||||
|  |  | ||||||
|         let exit = Arc::new(AtomicBool::new(false)); |         let exit = Arc::new(AtomicBool::new(false)); | ||||||
|         let (s_reader, r_reader) = channel(); |         let (s_reader, r_reader) = channel(); | ||||||
|         let repair_socket = Arc::new(bind_in_range(ip_addr, VALIDATOR_PORT_RANGE).unwrap().1); |         let repair_socket = Arc::new(bind_in_range(VALIDATOR_PORT_RANGE).unwrap().1); | ||||||
|         let t_receiver = receiver( |         let t_receiver = receiver( | ||||||
|             repair_socket.clone(), |             repair_socket.clone(), | ||||||
|             &exit, |             &exit, | ||||||
| @@ -805,10 +834,10 @@ impl Archiver { | |||||||
|             Recycler::default(), |             Recycler::default(), | ||||||
|             "archiver_reeciver", |             "archiver_reeciver", | ||||||
|         ); |         ); | ||||||
|         let id = serve_repair.keypair().pubkey(); |         let id = cluster_info.read().unwrap().id(); | ||||||
|         info!( |         info!( | ||||||
|             "Sending repair requests from: {} to: {}", |             "Sending repair requests from: {} to: {}", | ||||||
|             serve_repair.my_info().id, |             cluster_info.read().unwrap().my_data().id, | ||||||
|             archiver_info.gossip |             archiver_info.gossip | ||||||
|         ); |         ); | ||||||
|         let repair_slot_range = RepairSlotRange { |         let repair_slot_range = RepairSlotRange { | ||||||
| @@ -823,14 +852,15 @@ impl Archiver { | |||||||
|                 repair_service::MAX_REPAIR_LENGTH, |                 repair_service::MAX_REPAIR_LENGTH, | ||||||
|                 &repair_slot_range, |                 &repair_slot_range, | ||||||
|             ); |             ); | ||||||
|             let mut repair_stats = RepairStats::default(); |  | ||||||
|             //iter over the repairs and send them |             //iter over the repairs and send them | ||||||
|             if let Ok(repairs) = repairs { |             if let Ok(repairs) = repairs { | ||||||
|                 let reqs: Vec<_> = repairs |                 let reqs: Vec<_> = repairs | ||||||
|                     .into_iter() |                     .into_iter() | ||||||
|                     .filter_map(|repair_request| { |                     .filter_map(|repair_request| { | ||||||
|                         serve_repair |                         cluster_info | ||||||
|                             .map_repair_request(&repair_request, &mut repair_stats) |                             .read() | ||||||
|  |                             .unwrap() | ||||||
|  |                             .map_repair_request(&repair_request) | ||||||
|                             .map(|result| ((archiver_info.gossip, result), repair_request)) |                             .map(|result| ((archiver_info.gossip, result), repair_request)) | ||||||
|                             .ok() |                             .ok() | ||||||
|                     }) |                     }) | ||||||
| @@ -895,8 +925,8 @@ impl Archiver { | |||||||
|         true |         true | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     fn get_archiver_segment_slot(bind_ip_addr: IpAddr, to: SocketAddr) -> u64 { |     fn get_archiver_segment_slot(to: SocketAddr) -> u64 { | ||||||
|         let (_port, socket) = bind_in_range(bind_ip_addr, VALIDATOR_PORT_RANGE).unwrap(); |         let (_port, socket) = bind_in_range(VALIDATOR_PORT_RANGE).unwrap(); | ||||||
|         socket |         socket | ||||||
|             .set_read_timeout(Some(Duration::from_secs(5))) |             .set_read_timeout(Some(Duration::from_secs(5))) | ||||||
|             .unwrap(); |             .unwrap(); | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | use serde_json; | ||||||
| use solana_client::client_error; | use solana_client::client_error; | ||||||
| use solana_ledger::blockstore; | use solana_ledger::blockstore; | ||||||
| use solana_sdk::transport; | use solana_sdk::transport; | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| [package] | [package] | ||||||
| name = "solana-archiver-utils" | name = "solana-archiver-utils" | ||||||
| version = "1.1.12" | version = "0.23.1" | ||||||
| description = "Solana Archiver Utils" | description = "Solana Archiver Utils" | ||||||
| authors = ["Solana Maintainers <maintainers@solana.com>"] | authors = ["Solana Maintainers <maintainers@solana.com>"] | ||||||
| repository = "https://github.com/solana-labs/solana" | repository = "https://github.com/solana-labs/solana" | ||||||
| @@ -10,19 +10,17 @@ edition = "2018" | |||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
| log = "0.4.8" | log = "0.4.8" | ||||||
| rand = "0.7.0" | rand = "0.6.5" | ||||||
| solana-chacha = { path = "../chacha", version = "1.1.12" } | rand_chacha = "0.1.1" | ||||||
| solana-chacha-sys = { path = "../chacha-sys", version = "1.1.12" } | solana-chacha = { path = "../chacha", version = "0.23.1" } | ||||||
| solana-ledger = { path = "../ledger", version = "1.1.12" } | solana-chacha-sys = { path = "../chacha-sys", version = "0.23.1" } | ||||||
| solana-logger = { path = "../logger", version = "1.1.12" } | solana-ledger = { path = "../ledger", version = "0.23.1" } | ||||||
| solana-perf = { path = "../perf", version = "1.1.12" } | solana-logger = { path = "../logger", version = "0.23.1" } | ||||||
| solana-sdk = { path = "../sdk", version = "1.1.12" } | solana-perf = { path = "../perf", version = "0.23.1" } | ||||||
|  | solana-sdk = { path = "../sdk", version = "0.23.1" } | ||||||
|  |  | ||||||
| [dev-dependencies] | [dev-dependencies] | ||||||
| hex = "0.4.2" | hex = "0.4.0" | ||||||
|  |  | ||||||
| [lib] | [lib] | ||||||
| name = "solana_archiver_utils" | name = "solana_archiver_utils" | ||||||
|  |  | ||||||
| [package.metadata.docs.rs] |  | ||||||
| targets = ["x86_64-unknown-linux-gnu"] |  | ||||||
|   | |||||||
| @@ -2,22 +2,19 @@ | |||||||
| authors = ["Solana Maintainers <maintainers@solana.com>"] | authors = ["Solana Maintainers <maintainers@solana.com>"] | ||||||
| edition = "2018" | edition = "2018" | ||||||
| name = "solana-archiver" | name = "solana-archiver" | ||||||
| version = "1.1.12" | version = "0.23.1" | ||||||
| repository = "https://github.com/solana-labs/solana" | repository = "https://github.com/solana-labs/solana" | ||||||
| license = "Apache-2.0" | license = "Apache-2.0" | ||||||
| homepage = "https://solana.com/" | homepage = "https://solana.com/" | ||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
| clap = "2.33.0" | clap = "2.33.0" | ||||||
| console = "0.10.0" | console = "0.9.1" | ||||||
| solana-clap-utils = { path = "../clap-utils", version = "1.1.12" } | solana-clap-utils = { path = "../clap-utils", version = "0.23.1" } | ||||||
| solana-core = { path = "../core", version = "1.1.12" } | solana-core = { path = "../core", version = "0.23.1" } | ||||||
| solana-logger = { path = "../logger", version = "1.1.12" } | solana-logger = { path = "../logger", version = "0.23.1" } | ||||||
| solana-metrics = { path = "../metrics", version = "1.1.12" } | solana-metrics = { path = "../metrics", version = "0.23.1" } | ||||||
| solana-archiver-lib = { path = "../archiver-lib", version = "1.1.12" } | solana-archiver-lib = { path = "../archiver-lib", version = "0.23.1" } | ||||||
| solana-net-utils = { path = "../net-utils", version = "1.1.12" } | solana-net-utils = { path = "../net-utils", version = "0.23.1" } | ||||||
| solana-sdk = { path = "../sdk", version = "1.1.12" } | solana-sdk = { path = "../sdk", version = "0.23.1" } | ||||||
|  |  | ||||||
|  |  | ||||||
| [package.metadata.docs.rs] |  | ||||||
| targets = ["x86_64-unknown-linux-gnu"] |  | ||||||
|   | |||||||
| @@ -2,22 +2,18 @@ use clap::{crate_description, crate_name, App, Arg}; | |||||||
| use console::style; | use console::style; | ||||||
| use solana_archiver_lib::archiver::Archiver; | use solana_archiver_lib::archiver::Archiver; | ||||||
| use solana_clap_utils::{ | use solana_clap_utils::{ | ||||||
|     input_parsers::keypair_of, input_validators::is_keypair_or_ask_keyword, |     input_validators::is_keypair, | ||||||
|     keypair::SKIP_SEED_PHRASE_VALIDATION_ARG, |     keypair::{ | ||||||
|  |         self, keypair_input, KeypairWithSource, ASK_SEED_PHRASE_ARG, | ||||||
|  |         SKIP_SEED_PHRASE_VALIDATION_ARG, | ||||||
|  |     }, | ||||||
| }; | }; | ||||||
| use solana_core::{ | use solana_core::{ | ||||||
|     cluster_info::{Node, VALIDATOR_PORT_RANGE}, |     cluster_info::{Node, VALIDATOR_PORT_RANGE}, | ||||||
|     contact_info::ContactInfo, |     contact_info::ContactInfo, | ||||||
| }; | }; | ||||||
| use solana_sdk::{ | use solana_sdk::{commitment_config::CommitmentConfig, signature::KeypairUtil}; | ||||||
|     commitment_config::CommitmentConfig, | use std::{net::SocketAddr, path::PathBuf, process::exit, sync::Arc}; | ||||||
|     signature::{Keypair, Signer}, |  | ||||||
| }; |  | ||||||
| use std::{ |  | ||||||
|     net::{IpAddr, Ipv4Addr, SocketAddr}, |  | ||||||
|     path::PathBuf, |  | ||||||
|     sync::Arc, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| fn main() { | fn main() { | ||||||
|     solana_logger::setup(); |     solana_logger::setup(); | ||||||
| @@ -28,10 +24,10 @@ fn main() { | |||||||
|         .arg( |         .arg( | ||||||
|             Arg::with_name("identity_keypair") |             Arg::with_name("identity_keypair") | ||||||
|                 .short("i") |                 .short("i") | ||||||
|                 .long("identity") |                 .long("identity-keypair") | ||||||
|                 .value_name("PATH") |                 .value_name("PATH") | ||||||
|                 .takes_value(true) |                 .takes_value(true) | ||||||
|                 .validator(is_keypair_or_ask_keyword) |                 .validator(is_keypair) | ||||||
|                 .help("File containing an identity (keypair)"), |                 .help("File containing an identity (keypair)"), | ||||||
|         ) |         ) | ||||||
|         .arg( |         .arg( | ||||||
| @@ -59,27 +55,48 @@ fn main() { | |||||||
|                 .long("storage-keypair") |                 .long("storage-keypair") | ||||||
|                 .value_name("PATH") |                 .value_name("PATH") | ||||||
|                 .takes_value(true) |                 .takes_value(true) | ||||||
|                 .validator(is_keypair_or_ask_keyword) |                 .validator(is_keypair) | ||||||
|                 .help("File containing the storage account keypair"), |                 .help("File containing the storage account keypair"), | ||||||
|         ) |         ) | ||||||
|  |         .arg( | ||||||
|  |             Arg::with_name(ASK_SEED_PHRASE_ARG.name) | ||||||
|  |                 .long(ASK_SEED_PHRASE_ARG.long) | ||||||
|  |                 .value_name("KEYPAIR NAME") | ||||||
|  |                 .multiple(true) | ||||||
|  |                 .takes_value(true) | ||||||
|  |                 .possible_values(&["identity-keypair", "storage-keypair"]) | ||||||
|  |                 .help(ASK_SEED_PHRASE_ARG.help), | ||||||
|  |         ) | ||||||
|         .arg( |         .arg( | ||||||
|             Arg::with_name(SKIP_SEED_PHRASE_VALIDATION_ARG.name) |             Arg::with_name(SKIP_SEED_PHRASE_VALIDATION_ARG.name) | ||||||
|                 .long(SKIP_SEED_PHRASE_VALIDATION_ARG.long) |                 .long(SKIP_SEED_PHRASE_VALIDATION_ARG.long) | ||||||
|  |                 .requires(ASK_SEED_PHRASE_ARG.name) | ||||||
|                 .help(SKIP_SEED_PHRASE_VALIDATION_ARG.help), |                 .help(SKIP_SEED_PHRASE_VALIDATION_ARG.help), | ||||||
|         ) |         ) | ||||||
|         .get_matches(); |         .get_matches(); | ||||||
|  |  | ||||||
|     let ledger_path = PathBuf::from(matches.value_of("ledger").unwrap()); |     let ledger_path = PathBuf::from(matches.value_of("ledger").unwrap()); | ||||||
|  |  | ||||||
|     let identity_keypair = keypair_of(&matches, "identity_keypair").unwrap_or_else(Keypair::new); |     let identity_keypair = keypair_input(&matches, "identity_keypair") | ||||||
|  |         .unwrap_or_else(|err| { | ||||||
|     let storage_keypair = keypair_of(&matches, "storage_keypair").unwrap_or_else(|| { |             eprintln!("Identity keypair input failed: {}", err); | ||||||
|  |             exit(1); | ||||||
|  |         }) | ||||||
|  |         .keypair; | ||||||
|  |     let KeypairWithSource { | ||||||
|  |         keypair: storage_keypair, | ||||||
|  |         source: storage_keypair_source, | ||||||
|  |     } = keypair_input(&matches, "storage_keypair").unwrap_or_else(|err| { | ||||||
|  |         eprintln!("Storage keypair input failed: {}", err); | ||||||
|  |         exit(1); | ||||||
|  |     }); | ||||||
|  |     if storage_keypair_source == keypair::Source::Generated { | ||||||
|         clap::Error::with_description( |         clap::Error::with_description( | ||||||
|             "The `storage-keypair` argument was not found", |             "The `storage-keypair` argument was not found", | ||||||
|             clap::ErrorKind::ArgumentNotFound, |             clap::ErrorKind::ArgumentNotFound, | ||||||
|         ) |         ) | ||||||
|         .exit(); |         .exit(); | ||||||
|     }); |     } | ||||||
|  |  | ||||||
|     let entrypoint_addr = matches |     let entrypoint_addr = matches | ||||||
|         .value_of("entrypoint") |         .value_of("entrypoint") | ||||||
| @@ -99,7 +116,6 @@ fn main() { | |||||||
|         &identity_keypair.pubkey(), |         &identity_keypair.pubkey(), | ||||||
|         &gossip_addr, |         &gossip_addr, | ||||||
|         VALIDATOR_PORT_RANGE, |         VALIDATOR_PORT_RANGE, | ||||||
|         IpAddr::V4(Ipv4Addr::new(0, 0, 0, 0)), |  | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|     println!( |     println!( | ||||||
|   | |||||||
| @@ -2,24 +2,19 @@ | |||||||
| authors = ["Solana Maintainers <maintainers@solana.com>"] | authors = ["Solana Maintainers <maintainers@solana.com>"] | ||||||
| edition = "2018" | edition = "2018" | ||||||
| name = "solana-banking-bench" | name = "solana-banking-bench" | ||||||
| version = "1.1.12" | version = "0.23.1" | ||||||
| repository = "https://github.com/solana-labs/solana" | repository = "https://github.com/solana-labs/solana" | ||||||
| license = "Apache-2.0" | license = "Apache-2.0" | ||||||
| homepage = "https://solana.com/" | homepage = "https://solana.com/" | ||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
| log = "0.4.6" | log = "0.4.6" | ||||||
| rayon = "1.3.0" | rayon = "1.2.0" | ||||||
| solana-core = { path = "../core", version = "1.1.12" } | solana-core = { path = "../core", version = "0.23.1" } | ||||||
| solana-streamer = { path = "../streamer", version = "1.1.12" } | solana-ledger = { path = "../ledger", version = "0.23.1" } | ||||||
| solana-perf = { path = "../perf", version = "1.1.12" } | solana-logger = { path = "../logger", version = "0.23.1" } | ||||||
| solana-ledger = { path = "../ledger", version = "1.1.12" } | solana-runtime = { path = "../runtime", version = "0.23.1" } | ||||||
| solana-logger = { path = "../logger", version = "1.1.12" } | solana-measure = { path = "../measure", version = "0.23.1" } | ||||||
| solana-runtime = { path = "../runtime", version = "1.1.12" } | solana-sdk = { path = "../sdk", version = "0.23.1" } | ||||||
| solana-measure = { path = "../measure", version = "1.1.12" } | rand = "0.6.5" | ||||||
| solana-sdk = { path = "../sdk", version = "1.1.12" } | crossbeam-channel = "0.3" | ||||||
| rand = "0.7.0" |  | ||||||
| crossbeam-channel = "0.4" |  | ||||||
|  |  | ||||||
| [package.metadata.docs.rs] |  | ||||||
| targets = ["x86_64-unknown-linux-gnu"] |  | ||||||
|   | |||||||
| @@ -2,36 +2,29 @@ use crossbeam_channel::unbounded; | |||||||
| use log::*; | use log::*; | ||||||
| use rand::{thread_rng, Rng}; | use rand::{thread_rng, Rng}; | ||||||
| use rayon::prelude::*; | use rayon::prelude::*; | ||||||
| use solana_core::{ | use solana_core::banking_stage::{create_test_recorder, BankingStage}; | ||||||
|     banking_stage::{create_test_recorder, BankingStage}, | use solana_core::cluster_info::ClusterInfo; | ||||||
|     cluster_info::ClusterInfo, | use solana_core::cluster_info::Node; | ||||||
|     cluster_info::Node, | use solana_core::genesis_utils::{create_genesis_config, GenesisConfigInfo}; | ||||||
|     poh_recorder::PohRecorder, | use solana_core::packet::to_packets_chunked; | ||||||
|     poh_recorder::WorkingBankEntry, | use solana_core::poh_recorder::PohRecorder; | ||||||
| }; | use solana_core::poh_recorder::WorkingBankEntry; | ||||||
| use solana_ledger::{ | use solana_ledger::bank_forks::BankForks; | ||||||
|     bank_forks::BankForks, | use solana_ledger::{blockstore::Blockstore, get_tmp_ledger_path}; | ||||||
|     blockstore::Blockstore, |  | ||||||
|     genesis_utils::{create_genesis_config, GenesisConfigInfo}, |  | ||||||
|     get_tmp_ledger_path, |  | ||||||
| }; |  | ||||||
| use solana_measure::measure::Measure; | use solana_measure::measure::Measure; | ||||||
| use solana_perf::packet::to_packets_chunked; |  | ||||||
| use solana_runtime::bank::Bank; | use solana_runtime::bank::Bank; | ||||||
| use solana_sdk::{ | use solana_sdk::hash::Hash; | ||||||
|     hash::Hash, | use solana_sdk::pubkey::Pubkey; | ||||||
|     pubkey::Pubkey, | use solana_sdk::signature::Keypair; | ||||||
|     signature::Keypair, | use solana_sdk::signature::Signature; | ||||||
|     signature::Signature, | use solana_sdk::system_transaction; | ||||||
|     system_transaction, | use solana_sdk::timing::{duration_as_us, timestamp}; | ||||||
|     timing::{duration_as_us, timestamp}, | use solana_sdk::transaction::Transaction; | ||||||
|     transaction::Transaction, | use std::sync::atomic::Ordering; | ||||||
| }; | use std::sync::mpsc::Receiver; | ||||||
| use std::{ | use std::sync::{Arc, Mutex, RwLock}; | ||||||
|     sync::{atomic::Ordering, mpsc::Receiver, Arc, Mutex}, | use std::thread::sleep; | ||||||
|     thread::sleep, | use std::time::{Duration, Instant}; | ||||||
|     time::{Duration, Instant}, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| fn check_txs( | fn check_txs( | ||||||
|     receiver: &Arc<Receiver<WorkingBankEntry>>, |     receiver: &Arc<Receiver<WorkingBankEntry>>, | ||||||
| @@ -152,7 +145,7 @@ fn main() { | |||||||
|         let (exit, poh_recorder, poh_service, signal_receiver) = |         let (exit, poh_recorder, poh_service, signal_receiver) = | ||||||
|             create_test_recorder(&bank, &blockstore, None); |             create_test_recorder(&bank, &blockstore, None); | ||||||
|         let cluster_info = ClusterInfo::new_with_invalid_keypair(Node::new_localhost().info); |         let cluster_info = ClusterInfo::new_with_invalid_keypair(Node::new_localhost().info); | ||||||
|         let cluster_info = Arc::new(cluster_info); |         let cluster_info = Arc::new(RwLock::new(cluster_info)); | ||||||
|         let banking_stage = BankingStage::new( |         let banking_stage = BankingStage::new( | ||||||
|             &cluster_info, |             &cluster_info, | ||||||
|             &poh_recorder, |             &poh_recorder, | ||||||
| @@ -253,7 +246,7 @@ fn main() { | |||||||
|                 poh_recorder.lock().unwrap().set_bank(&bank); |                 poh_recorder.lock().unwrap().set_bank(&bank); | ||||||
|                 assert!(poh_recorder.lock().unwrap().bank().is_some()); |                 assert!(poh_recorder.lock().unwrap().bank().is_some()); | ||||||
|                 if bank.slot() > 32 { |                 if bank.slot() > 32 { | ||||||
|                     bank_forks.set_root(root, &None, None); |                     bank_forks.set_root(root, &None); | ||||||
|                     root += 1; |                     root += 1; | ||||||
|                 } |                 } | ||||||
|                 debug!( |                 debug!( | ||||||
|   | |||||||
| @@ -2,36 +2,40 @@ | |||||||
| authors = ["Solana Maintainers <maintainers@solana.com>"] | authors = ["Solana Maintainers <maintainers@solana.com>"] | ||||||
| edition = "2018" | edition = "2018" | ||||||
| name = "solana-bench-exchange" | name = "solana-bench-exchange" | ||||||
| version = "1.1.12" | version = "0.23.1" | ||||||
| repository = "https://github.com/solana-labs/solana" | repository = "https://github.com/solana-labs/solana" | ||||||
| license = "Apache-2.0" | license = "Apache-2.0" | ||||||
| homepage = "https://solana.com/" | homepage = "https://solana.com/" | ||||||
| publish = false | publish = false | ||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
|  | bincode = "1.2.1" | ||||||
|  | bs58 = "0.3.0" | ||||||
| clap = "2.32.0" | clap = "2.32.0" | ||||||
| itertools = "0.9.0" | env_logger = "0.7.1" | ||||||
|  | itertools = "0.8.2" | ||||||
| log = "0.4.8" | log = "0.4.8" | ||||||
| num-derive = "0.3" | num-derive = "0.3" | ||||||
| num-traits = "0.2" | num-traits = "0.2" | ||||||
| rand = "0.7.0" | rand = "0.6.5" | ||||||
| rayon = "1.3.0" | rayon = "1.2.0" | ||||||
| serde_json = "1.0.48" | serde = "1.0.104" | ||||||
|  | serde_derive = "1.0.103" | ||||||
|  | serde_json = "1.0.44" | ||||||
| serde_yaml = "0.8.11" | serde_yaml = "0.8.11" | ||||||
| solana-clap-utils = { path = "../clap-utils", version = "1.1.12" } | solana-clap-utils = { path = "../clap-utils", version = "0.23.1" } | ||||||
| solana-core = { path = "../core", version = "1.1.12" } | solana-core = { path = "../core", version = "0.23.1" } | ||||||
| solana-genesis = { path = "../genesis", version = "1.1.12" } | solana-genesis = { path = "../genesis", version = "0.23.1" } | ||||||
| solana-client = { path = "../client", version = "1.1.12" } | solana-client = { path = "../client", version = "0.23.1" } | ||||||
| solana-faucet = { path = "../faucet", version = "1.1.12" } | solana-faucet = { path = "../faucet", version = "0.23.1" } | ||||||
| solana-exchange-program = { path = "../programs/exchange", version = "1.1.12" } | solana-exchange-program = { path = "../programs/exchange", version = "0.23.1" } | ||||||
| solana-logger = { path = "../logger", version = "1.1.12" } | solana-logger = { path = "../logger", version = "0.23.1" } | ||||||
| solana-metrics = { path = "../metrics", version = "1.1.12" } | solana-metrics = { path = "../metrics", version = "0.23.1" } | ||||||
| solana-net-utils = { path = "../net-utils", version = "1.1.12" } | solana-net-utils = { path = "../net-utils", version = "0.23.1" } | ||||||
| solana-runtime = { path = "../runtime", version = "1.1.12" } | solana-runtime = { path = "../runtime", version = "0.23.1" } | ||||||
| solana-sdk = { path = "../sdk", version = "1.1.12" } | solana-sdk = { path = "../sdk", version = "0.23.1" } | ||||||
|  | untrusted = "0.7.0" | ||||||
|  | ws = "0.9.1" | ||||||
|  |  | ||||||
| [dev-dependencies] | [dev-dependencies] | ||||||
| solana-local-cluster = { path = "../local-cluster", version = "1.1.12" } | solana-local-cluster = { path = "../local-cluster", version = "0.23.1" } | ||||||
|  |  | ||||||
| [package.metadata.docs.rs] |  | ||||||
| targets = ["x86_64-unknown-linux-gnu"] |  | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ use solana_sdk::{ | |||||||
|     client::{Client, SyncClient}, |     client::{Client, SyncClient}, | ||||||
|     commitment_config::CommitmentConfig, |     commitment_config::CommitmentConfig, | ||||||
|     pubkey::Pubkey, |     pubkey::Pubkey, | ||||||
|     signature::{Keypair, Signer}, |     signature::{Keypair, KeypairUtil}, | ||||||
|     timing::{duration_as_ms, duration_as_s}, |     timing::{duration_as_ms, duration_as_s}, | ||||||
|     transaction::Transaction, |     transaction::Transaction, | ||||||
|     {system_instruction, system_program}, |     {system_instruction, system_program}, | ||||||
| @@ -701,7 +701,7 @@ fn verify_funding_transfer<T: SyncClient + ?Sized>( | |||||||
|     false |     false | ||||||
| } | } | ||||||
|  |  | ||||||
| pub fn fund_keys<T: Client>(client: &T, source: &Keypair, dests: &[Arc<Keypair>], lamports: u64) { | pub fn fund_keys(client: &dyn Client, source: &Keypair, dests: &[Arc<Keypair>], lamports: u64) { | ||||||
|     let total = lamports * (dests.len() as u64 + 1); |     let total = lamports * (dests.len() as u64 + 1); | ||||||
|     let mut funded: Vec<(&Keypair, u64)> = vec![(source, total)]; |     let mut funded: Vec<(&Keypair, u64)> = vec![(source, total)]; | ||||||
|     let mut notfunded: Vec<&Arc<Keypair>> = dests.iter().collect(); |     let mut notfunded: Vec<&Arc<Keypair>> = dests.iter().collect(); | ||||||
| @@ -824,11 +824,7 @@ pub fn fund_keys<T: Client>(client: &T, source: &Keypair, dests: &[Arc<Keypair>] | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| pub fn create_token_accounts<T: Client>( | pub fn create_token_accounts(client: &dyn Client, signers: &[Arc<Keypair>], accounts: &[Keypair]) { | ||||||
|     client: &T, |  | ||||||
|     signers: &[Arc<Keypair>], |  | ||||||
|     accounts: &[Keypair], |  | ||||||
| ) { |  | ||||||
|     let mut notfunded: Vec<(&Arc<Keypair>, &Keypair)> = signers.iter().zip(accounts).collect(); |     let mut notfunded: Vec<(&Arc<Keypair>, &Keypair)> = signers.iter().zip(accounts).collect(); | ||||||
|  |  | ||||||
|     while !notfunded.is_empty() { |     while !notfunded.is_empty() { | ||||||
| @@ -972,12 +968,7 @@ fn generate_keypairs(num: u64) -> Vec<Keypair> { | |||||||
|     rnd.gen_n_keypairs(num) |     rnd.gen_n_keypairs(num) | ||||||
| } | } | ||||||
|  |  | ||||||
| pub fn airdrop_lamports<T: Client>( | pub fn airdrop_lamports(client: &dyn Client, faucet_addr: &SocketAddr, id: &Keypair, amount: u64) { | ||||||
|     client: &T, |  | ||||||
|     faucet_addr: &SocketAddr, |  | ||||||
|     id: &Keypair, |  | ||||||
|     amount: u64, |  | ||||||
| ) { |  | ||||||
|     let balance = client.get_balance_with_commitment(&id.pubkey(), CommitmentConfig::recent()); |     let balance = client.get_balance_with_commitment(&id.pubkey(), CommitmentConfig::recent()); | ||||||
|     let balance = balance.unwrap_or(0); |     let balance = balance.unwrap_or(0); | ||||||
|     if balance >= amount { |     if balance >= amount { | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| use clap::{crate_description, crate_name, value_t, App, Arg, ArgMatches}; | use clap::{crate_description, crate_name, value_t, App, Arg, ArgMatches}; | ||||||
| use solana_core::gen_keys::GenKeys; | use solana_core::gen_keys::GenKeys; | ||||||
| use solana_faucet::faucet::FAUCET_PORT; | use solana_faucet::faucet::FAUCET_PORT; | ||||||
| use solana_sdk::signature::{read_keypair_file, Keypair}; | use solana_sdk::signature::{read_keypair_file, Keypair, KeypairUtil}; | ||||||
| use std::net::SocketAddr; | use std::net::SocketAddr; | ||||||
| use std::process::exit; | use std::process::exit; | ||||||
| use std::time::Duration; | use std::time::Duration; | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ pub mod order_book; | |||||||
| use crate::bench::{airdrop_lamports, create_client_accounts_file, do_bench_exchange, Config}; | use crate::bench::{airdrop_lamports, create_client_accounts_file, do_bench_exchange, Config}; | ||||||
| use log::*; | use log::*; | ||||||
| use solana_core::gossip_service::{discover_cluster, get_multi_client}; | use solana_core::gossip_service::{discover_cluster, get_multi_client}; | ||||||
| use solana_sdk::signature::Signer; | use solana_sdk::signature::KeypairUtil; | ||||||
|  |  | ||||||
| fn main() { | fn main() { | ||||||
|     solana_logger::setup(); |     solana_logger::setup(); | ||||||
|   | |||||||
| @@ -10,13 +10,12 @@ use solana_local_cluster::local_cluster::{ClusterConfig, LocalCluster}; | |||||||
| use solana_runtime::bank::Bank; | use solana_runtime::bank::Bank; | ||||||
| use solana_runtime::bank_client::BankClient; | use solana_runtime::bank_client::BankClient; | ||||||
| use solana_sdk::genesis_config::create_genesis_config; | use solana_sdk::genesis_config::create_genesis_config; | ||||||
| use solana_sdk::signature::{Keypair, Signer}; | use solana_sdk::signature::{Keypair, KeypairUtil}; | ||||||
| use std::process::exit; | use std::process::exit; | ||||||
| use std::sync::mpsc::channel; | use std::sync::mpsc::channel; | ||||||
| use std::time::Duration; | use std::time::Duration; | ||||||
|  |  | ||||||
| #[test] | #[test] | ||||||
| #[ignore] |  | ||||||
| fn test_exchange_local_cluster() { | fn test_exchange_local_cluster() { | ||||||
|     solana_logger::setup(); |     solana_logger::setup(); | ||||||
|  |  | ||||||
| @@ -86,7 +85,7 @@ fn test_exchange_bank_client() { | |||||||
|     solana_logger::setup(); |     solana_logger::setup(); | ||||||
|     let (genesis_config, identity) = create_genesis_config(100_000_000_000_000); |     let (genesis_config, identity) = create_genesis_config(100_000_000_000_000); | ||||||
|     let mut bank = Bank::new(&genesis_config); |     let mut bank = Bank::new(&genesis_config); | ||||||
|     bank.add_static_program("exchange_program", id(), process_instruction); |     bank.add_instruction_processor(id(), process_instruction); | ||||||
|     let clients = vec![BankClient::new(bank)]; |     let clients = vec![BankClient::new(bank)]; | ||||||
|  |  | ||||||
|     let mut config = Config::default(); |     let mut config = Config::default(); | ||||||
|   | |||||||
| @@ -2,17 +2,14 @@ | |||||||
| authors = ["Solana Maintainers <maintainers@solana.com>"] | authors = ["Solana Maintainers <maintainers@solana.com>"] | ||||||
| edition = "2018" | edition = "2018" | ||||||
| name = "solana-bench-streamer" | name = "solana-bench-streamer" | ||||||
| version = "1.1.12" | version = "0.23.1" | ||||||
| repository = "https://github.com/solana-labs/solana" | repository = "https://github.com/solana-labs/solana" | ||||||
| license = "Apache-2.0" | license = "Apache-2.0" | ||||||
| homepage = "https://solana.com/" | homepage = "https://solana.com/" | ||||||
|  |  | ||||||
| [dependencies] | [dependencies] | ||||||
| clap = "2.33.0" | clap = "2.33.0" | ||||||
| solana-clap-utils = { path = "../clap-utils", version = "1.1.12" } | solana-clap-utils = { path = "../clap-utils", version = "0.23.1" } | ||||||
| solana-streamer = { path = "../streamer", version = "1.1.12" } | solana-core = { path = "../core", version = "0.23.1" } | ||||||
| solana-logger = { path = "../logger", version = "1.1.12" } | solana-logger = { path = "../logger", version = "0.23.1" } | ||||||
| solana-net-utils = { path = "../net-utils", version = "1.1.12" } | solana-net-utils = { path = "../net-utils", version = "0.23.1" } | ||||||
|  |  | ||||||
| [package.metadata.docs.rs] |  | ||||||
| targets = ["x86_64-unknown-linux-gnu"] |  | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| use clap::{crate_description, crate_name, App, Arg}; | use clap::{crate_description, crate_name, App, Arg}; | ||||||
| use solana_streamer::packet::{Packet, Packets, PacketsRecycler, PACKET_DATA_SIZE}; | use solana_core::packet::{Packet, Packets, PacketsRecycler, PACKET_DATA_SIZE}; | ||||||
| use solana_streamer::streamer::{receiver, PacketReceiver}; | use solana_core::streamer::{receiver, PacketReceiver}; | ||||||
| use std::cmp::max; | use std::cmp::max; | ||||||
| use std::net::{IpAddr, Ipv4Addr, SocketAddr, UdpSocket}; | use std::net::{IpAddr, Ipv4Addr, SocketAddr, UdpSocket}; | ||||||
| use std::sync::atomic::{AtomicBool, AtomicUsize, Ordering}; | use std::sync::atomic::{AtomicBool, AtomicUsize, Ordering}; | ||||||
| @@ -67,8 +67,7 @@ fn main() -> Result<()> { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     let mut port = 0; |     let mut port = 0; | ||||||
|     let ip_addr = IpAddr::V4(Ipv4Addr::new(0, 0, 0, 0)); |     let mut addr = SocketAddr::new(IpAddr::V4(Ipv4Addr::new(0, 0, 0, 0)), 0); | ||||||
|     let mut addr = SocketAddr::new(ip_addr, 0); |  | ||||||
|  |  | ||||||
|     let exit = Arc::new(AtomicBool::new(false)); |     let exit = Arc::new(AtomicBool::new(false)); | ||||||
|  |  | ||||||
| @@ -76,7 +75,7 @@ fn main() -> Result<()> { | |||||||
|     let mut read_threads = Vec::new(); |     let mut read_threads = Vec::new(); | ||||||
|     let recycler = PacketsRecycler::default(); |     let recycler = PacketsRecycler::default(); | ||||||
|     for _ in 0..num_sockets { |     for _ in 0..num_sockets { | ||||||
|         let read = solana_net_utils::bind_to(ip_addr, port, false).unwrap(); |         let read = solana_net_utils::bind_to(port, false).unwrap(); | ||||||
|         read.set_read_timeout(Some(Duration::new(1, 0))).unwrap(); |         read.set_read_timeout(Some(Duration::new(1, 0))).unwrap(); | ||||||
|  |  | ||||||
|         addr = read.local_addr().unwrap(); |         addr = read.local_addr().unwrap(); | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
| authors = ["Solana Maintainers <maintainers@solana.com>"] | authors = ["Solana Maintainers <maintainers@solana.com>"] | ||||||
| edition = "2018" | edition = "2018" | ||||||
| name = "solana-bench-tps" | name = "solana-bench-tps" | ||||||
| version = "1.1.12" | version = "0.23.1" | ||||||
| repository = "https://github.com/solana-labs/solana" | repository = "https://github.com/solana-labs/solana" | ||||||
| license = "Apache-2.0" | license = "Apache-2.0" | ||||||
| homepage = "https://solana.com/" | homepage = "https://solana.com/" | ||||||
| @@ -11,30 +11,29 @@ homepage = "https://solana.com/" | |||||||
| bincode = "1.2.1" | bincode = "1.2.1" | ||||||
| clap = "2.33.0" | clap = "2.33.0" | ||||||
| log = "0.4.8" | log = "0.4.8" | ||||||
| rayon = "1.3.0" | rayon = "1.2.0" | ||||||
| serde_json = "1.0.48" | serde = "1.0.104" | ||||||
|  | serde_derive = "1.0.103" | ||||||
|  | serde_json = "1.0.44" | ||||||
| serde_yaml = "0.8.11" | serde_yaml = "0.8.11" | ||||||
| solana-clap-utils = { path = "../clap-utils", version = "1.1.12" } | solana-clap-utils = { path = "../clap-utils", version = "0.23.1" } | ||||||
| solana-core = { path = "../core", version = "1.1.12" } | solana-core = { path = "../core", version = "0.23.1" } | ||||||
| solana-genesis = { path = "../genesis", version = "1.1.12" } | solana-genesis = { path = "../genesis", version = "0.23.1" } | ||||||
| solana-client = { path = "../client", version = "1.1.12" } | solana-client = { path = "../client", version = "0.23.1" } | ||||||
| solana-faucet = { path = "../faucet", version = "1.1.12" } | solana-faucet = { path = "../faucet", version = "0.23.1" } | ||||||
| #solana-librapay = { path = "../programs/librapay", version = "1.1.8", optional = true } | solana-librapay = { path = "../programs/librapay", version = "0.23.1", optional = true } | ||||||
| solana-logger = { path = "../logger", version = "1.1.12" } | solana-logger = { path = "../logger", version = "0.23.1" } | ||||||
| solana-metrics = { path = "../metrics", version = "1.1.12" } | solana-metrics = { path = "../metrics", version = "0.23.1" } | ||||||
| solana-measure = { path = "../measure", version = "1.1.12" } | solana-measure = { path = "../measure", version = "0.23.1" } | ||||||
| solana-net-utils = { path = "../net-utils", version = "1.1.12" } | solana-net-utils = { path = "../net-utils", version = "0.23.1" } | ||||||
| solana-runtime = { path = "../runtime", version = "1.1.12" } | solana-runtime = { path = "../runtime", version = "0.23.1" } | ||||||
| solana-sdk = { path = "../sdk", version = "1.1.12" } | solana-sdk = { path = "../sdk", version = "0.23.1" } | ||||||
| #solana-move-loader-program = { path = "../programs/move_loader", version = "1.1.8", optional = true } | solana-move-loader-program = { path = "../programs/move_loader", version = "0.23.1", optional = true } | ||||||
|  |  | ||||||
| [dev-dependencies] | [dev-dependencies] | ||||||
| serial_test = "0.4.0" | serial_test = "0.3.2" | ||||||
| serial_test_derive = "0.4.0" | serial_test_derive = "0.3.1" | ||||||
| solana-local-cluster = { path = "../local-cluster", version = "1.1.12" } | solana-local-cluster = { path = "../local-cluster", version = "0.23.1" } | ||||||
|  |  | ||||||
| #[features] | [features] | ||||||
| #move = ["solana-librapay", "solana-move-loader-program"] | move = ["solana-librapay", "solana-move-loader-program"] | ||||||
|  |  | ||||||
| [package.metadata.docs.rs] |  | ||||||
| targets = ["x86_64-unknown-linux-gnu"] |  | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ use solana_faucet::faucet::request_airdrop_transaction; | |||||||
| #[cfg(feature = "move")] | #[cfg(feature = "move")] | ||||||
| use solana_librapay::{create_genesis, upload_mint_script, upload_payment_script}; | use solana_librapay::{create_genesis, upload_mint_script, upload_payment_script}; | ||||||
| use solana_measure::measure::Measure; | use solana_measure::measure::Measure; | ||||||
| use solana_metrics::{self, datapoint_info}; | use solana_metrics::{self, datapoint_debug}; | ||||||
| use solana_sdk::{ | use solana_sdk::{ | ||||||
|     client::Client, |     client::Client, | ||||||
|     clock::{DEFAULT_TICKS_PER_SECOND, DEFAULT_TICKS_PER_SLOT, MAX_PROCESSING_AGE}, |     clock::{DEFAULT_TICKS_PER_SECOND, DEFAULT_TICKS_PER_SLOT, MAX_PROCESSING_AGE}, | ||||||
| @@ -15,7 +15,7 @@ use solana_sdk::{ | |||||||
|     fee_calculator::FeeCalculator, |     fee_calculator::FeeCalculator, | ||||||
|     hash::Hash, |     hash::Hash, | ||||||
|     pubkey::Pubkey, |     pubkey::Pubkey, | ||||||
|     signature::{Keypair, Signer}, |     signature::{Keypair, KeypairUtil}, | ||||||
|     system_instruction, system_transaction, |     system_instruction, system_transaction, | ||||||
|     timing::{duration_as_ms, duration_as_s, duration_as_us, timestamp}, |     timing::{duration_as_ms, duration_as_s, duration_as_us, timestamp}, | ||||||
|     transaction::Transaction, |     transaction::Transaction, | ||||||
| @@ -244,7 +244,7 @@ where | |||||||
|  |  | ||||||
| fn metrics_submit_lamport_balance(lamport_balance: u64) { | fn metrics_submit_lamport_balance(lamport_balance: u64) { | ||||||
|     info!("Token balance: {}", lamport_balance); |     info!("Token balance: {}", lamport_balance); | ||||||
|     datapoint_info!( |     datapoint_debug!( | ||||||
|         "bench-tps-lamport_balance", |         "bench-tps-lamport_balance", | ||||||
|         ("balance", lamport_balance, i64) |         ("balance", lamport_balance, i64) | ||||||
|     ); |     ); | ||||||
| @@ -375,7 +375,7 @@ fn generate_txs( | |||||||
|         duration_as_ms(&duration), |         duration_as_ms(&duration), | ||||||
|         blockhash, |         blockhash, | ||||||
|     ); |     ); | ||||||
|     datapoint_info!( |     datapoint_debug!( | ||||||
|         "bench-tps-generate_txs", |         "bench-tps-generate_txs", | ||||||
|         ("duration", duration_as_us(&duration), i64) |         ("duration", duration_as_us(&duration), i64) | ||||||
|     ); |     ); | ||||||
| @@ -481,7 +481,7 @@ fn do_tx_transfers<T: Client>( | |||||||
|                 duration_as_ms(&transfer_start.elapsed()), |                 duration_as_ms(&transfer_start.elapsed()), | ||||||
|                 tx_len as f32 / duration_as_s(&transfer_start.elapsed()), |                 tx_len as f32 / duration_as_s(&transfer_start.elapsed()), | ||||||
|             ); |             ); | ||||||
|             datapoint_info!( |             datapoint_debug!( | ||||||
|                 "bench-tps-do_tx_transfers", |                 "bench-tps-do_tx_transfers", | ||||||
|                 ("duration", duration_as_us(&transfer_start.elapsed()), i64), |                 ("duration", duration_as_us(&transfer_start.elapsed()), i64), | ||||||
|                 ("count", tx_len, i64) |                 ("count", tx_len, i64) | ||||||
| @@ -1059,8 +1059,8 @@ pub fn generate_and_fund_keypairs<T: 'static + Client + Send + Sync>( | |||||||
|     //   pay for the transaction fees in a new run. |     //   pay for the transaction fees in a new run. | ||||||
|     let enough_lamports = 8 * lamports_per_account / 10; |     let enough_lamports = 8 * lamports_per_account / 10; | ||||||
|     if first_keypair_balance < enough_lamports || last_keypair_balance < enough_lamports { |     if first_keypair_balance < enough_lamports || last_keypair_balance < enough_lamports { | ||||||
|         let fee_rate_governor = client.get_fee_rate_governor().unwrap(); |         let (_blockhash, fee_calculator) = get_recent_blockhash(client.as_ref()); | ||||||
|         let max_fee = fee_rate_governor.max_lamports_per_signature; |         let max_fee = fee_calculator.max_lamports_per_signature; | ||||||
|         let extra_fees = extra * max_fee; |         let extra_fees = extra * max_fee; | ||||||
|         let total_keypairs = keypairs.len() as u64 + 1; // Add one for funding keypair |         let total_keypairs = keypairs.len() as u64 + 1; // Add one for funding keypair | ||||||
|         let mut total = lamports_per_account * total_keypairs + extra_fees; |         let mut total = lamports_per_account * total_keypairs + extra_fees; | ||||||
| @@ -1134,7 +1134,7 @@ mod tests { | |||||||
|     use solana_runtime::bank::Bank; |     use solana_runtime::bank::Bank; | ||||||
|     use solana_runtime::bank_client::BankClient; |     use solana_runtime::bank_client::BankClient; | ||||||
|     use solana_sdk::client::SyncClient; |     use solana_sdk::client::SyncClient; | ||||||
|     use solana_sdk::fee_calculator::FeeRateGovernor; |     use solana_sdk::fee_calculator::FeeCalculator; | ||||||
|     use solana_sdk::genesis_config::create_genesis_config; |     use solana_sdk::genesis_config::create_genesis_config; | ||||||
|  |  | ||||||
|     #[test] |     #[test] | ||||||
| @@ -1181,8 +1181,8 @@ mod tests { | |||||||
|     #[test] |     #[test] | ||||||
|     fn test_bench_tps_fund_keys_with_fees() { |     fn test_bench_tps_fund_keys_with_fees() { | ||||||
|         let (mut genesis_config, id) = create_genesis_config(10_000); |         let (mut genesis_config, id) = create_genesis_config(10_000); | ||||||
|         let fee_rate_governor = FeeRateGovernor::new(11, 0); |         let fee_calculator = FeeCalculator::new(11, 0); | ||||||
|         genesis_config.fee_rate_governor = fee_rate_governor; |         genesis_config.fee_calculator = fee_calculator; | ||||||
|         let bank = Bank::new(&genesis_config); |         let bank = Bank::new(&genesis_config); | ||||||
|         let client = Arc::new(BankClient::new(bank)); |         let client = Arc::new(BankClient::new(bank)); | ||||||
|         let keypair_count = 20; |         let keypair_count = 20; | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| use clap::{crate_description, crate_name, App, Arg, ArgMatches}; | use clap::{crate_description, crate_name, App, Arg, ArgMatches}; | ||||||
| use solana_faucet::faucet::FAUCET_PORT; | use solana_faucet::faucet::FAUCET_PORT; | ||||||
| use solana_sdk::fee_calculator::FeeRateGovernor; | use solana_sdk::fee_calculator::FeeCalculator; | ||||||
| use solana_sdk::signature::{read_keypair_file, Keypair}; | use solana_sdk::signature::{read_keypair_file, Keypair, KeypairUtil}; | ||||||
| use std::{net::SocketAddr, process::exit, time::Duration}; | use std::{net::SocketAddr, process::exit, time::Duration}; | ||||||
|  |  | ||||||
| const NUM_LAMPORTS_PER_ACCOUNT_DEFAULT: u64 = solana_sdk::native_token::LAMPORTS_PER_SOL; | const NUM_LAMPORTS_PER_ACCOUNT_DEFAULT: u64 = solana_sdk::native_token::LAMPORTS_PER_SOL; | ||||||
| @@ -43,7 +43,7 @@ impl Default for Config { | |||||||
|             client_ids_and_stake_file: String::new(), |             client_ids_and_stake_file: String::new(), | ||||||
|             write_to_client_file: false, |             write_to_client_file: false, | ||||||
|             read_from_client_file: false, |             read_from_client_file: false, | ||||||
|             target_lamports_per_signature: FeeRateGovernor::default().target_lamports_per_signature, |             target_lamports_per_signature: FeeCalculator::default().target_lamports_per_signature, | ||||||
|             multi_client: true, |             multi_client: true, | ||||||
|             use_move: false, |             use_move: false, | ||||||
|             num_lamports_per_account: NUM_LAMPORTS_PER_ACCOUNT_DEFAULT, |             num_lamports_per_account: NUM_LAMPORTS_PER_ACCOUNT_DEFAULT, | ||||||
|   | |||||||
| @@ -3,8 +3,8 @@ use solana_bench_tps::bench::{do_bench_tps, generate_and_fund_keypairs, generate | |||||||
| use solana_bench_tps::cli; | use solana_bench_tps::cli; | ||||||
| use solana_core::gossip_service::{discover_cluster, get_client, get_multi_client}; | use solana_core::gossip_service::{discover_cluster, get_client, get_multi_client}; | ||||||
| use solana_genesis::Base64Account; | use solana_genesis::Base64Account; | ||||||
| use solana_sdk::fee_calculator::FeeRateGovernor; | use solana_sdk::fee_calculator::FeeCalculator; | ||||||
| use solana_sdk::signature::{Keypair, Signer}; | use solana_sdk::signature::{Keypair, KeypairUtil}; | ||||||
| use solana_sdk::system_program; | use solana_sdk::system_program; | ||||||
| use std::{collections::HashMap, fs::File, io::prelude::*, path::Path, process::exit, sync::Arc}; | use std::{collections::HashMap, fs::File, io::prelude::*, path::Path, process::exit, sync::Arc}; | ||||||
|  |  | ||||||
| @@ -41,7 +41,7 @@ fn main() { | |||||||
|         let (keypairs, _) = generate_keypairs(&id, keypair_count as u64); |         let (keypairs, _) = generate_keypairs(&id, keypair_count as u64); | ||||||
|         let num_accounts = keypairs.len() as u64; |         let num_accounts = keypairs.len() as u64; | ||||||
|         let max_fee = |         let max_fee = | ||||||
|             FeeRateGovernor::new(*target_lamports_per_signature, 0).max_lamports_per_signature; |             FeeCalculator::new(*target_lamports_per_signature, 0).max_lamports_per_signature; | ||||||
|         let num_lamports_per_account = (num_accounts - 1 + NUM_SIGNATURES_FOR_TXS * max_fee) |         let num_lamports_per_account = (num_accounts - 1 + NUM_SIGNATURES_FOR_TXS * max_fee) | ||||||
|             / num_accounts |             / num_accounts | ||||||
|             + num_lamports_per_account; |             + num_lamports_per_account; | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ use solana_faucet::faucet::run_local_faucet; | |||||||
| use solana_local_cluster::local_cluster::{ClusterConfig, LocalCluster}; | use solana_local_cluster::local_cluster::{ClusterConfig, LocalCluster}; | ||||||
| #[cfg(feature = "move")] | #[cfg(feature = "move")] | ||||||
| use solana_sdk::move_loader::solana_move_loader_program; | use solana_sdk::move_loader::solana_move_loader_program; | ||||||
| use solana_sdk::signature::{Keypair, Signer}; | use solana_sdk::signature::{Keypair, KeypairUtil}; | ||||||
| use std::sync::{mpsc::channel, Arc}; | use std::sync::{mpsc::channel, Arc}; | ||||||
| use std::time::Duration; | use std::time::Duration; | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								book/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								book/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | Building the Solana book | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | Install the book's dependnecies, build, and test the book: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | $ ./build.sh | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Run any Rust tests in the markdown: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | $ make test | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Render markdown as HTML: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | $ make build | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Render and view the book: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | $ make open | ||||||
|  | ``` | ||||||
							
								
								
									
										22
									
								
								book/art/tvu.bob
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								book/art/tvu.bob
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  |                                                   .--------. | ||||||
|  |                                                   | Leader | | ||||||
|  |                                                   `--------` | ||||||
|  |                                                        ^ | ||||||
|  |                                                        | | ||||||
|  |                   .------------------------------------|--------------------. | ||||||
|  |                   |  TVU                               |                    | | ||||||
|  |                   |                                    |                    | | ||||||
|  |                   |  .-------.   .------------.   .----+---.   .---------.  | | ||||||
|  |  .------------.   |  | Shred |   | Retransmit |   | Replay |   | Storage |  | | ||||||
|  |  | Upstream   +----->| Fetch +-->| Stage      +-->| Stage  +-->| Stage   |  | | ||||||
|  |  | Validators |   |  | Stage |   |            |   |        |   |         |  | | ||||||
|  |  `------------`   |  `-------`   `----+-------`   `----+---`   `---------`  | | ||||||
|  |                   |        ^          |                |                    | | ||||||
|  |                   |        |          |                |                    | | ||||||
|  |                   `--------|----------|----------------|--------------------` | ||||||
|  |                            |          |                | | ||||||
|  |                            |          V                v | ||||||
|  |                           .+-----------.            .------. | ||||||
|  |                           | Gossip     |            | Bank | | ||||||
|  |                           | Service    |            `------` | ||||||
|  |                           `------------` | ||||||
| @@ -8,5 +8,3 @@ create-missing = false | |||||||
| 
 | 
 | ||||||
| [output.html] | [output.html] | ||||||
| theme = "theme" | theme = "theme" | ||||||
| 
 |  | ||||||
| [output.linkcheck] |  | ||||||
| @@ -3,9 +3,7 @@ set -e | |||||||
| 
 | 
 | ||||||
| cd "$(dirname "$0")" | cd "$(dirname "$0")" | ||||||
| 
 | 
 | ||||||
| : "${rust_stable:=}" # Pacify shellcheck | usage=$(cargo -q run -p solana-cli -- -C ~/.foo --help | sed 's|'"$HOME"'|~|g') | ||||||
| 
 |  | ||||||
| usage=$(cargo +"$rust_stable" -q run -p solana-cli -- -C ~/.foo --help | sed -e 's|'"$HOME"'|~|g' -e 's/[[:space:]]\+$//') |  | ||||||
| 
 | 
 | ||||||
| out=${1:-src/cli/usage.md} | out=${1:-src/cli/usage.md} | ||||||
| 
 | 
 | ||||||
| @@ -27,12 +25,10 @@ section() { | |||||||
| 
 | 
 | ||||||
| section "$usage" >> "$out" | section "$usage" >> "$out" | ||||||
| 
 | 
 | ||||||
| usage=$(sed -e '/^ \{5,\}/d' <<<"$usage") |  | ||||||
| 
 |  | ||||||
| in_subcommands=0 | in_subcommands=0 | ||||||
| while read -r subcommand rest; do | while read -r subcommand rest; do | ||||||
|   [[ $subcommand == "SUBCOMMANDS:" ]] && in_subcommands=1 && continue |   [[ $subcommand == "SUBCOMMANDS:" ]] && in_subcommands=1 && continue | ||||||
|   if ((in_subcommands)); then |   if ((in_subcommands)); then | ||||||
|       section "$(cargo +"$rust_stable" -q run -p solana-cli -- help "$subcommand" | sed -e 's|'"$HOME"'|~|g' -e 's/[[:space:]]\+$//')" "####" >> "$out" |       section "$(cargo -q run -p solana-cli -- help "$subcommand" | sed 's|'"$HOME"'|~|g')" "####" >> "$out" | ||||||
|   fi |   fi | ||||||
| done <<<"$usage">>"$out" | done <<<"$usage">>"$out" | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| BOB_SRCS=$(wildcard art/*.bob) | BOB_SRCS=$(wildcard art/*.bob) | ||||||
| MSC_SRCS=$(wildcard art/*.msc) | MSC_SRCS=$(wildcard art/*.msc) | ||||||
| MD_SRCS=$(wildcard src/*.md src/*/*.md) src/cli/usage.md | MD_SRCS=$(wildcard src/*.md src/*/*.md) | ||||||
| 
 | 
 | ||||||
| SVG_IMGS=$(BOB_SRCS:art/%.bob=src/.gitbook/assets/%.svg) $(MSC_SRCS:art/%.msc=src/.gitbook/assets/%.svg) | SVG_IMGS=$(BOB_SRCS:art/%.bob=src/.gitbook/assets/%.svg) $(MSC_SRCS:art/%.msc=src/.gitbook/assets/%.svg) | ||||||
| 
 | 
 | ||||||
| @@ -8,7 +8,6 @@ TARGET=html/index.html | |||||||
| TEST_STAMP=src/tests.ok | TEST_STAMP=src/tests.ok | ||||||
| 
 | 
 | ||||||
| all: $(TARGET) | all: $(TARGET) | ||||||
| 	./set-solana-release-tag.sh |  | ||||||
| 
 | 
 | ||||||
| svg: $(SVG_IMGS) | svg: $(SVG_IMGS) | ||||||
| 
 | 
 | ||||||
| @@ -28,12 +27,6 @@ src/.gitbook/assets/%.svg: art/%.msc | |||||||
| 	@mkdir -p $(@D) | 	@mkdir -p $(@D) | ||||||
| 	mscgen -T svg -i $< -o $@ | 	mscgen -T svg -i $< -o $@ | ||||||
| 
 | 
 | ||||||
| ../target/debug/solana: |  | ||||||
| 	cd ../cli && cargo build |  | ||||||
| 
 |  | ||||||
| src/cli/usage.md: build-cli-usage.sh ../target/debug/solana |  | ||||||
| 	./$< |  | ||||||
| 
 |  | ||||||
| src/%.md: %.md | src/%.md: %.md | ||||||
| 	@mkdir -p $(@D) | 	@mkdir -p $(@D) | ||||||
| 	@cp $< $@ | 	@cp $< $@ | ||||||
| Before Width: | Height: | Size: 542 KiB After Width: | Height: | Size: 542 KiB | 
| Before Width: | Height: | Size: 256 KiB After Width: | Height: | Size: 256 KiB | 
| Before Width: | Height: | Size: 269 KiB After Width: | Height: | Size: 269 KiB | 
| @@ -1,58 +1,31 @@ | |||||||
| # Table of contents | # Table of contents | ||||||
| 
 | 
 | ||||||
| * [Introduction](introduction.md) | * [Introduction](introduction.md) | ||||||
| * [Wallet Guide](wallet-guide/README.md) | * [Using Solana from the Command-line](cli/README.md) | ||||||
|   * [App Wallets](wallet-guide/apps.md) |   * [Command-line Usage](cli/usage.md) | ||||||
|     * [Trust Wallet](wallet-guide/trust-wallet.md) |   * [Paper Wallet](paper-wallet/README.md) | ||||||
|     * [Ledger Live](wallet-guide/ledger-live.md) |     * [Installation](paper-wallet/installation.md) | ||||||
|   * [Command-line Wallets](wallet-guide/cli.md) |     * [Paper Wallet Usage](paper-wallet/usage.md) | ||||||
|     * [Paper Wallet](paper-wallet/README.md) |  | ||||||
|       * [Paper Wallet Usage](paper-wallet/paper-wallet-usage.md) |  | ||||||
|     * [Hardware Wallets](hardware-wallets/README.md) |  | ||||||
|       * [Ledger Hardware Wallet](hardware-wallets/ledger.md) |  | ||||||
|     * [File System Wallet](file-system-wallet/README.md) |  | ||||||
|   * [Support / Troubleshooting](wallet-guide/support.md) |  | ||||||
| * [Command Line Guide](cli/README.md) |  | ||||||
|   * [Install the Solana Command Line Tool Suite](cli/install-solana-cli-tools.md) |  | ||||||
|   * [Command Line Conventions](cli/conventions.md) |  | ||||||
|   * [Choose a Cluster](cli/choose-a-cluster.md) |  | ||||||
|   * [Send and Receive Tokens](cli/transfer-tokens.md) |  | ||||||
|   * [Delegate Stake](cli/delegate-stake.md) |  | ||||||
|   * [Manage Stake Accounts](cli/manage-stake-accounts.md) |  | ||||||
|   * [Offline Signing](offline-signing/README.md) |   * [Offline Signing](offline-signing/README.md) | ||||||
|     * [Durable Transaction Nonces](offline-signing/durable-nonce.md) |     * [Durable Transaction Nonces](offline-signing/durable-nonce.md) | ||||||
|   * [Command-line Reference](cli/usage.md) | * [Developing Applications](apps/README.md) | ||||||
| * [Solana Clusters](clusters.md) |  | ||||||
| * [Develop Applications](apps/README.md) |  | ||||||
|   * [Example: Web Wallet](apps/webwallet.md) |   * [Example: Web Wallet](apps/webwallet.md) | ||||||
|   * [Example: Tic-Tac-Toe](apps/tictactoe.md) |   * [Example: Tic-Tac-Toe](apps/tictactoe.md) | ||||||
|   * [Drones](apps/drones.md) |   * [Drones](apps/drones.md) | ||||||
|   * [Anatomy of a Transaction](transaction.md) |   * [Anatomy of a Transaction](transaction.md) | ||||||
|   * [JSON RPC API](apps/jsonrpc-api.md) |   * [JSON RPC API](apps/jsonrpc-api.md) | ||||||
|   * [JavaScript API](apps/javascript-api.md) |   * [JavaScript API](apps/javascript-api.md) | ||||||
| * [Run a Validator](running-validator/README.md) | * [Running a Validator](running-validator/README.md) | ||||||
|   * [Validator Requirements](running-validator/validator-reqs.md) |   * [Validator Requirements](running-validator/validator-reqs.md) | ||||||
|   * [Start a Validator](running-validator/validator-start.md) |   * [Choosing a Testnet](running-validator/validator-testnet.md) | ||||||
|   * [Stake](running-validator/validator-stake.md) |   * [Installing the Validator Software](running-validator/validator-software.md) | ||||||
|   * [Monitor a Validator](running-validator/validator-monitor.md) |   * [Starting a Validator](running-validator/validator-start.md) | ||||||
|   * [Publish Validator Info](running-validator/validator-info.md) |   * [Staking](running-validator/validator-stake.md) | ||||||
|   * [Troubleshoot](running-validator/validator-troubleshoot.md) |   * [Monitoring a Validator](running-validator/validator-monitor.md) | ||||||
| * [Participate in Tour de SOL](tour-de-sol/README.md) |   * [Publishing Validator Info](running-validator/validator-info.md) | ||||||
|   * [Useful Links & Discussion](tour-de-sol/useful-links.md) |   * [Troubleshooting](running-validator/validator-troubleshoot.md) | ||||||
|   * [Registration](tour-de-sol/registration/README.md) | * [Running an Archiver](running-archiver.md) | ||||||
|     * [How To Register](tour-de-sol/registration/how-to-register.md) | * [Understanding Solana's Architecture](cluster/README.md) | ||||||
|     * [Terms of Participation](tour-de-sol/registration/terms-of-participation.md) |  | ||||||
|     * [Rewards](tour-de-sol/registration/rewards.md) |  | ||||||
|     * [Confidentiality](tour-de-sol/registration/confidentiality.md) |  | ||||||
|     * [Registration FAQ](tour-de-sol/registration/validator-registration-and-rewards-faq.md) |  | ||||||
|   * [Participation](tour-de-sol/participation/README.md) |  | ||||||
|     * [Requirements to run a validator](tour-de-sol/participation/validator-technical-requirements.md) |  | ||||||
|     * [Create a validator public key](tour-de-sol/participation/validator-public-key-registration.md) |  | ||||||
|     * [Steps to create a validator](tour-de-sol/participation/steps-to-create-a-validator.md) |  | ||||||
|   * [Submit Bug Reports](tour-de-sol/submitting-bugs.md) |  | ||||||
| * [Benchmark a Cluster](cluster/bench-tps.md) |  | ||||||
|   * [Performance Metrics](cluster/performance-metrics.md) |  | ||||||
| * [Solana's Architecture](cluster/README.md) |  | ||||||
|   * [Synchronization](cluster/synchronization.md) |   * [Synchronization](cluster/synchronization.md) | ||||||
|   * [Leader Rotation](cluster/leader-rotation.md) |   * [Leader Rotation](cluster/leader-rotation.md) | ||||||
|   * [Fork Generation](cluster/fork-generation.md) |   * [Fork Generation](cluster/fork-generation.md) | ||||||
| @@ -61,14 +34,15 @@ | |||||||
|   * [Ledger Replication](cluster/ledger-replication.md) |   * [Ledger Replication](cluster/ledger-replication.md) | ||||||
|   * [Secure Vote Signing](cluster/vote-signing.md) |   * [Secure Vote Signing](cluster/vote-signing.md) | ||||||
|   * [Stake Delegation and Rewards](cluster/stake-delegation-and-rewards.md) |   * [Stake Delegation and Rewards](cluster/stake-delegation-and-rewards.md) | ||||||
|  |   * [Performance Metrics](cluster/performance-metrics.md) | ||||||
| * [Anatomy of a Validator](validator/README.md) | * [Anatomy of a Validator](validator/README.md) | ||||||
|   * [TPU](validator/tpu.md) |   * [TPU](validator/tpu.md) | ||||||
|   * [TVU](validator/tvu.md) |   * [TVU](validator/tvu.md) | ||||||
|   * [Blockstore](validator/blockstore.md) |   * [Blockstore](validator/blockstore.md) | ||||||
|   * [Gossip Service](validator/gossip.md) |   * [Gossip Service](validator/gossip.md) | ||||||
|   * [The Runtime](validator/runtime.md) |   * [The Runtime](validator/runtime.md) | ||||||
|  | * [Building from Source](building-from-source.md) | ||||||
| * [Terminology](terminology.md) | * [Terminology](terminology.md) | ||||||
| * [History](history.md) |  | ||||||
| * [Implemented Design Proposals](implemented-proposals/README.md) | * [Implemented Design Proposals](implemented-proposals/README.md) | ||||||
|   * [Cluster Software Installation and Updates](implemented-proposals/installer.md) |   * [Cluster Software Installation and Updates](implemented-proposals/installer.md) | ||||||
|   * [Cluster Economics](implemented-proposals/ed_overview/README.md) |   * [Cluster Economics](implemented-proposals/ed_overview/README.md) | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| # Drones | # Drones | ||||||
| 
 | 
 | ||||||
| This section defines an off-chain service called a _drone_, which acts as custodian of a user's private key. In its simplest form, it can be used to create _airdrop_ transactions, a token transfer from the drone's account to a client's account. | This chapter defines an off-chain service called a _drone_, which acts as custodian of a user's private key. In its simplest form, it can be used to create _airdrop_ transactions, a token transfer from the drone's account to a client's account. | ||||||
| 
 | 
 | ||||||
| ## Signing Service | ## Signing Service | ||||||
| 
 | 
 | ||||||
| @@ -14,6 +14,7 @@ To interact with a Solana node inside a JavaScript application, use the [solana- | |||||||
| 
 | 
 | ||||||
| ## Methods | ## Methods | ||||||
| 
 | 
 | ||||||
|  | * [confirmTransaction](jsonrpc-api.md#confirmtransaction) | ||||||
| * [getAccountInfo](jsonrpc-api.md#getaccountinfo) | * [getAccountInfo](jsonrpc-api.md#getaccountinfo) | ||||||
| * [getBalance](jsonrpc-api.md#getbalance) | * [getBalance](jsonrpc-api.md#getbalance) | ||||||
| * [getBlockCommitment](jsonrpc-api.md#getblockcommitment) | * [getBlockCommitment](jsonrpc-api.md#getblockcommitment) | ||||||
| @@ -21,30 +22,25 @@ To interact with a Solana node inside a JavaScript application, use the [solana- | |||||||
| * [getClusterNodes](jsonrpc-api.md#getclusternodes) | * [getClusterNodes](jsonrpc-api.md#getclusternodes) | ||||||
| * [getConfirmedBlock](jsonrpc-api.md#getconfirmedblock) | * [getConfirmedBlock](jsonrpc-api.md#getconfirmedblock) | ||||||
| * [getConfirmedBlocks](jsonrpc-api.md#getconfirmedblocks) | * [getConfirmedBlocks](jsonrpc-api.md#getconfirmedblocks) | ||||||
| * [getConfirmedSignaturesForAddress](jsonrpc-api.md#getconfirmedsignaturesforaddress) |  | ||||||
| * [getConfirmedTransaction](jsonrpc-api.md#getconfirmedtransaction) |  | ||||||
| * [getEpochInfo](jsonrpc-api.md#getepochinfo) | * [getEpochInfo](jsonrpc-api.md#getepochinfo) | ||||||
| * [getEpochSchedule](jsonrpc-api.md#getepochschedule) | * [getEpochSchedule](jsonrpc-api.md#getepochschedule) | ||||||
| * [getFeeCalculatorForBlockhash](jsonrpc-api.md#getfeecalculatorforblockhash) |  | ||||||
| * [getFeeRateGovernor](jsonrpc-api.md#getfeerategovernor) |  | ||||||
| * [getFirstAvailableBlock](jsonrpc-api.md#getfirstavailableblock) |  | ||||||
| * [getGenesisHash](jsonrpc-api.md#getgenesishash) | * [getGenesisHash](jsonrpc-api.md#getgenesishash) | ||||||
| * [getIdentity](jsonrpc-api.md#getidentity) |  | ||||||
| * [getInflation](jsonrpc-api.md#getinflation) | * [getInflation](jsonrpc-api.md#getinflation) | ||||||
| * [getLargestAccounts](jsonrpc-api.md#getlargestaccounts) |  | ||||||
| * [getLeaderSchedule](jsonrpc-api.md#getleaderschedule) | * [getLeaderSchedule](jsonrpc-api.md#getleaderschedule) | ||||||
| * [getMinimumBalanceForRentExemption](jsonrpc-api.md#getminimumbalanceforrentexemption) | * [getMinimumBalanceForRentExemption](jsonrpc-api.md#getminimumbalanceforrentexemption) | ||||||
|  | * [getNumBlocksSinceSignatureConfirmation](jsonrpc-api.md#getnumblockssincesignatureconfirmation) | ||||||
| * [getProgramAccounts](jsonrpc-api.md#getprogramaccounts) | * [getProgramAccounts](jsonrpc-api.md#getprogramaccounts) | ||||||
| * [getRecentBlockhash](jsonrpc-api.md#getrecentblockhash) | * [getRecentBlockhash](jsonrpc-api.md#getrecentblockhash) | ||||||
| * [getSignatureStatuses](jsonrpc-api.md#getsignaturestatuses) | * [getSignatureConfirmation](jsonrpc-api.md#getsignatureconfirmation) | ||||||
|  | * [getSignatureStatus](jsonrpc-api.md#getsignaturestatus) | ||||||
| * [getSlot](jsonrpc-api.md#getslot) | * [getSlot](jsonrpc-api.md#getslot) | ||||||
| * [getSlotLeader](jsonrpc-api.md#getslotleader) | * [getSlotLeader](jsonrpc-api.md#getslotleader) | ||||||
| * [getSlotsPerSegment](jsonrpc-api.md#getslotspersegment) | * [getSlotsPerSegment](jsonrpc-api.md#getslotspersegment) | ||||||
| * [getStoragePubkeysForSlot](jsonrpc-api.md#getstoragepubkeysforslot) | * [getStoragePubkeysForSlot](jsonrpc-api.md#getstoragepubkeysforslot) | ||||||
| * [getStorageTurn](jsonrpc-api.md#getstorageturn) | * [getStorageTurn](jsonrpc-api.md#getstorageturn) | ||||||
| * [getStorageTurnRate](jsonrpc-api.md#getstorageturnrate) | * [getStorageTurnRate](jsonrpc-api.md#getstorageturnrate) | ||||||
| * [getSupply](jsonrpc-api.md#getsupply) |  | ||||||
| * [getTransactionCount](jsonrpc-api.md#gettransactioncount) | * [getTransactionCount](jsonrpc-api.md#gettransactioncount) | ||||||
|  | * [getTotalSupply](jsonrpc-api.md#gettotalsupply) | ||||||
| * [getVersion](jsonrpc-api.md#getversion) | * [getVersion](jsonrpc-api.md#getversion) | ||||||
| * [getVoteAccounts](jsonrpc-api.md#getvoteaccounts) | * [getVoteAccounts](jsonrpc-api.md#getvoteaccounts) | ||||||
| * [minimumLedgerSlot](jsonrpc-api.md#minimumledgerslot) | * [minimumLedgerSlot](jsonrpc-api.md#minimumledgerslot) | ||||||
| @@ -96,8 +92,7 @@ Requests can be sent in batches by sending an array of JSON-RPC request objects | |||||||
| 
 | 
 | ||||||
| Solana nodes choose which bank state to query based on a commitment requirement | Solana nodes choose which bank state to query based on a commitment requirement | ||||||
| set by the client. Clients may specify either: | set by the client. Clients may specify either: | ||||||
| * `{"commitment":"max"}` - the node will query the most recent bank confirmed by the cluster as having reached `MAX_LOCKOUT_HISTORY` confirmations | * `{"commitment":"max"}` - the node will query the most recent bank having reached `MAX_LOCKOUT_HISTORY` confirmations | ||||||
| * `{"commitment":"root"}` - the node will query the most recent bank having reached `MAX_LOCKOUT_HISTORY` confirmations on this node |  | ||||||
| * `{"commitment":"recent"}` - the node will query its most recent bank state | * `{"commitment":"recent"}` - the node will query its most recent bank state | ||||||
| 
 | 
 | ||||||
| The commitment parameter should be included as the last element in the `params` array: | The commitment parameter should be included as the last element in the `params` array: | ||||||
| @@ -118,18 +113,31 @@ Many methods that take a commitment parameter return an RpcResponse JSON object | |||||||
| * `value` : The value returned by the operation itself. | * `value` : The value returned by the operation itself. | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| ## Health Check |  | ||||||
| Although not a JSON RPC API, a `GET /heath` at the RPC HTTP Endpoint provides a |  | ||||||
| health-check mechanism for use by load balancers or other network |  | ||||||
| infrastructure.  This request will always return a HTTP 200 OK response with a body of |  | ||||||
| "ok" or "behind" based on the following conditions: |  | ||||||
| 1. If one or more `--trusted-validator` arguments are provided to `solana-validator`, "ok" is returned |  | ||||||
|    when the node has within `HEALTH_CHECK_SLOT_DISTANCE` slots of the highest trusted validator, |  | ||||||
|    otherwise "behind" is returned. |  | ||||||
| 2. "ok" is always returned if no trusted validators are provided. |  | ||||||
| 
 |  | ||||||
| ## JSON RPC API Reference | ## JSON RPC API Reference | ||||||
| 
 | 
 | ||||||
|  | ### confirmTransaction | ||||||
|  | 
 | ||||||
|  | Returns a transaction receipt | ||||||
|  | 
 | ||||||
|  | #### Parameters: | ||||||
|  | 
 | ||||||
|  | * `<string>` - Signature of Transaction to confirm, as base-58 encoded string | ||||||
|  | * `<object>` - (optional) [Commitment](jsonrpc-api.md#configuring-state-commitment) | ||||||
|  | 
 | ||||||
|  | #### Results: | ||||||
|  | 
 | ||||||
|  | * `RpcResponse<bool>` - RpcResponse JSON object with `value` field set to Transaction status, boolean true if Transaction is confirmed | ||||||
|  | 
 | ||||||
|  | #### Example: | ||||||
|  | 
 | ||||||
|  | ```bash | ||||||
|  | // Request | ||||||
|  | curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0", "id":1, "method":"confirmTransaction", "params":["5VERv8NMvzbJMEkV8xnrLkEaWRtSz9CosKDYjCJjBRnbJLgp8uirBgmQpjKhoR4tjF3ZpRzrFmBV6UjKdiSZkQUW"]}' http://localhost:8899 | ||||||
|  | 
 | ||||||
|  | // Result | ||||||
|  | {"jsonrpc":"2.0","result":{"context":{"slot":1},"value":true},"id":1} | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
| ### getAccountInfo | ### getAccountInfo | ||||||
| 
 | 
 | ||||||
| Returns all information associated with the account of provided Pubkey | Returns all information associated with the account of provided Pubkey | ||||||
| @@ -141,15 +149,14 @@ Returns all information associated with the account of provided Pubkey | |||||||
| 
 | 
 | ||||||
| #### Results: | #### Results: | ||||||
| 
 | 
 | ||||||
| The result will be an RpcResponse JSON object with `value` equal to: | The result value will be an RpcResponse JSON object containing an AccountInfo JSON object. | ||||||
| 
 | 
 | ||||||
| * `<null>` - if the requested account doesn't exist | * `RpcResponse<AccountInfo>`, RpcResponse JSON object with `value` field set to AccountInfo, a JSON object containing: | ||||||
| * `<object>` - otherwise, a JSON object containing: | * `lamports: <u64>`, number of lamports assigned to this account, as a u64 | ||||||
|   * `lamports: <u64>`, number of lamports assigned to this account, as a u64 | * `owner: <string>`, base-58 encoded Pubkey of the program this account has been assigned to | ||||||
|   * `owner: <string>`, base-58 encoded Pubkey of the program this account has been assigned to | * `data: <string>`, base-58 encoded data associated with the account | ||||||
|   * `data: <string>`, base-58 encoded data associated with the account | * `executable: <bool>`, boolean indicating if the account contains a program \(and is strictly read-only\) | ||||||
|   * `executable: <bool>`, boolean indicating if the account contains a program \(and is strictly read-only\) | * `rentEpoch`: <u64>, the epoch at which this account will next owe rent, as u64 | ||||||
|   * `rentEpoch`: <u64>, the epoch at which this account will next owe rent, as u64 |  | ||||||
| 
 | 
 | ||||||
| #### Example: | #### Example: | ||||||
| 
 | 
 | ||||||
| @@ -198,7 +205,7 @@ The result field will be a JSON object containing: | |||||||
| 
 | 
 | ||||||
| * `commitment` - commitment, comprising either: | * `commitment` - commitment, comprising either: | ||||||
|   * `<null>` - Unknown block |   * `<null>` - Unknown block | ||||||
|   * `<array>` - commitment, array of u64 integers logging the amount of cluster stake in lamports that has voted on the block at each depth from 0 to `MAX_LOCKOUT_HISTORY` + 1 |   * `<array>` - commitment, array of u64 integers logging the amount of cluster stake in lamports that has voted on the block at each depth from 0 to `MAX_LOCKOUT_HISTORY` | ||||||
| * `totalStake` - total active stake, in lamports, of the current epoch | * `totalStake` - total active stake, in lamports, of the current epoch | ||||||
| 
 | 
 | ||||||
| #### Example: | #### Example: | ||||||
| @@ -208,7 +215,7 @@ The result field will be a JSON object containing: | |||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "method":"getBlockCommitment","params":[5]}' http://localhost:8899 | curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "method":"getBlockCommitment","params":[5]}' http://localhost:8899 | ||||||
| 
 | 
 | ||||||
| // Result | // Result | ||||||
| {"jsonrpc":"2.0","result":{"commitment":[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,10,32],"totalStake": 42},"id":1} | {"jsonrpc":"2.0","result":{"commitment":[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,10,32],"totalStake": 42},"id":1} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ### getBlockTime | ### getBlockTime | ||||||
| @@ -259,8 +266,7 @@ The result field will be an array of JSON objects, each with the following sub f | |||||||
| * `pubkey: <string>` - Node public key, as base-58 encoded string | * `pubkey: <string>` - Node public key, as base-58 encoded string | ||||||
| * `gossip: <string>` - Gossip network address for the node | * `gossip: <string>` - Gossip network address for the node | ||||||
| * `tpu: <string>` - TPU network address for the node | * `tpu: <string>` - TPU network address for the node | ||||||
| * `rpc: <string>|null` - JSON RPC network address for the node, or `null` if the JSON RPC service is not enabled | * `rpc: <string>` - JSON RPC network address for the node, or `null` if the JSON RPC service is not enabled | ||||||
| * `version: <string>|null` - The software version of the node, or `null` if the version information is not available |  | ||||||
| 
 | 
 | ||||||
| #### Example: | #### Example: | ||||||
| 
 | 
 | ||||||
| @@ -269,7 +275,7 @@ The result field will be an array of JSON objects, each with the following sub f | |||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0", "id":1, "method":"getClusterNodes"}' http://localhost:8899 | curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0", "id":1, "method":"getClusterNodes"}' http://localhost:8899 | ||||||
| 
 | 
 | ||||||
| // Result | // Result | ||||||
| {"jsonrpc":"2.0","result":[{"gossip":"10.239.6.48:8001","pubkey":"9QzsJf7LPLj8GkXbYT3LFDKqsj2hHG7TA3xinJHu8epQ","rpc":"10.239.6.48:8899","tpu":"10.239.6.48:8856"},"version":"1.0.0 c375ce1f"],"id":1} | {"jsonrpc":"2.0","result":[{"gossip":"10.239.6.48:8001","pubkey":"9QzsJf7LPLj8GkXbYT3LFDKqsj2hHG7TA3xinJHu8epQ","rpc":"10.239.6.48:8899","tpu":"10.239.6.48:8856"}],"id":1} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ### getConfirmedBlock | ### getConfirmedBlock | ||||||
| @@ -285,24 +291,18 @@ Returns identity and transaction information about a confirmed block in the ledg | |||||||
| 
 | 
 | ||||||
| The result field will be an object with the following fields: | The result field will be an object with the following fields: | ||||||
| 
 | 
 | ||||||
| * `<null>` - if specified block is not confirmed | * `blockhash: <string>` - the blockhash of this block, as base-58 encoded string | ||||||
| * `<object>` - if block is confirmed, an object with the following fields: | * `previousBlockhash: <string>` - the blockhash of this block's parent, as base-58 encoded string | ||||||
|   * `blockhash: <string>` - the blockhash of this block, as base-58 encoded string | * `parentSlot: <u64>` - the slot index of this block's parent | ||||||
|   * `previousBlockhash: <string>` - the blockhash of this block's parent, as base-58 encoded string; if the parent block is not available due to ledger cleanup, this field will return "11111111111111111111111111111111" | * `transactions: <array>` - an array of JSON objects containing: | ||||||
|   * `parentSlot: <u64>` - the slot index of this block's parent |   * `transaction: <object|string>` - [Transaction](transaction-api.md) object, either in JSON format or base-58 encoded binary data, depending on encoding parameter | ||||||
|   * `transactions: <array>` - an array of JSON objects containing: |   * `meta: <object>` - transaction status metadata object, containing `null` or: | ||||||
|     * `transaction: <object|string>` - [Transaction](#transaction-structure) object, either in JSON format or base-58 encoded binary data, depending on encoding parameter |      * `status: <object>` - Transaction status: | ||||||
|     * `meta: <object>` - transaction status metadata object, containing `null` or: |        * `"Ok": null` - Transaction was successful | ||||||
|       * `err: <object | null>` - Error if transaction failed, null if transaction succeeded. [TransactionError definitions](https://github.com/solana-labs/solana/blob/master/sdk/src/transaction.rs#L14) |        * `"Err": <ERR>` - Transaction failed with TransactionError  [TransactionError definitions](https://github.com/solana-labs/solana/blob/master/sdk/src/transaction.rs#L14) | ||||||
|       * `fee: <u64>` - fee this transaction was charged, as u64 integer |      * `fee: <u64>` - fee this transaction was charged, as u64 integer | ||||||
|       * `preBalances: <array>` - array of u64 account balances from before the transaction was processed |      * `preBalances: <array>` - array of u64 account balances from before the transaction was processed | ||||||
|       * `postBalances: <array>` - array of u64 account balances after the transaction was processed |      * `postBalances: <array>` - array of u64 account balances after the transaction was processed | ||||||
|       * DEPRECATED: `status: <object>` - Transaction status |  | ||||||
|         * `"Ok": <null>` - Transaction was successful |  | ||||||
|         * `"Err": <ERR>` - Transaction failed with TransactionError |  | ||||||
|   * `rewards: <array>` - an array of JSON objects containing: |  | ||||||
|     * `pubkey: <string>` - The public key, as base-58 encoded string, of the account that received the reward |  | ||||||
|     * `lamports: <i64>`- number of reward lamports credited or debited by the account, as a i64 |  | ||||||
| 
 | 
 | ||||||
| #### Example: | #### Example: | ||||||
| 
 | 
 | ||||||
| @@ -311,34 +311,15 @@ The result field will be an object with the following fields: | |||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc": "2.0","id":1,"method":"getConfirmedBlock","params":[430, "json"]}' localhost:8899 | curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc": "2.0","id":1,"method":"getConfirmedBlock","params":[430, "json"]}' localhost:8899 | ||||||
| 
 | 
 | ||||||
| // Result | // Result | ||||||
| {"jsonrpc":"2.0","result":{"blockhash":"Gp3t5bfDsJv1ovP8cB1SuRhXVuoTqDv7p3tymyubYg5","parentSlot":429,"previousBlockhash":"EFejToxii1L5aUF2NrK9dsbAEmZSNyN5nsipmZHQR1eA","transactions":[{"transaction":{"message":{"accountKeys":["6H94zdiaYfRfPfKjYLjyr2VFBg6JHXygy84r3qhc3NsC","39UAy8hsoYPywGPGdmun747omSr79zLSjqvPJN3zetoH","SysvarS1otHashes111111111111111111111111111","SysvarC1ock11111111111111111111111111111111","Vote111111111111111111111111111111111111111"],"header":{"numReadonlySignedAccounts":0,"numReadonlyUnsignedAccounts":3,"numRequiredSignatures":2},"instructions":[{"accounts":[1,2,3],"data":"29z5mr1JoRmJYQ6ynmk3pf31cGFRziAF1M3mT3L6sFXf5cKLdkEaMXMT8AqLpD4CpcupHmuMEmtZHpomrwfdZetSomNy3d","programIdIndex":4}],"recentBlockhash":"EFejToxii1L5aUF2NrK9dsbAEmZSNyN5nsipmZHQR1eA"},"signatures":["35YGay1Lwjwgxe9zaH6APSHbt9gYQUCtBWTNL3aVwVGn9xTFw2fgds7qK5AL29mP63A9j3rh8KpN1TgSR62XCaby","4vANMjSKiwEchGSXwVrQkwHnmsbKQmy9vdrsYxWdCup1bLsFzX8gKrFTSVDCZCae2dbxJB9mPNhqB2sD1vvr4sAD"]},"meta":{"err":null,"fee":18000,"postBalances":[499999972500,15298080,1,1,1],"preBalances":[499999990500,15298080,1,1,1],"status":{"Ok":null}}}]},"id":1} | {"jsonrpc":"2.0","result":{"blockhash":"Gp3t5bfDsJv1ovP8cB1SuRhXVuoTqDv7p3tymyubYg5","parentSlot":429,"previousBlockhash":"EFejToxii1L5aUF2NrK9dsbAEmZSNyN5nsipmZHQR1eA","transactions":[{"transaction":{"message":{"accountKeys":["6H94zdiaYfRfPfKjYLjyr2VFBg6JHXygy84r3qhc3NsC","39UAy8hsoYPywGPGdmun747omSr79zLSjqvPJN3zetoH","SysvarS1otHashes111111111111111111111111111","SysvarC1ock11111111111111111111111111111111","Vote111111111111111111111111111111111111111"],"header":{"numReadonlySignedAccounts":0,"numReadonlyUnsignedAccounts":3,"numRequiredSignatures":2},"instructions":[{"accounts":[1,2,3],"data":"29z5mr1JoRmJYQ6ynmk3pf31cGFRziAF1M3mT3L6sFXf5cKLdkEaMXMT8AqLpD4CpcupHmuMEmtZHpomrwfdZetSomNy3d","programIdIndex":4}],"recentBlockhash":"EFejToxii1L5aUF2NrK9dsbAEmZSNyN5nsipmZHQR1eA"},"signatures":["35YGay1Lwjwgxe9zaH6APSHbt9gYQUCtBWTNL3aVwVGn9xTFw2fgds7qK5AL29mP63A9j3rh8KpN1TgSR62XCaby","4vANMjSKiwEchGSXwVrQkwHnmsbKQmy9vdrsYxWdCup1bLsFzX8gKrFTSVDCZCae2dbxJB9mPNhqB2sD1vvr4sAD"]},"meta":{"fee":18000,"postBalances":[499999972500,15298080,1,1,1],"preBalances":[499999990500,15298080,1,1,1],"status":{"Ok":null}}}]},"id":1} | ||||||
| 
 | 
 | ||||||
| // Request | // Request | ||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc": "2.0","id":1,"method":"getConfirmedBlock","params":[430, "binary"]}' localhost:8899 | curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc": "2.0","id":1,"method":"getConfirmedBlock","params":[430, "binary"]}' localhost:8899 | ||||||
| 
 | 
 | ||||||
| // Result | // Result | ||||||
| {"jsonrpc":"2.0","result":{"blockhash":"Gp3t5bfDsJv1ovP8cB1SuRhXVuoTqDv7p3tymyubYg5","parentSlot":429,"previousBlockhash":"EFejToxii1L5aUF2NrK9dsbAEmZSNyN5nsipmZHQR1eA","transactions":[{"transaction":"81UZJt4dh4Do66jDhrgkQudS8J2N6iG3jaVav7gJrqJSFY4Ug53iA9JFJZh2gxKWcaFdLJwhHx9mRdg9JwDAWB4ywiu5154CRwXV4FMdnPLg7bhxRLwhhYaLsVgMF5AyNRcTzjCVoBvqFgDU7P8VEKDEiMvD3qxzm1pLZVxDG1LTQpT3Dz4Uviv4KQbFQNuC22KupBoyHFB7Zh6KFdMqux4M9PvhoqcoJsJKwXjWpKu7xmEKnnrSbfLadkgjBmmjhW3fdTrFvnhQdTkhtdJxUL1xS9GMuJQer8YgSKNtUXB1eXZQwXU8bU2BjYkZE6Q5Xww8hu9Z4E4Mo4QsooVtHoP6BM3NKw8zjVbWfoCQqxTrwuSzrNCWCWt58C24LHecH67CTt2uXbYSviixvrYkK7A3t68BxTJcF1dXJitEPTFe2ceTkauLJqrJgnER4iUrsjr26T8YgWvpY9wkkWFSviQW6wV5RASTCUasVEcrDiaKj8EQMkgyDoe9HyKitSVg67vMWJFpUXpQobseWJUs5FTWWzmfHmFp8FZ","meta":{"err":null,"fee":18000,"postBalances":[499999972500,15298080,1,1,1],"preBalances":[499999990500,15298080,1,1,1],"status":{"Ok":null}}}]},"id":1} | {"jsonrpc":"2.0","result":{"blockhash":"Gp3t5bfDsJv1ovP8cB1SuRhXVuoTqDv7p3tymyubYg5","parentSlot":429,"previousBlockhash":"EFejToxii1L5aUF2NrK9dsbAEmZSNyN5nsipmZHQR1eA","transactions":[{"transaction":"81UZJt4dh4Do66jDhrgkQudS8J2N6iG3jaVav7gJrqJSFY4Ug53iA9JFJZh2gxKWcaFdLJwhHx9mRdg9JwDAWB4ywiu5154CRwXV4FMdnPLg7bhxRLwhhYaLsVgMF5AyNRcTzjCVoBvqFgDU7P8VEKDEiMvD3qxzm1pLZVxDG1LTQpT3Dz4Uviv4KQbFQNuC22KupBoyHFB7Zh6KFdMqux4M9PvhoqcoJsJKwXjWpKu7xmEKnnrSbfLadkgjBmmjhW3fdTrFvnhQdTkhtdJxUL1xS9GMuJQer8YgSKNtUXB1eXZQwXU8bU2BjYkZE6Q5Xww8hu9Z4E4Mo4QsooVtHoP6BM3NKw8zjVbWfoCQqxTrwuSzrNCWCWt58C24LHecH67CTt2uXbYSviixvrYkK7A3t68BxTJcF1dXJitEPTFe2ceTkauLJqrJgnER4iUrsjr26T8YgWvpY9wkkWFSviQW6wV5RASTCUasVEcrDiaKj8EQMkgyDoe9HyKitSVg67vMWJFpUXpQobseWJUs5FTWWzmfHmFp8FZ","meta":{"fee":18000,"postBalances":[499999972500,15298080,1,1,1],"preBalances":[499999990500,15298080,1,1,1],"status":{"Ok":null}}}]},"id":1} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| #### Transaction Structure |  | ||||||
| 
 |  | ||||||
| Transactions are quite different from those on other blockchains. Be sure to review [Anatomy of a Transaction](../transaction.md) to learn about transactions on Solana. |  | ||||||
| 
 |  | ||||||
| The JSON structure of a transaction is defined as follows: |  | ||||||
| 
 |  | ||||||
| * `signatures: <array[string]>` - A list of base-58 encoded signatures applied to the transaction. The list is always of length `message.header.numRequiredSignatures`, and the signature at index `i` corresponds to the public key at index `i` in `message.account_keys`. |  | ||||||
| * `message: <object>` - Defines the content of the transaction. |  | ||||||
|   * `accountKeys: <array[string]>` - List of base-58 encoded public keys used by the transaction, including by the instructions and for signatures. The first `message.header.numRequiredSignatures` public keys must sign the transaction. |  | ||||||
|   * `header: <object>` - Details the account types and signatures required by the transaction. |  | ||||||
|     * `numRequiredSignatures: <number>` - The total number of signatures required to make the transaction valid. The signatures must match the first `numRequiredSignatures` of `message.account_keys`. |  | ||||||
|     * `numReadonlySignedAccounts: <number>` - The last `numReadonlySignedAccounts` of the signed keys are read-only accounts. Programs may process multiple transactions that load read-only accounts within a single PoH entry, but are not permitted to credit or debit lamports or modify account data. Transactions targeting the same read-write account are evaluated sequentially. |  | ||||||
|     * `numReadonlyUnsignedAccounts: <number>` - The last `numReadonlyUnsignedAccounts` of the unsigned keys are read-only accounts. |  | ||||||
|   * `recentBlockhash: <string>` - A base-58 encoded hash of a recent block in the ledger used to prevent transaction duplication and to give transactions lifetimes. |  | ||||||
|   * `instructions: <array[object]>` - List of program instructions that will be executed in sequence and committed in one atomic transaction if all succeed. |  | ||||||
|     * `programIdIndex: <number>` - Index into the `message.accountKeys` array indicating the program account that executes this instruction. |  | ||||||
|     * `accounts: <array[number]>` - List of ordered indices into the `message.accountKeys` array indicating which accounts to pass to the program. |  | ||||||
|     * `data: <string>` - The program input data encoded in a base-58 string. |  | ||||||
| 
 |  | ||||||
| ### getConfirmedBlocks | ### getConfirmedBlocks | ||||||
| 
 | 
 | ||||||
| Returns a list of confirmed blocks | Returns a list of confirmed blocks | ||||||
| @@ -364,73 +345,6 @@ curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc": "2.0","id":1,"m | |||||||
| {"jsonrpc":"2.0","result":[5,6,7,8,9,10],"id":1} | {"jsonrpc":"2.0","result":[5,6,7,8,9,10],"id":1} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ### getConfirmedSignaturesForAddress |  | ||||||
| 
 |  | ||||||
| Returns a list of all the confirmed signatures for transactions involving an address, within a specified Slot range. Max range allowed is 10_000 Slots. |  | ||||||
| 
 |  | ||||||
| #### Parameters: |  | ||||||
| 
 |  | ||||||
| * `<string>` - account address as base-58 encoded string |  | ||||||
| * `<u64>` - start slot, inclusive |  | ||||||
| * `<u64>` - end slot, inclusive |  | ||||||
| 
 |  | ||||||
| #### Results: |  | ||||||
| 
 |  | ||||||
| The result field will be an array of: |  | ||||||
| * `<string>` - transaction signature as base-58 encoded string |  | ||||||
| 
 |  | ||||||
| The signatures will be ordered based on the Slot in which they were confirmed in, from lowest to highest Slot |  | ||||||
| 
 |  | ||||||
| #### Example: |  | ||||||
| 
 |  | ||||||
| ```bash |  | ||||||
| // Request |  | ||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc": "2.0","id":1,"method":"getConfirmedSignaturesForAddress","params":["6H94zdiaYfRfPfKjYLjyr2VFBg6JHXygy84r3qhc3NsC", 0, 100]}' localhost:8899 |  | ||||||
| 
 |  | ||||||
| // Result |  | ||||||
| {"jsonrpc":"2.0","result":{["35YGay1Lwjwgxe9zaH6APSHbt9gYQUCtBWTNL3aVwVGn9xTFw2fgds7qK5AL29mP63A9j3rh8KpN1TgSR62XCaby","4bJdGN8Tt2kLWZ3Fa1dpwPSEkXWWTSszPSf1rRVsCwNjxbbUdwTeiWtmi8soA26YmwnKD4aAxNp8ci1Gjpdv4gsr","4LQ14a7BYY27578Uj8LPCaVhSdJGLn9DJqnUJHpy95FMqdKf9acAhUhecPQNjNUy6VoNFUbvwYkPociFSf87cWbG"]},"id":1} |  | ||||||
| ``` |  | ||||||
| 
 |  | ||||||
| ### getConfirmedTransaction |  | ||||||
| 
 |  | ||||||
| Returns transaction details for a confirmed transaction |  | ||||||
| 
 |  | ||||||
| #### Parameters: |  | ||||||
| 
 |  | ||||||
| * `<string>` - transaction signature as base-58 encoded string |  | ||||||
| * `<string>` - (optional) encoding for the returned Transaction, either "json" or "binary". If not provided, the default encoding is JSON. |  | ||||||
| 
 |  | ||||||
| #### Results: |  | ||||||
| 
 |  | ||||||
| * `<null>` - if transaction is not found or not confirmed |  | ||||||
| * `<object>` - if transaction is confirmed, an object with the following fields: |  | ||||||
|   * `slot: <u64>` - the slot this transaction was processed in |  | ||||||
|   * `transaction: <object|string>` - [Transaction](#transaction-structure) object, either in JSON format or base-58 encoded binary data, depending on encoding parameter |  | ||||||
|   * `meta: <object | null>` - transaction status metadata object: |  | ||||||
|     * `err: <object | null>` - Error if transaction failed, null if transaction succeeded. [TransactionError definitions](https://github.com/solana-labs/solana/blob/master/sdk/src/transaction.rs#L14) |  | ||||||
|     * `fee: <u64>` - fee this transaction was charged, as u64 integer |  | ||||||
|     * `preBalances: <array>` - array of u64 account balances from before the transaction was processed |  | ||||||
|     * `postBalances: <array>` - array of u64 account balances after the transaction was processed |  | ||||||
|     * DEPRECATED: `status: <object>` - Transaction status |  | ||||||
|       * `"Ok": <null>` - Transaction was successful |  | ||||||
|       * `"Err": <ERR>` - Transaction failed with TransactionError |  | ||||||
| 
 |  | ||||||
| #### Example: |  | ||||||
| 
 |  | ||||||
| ```bash |  | ||||||
| // Request |  | ||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc": "2.0","id":1,"method":"getConfirmedTransaction","params":["35YGay1Lwjwgxe9zaH6APSHbt9gYQUCtBWTNL3aVwVGn9xTFw2fgds7qK5AL29mP63A9j3rh8KpN1TgSR62XCaby", "json"]}' localhost:8899 |  | ||||||
| 
 |  | ||||||
| // Result |  | ||||||
| {"jsonrpc":"2.0","result":{"slot":430,"transaction":{"message":{"accountKeys":["6H94zdiaYfRfPfKjYLjyr2VFBg6JHXygy84r3qhc3NsC","39UAy8hsoYPywGPGdmun747omSr79zLSjqvPJN3zetoH","SysvarS1otHashes111111111111111111111111111","SysvarC1ock11111111111111111111111111111111","Vote111111111111111111111111111111111111111"],"header":{"numReadonlySignedAccounts":0,"numReadonlyUnsignedAccounts":3,"numRequiredSignatures":2},"instructions":[{"accounts":[1,2,3],"data":"29z5mr1JoRmJYQ6ynmk3pf31cGFRziAF1M3mT3L6sFXf5cKLdkEaMXMT8AqLpD4CpcupHmuMEmtZHpomrwfdZetSomNy3d","programIdIndex":4}],"recentBlockhash":"EFejToxii1L5aUF2NrK9dsbAEmZSNyN5nsipmZHQR1eA"},"signatures":["35YGay1Lwjwgxe9zaH6APSHbt9gYQUCtBWTNL3aVwVGn9xTFw2fgds7qK5AL29mP63A9j3rh8KpN1TgSR62XCaby","4vANMjSKiwEchGSXwVrQkwHnmsbKQmy9vdrsYxWdCup1bLsFzX8gKrFTSVDCZCae2dbxJB9mPNhqB2sD1vvr4sAD"]},"meta":{"err":null,"fee":18000,"postBalances":[499999972500,15298080,1,1,1],"preBalances":[499999990500,15298080,1,1,1],"status":{"Ok":null}}},"id":1} |  | ||||||
| 
 |  | ||||||
| // Request |  | ||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc": "2.0","id":1,"method":"getConfirmedTransaction","params":["35YGay1Lwjwgxe9zaH6APSHbt9gYQUCtBWTNL3aVwVGn9xTFw2fgds7qK5AL29mP63A9j3rh8KpN1TgSR62XCaby", "binary"]}' localhost:8899 |  | ||||||
| 
 |  | ||||||
| // Result |  | ||||||
| {"jsonrpc":"2.0","result":{"slot":430,"transaction":"81UZJt4dh4Do66jDhrgkQudS8J2N6iG3jaVav7gJrqJSFY4Ug53iA9JFJZh2gxKWcaFdLJwhHx9mRdg9JwDAWB4ywiu5154CRwXV4FMdnPLg7bhxRLwhhYaLsVgMF5AyNRcTzjCVoBvqFgDU7P8VEKDEiMvD3qxzm1pLZVxDG1LTQpT3Dz4Uviv4KQbFQNuC22KupBoyHFB7Zh6KFdMqux4M9PvhoqcoJsJKwXjWpKu7xmEKnnrSbfLadkgjBmmjhW3fdTrFvnhQdTkhtdJxUL1xS9GMuJQer8YgSKNtUXB1eXZQwXU8bU2BjYkZE6Q5Xww8hu9Z4E4Mo4QsooVtHoP6BM3NKw8zjVbWfoCQqxTrwuSzrNCWCWt58C24LHecH67CTt2uXbYSviixvrYkK7A3t68BxTJcF1dXJitEPTFe2ceTkauLJqrJgnER4iUrsjr26T8YgWvpY9wkkWFSviQW6wV5RASTCUasVEcrDiaKj8EQMkgyDoe9HyKitSVg67vMWJFpUXpQobseWJUs5FTWWzmfHmFp8FZ","meta":{"err":null,"fee":18000,"postBalances":[499999972500,15298080,1,1,1],"preBalances":[499999990500,15298080,1,1,1],"status":{"Ok":null}}},"id":1} |  | ||||||
| ``` |  | ||||||
| 
 |  | ||||||
| ### getEpochInfo | ### getEpochInfo | ||||||
| 
 | 
 | ||||||
| Returns information about the current epoch | Returns information about the current epoch | ||||||
| @@ -486,82 +400,6 @@ curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "m | |||||||
| {"jsonrpc":"2.0","result":{"firstNormalEpoch":8,"firstNormalSlot":8160,"leaderScheduleSlotOffset":8192,"slotsPerEpoch":8192,"warmup":true},"id":1} | {"jsonrpc":"2.0","result":{"firstNormalEpoch":8,"firstNormalSlot":8160,"leaderScheduleSlotOffset":8192,"slotsPerEpoch":8192,"warmup":true},"id":1} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ### getFeeCalculatorForBlockhash |  | ||||||
| 
 |  | ||||||
| Returns the fee calculator associated with the query blockhash, or `null` if the blockhash has expired |  | ||||||
| 
 |  | ||||||
| #### Parameters: |  | ||||||
| 
 |  | ||||||
| * `blockhash: <string>`, query blockhash as a Base58 encoded string |  | ||||||
| 
 |  | ||||||
| #### Results: |  | ||||||
| 
 |  | ||||||
| The result will be an RpcResponse JSON object with `value` equal to: |  | ||||||
| 
 |  | ||||||
| * `<null>` - if the query blockhash has expired |  | ||||||
| * `<object>` - otherwise, a JSON object containing: |  | ||||||
|   * `feeCalculator: <object>`, `FeeCalculator` object describing the cluster fee rate at the queried blockhash |  | ||||||
| 
 |  | ||||||
| #### Example: |  | ||||||
| 
 |  | ||||||
| ```bash |  | ||||||
| // Request |  | ||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "method":"getFeeCalculatorForBlockhash", "params":["GJxqhuxcgfn5Tcj6y3f8X4FeCDd2RQ6SnEMo1AAxrPRZ"]}' http://localhost:8899 |  | ||||||
| 
 |  | ||||||
| // Result |  | ||||||
| {"jsonrpc":"2.0","result":{"context":{"slot":221},"value":{"feeCalculator":{"lamportsPerSignature":5000}}},"id":1} |  | ||||||
| ``` |  | ||||||
| 
 |  | ||||||
| ### getFeeRateGovernor |  | ||||||
| 
 |  | ||||||
| Returns the fee rate governor information from the root bank |  | ||||||
| 
 |  | ||||||
| #### Parameters: |  | ||||||
| 
 |  | ||||||
| None |  | ||||||
| 
 |  | ||||||
| #### Results: |  | ||||||
| 
 |  | ||||||
| The `result` field will be an `object` with the following fields: |  | ||||||
| 
 |  | ||||||
| * `burnPercent: <u8>`, Percentage of fees collected to be destroyed |  | ||||||
| * `maxLamportsPerSignature: <u64>`, Largest value `lamportsPerSignature` can attain for the next slot |  | ||||||
| * `minLamportsPerSignature: <u64>`, Smallest value `lamportsPerSignature` can attain for the next slot |  | ||||||
| * `targetLamportsPerSignature: <u64>`, Desired fee rate for the cluster |  | ||||||
| * `targetSignaturesPerSlot: <u64>`, Desired signature rate for the cluster |  | ||||||
| 
 |  | ||||||
| #### Example: |  | ||||||
| 
 |  | ||||||
| ```bash |  | ||||||
| // Request |  | ||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "method":"getFeeRateGovernor"}' http://localhost:8899 |  | ||||||
| 
 |  | ||||||
| // Result |  | ||||||
| {"jsonrpc":"2.0","result":{"context":{"slot":54},"value":{"feeRateGovernor":{"burnPercent":50,"maxLamportsPerSignature":100000,"minLamportsPerSignature":5000,"targetLamportsPerSignature":10000,"targetSignaturesPerSlot":20000}}},"id":1} |  | ||||||
| ``` |  | ||||||
| 
 |  | ||||||
| ### getFirstAvailableBlock |  | ||||||
| 
 |  | ||||||
| Returns the slot of the lowest confirmed block that has not been purged from the ledger |  | ||||||
| 
 |  | ||||||
| #### Parameters: |  | ||||||
| 
 |  | ||||||
| None |  | ||||||
| 
 |  | ||||||
| #### Results: |  | ||||||
| 
 |  | ||||||
| * `<u64>` - Slot |  | ||||||
| 
 |  | ||||||
| #### Example: |  | ||||||
| 
 |  | ||||||
| ```bash |  | ||||||
| // Request |  | ||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "method":"getFirstAvailableBlock"}' http://localhost:8899 |  | ||||||
| 
 |  | ||||||
| // Result |  | ||||||
| {"jsonrpc":"2.0","result":250000,"id":1} |  | ||||||
| ``` |  | ||||||
| 
 |  | ||||||
| ### getGenesisHash | ### getGenesisHash | ||||||
| 
 | 
 | ||||||
| Returns the genesis hash | Returns the genesis hash | ||||||
| @@ -584,29 +422,6 @@ curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "m | |||||||
| {"jsonrpc":"2.0","result":"GH7ome3EiwEr7tu9JuTh2dpYWBJK3z69Xm1ZE3MEE6JC","id":1} | {"jsonrpc":"2.0","result":"GH7ome3EiwEr7tu9JuTh2dpYWBJK3z69Xm1ZE3MEE6JC","id":1} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ### getIdentity |  | ||||||
| 
 |  | ||||||
| Returns the identity pubkey for the current node |  | ||||||
| 
 |  | ||||||
| #### Parameters: |  | ||||||
| 
 |  | ||||||
| None |  | ||||||
| 
 |  | ||||||
| #### Results: |  | ||||||
| 
 |  | ||||||
| The result field will be a JSON object with the following fields: |  | ||||||
| 
 |  | ||||||
| * `identity`, the identity pubkey of the current node \(as a base-58 encoded string\) |  | ||||||
| 
 |  | ||||||
| #### Example: |  | ||||||
| 
 |  | ||||||
| ```bash |  | ||||||
| // Request |  | ||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "method":"getIdentity"}' http://localhost:8899 |  | ||||||
| // Result |  | ||||||
| {"jsonrpc":"2.0","result":{"identity": "2r1F4iWqVcb8M1DbAjQuFpebkQHY9hcVU4WuW2DJBppN"},"id":1} |  | ||||||
| ``` |  | ||||||
| 
 |  | ||||||
| ### getInflation | ### getInflation | ||||||
| 
 | 
 | ||||||
| Returns the inflation configuration of the cluster | Returns the inflation configuration of the cluster | ||||||
| @@ -636,32 +451,6 @@ curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "m | |||||||
| {"jsonrpc":"2.0","result":{"foundation":0.05,"foundationTerm":7.0,"initial":0.15,"storage":0.1,"taper":0.15,"terminal":0.015},"id":1} | {"jsonrpc":"2.0","result":{"foundation":0.05,"foundationTerm":7.0,"initial":0.15,"storage":0.1,"taper":0.15,"terminal":0.015},"id":1} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ### getLargestAccounts |  | ||||||
| 
 |  | ||||||
| Returns the 20 largest accounts, by lamport balance |  | ||||||
| 
 |  | ||||||
| #### Parameters: |  | ||||||
| 
 |  | ||||||
| * `<object>` - (optional) [Commitment](jsonrpc-api.md#configuring-state-commitment) |  | ||||||
| 
 |  | ||||||
| #### Results: |  | ||||||
| 
 |  | ||||||
| The result will be an RpcResponse JSON object with `value` equal to an array of: |  | ||||||
| 
 |  | ||||||
| * `<object>` - otherwise, a JSON object containing: |  | ||||||
|   * `address: <string>`, base-58 encoded address of the account |  | ||||||
|   * `lamports: <u64>`, number of lamports in the account, as a u64 |  | ||||||
| 
 |  | ||||||
| #### Example: |  | ||||||
| 
 |  | ||||||
| ```bash |  | ||||||
| // Request |  | ||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "method":"getLargestAccounts"}' http://localhost:8899 |  | ||||||
| 
 |  | ||||||
| // Result |  | ||||||
| {"jsonrpc":"2.0","result":{"context":{"slot":54},"value":[{"lamports":999974,"address":"99P8ZgtJYe1buSK8JXkvpLh8xPsCFuLYhz9hQFNw93WJ"},{"lamports":42,"address":"uPwWLo16MVehpyWqsLkK3Ka8nLowWvAHbBChqv2FZeL"},{"lamports":42,"address":"aYJCgU7REfu3XF8b3QhkqgqQvLizx8zxuLBHA25PzDS"},{"lamports":42,"address":"CTvHVtQ4gd4gUcw3bdVgZJJqApXE9nCbbbP4VTS5wE1D"},{"lamports":20,"address":"4fq3xJ6kfrh9RkJQsmVd5gNMvJbuSHfErywvEjNQDPxu"},{"lamports":4,"address":"AXJADheGVp9cruP8WYu46oNkRbeASngN5fPCMVGQqNHa"},{"lamports":2,"address":"8NT8yS6LiwNprgW4yM1jPPow7CwRUotddBVkrkWgYp24"},{"lamports":1,"address":"SysvarEpochSchedu1e111111111111111111111111"},{"lamports":1,"address":"11111111111111111111111111111111"},{"lamports":1,"address":"Stake11111111111111111111111111111111111111"},{"lamports":1,"address":"SysvarC1ock11111111111111111111111111111111"},{"lamports":1,"address":"StakeConfig11111111111111111111111111111111"},{"lamports":1,"address":"SysvarRent111111111111111111111111111111111"},{"lamports":1,"address":"Config1111111111111111111111111111111111111"},{"lamports":1,"address":"SysvarStakeHistory1111111111111111111111111"},{"lamports":1,"address":"SysvarRecentB1ockHashes11111111111111111111"},{"lamports":1,"address":"SysvarFees111111111111111111111111111111111"},{"lamports":1,"address":"Vote111111111111111111111111111111111111111"}]},"id":1} |  | ||||||
| ``` |  | ||||||
| 
 |  | ||||||
| ### getLeaderSchedule | ### getLeaderSchedule | ||||||
| 
 | 
 | ||||||
| Returns the leader schedule for an epoch | Returns the leader schedule for an epoch | ||||||
| @@ -673,10 +462,9 @@ Returns the leader schedule for an epoch | |||||||
| 
 | 
 | ||||||
| #### Results: | #### Results: | ||||||
| 
 | 
 | ||||||
| * `<null>` - if requested epoch is not found | The result field will be a dictionary of leader public keys \(as base-58 encoded | ||||||
| * `<object>` - otherwise, the result field will be a dictionary of leader public keys | strings\) and their corresponding leader slot indices as values (indices are to | ||||||
|   \(as base-58 encoded strings\) and their corresponding leader slot indices as values | the first slot in the requested epoch) | ||||||
|   (indices are to the first slot in the requested epoch) |  | ||||||
| 
 | 
 | ||||||
| #### Example: | #### Example: | ||||||
| 
 | 
 | ||||||
| @@ -711,6 +499,29 @@ curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0", "id":1, " | |||||||
| {"jsonrpc":"2.0","result":500,"id":1} | {"jsonrpc":"2.0","result":500,"id":1} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
|  | ### getNumBlocksSinceSignatureConfirmation | ||||||
|  | 
 | ||||||
|  | Returns the current number of blocks since signature has been confirmed. | ||||||
|  | 
 | ||||||
|  | #### Parameters: | ||||||
|  | 
 | ||||||
|  | * `<string>` - Signature of Transaction to confirm, as base-58 encoded string | ||||||
|  | * `<object>` - (optional) [Commitment](jsonrpc-api.md#configuring-state-commitment) | ||||||
|  | 
 | ||||||
|  | #### Results: | ||||||
|  | 
 | ||||||
|  | * `<u64>` - count, or null if signature not found | ||||||
|  | 
 | ||||||
|  | #### Example: | ||||||
|  | 
 | ||||||
|  | ```bash | ||||||
|  | // Request | ||||||
|  | curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0", "id":1, "method":"getNumBlocksSinceSignatureConfirmation", "params":["5VERv8NMvzbJMEkV8xnrLkEaWRtSz9CosKDYjCJjBRnbJLgp8uirBgmQpjKhoR4tjF3ZpRzrFmBV6UjKdiSZkQUW"]}' http://localhost:8899 | ||||||
|  | 
 | ||||||
|  | // Result | ||||||
|  | {"jsonrpc":"2.0","result":8,"id":1} | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
| ### getProgramAccounts | ### getProgramAccounts | ||||||
| 
 | 
 | ||||||
| Returns all accounts owned by the provided program Pubkey | Returns all accounts owned by the provided program Pubkey | ||||||
| @@ -768,50 +579,57 @@ curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "m | |||||||
| {"jsonrpc":"2.0","result":{"context":{"slot":1},"value":{"blockhash":"CSymwgTNX1j3E4qhKfJAUE41nBWEwXufoYryPbkde5RR","feeCalculator":{"burnPercent":50,"lamportsPerSignature":5000,"maxLamportsPerSignature":100000,"minLamportsPerSignature":5000,"targetLamportsPerSignature":10000,"targetSignaturesPerSlot":20000}}},"id":1} | {"jsonrpc":"2.0","result":{"context":{"slot":1},"value":{"blockhash":"CSymwgTNX1j3E4qhKfJAUE41nBWEwXufoYryPbkde5RR","feeCalculator":{"burnPercent":50,"lamportsPerSignature":5000,"maxLamportsPerSignature":100000,"minLamportsPerSignature":5000,"targetLamportsPerSignature":10000,"targetSignaturesPerSlot":20000}}},"id":1} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ### getSignatureStatuses | ### getSignatureConfirmation | ||||||
| 
 |  | ||||||
| Returns the statuses of a list of signatures. Unless the |  | ||||||
| `searchTransactionHistory` configuration parameter is included, this method only |  | ||||||
| searches the recent status cache of signatures, which retains statuses for all |  | ||||||
| active slots plus `MAX_RECENT_BLOCKHASHES` rooted slots. |  | ||||||
| 
 | 
 | ||||||
|  | Returns the status and number of confirmations of a given signature. | ||||||
| #### Parameters: | #### Parameters: | ||||||
| 
 | 
 | ||||||
| * `<array>` - An array of transaction signatures to confirm, as base-58 encoded strings | * `<string>` - Signature of Transaction to confirm, as base-58 encoded string | ||||||
| * `<object>` - (optional) Configuration object containing the following field: | * `<object>` - (optional) [Commitment](jsonrpc-api.md#configuring-state-commitment) | ||||||
|   * `searchTransactionHistory: <bool>` - if true, a Solana node will search its ledger cache for any signatures not found in the recent status cache |  | ||||||
| 
 | 
 | ||||||
| #### Results: | #### Results: | ||||||
| 
 | 
 | ||||||
| An RpcResponse containing a JSON object consisting of an array of TransactionStatus objects. |  | ||||||
| 
 |  | ||||||
| * `RpcResponse<object>` - RpcResponse JSON object with `value` field: |  | ||||||
| 
 |  | ||||||
| An array of: |  | ||||||
| 
 |  | ||||||
| * `<null>` - Unknown transaction | * `<null>` - Unknown transaction | ||||||
| * `<object>` | * `<object>` - Transaction confirmations and status: | ||||||
|   * `slot: <u64>` - The slot the transaction was processed |   * `confirmations: <u64>` - count of confirmations since transaction was processed | ||||||
|   * `confirmations: <usize | null>` - Number of blocks since signature confirmation, null if rooted, as well as finalized by a supermajority of the cluster |   * `status: <object>` - | ||||||
|   * `err: <object | null>` - Error if transaction failed, null if transaction succeeded. [TransactionError definitions](https://github.com/solana-labs/solana/blob/master/sdk/src/transaction.rs#L14) |  | ||||||
|   * DEPRECATED: `status: <object>` - Transaction status |  | ||||||
|     * `"Ok": <null>` - Transaction was successful |     * `"Ok": <null>` - Transaction was successful | ||||||
|     * `"Err": <ERR>` - Transaction failed with TransactionError |     * `"Err": <ERR>` - Transaction failed with TransactionError  [TransactionError definitions](https://github.com/solana-labs/solana/blob/master/sdk/src/transaction.rs#L14) | ||||||
| 
 | 
 | ||||||
| #### Example: | #### Example: | ||||||
| 
 | 
 | ||||||
| ```bash | ```bash | ||||||
| // Request | // Request | ||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0", "id":1, "method":"getSignatureStatuses", "params":[["5VERv8NMvzbJMEkV8xnrLkEaWRtSz9CosKDYjCJjBRnbJLgp8uirBgmQpjKhoR4tjF3ZpRzrFmBV6UjKdiSZkQUW", "5j7s6NiJS3JAkvgkoc18WVAsiSaci2pxB2A6ueCJP4tprA2TFg9wSyTLeYouxPBJEMzJinENTkpA52YStRW5Dia7"]]}' http://localhost:8899 | curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0", "id":1, "method":"getSignatureConfirmation", "params":["5VERv8NMvzbJMEkV8xnrLkEaWRtSz9CosKDYjCJjBRnbJLgp8uirBgmQpjKhoR4tjF3ZpRzrFmBV6UjKdiSZkQUW"]}' http://localhost:8899 | ||||||
| 
 |  | ||||||
| // Request with configuration |  | ||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0", "id":1, "method":"getSignatureStatuses", "params":[["5VERv8NMvzbJMEkV8xnrLkEaWRtSz9CosKDYjCJjBRnbJLgp8uirBgmQpjKhoR4tjF3ZpRzrFmBV6UjKdiSZkQUW"], {"searchTransactionHistory": true}]}' http://localhost:8899 |  | ||||||
| 
 | 
 | ||||||
| // Result | // Result | ||||||
| {"jsonrpc":"2.0","result":{"context":{"slot":82},"value":[{"slot": 72, "confirmations": 10, "err": null, "status": {"Ok": null}}, null]},"id":1} | {"jsonrpc":"2.0","result":{"confirmations":12,"status":{"Ok": null}},"id":1} | ||||||
|  | ``` | ||||||
| 
 | 
 | ||||||
| // Result, first transaction rooted | ### getSignatureStatus | ||||||
| {"jsonrpc":"2.0","result":{"context":{"slot":82},"value":[{"slot": 48, "confirmations": null, "err": null, "status": {"Ok": null}}, null]},"id":1} | 
 | ||||||
|  | Returns the status of a given signature. This method is similar to [confirmTransaction](jsonrpc-api.md#confirmtransaction) but provides more resolution for error events. | ||||||
|  | 
 | ||||||
|  | #### Parameters: | ||||||
|  | 
 | ||||||
|  | * `<string>` - Signature of Transaction to confirm, as base-58 encoded string | ||||||
|  | * `<object>` - (optional) [Commitment](jsonrpc-api.md#configuring-state-commitment) | ||||||
|  | 
 | ||||||
|  | #### Results: | ||||||
|  | 
 | ||||||
|  | * `<null>` - Unknown transaction | ||||||
|  | * `<object>` - Transaction status: | ||||||
|  |   * `"Ok": <null>` - Transaction was successful | ||||||
|  |   * `"Err": <ERR>` - Transaction failed with TransactionError  [TransactionError definitions](https://github.com/solana-labs/solana/blob/master/sdk/src/transaction.rs#L14) | ||||||
|  | 
 | ||||||
|  | #### Example: | ||||||
|  | 
 | ||||||
|  | ```bash | ||||||
|  | // Request | ||||||
|  | curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0", "id":1, "method":"getSignatureStatus", "params":["5VERv8NMvzbJMEkV8xnrLkEaWRtSz9CosKDYjCJjBRnbJLgp8uirBgmQpjKhoR4tjF3ZpRzrFmBV6UjKdiSZkQUW"]}' http://localhost:8899 | ||||||
|  | 
 | ||||||
|  | // Result | ||||||
|  | {"jsonrpc":"2.0","result":{"Ok": null},"id":1} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ### getSlot | ### getSlot | ||||||
| @@ -945,32 +763,6 @@ curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "m | |||||||
| {"jsonrpc":"2.0","result":1024,"id":1} | {"jsonrpc":"2.0","result":1024,"id":1} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ### getSupply |  | ||||||
| 
 |  | ||||||
| Returns information about the current supply. |  | ||||||
| 
 |  | ||||||
| #### Parameters: |  | ||||||
| 
 |  | ||||||
| * `<object>` - (optional) [Commitment](jsonrpc-api.md#configuring-state-commitment) |  | ||||||
| 
 |  | ||||||
| #### Results: |  | ||||||
| 
 |  | ||||||
| The result will be an RpcResponse JSON object with `value` equal to a JSON object containing: |  | ||||||
| 
 |  | ||||||
| * `total: <u64>` - Total supply in lamports |  | ||||||
| * `circulating: <u64>` - Circulating supply in lamports |  | ||||||
| * `nonCirculating: <u64>` - Non-circulating supply in lamports |  | ||||||
| * `nonCirculatingAccounts: <array>` - an array of account addresses of non-circulating accounts, as strings |  | ||||||
| 
 |  | ||||||
| #### Example: |  | ||||||
| 
 |  | ||||||
| ```bash |  | ||||||
| // Request |  | ||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0", "id":1, "method":"getCirculatingSupply"}' http://localhost:8899 |  | ||||||
| // Result |  | ||||||
| {"jsonrpc":"2.0","result":{"context":{"slot":1114},"value":{"circulating":16000,"nonCirculating":1000000,"nonCirculatingAccounts":["FEy8pTbP5fEoqMV1GdTz83byuA8EKByqYat1PKDgVAq5","9huDUZfxoJ7wGMTffUE7vh1xePqef7gyrLJu9NApncqA","3mi1GmwEE3zo2jmfDuzvjSX9ovRXsDUKHvsntpkhuLJ9","BYxEJTDerkaRWBem3XgnVcdhppktBXa2HbkHPKj2Ui4Z],total:1016000}},"id":1} |  | ||||||
| ``` |  | ||||||
| 
 |  | ||||||
| ### getTransactionCount | ### getTransactionCount | ||||||
| 
 | 
 | ||||||
| Returns the current Transaction count from the ledger | Returns the current Transaction count from the ledger | ||||||
| @@ -993,6 +785,28 @@ curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "m | |||||||
| {"jsonrpc":"2.0","result":268,"id":1} | {"jsonrpc":"2.0","result":268,"id":1} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
|  | ### getTotalSupply | ||||||
|  | 
 | ||||||
|  | Returns the current total supply in lamports | ||||||
|  | 
 | ||||||
|  | #### Parameters: | ||||||
|  | 
 | ||||||
|  | * `<object>` - (optional) [Commitment](jsonrpc-api.md#configuring-state-commitment) | ||||||
|  | 
 | ||||||
|  | #### Results: | ||||||
|  | 
 | ||||||
|  | * `<u64>` - Total supply | ||||||
|  | 
 | ||||||
|  | #### Example: | ||||||
|  | 
 | ||||||
|  | ```bash | ||||||
|  | // Request | ||||||
|  | curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "method":"getTotalSupply"}' http://localhost:8899 | ||||||
|  | 
 | ||||||
|  | // Result | ||||||
|  | {"jsonrpc":"2.0","result":10126,"id":1} | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
| ### getVersion | ### getVersion | ||||||
| 
 | 
 | ||||||
| Returns the current solana versions running on the node | Returns the current solana versions running on the node | ||||||
| @@ -1013,7 +827,7 @@ The result field will be a JSON object with the following fields: | |||||||
| // Request | // Request | ||||||
| curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "method":"getVersion"}' http://localhost:8899 | curl -X POST -H "Content-Type: application/json" -d '{"jsonrpc":"2.0","id":1, "method":"getVersion"}' http://localhost:8899 | ||||||
| // Result | // Result | ||||||
| {"jsonrpc":"2.0","result":{"solana-core": "1.1.12"},"id":1} | {"jsonrpc":"2.0","result":{"solana-core": "0.23.1"},"id":1} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ### getVoteAccounts | ### getVoteAccounts | ||||||
| @@ -1034,7 +848,7 @@ The result field will be a JSON object of `current` and `delinquent` accounts, e | |||||||
| * `epochVoteAccount: <bool>` - bool, whether the vote account is staked for this epoch | * `epochVoteAccount: <bool>` - bool, whether the vote account is staked for this epoch | ||||||
| * `commission: <number>`, percentage (0-100) of rewards payout owed to the vote account | * `commission: <number>`, percentage (0-100) of rewards payout owed to the vote account | ||||||
| * `lastVote: <u64>` - Most recent slot voted on by this vote account | * `lastVote: <u64>` - Most recent slot voted on by this vote account | ||||||
| * `epochCredits: <array>` - History of how many credits earned by the end of each epoch, as an array of arrays containing: `[epoch, credits, previousCredits]` | * `epochCredits: <array>` - History of how many credits earned by the end of each epoch, as an array of arrays containing: [epoch, credits, previousCredits] | ||||||
| 
 | 
 | ||||||
| #### Example: | #### Example: | ||||||
| 
 | 
 | ||||||
| @@ -1099,7 +913,7 @@ Creates new transaction | |||||||
| 
 | 
 | ||||||
| #### Parameters: | #### Parameters: | ||||||
| 
 | 
 | ||||||
| * `<string>` - fully-signed Transaction, as base-58 encoded string | * `<array>` - array of octets containing a fully-signed Transaction | ||||||
| 
 | 
 | ||||||
| #### Results: | #### Results: | ||||||
| 
 | 
 | ||||||
| @@ -1324,7 +1138,7 @@ Subscribe to a transaction signature to receive notification when the transactio | |||||||
| #### Notification Format: | #### Notification Format: | ||||||
| 
 | 
 | ||||||
| ```bash | ```bash | ||||||
| {"jsonrpc": "2.0","method": "signatureNotification", "params": {"result": {"err": null}, "subscription":0}} | {"jsonrpc": "2.0","method": "signatureNotification", "params": {"result": "Confirmed","subscription":0}} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ### signatureUnsubscribe | ### signatureUnsubscribe | ||||||
| @@ -1379,7 +1193,7 @@ None | |||||||
| 
 | 
 | ||||||
| ### slotUnsubscribe | ### slotUnsubscribe | ||||||
| 
 | 
 | ||||||
| Unsubscribe from slot notifications | Unsubscribe from signature confirmation notification | ||||||
| 
 | 
 | ||||||
| #### Parameters: | #### Parameters: | ||||||
| 
 | 
 | ||||||
| @@ -1398,55 +1212,3 @@ Unsubscribe from slot notifications | |||||||
| // Result | // Result | ||||||
| {"jsonrpc": "2.0","result": true,"id": 1} | {"jsonrpc": "2.0","result": true,"id": 1} | ||||||
| ``` | ``` | ||||||
| 
 |  | ||||||
| ### rootSubscribe |  | ||||||
| 
 |  | ||||||
| Subscribe to receive notification anytime a new root is set by the validator. |  | ||||||
| 
 |  | ||||||
| #### Parameters: |  | ||||||
| 
 |  | ||||||
| None |  | ||||||
| 
 |  | ||||||
| #### Results: |  | ||||||
| 
 |  | ||||||
| * `integer` - subscription id \(needed to unsubscribe\) |  | ||||||
| 
 |  | ||||||
| #### Example: |  | ||||||
| 
 |  | ||||||
| ```bash |  | ||||||
| // Request |  | ||||||
| {"jsonrpc":"2.0", "id":1, "method":"rootSubscribe"} |  | ||||||
| 
 |  | ||||||
| // Result |  | ||||||
| {"jsonrpc": "2.0","result": 0,"id": 1} |  | ||||||
| ``` |  | ||||||
| 
 |  | ||||||
| #### Notification Format: |  | ||||||
| 
 |  | ||||||
| The result is the latest root slot number. |  | ||||||
| 
 |  | ||||||
| ```bash |  | ||||||
| {"jsonrpc": "2.0","method": "rootNotification", "params": {"result":42,"subscription":0}} |  | ||||||
| ``` |  | ||||||
| 
 |  | ||||||
| ### rootUnsubscribe |  | ||||||
| 
 |  | ||||||
| Unsubscribe from root notifications |  | ||||||
| 
 |  | ||||||
| #### Parameters: |  | ||||||
| 
 |  | ||||||
| * `<integer>` - subscription id to cancel |  | ||||||
| 
 |  | ||||||
| #### Results: |  | ||||||
| 
 |  | ||||||
| * `<bool>` - unsubscribe success message |  | ||||||
| 
 |  | ||||||
| #### Example: |  | ||||||
| 
 |  | ||||||
| ```bash |  | ||||||
| // Request |  | ||||||
| {"jsonrpc":"2.0", "id":1, "method":"rootUnsubscribe", "params":[0]} |  | ||||||
| 
 |  | ||||||
| // Result |  | ||||||
| {"jsonrpc": "2.0","result": true,"id": 1} |  | ||||||
| ``` |  | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Benchmark a Cluster | # Building from Source | ||||||
| 
 | 
 | ||||||
| The Solana git repository contains all the scripts you might need to spin up your own local testnet. Depending on what you're looking to achieve, you may want to run a different variation, as the full-fledged, performance-enhanced multinode testnet is considerably more complex to set up than a Rust-only, singlenode testnode. If you are looking to develop high-level features, such as experimenting with smart contracts, save yourself some setup headaches and stick to the Rust-only singlenode demo. If you're doing performance optimization of the transaction pipeline, consider the enhanced singlenode demo. If you're doing consensus work, you'll need at least a Rust-only multinode demo. If you want to reproduce our TPS metrics, run the enhanced multinode demo. | The Solana git repository contains all the scripts you might need to spin up your own local testnet. Depending on what you're looking to achieve, you may want to run a different variation, as the full-fledged, performance-enhanced multinode testnet is considerably more complex to set up than a Rust-only, singlenode testnode. If you are looking to develop high-level features, such as experimenting with smart contracts, save yourself some setup headaches and stick to the Rust-only singlenode demo. If you're doing performance optimization of the transaction pipeline, consider the enhanced singlenode demo. If you're doing consensus work, you'll need at least a Rust-only multinode demo. If you want to reproduce our TPS metrics, run the enhanced multinode demo. | ||||||
| 
 | 
 | ||||||
| @@ -121,12 +121,40 @@ thread apply all bt | |||||||
| 
 | 
 | ||||||
| This will dump all the threads stack traces into gdb.txt | This will dump all the threads stack traces into gdb.txt | ||||||
| 
 | 
 | ||||||
| ## Developer Testnet | ### Blockstreamer | ||||||
|  | 
 | ||||||
|  | Solana supports a node type called an _blockstreamer_. This validator variation is intended for applications that need to observe the data plane without participating in transaction validation or ledger replication. | ||||||
|  | 
 | ||||||
|  | A blockstreamer runs without a vote signer, and can optionally stream ledger entries out to a Unix domain socket as they are processed. The JSON-RPC service still functions as on any other node. | ||||||
|  | 
 | ||||||
|  | To run a blockstreamer, include the argument `no-signer` and \(optional\) `blockstream` socket location: | ||||||
|  | 
 | ||||||
|  | ```bash | ||||||
|  | $ NDEBUG=1 ./multinode-demo/validator-x.sh --no-signer --blockstream <SOCKET> | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | The stream will output a series of JSON objects: | ||||||
|  | 
 | ||||||
|  | * An Entry event JSON object is sent when each ledger entry is processed, with the following fields: | ||||||
|  |   * `dt`, the system datetime, as RFC3339-formatted string | ||||||
|  |   * `t`, the event type, always "entry" | ||||||
|  |   * `s`, the slot height, as unsigned 64-bit integer | ||||||
|  |   * `h`, the tick height, as unsigned 64-bit integer | ||||||
|  |   * `entry`, the entry, as JSON object | ||||||
|  | * A Block event JSON object is sent when a block is complete, with the following fields: | ||||||
|  |   * `dt`, the system datetime, as RFC3339-formatted string | ||||||
|  |   * `t`, the event type, always "block" | ||||||
|  |   * `s`, the slot height, as unsigned 64-bit integer | ||||||
|  |   * `h`, the tick height, as unsigned 64-bit integer | ||||||
|  |   * `l`, the slot leader id, as base-58 encoded string | ||||||
|  |   * `hash`, the [blockhash](terminology.md#blockhash), as base-58 encoded string | ||||||
|  | 
 | ||||||
|  | ## Public Testnet | ||||||
| 
 | 
 | ||||||
| In this example the client connects to our public testnet. To run validators on the testnet you would need to open udp ports `8000-10000`. | In this example the client connects to our public testnet. To run validators on the testnet you would need to open udp ports `8000-10000`. | ||||||
| 
 | 
 | ||||||
| ```bash | ```bash | ||||||
| $ NDEBUG=1 ./multinode-demo/bench-tps.sh --entrypoint devnet.solana.com:8001 --faucet devnet.solana.com:9900 --duration 60 --tx_count 50 | $ NDEBUG=1 ./multinode-demo/bench-tps.sh --entrypoint testnet.solana.com:8001 --faucet testnet.solana.com:9900 --duration 60 --tx_count 50 | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| You can observe the effects of your client's transactions on our [metrics dashboard](https://metrics.solana.com:3000/d/monitor/cluster-telemetry?var-testnet=devnet) | You can observe the effects of your client's transactions on our [dashboard](https://metrics.solana.com:3000/d/testnet/testnet-hud?orgId=2&from=now-30m&to=now&refresh=5s&var-testnet=testnet) | ||||||
| @@ -8,7 +8,7 @@ The [solana-cli crate](https://crates.io/crates/solana-cli) provides a command-l | |||||||
| 
 | 
 | ||||||
| ```bash | ```bash | ||||||
| // Command | // Command | ||||||
| $ solana-keygen pubkey | $ solana address | ||||||
| 
 | 
 | ||||||
| // Return | // Return | ||||||
| <PUBKEY> | <PUBKEY> | ||||||
| @@ -22,6 +22,12 @@ $ solana airdrop 2 | |||||||
| 
 | 
 | ||||||
| // Return | // Return | ||||||
| "2.00000000 SOL" | "2.00000000 SOL" | ||||||
|  | 
 | ||||||
|  | // Command | ||||||
|  | $ solana airdrop 123 --lamports | ||||||
|  | 
 | ||||||
|  | // Return | ||||||
|  | "123 lamports" | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ### Get Balance | ### Get Balance | ||||||
							
								
								
									
										5
									
								
								book/src/cli/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								book/src/cli/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | # Using Solana from the Command-line | ||||||
|  |  | ||||||
|  | This chapter describes the command-line tools for interacting with Solana. One | ||||||
|  | could use these tools to send payments, stake validators, and check account | ||||||
|  | balances. | ||||||
							
								
								
									
										1746
									
								
								book/src/cli/usage.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1746
									
								
								book/src/cli/usage.md
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,6 +1,6 @@ | |||||||
| # A Solana Cluster | # A Solana Cluster | ||||||
| 
 | 
 | ||||||
| A Solana cluster is a set of validators working together to serve client transactions and maintain the integrity of the ledger. Many clusters may coexist. When two clusters share a common genesis block, they attempt to converge. Otherwise, they simply ignore the existence of the other. Transactions sent to the wrong one are quietly rejected. In this section, we'll discuss how a cluster is created, how nodes join the cluster, how they share the ledger, how they ensure the ledger is replicated, and how they cope with buggy and malicious nodes. | A Solana cluster is a set of validators working together to serve client transactions and maintain the integrity of the ledger. Many clusters may coexist. When two clusters share a common genesis block, they attempt to converge. Otherwise, they simply ignore the existence of the other. Transactions sent to the wrong one are quietly rejected. In this chapter, we'll discuss how a cluster is created, how nodes join the cluster, how they share the ledger, how they ensure the ledger is replicated, and how they cope with buggy and malicious nodes. | ||||||
| 
 | 
 | ||||||
| ## Creating a Cluster | ## Creating a Cluster | ||||||
| 
 | 
 | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| # Fork Generation | # Fork Generation | ||||||
| 
 | 
 | ||||||
| This section describes how forks naturally occur as a consequence of [leader rotation](leader-rotation.md). | The chapter describes how forks naturally occur as a consequence of [leader rotation](leader-rotation.md). | ||||||
| 
 | 
 | ||||||
| ## Overview | ## Overview | ||||||
| 
 | 
 | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| # Stake Delegation and Rewards | # Stake Delegation and Rewards | ||||||
| 
 | 
 | ||||||
| Stakers are rewarded for helping to validate the ledger. They do this by delegating their stake to validator nodes. Those validators do the legwork of replaying the ledger and send votes to a per-node vote account to which stakers can delegate their stakes. The rest of the cluster uses those stake-weighted votes to select a block when forks arise. Both the validator and staker need some economic incentive to play their part. The validator needs to be compensated for its hardware and the staker needs to be compensated for the risk of getting its stake slashed. The economics are covered in [staking rewards](../implemented-proposals/staking-rewards.md). This section, on the other hand, describes the underlying mechanics of its implementation. | Stakers are rewarded for helping to validate the ledger. They do this by delegating their stake to validator nodes. Those validators do the legwork of replaying the ledger and send votes to a per-node vote account to which stakers can delegate their stakes. The rest of the cluster uses those stake-weighted votes to select a block when forks arise. Both the validator and staker need some economic incentive to play their part. The validator needs to be compensated for its hardware and the staker needs to be compensated for the risk of getting its stake slashed. The economics are covered in [staking rewards](../implemented-proposals/staking-rewards.md). This chapter, on the other hand, describes the underlying mechanics of its implementation. | ||||||
| 
 | 
 | ||||||
| ## Basic Design | ## Basic Design | ||||||
| 
 | 
 | ||||||
| @@ -29,7 +29,11 @@ VoteState is the current state of all the votes the validator has submitted to t | |||||||
| * Account::lamports - The accumulated lamports from the commission. These do not count as stakes. | * Account::lamports - The accumulated lamports from the commission. These do not count as stakes. | ||||||
| * `authorized_voter` - Only this identity is authorized to submit votes. This field can only modified by this identity. | * `authorized_voter` - Only this identity is authorized to submit votes. This field can only modified by this identity. | ||||||
| * `node_pubkey` - The Solana node that votes in this account. | * `node_pubkey` - The Solana node that votes in this account. | ||||||
| * `authorized_withdrawer` - the identity of the entity in charge of the lamports of this account, separate from the account's address and the authorized vote signer | * `authorized_withdrawer` - the identity of the entity in charge of the lamports of this account, separate from the account's | ||||||
|  | 
 | ||||||
|  |   ```text | ||||||
|  |                        address and the authorized vote signer | ||||||
|  |   ``` | ||||||
| 
 | 
 | ||||||
| ### VoteInstruction::Initialize\(VoteInit\) | ### VoteInstruction::Initialize\(VoteInit\) | ||||||
| 
 | 
 | ||||||
| @@ -44,11 +48,13 @@ VoteState is the current state of all the votes the validator has submitted to t | |||||||
| Updates the account with a new authorized voter or withdrawer, according to the VoteAuthorize parameter \(`Voter` or `Withdrawer`\). The transaction must be by signed by the Vote account's current `authorized_voter` or `authorized_withdrawer`. | Updates the account with a new authorized voter or withdrawer, according to the VoteAuthorize parameter \(`Voter` or `Withdrawer`\). The transaction must be by signed by the Vote account's current `authorized_voter` or `authorized_withdrawer`. | ||||||
| 
 | 
 | ||||||
| * `account[0]` - RW - The VoteState | * `account[0]` - RW - The VoteState | ||||||
|  | 
 | ||||||
|   `VoteState::authorized_voter` or `authorized_withdrawer` is set to to `Pubkey`. |   `VoteState::authorized_voter` or `authorized_withdrawer` is set to to `Pubkey`. | ||||||
| 
 | 
 | ||||||
| ### VoteInstruction::Vote\(Vote\) | ### VoteInstruction::Vote\(Vote\) | ||||||
| 
 | 
 | ||||||
| * `account[0]` - RW - The VoteState | * `account[0]` - RW - The VoteState | ||||||
|  | 
 | ||||||
|   `VoteState::lockouts` and `VoteState::credits` are updated according to voting lockout rules see [Tower BFT](../implemented-proposals/tower-bft.md) |   `VoteState::lockouts` and `VoteState::credits` are updated according to voting lockout rules see [Tower BFT](../implemented-proposals/tower-bft.md) | ||||||
| 
 | 
 | ||||||
| * `account[1]` - RO - `sysvar::slot_hashes` A list of some N most recent slots and their hashes for the vote to be verified against. | * `account[1]` - RO - `sysvar::slot_hashes` A list of some N most recent slots and their hashes for the vote to be verified against. | ||||||
| @@ -67,10 +73,18 @@ StakeState::Stake is the current delegation preference of the **staker** and con | |||||||
| * `voter_pubkey` - The pubkey of the VoteState instance the lamports are delegated to. | * `voter_pubkey` - The pubkey of the VoteState instance the lamports are delegated to. | ||||||
| * `credits_observed` - The total credits claimed over the lifetime of the program. | * `credits_observed` - The total credits claimed over the lifetime of the program. | ||||||
| * `activated` - the epoch at which this stake was activated/delegated. The full stake will be counted after warm up. | * `activated` - the epoch at which this stake was activated/delegated. The full stake will be counted after warm up. | ||||||
| * `deactivated` - the epoch at which this stake was de-activated, some cool down epochs are required before the account is fully deactivated, and the stake available for withdrawal | * `deactivated` - the epoch at which this stake was de-activated, some cool down epochs are required before the account | ||||||
|  | 
 | ||||||
|  |   ```text | ||||||
|  |               is fully deactivated, and the stake available for withdrawal | ||||||
|  |   ``` | ||||||
| 
 | 
 | ||||||
| * `authorized_staker` - the pubkey of the entity that must sign delegation, activation, and deactivation transactions | * `authorized_staker` - the pubkey of the entity that must sign delegation, activation, and deactivation transactions | ||||||
| * `authorized_withdrawer` - the identity of the entity in charge of the lamports of this account, separate from the account's address, and the authorized staker | * `authorized_withdrawer` - the identity of the entity in charge of the lamports of this account, separate from the account's | ||||||
|  | 
 | ||||||
|  |   ```text | ||||||
|  |                        address, and the authorized staker | ||||||
|  |   ``` | ||||||
| 
 | 
 | ||||||
| ### StakeState::RewardsPool | ### StakeState::RewardsPool | ||||||
| 
 | 
 | ||||||
| @@ -154,7 +168,7 @@ Stakers who have delegated to that validator earn points in proportion to their | |||||||
| 
 | 
 | ||||||
| Stakes, once delegated, do not become effective immediately. They must first pass through a warm up period. During this period some portion of the stake is considered "effective", the rest is considered "activating". Changes occur on epoch boundaries. | Stakes, once delegated, do not become effective immediately. They must first pass through a warm up period. During this period some portion of the stake is considered "effective", the rest is considered "activating". Changes occur on epoch boundaries. | ||||||
| 
 | 
 | ||||||
| The stake program limits the rate of change to total network stake, reflected in the stake program's `config::warmup_rate` \(set to 25% per epoch in the current implementation\). | The stake program limits the rate of change to total network stake, reflected in the stake program's `config::warmup_rate` \(typically 25% per epoch\). | ||||||
| 
 | 
 | ||||||
| The amount of stake that can be warmed up each epoch is a function of the previous epoch's total effective stake, total activating stake, and the stake program's configured warmup rate. | The amount of stake that can be warmed up each epoch is a function of the previous epoch's total effective stake, total activating stake, and the stake program's configured warmup rate. | ||||||
| 
 | 
 | ||||||
| @@ -6,7 +6,7 @@ Solana takes a very different approach, which it calls _Proof of History_ or _Po | |||||||
| 
 | 
 | ||||||
| Solana technically never sends a _block_, but uses the term to describe the sequence of entries that validators vote on to achieve _confirmation_. In that way, Solana's confirmation times can be compared apples to apples to block-based systems. The current implementation sets block time to 800ms. | Solana technically never sends a _block_, but uses the term to describe the sequence of entries that validators vote on to achieve _confirmation_. In that way, Solana's confirmation times can be compared apples to apples to block-based systems. The current implementation sets block time to 800ms. | ||||||
| 
 | 
 | ||||||
| What's happening under the hood is that entries are streamed to validators as quickly as a leader node can batch a set of valid transactions into an entry. Validators process those entries long before it is time to vote on their validity. By processing the transactions optimistically, there is effectively no delay between the time the last entry is received and the time when the node can vote. In the event consensus is **not** achieved, a node simply rolls back its state. This optimisic processing technique was introduced in 1981 and called [Optimistic Concurrency Control](http://citeseerx.ist.psu.edu/viewdoc/summary?doi=10.1.1.125.4735). It can be applied to blockchain architecture where a cluster votes on a hash that represents the full ledger up to some _block height_. In Solana, it is implemented trivially using the last entry's PoH hash. | What's happening under the hood is that entries are streamed to validators as quickly as a leader node can batch a set of valid transactions into an entry. Validators process those entries long before it is time to vote on their validity. By processing the transactions optimistically, there is effectively no delay between the time the last entry is received and the time when the node can vote. In the event consensus is **not** achieved, a node simply rolls back its state. This optimisic processing technique was introduced in 1981 and called [Optimistic Concurrency Control](http://citeseerx.ist.psu.edu/viewdoc/summary?doi=10.1.1.65.4735). It can be applied to blockchain architecture where a cluster votes on a hash that represents the full ledger up to some _block height_. In Solana, it is implemented trivially using the last entry's PoH hash. | ||||||
| 
 | 
 | ||||||
| ## Relationship to VDFs | ## Relationship to VDFs | ||||||
| 
 | 
 | ||||||
| @@ -8,7 +8,7 @@ During its slot, the leader node distributes shreds between the validator nodes | |||||||
| 
 | 
 | ||||||
| In order for data plane fanout to work, the entire cluster must agree on how the cluster is divided into neighborhoods. To achieve this, all the recognized validator nodes \(the TVU peers\) are sorted by stake and stored in a list. This list is then indexed in different ways to figure out neighborhood boundaries and retransmit peers. For example, the leader will simply select the first nodes to make up layer 0. These will automatically be the highest stake holders, allowing the heaviest votes to come back to the leader first. Layer-0 and lower-layer nodes use the same logic to find their neighbors and next layer peers. | In order for data plane fanout to work, the entire cluster must agree on how the cluster is divided into neighborhoods. To achieve this, all the recognized validator nodes \(the TVU peers\) are sorted by stake and stored in a list. This list is then indexed in different ways to figure out neighborhood boundaries and retransmit peers. For example, the leader will simply select the first nodes to make up layer 0. These will automatically be the highest stake holders, allowing the heaviest votes to come back to the leader first. Layer-0 and lower-layer nodes use the same logic to find their neighbors and next layer peers. | ||||||
| 
 | 
 | ||||||
| To reduce the possibility of attack vectors, each shred is transmitted over a random tree of neighborhoods. Each node uses the same set of nodes representing the cluster. A random tree is generated from the set for each shred using a seed derived from the leader id, slot and shred index. | To reduce the possibility of attack vectors, each shred is transmitted over a random tree of neighborhoods. Each node uses the same set of nodes representing the cluster. A random tree is generated from the set for each shred using randomness derived from the shred itself. Since the random seed is not known in advance, attacks that try to eclipse neighborhoods from certain leaders or blocks become very difficult, and should require almost complete control of the stake in the cluster. | ||||||
| 
 | 
 | ||||||
| ## Layer and Neighborhood Structure | ## Layer and Neighborhood Structure | ||||||
| 
 | 
 | ||||||
| @@ -10,8 +10,6 @@ These protocol-based rewards, to be distributed to participating validation and | |||||||
| 
 | 
 | ||||||
| Transaction fees are market-based participant-to-participant transfers, attached to network interactions as a necessary motivation and compensation for the inclusion and execution of a proposed transaction \(be it a state execution or proof-of-replication verification\). A mechanism for long-term economic stability and forking protection through partial burning of each transaction fee is also discussed below. | Transaction fees are market-based participant-to-participant transfers, attached to network interactions as a necessary motivation and compensation for the inclusion and execution of a proposed transaction \(be it a state execution or proof-of-replication verification\). A mechanism for long-term economic stability and forking protection through partial burning of each transaction fee is also discussed below. | ||||||
| 
 | 
 | ||||||
| A high-level schematic of Solana’s crypto-economic design is shown below in **Figure 1**. The specifics of validation-client economics are described in sections: [Validation-client Economics](ed_validation_client_economics/README.md), [State-validation Protocol-based Rewards](ed_validation_client_economics/ed_vce_state_validation_protocol_based_rewards.md), [State-validation Transaction Fees](ed_validation_client_economics/ed_vce_state_validation_transaction_fees.md) and [Replication-validation Transaction Fees](ed_validation_client_economics/ed_vce_replication_validation_transaction_fees.md). Also, the section titled [Validation Stake Delegation](ed_validation_client_economics/ed_vce_validation_stake_delegation.md) closes with a discussion of validator delegation opportunities and marketplace. Additionally, in [Storage Rent Economics](ed_storage_rent_economics.md), we describe an implementation of storage rent to account for the externality costs of maintaining the active state of the ledger. [Replication-client Economics](ed_replication_client_economics/README.md) will review the Solana network design for global ledger storage/redundancy and archiver-client economics \([Storage-replication rewards](ed_replication_client_economics/ed_rce_storage_replication_rewards.md)\) along with an archiver-to-validator delegation mechanism designed to aide participant on-boarding into the Solana economy discussed in [Replication-client Reward Auto-delegation](ed_replication_client_economics/ed_rce_replication_client_reward_auto_delegation.md). An outline of features for an MVP economic design is discussed in the [Economic Design MVP](ed_mvp.md) section. Finally, in [Attack Vectors](ed_attack_vectors.md), various attack vectors will be described and potential vulnerabilities explored and parameterized. | A high-level schematic of Solana’s crypto-economic design is shown below in **Figure 1**. The specifics of validation-client economics are described in sections: [Validation-client Economics](ed_validation_client_economics/), [State-validation Protocol-based Rewards](ed_validation_client_economics/ed_vce_state_validation_protocol_based_rewards.md), [State-validation Transaction Fees](ed_validation_client_economics/ed_vce_state_validation_transaction_fees.md) and [Replication-validation Transaction Fees](ed_validation_client_economics/ed_vce_replication_validation_transaction_fees.md). Also, the chapter titled [Validation Stake Delegation](ed_validation_client_economics/ed_vce_validation_stake_delegation.md) closes with a discussion of validator delegation opportunties and marketplace. Additionally, in [Storage Rent Economics](ed_storage_rent_economics.md), we describe an implementation of storage rent to account for the externality costs of maintaining the active state of the ledger. The [Replication-client Economics](ed_replication_client_economics/) chapter will review the Solana network design for global ledger storage/redundancy and archiver-client economics \([Storage-replication rewards](ed_replication_client_economics/ed_rce_storage_replication_rewards.md)\) along with an archiver-to-validator delegation mechanism designed to aide participant on-boarding into the Solana economy discussed in [Replication-client Reward Auto-delegation](ed_replication_client_economics/ed_rce_replication_client_reward_auto_delegation.md). An outline of features for an MVP economic design is discussed in the [Economic Design MVP](ed_mvp.md) section. Finally, in chapter [Attack Vectors](ed_attack_vectors.md), various attack vectors will be described and potential vulnerabilities explored and parameterized. | ||||||
| 
 |  | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| **Figure 1**: Schematic overview of Solana economic incentive design. | **Figure 1**: Schematic overview of Solana economic incentive design. | ||||||
| @@ -2,15 +2,13 @@ | |||||||
| 
 | 
 | ||||||
| **Subject to change.** | **Subject to change.** | ||||||
| 
 | 
 | ||||||
| Long term economic sustainability is one of the guiding principles of Solana’s economic design. While it is impossible to predict how decentralized economies will develop over time, especially economies with flexible decentralized governances, we can arrange economic components such that, under certain conditions, a sustainable economy may take shape in the long term. In the case of Solana’s network, these components take the form of token issuance \(via inflation\) and token burning. | Long term economic sustainability is one of the guiding principles of Solana’s economic design. While it is impossible to predict how decentralized economies will develop over time, especially economies with flexible decentralized governances, we can arrange economic components such that, under certain conditions, a sustainable economy may take shape in the long term. In the case of Solana’s network, these components take the form of token issuance \(via inflation\) and token burning’. | ||||||
| 
 | 
 | ||||||
| The dominant remittances from the Solana mining pool are validator and archiver rewards. The disinflationary mechanism is a flat, protocol-specified and adjusted, % of each transaction fee. | The dominant remittances from the Solana mining pool are validator and archiver rewards. The disinflationary mechanism is a flat, protocol-specified and adjusted, % of each transaction fee. | ||||||
| 
 | 
 | ||||||
| The Archiver rewards are to be delivered to archivers as a portion of the network inflation after successful PoRep validation. The per-PoRep reward amount is determined as a function of the total network storage redundancy at the time of the PoRep validation and the network goal redundancy. This function is likely to take the form of a discount from a base reward to be delivered when the network has achieved and maintained its goal redundancy. An example of such a reward function is shown in **Figure 1** | The Archiver rewards are to be delivered to archivers as a portion of the network inflation after successful PoRep validation. The per-PoRep reward amount is determined as a function of the total network storage redundancy at the time of the PoRep validation and the network goal redundancy. This function is likely to take the form of a discount from a base reward to be delivered when the network has achieved and maintained its goal redundancy. An example of such a reward function is shown in **Figure 3** | ||||||
| 
 | 
 | ||||||
|  | **Figure 3**: Example PoRep reward design as a function of global network storage redundancy. | ||||||
| 
 | 
 | ||||||
| **Figure 1**: Example PoRep reward design as a function of global network storage redundancy. | In the example shown in Figure 1, multiple per PoRep base rewards are explored \(as a % of Tx Fee\) to be delivered when the global ledger replication redundancy meets 10X. When the global ledger replication redundancy is less than 10X, the base reward is discounted as a function of the square of the ratio of the actual ledger replication redundancy to the goal redundancy \(i.e. 10X\). | ||||||
| 
 |  | ||||||
| In the example shown in **Figure 1**, multiple per PoRep base rewards are explored \(as a % of Tx Fee\) to be delivered when the global ledger replication redundancy meets 10X. When the global ledger replication redundancy is less than 10X, the base reward is discounted as a function of the square of the ratio of the actual ledger replication redundancy to the goal redundancy \(i.e. 10X\). |  | ||||||
| 
 | 
 | ||||||
| @@ -2,7 +2,7 @@ | |||||||
| 
 | 
 | ||||||
| **Subject to change.** | **Subject to change.** | ||||||
| 
 | 
 | ||||||
| The preceding sections, outlined in the [Economic Design Overview](../README.md), describe a long-term vision of a sustainable Solana economy. Of course, we don't expect the final implementation to perfectly match what has been described above. We intend to fully engage with network stakeholders throughout the implementation phases \(i.e. pre-testnet, testnet, mainnet\) to ensure the system supports, and is representative of, the various network participants' interests. The first step toward this goal, however, is outlining a some desired MVP economic features to be available for early pre-testnet and testnet participants. Below is a rough sketch outlining basic economic functionality from which a more complete and functional system can be developed. | The preceeding sections, outlined in the [Economic Design Overview](./), describe a long-term vision of a sustainable Solana economy. Of course, we don't expect the final implementation to perfectly match what has been described above. We intend to fully engage with network stakeholders throughout the implementation phases \(i.e. pre-testnet, testnet, mainnet\) to ensure the system supports, and is representative of, the various network participants' interests. The first step toward this goal, however, is outlining a some desired MVP economic features to be available for early pre-testnet and testnet participants. Below is a rough sketch outlining basic economic functionality from which a more complete and functional system can be developed. | ||||||
| 
 | 
 | ||||||
| ## MVP Economic Features | ## MVP Economic Features | ||||||
| 
 | 
 | ||||||
| @@ -2,5 +2,5 @@ | |||||||
| 
 | 
 | ||||||
| **Subject to change.** | **Subject to change.** | ||||||
| 
 | 
 | ||||||
| Replication-clients should be rewarded for providing the network with storage space. Incentivization of the set of archivers provides data security through redundancy of the historical ledger. Replication nodes are rewarded in proportion to the amount of ledger data storage provided, as proved by successfully submitting Proofs-of-Replication to the cluster.. These rewards are captured by generating and entering Proofs of Replication \(PoReps\) into the PoH stream which can be validated by Validation nodes as described in [Replication-validation Transaction Fees](../ed_validation_client_economics/ed_vce_replication_validation_transaction_fees.md). | Replication-clients should be rewarded for providing the network with storage space. Incentivization of the set of archivers provides data security through redundancy of the historical ledger. Replication nodes are rewarded in proportion to the amount of ledger data storage provided, as proved by successfully submitting Proofs-of-Replication to the cluster.. These rewards are captured by generating and entering Proofs of Replication \(PoReps\) into the PoH stream which can be validated by Validation nodes as described above in the [Replication-validation Transaction Fees](../ed_validation_client_economics/ed_vce_replication_validation_transaction_fees.md) chapter. | ||||||
| 
 | 
 | ||||||
| @@ -1,18 +1,18 @@ | |||||||
| ## Storage Rent Economics | ## Storage Rent Economics | ||||||
| 
 | 
 | ||||||
| Each transaction that is submitted to the Solana ledger imposes costs. Transaction fees paid by the submitter, and collected by a validator, in theory, account for the acute, transactional, costs of validating and adding that data to the ledger. At the same time, our compensation design for archivers (see [Replication-client Economics](ed_replication_client_economics/README.md)), in theory, accounts for the long term storage of the historical ledger. Unaccounted in this process is the mid-term storage of active ledger state, necessarily maintained by the rotating validator set. This type of storage imposes costs not only to validators but also to the broader network as active state grows so does data transmission and validation overhead. To account for these costs, we describe here our preliminary design and implementation of storage rent. | Each transaction that is submitted to the Solana ledger imposes costs. Transaction fees paid by the submitter, and collected by a validator, in theory, account for the acute, transacitonal, costs of validating and adding that data to the ledger. At the same time, our compensation design for archivers (see [Replication-client Economics](ed_replication_client_economics.md)), in theory, accounts for the long term storage of the historical ledger. Unaccounted in this process is the mid-term storage of active ledger state, necessarily maintined by the rotating validator set. This type of storage imposes costs not only to validators but also to the broader network as active state grows so does data transmission and validation overhead. To account for these costs, we describe here our preliminary design and implementation of storage rent.  | ||||||
| 
 | 
 | ||||||
| Storage rent can be paid via one of two methods: | Storage rent can be paid via one of two methods: | ||||||
| 
 | 
 | ||||||
| Method 1: Set it and forget it | Method 1: Set it and forget it | ||||||
| 
 | 
 | ||||||
| With this approach, accounts with two-years worth of rent deposits secured are exempt from network rent charges. By maintaining this minimum-balance, the broader network benefits from reduced liquidity and the account holder can trust that their `Account::data` will be retained for continual access/usage. | With this approach, accounts with two-years worth of rent deposits secured are exempt from network rent charges. By maintaining this minimum-balance, the broader network benefits from reduced liquitity and the account holder can trust that their `Account::data` will be retained for continual access/usage.  | ||||||
| 
 | 
 | ||||||
| Method 2: Pay per byte | Method 2: Pay per byte | ||||||
| 
 | 
 | ||||||
| If an account has less than two-years worth of deposited rent the network charges rent on a per-epoch basis, in credit for the next epoch (but in arrears when necessary). This rent is deducted at a rate specified in genesis, in lamports per kilobyte-year. | If an account has less than two-years worth of deposited rent the network charges rent on a per-epoch basis, in credit for the next epoch (but in arrears when necessary). This rent is deducted at a rate specified in genesis, in lamports per kilobyte-year. | ||||||
| 
 | 
 | ||||||
| For information on the technical implementation details of this design, see the [Rent](../rent.md) section. | For information on the technical implementation details of this design, see the [Rent](rent.md) section.  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @@ -0,0 +1,11 @@ | |||||||
|  | # Replication-validation Transaction Fees | ||||||
|  |  | ||||||
|  | **Subject to change.** | ||||||
|  |  | ||||||
|  | As previously mentioned, validator-clients will also be responsible for validating PoReps submitted into the PoH stream by archiver-clients. In this case, validators are providing compute \(CPU/GPU\) and light storage resources to confirm that these replication proofs could only be generated by a client that is storing the referenced PoH leger block. | ||||||
|  |  | ||||||
|  | While replication-clients are incentivized and rewarded through protocol-based rewards schedule \(see [Replication-client Economics](../ed_replication_client_economics/)\), validator-clients will be incentivized to include and validate PoReps in PoH through collection of transaction fees associated with the submitted PoReps and distribution of protocol rewards proportional to the validated PoReps. As will be described in detail in the Section 3.1, replication-client rewards are protocol-based and designed to reward based on a global data redundancy factor. I.e. the protocol will incentivize replication-client participation through rewards based on a target ledger redundancy \(e.g. 10x data redundancy\). | ||||||
|  |  | ||||||
|  | The validation of PoReps by validation-clients is computationally more expensive than state-validation \(detail in the [Economic Sustainability](../ed_economic_sustainability.md) chapter\), thus the transaction fees are expected to be proportionally higher. | ||||||
|  |  | ||||||
|  | There are various attack vectors available for colluding validation and replication clients, also described in detail below in [Economic Sustainability](../ed_economic_sustainability/README.md). To protect against various collusion attack vectors, for a given epoch, validator rewards are distributed across participating validation-clients in proportion to the number of validated PoReps in the epoch less the number of PoReps that mismatch the archivers challenge. The PoRep challenge game is described in [Ledger Replication](https://github.com/solana-labs/solana/blob/master/book/src/ledger-replication.md#the-porep-game). This design rewards validators proportional to the number of PoReps they process and validate, while providing negative pressure for validation-clients to submit lazy or malicious invalid votes on submitted PoReps \(note that it is computationally prohibitive to determine whether a validator-client has marked a valid PoRep as invalid\). | ||||||
| @@ -4,31 +4,27 @@ | |||||||
| 
 | 
 | ||||||
| Validator-clients have two functional roles in the Solana network: | Validator-clients have two functional roles in the Solana network: | ||||||
| 
 | 
 | ||||||
| * Validate \(vote\) the current global state of that PoH along with any Proofs-of-Replication \(see [Replication Client Economics](../ed_replication_client_economics/README.md)\) that they are eligible to validate. | * Validate \(vote\) the current global state of that PoH along with any Proofs-of-Replication \(see [Replication Client Economics](../ed_replication_client_economics/)\) that they are eligible to validate. | ||||||
| * Be elected as ‘leader’ on a stake-weighted round-robin schedule during which time they are responsible for collecting outstanding transactions and Proofs-of-Replication and incorporating them into the PoH, thus updating the global state of the network and providing chain continuity. | * Be elected as ‘leader’ on a stake-weighted round-robin schedule during which time they are responsible for collecting outstanding transactions and Proofs-of-Replication and incorporating them into the PoH, thus updating the global state of the network and providing chain continuity. | ||||||
| 
 | 
 | ||||||
| Validator-client rewards for these services are to be distributed at the end of each Solana epoch. As previously discussed, compensation for validator-clients is provided via a protocol-based annual inflation rate dispersed in proportion to the stake-weight of each validator \(see below\) along with leader-claimed transaction fees available during each leader rotation. I.e. during the time a given validator-client is elected as leader, it has the opportunity to keep a portion of each transaction fee, less a protocol-specified amount that is destroyed \(see [Validation-client State Transaction Fees](ed_vce_state_validation_transaction_fees.md)\). PoRep transaction fees are also collected by the leader client and validator PoRep rewards are distributed in proportion to the number of validated PoReps less the number of PoReps that mismatch an archiver's challenge. \(see [Replication-client Transaction Fees](ed_vce_replication_validation_transaction_fees.md)\) | Validator-client rewards for these services are to be distributed at the end of each Solana epoch. As previously discussed, compensation for validator-clients is provided via a protocol-based annual inflation rate dispersed in proportion to the stake-weight of each validator \(see below\) along with leader-claimed transaction fees available during each leader rotation. I.e. during the time a given validator-client is elected as leader, it has the opportunity to keep a portion of each transaction fee, less a protocol-specified amount that is destroyed \(see [Validation-client State Transaction Fees](ed_vce_state_validation_transaction_fees.md)\). PoRep transaction fees are also collected by the leader client and validator PoRep rewards are distributed in proportion to the number of validated PoReps less the number of PoReps that mismatch an archiver's challenge. \(see [Replication-client Transaction Fees](ed_vce_replication_validation_transaction_fees.md)\) | ||||||
| 
 | 
 | ||||||
| The effective protocol-based annual interest rate \(%\) per epoch received by validation-clients is to be a function of: | The effective protocol-based annual interest rate \(%\) per epoch received by validation-clients is to be a function of: | ||||||
| 
 | 
 | ||||||
| * the current global inflation rate, derived from the pre-determined dis-inflationary issuance schedule \(see [Validation-client Economics](README.md)\) | * the current global inflation rate, derived from the pre-determined dis-inflationary issuance schedule \(see [Validation-client Economics](.)\) | ||||||
| * the fraction of staked SOLs out of the current total circulating supply, | * the fraction of staked SOLs out of the current total circulating supply, | ||||||
| * the up-time/participation \[% of available slots that validator had opportunity to vote on\] of a given validator over the previous epoch. | * the up-time/participation \[% of available slots that validator had opportunity to vote on\] of a given validator over the previous epoch. | ||||||
| 
 | 
 | ||||||
| The first factor is a function of protocol parameters only \(i.e. independent of validator behavior in a given epoch\) and results in a global validation reward schedule designed to incentivize early participation, provide clear monetary stability and provide optimal security in the network. | The first factor is a function of protocol parameters only \(i.e. independent of validator behavior in a given epoch\) and results in a global validation reward schedule designed to incentivize early participation, provide clear montetary stability and provide optimal security in the network. | ||||||
| 
 | 
 | ||||||
| At any given point in time, a specific validator's interest rate can be determined based on the proportion of circulating supply that is staked by the network and the validator's uptime/activity in the previous epoch. For example, consider a hypothetical instance of the network with an initial circulating token supply of 250MM tokens with an additional 250MM vesting over 3 years. Additionally an inflation rate is specified at network launch of 7.5%, and a disinflationary schedule of 20% decrease in inflation rate per year \(the actual rates to be implemented are to be worked out during the testnet experimentation phase of mainnet launch\). With these broad assumptions, the 10-year inflation rate \(adjusted daily for this example\) is shown in **Figure 1**, while the total circulating token supply is illustrated in **Figure 2**. Neglected in this toy-model is the inflation suppression due to the portion of each transaction fee that is to be destroyed. | At any given point in time, a specific validator's interest rate can be determined based on the porportion of circulating supply that is staked by the network and the validator's uptime/activity in the previous epoch. For example, consider a hypothetical instance of the network with an initial circulating token supply of 250MM tokens with an additional 250MM vesting over 3 years. Additionally an inflation rate is specified at network launch of 7.5%, and a disinflationary schedule of 20% decrease in inflation rate per year \(the actual rates to be implemented are to be worked out during the testnet experimentation phase of mainnet launch\). With these broad assumptions, the 10-year inflation rate \(adjusted daily for this example\) is shown in **Figure 2**, while the total circulating token supply is illustrated in **Figure 3**. Neglected in this toy-model is the inflation supression due to the portion of each transaction fee that is to be destroyed. | ||||||
| 
 | 
 | ||||||
|  |  \*\*Figure 2:\*\* In this example schedule, the annual inflation rate \[%\] reduces at around 20% per year, until it reaches the long-term, fixed, 1.5% rate. | ||||||
| 
 | 
 | ||||||
| **Figure 1:** In this example schedule, the annual inflation rate \[%\] reduces at around 20% per year, until it reaches the long-term, fixed, 1.5% rate. |  \*\*Figure 3:\*\* The total token supply over a 10-year period, based on an initial 250MM tokens with the disinflationary inflation schedule as shown in \*\*Figure 2\*\* Over time, the interest rate, at a fixed network staked percentage, will reduce concordant with network inflation. Validation-client interest rates are designed to be higher in the early days of the network to incentivize participation and jumpstart the network economy. As previously mentioned, the inflation rate is expected to stabalize near 1-2% which also results in a fixed, long-term, interest rate to be provided to validator-clients. This value does not represent the total interest available to validator-clients as transaction fees for state-validation and ledger storage replication \(PoReps\) are not accounted for here. Given these example parameters, annualized validator-specific interest rates can be determined based on the global fraction of tokens bonded as stake, as well as their uptime/activity in the previous epoch. For the purpose of this example, we assume 100% uptime for all validators and a split in interest-based rewards between validators and archiver nodes of 80%/20%. Additionally, the fraction of staked circulating supply is assummed to be constant. Based on these assumptions, an annualized validation-client interest rate schedule as a function of % circulating token supply that is staked is shown in\*\* Figure 4\*\*. | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
| 
 | 
 | ||||||
| **Figure 2:** The total token supply over a 10-year period, based on an initial 250MM tokens with the disinflationary inflation schedule as shown in **Figure 1**. Over time, the interest rate, at a fixed network staked percentage, will reduce concordant with network inflation. Validation-client interest rates are designed to be higher in the early days of the network to incentivize participation and jumpstart the network economy. As previously mentioned, the inflation rate is expected to stabilize near 1-2% which also results in a fixed, long-term, interest rate to be provided to validator-clients. This value does not represent the total interest available to validator-clients as transaction fees for state-validation and ledger storage replication \(PoReps\) are not accounted for here. Given these example parameters, annualized validator-specific interest rates can be determined based on the global fraction of tokens bonded as stake, as well as their uptime/activity in the previous epoch. For the purpose of this example, we assume 100% uptime for all validators and a split in interest-based rewards between validators and archiver nodes of 80%/20%. Additionally, the fraction of staked circulating supply is assumed to be constant. Based on these assumptions, an annualized validation-client interest rate schedule as a function of % circulating token supply that is staked is shown in **Figure 3**. | **Figure 4:** Shown here are example validator interest rates over time, neglecting transaction fees, segmented by fraction of total circulating supply bonded as stake. | ||||||
| 
 |  | ||||||
|  |  | ||||||
| 
 |  | ||||||
| **Figure 3:** Shown here are example validator interest rates over time, neglecting transaction fees, segmented by fraction of total circulating supply bonded as stake. |  | ||||||
| 
 | 
 | ||||||
| This epoch-specific protocol-defined interest rate sets an upper limit of _protocol-generated_ annual interest rate \(not absolute total interest rate\) possible to be delivered to any validator-client per epoch. The distributed interest rate per epoch is then discounted from this value based on the participation of the validator-client during the previous epoch. | This epoch-specific protocol-defined interest rate sets an upper limit of _protocol-generated_ annual interest rate \(not absolute total interest rate\) possible to be delivered to any validator-client per epoch. The distributed interest rate per epoch is then discounted from this value based on the participation of the validator-client during the previous epoch. | ||||||
| @@ -11,8 +11,8 @@ Each transaction sent through the network, to be processed by the current leader | |||||||
| 
 | 
 | ||||||
| Many current blockchain economies \(e.g. Bitcoin, Ethereum\), rely on protocol-based rewards to support the economy in the short term, with the assumption that the revenue generated through transaction fees will support the economy in the long term, when the protocol derived rewards expire. In an attempt to create a sustainable economy through protocol-based rewards and transaction fees, a fixed portion of each transaction fee is destroyed, with the remaining fee going to the current leader processing the transaction. A scheduled global inflation rate provides a source for rewards distributed to validation-clients, through the process described above, and replication-clients, as discussed below. | Many current blockchain economies \(e.g. Bitcoin, Ethereum\), rely on protocol-based rewards to support the economy in the short term, with the assumption that the revenue generated through transaction fees will support the economy in the long term, when the protocol derived rewards expire. In an attempt to create a sustainable economy through protocol-based rewards and transaction fees, a fixed portion of each transaction fee is destroyed, with the remaining fee going to the current leader processing the transaction. A scheduled global inflation rate provides a source for rewards distributed to validation-clients, through the process described above, and replication-clients, as discussed below. | ||||||
| 
 | 
 | ||||||
| Transaction fees are set by the network cluster based on recent historical throughput, see [Congestion Driven Fees](../../transaction-fees.md#congestion-driven-fees). This minimum portion of each transaction fee can be dynamically adjusted depending on historical gas usage. In this way, the protocol can use the minimum fee to target a desired hardware utilization. By monitoring a protocol specified gas usage with respect to a desired, target usage amount, the minimum fee can be raised/lowered which should, in turn, lower/raise the actual gas usage per block until it reaches the target amount. This adjustment process can be thought of as similar to the difficulty adjustment algorithm in the Bitcoin protocol, however in this case it is adjusting the minimum transaction fee to guide the transaction processing hardware usage to a desired level. | Transaction fees are set by the network cluster based on recent historical throughput, see [Congestion Driven Fees](../../transaction-fees.md#congestion-driven-fees). This minimum portion of each transaction fee can be dynamically adjusted depending on historical gas usage. In this way, the protocol can use the minimum fee to target a desired hardware utilisation. By monitoring a protocol specified gas usage with respect to a desired, target usage amount, the minimum fee can be raised/lowered which should, in turn, lower/raise the actual gas usage per block until it reaches the target amount. This adjustment process can be thought of as similar to the difficulty adjustment algorithm in the Bitcoin protocol, however in this case it is adjusting the minimum transaction fee to guide the transaction processing hardware usage to a desired level. | ||||||
| 
 | 
 | ||||||
| As mentioned, a fixed-proportion of each transaction fee is to be destroyed. The intent of this design is to retain leader incentive to include as many transactions as possible within the leader-slot time, while providing an inflation limiting mechanism that protects against "tax evasion" attacks \(i.e. side-channel fee payments\)[1](../ed_references.md). | As mentioned, a fixed-proportion of each transaction fee is to be destroyed. The intent of this design is to retain leader incentive to include as many transactions as possible within the leader-slot time, while providing an inflation limiting mechansim that protects against "tax evasion" attacks \(i.e. side-channel fee payments\)[1](../ed_references.md). | ||||||
| 
 | 
 | ||||||
| Additionally, the burnt fees can be a consideration in fork selection. In the case of a PoH fork with a malicious, censoring leader, we would expect the total fees destroyed to be less than a comparable honest fork, due to the fees lost from censoring. If the censoring leader is to compensate for these lost protocol fees, they would have to replace the burnt fees on their fork themselves, thus potentially reducing the incentive to censor in the first place. | Additionally, the burnt fees can be a consideration in fork selection. In the case of a PoH fork with a malicious, censoring leader, we would expect the total fees destroyed to be less than a comparable honest fork, due to the fees lost from censoring. If the censoring leader is to compensate for these lost protocol fees, they would have to replace the burnt fees on their fork themselves, thus potentially reducing the incentive to censor in the first place. | ||||||
| @@ -24,7 +24,7 @@ Running a Solana validation-client required relatively modest upfront hardware c | |||||||
| Despite the low-barrier to entry as a validation-client, from a capital investment perspective, as in any developing economy, there will be much opportunity and need for trusted validation services as evidenced by node reliability, UX/UI, APIs and other software accessibility tools. Additionally, although Solana’s validator node startup costs are nominal when compared to similar networks, they may still be somewhat restrictive for some potential participants. In the spirit of developing a true decentralized, permissionless network, these interested parties still have two options to become involved in the Solana network/economy: | Despite the low-barrier to entry as a validation-client, from a capital investment perspective, as in any developing economy, there will be much opportunity and need for trusted validation services as evidenced by node reliability, UX/UI, APIs and other software accessibility tools. Additionally, although Solana’s validator node startup costs are nominal when compared to similar networks, they may still be somewhat restrictive for some potential participants. In the spirit of developing a true decentralized, permissionless network, these interested parties still have two options to become involved in the Solana network/economy: | ||||||
| 
 | 
 | ||||||
| 1. Delegation of previously acquired tokens with a reliable validation node to earn a portion of interest generated | 1. Delegation of previously acquired tokens with a reliable validation node to earn a portion of interest generated | ||||||
| 2. Provide local storage space as a replication-client and receive rewards by submitting Proof-of-Replication \(see [Replication-client Economics](../ed_replication_client_economics/README.md)\). | 2. Provide local storage space as a replication-client and receive rewards by submitting Proof-of-Replication \(see [Replication-client Economics](../ed_replication_client_economics/)\). | ||||||
| 
 | 
 | ||||||
|    a. This participant has the additional option to directly delegate their earned storage rewards \([Replication-client Reward Auto-delegation](../ed_replication_client_economics/ed_rce_replication_client_reward_auto_delegation.md)\) |    a. This participant has the additional option to directly delegate their earned storage rewards \([Replication-client Reward Auto-delegation](../ed_replication_client_economics/ed_rce_replication_client_reward_auto_delegation.md)\) | ||||||
| 
 | 
 | ||||||
| @@ -11,7 +11,7 @@ This document proposes an easy to use software install and updater that can be u | |||||||
| The easiest install method for supported platforms: | The easiest install method for supported platforms: | ||||||
| 
 | 
 | ||||||
| ```bash | ```bash | ||||||
| $ curl -sSf https://raw.githubusercontent.com/solana-labs/solana/v1.0.0/install/solana-install-init.sh | sh | $ curl -sSf https://raw.githubusercontent.com/solana-labs/solana/v0.18.0/install/solana-install-init.sh | sh | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| This script will check github for the latest tagged release and download and run the `solana-install-init` binary from there. | This script will check github for the latest tagged release and download and run the `solana-install-init` binary from there. | ||||||
| @@ -20,7 +20,7 @@ If additional arguments need to be specified during the installation, the follow | |||||||
| 
 | 
 | ||||||
| ```bash | ```bash | ||||||
| $ init_args=.... # arguments for `solana-install-init ...` | $ init_args=.... # arguments for `solana-install-init ...` | ||||||
| $ curl -sSf https://raw.githubusercontent.com/solana-labs/solana/v1.0.0/install/solana-install-init.sh | sh -s - ${init_args} | $ curl -sSf https://raw.githubusercontent.com/solana-labs/solana/v0.18.0/install/solana-install-init.sh | sh -s - ${init_args} | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ### Fetch and run a pre-built installer from a Github release | ### Fetch and run a pre-built installer from a Github release | ||||||
| @@ -28,7 +28,7 @@ $ curl -sSf https://raw.githubusercontent.com/solana-labs/solana/v1.0.0/install/ | |||||||
| With a well-known release URL, a pre-built binary can be obtained for supported platforms: | With a well-known release URL, a pre-built binary can be obtained for supported platforms: | ||||||
| 
 | 
 | ||||||
| ```bash | ```bash | ||||||
| $ curl -o solana-install-init https://github.com/solana-labs/solana/releases/download/v1.0.0/solana-install-init-x86_64-apple-darwin | $ curl -o solana-install-init https://github.com/solana-labs/solana/releases/download/v0.18.0/solana-install-init-x86_64-apple-darwin | ||||||
| $ chmod +x ./solana-install-init | $ chmod +x ./solana-install-init | ||||||
| $ ./solana-install-init --help | $ ./solana-install-init --help | ||||||
| ``` | ``` | ||||||
| @@ -77,7 +77,7 @@ pub struct UpdateManifest { | |||||||
|     pub download_sha256: String, // SHA256 digest of the release tar.bz2 file |     pub download_sha256: String, // SHA256 digest of the release tar.bz2 file | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Data of an Update Manifest program Account. | /// Userdata of an Update Manifest program Account. | ||||||
| #[derive(Serialize, Deserialize, Default, Debug, PartialEq)] | #[derive(Serialize, Deserialize, Default, Debug, PartialEq)] | ||||||
| pub struct SignedUpdateManifest { | pub struct SignedUpdateManifest { | ||||||
|     pub manifest: UpdateManifest, |     pub manifest: UpdateManifest, | ||||||
| @@ -154,7 +154,7 @@ FLAGS: | |||||||
| 
 | 
 | ||||||
| OPTIONS: | OPTIONS: | ||||||
|     -d, --data_dir <PATH>    Directory to store install data [default: .../Library/Application Support/solana] |     -d, --data_dir <PATH>    Directory to store install data [default: .../Library/Application Support/solana] | ||||||
|     -u, --url <URL>          JSON RPC URL for the solana cluster [default: http://devnet.solana.com] |     -u, --url <URL>          JSON RPC URL for the solana cluster [default: http://testnet.solana.com:8899] | ||||||
|     -p, --pubkey <PUBKEY>    Public key of the update manifest [default: 9XX329sPuskWhH4DQh6k16c87dHKhXLBZTL3Gxmve8Gp] |     -p, --pubkey <PUBKEY>    Public key of the update manifest [default: 9XX329sPuskWhH4DQh6k16c87dHKhXLBZTL3Gxmve8Gp] | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user