Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/mcf-rocks/solana-upload

Upload a file to the Solana blockchain
https://github.com/mcf-rocks/solana-upload

Last synced: 2 months ago
JSON representation

Upload a file to the Solana blockchain

Awesome Lists containing this project

README

        

# Solana Upload File

Upload a file to the Solana blockchain, download a file from the Solana blockchain.

WARNING: already pointing at mainnet. Can change with "npm run cluster_local" / "npm run cluster_mainnet" -- see notes at end.

There is a bunch of stuff you need to setup to make this work. This is not a tutorial, but there is a [Solana dapp tutorial here](https://medium.com/@smith_10562/a-simple-solana-dapp-tutorial-6dedbdf65444)

## To Deploy the Program

NB: You don't need to deploy the program, as I already have... address is 5jSTGSbX8NsvZv2ZrqtTqgEBdQLRP1KCLai8j9fshzH5

If you just want to upload a file, go to the next section.

Deploy contract program:

```
npm install
npm run keypair
npm run balance
npm run build_upload
npm run deploy_upload
```

## To Upload a File

Place the file in the top level of the project, for example UK.png, then:

```
npm run balance
npm run upload_file UK.png
```

UK.png is a nice small file you can test on, before you upload that video that should never have been made.

Remember to record the address (public key) of the upload account.

## To Download a File

```
npm run download_file


```

For example this will get you the UK.png file:

```
npm run download_file Ej7YU6LD8D2jvP2dNFHXiHq6x7CoS7r94RaZjCGKTYEG
```

The file will be in the top level of the project.

- Also check out ya boi: tzz2AdP4kkdyebywvJfe6LCqSPwwYKBMYww3znSN8Bg
- And ya boi's sunflower: BHhRmkamZH2jNRPNWoAWxXkzajuExUJPTtEwdBfa1K4c

Or directly:

```
solana -um account tzz2AdP4kkdyebywvJfe6LCqSPwwYKBMYww3znSN8Bg --output json | jq -r .account.data[0] | base64 -d > mcf.jpg && xdg-open mcf.jpg
```

## How it Works

1. Client code (JS) reads file into byte buffer.
2. Creates account of same size on blockchain.
3. Accounts are zero initialised and can only be written via on-chain program
3. Solana has a 1232 transaction limit, therefore:
a. Client calculates maximum data that can be crammed into one transaction,
b. Splits data into X chunks,
c. Calls on-chain program (transaction), which writes that chunk to the account at the right place.

NB: program ensures account is signer to prevent hijack, only client that created account can write.

WARNING: The client does each transaction sequentially (no real reason, except atm mainnet has rate limiting on the RPC node), consequently it will take a long time to upload a large file, if you run out of funds, get an internet outage, accidentally ctrl-c, etc... it's dead, you need to start again.

How long each transaction waits, is in upload_file.js, search for "impatient"

Download just reads the data from the chain and writes it to a file.

## Troubleshooting and Warnings

To upload you must have sufficient funds on the keypair.json in the top level of the project. The upload_file script will estimate the total cost, it asks you to press a key before it starts.

There is a limit of how much data can be stored in an account, I think it's 10MB, but not really sure.

Technically if you have the secret key, you can overwrite, edit scripts as you wish.

Nothing that happens because of this software is my responsibility.

** The following component combination works for a local docker node:

1. package.json
- "testnetDefaultChannel": "v1.3.9",
- "@solana/web3.js": "^0.73.0",
2. Using the BPF_LOADER_DEPRECATED_PROGRAM_ID in deploy.js
3. Cargo.toml
- solana-sdk = { version = "=1.3.9", default-features = false }
- Using the entrypoint_deprecated in lib.rs

** The following component combination works for mainnet:

1. package.json
- "testnetDefaultChannel": "v1.3.4",
- "@solana/web3.js": "^0.71.9",
2. Using the BPF_LOADER_DEPRECATED_PROGRAM_ID in deploy.js
3. Cargo.toml
- solana-sdk = { version = "=1.3.4", default-features = false }
- Using the entrypoint in lib.rs (!)