Compare commits
49 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 082a8f52ef | |||
| bcd3d50b42 | |||
| f695ac3a1c | |||
| a6125b3d0b | |||
| e01d9fe3d7 | |||
| 6294bb0a7b | |||
| fbb2ed8a57 | |||
| aef6c07371 | |||
| ed75198528 | |||
| d0c94b7316 | |||
| 63f338075b | |||
| 4cb0bcd3b9 | |||
| 01b1b39c42 | |||
| 8558860f0a | |||
| b4ebfc7c3f | |||
| a47de8fcb5 | |||
| e9ebe33d51 | |||
| 4c06ff5d8e | |||
| 999399fb08 | |||
| a00ca77b3e | |||
| cae90c1a82 | |||
| 7f169bbabd | |||
| a5d4ecf045 | |||
| 1e67de0242 | |||
| 8cbd812a2c | |||
| b3f521ca20 | |||
| 79bb315401 | |||
| 5871223203 | |||
| cc91f1d64c | |||
| e287845f7c | |||
| 16ffffb0c4 | |||
| 080d9f2c2a | |||
| 4f9abc614e | |||
| 20a051b658 | |||
| 0c2ac0c454 | |||
| 8802d20555 | |||
| 7fa1cb0ccf | |||
| bab08e1df2 | |||
| d91c334cf8 | |||
| bce93ce3cd | |||
| 8367f2b76a | |||
| 055a3002b3 | |||
| c9c4e7d7d1 | |||
| d97bc27c14 | |||
| e215c61ea1 | |||
| 8298d0bc66 | |||
| fac72b1299 | |||
| e780b971d9 | |||
| 90f9f91ddb |
@@ -9,7 +9,6 @@
|
|||||||
"file-loader",
|
"file-loader",
|
||||||
"ts-node",
|
"ts-node",
|
||||||
"webpack-cli",
|
"webpack-cli",
|
||||||
"assert",
|
|
||||||
"buffer",
|
"buffer",
|
||||||
"crypto*",
|
"crypto*",
|
||||||
"stream*",
|
"stream*",
|
||||||
|
|||||||
@@ -18,7 +18,6 @@ jobs:
|
|||||||
|
|
||||||
env:
|
env:
|
||||||
REACT_APP_BEE_HOST: https://api.test-node.staging.ethswarm.org/
|
REACT_APP_BEE_HOST: https://api.test-node.staging.ethswarm.org/
|
||||||
REACT_APP_BEE_DEBUG_HOST: https://debug.test-node.staging.ethswarm.org/
|
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
@@ -57,13 +56,6 @@ jobs:
|
|||||||
- name: Types check
|
- name: Types check
|
||||||
run: npm run check:types
|
run: npm run check:types
|
||||||
|
|
||||||
- name: Update supported Bee action
|
|
||||||
uses: ethersphere/update-supported-bee-action@v1
|
|
||||||
if: github.ref == 'refs/heads/master'
|
|
||||||
with:
|
|
||||||
updateEngine: true
|
|
||||||
token: ${{ secrets.GHA_PAT_BASIC }}
|
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
run: npm run build
|
run: npm run build
|
||||||
|
|
||||||
@@ -71,16 +63,16 @@ jobs:
|
|||||||
run: npm run build:component
|
run: npm run build:component
|
||||||
|
|
||||||
- name: Create preview
|
- name: Create preview
|
||||||
uses: ethersphere/swarm-actions/pr-preview@v0
|
uses: ethersphere/swarm-actions/pr-preview@v1
|
||||||
continue-on-error: true
|
continue-on-error: true
|
||||||
with:
|
with:
|
||||||
bee-url: https://unlimited.gateway.ethswarm.org
|
bee-url: https://unlimited.gateway.ethswarm.org
|
||||||
token: ${{ secrets.GHA_PAT_BASIC }}
|
token: ${{ secrets.GHA_PAT_BASIC }}
|
||||||
error-document: index.html
|
error-document: index.html
|
||||||
headers: "${{ secrets.GATEWAY_AUTHORIZATION_HEADER }}"
|
headers: '${{ secrets.GATEWAY_AUTHORIZATION_HEADER }}'
|
||||||
|
|
||||||
- name: Upload to testnet
|
- name: Upload to testnet
|
||||||
uses: ethersphere/swarm-actions/upload-dir@v0
|
uses: ethersphere/swarm-actions/upload-dir@v1
|
||||||
continue-on-error: true
|
continue-on-error: true
|
||||||
with:
|
with:
|
||||||
index-document: index.html
|
index-document: index.html
|
||||||
|
|||||||
+127
@@ -1,5 +1,132 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
|
## [0.32.0](https://github.com/ethersphere/bee-dashboard/compare/v0.31.0...v0.32.0) (2025-02-12)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* vod display ([#686](https://github.com/ethersphere/bee-dashboard/issues/686)) ([bcd3d50](https://github.com/ethersphere/bee-dashboard/commit/bcd3d50b4209a4f66a259b8a3f6ea5ffd908471f))
|
||||||
|
|
||||||
|
## [0.31.0](https://github.com/ethersphere/bee-dashboard/compare/v0.30.0...v0.31.0) (2025-01-13)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* remove experimental FDP menu item ([#687](https://github.com/ethersphere/bee-dashboard/issues/687)) ([a6125b3](https://github.com/ethersphere/bee-dashboard/commit/a6125b3d0b0b680a9fa61a8edcd75b2ae6c153e0))
|
||||||
|
|
||||||
|
## [0.30.0](https://github.com/ethersphere/bee-dashboard/compare/v0.29.0...v0.30.0) (2024-11-25)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add experimental fdp ([#681](https://github.com/ethersphere/bee-dashboard/issues/681)) ([d0c94b7](https://github.com/ethersphere/bee-dashboard/commit/d0c94b7316ea2b139bddc5481132ea7de7cb840d))
|
||||||
|
* update map data ([#684](https://github.com/ethersphere/bee-dashboard/issues/684)) ([fbb2ed8](https://github.com/ethersphere/bee-dashboard/commit/fbb2ed8a576f3519883e71382b7f4e8505fbe139))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* allow changing api url ([#676](https://github.com/ethersphere/bee-dashboard/issues/676)) ([6294bb0](https://github.com/ethersphere/bee-dashboard/commit/6294bb0a7be6b9b82354c42da8c84e767fad899e))
|
||||||
|
* explicitly define type 0 transaction ([#674](https://github.com/ethersphere/bee-dashboard/issues/674)) ([63f3380](https://github.com/ethersphere/bee-dashboard/commit/63f338075b919cb70d79665c3d86537f2ac1d2e9))
|
||||||
|
|
||||||
|
## [0.29.0](https://github.com/ethersphere/bee-dashboard/compare/v0.28.0...v0.29.0) (2024-07-17)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* clarify labels and syncing ([#670](https://github.com/ethersphere/bee-dashboard/issues/670)) ([01b1b39](https://github.com/ethersphere/bee-dashboard/commit/01b1b39c42cc5b68a0132c3696c3c42a27ea2ee4))
|
||||||
|
* polish app ([#669](https://github.com/ethersphere/bee-dashboard/issues/669)) ([8558860](https://github.com/ethersphere/bee-dashboard/commit/8558860f0a3baa82c31c091a44c78bb8e97de70d))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* clarify withdraw and deposit message ([#654](https://github.com/ethersphere/bee-dashboard/issues/654)) ([b4ebfc7](https://github.com/ethersphere/bee-dashboard/commit/b4ebfc7c3fd449807db47fa25763df464cc45618))
|
||||||
|
|
||||||
|
## [0.28.0](https://github.com/ethersphere/bee-dashboard/compare/v0.27.0...v0.28.0) (2024-06-18)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* upgrade bee-js to 7.0.3 ([#666](https://github.com/ethersphere/bee-dashboard/issues/666)) ([e9ebe33](https://github.com/ethersphere/bee-dashboard/commit/e9ebe33d51aa525921eacfad683577605e591531))
|
||||||
|
|
||||||
|
## [0.27.0](https://github.com/ethersphere/bee-dashboard/compare/v0.26.2...v0.27.0) (2024-06-11)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add redeem shortcut to sidebar ([999399f](https://github.com/ethersphere/bee-dashboard/commit/999399fb08c1a47a671ba0ad50409624654a1082))
|
||||||
|
|
||||||
|
## [0.26.2](https://github.com/ethersphere/bee-dashboard/compare/v0.26.1...v0.26.2) (2024-06-05)
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* merge version and health check ([#662](https://github.com/ethersphere/bee-dashboard/issues/662)) ([cae90c1](https://github.com/ethersphere/bee-dashboard/commit/cae90c1a82e16ee8c7908c43e2fd17f7130eb89d))
|
||||||
|
|
||||||
|
## [0.26.1](https://github.com/ethersphere/bee-dashboard/compare/v0.26.0...v0.26.1) (2024-06-03)
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* add bee version ([#659](https://github.com/ethersphere/bee-dashboard/issues/659)) ([a5d4ecf](https://github.com/ethersphere/bee-dashboard/commit/a5d4ecf045f691b9059fcca925d0f30675d12db0))
|
||||||
|
|
||||||
|
## [0.26.0](https://github.com/ethersphere/bee-dashboard/compare/v0.25.0...v0.26.0) (2024-06-03)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* display effective capacity ([#643](https://github.com/ethersphere/bee-dashboard/issues/643)) ([5871223](https://github.com/ethersphere/bee-dashboard/commit/58712232031e084195adf92c40cd41a98eaf16cf))
|
||||||
|
* merge api ([#658](https://github.com/ethersphere/bee-dashboard/issues/658)) ([8cbd812](https://github.com/ethersphere/bee-dashboard/commit/8cbd812a2c04706f8f46de5355209b96783723b9))
|
||||||
|
* show syncing info ([#647](https://github.com/ethersphere/bee-dashboard/issues/647)) ([cc91f1d](https://github.com/ethersphere/bee-dashboard/commit/cc91f1d64cd48a845fa9fa45ec4b58335eab3893))
|
||||||
|
* wait for upload sync ([#649](https://github.com/ethersphere/bee-dashboard/issues/649)) ([79bb315](https://github.com/ethersphere/bee-dashboard/commit/79bb31540196b74f3bc0220b8c844fbd5aaaf488))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* correct the bee version detection ([#645](https://github.com/ethersphere/bee-dashboard/issues/645)) ([b3f521c](https://github.com/ethersphere/bee-dashboard/commit/b3f521ca2055b91d7adddf96563cca6bf92e3d59))
|
||||||
|
|
||||||
|
## [0.25.0](https://github.com/ethersphere/bee-dashboard/compare/v0.24.1...v0.25.0) (2023-12-04)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* improve topup and dilute ux ([0c2ac0c](https://github.com/ethersphere/bee-dashboard/commit/0c2ac0c454ad02200a2762958c5bc5abbdfe8005))
|
||||||
|
* update postage stamp creation screen ([#641](https://github.com/ethersphere/bee-dashboard/issues/641)) ([4f9abc6](https://github.com/ethersphere/bee-dashboard/commit/4f9abc614eedd5ce3a279a4686cc832c4d1e62c7))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* add missing stamp labels and fix inputs ([#634](https://github.com/ethersphere/bee-dashboard/issues/634)) ([7fa1cb0](https://github.com/ethersphere/bee-dashboard/commit/7fa1cb0ccf9f2a32263e84aa76732ebd2fc7fb22))
|
||||||
|
* put stamp input error handling in state ([#640](https://github.com/ethersphere/bee-dashboard/issues/640)) ([20a051b](https://github.com/ethersphere/bee-dashboard/commit/20a051b6589c22397a7305d722a56df0604ff7a4))
|
||||||
|
|
||||||
|
## [0.24.1](https://github.com/ethersphere/bee-dashboard/compare/v0.24.0...v0.24.1) (2023-10-18)
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* update `swap-endpoint` to `blockchain-rpc-endpoint` ([#628](https://github.com/ethersphere/bee-dashboard/issues/628)) ([bce93ce](https://github.com/ethersphere/bee-dashboard/commit/bce93ce3cdc1ef4b1f50fcf274591ba00726be16))
|
||||||
|
|
||||||
|
## [0.24.0](https://github.com/ethersphere/bee-dashboard/compare/v0.23.0...v0.24.0) (2023-08-11)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add stamp dilute and topup ([#619](https://github.com/ethersphere/bee-dashboard/issues/619)) ([055a300](https://github.com/ethersphere/bee-dashboard/commit/055a3002b303df45c7010ef4d365e14b979e9084))
|
||||||
|
|
||||||
|
## [0.23.0](https://github.com/ethersphere/bee-dashboard/compare/v0.22.0...v0.23.0) (2023-02-21)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add staking for full nodes ([#590](https://github.com/ethersphere/bee-dashboard/issues/590)) ([fac72b1](https://github.com/ethersphere/bee-dashboard/commit/fac72b1299353c104231aa038c1bab9df78c1355))
|
||||||
|
* upgrade bee-js to 5.2.0 ([#611](https://github.com/ethersphere/bee-dashboard/issues/611)) ([e215c61](https://github.com/ethersphere/bee-dashboard/commit/e215c61ea1619fc388fe8b1904d160b04a1a5c0d))
|
||||||
|
|
||||||
|
## [0.22.0](https://github.com/ethersphere/bee-dashboard/compare/v0.21.1...v0.22.0) (2023-01-19)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add node connecting status ([#603](https://github.com/ethersphere/bee-dashboard/issues/603)) ([90f9f91](https://github.com/ethersphere/bee-dashboard/commit/90f9f91ddbefb47b40c7e567125972b800d81972))
|
||||||
|
|
||||||
## [0.21.1](https://github.com/ethersphere/bee-dashboard/compare/v0.21.0...v0.21.1) (2022-12-21)
|
## [0.21.1](https://github.com/ethersphere/bee-dashboard/compare/v0.21.0...v0.21.1) (2022-12-21)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
+1
-1
@@ -1 +1 @@
|
|||||||
* @Cafe137 @vojtechsimetka
|
* @Cafe137
|
||||||
|
|||||||
@@ -13,15 +13,13 @@
|
|||||||
**Warning: This project is in alpha state. There might (and most probably will) be changes in the future to its API and
|
**Warning: This project is in alpha state. There might (and most probably will) be changes in the future to its API and
|
||||||
working. Also, no guarantees can be made about its stability, efficiency, and security at this stage.**
|
working. Also, no guarantees can be made about its stability, efficiency, and security at this stage.**
|
||||||
|
|
||||||
This project is intended to be used with **Bee version <!-- SUPPORTED_BEE_START -->1.9.0-13a47043<!-- SUPPORTED_BEE_END -->**.
|
Stay up to date by joining the [official Discord](https://discord.gg/GU22h2utj6) and by keeping an eye on the
|
||||||
Using it with older or newer Bee versions is not recommended and may not work. Stay up to date by joining the
|
|
||||||
[official Discord](https://discord.gg/GU22h2utj6) and by keeping an eye on the
|
|
||||||
[releases tab](https://github.com/ethersphere/bee-dashboard/releases).
|
[releases tab](https://github.com/ethersphere/bee-dashboard/releases).
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
| Node Setup | Upload Files | Download Content | Accounting | Settings |
|
| Node Setup | Upload Files | Download Content | Accounting | Settings |
|
||||||
| ------------------------------------ | -------------------------------------- | ------------------------------------------ | ----------------------------------------- | ---------------------------------------- |
|
| ------------------------------------ | -------------------------------------- | ------------------------------------------ | ----------------------------------------- | ------------------------------------- |
|
||||||
|  |  |  |  |  |
|
|  |  |  |  |  |
|
||||||
|
|
||||||
## Table of Contents
|
## Table of Contents
|
||||||
@@ -45,9 +43,9 @@ npm install -g @ethersphere/bee-dashboard
|
|||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
:warning: To successfully connect to the Bee node, you will need to enable the Debug API and CORS. You can do so by
|
:warning: To successfully connect to the Bee node, you will need to enable CORS. You can do so by setting
|
||||||
setting `cors-allowed-origins: ['*']` and `debug-api-enable: true` in the Bee config file and then restart the Bee node.
|
`cors-allowed-origins: ['*']` in the Bee config file and then restart the Bee node. To see where the config file is,
|
||||||
To see where the config file is, consult the
|
consult the
|
||||||
[official Bee documentation](https://docs.ethswarm.org/docs/working-with-bee/configuration#configuring-bee-installed-using-a-package-manager)
|
[official Bee documentation](https://docs.ethswarm.org/docs/working-with-bee/configuration#configuring-bee-installed-using-a-package-manager)
|
||||||
|
|
||||||
### Terminal
|
### Terminal
|
||||||
@@ -96,18 +94,22 @@ The Bee Dashboard runs in development mode on [http://localhost:3031/](http://lo
|
|||||||
|
|
||||||
#### Environmental variables
|
#### Environmental variables
|
||||||
|
|
||||||
The CRA supports to specify "environmental variables" during build time which are then hardcoded into the served static files.
|
The CRA supports to specify "environmental variables" during build time which are then hardcoded into the served static
|
||||||
We support following variables:
|
files. We support following variables:
|
||||||
|
|
||||||
- `REACT_APP_BEE_DESKTOP_ENABLED` (`boolean`) that toggles if the Dashboard is in Desktop mode or not.
|
- `REACT_APP_BEE_DESKTOP_ENABLED` (`boolean`) that toggles if the Dashboard is in Desktop mode or not.
|
||||||
- `REACT_APP_BEE_DESKTOP_URL` (`string`) defines custom URL where the Desktop API is expected. By default, it is same origin under which the Dashboard is served.
|
- `REACT_APP_BEE_DESKTOP_URL` (`string`) defines custom URL where the Desktop API is expected. By default, it is same
|
||||||
- `REACT_APP_BEE_HOST` (`string`) defines custom Bee API URL to be used as default one. By default, the `http://localhost:1633` is used.
|
origin under which the Dashboard is served.
|
||||||
- `REACT_APP_BEE_DEBUG_HOST` (`string`) defines custom Bee Debug API URL to be used as default one. By default, the `http://localhost:1635` is used.
|
- `REACT_APP_BEE_HOST` (`string`) defines custom Bee API URL to be used as default one. By default, the
|
||||||
- `REACT_APP_DEFAULT_RPC_URL` (`string`) defines the default RPC provider URL. Be aware, that his only configures the default value. The user can override this in Settings, which is then persisted in local store and has priority over the value set in this env. variable. By default `https://xdai.fairdatasociety.org` is used.
|
`http://localhost:1633` is used.
|
||||||
|
- `REACT_APP_DEFAULT_RPC_URL` (`string`) defines the default RPC provider URL. Be aware, that his only configures the
|
||||||
|
default value. The user can override this in Settings, which is then persisted in local store and has priority over
|
||||||
|
the value set in this env. variable. By default `https://xdai.fairdatasociety.org` is used.
|
||||||
|
|
||||||
#### Swarm Desktop development
|
#### Swarm Desktop development
|
||||||
|
|
||||||
If you want to develop Bee Dashboard in the Swarm Desktop mode, then spin up `swarm-desktop` to the point where Desktop is initialized (eq. the splash screen disappear) and:
|
If you want to develop Bee Dashboard in the Swarm Desktop mode, then spin up `swarm-desktop` to the point where Desktop
|
||||||
|
is initialized (eq. the splash screen disappear) and:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
echo "REACT_APP_BEE_DESKTOP_URL=http://localhost:3054
|
echo "REACT_APP_BEE_DESKTOP_URL=http://localhost:3054
|
||||||
@@ -128,7 +130,6 @@ There are some ways you can make this module better:
|
|||||||
|
|
||||||
## Maintainers
|
## Maintainers
|
||||||
|
|
||||||
- [vojtechsimetka](https://github.com/vojtechsimetka)
|
|
||||||
- [Cafe137](https://github.com/Cafe137)
|
- [Cafe137](https://github.com/Cafe137)
|
||||||
|
|
||||||
See what "Maintainer" means [here](https://github.com/ethersphere/repo-maintainer).
|
See what "Maintainer" means [here](https://github.com/ethersphere/repo-maintainer).
|
||||||
@@ -137,5 +138,4 @@ See what "Maintainer" means [here](https://github.com/ethersphere/repo-maintaine
|
|||||||
|
|
||||||
[BSD-3-Clause](./LICENSE)
|
[BSD-3-Clause](./LICENSE)
|
||||||
|
|
||||||
|
|
||||||
[](https://app.fossa.com/projects/git%2Bgithub.com%2Fethersphere%2Fbee-dashboard?ref=badge_large)
|
[](https://app.fossa.com/projects/git%2Bgithub.com%2Fethersphere%2Fbee-dashboard?ref=badge_large)
|
||||||
|
|||||||
Generated
+1564
-2220
File diff suppressed because it is too large
Load Diff
+18
-20
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@ethersphere/bee-dashboard",
|
"name": "@ethersphere/bee-dashboard",
|
||||||
"version": "0.21.1",
|
"version": "0.32.0",
|
||||||
"description": "An app which helps users to setup their Bee node and do actions like cash out cheques",
|
"description": "An app which helps users to setup their Bee node and do actions like cash out cheques",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
"bee",
|
"bee",
|
||||||
@@ -26,37 +26,36 @@
|
|||||||
"url": "https://github.com/ethersphere/bee-dashboard.git"
|
"url": "https://github.com/ethersphere/bee-dashboard.git"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@ethersphere/bee-js": "^5.1.0",
|
"@ethersphere/bee-js": "^8.3.1",
|
||||||
"@ethersphere/manifest-js": "1.2.1",
|
|
||||||
"@ethersphere/swarm-cid": "^0.1.0",
|
"@ethersphere/swarm-cid": "^0.1.0",
|
||||||
|
"@fairdatasociety/fdp-storage": "^0.19.0",
|
||||||
"@material-ui/core": "4.12.3",
|
"@material-ui/core": "4.12.3",
|
||||||
"@material-ui/icons": "4.11.2",
|
"@material-ui/icons": "4.11.2",
|
||||||
"@material-ui/lab": "4.0.0-alpha.57",
|
"@material-ui/lab": "4.0.0-alpha.57",
|
||||||
"assert": "^2.0.0",
|
"axios": "^0.28.1",
|
||||||
"axios": "0.24.0",
|
"bignumber.js": "^9.1.2",
|
||||||
"bignumber.js": "9.0.1",
|
|
||||||
"buffer": "^6.0.3",
|
"buffer": "^6.0.3",
|
||||||
"crypto": "npm:crypto-browserify",
|
"crypto": "npm:crypto-browserify",
|
||||||
"crypto-browserify": "^3.12.0",
|
"crypto-browserify": "^3.12.0",
|
||||||
"dotted-map": "^2.2.3",
|
"dotted-map": "^2.2.3",
|
||||||
"ethers": "^5.6.4",
|
"ethers": "^5.7.2",
|
||||||
"file-saver": "^2.0.5",
|
"file-saver": "^2.0.5",
|
||||||
"formik": "2.2.9",
|
"formik": "2.2.9",
|
||||||
"formik-material-ui": "3.0.1",
|
"formik-material-ui": "3.0.1",
|
||||||
"jszip": "^3.7.1",
|
"jszip": "^3.10.1",
|
||||||
|
"mantaray-js": "^1.0.3",
|
||||||
"material-ui-dropzone": "3.5.0",
|
"material-ui-dropzone": "3.5.0",
|
||||||
"notistack": "1.0.10",
|
"notistack": "^3.0.1",
|
||||||
"opener": "1.5.2",
|
"opener": "1.5.2",
|
||||||
"qrcode.react": "1.0.1",
|
"qrcode.react": "1.0.1",
|
||||||
"react": ">=17.0.0 || >=18.0.0",
|
"react": ">= 17.0.2",
|
||||||
"react-copy-to-clipboard": "5.0.4",
|
"react-copy-to-clipboard": "^5.1.0",
|
||||||
"react-dom": ">=17.0.0 || >=18.0.0",
|
"react-dom": ">= 17.0.2",
|
||||||
"react-identicons": "1.2.5",
|
"react-identicons": "1.2.5",
|
||||||
"react-router": "6.2.1",
|
"react-router": "6.2.1",
|
||||||
"react-router-dom": "6.2.1",
|
"react-router-dom": "6.2.1",
|
||||||
"react-syntax-highlighter": "15.4.4",
|
"react-syntax-highlighter": "15.4.4",
|
||||||
"remixicon-react": "^1.0.0",
|
"remixicon-react": "^1.0.0",
|
||||||
"semver": "7.3.5",
|
|
||||||
"serve-handler": "6.1.3",
|
"serve-handler": "6.1.3",
|
||||||
"stream": "npm:stream-browserify",
|
"stream": "npm:stream-browserify",
|
||||||
"stream-browserify": "^3.0.0"
|
"stream-browserify": "^3.0.0"
|
||||||
@@ -78,16 +77,15 @@
|
|||||||
"@types/jest": "27.0.2",
|
"@types/jest": "27.0.2",
|
||||||
"@types/qrcode.react": "1.0.2",
|
"@types/qrcode.react": "1.0.2",
|
||||||
"@types/react": "17.0.34",
|
"@types/react": "17.0.34",
|
||||||
"@types/react-copy-to-clipboard": "5.0.2",
|
"@types/react-copy-to-clipboard": "^5.0.2",
|
||||||
"@types/react-dom": "17.0.11",
|
"@types/react-dom": "17.0.11",
|
||||||
"@types/react-router": "5.1.18",
|
"@types/react-router": "5.1.18",
|
||||||
"@types/react-router-dom": "5.3.2",
|
"@types/react-router-dom": "5.3.2",
|
||||||
"@types/react-syntax-highlighter": "13.5.2",
|
"@types/react-syntax-highlighter": "13.5.2",
|
||||||
"@types/semver": "7.3.9",
|
|
||||||
"@typescript-eslint/eslint-plugin": "5.28.0",
|
"@typescript-eslint/eslint-plugin": "5.28.0",
|
||||||
"@typescript-eslint/parser": "5.28.0",
|
"@typescript-eslint/parser": "5.28.0",
|
||||||
"babel-eslint": "10.1.0",
|
"babel-eslint": "^10.1.0",
|
||||||
"babel-loader": "8.1.0",
|
"babel-loader": "^8.1.0",
|
||||||
"babel-plugin-syntax-dynamic-import": "6.18.0",
|
"babel-plugin-syntax-dynamic-import": "6.18.0",
|
||||||
"babel-plugin-tsconfig-paths": "1.0.2",
|
"babel-plugin-tsconfig-paths": "1.0.2",
|
||||||
"base64-inline-loader": "^2.0.1",
|
"base64-inline-loader": "^2.0.1",
|
||||||
@@ -115,7 +113,7 @@
|
|||||||
"ts-node": "^10.8.1",
|
"ts-node": "^10.8.1",
|
||||||
"typescript": "4.8.3",
|
"typescript": "4.8.3",
|
||||||
"web-vitals": "2.1.2",
|
"web-vitals": "2.1.2",
|
||||||
"webpack": "^5.73.0",
|
"webpack": "^5.93.0",
|
||||||
"webpack-cli": "^4.10.0"
|
"webpack-cli": "^4.10.0"
|
||||||
},
|
},
|
||||||
"peerDependencies": {
|
"peerDependencies": {
|
||||||
@@ -137,7 +135,7 @@
|
|||||||
"lint:check": "eslint \"src/**/*.ts\" \"src/**/*.tsx\" && prettier --check \"src/**/*.ts\" \"src/**/*.tsx\"",
|
"lint:check": "eslint \"src/**/*.ts\" \"src/**/*.tsx\" && prettier --check \"src/**/*.ts\" \"src/**/*.tsx\"",
|
||||||
"check:types": "tsc --project tsconfig.lib.json",
|
"check:types": "tsc --project tsconfig.lib.json",
|
||||||
"update-map-data": "node ./utils/update-map-data.js",
|
"update-map-data": "node ./utils/update-map-data.js",
|
||||||
"bee": "bee-factory start"
|
"bee": "npx bee-factory start"
|
||||||
},
|
},
|
||||||
"files": [
|
"files": [
|
||||||
"lib",
|
"lib",
|
||||||
@@ -159,6 +157,6 @@
|
|||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=14.0.0",
|
"node": ">=14.0.0",
|
||||||
"npm": ">=6.9.0",
|
"npm": ">=6.9.0",
|
||||||
"bee": "1.9.0-13a47043"
|
"bee": "1.16.1-8e269c8"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+1
-4
@@ -18,7 +18,6 @@ import { theme } from './theme'
|
|||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
beeApiUrl?: string
|
beeApiUrl?: string
|
||||||
beeDebugApiUrl?: string
|
|
||||||
defaultRpcUrl?: string
|
defaultRpcUrl?: string
|
||||||
lockedApiSettings?: boolean
|
lockedApiSettings?: boolean
|
||||||
isDesktop?: boolean
|
isDesktop?: boolean
|
||||||
@@ -28,7 +27,6 @@ interface Props {
|
|||||||
|
|
||||||
const App = ({
|
const App = ({
|
||||||
beeApiUrl,
|
beeApiUrl,
|
||||||
beeDebugApiUrl,
|
|
||||||
defaultRpcUrl,
|
defaultRpcUrl,
|
||||||
lockedApiSettings,
|
lockedApiSettings,
|
||||||
isDesktop,
|
isDesktop,
|
||||||
@@ -40,14 +38,13 @@ const App = ({
|
|||||||
<ThemeProvider theme={theme}>
|
<ThemeProvider theme={theme}>
|
||||||
<SettingsProvider
|
<SettingsProvider
|
||||||
beeApiUrl={beeApiUrl}
|
beeApiUrl={beeApiUrl}
|
||||||
beeDebugApiUrl={beeDebugApiUrl}
|
|
||||||
defaultRpcUrl={defaultRpcUrl}
|
defaultRpcUrl={defaultRpcUrl}
|
||||||
lockedApiSettings={lockedApiSettings}
|
lockedApiSettings={lockedApiSettings}
|
||||||
isDesktop={isDesktop}
|
isDesktop={isDesktop}
|
||||||
desktopUrl={desktopUrl}
|
desktopUrl={desktopUrl}
|
||||||
>
|
>
|
||||||
<TopUpProvider>
|
<TopUpProvider>
|
||||||
<BeeProvider isDesktop={isDesktop}>
|
<BeeProvider>
|
||||||
<BalanceProvider>
|
<BalanceProvider>
|
||||||
<StampsProvider>
|
<StampsProvider>
|
||||||
<FileProvider>
|
<FileProvider>
|
||||||
|
|||||||
+51739
-7115
File diff suppressed because it is too large
Load Diff
@@ -2,6 +2,7 @@ import { createStyles, makeStyles, Theme, Typography } from '@material-ui/core'
|
|||||||
import { ReactElement } from 'react'
|
import { ReactElement } from 'react'
|
||||||
import Check from 'remixicon-react/CheckLineIcon'
|
import Check from 'remixicon-react/CheckLineIcon'
|
||||||
import AlertCircle from 'remixicon-react/ErrorWarningFillIcon'
|
import AlertCircle from 'remixicon-react/ErrorWarningFillIcon'
|
||||||
|
import Connecting from 'remixicon-react/LinksLineIcon'
|
||||||
import RefreshLine from 'remixicon-react/RefreshLineIcon'
|
import RefreshLine from 'remixicon-react/RefreshLineIcon'
|
||||||
import { SwarmButton, SwarmButtonProps } from './SwarmButton'
|
import { SwarmButton, SwarmButtonProps } from './SwarmButton'
|
||||||
|
|
||||||
@@ -10,7 +11,7 @@ interface Props {
|
|||||||
title: string
|
title: string
|
||||||
subtitle: string
|
subtitle: string
|
||||||
buttonProps: SwarmButtonProps
|
buttonProps: SwarmButtonProps
|
||||||
status: 'ok' | 'error' | 'loading'
|
status: 'ok' | 'error' | 'loading' | 'connecting'
|
||||||
}
|
}
|
||||||
|
|
||||||
const useStyles = (backgroundColor: string) =>
|
const useStyles = (backgroundColor: string) =>
|
||||||
@@ -65,6 +66,8 @@ export default function Card({ buttonProps, icon, title, subtitle, status }: Pro
|
|||||||
statusIcon = <AlertCircle size="13" color="#f44336" />
|
statusIcon = <AlertCircle size="13" color="#f44336" />
|
||||||
} else if (status === 'loading') {
|
} else if (status === 'loading') {
|
||||||
statusIcon = <RefreshLine size="13" color="orange" />
|
statusIcon = <RefreshLine size="13" color="orange" />
|
||||||
|
} else if (status === 'connecting') {
|
||||||
|
statusIcon = <Connecting size="13" color="#0074D9" />
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -19,8 +19,8 @@ interface Props {
|
|||||||
export default function CheckoutModal({ peerId, uncashedAmount }: Props): ReactElement {
|
export default function CheckoutModal({ peerId, uncashedAmount }: Props): ReactElement {
|
||||||
const [open, setOpen] = useState<boolean>(false)
|
const [open, setOpen] = useState<boolean>(false)
|
||||||
const [loadingCashout, setLoadingCashout] = useState<boolean>(false)
|
const [loadingCashout, setLoadingCashout] = useState<boolean>(false)
|
||||||
|
const { beeApi } = useContext(SettingsContext)
|
||||||
const { enqueueSnackbar } = useSnackbar()
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
const { beeDebugApi } = useContext(SettingsContext)
|
|
||||||
|
|
||||||
const handleClickOpen = () => {
|
const handleClickOpen = () => {
|
||||||
setOpen(true)
|
setOpen(true)
|
||||||
@@ -31,11 +31,9 @@ export default function CheckoutModal({ peerId, uncashedAmount }: Props): ReactE
|
|||||||
}
|
}
|
||||||
|
|
||||||
const handleCashout = () => {
|
const handleCashout = () => {
|
||||||
if (!beeDebugApi) return
|
if (peerId && beeApi) {
|
||||||
|
|
||||||
if (peerId) {
|
|
||||||
setLoadingCashout(true)
|
setLoadingCashout(true)
|
||||||
beeDebugApi
|
beeApi
|
||||||
.cashoutLastCheque(peerId)
|
.cashoutLastCheque(peerId)
|
||||||
.then(res => {
|
.then(res => {
|
||||||
setOpen(false)
|
setOpen(false)
|
||||||
|
|||||||
@@ -0,0 +1,25 @@
|
|||||||
|
import { ChainState } from '@ethersphere/bee-js'
|
||||||
|
import { useContext, useEffect, useState } from 'react'
|
||||||
|
import { Context } from '../providers/Settings'
|
||||||
|
import ExpandableListItem from './ExpandableListItem'
|
||||||
|
|
||||||
|
export function ChainSync() {
|
||||||
|
const { beeApi } = useContext(Context)
|
||||||
|
const [chainState, setChainState] = useState<ChainState | null>(null)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const interval = setInterval(() => {
|
||||||
|
if (!beeApi) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
beeApi.getChainState().then(setChainState).catch(console.error) // eslint-disable-line
|
||||||
|
}, 3_000)
|
||||||
|
|
||||||
|
return () => clearInterval(interval)
|
||||||
|
})
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ExpandableListItem label="Chain state" value={chainState ? `${chainState.block} / ${chainState.chainTip}` : '-'} />
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -1,9 +1,11 @@
|
|||||||
|
import { Utils } from '@ethersphere/bee-js'
|
||||||
import { Typography } from '@material-ui/core/'
|
import { Typography } from '@material-ui/core/'
|
||||||
import { ReactElement } from 'react'
|
import { ReactElement } from 'react'
|
||||||
import Identicon from 'react-identicons'
|
import Identicon from 'react-identicons'
|
||||||
import ClipboardCopy from './ClipboardCopy'
|
|
||||||
import QRCodeModal from './QRCodeModal'
|
|
||||||
import { BLOCKCHAIN_EXPLORER_URL } from '../constants'
|
import { BLOCKCHAIN_EXPLORER_URL } from '../constants'
|
||||||
|
import ClipboardCopy from './ClipboardCopy'
|
||||||
|
import { Flex } from './Flex'
|
||||||
|
import QRCodeModal from './QRCodeModal'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
address: string | undefined
|
address: string | undefined
|
||||||
@@ -16,10 +18,10 @@ export default function EthereumAddress(props: Props): ReactElement {
|
|||||||
return (
|
return (
|
||||||
<Typography component="div" variant="subtitle1">
|
<Typography component="div" variant="subtitle1">
|
||||||
{props.address ? (
|
{props.address ? (
|
||||||
<div style={{ display: 'flex' }}>
|
<Flex>
|
||||||
{props.hideBlockie ? null : (
|
{props.hideBlockie ? null : (
|
||||||
<div style={{ paddingTop: '5px', marginRight: '10px' }}>
|
<div style={{ paddingTop: '5px', marginRight: '10px' }}>
|
||||||
<Identicon size={20} string={props.address} />
|
<Identicon size={20} string={Utils.capitalizeAddressERC55(props.address)} />
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
<div>
|
<div>
|
||||||
@@ -43,9 +45,9 @@ export default function EthereumAddress(props: Props): ReactElement {
|
|||||||
{props.address}
|
{props.address}
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
<QRCodeModal value={props.address} label={'Ethereum Address'} />
|
<QRCodeModal value={Utils.capitalizeAddressERC55(props.address)} label={'Ethereum Address'} />
|
||||||
<ClipboardCopy value={props.address} />
|
<ClipboardCopy value={Utils.capitalizeAddressERC55(props.address)} />
|
||||||
</div>
|
</Flex>
|
||||||
) : (
|
) : (
|
||||||
'-'
|
'-'
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -24,6 +24,9 @@ const useStyles = makeStyles((theme: Theme) =>
|
|||||||
},
|
},
|
||||||
contentLevel12: {
|
contentLevel12: {
|
||||||
marginTop: theme.spacing(0.25),
|
marginTop: theme.spacing(0.25),
|
||||||
|
'& > li:last-of-type': {
|
||||||
|
marginBottom: theme.spacing(2),
|
||||||
|
},
|
||||||
},
|
},
|
||||||
infoText: {
|
infoText: {
|
||||||
color: '#c9c9c9',
|
color: '#c9c9c9',
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
import { ReactElement, ReactNode, useState } from 'react'
|
|
||||||
import { makeStyles, Theme, createStyles } from '@material-ui/core/styles'
|
|
||||||
import { Collapse, ListItem, ListItemText, Typography } from '@material-ui/core'
|
import { Collapse, ListItem, ListItemText, Typography } from '@material-ui/core'
|
||||||
|
import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
||||||
import { ExpandLess, ExpandMore } from '@material-ui/icons'
|
import { ExpandLess, ExpandMore } from '@material-ui/icons'
|
||||||
|
import { ReactElement, ReactNode, useState } from 'react'
|
||||||
|
import { Flex } from './Flex'
|
||||||
|
|
||||||
const useStyles = makeStyles((theme: Theme) =>
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
createStyles({
|
createStyles({
|
||||||
@@ -65,14 +66,14 @@ export default function ExpandableList({ children, label, level, defaultOpen, in
|
|||||||
<div className={`${classes.root} ${rootLevelClass}`}>
|
<div className={`${classes.root} ${rootLevelClass}`}>
|
||||||
<ListItem button onClick={handleClick} className={classes.header}>
|
<ListItem button onClick={handleClick} className={classes.header}>
|
||||||
<ListItemText primary={<Typography variant={typographyVariant}>{label}</Typography>} />
|
<ListItemText primary={<Typography variant={typographyVariant}>{label}</Typography>} />
|
||||||
<div style={{ display: 'flex' }}>
|
<Flex>
|
||||||
{!open && (
|
{!open && (
|
||||||
<Typography variant="body2" className={classes.infoText}>
|
<Typography variant="body2" className={classes.infoText}>
|
||||||
{info}
|
{info}
|
||||||
</Typography>
|
</Typography>
|
||||||
)}
|
)}
|
||||||
{open ? <ExpandLess /> : <ExpandMore />}
|
{open ? <ExpandLess /> : <ExpandMore />}
|
||||||
</div>
|
</Flex>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
<Collapse in={open} timeout="auto" unmountOnExit>
|
<Collapse in={open} timeout="auto" unmountOnExit>
|
||||||
<div className={contentLevelClass}>{children}</div>
|
<div className={contentLevelClass}>{children}</div>
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
import { ReactElement, ReactNode } from 'react'
|
import { Grid, IconButton, Tooltip, Typography } from '@material-ui/core'
|
||||||
import { makeStyles, Theme, createStyles } from '@material-ui/core/styles'
|
|
||||||
import { Typography, Grid, IconButton, Tooltip } from '@material-ui/core'
|
|
||||||
import Info from 'remixicon-react/InformationLineIcon'
|
|
||||||
import ListItem from '@material-ui/core/ListItem'
|
import ListItem from '@material-ui/core/ListItem'
|
||||||
|
import { Theme, createStyles, makeStyles } from '@material-ui/core/styles'
|
||||||
|
import { ReactElement, ReactNode } from 'react'
|
||||||
|
import Info from 'remixicon-react/InformationLineIcon'
|
||||||
|
|
||||||
const useStyles = makeStyles((theme: Theme) =>
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
createStyles({
|
createStyles({
|
||||||
|
|||||||
@@ -1,15 +1,15 @@
|
|||||||
import { Grid, IconButton, InputBase, ListItem, Typography } from '@material-ui/core'
|
import { Box, Grid, IconButton, InputBase, ListItem, Typography } from '@material-ui/core'
|
||||||
import Collapse from '@material-ui/core/Collapse'
|
import Collapse from '@material-ui/core/Collapse'
|
||||||
import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
||||||
import { ChangeEvent, ReactElement, useState } from 'react'
|
import { ChangeEvent, ReactElement, useState } from 'react'
|
||||||
|
import type { RemixiconReactIconProps } from 'remixicon-react'
|
||||||
import Check from 'remixicon-react/CheckLineIcon'
|
import Check from 'remixicon-react/CheckLineIcon'
|
||||||
|
import X from 'remixicon-react/CloseLineIcon'
|
||||||
import Edit from 'remixicon-react/PencilLineIcon'
|
import Edit from 'remixicon-react/PencilLineIcon'
|
||||||
import Minus from 'remixicon-react/SubtractLineIcon'
|
import Minus from 'remixicon-react/SubtractLineIcon'
|
||||||
import X from 'remixicon-react/CloseLineIcon'
|
|
||||||
import ExpandableListItemActions from './ExpandableListItemActions'
|
import ExpandableListItemActions from './ExpandableListItemActions'
|
||||||
import ExpandableListItemNote from './ExpandableListItemNote'
|
import ExpandableListItemNote from './ExpandableListItemNote'
|
||||||
import { SwarmButton } from './SwarmButton'
|
import { SwarmButton } from './SwarmButton'
|
||||||
import type { RemixiconReactIconProps } from 'remixicon-react'
|
|
||||||
|
|
||||||
const useStyles = makeStyles((theme: Theme) =>
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
createStyles({
|
createStyles({
|
||||||
@@ -108,12 +108,8 @@ export default function ExpandableListItemKey({
|
|||||||
<div>
|
<div>
|
||||||
{!open && value}
|
{!open && value}
|
||||||
{!expandedOnly && !locked && (
|
{!expandedOnly && !locked && (
|
||||||
<IconButton size="small" className={classes.copyValue}>
|
<IconButton size="small" className={classes.copyValue} onClick={toggleOpen}>
|
||||||
{open ? (
|
{open ? <Minus strokeWidth={1} /> : <Edit strokeWidth={1} />}
|
||||||
<Minus onClick={toggleOpen} strokeWidth={1} />
|
|
||||||
) : (
|
|
||||||
<Edit onClick={toggleOpen} strokeWidth={1} />
|
|
||||||
)}
|
|
||||||
</IconButton>
|
</IconButton>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
@@ -134,31 +130,33 @@ export default function ExpandableListItemKey({
|
|||||||
</ListItem>
|
</ListItem>
|
||||||
<Collapse in={open} timeout="auto" unmountOnExit>
|
<Collapse in={open} timeout="auto" unmountOnExit>
|
||||||
{helperText && <ExpandableListItemNote>{helperText}</ExpandableListItemNote>}
|
{helperText && <ExpandableListItemNote>{helperText}</ExpandableListItemNote>}
|
||||||
<ExpandableListItemActions>
|
<Box mt={2}>
|
||||||
<SwarmButton
|
<ExpandableListItemActions>
|
||||||
disabled={
|
<SwarmButton
|
||||||
loading ||
|
disabled={
|
||||||
inputValue === value ||
|
loading ||
|
||||||
Boolean(confirmLabelDisabled) || // Disable if external validation is provided
|
inputValue === value ||
|
||||||
(inputValue === '' && value === undefined) // Disable if no initial value was not provided and the field is empty. The undefined check is improtant so that it is possible to submit with empty input in other cases
|
Boolean(confirmLabelDisabled) || // Disable if external validation is provided
|
||||||
}
|
(inputValue === '' && value === undefined) // Disable if no initial value was not provided and the field is empty. The undefined check is improtant so that it is possible to submit with empty input in other cases
|
||||||
loading={loading}
|
}
|
||||||
iconType={confirmIcon ?? Check}
|
loading={loading}
|
||||||
onClick={() => {
|
iconType={confirmIcon ?? Check}
|
||||||
if (onConfirm) onConfirm(inputValue)
|
onClick={() => {
|
||||||
}}
|
if (onConfirm) onConfirm(inputValue)
|
||||||
>
|
}}
|
||||||
{confirmLabel || 'Save'}
|
>
|
||||||
</SwarmButton>
|
{confirmLabel || 'Save'}
|
||||||
<SwarmButton
|
</SwarmButton>
|
||||||
disabled={loading || inputValue === value || inputValue === ''}
|
<SwarmButton
|
||||||
iconType={X}
|
disabled={loading || inputValue === value || inputValue === ''}
|
||||||
onClick={() => setInputValue(value || '')}
|
iconType={X}
|
||||||
cancel
|
onClick={() => setInputValue(value || '')}
|
||||||
>
|
cancel
|
||||||
Cancel
|
>
|
||||||
</SwarmButton>
|
Cancel
|
||||||
</ExpandableListItemActions>
|
</SwarmButton>
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
</Box>
|
||||||
</Collapse>
|
</Collapse>
|
||||||
</>
|
</>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -77,20 +77,18 @@ export default function ExpandableListItemKey({ label, value, expanded }: Props)
|
|||||||
<Grid container direction="row" justifyContent="space-between" alignItems="center">
|
<Grid container direction="row" justifyContent="space-between" alignItems="center">
|
||||||
{label && <Typography variant="body1">{label}</Typography>}
|
{label && <Typography variant="body1">{label}</Typography>}
|
||||||
<Typography variant="body2">
|
<Typography variant="body2">
|
||||||
<div>
|
{!open && (
|
||||||
{!open && (
|
<span className={classes.copyValue}>
|
||||||
<span className={classes.copyValue}>
|
<Tooltip title={copied ? 'Copied' : 'Copy'} placement="top" arrow onClose={tooltipCloseHandler}>
|
||||||
<Tooltip title={copied ? 'Copied' : 'Copy'} placement="top" arrow onClose={tooltipCloseHandler}>
|
<CopyToClipboard text={value}>
|
||||||
<CopyToClipboard text={value}>
|
<span onClick={tooltipClickHandler}>{value ? spanText : ''}</span>
|
||||||
<span onClick={tooltipClickHandler}>{value ? spanText : ''}</span>
|
</CopyToClipboard>
|
||||||
</CopyToClipboard>
|
</Tooltip>
|
||||||
</Tooltip>
|
</span>
|
||||||
</span>
|
)}
|
||||||
)}
|
<IconButton size="small" className={classes.copyValue} onClick={toggleOpen}>
|
||||||
<IconButton size="small" className={classes.copyValue}>
|
{open ? <Minus strokeWidth={1} /> : <Eye strokeWidth={1} />}
|
||||||
{open ? <Minus onClick={toggleOpen} strokeWidth={1} /> : <Eye onClick={toggleOpen} strokeWidth={1} />}
|
</IconButton>
|
||||||
</IconButton>
|
|
||||||
</div>
|
|
||||||
</Typography>
|
</Typography>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Collapse in={open} timeout="auto" unmountOnExit>
|
<Collapse in={open} timeout="auto" unmountOnExit>
|
||||||
|
|||||||
@@ -82,22 +82,20 @@ export default function ExpandableListItemLink({
|
|||||||
<Grid container direction="row" justifyContent="space-between" alignItems="center">
|
<Grid container direction="row" justifyContent="space-between" alignItems="center">
|
||||||
{label && <Typography variant="body1">{label}</Typography>}
|
{label && <Typography variant="body1">{label}</Typography>}
|
||||||
<Typography variant="body2">
|
<Typography variant="body2">
|
||||||
<div>
|
{allowClipboard && (
|
||||||
{allowClipboard && (
|
<span className={classes.copyValue}>
|
||||||
<span className={classes.copyValue}>
|
<Tooltip title={copied ? 'Copied' : 'Copy'} placement="top" arrow onClose={tooltipCloseHandler}>
|
||||||
<Tooltip title={copied ? 'Copied' : 'Copy'} placement="top" arrow onClose={tooltipCloseHandler}>
|
<CopyToClipboard text={value}>
|
||||||
<CopyToClipboard text={value}>
|
<span onClick={tooltipClickHandler}>{displayValue}</span>
|
||||||
<span onClick={tooltipClickHandler}>{displayValue}</span>
|
</CopyToClipboard>
|
||||||
</CopyToClipboard>
|
</Tooltip>
|
||||||
</Tooltip>
|
</span>
|
||||||
</span>
|
)}
|
||||||
)}
|
{!allowClipboard && <span onClick={onNavigation}>{displayValue}</span>}
|
||||||
{!allowClipboard && <span onClick={onNavigation}>{displayValue}</span>}
|
<IconButton size="small" className={classes.openLinkIcon} onClick={onNavigation}>
|
||||||
<IconButton size="small" className={classes.openLinkIcon}>
|
{navigationType === 'NEW_WINDOW' && <OpenInNewSharp strokeWidth={1} />}
|
||||||
{navigationType === 'NEW_WINDOW' && <OpenInNewSharp onClick={onNavigation} strokeWidth={1} />}
|
{navigationType === 'HISTORY_PUSH' && <ArrowForward strokeWidth={1} />}
|
||||||
{navigationType === 'HISTORY_PUSH' && <ArrowForward onClick={onNavigation} strokeWidth={1} />}
|
</IconButton>
|
||||||
</IconButton>
|
|
||||||
</div>
|
|
||||||
</Typography>
|
</Typography>
|
||||||
</Grid>
|
</Grid>
|
||||||
</Grid>
|
</Grid>
|
||||||
|
|||||||
@@ -0,0 +1,29 @@
|
|||||||
|
import { createStyles, makeStyles } from '@material-ui/core'
|
||||||
|
import { ReactElement } from 'react'
|
||||||
|
|
||||||
|
const useStyles = makeStyles(() =>
|
||||||
|
createStyles({
|
||||||
|
video: {
|
||||||
|
width: '100%',
|
||||||
|
height: '100%',
|
||||||
|
objectFit: 'cover',
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
interface VideoProps {
|
||||||
|
src: string | undefined
|
||||||
|
maxHeight?: string
|
||||||
|
maxWidth?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function FitVideo(props: VideoProps): ReactElement {
|
||||||
|
const classes = useStyles()
|
||||||
|
|
||||||
|
const inlineStyles: Record<string, string> = {}
|
||||||
|
|
||||||
|
props.maxHeight && (inlineStyles.maxHeight = props.maxHeight)
|
||||||
|
props.maxWidth && (inlineStyles.maxWidth = props.maxWidth)
|
||||||
|
|
||||||
|
return <video className={classes.video} src={props.src} style={inlineStyles} controls />
|
||||||
|
}
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
import { ReactNode } from 'react'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
children: ReactNode
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Flex({ children }: Props) {
|
||||||
|
return <div style={{ display: 'flex' }}>{children}</div>
|
||||||
|
}
|
||||||
@@ -1,9 +1,9 @@
|
|||||||
import { ReactElement, CSSProperties, useContext, useState, useEffect } from 'react'
|
|
||||||
import type { Peer } from '@ethersphere/bee-js'
|
import type { Peer } from '@ethersphere/bee-js'
|
||||||
import DottedMap, { DottedMapWithoutCountriesLib } from 'dotted-map/without-countries'
|
import DottedMap, { DottedMapWithoutCountriesLib } from 'dotted-map/without-countries'
|
||||||
|
import { CSSProperties, ReactElement, useContext, useEffect, useState } from 'react'
|
||||||
|
import mapData from '../assets/data/map-data.json'
|
||||||
import nodesDb from '../assets/data/nodes-db.json'
|
import nodesDb from '../assets/data/nodes-db.json'
|
||||||
import { Context } from '../providers/Bee'
|
import { Context } from '../providers/Bee'
|
||||||
import mapData from '../assets/data/map-data.json'
|
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
style?: CSSProperties
|
style?: CSSProperties
|
||||||
|
|||||||
@@ -0,0 +1,22 @@
|
|||||||
|
import React, { ReactElement } from 'react'
|
||||||
|
import LinearProgress, { LinearProgressProps } from '@material-ui/core/LinearProgress'
|
||||||
|
import Typography from '@material-ui/core/Typography'
|
||||||
|
import Box from '@material-ui/core/Box'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
linearProgressProps?: LinearProgressProps
|
||||||
|
value: number
|
||||||
|
}
|
||||||
|
|
||||||
|
export function LinearProgressWithLabel(props: Props): ReactElement {
|
||||||
|
return (
|
||||||
|
<Box display="flex" alignItems="center">
|
||||||
|
<Box width="100%" mr={1}>
|
||||||
|
<LinearProgress variant="determinate" {...props} />
|
||||||
|
</Box>
|
||||||
|
<Box minWidth={35}>
|
||||||
|
<Typography variant="body2" color="textSecondary">{`${Math.round(props.value)}%`}</Typography>
|
||||||
|
</Box>
|
||||||
|
</Box>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -1,22 +1,23 @@
|
|||||||
import { Divider, Drawer, Grid, Link as MUILink, List } from '@material-ui/core'
|
import { BeeModes } from '@ethersphere/bee-js'
|
||||||
import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
import { Box, Divider, Drawer, Grid, List, Link as MUILink, Typography } from '@material-ui/core'
|
||||||
|
import { Theme, createStyles, makeStyles } from '@material-ui/core/styles'
|
||||||
import { ReactElement, useContext } from 'react'
|
import { ReactElement, useContext } from 'react'
|
||||||
import { Link } from 'react-router-dom'
|
import { Link } from 'react-router-dom'
|
||||||
import FilesIcon from 'remixicon-react/ArrowUpDownLineIcon'
|
import FilesIcon from 'remixicon-react/ArrowUpDownLineIcon'
|
||||||
import DocsIcon from 'remixicon-react/BookOpenLineIcon'
|
import DocsIcon from 'remixicon-react/BookOpenLineIcon'
|
||||||
import ExternalLinkIcon from 'remixicon-react/ExternalLinkLineIcon'
|
import ExternalLinkIcon from 'remixicon-react/ExternalLinkLineIcon'
|
||||||
|
import GithubIcon from 'remixicon-react/GithubFillIcon'
|
||||||
import HomeIcon from 'remixicon-react/Home3LineIcon'
|
import HomeIcon from 'remixicon-react/Home3LineIcon'
|
||||||
import SettingsIcon from 'remixicon-react/Settings2LineIcon'
|
import SettingsIcon from 'remixicon-react/Settings2LineIcon'
|
||||||
import AccountIcon from 'remixicon-react/Wallet3LineIcon'
|
import AccountIcon from 'remixicon-react/Wallet3LineIcon'
|
||||||
import { Context as BeeContext } from '../providers/Bee'
|
|
||||||
import { Context as SettingsContext } from '../providers/Settings'
|
|
||||||
import DashboardLogo from '../assets/dashboard-logo.svg'
|
import DashboardLogo from '../assets/dashboard-logo.svg'
|
||||||
import DesktopLogo from '../assets/desktop-logo.svg'
|
import DesktopLogo from '../assets/desktop-logo.svg'
|
||||||
|
import { BEE_DOCS_HOST, GITHUB_BEE_DASHBOARD_URL } from '../constants'
|
||||||
|
import { Context as BeeContext } from '../providers/Bee'
|
||||||
|
import { Context as SettingsContext } from '../providers/Settings'
|
||||||
import { ROUTES } from '../routes'
|
import { ROUTES } from '../routes'
|
||||||
import SideBarItem from './SideBarItem'
|
import SideBarItem from './SideBarItem'
|
||||||
import SideBarStatus from './SideBarStatus'
|
import SideBarStatus from './SideBarStatus'
|
||||||
import { BeeModes } from '@ethersphere/bee-js'
|
|
||||||
import { BEE_DOCS_HOST } from '../constants'
|
|
||||||
|
|
||||||
const drawerWidth = 300
|
const drawerWidth = 300
|
||||||
|
|
||||||
@@ -126,6 +127,22 @@ export default function SideBar(): ReactElement {
|
|||||||
/>
|
/>
|
||||||
</MUILink>
|
</MUILink>
|
||||||
</List>
|
</List>
|
||||||
|
<Divider className={classes.divider} />
|
||||||
|
<List>
|
||||||
|
<MUILink href={GITHUB_BEE_DASHBOARD_URL} target="_blank" className={classes.link}>
|
||||||
|
<SideBarItem
|
||||||
|
iconStart={<GithubIcon className={classes.icon} />}
|
||||||
|
iconEnd={<ExternalLinkIcon className={classes.icon} color="#595959" />}
|
||||||
|
label={<span>GitHub</span>}
|
||||||
|
/>
|
||||||
|
</MUILink>
|
||||||
|
</List>
|
||||||
|
<Divider className={classes.divider} />
|
||||||
|
<Box mt={4}>
|
||||||
|
<Link to={ROUTES.TOP_UP_GIFT_CODE}>
|
||||||
|
<Typography align="center">Redeem gift code</Typography>
|
||||||
|
</Link>
|
||||||
|
</Box>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<List>
|
<List>
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
import { ReactElement, useContext } from 'react'
|
import { ReactElement, useContext } from 'react'
|
||||||
import { useLocation, matchPath } from 'react-router-dom'
|
import { matchPath, useLocation } from 'react-router-dom'
|
||||||
import ArrowRight from 'remixicon-react/ArrowRightLineIcon'
|
import ArrowRight from 'remixicon-react/ArrowRightLineIcon'
|
||||||
|
|
||||||
import { createStyles, Theme, makeStyles } from '@material-ui/core/styles'
|
import { ListItem, ListItemIcon, ListItemText, Typography } from '@material-ui/core'
|
||||||
import { ListItemText, ListItemIcon, ListItem, Typography } from '@material-ui/core'
|
import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
||||||
import { Context } from '../providers/Bee'
|
import { Context } from '../providers/Bee'
|
||||||
import StatusIcon from './StatusIcon'
|
import StatusIcon from './StatusIcon'
|
||||||
|
|
||||||
@@ -44,6 +44,7 @@ const useStyles = makeStyles((theme: Theme) =>
|
|||||||
},
|
},
|
||||||
smallerText: {
|
smallerText: {
|
||||||
fontSize: '0.9rem',
|
fontSize: '0.9rem',
|
||||||
|
whiteSpace: 'nowrap',
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -0,0 +1,88 @@
|
|||||||
|
import { Bee } from '@ethersphere/bee-js'
|
||||||
|
import { Box } from '@material-ui/core'
|
||||||
|
import Button from '@material-ui/core/Button'
|
||||||
|
import Dialog from '@material-ui/core/Dialog'
|
||||||
|
import DialogActions from '@material-ui/core/DialogActions'
|
||||||
|
import DialogContent from '@material-ui/core/DialogContent'
|
||||||
|
import DialogTitle from '@material-ui/core/DialogTitle'
|
||||||
|
import Input from '@material-ui/core/Input'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
import { ReactElement, ReactNode, useState } from 'react'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
type: 'Topup' | 'Dilute'
|
||||||
|
icon: ReactNode
|
||||||
|
bee: Bee
|
||||||
|
stamp: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function StampExtensionModal({ type, icon, bee, stamp }: Props): ReactElement {
|
||||||
|
const [open, setOpen] = useState(false)
|
||||||
|
const [amount, setAmount] = useState('')
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
const label = `${type} ${stamp.substring(0, 8)}`
|
||||||
|
|
||||||
|
const handleClickOpen = (e: React.MouseEvent<HTMLButtonElement>) => {
|
||||||
|
setOpen(true)
|
||||||
|
e.stopPropagation()
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleClose = () => {
|
||||||
|
setOpen(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleAction = async () => {
|
||||||
|
if (type === 'Topup') {
|
||||||
|
try {
|
||||||
|
await bee.topUpBatch(stamp, amount)
|
||||||
|
enqueueSnackbar(`Successfully topped up stamp, your changes will appear soon`, { variant: 'success' })
|
||||||
|
} catch (error) {
|
||||||
|
enqueueSnackbar(`Failed to topup stamp: ${error || 'Unknown reason'}`, { variant: 'error' })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (type === 'Dilute') {
|
||||||
|
try {
|
||||||
|
await bee.diluteBatch(stamp, parseInt(amount, 10))
|
||||||
|
enqueueSnackbar(`Successfully diluted stamp, your changes will appear soon`, { variant: 'success' })
|
||||||
|
} catch (error) {
|
||||||
|
enqueueSnackbar(`Failed to dilute stamp: ${error || 'Unknown reason'}`, { variant: 'error' })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleChange = (event: React.ChangeEvent<HTMLTextAreaElement | HTMLInputElement>) => {
|
||||||
|
setAmount(event.target.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Box mb={2}>
|
||||||
|
<Button variant="contained" onClick={handleClickOpen} startIcon={icon}>
|
||||||
|
{type}
|
||||||
|
</Button>
|
||||||
|
<Dialog open={open} onClose={handleClose} aria-labelledby="form-dialog-title">
|
||||||
|
<DialogTitle id="form-dialog-title">{label}</DialogTitle>
|
||||||
|
<DialogContent>
|
||||||
|
<Input
|
||||||
|
autoFocus
|
||||||
|
margin="dense"
|
||||||
|
id="name"
|
||||||
|
type="text"
|
||||||
|
placeholder={type === 'Topup' ? 'Amount to add' : 'New depth to dilute'}
|
||||||
|
fullWidth
|
||||||
|
value={amount}
|
||||||
|
onChange={handleChange}
|
||||||
|
/>
|
||||||
|
</DialogContent>
|
||||||
|
<DialogActions>
|
||||||
|
<Button onClick={handleClose} color="primary">
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
<Button disabled={amount === ''} onClick={handleAction} color="primary">
|
||||||
|
{type}
|
||||||
|
</Button>
|
||||||
|
</DialogActions>
|
||||||
|
</Dialog>
|
||||||
|
</Box>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import type { ReactElement } from 'react'
|
|
||||||
import { CircularProgress } from '@material-ui/core'
|
import { CircularProgress } from '@material-ui/core'
|
||||||
|
import type { ReactElement } from 'react'
|
||||||
import { CheckState } from '../providers/Bee'
|
import { CheckState } from '../providers/Bee'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
@@ -28,6 +28,9 @@ export default function StatusIcon({ checkState, size, className, isLoading }: P
|
|||||||
case CheckState.STARTING:
|
case CheckState.STARTING:
|
||||||
backgroundColor = 'orange'
|
backgroundColor = 'orange'
|
||||||
break
|
break
|
||||||
|
case CheckState.CONNECTING:
|
||||||
|
backgroundColor = '#0074D9'
|
||||||
|
break
|
||||||
default:
|
default:
|
||||||
// Default is error
|
// Default is error
|
||||||
backgroundColor = '#ff3a52'
|
backgroundColor = '#ff3a52'
|
||||||
|
|||||||
@@ -30,6 +30,7 @@ interface Props {
|
|||||||
formik?: boolean
|
formik?: boolean
|
||||||
defaultValue?: string
|
defaultValue?: string
|
||||||
placeholder?: string
|
placeholder?: string
|
||||||
|
disabled?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
const useStyles = makeStyles((theme: Theme) =>
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
@@ -60,6 +61,7 @@ export function SwarmSelect({
|
|||||||
onChange,
|
onChange,
|
||||||
label,
|
label,
|
||||||
placeholder,
|
placeholder,
|
||||||
|
disabled = false,
|
||||||
}: Props): ReactElement {
|
}: Props): ReactElement {
|
||||||
const classes = useStyles()
|
const classes = useStyles()
|
||||||
|
|
||||||
@@ -69,6 +71,7 @@ export function SwarmSelect({
|
|||||||
{label && <FormHelperText>{label}</FormHelperText>}
|
{label && <FormHelperText>{label}</FormHelperText>}
|
||||||
<Field
|
<Field
|
||||||
required
|
required
|
||||||
|
disabled={disabled}
|
||||||
component={Select}
|
component={Select}
|
||||||
name={name}
|
name={name}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -94,6 +97,7 @@ export function SwarmSelect({
|
|||||||
{label && <FormHelperText>{label}</FormHelperText>}
|
{label && <FormHelperText>{label}</FormHelperText>}
|
||||||
<MuiSelect
|
<MuiSelect
|
||||||
required
|
required
|
||||||
|
disabled={disabled}
|
||||||
name={name}
|
name={name}
|
||||||
fullWidth
|
fullWidth
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
|
|||||||
+2
-3
@@ -1,14 +1,13 @@
|
|||||||
export const META_FILE_NAME = '.swarmgatewaymeta.json'
|
export const META_FILE_NAME = 'metadata'
|
||||||
export const PREVIEW_FILE_NAME = '.swarmgatewaypreview.jpeg'
|
|
||||||
export const PREVIEW_DIMENSIONS = { maxWidth: 250, maxHeight: 175 }
|
export const PREVIEW_DIMENSIONS = { maxWidth: 250, maxHeight: 175 }
|
||||||
export const BZZ_LINK_DOMAIN = 'bzz.link'
|
export const BZZ_LINK_DOMAIN = 'bzz.link'
|
||||||
export const BLOCKCHAIN_EXPLORER_URL = 'https://blockscout.com/xdai/mainnet'
|
export const BLOCKCHAIN_EXPLORER_URL = 'https://blockscout.com/xdai/mainnet'
|
||||||
export const BEE_DOCS_HOST = 'https://docs.ethswarm.org/docs/'
|
export const BEE_DOCS_HOST = 'https://docs.ethswarm.org/docs/'
|
||||||
export const BEE_DISCORD_HOST = 'https://discord.gg/eKr9XPv7'
|
export const BEE_DISCORD_HOST = 'https://discord.gg/eKr9XPv7'
|
||||||
export const GITHUB_REPO_URL = 'https://api.github.com/repos/ethersphere/bee'
|
export const GITHUB_REPO_URL = 'https://api.github.com/repos/ethersphere/bee'
|
||||||
|
export const GITHUB_BEE_DASHBOARD_URL = 'https://github.com/ethersphere/bee-dashboard.git'
|
||||||
export const BEE_DESKTOP_LATEST_RELEASE_PAGE = 'https://github.com/ethersphere/bee-desktop/releases/latest'
|
export const BEE_DESKTOP_LATEST_RELEASE_PAGE = 'https://github.com/ethersphere/bee-desktop/releases/latest'
|
||||||
export const BEE_DESKTOP_LATEST_RELEASE_PAGE_API =
|
export const BEE_DESKTOP_LATEST_RELEASE_PAGE_API =
|
||||||
'https://api.github.com/repos/ethersphere/bee-desktop/releases/latest'
|
'https://api.github.com/repos/ethersphere/bee-desktop/releases/latest'
|
||||||
export const DEFAULT_BEE_API_HOST = 'http://localhost:1633'
|
export const DEFAULT_BEE_API_HOST = 'http://localhost:1633'
|
||||||
export const DEFAULT_BEE_DEBUG_API_HOST = 'http://localhost:1635'
|
|
||||||
export const DEFAULT_RPC_URL = 'https://xdai.fairdatasociety.org'
|
export const DEFAULT_RPC_URL = 'https://xdai.fairdatasociety.org'
|
||||||
|
|||||||
@@ -6,21 +6,23 @@ import { Context as BeeContext } from '../providers/Bee'
|
|||||||
import { Context as SettingsContext } from '../providers/Settings'
|
import { Context as SettingsContext } from '../providers/Settings'
|
||||||
|
|
||||||
export default function DepositModal(): ReactElement {
|
export default function DepositModal(): ReactElement {
|
||||||
const { beeDebugApi } = useContext(SettingsContext)
|
const { beeApi } = useContext(SettingsContext)
|
||||||
const { refresh } = useContext(BeeContext)
|
const { refresh } = useContext(BeeContext)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<WithdrawDepositModal
|
<WithdrawDepositModal
|
||||||
successMessage="Successful deposit."
|
successMessage="Successful deposit."
|
||||||
errorMessage="Error with depositing"
|
errorMessage="Error with depositing"
|
||||||
dialogMessage="Specify the amount of xBZZ you would like to deposit to your node."
|
dialogMessage="Amount of xBZZ to deposit to the checkbook, from your node."
|
||||||
label="Deposit"
|
label="Deposit"
|
||||||
icon={<Download size="1rem" />}
|
icon={<Download size="1rem" />}
|
||||||
min={new BigNumber(0)}
|
min={new BigNumber(0)}
|
||||||
action={async (amount: bigint) => {
|
action={async (amount: bigint) => {
|
||||||
if (!beeDebugApi) throw new Error('Bee Debug URL is not valid')
|
if (!beeApi) {
|
||||||
|
throw new Error('Bee URL is not valid')
|
||||||
|
}
|
||||||
|
|
||||||
const transactionHash = await beeDebugApi.depositTokens(amount.toString())
|
const transactionHash = await beeApi.depositTokens(amount.toString())
|
||||||
refresh()
|
refresh()
|
||||||
|
|
||||||
return transactionHash
|
return transactionHash
|
||||||
|
|||||||
@@ -0,0 +1,43 @@
|
|||||||
|
import { BigNumber } from 'bignumber.js'
|
||||||
|
import { ReactElement, useContext } from 'react'
|
||||||
|
import Download from 'remixicon-react/DownloadLineIcon'
|
||||||
|
import WithdrawDepositModal from '../components/WithdrawDepositModal'
|
||||||
|
import { Context as BeeContext } from '../providers/Bee'
|
||||||
|
import { Context as SettingsContext } from '../providers/Settings'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
onStarted: () => void
|
||||||
|
onFinished: () => void
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function StakeModal({ onStarted, onFinished }: Props): ReactElement {
|
||||||
|
const { beeApi } = useContext(SettingsContext)
|
||||||
|
const { refresh } = useContext(BeeContext)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<WithdrawDepositModal
|
||||||
|
successMessage="Successfully deposited stake."
|
||||||
|
errorMessage="Error with depositing"
|
||||||
|
dialogMessage="Specify the amount of xBZZ you would like to stake. Your first stake must be at least 10 xBZZ. This will lock your tokens."
|
||||||
|
label="Stake"
|
||||||
|
icon={<Download size="1rem" />}
|
||||||
|
min={new BigNumber(0)}
|
||||||
|
action={async (amount: bigint) => {
|
||||||
|
if (!beeApi) {
|
||||||
|
throw new Error('Bee URL is not valid')
|
||||||
|
}
|
||||||
|
|
||||||
|
onStarted()
|
||||||
|
|
||||||
|
try {
|
||||||
|
await beeApi.depositStake(amount.toString())
|
||||||
|
} finally {
|
||||||
|
refresh()
|
||||||
|
onFinished()
|
||||||
|
}
|
||||||
|
|
||||||
|
return 'unknown'
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -6,21 +6,23 @@ import { Context as BeeContext } from '../providers/Bee'
|
|||||||
import { Context as SettingsContext } from '../providers/Settings'
|
import { Context as SettingsContext } from '../providers/Settings'
|
||||||
|
|
||||||
export default function WithdrawModal(): ReactElement {
|
export default function WithdrawModal(): ReactElement {
|
||||||
const { beeDebugApi } = useContext(SettingsContext)
|
const { beeApi } = useContext(SettingsContext)
|
||||||
const { refresh } = useContext(BeeContext)
|
const { refresh } = useContext(BeeContext)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<WithdrawDepositModal
|
<WithdrawDepositModal
|
||||||
successMessage="Successful withdrawal."
|
successMessage="Successful withdrawal."
|
||||||
errorMessage="Error with withdrawing."
|
errorMessage="Error with withdrawing."
|
||||||
dialogMessage="Specify the amount of xBZZ you would like to withdraw from your node."
|
dialogMessage="Amount of xBZZ to withdraw from the checkbook to your node."
|
||||||
label="Withdraw"
|
label="Withdraw"
|
||||||
icon={<Upload size="1rem" />}
|
icon={<Upload size="1rem" />}
|
||||||
min={new BigNumber(0)}
|
min={new BigNumber(0)}
|
||||||
action={async (amount: bigint) => {
|
action={async (amount: bigint) => {
|
||||||
if (!beeDebugApi) throw new Error('Bee Debug URL is not valid')
|
if (!beeApi) {
|
||||||
|
throw new Error('Bee URL is not valid')
|
||||||
|
}
|
||||||
|
|
||||||
const transactionHash = await beeDebugApi.withdrawTokens(amount.toString())
|
const transactionHash = await beeApi.withdrawTokens(amount.toString())
|
||||||
refresh()
|
refresh()
|
||||||
|
|
||||||
return transactionHash
|
return transactionHash
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
import { LastCashoutActionResponse, BeeDebug } from '@ethersphere/bee-js'
|
import { Bee, LastCashoutActionResponse } from '@ethersphere/bee-js'
|
||||||
import { useEffect, useState } from 'react'
|
import { useEffect, useState } from 'react'
|
||||||
import { Token } from '../models/Token'
|
import { Token } from '../models/Token'
|
||||||
|
import { Balance, Settlement, Settlements } from '../types'
|
||||||
import { makeRetriablePromise, unwrapPromiseSettlements } from '../utils'
|
import { makeRetriablePromise, unwrapPromiseSettlements } from '../utils'
|
||||||
import { Balance, Settlements, Settlement } from '../types'
|
|
||||||
|
|
||||||
interface UseAccountingHook {
|
interface UseAccountingHook {
|
||||||
isLoadingUncashed: boolean
|
isLoadingUncashed: boolean
|
||||||
@@ -79,7 +79,7 @@ function mergeAccounting(
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const useAccounting = (
|
export const useAccounting = (
|
||||||
beeDebugApi: BeeDebug | null,
|
beeApi: Bee | null,
|
||||||
settlements: Settlements | null,
|
settlements: Settlements | null,
|
||||||
balances: Balance[] | null,
|
balances: Balance[] | null,
|
||||||
): UseAccountingHook => {
|
): UseAccountingHook => {
|
||||||
@@ -88,19 +88,19 @@ export const useAccounting = (
|
|||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
// We don't have any settlements loaded yet or we are already loading/have loaded the uncashed amounts
|
// We don't have any settlements loaded yet or we are already loading/have loaded the uncashed amounts
|
||||||
if (isLoadingUncashed || !beeDebugApi || !settlements || uncashedAmounts) return
|
if (isLoadingUncashed || !beeApi || !settlements || uncashedAmounts) return
|
||||||
|
|
||||||
setIsloadingUncashed(true)
|
setIsloadingUncashed(true)
|
||||||
const promises = settlements.settlements
|
const promises = settlements.settlements
|
||||||
.filter(({ received }) => received.toBigNumber.gt('0'))
|
.filter(({ received }) => received.toBigNumber.gt('0'))
|
||||||
.map(({ peer }) => makeRetriablePromise(() => beeDebugApi.getLastCashoutAction(peer)))
|
.map(({ peer }) => makeRetriablePromise(() => beeApi.getLastCashoutAction(peer)))
|
||||||
|
|
||||||
Promise.allSettled(promises).then(settlements => {
|
Promise.allSettled(promises).then(settlements => {
|
||||||
const results = unwrapPromiseSettlements(settlements)
|
const results = unwrapPromiseSettlements(settlements)
|
||||||
setUncashedAmounts(results.fulfilled)
|
setUncashedAmounts(results.fulfilled)
|
||||||
setIsloadingUncashed(false)
|
setIsloadingUncashed(false)
|
||||||
})
|
})
|
||||||
}, [settlements, isLoadingUncashed, uncashedAmounts, beeDebugApi])
|
}, [settlements, isLoadingUncashed, uncashedAmounts, beeApi])
|
||||||
|
|
||||||
const accounting = mergeAccounting(balances, settlements?.settlements, uncashedAmounts)
|
const accounting = mergeAccounting(balances, settlements?.settlements, uncashedAmounts)
|
||||||
|
|
||||||
|
|||||||
+2
-9
@@ -1,24 +1,17 @@
|
|||||||
import React from 'react'
|
import React from 'react'
|
||||||
import ReactDOM from 'react-dom'
|
import ReactDOM from 'react-dom'
|
||||||
import './index.css'
|
|
||||||
import App from './App'
|
import App from './App'
|
||||||
|
import './index.css'
|
||||||
import reportWebVitals from './reportWebVitals'
|
import reportWebVitals from './reportWebVitals'
|
||||||
|
|
||||||
const desktopEnabled = Boolean(process.env.REACT_APP_BEE_DESKTOP_ENABLED)
|
const desktopEnabled = Boolean(process.env.REACT_APP_BEE_DESKTOP_ENABLED)
|
||||||
const desktopUrl = process.env.REACT_APP_BEE_DESKTOP_URL
|
const desktopUrl = process.env.REACT_APP_BEE_DESKTOP_URL
|
||||||
const beeApiUrl = process.env.REACT_APP_BEE_HOST
|
const beeApiUrl = process.env.REACT_APP_BEE_HOST
|
||||||
const beeDebugApiUrl = process.env.REACT_APP_BEE_DEBUG_HOST
|
|
||||||
const defaultRpcUrl = process.env.REACT_APP_DEFAULT_RPC_URL
|
const defaultRpcUrl = process.env.REACT_APP_DEFAULT_RPC_URL
|
||||||
|
|
||||||
ReactDOM.render(
|
ReactDOM.render(
|
||||||
<React.StrictMode>
|
<React.StrictMode>
|
||||||
<App
|
<App isDesktop={desktopEnabled} desktopUrl={desktopUrl} beeApiUrl={beeApiUrl} defaultRpcUrl={defaultRpcUrl} />
|
||||||
isDesktop={desktopEnabled}
|
|
||||||
desktopUrl={desktopUrl}
|
|
||||||
beeApiUrl={beeApiUrl}
|
|
||||||
beeDebugApiUrl={beeDebugApiUrl}
|
|
||||||
defaultRpcUrl={defaultRpcUrl}
|
|
||||||
/>
|
|
||||||
</React.StrictMode>,
|
</React.StrictMode>,
|
||||||
document.getElementById('root'),
|
document.getElementById('root'),
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,14 +1,15 @@
|
|||||||
import { Button, CircularProgress, Container, IconButton } from '@material-ui/core'
|
import { Button, CircularProgress, Container, IconButton } from '@material-ui/core'
|
||||||
import { createStyles, makeStyles, Theme } from '@material-ui/core/styles'
|
import { Theme, createStyles, makeStyles } from '@material-ui/core/styles'
|
||||||
import React, { ReactElement, useContext, useEffect } from 'react'
|
|
||||||
import { useSnackbar } from 'notistack'
|
import { useSnackbar } from 'notistack'
|
||||||
|
import React, { ReactElement, useContext, useEffect } from 'react'
|
||||||
import CloseIcon from 'remixicon-react/CloseCircleLineIcon'
|
import CloseIcon from 'remixicon-react/CloseCircleLineIcon'
|
||||||
import ErrorBoundary from '../components/ErrorBoundary'
|
import ErrorBoundary from '../components/ErrorBoundary'
|
||||||
|
import { Flex } from '../components/Flex'
|
||||||
import SideBar from '../components/SideBar'
|
import SideBar from '../components/SideBar'
|
||||||
|
import { BEE_DESKTOP_LATEST_RELEASE_PAGE } from '../constants'
|
||||||
|
import { useBeeDesktop, useNewBeeDesktopVersion } from '../hooks/apiHooks'
|
||||||
import { Context as BeeContext } from '../providers/Bee'
|
import { Context as BeeContext } from '../providers/Bee'
|
||||||
import { Context as SettingsContext } from '../providers/Settings'
|
import { Context as SettingsContext } from '../providers/Settings'
|
||||||
import { useBeeDesktop, useNewBeeDesktopVersion } from '../hooks/apiHooks'
|
|
||||||
import { BEE_DESKTOP_LATEST_RELEASE_PAGE } from '../constants'
|
|
||||||
|
|
||||||
const useStyles = makeStyles((theme: Theme) =>
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
createStyles({
|
createStyles({
|
||||||
@@ -27,53 +28,12 @@ interface Props {
|
|||||||
const Dashboard = (props: Props): ReactElement => {
|
const Dashboard = (props: Props): ReactElement => {
|
||||||
const classes = useStyles()
|
const classes = useStyles()
|
||||||
|
|
||||||
const { isLoading, isLatestBeeVersion, latestBeeRelease, latestBeeVersionUrl, latestUserVersion } =
|
const { isLoading } = useContext(BeeContext)
|
||||||
useContext(BeeContext)
|
|
||||||
const { isDesktop, desktopUrl } = useContext(SettingsContext)
|
const { isDesktop, desktopUrl } = useContext(SettingsContext)
|
||||||
const { desktopAutoUpdateEnabled } = useBeeDesktop(isDesktop, desktopUrl)
|
const { desktopAutoUpdateEnabled } = useBeeDesktop(isDesktop, desktopUrl)
|
||||||
const { newBeeDesktopVersion } = useNewBeeDesktopVersion(isDesktop, desktopUrl, desktopAutoUpdateEnabled)
|
const { newBeeDesktopVersion } = useNewBeeDesktopVersion(isDesktop, desktopUrl, desktopAutoUpdateEnabled)
|
||||||
const { enqueueSnackbar, closeSnackbar } = useSnackbar()
|
const { enqueueSnackbar, closeSnackbar } = useSnackbar()
|
||||||
|
|
||||||
// New version of Bee client notification
|
|
||||||
useEffect(() => {
|
|
||||||
if (!isLoading && !isDesktop && !isLatestBeeVersion && latestBeeRelease && latestUserVersion) {
|
|
||||||
enqueueSnackbar(`There is new Bee version ${latestBeeRelease?.name}!`, {
|
|
||||||
variant: 'warning',
|
|
||||||
preventDuplicate: true,
|
|
||||||
key: 'beeNewVersion',
|
|
||||||
persist: true,
|
|
||||||
action: key => (
|
|
||||||
<React.Fragment>
|
|
||||||
<Button
|
|
||||||
onClick={() => {
|
|
||||||
window.open(latestBeeVersionUrl)
|
|
||||||
closeSnackbar(key)
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
Download release
|
|
||||||
</Button>
|
|
||||||
<IconButton
|
|
||||||
onClick={() => {
|
|
||||||
closeSnackbar(key)
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<CloseIcon />
|
|
||||||
</IconButton>
|
|
||||||
</React.Fragment>
|
|
||||||
),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}, [
|
|
||||||
closeSnackbar,
|
|
||||||
enqueueSnackbar,
|
|
||||||
isLatestBeeVersion,
|
|
||||||
isDesktop,
|
|
||||||
latestBeeRelease,
|
|
||||||
latestBeeVersionUrl,
|
|
||||||
isLoading,
|
|
||||||
latestUserVersion,
|
|
||||||
])
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
// When autoupdate is enabled then we leave the version check for the built-in Electron update mechanism
|
// When autoupdate is enabled then we leave the version check for the built-in Electron update mechanism
|
||||||
if (desktopAutoUpdateEnabled) {
|
if (desktopAutoUpdateEnabled) {
|
||||||
@@ -122,13 +82,13 @@ const Dashboard = (props: Props): ReactElement => {
|
|||||||
)
|
)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div style={{ display: 'flex' }}>
|
<Flex>
|
||||||
<SideBar />
|
<SideBar />
|
||||||
<Container className={classes.content}>
|
<Container className={classes.content}>
|
||||||
{' '}
|
{' '}
|
||||||
<ErrorBoundary errorReporting={props.errorReporting}>{content}</ErrorBoundary>
|
<ErrorBoundary errorReporting={props.errorReporting}>{content}</ErrorBoundary>
|
||||||
</Container>
|
</Container>
|
||||||
</div>
|
</Flex>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
+8
-7
@@ -81,17 +81,18 @@ export class Token {
|
|||||||
return asString.slice(0, indexOfSignificantDigit + digits)
|
return asString.slice(0, indexOfSignificantDigit + digits)
|
||||||
}
|
}
|
||||||
|
|
||||||
minusBaseUnits(amount: string): Token {
|
minusBaseUnits(amount: string | BigNumber | bigint): Token {
|
||||||
|
const baseUnits = makeBigNumber(amount)
|
||||||
|
|
||||||
return new Token(
|
return new Token(
|
||||||
this.toBigNumber.minus(new BigNumber(amount).multipliedBy(new BigNumber(10).pow(this.decimals))),
|
this.toBigNumber.minus(baseUnits.multipliedBy(new BigNumber(10).pow(this.decimals))),
|
||||||
this.decimals,
|
this.decimals,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
plusBaseUnits(amount: string): Token {
|
plusBaseUnits(amount: string | BigNumber | bigint): Token {
|
||||||
return new Token(
|
const baseUnits = makeBigNumber(amount)
|
||||||
this.toBigNumber.plus(new BigNumber(amount).multipliedBy(new BigNumber(10).pow(this.decimals))),
|
|
||||||
this.decimals,
|
return new Token(this.toBigNumber.plus(baseUnits.multipliedBy(new BigNumber(10).pow(this.decimals))), this.decimals)
|
||||||
)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
|
import { BeeModes } from '@ethersphere/bee-js'
|
||||||
import { createStyles, makeStyles, Tab, Tabs, Theme } from '@material-ui/core'
|
import { createStyles, makeStyles, Tab, Tabs, Theme } from '@material-ui/core'
|
||||||
import { ReactElement } from 'react'
|
import { ReactElement, useContext } from 'react'
|
||||||
import { useNavigate } from 'react-router-dom'
|
import { useNavigate } from 'react-router-dom'
|
||||||
|
import { Context } from '../../providers/Bee'
|
||||||
import { ACCOUNT_TABS } from '../../routes'
|
import { ACCOUNT_TABS } from '../../routes'
|
||||||
|
|
||||||
const tabMap = {
|
const tabMap = {
|
||||||
@@ -8,10 +10,11 @@ const tabMap = {
|
|||||||
CHEQUEBOOK: 1,
|
CHEQUEBOOK: 1,
|
||||||
STAMPS: 2,
|
STAMPS: 2,
|
||||||
FEEDS: 3,
|
FEEDS: 3,
|
||||||
|
STAKING: 4,
|
||||||
}
|
}
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
active: 'WALLET' | 'CHEQUEBOOK' | 'STAMPS' | 'FEEDS'
|
active: 'WALLET' | 'CHEQUEBOOK' | 'STAMPS' | 'FEEDS' | 'STAKING'
|
||||||
}
|
}
|
||||||
|
|
||||||
const useStyles = makeStyles((theme: Theme) =>
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
@@ -20,16 +23,12 @@ const useStyles = makeStyles((theme: Theme) =>
|
|||||||
flexGrow: 1,
|
flexGrow: 1,
|
||||||
marginBottom: theme.spacing(4),
|
marginBottom: theme.spacing(4),
|
||||||
textTransform: 'none',
|
textTransform: 'none',
|
||||||
|
marginLeft: theme.spacing(-0.25),
|
||||||
|
marginRight: theme.spacing(-0.25),
|
||||||
},
|
},
|
||||||
leftTab: {
|
tab: {
|
||||||
marginRight: theme.spacing(0.125),
|
marginLeft: theme.spacing(0.25),
|
||||||
},
|
marginRight: theme.spacing(0.25),
|
||||||
centerTab: {
|
|
||||||
marginLeft: theme.spacing(0.125),
|
|
||||||
marginRight: theme.spacing(0.125),
|
|
||||||
},
|
|
||||||
rightTab: {
|
|
||||||
marginLeft: theme.spacing(0.125),
|
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
@@ -37,6 +36,7 @@ const useStyles = makeStyles((theme: Theme) =>
|
|||||||
export function AccountNavigation({ active }: Props): ReactElement {
|
export function AccountNavigation({ active }: Props): ReactElement {
|
||||||
const classes = useStyles()
|
const classes = useStyles()
|
||||||
const navigate = useNavigate()
|
const navigate = useNavigate()
|
||||||
|
const { nodeInfo } = useContext(Context)
|
||||||
|
|
||||||
function onChange(event: React.ChangeEvent<Record<string, never>>, newValue: number) {
|
function onChange(event: React.ChangeEvent<Record<string, never>>, newValue: number) {
|
||||||
navigate(ACCOUNT_TABS[newValue])
|
navigate(ACCOUNT_TABS[newValue])
|
||||||
@@ -45,10 +45,11 @@ export function AccountNavigation({ active }: Props): ReactElement {
|
|||||||
return (
|
return (
|
||||||
<div className={classes.root}>
|
<div className={classes.root}>
|
||||||
<Tabs value={tabMap[active]} onChange={onChange} variant="fullWidth">
|
<Tabs value={tabMap[active]} onChange={onChange} variant="fullWidth">
|
||||||
<Tab className={classes.leftTab} key="WALLET" label="Wallet" />
|
<Tab className={classes.tab} key="WALLET" label="Wallet" />
|
||||||
<Tab className={classes.centerTab} key="CHEQUEBOOK" label="Chequebook" />
|
<Tab className={classes.tab} key="CHEQUEBOOK" label="Chequebook" />
|
||||||
<Tab className={classes.centerTab} key="STAMPS" label="Stamps" />
|
<Tab className={classes.tab} key="STAMPS" label="Stamps" />
|
||||||
<Tab className={classes.rightTab} key="FEEDS" label="Feeds" />
|
<Tab className={classes.tab} key="FEEDS" label="Feeds" />
|
||||||
|
{nodeInfo?.beeMode === BeeModes.FULL ? <Tab className={classes.tab} key="STAKING" label="Staking" /> : null}
|
||||||
</Tabs>
|
</Tabs>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import { Utils } from '@ethersphere/bee-js'
|
||||||
|
import { Box } from '@material-ui/core'
|
||||||
import { ReactElement, useContext } from 'react'
|
import { ReactElement, useContext } from 'react'
|
||||||
import ExpandableList from '../../../components/ExpandableList'
|
import ExpandableList from '../../../components/ExpandableList'
|
||||||
import ExpandableListItem from '../../../components/ExpandableListItem'
|
import ExpandableListItem from '../../../components/ExpandableListItem'
|
||||||
@@ -7,7 +9,7 @@ import TroubleshootConnectionCard from '../../../components/TroubleshootConnecti
|
|||||||
import DepositModal from '../../../containers/DepositModal'
|
import DepositModal from '../../../containers/DepositModal'
|
||||||
import WithdrawModal from '../../../containers/WithdrawModal'
|
import WithdrawModal from '../../../containers/WithdrawModal'
|
||||||
import { useAccounting } from '../../../hooks/accounting'
|
import { useAccounting } from '../../../hooks/accounting'
|
||||||
import { CheckState, Context as BeeContext } from '../../../providers/Bee'
|
import { Context as BeeContext, CheckState } from '../../../providers/Bee'
|
||||||
import { Context as SettingsContext } from '../../../providers/Settings'
|
import { Context as SettingsContext } from '../../../providers/Settings'
|
||||||
import PeerBalances from '../../accounting/PeerBalances'
|
import PeerBalances from '../../accounting/PeerBalances'
|
||||||
import { AccountNavigation } from '../AccountNavigation'
|
import { AccountNavigation } from '../AccountNavigation'
|
||||||
@@ -16,9 +18,9 @@ import { Header } from '../Header'
|
|||||||
export function AccountChequebook(): ReactElement {
|
export function AccountChequebook(): ReactElement {
|
||||||
const { status, nodeAddresses, chequebookAddress, chequebookBalance, settlements, peerBalances } =
|
const { status, nodeAddresses, chequebookAddress, chequebookBalance, settlements, peerBalances } =
|
||||||
useContext(BeeContext)
|
useContext(BeeContext)
|
||||||
const { beeDebugApi } = useContext(SettingsContext)
|
const { beeApi } = useContext(SettingsContext)
|
||||||
|
|
||||||
const { accounting, totalUncashed, isLoadingUncashed } = useAccounting(beeDebugApi, settlements, peerBalances)
|
const { accounting, totalUncashed, isLoadingUncashed } = useAccounting(beeApi, settlements, peerBalances)
|
||||||
|
|
||||||
if (status.all === CheckState.ERROR) return <TroubleshootConnectionCard />
|
if (status.all === CheckState.ERROR) return <TroubleshootConnectionCard />
|
||||||
|
|
||||||
@@ -43,10 +45,12 @@ export function AccountChequebook(): ReactElement {
|
|||||||
label="Total Cheques Amount Sent"
|
label="Total Cheques Amount Sent"
|
||||||
value={`${settlements?.totalSent.toFixedDecimal()} xBZZ`}
|
value={`${settlements?.totalSent.toFixedDecimal()} xBZZ`}
|
||||||
/>
|
/>
|
||||||
<ExpandableListItem
|
<Box mb={2}>
|
||||||
label="Total Cheques Amount Received"
|
<ExpandableListItem
|
||||||
value={`${settlements?.totalReceived.toFixedDecimal()} xBZZ`}
|
label="Total Cheques Amount Received"
|
||||||
/>
|
value={`${settlements?.totalReceived.toFixedDecimal()} xBZZ`}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
<ExpandableListItemActions>
|
<ExpandableListItemActions>
|
||||||
<WithdrawModal />
|
<WithdrawModal />
|
||||||
<DepositModal />
|
<DepositModal />
|
||||||
@@ -54,7 +58,10 @@ export function AccountChequebook(): ReactElement {
|
|||||||
</ExpandableList>
|
</ExpandableList>
|
||||||
)}
|
)}
|
||||||
<ExpandableList label="Blockchain" defaultOpen>
|
<ExpandableList label="Blockchain" defaultOpen>
|
||||||
<ExpandableListItemKey label="Ethereum address" value={nodeAddresses?.ethereum || ''} />
|
<ExpandableListItemKey
|
||||||
|
label="Ethereum address"
|
||||||
|
value={nodeAddresses?.ethereum ? Utils.capitalizeAddressERC55(nodeAddresses.ethereum) : ''}
|
||||||
|
/>
|
||||||
<ExpandableListItemKey
|
<ExpandableListItemKey
|
||||||
label="Chequebook contract address"
|
label="Chequebook contract address"
|
||||||
value={chequebookAddress?.chequebookAddress || ''}
|
value={chequebookAddress?.chequebookAddress || ''}
|
||||||
|
|||||||
@@ -0,0 +1,53 @@
|
|||||||
|
import { ReactElement, useContext, useState } from 'react'
|
||||||
|
import ExpandableList from '../../../components/ExpandableList'
|
||||||
|
import ExpandableListItem from '../../../components/ExpandableListItem'
|
||||||
|
import ExpandableListItemActions from '../../../components/ExpandableListItemActions'
|
||||||
|
import { Loading } from '../../../components/Loading'
|
||||||
|
import TroubleshootConnectionCard from '../../../components/TroubleshootConnectionCard'
|
||||||
|
import StakeModal from '../../../containers/StakeModal'
|
||||||
|
import { CheckState, Context as BeeContext } from '../../../providers/Bee'
|
||||||
|
import { Context as BalanceContext } from '../../../providers/WalletBalance'
|
||||||
|
import { AccountNavigation } from '../AccountNavigation'
|
||||||
|
import { Header } from '../Header'
|
||||||
|
|
||||||
|
export function AccountStaking(): ReactElement {
|
||||||
|
const [loading, setLoading] = useState(false)
|
||||||
|
|
||||||
|
const { status, stake } = useContext(BeeContext)
|
||||||
|
const { balance } = useContext(BalanceContext)
|
||||||
|
|
||||||
|
if (status.all === CheckState.ERROR) return <TroubleshootConnectionCard />
|
||||||
|
|
||||||
|
function onStarted() {
|
||||||
|
setLoading(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
function onFinished() {
|
||||||
|
setLoading(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Header />
|
||||||
|
<AccountNavigation active="STAKING" />
|
||||||
|
<div>
|
||||||
|
{loading || stake?.toDecimal === undefined ? (
|
||||||
|
<Loading />
|
||||||
|
) : (
|
||||||
|
<ExpandableList label="Staking" defaultOpen>
|
||||||
|
<ExpandableListItem label="Staked BZZ" value={`${stake?.toSignificantDigits()} xBZZ`} />
|
||||||
|
{balance?.bzz ? (
|
||||||
|
<ExpandableListItem
|
||||||
|
label="Available xBZZ balance"
|
||||||
|
value={`${balance?.bzz.toSignificantDigits(4)} xBZZ`}
|
||||||
|
/>
|
||||||
|
) : null}
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
<StakeModal onStarted={onStarted} onFinished={onFinished} />
|
||||||
|
</ExpandableListItemActions>
|
||||||
|
</ExpandableList>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -1,10 +1,12 @@
|
|||||||
import { CircularProgress, Container, createStyles, makeStyles } from '@material-ui/core'
|
import { CircularProgress, Container, createStyles, makeStyles } from '@material-ui/core'
|
||||||
import { ReactElement, useContext, useEffect } from 'react'
|
import { ReactElement, useContext, useEffect } from 'react'
|
||||||
import PlusSquare from 'remixicon-react/AddBoxLineIcon'
|
|
||||||
import { useNavigate } from 'react-router'
|
import { useNavigate } from 'react-router'
|
||||||
|
import PlusSquare from 'remixicon-react/AddBoxLineIcon'
|
||||||
|
import { ChainSync } from '../../../components/ChainSync'
|
||||||
|
import { Loading } from '../../../components/Loading'
|
||||||
import { SwarmButton } from '../../../components/SwarmButton'
|
import { SwarmButton } from '../../../components/SwarmButton'
|
||||||
import TroubleshootConnectionCard from '../../../components/TroubleshootConnectionCard'
|
import TroubleshootConnectionCard from '../../../components/TroubleshootConnectionCard'
|
||||||
import { CheckState, Context as BeeContext } from '../../../providers/Bee'
|
import { Context as BeeContext, CheckState } from '../../../providers/Bee'
|
||||||
import { Context as StampsContext } from '../../../providers/Stamps'
|
import { Context as StampsContext } from '../../../providers/Stamps'
|
||||||
import { ROUTES } from '../../../routes'
|
import { ROUTES } from '../../../routes'
|
||||||
import StampsTable from '../../stamps/StampsTable'
|
import StampsTable from '../../stamps/StampsTable'
|
||||||
@@ -45,7 +47,7 @@ export function AccountStamps(): ReactElement {
|
|||||||
if (status.all === CheckState.ERROR) return <TroubleshootConnectionCard />
|
if (status.all === CheckState.ERROR) return <TroubleshootConnectionCard />
|
||||||
|
|
||||||
function navigateToNewStamp() {
|
function navigateToNewStamp() {
|
||||||
navigate(ROUTES.ACCOUNT_STAMPS_NEW)
|
navigate(ROUTES.ACCOUNT_STAMPS_NEW_STANDARD)
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -55,7 +57,8 @@ export function AccountStamps(): ReactElement {
|
|||||||
<div className={classes.root}>
|
<div className={classes.root}>
|
||||||
{error && (
|
{error && (
|
||||||
<Container style={{ textAlign: 'center', padding: '50px' }}>
|
<Container style={{ textAlign: 'center', padding: '50px' }}>
|
||||||
Error loading postage stamps details: {error.message}
|
<Loading />
|
||||||
|
<ChainSync />
|
||||||
</Container>
|
</Container>
|
||||||
)}
|
)}
|
||||||
{!error && (
|
{!error && (
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { BeeModes } from '@ethersphere/bee-js'
|
import { BeeModes, Utils } from '@ethersphere/bee-js'
|
||||||
import { Box, Grid, Typography } from '@material-ui/core'
|
import { Box, Grid, Typography } from '@material-ui/core'
|
||||||
import { ReactElement, useContext } from 'react'
|
import { ReactElement, useContext } from 'react'
|
||||||
import { useNavigate } from 'react-router'
|
import { useNavigate } from 'react-router'
|
||||||
@@ -11,7 +11,7 @@ import ExpandableListItemKey from '../../../components/ExpandableListItemKey'
|
|||||||
import { Loading } from '../../../components/Loading'
|
import { Loading } from '../../../components/Loading'
|
||||||
import { SwarmButton } from '../../../components/SwarmButton'
|
import { SwarmButton } from '../../../components/SwarmButton'
|
||||||
import TroubleshootConnectionCard from '../../../components/TroubleshootConnectionCard'
|
import TroubleshootConnectionCard from '../../../components/TroubleshootConnectionCard'
|
||||||
import { CheckState, Context as BeeContext } from '../../../providers/Bee'
|
import { Context as BeeContext, CheckState } from '../../../providers/Bee'
|
||||||
import { Context as SettingsContext } from '../../../providers/Settings'
|
import { Context as SettingsContext } from '../../../providers/Settings'
|
||||||
import { Context as BalanceProvider } from '../../../providers/WalletBalance'
|
import { Context as BalanceProvider } from '../../../providers/WalletBalance'
|
||||||
import { ROUTES } from '../../../routes'
|
import { ROUTES } from '../../../routes'
|
||||||
@@ -26,7 +26,7 @@ export function AccountWallet(): ReactElement {
|
|||||||
const navigate = useNavigate()
|
const navigate = useNavigate()
|
||||||
|
|
||||||
function onCheckTransactions() {
|
function onCheckTransactions() {
|
||||||
window.open(`https://blockscout.com/xdai/mainnet/address/${nodeAddresses?.ethereum}/transactions`, '_blank')
|
window.open(`https://gnosisscan.io/address/${nodeAddresses?.ethereum}`, '_blank')
|
||||||
}
|
}
|
||||||
|
|
||||||
function onInvite() {
|
function onInvite() {
|
||||||
@@ -56,7 +56,11 @@ export function AccountWallet(): ReactElement {
|
|||||||
{balance && nodeAddresses ? (
|
{balance && nodeAddresses ? (
|
||||||
<>
|
<>
|
||||||
<Box mb={0.25}>
|
<Box mb={0.25}>
|
||||||
<ExpandableListItemKey label="Node wallet address" value={nodeAddresses.ethereum} expanded />
|
<ExpandableListItemKey
|
||||||
|
label="Node wallet address"
|
||||||
|
value={Utils.capitalizeAddressERC55(nodeAddresses.ethereum)}
|
||||||
|
expanded
|
||||||
|
/>
|
||||||
</Box>
|
</Box>
|
||||||
<Box mb={0.25}>
|
<Box mb={0.25}>
|
||||||
<ExpandableListItem label="xDAI balance" value={`${balance.dai.toSignificantDigits(4)} xDAI`} />
|
<ExpandableListItem label="xDAI balance" value={`${balance.dai.toSignificantDigits(4)} xDAI`} />
|
||||||
@@ -72,7 +76,7 @@ export function AccountWallet(): ReactElement {
|
|||||||
)}
|
)}
|
||||||
<ExpandableListItemActions>
|
<ExpandableListItemActions>
|
||||||
<SwarmButton onClick={onCheckTransactions} iconType={Link}>
|
<SwarmButton onClick={onCheckTransactions} iconType={Link}>
|
||||||
Check transactions on Blockscout
|
Check transactions
|
||||||
</SwarmButton>
|
</SwarmButton>
|
||||||
{isDesktop && (
|
{isDesktop && (
|
||||||
<SwarmButton onClick={onInvite} iconType={Gift}>
|
<SwarmButton onClick={onInvite} iconType={Gift}>
|
||||||
|
|||||||
@@ -14,13 +14,15 @@ interface Props {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function PeerBalances({ accounting, isLoadingUncashed, totalUncashed }: Props): ReactElement | null {
|
export default function PeerBalances({ accounting, isLoadingUncashed, totalUncashed }: Props): ReactElement | null {
|
||||||
|
const uncashedPeers = accounting?.filter(({ uncashedAmount }) => uncashedAmount.toBigNumber.isGreaterThan('0')) || []
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ExpandableList
|
<ExpandableList
|
||||||
label={`Peers (${accounting?.length || 0})`}
|
label={`Peers (${uncashedPeers.length})`}
|
||||||
info={`${totalUncashed.toFixedDecimal()} xBZZ (uncashed)`}
|
info={`${totalUncashed.toFixedDecimal()} xBZZ (uncashed)`}
|
||||||
>
|
>
|
||||||
<ExpandableListItem label="Uncashed Amount Total" value={`${totalUncashed.toFixedDecimal()} xBZZ`} />
|
<ExpandableListItem label="Uncashed Amount Total" value={`${totalUncashed.toFixedDecimal()} xBZZ`} />
|
||||||
{accounting?.map(({ peer, balance, received, sent, uncashedAmount, total }) => (
|
{uncashedPeers.map(({ peer, balance, received, sent, uncashedAmount, total }) => (
|
||||||
<ExpandableList
|
<ExpandableList
|
||||||
key={peer}
|
key={peer}
|
||||||
label={`Peer ${peer.slice(0, 8)}[…]`}
|
label={`Peer ${peer.slice(0, 8)}[…]`}
|
||||||
|
|||||||
@@ -0,0 +1,106 @@
|
|||||||
|
import { FdpStorage } from '@fairdatasociety/fdp-storage'
|
||||||
|
import { Checkbox, InputBase, Typography } from '@material-ui/core'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
import { useEffect, useState } from 'react'
|
||||||
|
import RegisterIcon from 'remixicon-react/AddBoxLineIcon'
|
||||||
|
import LoginIcon from 'remixicon-react/LoginBoxLineIcon'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { Horizontal } from './Horizontal'
|
||||||
|
import { Vertical } from './Vertical'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
fdp: FdpStorage
|
||||||
|
onSuccessfulLogin: () => void
|
||||||
|
}
|
||||||
|
|
||||||
|
export function FdpLogin({ fdp, onSuccessfulLogin }: Props) {
|
||||||
|
const [username, setUsername] = useState<string>('')
|
||||||
|
const [password, setPassword] = useState<string>('')
|
||||||
|
const [remember, setRemember] = useState<boolean>(false)
|
||||||
|
const [sepolia, setSepolia] = useState<string>('https://sepolia.drpc.org')
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
|
||||||
|
const inputStyle = { background: 'white', padding: '2px 8px', width: '100%' }
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const storedSepolia = localStorage.getItem('sepolia')
|
||||||
|
|
||||||
|
if (storedSepolia) {
|
||||||
|
setSepolia(storedSepolia)
|
||||||
|
}
|
||||||
|
const fdpCredentials = localStorage.getItem('fdpCredentials')
|
||||||
|
|
||||||
|
if (fdpCredentials) {
|
||||||
|
const { username, password } = JSON.parse(fdpCredentials)
|
||||||
|
setUsername(username)
|
||||||
|
setPassword(password)
|
||||||
|
setRemember(true)
|
||||||
|
}
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
async function onLogin() {
|
||||||
|
localStorage.setItem('sepolia', sepolia)
|
||||||
|
|
||||||
|
if (remember) {
|
||||||
|
localStorage.setItem('fdpCredentials', JSON.stringify({ username, password }))
|
||||||
|
} else {
|
||||||
|
localStorage.removeItem('fdpCredentials')
|
||||||
|
}
|
||||||
|
enqueueSnackbar('Logging in...', { variant: 'info' })
|
||||||
|
try {
|
||||||
|
await fdp.account.login(username, password)
|
||||||
|
enqueueSnackbar('Logged in successfully', { variant: 'success' })
|
||||||
|
onSuccessfulLogin()
|
||||||
|
} catch {
|
||||||
|
enqueueSnackbar('Login failed', { variant: 'error' })
|
||||||
|
} finally {
|
||||||
|
setUsername('')
|
||||||
|
setPassword('')
|
||||||
|
setRemember(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function onRegister() {
|
||||||
|
window.open('https://create.fairdatasociety.org/', '_blank')
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
maxWidth: '500px',
|
||||||
|
margin: 'auto',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Vertical gap={16} full>
|
||||||
|
<Vertical gap={8} left full>
|
||||||
|
<Typography variant="body2">Sepolia JSON RPC</Typography>
|
||||||
|
<InputBase value={sepolia} onChange={e => setSepolia(e.target.value)} style={inputStyle} />
|
||||||
|
</Vertical>
|
||||||
|
<Vertical gap={8} left full>
|
||||||
|
<Typography variant="body2">Username</Typography>
|
||||||
|
<InputBase value={username} onChange={e => setUsername(e.target.value)} style={inputStyle} />
|
||||||
|
</Vertical>
|
||||||
|
<Vertical gap={8} left full>
|
||||||
|
<Typography variant="body2">Password</Typography>
|
||||||
|
<InputBase value={password} onChange={e => setPassword(e.target.value)} style={inputStyle} type="password" />
|
||||||
|
</Vertical>
|
||||||
|
<Vertical gap={8} left full>
|
||||||
|
<Horizontal>
|
||||||
|
<Checkbox checked={remember} onChange={e => setRemember(e.target.checked)} />
|
||||||
|
<Typography variant="body2">Remember me</Typography>
|
||||||
|
</Horizontal>
|
||||||
|
</Vertical>
|
||||||
|
<Vertical left full>
|
||||||
|
<Horizontal gap={4}>
|
||||||
|
<SwarmButton iconType={LoginIcon} onClick={onLogin}>
|
||||||
|
Login
|
||||||
|
</SwarmButton>
|
||||||
|
<SwarmButton iconType={RegisterIcon} onClick={onRegister}>
|
||||||
|
Registration
|
||||||
|
</SwarmButton>
|
||||||
|
</Horizontal>
|
||||||
|
</Vertical>
|
||||||
|
</Vertical>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,98 @@
|
|||||||
|
import { FdpStorage } from '@fairdatasociety/fdp-storage'
|
||||||
|
import { useState } from 'react'
|
||||||
|
import { CafeReactFs } from '../../react-fs/CafeReactFs'
|
||||||
|
import { FsItem, FsItemType } from '../../react-fs/CafeReactType'
|
||||||
|
import { joinUrl } from '../../react-fs/Utility'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
fdp: FdpStorage
|
||||||
|
name: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function FdpPod({ fdp, name }: Props) {
|
||||||
|
const [reloader, setReloader] = useState(0)
|
||||||
|
|
||||||
|
function reload() {
|
||||||
|
setReloader(reloader + 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<CafeReactFs
|
||||||
|
rootAlias={`/${name}`}
|
||||||
|
backgroundColor="#ffffff"
|
||||||
|
reloader={reloader}
|
||||||
|
onDeleteFile={async (path: string) => {
|
||||||
|
await fdp.file.delete(name, path)
|
||||||
|
reload()
|
||||||
|
}}
|
||||||
|
onDeleteDirectory={async (path: string) => {
|
||||||
|
await fdp.directory.delete(name, path)
|
||||||
|
reload()
|
||||||
|
}}
|
||||||
|
onUpload={(path: string) => {
|
||||||
|
const input = document.createElement('input')
|
||||||
|
input.type = 'file'
|
||||||
|
input.multiple = true
|
||||||
|
input.click()
|
||||||
|
|
||||||
|
return new Promise<void>(resolve => {
|
||||||
|
input.onchange = async () => {
|
||||||
|
if (!input.files || !input.files.length) {
|
||||||
|
resolve()
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
for (const file of Array.from(input.files)) {
|
||||||
|
const data = await file.arrayBuffer()
|
||||||
|
await fdp.file.uploadData(name, joinUrl(path, file.name), new Uint8Array(data))
|
||||||
|
}
|
||||||
|
reload()
|
||||||
|
resolve()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}}
|
||||||
|
onCreateDirectory={async (path: string) => {
|
||||||
|
// eslint-disable-next-line no-alert
|
||||||
|
const newDirectoryName = prompt('Directory name')
|
||||||
|
|
||||||
|
if (!newDirectoryName) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
await fdp.directory.create(name, joinUrl(path, newDirectoryName))
|
||||||
|
reload()
|
||||||
|
}}
|
||||||
|
// eslint-disable-next-line require-await
|
||||||
|
onSync={async () => {
|
||||||
|
setReloader(reloader + 1)
|
||||||
|
}}
|
||||||
|
download={async (path: string) => {
|
||||||
|
const data = await fdp.file.downloadData(name, path)
|
||||||
|
const url = URL.createObjectURL(new Blob([data]))
|
||||||
|
const a = document.createElement('a')
|
||||||
|
a.href = url
|
||||||
|
a.download = path.split('/').pop() || 'Untitled'
|
||||||
|
a.click()
|
||||||
|
}}
|
||||||
|
list={async (path: string) => {
|
||||||
|
const fdpResponse = await fdp.directory.read(name, path)
|
||||||
|
const items: FsItem[] = []
|
||||||
|
for (const directory of fdpResponse.directories) {
|
||||||
|
items.push({
|
||||||
|
name: directory.name,
|
||||||
|
$type: FsItemType.DIRECTORY,
|
||||||
|
id: directory.name,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
for (const file of fdpResponse.files) {
|
||||||
|
items.push({
|
||||||
|
name: file.name,
|
||||||
|
$type: FsItemType.FILE,
|
||||||
|
id: file.name,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return items
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
import { FdpStorage } from '@fairdatasociety/fdp-storage'
|
||||||
|
import { Pod } from '@fairdatasociety/fdp-storage/dist/pod/types'
|
||||||
|
import { CircularProgress, Typography } from '@material-ui/core'
|
||||||
|
import { FdpPod } from './FdpPod'
|
||||||
|
import { Vertical } from './Vertical'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
fdp: FdpStorage
|
||||||
|
pods: Pod[]
|
||||||
|
loadingPods: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
export function FdpPods({ fdp, pods, loadingPods }: Props) {
|
||||||
|
if (loadingPods) {
|
||||||
|
return (
|
||||||
|
<Vertical gap={32} full>
|
||||||
|
<CircularProgress />
|
||||||
|
<Typography>Loading your pods...</Typography>
|
||||||
|
</Vertical>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Vertical gap={16} full left>
|
||||||
|
{pods.map(pod => (
|
||||||
|
<FdpPod key={pod.index} fdp={fdp} name={pod.name} />
|
||||||
|
))}
|
||||||
|
</Vertical>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,22 @@
|
|||||||
|
interface Props {
|
||||||
|
children: React.ReactNode
|
||||||
|
p?: string
|
||||||
|
gap?: number
|
||||||
|
between?: boolean
|
||||||
|
background?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Horizontal({ children, p = '0', gap = 8, between, background }: Props) {
|
||||||
|
const style = {
|
||||||
|
display: 'flex',
|
||||||
|
flexDirection: 'row' as 'row', //eslint-disable-line
|
||||||
|
alignItems: 'center',
|
||||||
|
justifyContent: between ? 'space-between' : 'flex-start',
|
||||||
|
gap: `${gap}px`,
|
||||||
|
padding: p,
|
||||||
|
background,
|
||||||
|
width: between ? '100%' : 'auto',
|
||||||
|
}
|
||||||
|
|
||||||
|
return <div style={style}>{children}</div>
|
||||||
|
}
|
||||||
@@ -0,0 +1,20 @@
|
|||||||
|
interface Props {
|
||||||
|
children: React.ReactNode
|
||||||
|
p?: number
|
||||||
|
gap?: number
|
||||||
|
left?: boolean
|
||||||
|
full?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Vertical({ children, p = 0, gap = 0, left = false, full = false }: Props) {
|
||||||
|
const style = {
|
||||||
|
display: 'flex',
|
||||||
|
flexDirection: 'column' as 'column', //eslint-disable-line
|
||||||
|
alignItems: left ? 'flex-start' : 'center',
|
||||||
|
gap: `${gap}px`,
|
||||||
|
width: full ? '100%' : 'auto',
|
||||||
|
padding: `${p}px`,
|
||||||
|
}
|
||||||
|
|
||||||
|
return <div style={style}>{children}</div>
|
||||||
|
}
|
||||||
@@ -0,0 +1,163 @@
|
|||||||
|
import { Bee } from '@ethersphere/bee-js'
|
||||||
|
import { FdpStorage } from '@fairdatasociety/fdp-storage'
|
||||||
|
import { Pod } from '@fairdatasociety/fdp-storage/dist/pod/types'
|
||||||
|
import { CircularProgress, Typography } from '@material-ui/core'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
import { ReactElement, useEffect, useState } from 'react'
|
||||||
|
import ImportIcon from 'remixicon-react/AddBoxLineIcon'
|
||||||
|
import PlusCircle from 'remixicon-react/AddCircleLineIcon'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { joinUrl } from '../../react-fs/Utility'
|
||||||
|
import { ManifestJs } from '../../utils/manifest'
|
||||||
|
import { FdpLogin } from './FdpLogin'
|
||||||
|
import { FdpPods } from './FdpPods'
|
||||||
|
import { Horizontal } from './Horizontal'
|
||||||
|
import { Vertical } from './Vertical'
|
||||||
|
|
||||||
|
async function makeFdp(): Promise<FdpStorage | null> {
|
||||||
|
const bee = new Bee('http://localhost:1633')
|
||||||
|
const sepolia = localStorage.getItem('sepolia') ?? 'https://sepolia.drpc.org'
|
||||||
|
const postageBatches = await bee.getAllPostageBatch()
|
||||||
|
const usableBatches = postageBatches.filter(batch => batch.usable)
|
||||||
|
const highestCapacityBatch = usableBatches.length ? usableBatches.reduce((a, b) => (a.depth > b.depth ? a : b)) : null
|
||||||
|
|
||||||
|
if (!highestCapacityBatch) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
return new FdpStorage('http://localhost:1633', highestCapacityBatch.batchID, {
|
||||||
|
ensOptions: {
|
||||||
|
rpcUrl: sepolia,
|
||||||
|
contractAddresses: {
|
||||||
|
ensRegistry: '0x42a96D45d787685ac4b36292d218B106Fb39be7F',
|
||||||
|
fdsRegistrar: '0xFBF00389140C00384d88d458239833E3231a7414',
|
||||||
|
nameResolver: '0xE20ECe6Ea93c4edE41e4d3B973f6679F1E89986A',
|
||||||
|
publicResolver: '0xC904989B579c2B216A75723688C784038AA99B56',
|
||||||
|
reverseResolver: '0xbDC8D98d3cbFd68EA9c165E1f15Df6e77A2ae0C5',
|
||||||
|
},
|
||||||
|
gasEstimation: 1,
|
||||||
|
performChecks: true,
|
||||||
|
},
|
||||||
|
providerOptions: {
|
||||||
|
url: sepolia,
|
||||||
|
},
|
||||||
|
ensDomain: 'fds',
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function FDP(): ReactElement {
|
||||||
|
const [fdp, setFdp] = useState<FdpStorage | null>(null)
|
||||||
|
const [pods, setPods] = useState<Pod[]>([])
|
||||||
|
const [loggedIn, setLoggedIn] = useState<boolean>(false)
|
||||||
|
const [loadingPods, setLoadingPods] = useState<boolean>(false)
|
||||||
|
const [creatingPod, setCreatingPod] = useState<boolean>(false)
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
makeFdp().then(fdp => {
|
||||||
|
if (!fdp) {
|
||||||
|
enqueueSnackbar('FDP could not be initialized. Do you have a postage batch?', { variant: 'error' })
|
||||||
|
}
|
||||||
|
setFdp(fdp)
|
||||||
|
})
|
||||||
|
}, [enqueueSnackbar])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (fdp && loggedIn) {
|
||||||
|
setLoadingPods(true)
|
||||||
|
fdp.personalStorage.list().then(pods => {
|
||||||
|
setPods(pods.pods)
|
||||||
|
setLoadingPods(false)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}, [fdp, loggedIn])
|
||||||
|
|
||||||
|
function onSuccessfulLogin() {
|
||||||
|
setLoggedIn(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
function onCreatePod() {
|
||||||
|
if (!fdp) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (loadingPods || creatingPod) {
|
||||||
|
enqueueSnackbar('Please wait until the pods are loaded', { variant: 'info' })
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// eslint-disable-next-line no-alert
|
||||||
|
const name = prompt('Enter a name for the new pod')
|
||||||
|
|
||||||
|
if (name) {
|
||||||
|
setCreatingPod(true)
|
||||||
|
fdp.personalStorage.create(name).then(() => {
|
||||||
|
fdp.personalStorage.list().then(pods => {
|
||||||
|
setPods(pods.pods)
|
||||||
|
setCreatingPod(false)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function onImportPod() {
|
||||||
|
if (!fdp) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (loadingPods || creatingPod) {
|
||||||
|
enqueueSnackbar('Please wait until the pods are loaded', { variant: 'info' })
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// eslint-disable-next-line no-alert
|
||||||
|
const name = prompt('Enter a name for the new pod')
|
||||||
|
// eslint-disable-next-line no-alert
|
||||||
|
const importHash = prompt('Enter the Swarm reference')
|
||||||
|
|
||||||
|
if (!name || !importHash) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
setCreatingPod(true)
|
||||||
|
const bee = new Bee('http://localhost:1633')
|
||||||
|
const manifestJs = new ManifestJs(bee)
|
||||||
|
const entries = await manifestJs.getHashes(importHash)
|
||||||
|
await fdp.personalStorage.create(name)
|
||||||
|
for (const [path, hash] of Object.entries(entries)) {
|
||||||
|
await fdp.file.uploadData(name, joinUrl('/', path), await bee.downloadData(hash))
|
||||||
|
}
|
||||||
|
const pods = await fdp.personalStorage.list()
|
||||||
|
setPods(pods.pods)
|
||||||
|
setCreatingPod(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!fdp) {
|
||||||
|
return <CircularProgress />
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Vertical gap={32} full left>
|
||||||
|
<Horizontal between>
|
||||||
|
<Typography variant="h1">Files</Typography>
|
||||||
|
{loggedIn && (
|
||||||
|
<Horizontal gap={4}>
|
||||||
|
<SwarmButton onClick={onCreatePod} iconType={PlusCircle}>
|
||||||
|
Create
|
||||||
|
</SwarmButton>
|
||||||
|
<SwarmButton onClick={onImportPod} iconType={ImportIcon}>
|
||||||
|
Import
|
||||||
|
</SwarmButton>
|
||||||
|
</Horizontal>
|
||||||
|
)}
|
||||||
|
</Horizontal>
|
||||||
|
{!loggedIn && <FdpLogin fdp={fdp} onSuccessfulLogin={onSuccessfulLogin} />}
|
||||||
|
{loggedIn && <FdpPods fdp={fdp} pods={pods} loadingPods={loadingPods || creatingPod} />}
|
||||||
|
{loggedIn && !loadingPods && !creatingPod && pods.length === 0 && (
|
||||||
|
<Typography>
|
||||||
|
<strong>You do not have any pods yet.</strong> Get started by clicking the Create or Import button on the top
|
||||||
|
right.
|
||||||
|
</Typography>
|
||||||
|
)}
|
||||||
|
</Vertical>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -2,9 +2,9 @@ import { Box, Grid, Typography } from '@material-ui/core'
|
|||||||
import { Form, Formik } from 'formik'
|
import { Form, Formik } from 'formik'
|
||||||
import { useSnackbar } from 'notistack'
|
import { useSnackbar } from 'notistack'
|
||||||
import { ReactElement, useContext, useState } from 'react'
|
import { ReactElement, useContext, useState } from 'react'
|
||||||
|
import { useNavigate } from 'react-router'
|
||||||
import Check from 'remixicon-react/CheckLineIcon'
|
import Check from 'remixicon-react/CheckLineIcon'
|
||||||
import X from 'remixicon-react/CloseLineIcon'
|
import X from 'remixicon-react/CloseLineIcon'
|
||||||
import { useNavigate } from 'react-router'
|
|
||||||
import { DocumentationText } from '../../components/DocumentationText'
|
import { DocumentationText } from '../../components/DocumentationText'
|
||||||
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
||||||
@@ -30,7 +30,7 @@ const initialValues: FormValues = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function CreateNewFeed(): ReactElement {
|
export default function CreateNewFeed(): ReactElement {
|
||||||
const { beeApi, beeDebugApi } = useContext(SettingsContext)
|
const { beeApi } = useContext(SettingsContext)
|
||||||
const { identities, setIdentities } = useContext(FeedsContext)
|
const { identities, setIdentities } = useContext(FeedsContext)
|
||||||
const [loading, setLoading] = useState(false)
|
const [loading, setLoading] = useState(false)
|
||||||
const { enqueueSnackbar } = useSnackbar()
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
@@ -47,7 +47,7 @@ export default function CreateNewFeed(): ReactElement {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
const wallet = generateWallet()
|
const wallet = generateWallet()
|
||||||
const stamps = await beeDebugApi?.getAllPostageBatch()
|
const stamps = await beeApi.getAllPostageBatch()
|
||||||
|
|
||||||
if (!stamps || !stamps.length) {
|
if (!stamps || !stamps.length) {
|
||||||
enqueueSnackbar(<span>No stamp available</span>, { variant: 'error' })
|
enqueueSnackbar(<span>No stamp available</span>, { variant: 'error' })
|
||||||
|
|||||||
@@ -1,12 +1,10 @@
|
|||||||
import * as swarmCid from '@ethersphere/swarm-cid'
|
|
||||||
import { Box } from '@material-ui/core'
|
import { Box } from '@material-ui/core'
|
||||||
import { ReactElement, useContext, useEffect, useState } from 'react'
|
import { ReactElement, useContext, useEffect, useState } from 'react'
|
||||||
import X from 'remixicon-react/CloseLineIcon'
|
|
||||||
import { useNavigate, useParams } from 'react-router-dom'
|
import { useNavigate, useParams } from 'react-router-dom'
|
||||||
|
import X from 'remixicon-react/CloseLineIcon'
|
||||||
import { DocumentationText } from '../../components/DocumentationText'
|
import { DocumentationText } from '../../components/DocumentationText'
|
||||||
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
||||||
import ExpandableListItemLink from '../../components/ExpandableListItemLink'
|
|
||||||
import { HistoryHeader } from '../../components/HistoryHeader'
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
import { SwarmButton } from '../../components/SwarmButton'
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
import { Context as BeeContext } from '../../providers/Bee'
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
@@ -55,14 +53,8 @@ export function FeedSubpage(): ReactElement {
|
|||||||
<UploadArea showHelp={false} uploadOrigin={{ origin: 'FEED', uuid }} />
|
<UploadArea showHelp={false} uploadOrigin={{ origin: 'FEED', uuid }} />
|
||||||
{available && identity.feedHash ? (
|
{available && identity.feedHash ? (
|
||||||
<>
|
<>
|
||||||
<Box mb={0.25}>
|
|
||||||
<ExpandableListItemKey label="Feed hash" value={identity.feedHash} />
|
|
||||||
</Box>
|
|
||||||
<Box mb={4}>
|
<Box mb={4}>
|
||||||
<ExpandableListItemLink
|
<ExpandableListItemKey label="Feed hash" value={identity.feedHash} />
|
||||||
label="BZZ Link"
|
|
||||||
value={`https://${swarmCid.encodeFeedReference(identity.feedHash)}.bzz.link`}
|
|
||||||
/>
|
|
||||||
</Box>
|
</Box>
|
||||||
</>
|
</>
|
||||||
) : (
|
) : (
|
||||||
|
|||||||
@@ -1,16 +1,16 @@
|
|||||||
import { Box, Grid, Typography } from '@material-ui/core'
|
import { Box, Grid, Typography } from '@material-ui/core'
|
||||||
import { useSnackbar } from 'notistack'
|
import { useSnackbar } from 'notistack'
|
||||||
import { ReactElement, useContext, useEffect, useState } from 'react'
|
import { ReactElement, useContext, useEffect, useState } from 'react'
|
||||||
import X from 'remixicon-react/CloseLineIcon'
|
|
||||||
import Bookmark from 'remixicon-react/BookmarkLineIcon'
|
|
||||||
import { useNavigate, useParams } from 'react-router'
|
import { useNavigate, useParams } from 'react-router'
|
||||||
|
import Bookmark from 'remixicon-react/BookmarkLineIcon'
|
||||||
|
import X from 'remixicon-react/CloseLineIcon'
|
||||||
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
import { HistoryHeader } from '../../components/HistoryHeader'
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
import { SwarmButton } from '../../components/SwarmButton'
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
import { SelectEvent, SwarmSelect } from '../../components/SwarmSelect'
|
import { SelectEvent, SwarmSelect } from '../../components/SwarmSelect'
|
||||||
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
||||||
import { Context as BeeContext } from '../../providers/Bee'
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
import { Context as IdentityContext, Identity } from '../../providers/Feeds'
|
import { Identity, Context as IdentityContext } from '../../providers/Feeds'
|
||||||
import { Context as SettingsContext } from '../../providers/Settings'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
import { Context as StampContext } from '../../providers/Stamps'
|
import { Context as StampContext } from '../../providers/Stamps'
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
@@ -19,7 +19,7 @@ import { FeedPasswordDialog } from './FeedPasswordDialog'
|
|||||||
|
|
||||||
export default function UpdateFeed(): ReactElement {
|
export default function UpdateFeed(): ReactElement {
|
||||||
const { identities, setIdentities } = useContext(IdentityContext)
|
const { identities, setIdentities } = useContext(IdentityContext)
|
||||||
const { beeApi, beeDebugApi } = useContext(SettingsContext)
|
const { beeApi } = useContext(SettingsContext)
|
||||||
const { stamps, refresh } = useContext(StampContext)
|
const { stamps, refresh } = useContext(StampContext)
|
||||||
const { status } = useContext(BeeContext)
|
const { status } = useContext(BeeContext)
|
||||||
const { hash } = useParams()
|
const { hash } = useParams()
|
||||||
@@ -66,7 +66,7 @@ export default function UpdateFeed(): ReactElement {
|
|||||||
async function onFeedUpdate(identity: Identity, password?: string) {
|
async function onFeedUpdate(identity: Identity, password?: string) {
|
||||||
setLoading(true)
|
setLoading(true)
|
||||||
|
|
||||||
if (!beeApi || !beeDebugApi || !selectedStamp) {
|
if (!beeApi || !selectedStamp) {
|
||||||
enqueueSnackbar(<span>Bee API unavailabe</span>, { variant: 'error' })
|
enqueueSnackbar(<span>Bee API unavailabe</span>, { variant: 'error' })
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
|
|
||||||
@@ -74,7 +74,7 @@ export default function UpdateFeed(): ReactElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await updateFeed(beeApi, beeDebugApi, identity, hash!, selectedStamp, password as string) // eslint-disable-line
|
await updateFeed(beeApi, identity, hash!, selectedStamp, password as string) // eslint-disable-line
|
||||||
persistIdentity(identities, identity)
|
persistIdentity(identities, identity)
|
||||||
setIdentities([...identities])
|
setIdentities([...identities])
|
||||||
navigate(ROUTES.ACCOUNT_FEEDS_VIEW.replace(':uuid', identity.uuid))
|
navigate(ROUTES.ACCOUNT_FEEDS_VIEW.replace(':uuid', identity.uuid))
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { Box, Grid, Typography } from '@material-ui/core'
|
import { Box, Grid, Typography } from '@material-ui/core'
|
||||||
import { Web } from '@material-ui/icons'
|
import { Web } from '@material-ui/icons'
|
||||||
import { ReactElement } from 'react'
|
import { ReactElement, useMemo } from 'react'
|
||||||
import File from 'remixicon-react/FileLineIcon'
|
import File from 'remixicon-react/FileLineIcon'
|
||||||
import Folder from 'remixicon-react/FolderLineIcon'
|
import Folder from 'remixicon-react/FolderLineIcon'
|
||||||
import { FitImage } from '../../components/FitImage'
|
import { FitImage } from '../../components/FitImage'
|
||||||
@@ -8,35 +8,52 @@ import { shortenText } from '../../utils'
|
|||||||
import { getHumanReadableFileSize } from '../../utils/file'
|
import { getHumanReadableFileSize } from '../../utils/file'
|
||||||
import { shortenHash } from '../../utils/hash'
|
import { shortenHash } from '../../utils/hash'
|
||||||
import { AssetIcon } from './AssetIcon'
|
import { AssetIcon } from './AssetIcon'
|
||||||
|
import { FitVideo } from '../../components/FitVideo'
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
previewUri?: string
|
previewUri?: string
|
||||||
metadata?: Metadata
|
metadata?: Metadata
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: add optional prop for indexDocument when it is already known (e.g. downloading a manifest)
|
/* eslint-disable react/display-name */
|
||||||
|
const getPreviewComponent = (previewUri?: string, metadata?: Metadata) => {
|
||||||
|
if (metadata?.isVideo) {
|
||||||
|
return () => <FitVideo src={previewUri} maxWidth="250px" maxHeight="175px" />
|
||||||
|
}
|
||||||
|
|
||||||
export function AssetPreview({ metadata, previewUri }: Props): ReactElement | null {
|
if (metadata?.isImage) {
|
||||||
let previewComponent = <File />
|
return () => <FitImage maxWidth="250px" maxHeight="175px" alt="Upload Preview" src={previewUri} />
|
||||||
let type = metadata?.type
|
}
|
||||||
|
|
||||||
if (metadata?.isWebsite) {
|
if (metadata?.isWebsite) {
|
||||||
previewComponent = <Web />
|
return () => <AssetIcon icon={<Web />} />
|
||||||
type = 'Website'
|
|
||||||
} else if (metadata?.type === 'folder') {
|
|
||||||
previewComponent = <Folder />
|
|
||||||
type = 'Folder'
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (metadata?.type === 'folder') {
|
||||||
|
return () => <AssetIcon icon={<Folder />} />
|
||||||
|
}
|
||||||
|
|
||||||
|
return () => <AssetIcon icon={<File />} />
|
||||||
|
}
|
||||||
|
|
||||||
|
const getType = (metadata?: Metadata) => {
|
||||||
|
if (metadata?.isWebsite) return 'Website'
|
||||||
|
|
||||||
|
if (metadata?.type === 'folder') return 'Folder'
|
||||||
|
|
||||||
|
return metadata?.type
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: add optional prop for indexDocument when it is already known (e.g. downloading a manifest)
|
||||||
|
export function AssetPreview({ metadata, previewUri }: Props): ReactElement | null {
|
||||||
|
const PreviewAssetComponent = useMemo(() => getPreviewComponent(previewUri, metadata), [metadata, previewUri])
|
||||||
|
const type = useMemo(() => getType(metadata), [metadata])
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box mb={4}>
|
<Box mb={4}>
|
||||||
<Box bgcolor="background.paper">
|
<Box bgcolor="background.paper">
|
||||||
<Grid container direction="row">
|
<Grid container direction="row">
|
||||||
{previewUri ? (
|
<PreviewAssetComponent />
|
||||||
<FitImage maxWidth="250px" maxHeight="175px" alt="Upload Preview" src={previewUri} />
|
|
||||||
) : (
|
|
||||||
<AssetIcon icon={previewComponent} />
|
|
||||||
)}
|
|
||||||
<Box p={2}>
|
<Box p={2}>
|
||||||
{metadata?.hash && <Typography>Swarm Hash: {shortenHash(metadata.hash)}</Typography>}
|
{metadata?.hash && <Typography>Swarm Hash: {shortenHash(metadata.hash)}</Typography>}
|
||||||
{metadata?.name && metadata?.name !== metadata?.hash && (
|
{metadata?.name && metadata?.name !== metadata?.hash && (
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
import * as swarmCid from '@ethersphere/swarm-cid'
|
import { Utils } from '@ethersphere/bee-js'
|
||||||
import { Box } from '@material-ui/core'
|
import { Box } from '@material-ui/core'
|
||||||
import { ReactElement } from 'react'
|
import { ReactElement } from 'react'
|
||||||
import { Utils } from '@ethersphere/bee-js'
|
|
||||||
import { DocumentationText } from '../../components/DocumentationText'
|
import { DocumentationText } from '../../components/DocumentationText'
|
||||||
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
||||||
import ExpandableListItemLink from '../../components/ExpandableListItemLink'
|
import ExpandableListItemLink from '../../components/ExpandableListItemLink'
|
||||||
@@ -19,16 +18,6 @@ export function AssetSummary({ isWebsite, reference }: Props): ReactElement {
|
|||||||
<Box mb={4}>
|
<Box mb={4}>
|
||||||
{isHash && <ExpandableListItemKey label="Swarm hash" value={reference} />}
|
{isHash && <ExpandableListItemKey label="Swarm hash" value={reference} />}
|
||||||
{!isHash && <ExpandableListItemLink label="ENS" value={reference} />}
|
{!isHash && <ExpandableListItemLink label="ENS" value={reference} />}
|
||||||
<ExpandableListItemLink
|
|
||||||
label="Share on Swarm Gateway"
|
|
||||||
value={`https://gateway.ethswarm.org/access/${reference}`}
|
|
||||||
/>
|
|
||||||
{isWebsite && isHash && (
|
|
||||||
<ExpandableListItemLink
|
|
||||||
label="BZZ Link"
|
|
||||||
value={`https://${swarmCid.encodeManifestReference(reference).toString()}.bzz.link`}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</Box>
|
</Box>
|
||||||
<DocumentationText>
|
<DocumentationText>
|
||||||
The Swarm Gateway is graciously provided by the Swarm Foundation. This service is under development and provided
|
The Swarm Gateway is graciously provided by the Swarm Foundation. This service is under development and provided
|
||||||
|
|||||||
@@ -0,0 +1,92 @@
|
|||||||
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
|
import { Box } from '@material-ui/core'
|
||||||
|
import { ReactElement, useContext, useEffect, useRef, useState } from 'react'
|
||||||
|
import { DocumentationText } from '../../components/DocumentationText'
|
||||||
|
import { LinearProgressWithLabel } from '../../components/ProgressBar'
|
||||||
|
import { Tag } from '@ethersphere/bee-js'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
reference: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function AssetSyncing({ reference }: Props): ReactElement {
|
||||||
|
const { beeApi } = useContext(SettingsContext)
|
||||||
|
|
||||||
|
const syncTimer = useRef<NodeJS.Timer>()
|
||||||
|
const [isRetrieveChecking, setIsRetrieveChecking] = useState<boolean>(false)
|
||||||
|
const [syncProgress, setSyncProgress] = useState<number>(0)
|
||||||
|
|
||||||
|
const syncCheck = async () => {
|
||||||
|
if (!beeApi) return
|
||||||
|
|
||||||
|
let allTags: Tag[] = []
|
||||||
|
let offset = 0
|
||||||
|
const limit = 1000
|
||||||
|
let tagsBatch
|
||||||
|
|
||||||
|
do {
|
||||||
|
tagsBatch = await beeApi.getAllTags({ limit, offset })
|
||||||
|
allTags = allTags.concat(tagsBatch)
|
||||||
|
offset += limit
|
||||||
|
} while (tagsBatch.length === limit) // Continue if the batch is full, stop if fewer than the limit
|
||||||
|
|
||||||
|
const tag = allTags.find(t => t.address === reference)
|
||||||
|
|
||||||
|
if (tag) {
|
||||||
|
const progress = ((tag.seen + tag.synced) / tag.split) * 100
|
||||||
|
setSyncProgress(progress)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
syncTimer.current = setInterval(syncCheck, 2000)
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
if (syncTimer.current) {
|
||||||
|
clearInterval(syncTimer.current)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
|
}, [reference])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (syncProgress === 100 && syncTimer.current) {
|
||||||
|
clearInterval(syncTimer.current)
|
||||||
|
}
|
||||||
|
}, [syncProgress])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
/*
|
||||||
|
There are instances when it seems that the content isn't synchronized, despite being already available.
|
||||||
|
To ensure it's not due to invalid synchronization data,
|
||||||
|
verify availability from at least 70% using one of the stewardship endpoints.
|
||||||
|
*/
|
||||||
|
if (beeApi && !isRetrieveChecking && syncProgress > 10 && syncProgress < 100) {
|
||||||
|
// It's a long running task make sure only one run occurs at a time.
|
||||||
|
setIsRetrieveChecking(true)
|
||||||
|
|
||||||
|
beeApi.isReferenceRetrievable(reference).then(isRetriavable => {
|
||||||
|
if (isRetriavable) {
|
||||||
|
setSyncProgress(100)
|
||||||
|
}
|
||||||
|
|
||||||
|
setIsRetrieveChecking(false)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}, [syncProgress, isRetrieveChecking, beeApi, reference])
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Box mb={2}>
|
||||||
|
<DocumentationText>
|
||||||
|
Files are not immediately accessible on the Swarm network. Please wait until your upload is synced to the
|
||||||
|
network.{' '}
|
||||||
|
<a href="https://docs.ethswarm.org/docs/develop/access-the-swarm/syncing">Learn more about syncing</a>.
|
||||||
|
</DocumentationText>
|
||||||
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
<LinearProgressWithLabel value={syncProgress}></LinearProgressWithLabel>
|
||||||
|
</Box>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -1,17 +1,17 @@
|
|||||||
import { BeeModes, Utils } from '@ethersphere/bee-js'
|
import { BeeModes, Utils } from '@ethersphere/bee-js'
|
||||||
import { ManifestJs } from '@ethersphere/manifest-js'
|
|
||||||
import { useSnackbar } from 'notistack'
|
import { useSnackbar } from 'notistack'
|
||||||
import { ReactElement, useContext, useState } from 'react'
|
import { ReactElement, useContext, useState } from 'react'
|
||||||
import { useNavigate } from 'react-router-dom'
|
import { useNavigate } from 'react-router-dom'
|
||||||
import Search from 'remixicon-react/SearchLineIcon'
|
import Search from 'remixicon-react/SearchLineIcon'
|
||||||
import ExpandableListItemInput from '../../components/ExpandableListItemInput'
|
import ExpandableListItemInput from '../../components/ExpandableListItemInput'
|
||||||
import { History } from '../../components/History'
|
import { History } from '../../components/History'
|
||||||
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
import { Context as FileContext, defaultUploadOrigin } from '../../providers/File'
|
import { Context as FileContext, defaultUploadOrigin } from '../../providers/File'
|
||||||
import { Context as SettingsContext } from '../../providers/Settings'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
import { Context as BeeContext } from '../../providers/Bee'
|
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
import { recognizeEnsOrSwarmHash, regexpEns } from '../../utils'
|
import { recognizeEnsOrSwarmHash, regexpEns } from '../../utils'
|
||||||
import { determineHistoryName, HISTORY_KEYS, putHistory } from '../../utils/local-storage'
|
import { HISTORY_KEYS, determineHistoryName, putHistory } from '../../utils/local-storage'
|
||||||
|
import { ManifestJs } from '../../utils/manifest'
|
||||||
import { FileNavigation } from './FileNavigation'
|
import { FileNavigation } from './FileNavigation'
|
||||||
|
|
||||||
export function Download(): ReactElement {
|
export function Download(): ReactElement {
|
||||||
@@ -34,9 +34,7 @@ export function Download(): ReactElement {
|
|||||||
) {
|
) {
|
||||||
setReferenceError(undefined)
|
setReferenceError(undefined)
|
||||||
} else {
|
} else {
|
||||||
setReferenceError(
|
setReferenceError('Incorrect format of swarm hash. Expected 64 or 128 hexstring characters or ENS domain.')
|
||||||
'Incorrect format of swarm hash. Expected 64 or 128 hexstring characters, bzz.link url or ENS domain.',
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -86,7 +84,7 @@ export function Download(): ReactElement {
|
|||||||
<>
|
<>
|
||||||
{nodeInfo?.beeMode !== BeeModes.ULTRA_LIGHT && <FileNavigation active="DOWNLOAD" />}
|
{nodeInfo?.beeMode !== BeeModes.ULTRA_LIGHT && <FileNavigation active="DOWNLOAD" />}
|
||||||
<ExpandableListItemInput
|
<ExpandableListItemInput
|
||||||
label="Swarm Hash"
|
label="Swarm Hash or ENS"
|
||||||
onConfirm={value => onSwarmIdentifier(value)}
|
onConfirm={value => onSwarmIdentifier(value)}
|
||||||
onChange={validateChange}
|
onChange={validateChange}
|
||||||
helperText={referenceError}
|
helperText={referenceError}
|
||||||
|
|||||||
+29
-28
@@ -1,4 +1,3 @@
|
|||||||
import { ManifestJs } from '@ethersphere/manifest-js'
|
|
||||||
import { Box, Typography } from '@material-ui/core'
|
import { Box, Typography } from '@material-ui/core'
|
||||||
import { saveAs } from 'file-saver'
|
import { saveAs } from 'file-saver'
|
||||||
import JSZip from 'jszip'
|
import JSZip from 'jszip'
|
||||||
@@ -8,14 +7,16 @@ import { useNavigate, useParams } from 'react-router-dom'
|
|||||||
import { HistoryHeader } from '../../components/HistoryHeader'
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
import { Loading } from '../../components/Loading'
|
import { Loading } from '../../components/Loading'
|
||||||
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
||||||
import { META_FILE_NAME, PREVIEW_FILE_NAME } from '../../constants'
|
import { META_FILE_NAME } from '../../constants'
|
||||||
import { Context as BeeContext } from '../../providers/Bee'
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
import { Context as SettingsContext } from '../../providers/Settings'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
import { determineHistoryName, HISTORY_KEYS, putHistory } from '../../utils/local-storage'
|
import { determineHistoryName, HISTORY_KEYS, putHistory } from '../../utils/local-storage'
|
||||||
|
import { ManifestJs } from '../../utils/manifest'
|
||||||
import { AssetPreview } from './AssetPreview'
|
import { AssetPreview } from './AssetPreview'
|
||||||
import { AssetSummary } from './AssetSummary'
|
import { AssetSummary } from './AssetSummary'
|
||||||
import { DownloadActionBar } from './DownloadActionBar'
|
import { DownloadActionBar } from './DownloadActionBar'
|
||||||
|
import { AssetSyncing } from './AssetSyncing'
|
||||||
|
|
||||||
export function Share(): ReactElement {
|
export function Share(): ReactElement {
|
||||||
const { apiUrl, beeApi } = useContext(SettingsContext)
|
const { apiUrl, beeApi } = useContext(SettingsContext)
|
||||||
@@ -49,38 +50,35 @@ export function Share(): ReactElement {
|
|||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
const entries = await manifestJs.getHashes(reference)
|
|
||||||
|
const entries = await manifestJs.getHashes(reference, { exclude: [META_FILE_NAME] })
|
||||||
|
setSwarmEntries(entries)
|
||||||
|
|
||||||
const indexDocument = await manifestJs.getIndexDocumentPath(reference)
|
const indexDocument = await manifestJs.getIndexDocumentPath(reference)
|
||||||
setIndexDocument(indexDocument)
|
setIndexDocument(indexDocument)
|
||||||
|
|
||||||
const previewFile = entries[PREVIEW_FILE_NAME]
|
|
||||||
|
|
||||||
delete entries[META_FILE_NAME]
|
|
||||||
delete entries[PREVIEW_FILE_NAME]
|
|
||||||
setSwarmEntries(entries)
|
|
||||||
|
|
||||||
const count = Object.keys(entries).length
|
|
||||||
|
|
||||||
let metadata: Metadata | undefined = {
|
|
||||||
hash,
|
|
||||||
size: 0,
|
|
||||||
type: count > 1 ? 'folder' : 'unknown',
|
|
||||||
name: reference,
|
|
||||||
isWebsite: Boolean(indexDocument) && count > 1,
|
|
||||||
count,
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const mtdt = await beeApi.downloadFile(reference, META_FILE_NAME)
|
const remoteMetadata = await beeApi.downloadFile(reference, META_FILE_NAME)
|
||||||
const remoteMetadata = mtdt.data.text()
|
const formattedMetadata = JSON.parse(remoteMetadata.data.text()) as Metadata
|
||||||
metadata = { ...metadata, ...(JSON.parse(remoteMetadata) as Metadata) }
|
|
||||||
} catch (e) {} // eslint-disable-line no-empty
|
|
||||||
|
|
||||||
if (previewFile) {
|
if (formattedMetadata.isVideo || formattedMetadata.isImage) {
|
||||||
setPreview(`${apiUrl}/bzz/${reference}/${PREVIEW_FILE_NAME}`)
|
setPreview(`${apiUrl}/bzz/${reference}`)
|
||||||
|
}
|
||||||
|
setMetadata({ ...formattedMetadata, hash })
|
||||||
|
} catch (e) {
|
||||||
|
// if metadata is not available or invalid go with the default one
|
||||||
|
const count = Object.keys(entries).length
|
||||||
|
setMetadata({
|
||||||
|
hash,
|
||||||
|
type: count > 1 ? 'folder' : 'unknown',
|
||||||
|
name: reference,
|
||||||
|
count,
|
||||||
|
isWebsite: Boolean(indexDocument && /.*\.html?$/i.test(indexDocument)),
|
||||||
|
isVideo: Boolean(indexDocument && /.*\.(mp4|webm|ogg|mp3|ogg|wav)$/i.test(indexDocument)),
|
||||||
|
isImage: Boolean(indexDocument && /.*\.(jpg|jpeg|png|gif|webp|svg)$/i.test(indexDocument)),
|
||||||
|
// naive assumption based on indexDocument, we don't want to donwload the whole manifest
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
setMetadata(metadata)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function onOpen() {
|
function onOpen() {
|
||||||
@@ -152,6 +150,9 @@ export function Share(): ReactElement {
|
|||||||
<Box mb={4}>
|
<Box mb={4}>
|
||||||
<AssetSummary isWebsite={metadata?.isWebsite} reference={reference} />
|
<AssetSummary isWebsite={metadata?.isWebsite} reference={reference} />
|
||||||
</Box>
|
</Box>
|
||||||
|
<Box mb={4}>
|
||||||
|
<AssetSyncing reference={reference} />
|
||||||
|
</Box>
|
||||||
<DownloadActionBar
|
<DownloadActionBar
|
||||||
onOpen={onOpen}
|
onOpen={onOpen}
|
||||||
onCancel={onClose}
|
onCancel={onClose}
|
||||||
|
|||||||
+27
-38
@@ -6,19 +6,19 @@ import { DocumentationText } from '../../components/DocumentationText'
|
|||||||
import { HistoryHeader } from '../../components/HistoryHeader'
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
import { ProgressIndicator } from '../../components/ProgressIndicator'
|
import { ProgressIndicator } from '../../components/ProgressIndicator'
|
||||||
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
||||||
import { META_FILE_NAME, PREVIEW_FILE_NAME } from '../../constants'
|
import { META_FILE_NAME } from '../../constants'
|
||||||
import { CheckState, Context as BeeContext } from '../../providers/Bee'
|
import { Context as BeeContext, CheckState } from '../../providers/Bee'
|
||||||
import { Context as IdentityContext, Identity } from '../../providers/Feeds'
|
import { Identity, Context as IdentityContext } from '../../providers/Feeds'
|
||||||
import { Context as FileContext } from '../../providers/File'
|
import { Context as FileContext } from '../../providers/File'
|
||||||
import { Context as SettingsContext } from '../../providers/Settings'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
import { Context as StampsContext, EnrichedPostageBatch } from '../../providers/Stamps'
|
import { EnrichedPostageBatch, Context as StampsContext } from '../../providers/Stamps'
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
import { waitUntilStampUsable } from '../../utils'
|
import { waitUntilStampUsable } from '../../utils'
|
||||||
import { detectIndexHtml, getAssetNameFromFiles, packageFile } from '../../utils/file'
|
import { detectIndexHtml, getAssetNameFromFiles, packageFile } from '../../utils/file'
|
||||||
import { persistIdentity, updateFeed } from '../../utils/identity'
|
import { persistIdentity, updateFeed } from '../../utils/identity'
|
||||||
import { HISTORY_KEYS, putHistory } from '../../utils/local-storage'
|
import { HISTORY_KEYS, putHistory } from '../../utils/local-storage'
|
||||||
import { FeedPasswordDialog } from '../feeds/FeedPasswordDialog'
|
import { FeedPasswordDialog } from '../feeds/FeedPasswordDialog'
|
||||||
import { PostageStampCreation } from '../stamps/PostageStampCreation'
|
import { PostageStampAdvancedCreation } from '../stamps/PostageStampAdvancedCreation'
|
||||||
import { PostageStampSelector } from '../stamps/PostageStampSelector'
|
import { PostageStampSelector } from '../stamps/PostageStampSelector'
|
||||||
import { AssetPreview } from './AssetPreview'
|
import { AssetPreview } from './AssetPreview'
|
||||||
import { StampPreview } from './StampPreview'
|
import { StampPreview } from './StampPreview'
|
||||||
@@ -32,8 +32,8 @@ export function Upload(): ReactElement {
|
|||||||
const [showPasswordPrompt, setShowPasswordPrompt] = useState(false)
|
const [showPasswordPrompt, setShowPasswordPrompt] = useState(false)
|
||||||
|
|
||||||
const { stamps, refresh } = useContext(StampsContext)
|
const { stamps, refresh } = useContext(StampsContext)
|
||||||
const { beeApi, beeDebugApi } = useContext(SettingsContext)
|
const { beeApi } = useContext(SettingsContext)
|
||||||
const { files, setFiles, uploadOrigin, metadata, previewUri, previewBlob } = useContext(FileContext)
|
const { files, setFiles, uploadOrigin, metadata, previewUri } = useContext(FileContext)
|
||||||
const { identities, setIdentities } = useContext(IdentityContext)
|
const { identities, setIdentities } = useContext(IdentityContext)
|
||||||
const { status } = useContext(BeeContext)
|
const { status } = useContext(BeeContext)
|
||||||
|
|
||||||
@@ -98,52 +98,32 @@ export function Upload(): ReactElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const lastModified = files[0].lastModified
|
const lastModified = files[0].lastModified
|
||||||
|
|
||||||
// We want to store only some metadata
|
const metafile = new File([JSON.stringify(metadata)], META_FILE_NAME, {
|
||||||
const mtd: SwarmMetadata = {
|
|
||||||
name: metadata.name,
|
|
||||||
size: metadata.size,
|
|
||||||
}
|
|
||||||
|
|
||||||
// Type of the file only makes sense for a single file
|
|
||||||
if (files.length === 1) mtd.type = metadata.type
|
|
||||||
|
|
||||||
const metafile = new File([JSON.stringify(mtd)], META_FILE_NAME, {
|
|
||||||
type: 'application/json',
|
type: 'application/json',
|
||||||
lastModified,
|
lastModified,
|
||||||
})
|
})
|
||||||
fls.push(packageFile(metafile))
|
fls.push(packageFile(metafile))
|
||||||
|
|
||||||
if (previewBlob) {
|
|
||||||
const previewFile = new File([previewBlob], PREVIEW_FILE_NAME, {
|
|
||||||
type: 'image/jpeg',
|
|
||||||
lastModified,
|
|
||||||
})
|
|
||||||
fls.push(packageFile(previewFile))
|
|
||||||
}
|
|
||||||
|
|
||||||
setUploading(true)
|
setUploading(true)
|
||||||
|
|
||||||
if (beeDebugApi) {
|
await waitUntilStampUsable(stamp.batchID, beeApi)
|
||||||
await waitUntilStampUsable(stamp.batchID, beeDebugApi)
|
|
||||||
}
|
|
||||||
|
|
||||||
beeApi
|
beeApi
|
||||||
.uploadFiles(stamp.batchID, fls, { indexDocument })
|
.uploadFiles(stamp.batchID, fls, { indexDocument, deferred: true })
|
||||||
.then(hash => {
|
.then(hash => {
|
||||||
putHistory(HISTORY_KEYS.UPLOAD_HISTORY, hash.reference, getAssetNameFromFiles(files))
|
putHistory(HISTORY_KEYS.UPLOAD_HISTORY, hash.reference, getAssetNameFromFiles(files))
|
||||||
|
|
||||||
if (uploadOrigin.origin === 'UPLOAD') {
|
if (uploadOrigin.origin === 'UPLOAD') {
|
||||||
navigate(ROUTES.HASH.replace(':hash', hash.reference), { replace: true })
|
navigate(ROUTES.HASH.replace(':hash', hash.reference), { replace: true })
|
||||||
} else {
|
} else {
|
||||||
updateFeed(beeApi, beeDebugApi, identity as Identity, hash.reference, stamp.batchID, password as string).then(
|
updateFeed(beeApi, identity as Identity, hash.reference, stamp.batchID, password as string).then(() => {
|
||||||
() => {
|
persistIdentity(identities, identity as Identity)
|
||||||
persistIdentity(identities, identity as Identity)
|
setIdentities([...identities])
|
||||||
setIdentities([...identities])
|
navigate(ROUTES.ACCOUNT_FEEDS_VIEW.replace(':uuid', uploadOrigin.uuid as string), { replace: true })
|
||||||
navigate(ROUTES.ACCOUNT_FEEDS_VIEW.replace(':uuid', uploadOrigin.uuid as string), { replace: true })
|
})
|
||||||
},
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.catch(e => {
|
.catch(e => {
|
||||||
@@ -186,7 +166,7 @@ export function Upload(): ReactElement {
|
|||||||
{hasAnyStamps && stampMode === 'SELECT' ? (
|
{hasAnyStamps && stampMode === 'SELECT' ? (
|
||||||
<PostageStampSelector onSelect={stamp => setStamp(stamp)} defaultValue={stamp?.batchID} />
|
<PostageStampSelector onSelect={stamp => setStamp(stamp)} defaultValue={stamp?.batchID} />
|
||||||
) : (
|
) : (
|
||||||
<PostageStampCreation onFinished={() => setStampMode('SELECT')} />
|
<PostageStampAdvancedCreation onFinished={() => setStampMode('SELECT')} />
|
||||||
)}
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
<Box mb={4}>
|
<Box mb={4}>
|
||||||
@@ -204,7 +184,16 @@ export function Upload(): ReactElement {
|
|||||||
</Box>
|
</Box>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
{step === 2 && stamp && <StampPreview stamp={stamp} />}
|
{step === 2 && stamp && (
|
||||||
|
<>
|
||||||
|
<StampPreview stamp={stamp} />
|
||||||
|
<Box mb={4}>
|
||||||
|
<DocumentationText>
|
||||||
|
Please do not close the application until your file is uploaded to your local node!
|
||||||
|
</DocumentationText>
|
||||||
|
</Box>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
<UploadActionBar
|
<UploadActionBar
|
||||||
step={step}
|
step={step}
|
||||||
onCancel={reset}
|
onCancel={reset}
|
||||||
|
|||||||
@@ -1,22 +1,22 @@
|
|||||||
import { Box, Tooltip, Typography } from '@material-ui/core'
|
import { Box, Tooltip, Typography } from '@material-ui/core'
|
||||||
|
import { Wallet } from 'ethers'
|
||||||
import { useSnackbar } from 'notistack'
|
import { useSnackbar } from 'notistack'
|
||||||
import { ReactElement, useContext, useEffect, useState } from 'react'
|
import { ReactElement, useContext, useEffect, useState } from 'react'
|
||||||
|
import { useNavigate } from 'react-router'
|
||||||
import Check from 'remixicon-react/CheckLineIcon'
|
import Check from 'remixicon-react/CheckLineIcon'
|
||||||
import X from 'remixicon-react/CloseLineIcon'
|
import X from 'remixicon-react/CloseLineIcon'
|
||||||
import { useNavigate } from 'react-router'
|
|
||||||
import { Wallet } from 'ethers'
|
|
||||||
import ExpandableListItem from '../../components/ExpandableListItem'
|
import ExpandableListItem from '../../components/ExpandableListItem'
|
||||||
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
||||||
import { HistoryHeader } from '../../components/HistoryHeader'
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
import { Loading } from '../../components/Loading'
|
import { Loading } from '../../components/Loading'
|
||||||
import { SwarmButton } from '../../components/SwarmButton'
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
import { Context as BalanceProvider } from '../../providers/WalletBalance'
|
import { Token } from '../../models/Token'
|
||||||
import { Context as TopUpContext } from '../../providers/TopUp'
|
|
||||||
import { Context as SettingsContext } from '../../providers/Settings'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
|
import { Context as TopUpContext } from '../../providers/TopUp'
|
||||||
|
import { Context as BalanceProvider } from '../../providers/WalletBalance'
|
||||||
import { createGiftWallet } from '../../utils/desktop'
|
import { createGiftWallet } from '../../utils/desktop'
|
||||||
import { ResolvedWallet } from '../../utils/wallet'
|
import { ResolvedWallet } from '../../utils/wallet'
|
||||||
import { Token } from '../../models/Token'
|
|
||||||
|
|
||||||
const GIFT_WALLET_FUND_DAI_AMOUNT = Token.fromDecimal('0.1', 18)
|
const GIFT_WALLET_FUND_DAI_AMOUNT = Token.fromDecimal('0.1', 18)
|
||||||
const GIFT_WALLET_FUND_BZZ_AMOUNT = Token.fromDecimal('0.5', 16)
|
const GIFT_WALLET_FUND_BZZ_AMOUNT = Token.fromDecimal('0.5', 16)
|
||||||
@@ -31,6 +31,9 @@ export default function Index(): ReactElement {
|
|||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
async function mapGiftWallets() {
|
async function mapGiftWallets() {
|
||||||
|
if (!rpcProvider) {
|
||||||
|
return
|
||||||
|
}
|
||||||
const results = []
|
const results = []
|
||||||
for (const giftWallet of giftWallets) {
|
for (const giftWallet of giftWallets) {
|
||||||
results.push(await ResolvedWallet.make(giftWallet, rpcProvider))
|
results.push(await ResolvedWallet.make(giftWallet, rpcProvider))
|
||||||
|
|||||||
@@ -17,12 +17,12 @@ export function ChequebookInfoCard() {
|
|||||||
<Card
|
<Card
|
||||||
buttonProps={{
|
buttonProps={{
|
||||||
iconType: ExchangeFunds,
|
iconType: ExchangeFunds,
|
||||||
children: 'View chequebook',
|
children: 'Manage chequebook',
|
||||||
onClick: () => navigate(ROUTES.ACCOUNT_CHEQUEBOOK),
|
onClick: () => navigate(ROUTES.ACCOUNT_CHEQUEBOOK),
|
||||||
}}
|
}}
|
||||||
icon={<ExchangeFunds />}
|
icon={<ExchangeFunds />}
|
||||||
title={`${chequebookBalance?.availableBalance.toSignificantDigits(4)} xBZZ`}
|
title={`${chequebookBalance?.availableBalance.toSignificantDigits(4)} xBZZ`}
|
||||||
subtitle="Current chequebook balance."
|
subtitle="Network transfer balance."
|
||||||
status="ok"
|
status="ok"
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
@@ -32,7 +32,7 @@ export function ChequebookInfoCard() {
|
|||||||
<Card
|
<Card
|
||||||
buttonProps={{
|
buttonProps={{
|
||||||
iconType: ExchangeFunds,
|
iconType: ExchangeFunds,
|
||||||
children: 'View chequebook',
|
children: 'Manage chequebook',
|
||||||
onClick: () => navigate(ROUTES.ACCOUNT_CHEQUEBOOK),
|
onClick: () => navigate(ROUTES.ACCOUNT_CHEQUEBOOK),
|
||||||
}}
|
}}
|
||||||
icon={<ExchangeFunds />}
|
icon={<ExchangeFunds />}
|
||||||
|
|||||||
@@ -12,6 +12,18 @@ export default function NodeInfoCard(): ReactElement {
|
|||||||
const { status } = useContext(BeeContext)
|
const { status } = useContext(BeeContext)
|
||||||
const navigate = useNavigate()
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
if (status.all === CheckState.CONNECTING) {
|
||||||
|
return (
|
||||||
|
<Card
|
||||||
|
buttonProps={{ iconType: Settings, children: 'Open node setup', onClick: () => navigate(ROUTES.STATUS) }}
|
||||||
|
icon={<Globe />}
|
||||||
|
title="Connecting..."
|
||||||
|
subtitle="Attempting to establish connection to your Bee node."
|
||||||
|
status="connecting"
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
if (status.all === CheckState.STARTING) {
|
if (status.all === CheckState.STARTING) {
|
||||||
return (
|
return (
|
||||||
<Card
|
<Card
|
||||||
|
|||||||
+17
-42
@@ -1,5 +1,6 @@
|
|||||||
import { Button } from '@material-ui/core'
|
import { Button } from '@material-ui/core'
|
||||||
import { ReactElement, useContext } from 'react'
|
import { ReactElement, useContext } from 'react'
|
||||||
|
import { ChainSync } from '../../components/ChainSync'
|
||||||
import ExpandableListItem from '../../components/ExpandableListItem'
|
import ExpandableListItem from '../../components/ExpandableListItem'
|
||||||
import Map from '../../components/Map'
|
import Map from '../../components/Map'
|
||||||
import { BEE_DESKTOP_LATEST_RELEASE_PAGE } from '../../constants'
|
import { BEE_DESKTOP_LATEST_RELEASE_PAGE } from '../../constants'
|
||||||
@@ -12,38 +13,34 @@ import NodeInfoCard from './NodeInfoCard'
|
|||||||
import { WalletInfoCard } from './WalletInfoCard'
|
import { WalletInfoCard } from './WalletInfoCard'
|
||||||
|
|
||||||
export default function Status(): ReactElement {
|
export default function Status(): ReactElement {
|
||||||
const {
|
const { beeVersion, status, topology, nodeInfo, chainId } = useContext(BeeContext)
|
||||||
debugApiReadiness,
|
|
||||||
status,
|
|
||||||
latestUserVersion,
|
|
||||||
isLatestBeeVersion,
|
|
||||||
latestBeeVersionUrl,
|
|
||||||
topology,
|
|
||||||
nodeInfo,
|
|
||||||
chainId,
|
|
||||||
} = useContext(BeeContext)
|
|
||||||
const { isDesktop, desktopUrl } = useContext(SettingsContext)
|
const { isDesktop, desktopUrl } = useContext(SettingsContext)
|
||||||
const { beeDesktopVersion } = useBeeDesktop(isDesktop, desktopUrl)
|
const { beeDesktopVersion } = useBeeDesktop(isDesktop, desktopUrl)
|
||||||
const { newBeeDesktopVersion } = useNewBeeDesktopVersion(isDesktop, desktopUrl, false)
|
const { newBeeDesktopVersion } = useNewBeeDesktopVersion(isDesktop, desktopUrl, false)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<div style={{ display: 'flex', justifyContent: 'space-between', alignItems: 'stretch', alignContent: 'stretch' }}>
|
<div
|
||||||
|
style={{
|
||||||
|
display: 'flex',
|
||||||
|
justifyContent: 'space-between',
|
||||||
|
alignItems: 'stretch',
|
||||||
|
alignContent: 'stretch',
|
||||||
|
gap: '8px',
|
||||||
|
}}
|
||||||
|
>
|
||||||
<NodeInfoCard />
|
<NodeInfoCard />
|
||||||
{debugApiReadiness && (
|
<WalletInfoCard />
|
||||||
<>
|
<ChequebookInfoCard />
|
||||||
<div style={{ width: '8px' }}></div>
|
|
||||||
<WalletInfoCard />
|
|
||||||
<div style={{ width: '8px' }}></div>
|
|
||||||
<ChequebookInfoCard />
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
<div style={{ height: '16px' }} />
|
<div style={{ height: '16px' }} />
|
||||||
<Map error={status.topology.checkState !== 'OK'} />
|
<Map error={status.topology.checkState !== 'OK'} />
|
||||||
<div style={{ height: '2px' }} />
|
<div style={{ height: '2px' }} />
|
||||||
<ExpandableListItem label="Connected peers" value={topology?.connected ?? '-'} />
|
<ExpandableListItem label="Connected peers" value={topology?.connected ?? '-'} />
|
||||||
<ExpandableListItem label="Population" value={topology?.population ?? '-'} />
|
<ExpandableListItem label="Population" value={topology?.population ?? '-'} />
|
||||||
|
<ExpandableListItem label="Depth" value={topology?.depth ?? '-'} />
|
||||||
|
<ChainSync />
|
||||||
|
|
||||||
<div style={{ height: '16px' }} />
|
<div style={{ height: '16px' }} />
|
||||||
{isDesktop && (
|
{isDesktop && (
|
||||||
<ExpandableListItem
|
<ExpandableListItem
|
||||||
@@ -65,29 +62,7 @@ export default function Status(): ReactElement {
|
|||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
<ExpandableListItem
|
<ExpandableListItem label="Bee version" value={beeVersion} />
|
||||||
label="Bee version"
|
|
||||||
value={
|
|
||||||
<div>
|
|
||||||
<a href="https://github.com/ethersphere/bee" rel="noreferrer" target="_blank">
|
|
||||||
Bee
|
|
||||||
</a>
|
|
||||||
{` ${latestUserVersion ?? '-'} `}
|
|
||||||
{latestUserVersion && !isDesktop && (
|
|
||||||
<Button
|
|
||||||
size="small"
|
|
||||||
variant="outlined"
|
|
||||||
href={latestBeeVersionUrl}
|
|
||||||
disabled={isLatestBeeVersion}
|
|
||||||
target="_blank"
|
|
||||||
style={{ height: '26px' }}
|
|
||||||
>
|
|
||||||
{isLatestBeeVersion ? 'latest' : 'update'}
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
<ExpandableListItem label="Mode" value={nodeInfo?.beeMode} />
|
<ExpandableListItem label="Mode" value={nodeInfo?.beeMode} />
|
||||||
{chainId !== null && <ExpandableListItem label="Blockchain network" value={chainIdToName(chainId)} />}
|
{chainId !== null && <ExpandableListItem label="Blockchain network" value={chainIdToName(chainId)} />}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,35 +1,37 @@
|
|||||||
import { BeeModes } from '@ethersphere/bee-js'
|
import { BeeModes } from '@ethersphere/bee-js'
|
||||||
import { Box, Grid, Typography } from '@material-ui/core'
|
import { Box, Grid, Typography } from '@material-ui/core'
|
||||||
import { useSnackbar } from 'notistack'
|
import { useSnackbar } from 'notistack'
|
||||||
import { ReactElement, useContext, useEffect, useState } from 'react'
|
import { ReactElement, useContext, useEffect } from 'react'
|
||||||
import { useNavigate } from 'react-router'
|
import { useNavigate } from 'react-router'
|
||||||
|
import { ChainSync } from '../../components/ChainSync'
|
||||||
import { Waiting } from '../../components/Waiting'
|
import { Waiting } from '../../components/Waiting'
|
||||||
import { Context } from '../../providers/Bee'
|
import { Context } from '../../providers/Settings'
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
|
|
||||||
const STARTED_UPGRADE_AT = 'started-upgrade-at'
|
|
||||||
|
|
||||||
export default function LightModeRestart(): ReactElement {
|
export default function LightModeRestart(): ReactElement {
|
||||||
const [startedAt] = useState(Number.parseInt(localStorage.getItem(STARTED_UPGRADE_AT) ?? Date.now().toFixed()))
|
|
||||||
const { apiHealth, nodeInfo } = useContext(Context)
|
|
||||||
const navigate = useNavigate()
|
const navigate = useNavigate()
|
||||||
const { enqueueSnackbar } = useSnackbar()
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
const { beeApi } = useContext(Context)
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
localStorage.setItem(STARTED_UPGRADE_AT, startedAt.toFixed())
|
if (!beeApi) {
|
||||||
}, [startedAt])
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (Date.now() - startedAt < 45_000) {
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if (apiHealth && nodeInfo?.beeMode === BeeModes.LIGHT) {
|
const interval = setInterval(() => {
|
||||||
enqueueSnackbar('Upgraded to light node', { variant: 'success' })
|
beeApi
|
||||||
localStorage.removeItem(STARTED_UPGRADE_AT)
|
.getNodeInfo()
|
||||||
navigate(ROUTES.INFO)
|
.then(nodeInfo => {
|
||||||
}
|
if (nodeInfo.beeMode === BeeModes.LIGHT) {
|
||||||
}, [startedAt, navigate, nodeInfo, apiHealth, enqueueSnackbar])
|
enqueueSnackbar('Upgraded to light node', { variant: 'success' })
|
||||||
|
navigate(ROUTES.INFO)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(console.error) // eslint-disable-line
|
||||||
|
}, 3_000)
|
||||||
|
|
||||||
|
return () => clearInterval(interval)
|
||||||
|
}, [beeApi, enqueueSnackbar, navigate])
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Grid container direction="column" justifyContent="center" alignItems="center">
|
<Grid container direction="column" justifyContent="center" alignItems="center">
|
||||||
@@ -41,9 +43,12 @@ export default function LightModeRestart(): ReactElement {
|
|||||||
<strong>Upgrading Bee</strong>
|
<strong>Upgrading Bee</strong>
|
||||||
</Typography>
|
</Typography>
|
||||||
</Box>
|
</Box>
|
||||||
<Typography>
|
<Box mb={1}>
|
||||||
You will be redirected automatically once your node is up and running. This may take up to 10 minutes.
|
<Typography>
|
||||||
</Typography>
|
You will be redirected automatically once your node is up and running. This may take up to 10 minutes.
|
||||||
|
</Typography>
|
||||||
|
</Box>
|
||||||
|
<ChainSync />
|
||||||
</Grid>
|
</Grid>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,9 +10,7 @@ import { getDesktopConfiguration, restartBeeNode, setJsonRpcInDesktop } from '..
|
|||||||
export default function SettingsPage(): ReactElement {
|
export default function SettingsPage(): ReactElement {
|
||||||
const {
|
const {
|
||||||
apiUrl,
|
apiUrl,
|
||||||
apiDebugUrl,
|
|
||||||
setApiUrl,
|
setApiUrl,
|
||||||
setDebugApiUrl,
|
|
||||||
lockedApiSettings,
|
lockedApiSettings,
|
||||||
cors,
|
cors,
|
||||||
dataDir,
|
dataDir,
|
||||||
@@ -30,7 +28,7 @@ export default function SettingsPage(): ReactElement {
|
|||||||
try {
|
try {
|
||||||
setAndPersistJsonRpcProvider(value)
|
setAndPersistJsonRpcProvider(value)
|
||||||
|
|
||||||
const shouldUpdateDesktop = isDesktop && (await getDesktopConfiguration(desktopUrl))['swap-endpoint']
|
const shouldUpdateDesktop = isDesktop && (await getDesktopConfiguration(desktopUrl))['blockchain-rpc-endpoint']
|
||||||
|
|
||||||
if (shouldUpdateDesktop) {
|
if (shouldUpdateDesktop) {
|
||||||
await setJsonRpcInDesktop(desktopUrl, value)
|
await setJsonRpcInDesktop(desktopUrl, value)
|
||||||
@@ -68,12 +66,6 @@ export default function SettingsPage(): ReactElement {
|
|||||||
onConfirm={setApiUrl}
|
onConfirm={setApiUrl}
|
||||||
locked={lockedApiSettings || isDesktop}
|
locked={lockedApiSettings || isDesktop}
|
||||||
/>
|
/>
|
||||||
<ExpandableListItemInput
|
|
||||||
label="Bee Debug API"
|
|
||||||
value={apiDebugUrl}
|
|
||||||
onConfirm={setDebugApiUrl}
|
|
||||||
locked={lockedApiSettings || isDesktop}
|
|
||||||
/>
|
|
||||||
<ExpandableListItemInput
|
<ExpandableListItemInput
|
||||||
label="Blockchain RPC URL"
|
label="Blockchain RPC URL"
|
||||||
value={rpcProviderUrl}
|
value={rpcProviderUrl}
|
||||||
|
|||||||
+3
-3
@@ -2,7 +2,7 @@ import { ReactElement } from 'react'
|
|||||||
import { useNavigate } from 'react-router'
|
import { useNavigate } from 'react-router'
|
||||||
import { HistoryHeader } from '../../components/HistoryHeader'
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
import { PostageStampCreation } from './PostageStampCreation'
|
import { PostageStampAdvancedCreation } from './PostageStampAdvancedCreation'
|
||||||
|
|
||||||
export function CreatePostageStampPage(): ReactElement {
|
export function CreatePostageStampPage(): ReactElement {
|
||||||
const navigate = useNavigate()
|
const navigate = useNavigate()
|
||||||
@@ -13,8 +13,8 @@ export function CreatePostageStampPage(): ReactElement {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<HistoryHeader>Buy new postage stamp</HistoryHeader>
|
<HistoryHeader>Buy new postage stamp batch</HistoryHeader>
|
||||||
<PostageStampCreation onFinished={onFinished} />
|
<PostageStampAdvancedCreation onFinished={onFinished} />
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,20 @@
|
|||||||
|
import { ReactElement } from 'react'
|
||||||
|
import { useNavigate } from 'react-router'
|
||||||
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
import { PostageStampStandardCreation } from './PostageStampStandardCreation'
|
||||||
|
|
||||||
|
export function CreatePostageStampBasicPage(): ReactElement {
|
||||||
|
const navigate = useNavigate()
|
||||||
|
|
||||||
|
function onFinished() {
|
||||||
|
navigate(ROUTES.ACCOUNT_STAMPS)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<HistoryHeader>Buy new postage stamp batch</HistoryHeader>
|
||||||
|
<PostageStampStandardCreation onFinished={onFinished} />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,295 @@
|
|||||||
|
import { PostageBatchOptions, Utils } from '@ethersphere/bee-js'
|
||||||
|
import { Box, Grid, IconButton, Typography, createStyles, makeStyles } from '@material-ui/core'
|
||||||
|
import BigNumber from 'bignumber.js'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
import { ReactElement, useContext, useState } from 'react'
|
||||||
|
import { Link } from 'react-router-dom'
|
||||||
|
import Check from 'remixicon-react/CheckLineIcon'
|
||||||
|
import Info from 'remixicon-react/InformationLineIcon'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { SwarmSelect } from '../../components/SwarmSelect'
|
||||||
|
import { SwarmTextInput } from '../../components/SwarmTextInput'
|
||||||
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
|
import { Context as StampsContext } from '../../providers/Stamps'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
import { calculateStampPrice, convertAmountToSeconds, secondsToTimeString, waitUntilStampExists } from '../../utils'
|
||||||
|
import { getHumanReadableFileSize } from '../../utils/file'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
onFinished: () => void
|
||||||
|
}
|
||||||
|
|
||||||
|
const useStyles = makeStyles(() =>
|
||||||
|
createStyles({
|
||||||
|
link: {
|
||||||
|
color: '#dd7700',
|
||||||
|
textDecoration: 'underline',
|
||||||
|
'&:hover': {
|
||||||
|
textDecoration: 'none',
|
||||||
|
|
||||||
|
// https://github.com/mui-org/material-ui/issues/22543
|
||||||
|
'@media (hover: none)': {
|
||||||
|
textDecoration: 'none',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
stampVolumeWrapper: {
|
||||||
|
width: 'fit-content',
|
||||||
|
'& button': {
|
||||||
|
marginLeft: 4,
|
||||||
|
width: 24,
|
||||||
|
padding: 2,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
export function PostageStampAdvancedCreation({ onFinished }: Props): ReactElement {
|
||||||
|
const classes = useStyles()
|
||||||
|
const { chainState } = useContext(BeeContext)
|
||||||
|
const { refresh } = useContext(StampsContext)
|
||||||
|
const { beeApi } = useContext(SettingsContext)
|
||||||
|
|
||||||
|
const [depthInput, setDepthInput] = useState<string>('')
|
||||||
|
const [amountInput, setAmountInput] = useState<string>('')
|
||||||
|
const [labelInput, setLabelInput] = useState('')
|
||||||
|
const [immutable, setImmutable] = useState(false)
|
||||||
|
const [depthError, setDepthError] = useState<string>('')
|
||||||
|
const [amountError, setAmountError] = useState<string>('')
|
||||||
|
const [submitting, setSubmitting] = useState(false)
|
||||||
|
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
|
||||||
|
function getTtl(amount: number): string {
|
||||||
|
const isCurrentPriceAvailable = chainState && chainState.currentPrice
|
||||||
|
|
||||||
|
if (amount <= 0 || !isCurrentPriceAvailable) {
|
||||||
|
return '-'
|
||||||
|
}
|
||||||
|
|
||||||
|
const pricePerBlock = Number.parseInt(chainState.currentPrice, 10)
|
||||||
|
|
||||||
|
return `${secondsToTimeString(
|
||||||
|
convertAmountToSeconds(amount, pricePerBlock),
|
||||||
|
)} (with price of ${pricePerBlock.toFixed(0)} PLUR per block)`
|
||||||
|
}
|
||||||
|
|
||||||
|
function getPrice(depth: number, amount: bigint): string {
|
||||||
|
const hasInvalidInput = amount <= 0 || isNaN(depth) || depth < 17 || depth > 255
|
||||||
|
|
||||||
|
if (hasInvalidInput) {
|
||||||
|
return '-'
|
||||||
|
}
|
||||||
|
|
||||||
|
const price = calculateStampPrice(depth, amount)
|
||||||
|
|
||||||
|
return `${price.toSignificantDigits()} xBZZ`
|
||||||
|
}
|
||||||
|
|
||||||
|
async function submit() {
|
||||||
|
try {
|
||||||
|
// This is really just a typeguard, the validation pretty much guarantees these will have the right values
|
||||||
|
if (!depthInput || !amountInput) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!beeApi) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
setSubmitting(true)
|
||||||
|
const amount = BigInt(amountInput)
|
||||||
|
const depth = Number.parseInt(depthInput)
|
||||||
|
const options: PostageBatchOptions = {
|
||||||
|
waitForUsable: false,
|
||||||
|
label: labelInput || undefined,
|
||||||
|
immutableFlag: immutable,
|
||||||
|
}
|
||||||
|
|
||||||
|
const batchId = await beeApi.createPostageBatch(amount.toString(), depth, options)
|
||||||
|
await waitUntilStampExists(batchId, beeApi)
|
||||||
|
await refresh()
|
||||||
|
onFinished()
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e) // eslint-disable-line
|
||||||
|
enqueueSnackbar(`Error: ${(e as Error).message}`, { variant: 'error' })
|
||||||
|
}
|
||||||
|
setSubmitting(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
function validateAmountInput(amountInput: string) {
|
||||||
|
let validAmountInput = '0'
|
||||||
|
|
||||||
|
if (!amountInput) {
|
||||||
|
setAmountError('Required field')
|
||||||
|
} else {
|
||||||
|
if (amountInput.indexOf('.') > -1) {
|
||||||
|
setAmountError('Amount must be an integer')
|
||||||
|
} else {
|
||||||
|
const amount = new BigNumber(amountInput)
|
||||||
|
|
||||||
|
if (amount.isNaN()) {
|
||||||
|
setAmountError('Amount must contain only digits')
|
||||||
|
} else if (amount.isLessThanOrEqualTo(0)) {
|
||||||
|
setAmountError('Amount must be greater than 0')
|
||||||
|
} else {
|
||||||
|
setAmountError('')
|
||||||
|
validAmountInput = amountInput
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setAmountInput(validAmountInput)
|
||||||
|
}
|
||||||
|
|
||||||
|
function validateDepthInput(depthInput: string) {
|
||||||
|
let validDepthInput = '0'
|
||||||
|
|
||||||
|
if (!depthInput) {
|
||||||
|
setDepthError('Required field')
|
||||||
|
} else {
|
||||||
|
const depth = new BigNumber(depthInput)
|
||||||
|
|
||||||
|
if (!depth.isInteger()) {
|
||||||
|
setDepthError('Depth must be an integer')
|
||||||
|
} else if (depth.isLessThan(17)) {
|
||||||
|
setDepthError('Minimal depth is 17')
|
||||||
|
} else if (depth.isGreaterThan(255)) {
|
||||||
|
setDepthError('Depth has to be at most 255')
|
||||||
|
} else {
|
||||||
|
setDepthError('')
|
||||||
|
validDepthInput = depthInput
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setDepthInput(validDepthInput)
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderStampVolumesInfo() {
|
||||||
|
const depth = parseInt(depthInput, 10)
|
||||||
|
|
||||||
|
if (depthError || isNaN(depth) || depth < 17 || depth > 255) {
|
||||||
|
return '-'
|
||||||
|
}
|
||||||
|
|
||||||
|
const theoreticalMaximumVolume = getHumanReadableFileSize(Utils.getStampMaximumCapacityBytes(depth))
|
||||||
|
const effectiveVolume = getHumanReadableFileSize(Utils.getStampEffectiveBytes(depth))
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Grid item container alignItems="center" className={classes.stampVolumeWrapper}>
|
||||||
|
<Typography>
|
||||||
|
Theoretical: ~{theoreticalMaximumVolume} / Effective: ~{effectiveVolume}
|
||||||
|
</Typography>
|
||||||
|
<IconButton
|
||||||
|
onClick={() =>
|
||||||
|
window.open(
|
||||||
|
'https://docs.ethswarm.org/docs/learn/technology/contracts/postage-stamp/#effective-utilisation-table',
|
||||||
|
'_blank',
|
||||||
|
'noopener,noreferrer',
|
||||||
|
)
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<Info />
|
||||||
|
</IconButton>
|
||||||
|
</Grid>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Box mb={4}>
|
||||||
|
<Typography>
|
||||||
|
To upload data to Swarm network, you will need to purchase a postage stamp. If you're not familiar with
|
||||||
|
this, please read{' '}
|
||||||
|
<a
|
||||||
|
href="https://medium.com/ethereum-swarm/how-to-upload-data-to-the-swarm-network-c0766c3ae381"
|
||||||
|
target="_blank"
|
||||||
|
rel="noreferrer"
|
||||||
|
>
|
||||||
|
this guide
|
||||||
|
</a>
|
||||||
|
.
|
||||||
|
</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<SwarmTextInput name="depth" label="Depth" onChange={event => validateDepthInput(event.target.value)} />
|
||||||
|
<Box mt={0.25} sx={{ bgcolor: '#f6f6f6' }} p={2}>
|
||||||
|
<Grid container justifyContent="space-between" alignItems="center">
|
||||||
|
<Typography>Corresponding file size</Typography>
|
||||||
|
{renderStampVolumesInfo()}
|
||||||
|
</Grid>
|
||||||
|
</Box>
|
||||||
|
{depthError && <Typography>{depthError}</Typography>}
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<SwarmTextInput name="amount" label="Amount" onChange={event => validateAmountInput(event.target.value)} />
|
||||||
|
<Box mt={0.25} sx={{ bgcolor: '#f6f6f6' }} p={2}>
|
||||||
|
<Grid container justifyContent="space-between">
|
||||||
|
<Typography>Corresponding TTL (Time to live)</Typography>
|
||||||
|
<Typography>{!amountError && amountInput ? getTtl(Number.parseInt(amountInput, 10)) : '-'}</Typography>
|
||||||
|
</Grid>
|
||||||
|
</Box>
|
||||||
|
{amountError && <Typography>{amountError}</Typography>}
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<SwarmTextInput name="label" label="Label" optional onChange={event => setLabelInput(event.target.value)} />
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<SwarmSelect
|
||||||
|
label="Immutable"
|
||||||
|
defaultValue="No"
|
||||||
|
onChange={event => setImmutable(event.target.value === 'Yes')}
|
||||||
|
options={[
|
||||||
|
{ value: 'Yes', label: 'Yes' },
|
||||||
|
{ value: 'No', label: 'No' },
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
<Box mt={0.25} sx={{ bgcolor: '#f6f6f6' }} p={2}>
|
||||||
|
<Grid container justifyContent="space-between">
|
||||||
|
{immutable && (
|
||||||
|
<Typography>
|
||||||
|
Once an immutable stamp is maxed out, it disallows further content uploads, thereby safeguarding your
|
||||||
|
previously uploaded content from unintentional overwriting.
|
||||||
|
</Typography>
|
||||||
|
)}
|
||||||
|
{!immutable && (
|
||||||
|
<Typography>
|
||||||
|
When a mutable stamp reaches full capacity, it still permits new content uploads. However, this comes
|
||||||
|
with the caveat of overwriting previously uploaded content associated with the same stamp.
|
||||||
|
</Typography>
|
||||||
|
)}
|
||||||
|
</Grid>
|
||||||
|
</Box>
|
||||||
|
</Box>
|
||||||
|
<Box mb={4} sx={{ bgcolor: '#fcf2e8' }} p={2}>
|
||||||
|
<Grid container justifyContent="space-between">
|
||||||
|
<Typography>Indicative Price</Typography>
|
||||||
|
<Typography>
|
||||||
|
{!amountError && !depthError && amountInput && depthInput
|
||||||
|
? getPrice(parseInt(depthInput, 10), BigInt(amountInput))
|
||||||
|
: '-'}
|
||||||
|
</Typography>
|
||||||
|
</Grid>
|
||||||
|
</Box>
|
||||||
|
|
||||||
|
<Grid container justifyContent="space-between" alignItems="center">
|
||||||
|
<Grid item>
|
||||||
|
<SwarmButton
|
||||||
|
disabled={submitting || Boolean(depthError) || Boolean(amountError) || !depthInput || !amountInput}
|
||||||
|
onClick={submit}
|
||||||
|
iconType={Check}
|
||||||
|
loading={submitting}
|
||||||
|
>
|
||||||
|
Buy New Stamp
|
||||||
|
</SwarmButton>
|
||||||
|
</Grid>
|
||||||
|
<Grid item>
|
||||||
|
<Link to={ROUTES.ACCOUNT_STAMPS_NEW_STANDARD} className={classes.link}>
|
||||||
|
Standard mode
|
||||||
|
</Link>
|
||||||
|
</Grid>
|
||||||
|
</Grid>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -1,193 +0,0 @@
|
|||||||
import { PostageBatchOptions } from '@ethersphere/bee-js'
|
|
||||||
import { Box, Grid, Typography } from '@material-ui/core'
|
|
||||||
import BigNumber from 'bignumber.js'
|
|
||||||
import { Form, Formik, FormikHelpers } from 'formik'
|
|
||||||
import { useSnackbar } from 'notistack'
|
|
||||||
import { ReactElement, useContext } from 'react'
|
|
||||||
import Check from 'remixicon-react/CheckLineIcon'
|
|
||||||
import { SwarmButton } from '../../components/SwarmButton'
|
|
||||||
import { SwarmTextInput } from '../../components/SwarmTextInput'
|
|
||||||
import { Context as BeeContext } from '../../providers/Bee'
|
|
||||||
import { Context as SettingsContext } from '../../providers/Settings'
|
|
||||||
import { Context as StampsContext } from '../../providers/Stamps'
|
|
||||||
import {
|
|
||||||
calculateStampPrice,
|
|
||||||
convertAmountToSeconds,
|
|
||||||
convertDepthToBytes,
|
|
||||||
secondsToTimeString,
|
|
||||||
waitUntilStampExists,
|
|
||||||
} from '../../utils'
|
|
||||||
import { getHumanReadableFileSize } from '../../utils/file'
|
|
||||||
|
|
||||||
interface FormValues {
|
|
||||||
depth?: string
|
|
||||||
amount?: string
|
|
||||||
label?: string
|
|
||||||
}
|
|
||||||
type FormErrors = Partial<FormValues>
|
|
||||||
const initialFormValues: FormValues = {
|
|
||||||
depth: '',
|
|
||||||
amount: '',
|
|
||||||
label: '',
|
|
||||||
}
|
|
||||||
|
|
||||||
interface Props {
|
|
||||||
onFinished: () => void
|
|
||||||
}
|
|
||||||
|
|
||||||
export function PostageStampCreation({ onFinished }: Props): ReactElement {
|
|
||||||
const { chainState } = useContext(BeeContext)
|
|
||||||
const { refresh } = useContext(StampsContext)
|
|
||||||
const { beeDebugApi } = useContext(SettingsContext)
|
|
||||||
|
|
||||||
const { enqueueSnackbar } = useSnackbar()
|
|
||||||
|
|
||||||
function getFileSize(depth: number): string {
|
|
||||||
if (isNaN(depth) || depth < 17 || depth > 255) {
|
|
||||||
return '-'
|
|
||||||
}
|
|
||||||
|
|
||||||
return `~${getHumanReadableFileSize(convertDepthToBytes(depth))}`
|
|
||||||
}
|
|
||||||
|
|
||||||
function getTtl(amount: number): string {
|
|
||||||
const isCurrentPriceAvailable = chainState && chainState.currentPrice
|
|
||||||
|
|
||||||
if (amount <= 0 || !isCurrentPriceAvailable) {
|
|
||||||
return '-'
|
|
||||||
}
|
|
||||||
|
|
||||||
const pricePerBlock = Number.parseInt(chainState.currentPrice, 10)
|
|
||||||
|
|
||||||
return `${secondsToTimeString(
|
|
||||||
convertAmountToSeconds(amount, pricePerBlock),
|
|
||||||
)} (with price of ${pricePerBlock.toFixed(0)} per block)`
|
|
||||||
}
|
|
||||||
|
|
||||||
function getPrice(depth: number, amount: bigint): string {
|
|
||||||
const hasInvalidInput = amount <= 0 || isNaN(depth) || depth < 17 || depth > 255
|
|
||||||
|
|
||||||
if (hasInvalidInput) {
|
|
||||||
return '-'
|
|
||||||
}
|
|
||||||
|
|
||||||
const price = calculateStampPrice(depth, amount)
|
|
||||||
|
|
||||||
return `${price.toSignificantDigits()} xBZZ`
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<Box mb={4}>
|
|
||||||
<Typography>
|
|
||||||
To upload data to Swarm network, you will need to purchase a postage stamp. If you're not familiar with
|
|
||||||
this, please read{' '}
|
|
||||||
<a
|
|
||||||
href="https://medium.com/ethereum-swarm/how-to-upload-data-to-the-swarm-network-c0766c3ae381"
|
|
||||||
target="_blank"
|
|
||||||
rel="noreferrer"
|
|
||||||
>
|
|
||||||
this guide
|
|
||||||
</a>
|
|
||||||
.
|
|
||||||
</Typography>
|
|
||||||
</Box>
|
|
||||||
<Formik
|
|
||||||
initialValues={initialFormValues}
|
|
||||||
onSubmit={async (values: FormValues, actions: FormikHelpers<FormValues>) => {
|
|
||||||
try {
|
|
||||||
// This is really just a typeguard, the validation pretty much guarantees these will have the right values
|
|
||||||
if (!values.depth || !values.amount) return
|
|
||||||
|
|
||||||
if (!beeDebugApi) return
|
|
||||||
|
|
||||||
const amount = BigInt(values.amount)
|
|
||||||
const depth = Number.parseInt(values.depth)
|
|
||||||
const options: PostageBatchOptions = { waitForUsable: false, label: values.label || undefined }
|
|
||||||
const batchId = await beeDebugApi.createPostageBatch(amount.toString(), depth, options)
|
|
||||||
await waitUntilStampExists(batchId, beeDebugApi)
|
|
||||||
actions.resetForm()
|
|
||||||
await refresh()
|
|
||||||
onFinished()
|
|
||||||
} catch (e) {
|
|
||||||
console.error(e) // eslint-disable-line
|
|
||||||
enqueueSnackbar(`Error: ${(e as Error).message}`, { variant: 'error' })
|
|
||||||
actions.setSubmitting(false)
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
validate={(values: FormValues) => {
|
|
||||||
const errors: FormErrors = {}
|
|
||||||
|
|
||||||
// Depth
|
|
||||||
if (!values.depth) errors.depth = 'Required field'
|
|
||||||
else {
|
|
||||||
const depth = new BigNumber(values.depth)
|
|
||||||
|
|
||||||
if (!depth.isInteger()) errors.depth = 'Depth must be an integer'
|
|
||||||
else if (depth.isLessThan(17)) errors.depth = 'Minimal depth is 17'
|
|
||||||
else if (depth.isGreaterThan(255)) errors.depth = 'Depth has to be at most 255'
|
|
||||||
}
|
|
||||||
|
|
||||||
// Amount
|
|
||||||
if (!values.amount) errors.amount = 'Required field'
|
|
||||||
else {
|
|
||||||
const amount = new BigNumber(values.amount)
|
|
||||||
|
|
||||||
if (!amount.isInteger()) errors.amount = 'Amount must be an integer'
|
|
||||||
else if (amount.isLessThanOrEqualTo(0)) errors.amount = 'Amount must be greater than 0'
|
|
||||||
}
|
|
||||||
|
|
||||||
return errors
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{({ submitForm, isValid, isSubmitting, values, errors }) => (
|
|
||||||
<Form>
|
|
||||||
<Box mb={2}>
|
|
||||||
<SwarmTextInput name="depth" label="Depth" formik />
|
|
||||||
<Box mt={0.25} sx={{ bgcolor: '#f6f6f6' }} p={2}>
|
|
||||||
<Grid container justifyContent="space-between">
|
|
||||||
<Typography>Corresponding file size</Typography>
|
|
||||||
<Typography>
|
|
||||||
{!errors.depth && values.depth ? getFileSize(parseInt(values.depth, 10)) : '-'}
|
|
||||||
</Typography>
|
|
||||||
</Grid>
|
|
||||||
</Box>
|
|
||||||
</Box>
|
|
||||||
<Box mb={2}>
|
|
||||||
<SwarmTextInput name="amount" label="Amount" formik />
|
|
||||||
<Box mt={0.25} sx={{ bgcolor: '#f6f6f6' }} p={2}>
|
|
||||||
<Grid container justifyContent="space-between">
|
|
||||||
<Typography>Corresponding TTL (Time to live)</Typography>
|
|
||||||
<Typography>
|
|
||||||
{!errors.amount && values.amount ? getTtl(Number.parseInt(values.amount, 10)) : '-'}
|
|
||||||
</Typography>
|
|
||||||
</Grid>
|
|
||||||
</Box>
|
|
||||||
</Box>
|
|
||||||
<Box mb={2}>
|
|
||||||
<SwarmTextInput name="label" label="Label" optional formik />
|
|
||||||
</Box>
|
|
||||||
<Box mb={4} sx={{ bgcolor: '#fcf2e8' }} p={2}>
|
|
||||||
<Grid container justifyContent="space-between">
|
|
||||||
<Typography>Indicative Price</Typography>
|
|
||||||
<Typography>
|
|
||||||
{!errors.amount && !errors.depth && values.amount && values.depth
|
|
||||||
? getPrice(parseInt(values.depth, 10), BigInt(values.amount))
|
|
||||||
: '-'}
|
|
||||||
</Typography>
|
|
||||||
</Grid>
|
|
||||||
</Box>
|
|
||||||
<SwarmButton
|
|
||||||
disabled={isSubmitting || !isValid || !values.amount || !values.depth}
|
|
||||||
onClick={submitForm}
|
|
||||||
iconType={Check}
|
|
||||||
loading={isSubmitting}
|
|
||||||
>
|
|
||||||
Buy New Stamp
|
|
||||||
</SwarmButton>
|
|
||||||
</Form>
|
|
||||||
)}
|
|
||||||
</Formik>
|
|
||||||
</>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
@@ -23,7 +23,10 @@ export function PostageStampSelector({ onSelect, defaultValue }: Props): ReactEl
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<SwarmSelect
|
<SwarmSelect
|
||||||
options={(stamps || []).map(x => ({ label: x.batchID.slice(0, 8), value: x.batchID }))}
|
options={(stamps || []).map(x => ({
|
||||||
|
label: x.label ? x.batchID.slice(0, 8) + ' - ' + x.label : x.batchID.slice(0, 8),
|
||||||
|
value: x.batchID,
|
||||||
|
}))}
|
||||||
onChange={event => onChange(event.target.value as string)}
|
onChange={event => onChange(event.target.value as string)}
|
||||||
defaultValue={defaultValue}
|
defaultValue={defaultValue}
|
||||||
placeholder="Please select a postage stamp..."
|
placeholder="Please select a postage stamp..."
|
||||||
|
|||||||
@@ -0,0 +1,228 @@
|
|||||||
|
import { PostageBatchOptions, Utils } from '@ethersphere/bee-js'
|
||||||
|
import { Box, Button, Grid, Slider, Typography } from '@material-ui/core'
|
||||||
|
import { Theme, createStyles, makeStyles } from '@material-ui/core/styles'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
import { ReactElement, useContext, useState } from 'react'
|
||||||
|
import { Link } from 'react-router-dom'
|
||||||
|
import Check from 'remixicon-react/CheckLineIcon'
|
||||||
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
|
import { SwarmTextInput } from '../../components/SwarmTextInput'
|
||||||
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
|
import { Context as StampsContext } from '../../providers/Stamps'
|
||||||
|
import { ROUTES } from '../../routes'
|
||||||
|
import { calculateStampPrice, convertAmountToSeconds, secondsToTimeString, waitUntilStampExists } from '../../utils'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
onFinished: () => void
|
||||||
|
}
|
||||||
|
const useStyles = makeStyles((theme: Theme) =>
|
||||||
|
createStyles({
|
||||||
|
link: {
|
||||||
|
color: '#dd7700',
|
||||||
|
textDecoration: 'underline',
|
||||||
|
'&:hover': {
|
||||||
|
textDecoration: 'none',
|
||||||
|
|
||||||
|
// https://github.com/mui-org/material-ui/issues/22543
|
||||||
|
'@media (hover: none)': {
|
||||||
|
textDecoration: 'none',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
buttonSelected: {
|
||||||
|
color: 'white',
|
||||||
|
backgroundColor: theme.palette.primary.main,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
const marks = [
|
||||||
|
{ value: 1, label: '1 day' },
|
||||||
|
{ value: 365, label: '365 days' },
|
||||||
|
]
|
||||||
|
|
||||||
|
export function PostageStampStandardCreation({ onFinished }: Props): ReactElement {
|
||||||
|
const classes = useStyles()
|
||||||
|
const { refresh } = useContext(StampsContext)
|
||||||
|
const { beeApi } = useContext(SettingsContext)
|
||||||
|
|
||||||
|
const [depthInput, setDepthInput] = useState<number>(Utils.getDepthForCapacity(4))
|
||||||
|
const [amountInput, setAmountInput] = useState<string>(Utils.getAmountForTtl(30))
|
||||||
|
const [labelInput, setLabelInput] = useState('')
|
||||||
|
const [submitting, setSubmitting] = useState(false)
|
||||||
|
const [buttonValue, setButtonValue] = useState(4)
|
||||||
|
|
||||||
|
function sliderValueChange(_: unknown, newValue: number | number[]) {
|
||||||
|
if (typeof newValue !== 'number') {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const amountValue = Utils.getAmountForTtl(newValue)
|
||||||
|
setAmountInput(amountValue)
|
||||||
|
}
|
||||||
|
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
|
||||||
|
function getTtl(amount: string): string {
|
||||||
|
const pricePerBlock = 24000
|
||||||
|
|
||||||
|
return `${secondsToTimeString(
|
||||||
|
convertAmountToSeconds(parseInt(amount, 10), pricePerBlock),
|
||||||
|
)} (with price of ${pricePerBlock.toFixed(0)} PLUR per block)`
|
||||||
|
}
|
||||||
|
|
||||||
|
function getPrice(depth: number, amount: bigint): string {
|
||||||
|
const price = calculateStampPrice(depth, amount)
|
||||||
|
|
||||||
|
return `${price.toSignificantDigits()} xBZZ`
|
||||||
|
}
|
||||||
|
|
||||||
|
async function submit() {
|
||||||
|
try {
|
||||||
|
// This is really just a typeguard, the validation pretty much guarantees these will have the right values
|
||||||
|
if (!depthInput || !amountInput) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!beeApi) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
setSubmitting(true)
|
||||||
|
const amount = BigInt(amountInput)
|
||||||
|
const depth = depthInput
|
||||||
|
const options: PostageBatchOptions = {
|
||||||
|
waitForUsable: false,
|
||||||
|
label: labelInput || undefined,
|
||||||
|
immutableFlag: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
const batchId = await beeApi.createPostageBatch(amount.toString(), depth, options)
|
||||||
|
await waitUntilStampExists(batchId, beeApi)
|
||||||
|
await refresh()
|
||||||
|
onFinished()
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e) // eslint-disable-line
|
||||||
|
enqueueSnackbar(`Error: ${(e as Error).message}`, { variant: 'error' })
|
||||||
|
}
|
||||||
|
setSubmitting(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleBatchSize(gigabytes: number) {
|
||||||
|
setButtonValue(gigabytes)
|
||||||
|
const capacity = Utils.getDepthForCapacity(gigabytes)
|
||||||
|
setDepthInput(capacity)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Box mb={4}>
|
||||||
|
<Typography>
|
||||||
|
A postage stamp batch containes postage stamps that will give you the right to upload data to the Swarm
|
||||||
|
network. If you're not familiar with this, please read
|
||||||
|
<a
|
||||||
|
href="https://medium.com/ethereum-swarm/how-to-upload-data-to-the-swarm-network-c0766c3ae381"
|
||||||
|
target="_blank"
|
||||||
|
rel="noreferrer"
|
||||||
|
>
|
||||||
|
this guide
|
||||||
|
</a>
|
||||||
|
.
|
||||||
|
</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={1}>
|
||||||
|
<Typography variant="h2">Batch name</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<SwarmTextInput name="depth" label="Label" onChange={e => setLabelInput(e.target.value)} />
|
||||||
|
</Box>
|
||||||
|
<Box mb={1}>
|
||||||
|
<Typography variant="h2">Batch size</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<Grid container justifyContent="space-between" spacing={2}>
|
||||||
|
<Grid item xs={4}>
|
||||||
|
<Button
|
||||||
|
variant="contained"
|
||||||
|
fullWidth
|
||||||
|
onClick={() => handleBatchSize(4)}
|
||||||
|
className={buttonValue === 4 ? classes.buttonSelected : ''}
|
||||||
|
>
|
||||||
|
4 GB
|
||||||
|
</Button>
|
||||||
|
</Grid>
|
||||||
|
<Grid item xs={4}>
|
||||||
|
<Button
|
||||||
|
variant="contained"
|
||||||
|
fullWidth
|
||||||
|
onClick={() => handleBatchSize(32)}
|
||||||
|
className={buttonValue === 32 ? classes.buttonSelected : ''}
|
||||||
|
>
|
||||||
|
32 GB
|
||||||
|
</Button>
|
||||||
|
</Grid>
|
||||||
|
<Grid item xs={4}>
|
||||||
|
<Button
|
||||||
|
variant="contained"
|
||||||
|
fullWidth
|
||||||
|
onClick={() => handleBatchSize(256)}
|
||||||
|
className={buttonValue === 256 ? classes.buttonSelected : ''}
|
||||||
|
>
|
||||||
|
256 GB
|
||||||
|
</Button>
|
||||||
|
</Grid>
|
||||||
|
</Grid>
|
||||||
|
</Box>
|
||||||
|
<Box mb={1}>
|
||||||
|
<Typography variant="h2">Data persistence</Typography>
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<Slider
|
||||||
|
aria-label="Volume"
|
||||||
|
min={1}
|
||||||
|
max={365}
|
||||||
|
step={1}
|
||||||
|
marks={marks}
|
||||||
|
valueLabelDisplay="auto"
|
||||||
|
defaultValue={30}
|
||||||
|
onChange={sliderValueChange}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
<Box mb={2}>
|
||||||
|
<Box mt={0.25} sx={{ bgcolor: '#f6f6f6' }} p={2}>
|
||||||
|
<Grid container justifyContent="space-between">
|
||||||
|
<Typography>Corresponding TTL (Time to live)</Typography>
|
||||||
|
<Typography>{amountInput ? getTtl(amountInput) : '-'}</Typography>
|
||||||
|
</Grid>
|
||||||
|
</Box>
|
||||||
|
<Box display="flex" justifyContent={'right'} mt={0.5}>
|
||||||
|
<Typography style={{ fontSize: '10px', color: 'rgba(0, 0, 0, 0.26)' }}>
|
||||||
|
Current price of 24000 PLUR per block
|
||||||
|
</Typography>
|
||||||
|
</Box>
|
||||||
|
</Box>
|
||||||
|
<Box mb={4} sx={{ bgcolor: '#fcf2e8' }} p={2}>
|
||||||
|
<Grid container justifyContent="space-between">
|
||||||
|
<Typography>Indicative Price</Typography>
|
||||||
|
<Typography>{getPrice(depthInput, BigInt(amountInput))}</Typography>
|
||||||
|
</Grid>
|
||||||
|
</Box>
|
||||||
|
<Grid container justifyContent="space-between" alignItems="center">
|
||||||
|
<Grid item>
|
||||||
|
<SwarmButton
|
||||||
|
disabled={submitting || !depthInput || !amountInput}
|
||||||
|
onClick={submit}
|
||||||
|
iconType={Check}
|
||||||
|
loading={submitting}
|
||||||
|
>
|
||||||
|
Buy New Stamp
|
||||||
|
</SwarmButton>
|
||||||
|
</Grid>
|
||||||
|
<Grid item>
|
||||||
|
<Link to={ROUTES.ACCOUNT_STAMPS_NEW_ADVANCED} className={classes.link}>
|
||||||
|
Advanced mode
|
||||||
|
</Link>
|
||||||
|
</Grid>
|
||||||
|
</Grid>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -1,8 +1,13 @@
|
|||||||
import type { ReactElement } from 'react'
|
import { ReactElement, useContext } from 'react'
|
||||||
|
import TimerFlashFill from 'remixicon-react/TimerFlashFillIcon'
|
||||||
|
import TimerFlashLine from 'remixicon-react/TimerFlashLineIcon'
|
||||||
import ExpandableElement from '../../components/ExpandableElement'
|
import ExpandableElement from '../../components/ExpandableElement'
|
||||||
import ExpandableList from '../../components/ExpandableList'
|
import ExpandableList from '../../components/ExpandableList'
|
||||||
import ExpandableListItem from '../../components/ExpandableListItem'
|
import ExpandableListItem from '../../components/ExpandableListItem'
|
||||||
|
import ExpandableListItemActions from '../../components/ExpandableListItemActions'
|
||||||
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
import ExpandableListItemKey from '../../components/ExpandableListItemKey'
|
||||||
|
import StampExtensionModal from '../../components/StampExtensionModal'
|
||||||
|
import { Context } from '../../providers/Settings'
|
||||||
import { EnrichedPostageBatch } from '../../providers/Stamps'
|
import { EnrichedPostageBatch } from '../../providers/Stamps'
|
||||||
import { secondsToTimeString } from '../../utils'
|
import { secondsToTimeString } from '../../utils'
|
||||||
import { getHumanReadableFileSize } from '../../utils/file'
|
import { getHumanReadableFileSize } from '../../utils/file'
|
||||||
@@ -13,7 +18,11 @@ interface Props {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function StampsTable({ postageStamps }: Props): ReactElement | null {
|
function StampsTable({ postageStamps }: Props): ReactElement | null {
|
||||||
if (postageStamps === null) return null
|
const { beeApi } = useContext(Context)
|
||||||
|
|
||||||
|
if (!postageStamps || !beeApi) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ExpandableList label="Postage Stamps" defaultOpen>
|
<ExpandableList label="Postage Stamps" defaultOpen>
|
||||||
@@ -38,7 +47,22 @@ function StampsTable({ postageStamps }: Props): ReactElement | null {
|
|||||||
<ExpandableListItem label="Label" value={stamp.label} />
|
<ExpandableListItem label="Label" value={stamp.label} />
|
||||||
<ExpandableListItem label="Usable" value={stamp.usable ? 'yes' : 'no'} />
|
<ExpandableListItem label="Usable" value={stamp.usable ? 'yes' : 'no'} />
|
||||||
<ExpandableListItem label="Exists" value={stamp.exists ? 'yes' : 'no'} />
|
<ExpandableListItem label="Exists" value={stamp.exists ? 'yes' : 'no'} />
|
||||||
|
<ExpandableListItem label="Immutable" value={stamp.immutableFlag ? 'yes' : 'no'} />
|
||||||
<ExpandableListItem label="Purchase Block Number" value={stamp.blockNumber} />
|
<ExpandableListItem label="Purchase Block Number" value={stamp.blockNumber} />
|
||||||
|
<ExpandableListItemActions>
|
||||||
|
<StampExtensionModal
|
||||||
|
type="Topup"
|
||||||
|
icon={<TimerFlashFill size="1rem" />}
|
||||||
|
bee={beeApi}
|
||||||
|
stamp={stamp.batchID}
|
||||||
|
/>
|
||||||
|
<StampExtensionModal
|
||||||
|
type="Dilute"
|
||||||
|
icon={<TimerFlashLine size="1rem" />}
|
||||||
|
bee={beeApi}
|
||||||
|
stamp={stamp.batchID}
|
||||||
|
/>
|
||||||
|
</ExpandableListItemActions>
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -44,7 +44,7 @@ export default function Stamp(): ReactElement {
|
|||||||
if (status.all === CheckState.ERROR) return <TroubleshootConnectionCard />
|
if (status.all === CheckState.ERROR) return <TroubleshootConnectionCard />
|
||||||
|
|
||||||
function navigateToNewStamp() {
|
function navigateToNewStamp() {
|
||||||
navigate(ROUTES.ACCOUNT_STAMPS_NEW)
|
navigate(ROUTES.ACCOUNT_STAMPS_NEW_STANDARD)
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -1,19 +1,20 @@
|
|||||||
import { useContext } from 'react'
|
|
||||||
import DepositModal from '../../../containers/DepositModal'
|
|
||||||
import type { ReactElement, ReactNode } from 'react'
|
import type { ReactElement, ReactNode } from 'react'
|
||||||
|
import { useContext } from 'react'
|
||||||
import ExpandableList from '../../../components/ExpandableList'
|
import ExpandableList from '../../../components/ExpandableList'
|
||||||
import ExpandableListItemKey from '../../../components/ExpandableListItemKey'
|
|
||||||
import ExpandableListItemActions from '../../../components/ExpandableListItemActions'
|
import ExpandableListItemActions from '../../../components/ExpandableListItemActions'
|
||||||
|
import ExpandableListItemKey from '../../../components/ExpandableListItemKey'
|
||||||
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
||||||
import StatusIcon from '../../../components/StatusIcon'
|
import StatusIcon from '../../../components/StatusIcon'
|
||||||
|
import DepositModal from '../../../containers/DepositModal'
|
||||||
import { CheckState, Context } from '../../../providers/Bee'
|
import { CheckState, Context } from '../../../providers/Bee'
|
||||||
|
|
||||||
const ChequebookDeployFund = (): ReactElement | null => {
|
const ChequebookDeployFund = (): ReactElement | null => {
|
||||||
const { status, isLoading, chequebookAddress } = useContext(Context)
|
const { status, isLoading, chequebookAddress } = useContext(Context)
|
||||||
const { checkState, isEnabled } = status.chequebook
|
const { checkState, isEnabled } = status.chequebook
|
||||||
const { checkState: debugApiCheckState } = status.debugApiConnection
|
|
||||||
|
|
||||||
if (!isEnabled || debugApiCheckState === CheckState.ERROR) return null
|
if (!isEnabled) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
let text: ReactNode
|
let text: ReactNode
|
||||||
|
|
||||||
|
|||||||
@@ -1,88 +0,0 @@
|
|||||||
import MuiAlert from '@material-ui/lab/Alert'
|
|
||||||
import { ReactElement, useContext } from 'react'
|
|
||||||
import CodeBlockTabs from '../../../components/CodeBlockTabs'
|
|
||||||
import ExpandableList from '../../../components/ExpandableList'
|
|
||||||
import ExpandableListItem from '../../../components/ExpandableListItem'
|
|
||||||
import ExpandableListItemInput from '../../../components/ExpandableListItemInput'
|
|
||||||
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
|
||||||
import StatusIcon from '../../../components/StatusIcon'
|
|
||||||
import { CheckState, Context } from '../../../providers/Bee'
|
|
||||||
import { Context as SettingsContext } from '../../../providers/Settings'
|
|
||||||
|
|
||||||
export default function NodeConnectionCheck(): ReactElement | null {
|
|
||||||
const { status, isLoading } = useContext(Context)
|
|
||||||
const { setDebugApiUrl, apiDebugUrl, isDesktop } = useContext(SettingsContext)
|
|
||||||
const { checkState, isEnabled } = status.debugApiConnection
|
|
||||||
|
|
||||||
if (!isEnabled) return null
|
|
||||||
|
|
||||||
return (
|
|
||||||
<ExpandableList
|
|
||||||
label={
|
|
||||||
<>
|
|
||||||
<StatusIcon checkState={checkState} isLoading={isLoading} /> Connection to Bee Debug API
|
|
||||||
</>
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<ExpandableListItemNote>
|
|
||||||
{checkState === CheckState.OK
|
|
||||||
? 'The connection to the Bee node debug API has been successful'
|
|
||||||
: 'Could not connect to your Bee node debug API.'}
|
|
||||||
</ExpandableListItemNote>
|
|
||||||
<ExpandableListItemInput label="Bee Debug API" value={apiDebugUrl} onConfirm={setDebugApiUrl} />
|
|
||||||
|
|
||||||
{checkState === CheckState.ERROR && !isDesktop && (
|
|
||||||
<ExpandableList level={1} label="Troubleshoot">
|
|
||||||
<ExpandableListItem
|
|
||||||
label={
|
|
||||||
<ol>
|
|
||||||
<li>Check the status of your node by running the below command to see if your node is running.</li>
|
|
||||||
<CodeBlockTabs showLineNumbers linux={`sudo systemctl status bee`} mac={`brew services list`} />
|
|
||||||
<li>
|
|
||||||
If your node is running, check your firewall settings to make sure that port 1635 (or your custom
|
|
||||||
specified port) is bound to localhost. If your node is not running try executing the below command to
|
|
||||||
start your bee node
|
|
||||||
</li>
|
|
||||||
<MuiAlert
|
|
||||||
style={{ marginTop: '10px', marginBottom: '10px' }}
|
|
||||||
elevation={6}
|
|
||||||
variant="filled"
|
|
||||||
severity="error"
|
|
||||||
>
|
|
||||||
Your debug node API should never be completely open to the internet. If you want to connect remotely,
|
|
||||||
make sure your firewall settings are set to only allow specific trusted IP addresses and block all
|
|
||||||
other ports. A simple google search for "what is my ip" will show you your computers public
|
|
||||||
IP address to allow.
|
|
||||||
</MuiAlert>
|
|
||||||
<CodeBlockTabs
|
|
||||||
showLineNumbers
|
|
||||||
linux={`sudo systemctl start bee`}
|
|
||||||
mac={`brew services start swarm-bee`}
|
|
||||||
/>
|
|
||||||
<li>Run the commands to validate your node is running and see the log output.</li>
|
|
||||||
<CodeBlockTabs
|
|
||||||
showLineNumbers
|
|
||||||
linux={`sudo systemctl status bee \njournalctl --lines=100 --follow --unit bee`}
|
|
||||||
mac={`brew services list \ntail -f /usr/local/var/log/swarm-bee/bee.log`}
|
|
||||||
/>
|
|
||||||
<li>
|
|
||||||
Lastly, check your nodes configuration settings to validate the debug API is enabled and the Cross
|
|
||||||
Origin Resource Sharing (CORS) setting is configured to allow your host. Config parameter{' '}
|
|
||||||
<strong>debug-api-enable</strong> must be set to <strong>true</strong> and{' '}
|
|
||||||
<strong>cors-allowed-origins</strong> must be set to your host domain or IP (you can also use the
|
|
||||||
wildcard <code>{"cors-allowed-origins: ['*']"}</code>). If edits are made to the configuration run the
|
|
||||||
restart command below for changes to take effect.
|
|
||||||
</li>
|
|
||||||
<CodeBlockTabs
|
|
||||||
showLineNumbers
|
|
||||||
linux={`sudo vi /etc/bee/bee.yaml\nsudo systemctl restart bee`}
|
|
||||||
mac={`sudo vi /usr/local/etc/swarm-bee/bee.yaml \nbrew services restart swarm-bee`}
|
|
||||||
/>
|
|
||||||
</ol>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</ExpandableList>
|
|
||||||
)}
|
|
||||||
</ExpandableList>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
import { ReactElement, useContext } from 'react'
|
|
||||||
import ExpandableList from '../../../components/ExpandableList'
|
|
||||||
import ExpandableListItemKey from '../../../components/ExpandableListItemKey'
|
|
||||||
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
|
||||||
import StatusIcon from '../../../components/StatusIcon'
|
|
||||||
import { CheckState, Context } from '../../../providers/Bee'
|
|
||||||
|
|
||||||
export default function EthereumConnectionCheck(): ReactElement | null {
|
|
||||||
const { status, isLoading, nodeAddresses } = useContext(Context)
|
|
||||||
const { checkState, isEnabled } = status.blockchainConnection
|
|
||||||
|
|
||||||
if (!isEnabled) return null
|
|
||||||
|
|
||||||
return (
|
|
||||||
<ExpandableList
|
|
||||||
label={
|
|
||||||
<>
|
|
||||||
<StatusIcon checkState={checkState} isLoading={isLoading} /> Connection to Blockchain
|
|
||||||
</>
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<ExpandableListItemNote>
|
|
||||||
{checkState === CheckState.OK ? (
|
|
||||||
'Your node is connected to the xDai blockchain'
|
|
||||||
) : (
|
|
||||||
<>
|
|
||||||
Your Bee node must have access to the xDai blockchain, so that it can interact and deploy your chequebook
|
|
||||||
contract. You can run{' '}
|
|
||||||
<a href="https://www.xdaichain.com/" rel="noreferrer" target="_blank">
|
|
||||||
your own xDai node
|
|
||||||
</a>
|
|
||||||
, or use a provider instead - we recommend{' '}
|
|
||||||
<a href="https://getblock.io/" rel="noreferrer" target="_blank">
|
|
||||||
Getblock
|
|
||||||
</a>
|
|
||||||
. By default, Bee expects a local node at http://localhost:8545. To use a provider instead, simply change
|
|
||||||
the <strong>swap-endpoint</strong> in your configuration file.
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</ExpandableListItemNote>
|
|
||||||
{nodeAddresses?.ethereum && <ExpandableListItemKey label="Ethereum Address" value={nodeAddresses?.ethereum} />}
|
|
||||||
</ExpandableList>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
@@ -1,16 +1,17 @@
|
|||||||
import { ReactElement, ReactNode, useContext } from 'react'
|
import { ReactElement, ReactNode, useContext } from 'react'
|
||||||
import ExpandableList from '../../../components/ExpandableList'
|
import ExpandableList from '../../../components/ExpandableList'
|
||||||
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
||||||
import TopologyStats from '../../../components/TopologyStats'
|
|
||||||
import StatusIcon from '../../../components/StatusIcon'
|
import StatusIcon from '../../../components/StatusIcon'
|
||||||
|
import TopologyStats from '../../../components/TopologyStats'
|
||||||
import { CheckState, Context } from '../../../providers/Bee'
|
import { CheckState, Context } from '../../../providers/Bee'
|
||||||
|
|
||||||
export default function PeerConnection(): ReactElement | null {
|
export default function PeerConnection(): ReactElement | null {
|
||||||
const { status, isLoading, topology } = useContext(Context)
|
const { status, isLoading, topology } = useContext(Context)
|
||||||
const { isEnabled, checkState } = status.topology
|
const { isEnabled, checkState } = status.topology
|
||||||
const { checkState: debugApiCheckState } = status.debugApiConnection
|
|
||||||
|
|
||||||
if (!isEnabled || debugApiCheckState === CheckState.ERROR) return null
|
if (!isEnabled) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
let text: ReactNode
|
let text: ReactNode
|
||||||
switch (checkState) {
|
switch (checkState) {
|
||||||
|
|||||||
@@ -1,49 +0,0 @@
|
|||||||
import { ReactElement, useContext } from 'react'
|
|
||||||
import CodeBlockTabs from '../../../components/CodeBlockTabs'
|
|
||||||
import ExpandableList from '../../../components/ExpandableList'
|
|
||||||
import ExpandableListItem from '../../../components/ExpandableListItem'
|
|
||||||
import ExpandableListItemNote from '../../../components/ExpandableListItemNote'
|
|
||||||
import StatusIcon from '../../../components/StatusIcon'
|
|
||||||
import { CheckState, Context } from '../../../providers/Bee'
|
|
||||||
|
|
||||||
export default function VersionCheck(): ReactElement | null {
|
|
||||||
const { status, isLoading, latestUserVersion, latestPublishedVersion, latestBeeVersionUrl } = useContext(Context)
|
|
||||||
const { isEnabled, checkState } = status.version
|
|
||||||
|
|
||||||
if (!isEnabled) return null
|
|
||||||
|
|
||||||
return (
|
|
||||||
<ExpandableList
|
|
||||||
label={
|
|
||||||
<>
|
|
||||||
<StatusIcon checkState={checkState} isLoading={isLoading} /> Bee Version
|
|
||||||
</>
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<ExpandableListItemNote>
|
|
||||||
{checkState === CheckState.OK ? (
|
|
||||||
'You are running the latest version of Bee.'
|
|
||||||
) : (
|
|
||||||
<>
|
|
||||||
Your Bee version is out of date. Please update to the{' '}
|
|
||||||
<a href={latestBeeVersionUrl} rel="noreferrer" target="_blank">
|
|
||||||
latest
|
|
||||||
</a>{' '}
|
|
||||||
before continuing. Rerun the installation script below to upgrade. For more information please see the{' '}
|
|
||||||
<a href="https://docs.ethswarm.org/docs/installation/manual#upgrading-bee" rel="noreferrer" target="_blank">
|
|
||||||
Docs
|
|
||||||
</a>
|
|
||||||
.
|
|
||||||
<CodeBlockTabs
|
|
||||||
showLineNumbers
|
|
||||||
linux={`bee version\nwget https://github.com/ethersphere/bee/releases/download/${latestPublishedVersion}/bee_${latestPublishedVersion}_amd64.deb\nsudo dpkg -i bee_${latestPublishedVersion}_amd64.deb`}
|
|
||||||
mac={`bee version\nbrew tap ethersphere/tap\nbrew install swarm-bee\nbrew services start swarm-bee`}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</ExpandableListItemNote>
|
|
||||||
<ExpandableListItem label="Your Version" value={latestUserVersion || '-'} />
|
|
||||||
<ExpandableListItem label="Latest Version" value={latestPublishedVersion || '-'} />
|
|
||||||
</ExpandableList>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
@@ -1,12 +1,8 @@
|
|||||||
import { Context } from '../../providers/Settings'
|
|
||||||
import { ReactElement, useContext } from 'react'
|
import { ReactElement, useContext } from 'react'
|
||||||
|
import { Context } from '../../providers/Settings'
|
||||||
import DebugConnectionCheck from './SetupSteps/DebugConnectionCheck'
|
import ChequebookDeployFund from './SetupSteps/ChequebookDeployFund'
|
||||||
import DesktopConnection from './SetupSteps/DesktopConnectionCheck'
|
import DesktopConnection from './SetupSteps/DesktopConnectionCheck'
|
||||||
import NodeConnectionCheck from './SetupSteps/NodeConnectionCheck'
|
import NodeConnectionCheck from './SetupSteps/NodeConnectionCheck'
|
||||||
import VersionCheck from './SetupSteps/VersionCheck'
|
|
||||||
import EthereumConnectionCheck from './SetupSteps/EthereumConnectionCheck'
|
|
||||||
import ChequebookDeployFund from './SetupSteps/ChequebookDeployFund'
|
|
||||||
import PeerConnection from './SetupSteps/PeerConnection'
|
import PeerConnection from './SetupSteps/PeerConnection'
|
||||||
|
|
||||||
export default function NodeSetupWorkflow(): ReactElement {
|
export default function NodeSetupWorkflow(): ReactElement {
|
||||||
@@ -16,9 +12,6 @@ export default function NodeSetupWorkflow(): ReactElement {
|
|||||||
<div>
|
<div>
|
||||||
{isDesktop && <DesktopConnection />}
|
{isDesktop && <DesktopConnection />}
|
||||||
<NodeConnectionCheck />
|
<NodeConnectionCheck />
|
||||||
<DebugConnectionCheck />
|
|
||||||
{!isDesktop && <VersionCheck />}
|
|
||||||
<EthereumConnectionCheck />
|
|
||||||
<ChequebookDeployFund />
|
<ChequebookDeployFund />
|
||||||
<PeerConnection />
|
<PeerConnection />
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -13,13 +13,14 @@ import { Loading } from '../../components/Loading'
|
|||||||
import { SwarmButton } from '../../components/SwarmButton'
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
import { SwarmDivider } from '../../components/SwarmDivider'
|
import { SwarmDivider } from '../../components/SwarmDivider'
|
||||||
import { SwarmTextInput } from '../../components/SwarmTextInput'
|
import { SwarmTextInput } from '../../components/SwarmTextInput'
|
||||||
import { BzzToken, BZZ_DECIMAL_PLACES } from '../../models/BzzToken'
|
import { BZZ_DECIMAL_PLACES, BzzToken } from '../../models/BzzToken'
|
||||||
import { DaiToken } from '../../models/DaiToken'
|
import { DaiToken } from '../../models/DaiToken'
|
||||||
import { Context as BeeContext } from '../../providers/Bee'
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
import { Context as SettingsContext } from '../../providers/Settings'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
import { Context as BalanceProvider } from '../../providers/WalletBalance'
|
import { Context as BalanceProvider } from '../../providers/WalletBalance'
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
import { sleepMs } from '../../utils'
|
import { sleepMs } from '../../utils'
|
||||||
|
import { SwapError, isSwapError, wrapWithSwapError } from '../../utils/SwapError'
|
||||||
import {
|
import {
|
||||||
getBzzPriceAsDai,
|
getBzzPriceAsDai,
|
||||||
getDesktopConfiguration,
|
getDesktopConfiguration,
|
||||||
@@ -28,7 +29,6 @@ import {
|
|||||||
upgradeToLightNode,
|
upgradeToLightNode,
|
||||||
} from '../../utils/desktop'
|
} from '../../utils/desktop'
|
||||||
import { Rpc } from '../../utils/rpc'
|
import { Rpc } from '../../utils/rpc'
|
||||||
import { isSwapError, SwapError, wrapWithSwapError } from '../../utils/SwapError'
|
|
||||||
import { TopUpProgressIndicator } from './TopUpProgressIndicator'
|
import { TopUpProgressIndicator } from './TopUpProgressIndicator'
|
||||||
|
|
||||||
const MINIMUM_XDAI = '0.1'
|
const MINIMUM_XDAI = '0.1'
|
||||||
@@ -162,12 +162,12 @@ export function Swap({ header }: Props): ReactElement {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!desktopConfiguration['swap-endpoint']) {
|
if (!desktopConfiguration['blockchain-rpc-endpoint']) {
|
||||||
throw new SwapError('Swap endpoint is not configured in Swarm Desktop')
|
throw new SwapError('Blockchain RPC endpoint is not configured in Swarm Desktop')
|
||||||
}
|
}
|
||||||
await wrapWithSwapError(
|
await wrapWithSwapError(
|
||||||
Rpc.getNetworkChainId(desktopConfiguration['swap-endpoint']),
|
Rpc.getNetworkChainId(desktopConfiguration['blockchain-rpc-endpoint']),
|
||||||
`Swap endpoint not reachable at ${desktopConfiguration['swap-endpoint']}`,
|
`Blockchain RPC endpoint not reachable at ${desktopConfiguration['blockchain-rpc-endpoint']}`,
|
||||||
)
|
)
|
||||||
await wrapWithSwapError(sendSwapRequest(daiToSwap), GENERIC_SWAP_FAILED_ERROR_MESSAGE)
|
await wrapWithSwapError(sendSwapRequest(daiToSwap), GENERIC_SWAP_FAILED_ERROR_MESSAGE)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ import { HistoryHeader } from '../../components/HistoryHeader'
|
|||||||
import { Loading } from '../../components/Loading'
|
import { Loading } from '../../components/Loading'
|
||||||
import { SwarmButton } from '../../components/SwarmButton'
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
import TroubleshootConnectionCard from '../../components/TroubleshootConnectionCard'
|
||||||
import { CheckState, Context as BeeContext } from '../../providers/Bee'
|
import { Context as BeeContext, CheckState } from '../../providers/Bee'
|
||||||
import { Context as SettingsContext } from '../../providers/Settings'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
import { Context as BalanceProvider } from '../../providers/WalletBalance'
|
import { Context as BalanceProvider } from '../../providers/WalletBalance'
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
|
|||||||
+63
-122
@@ -2,26 +2,26 @@ import {
|
|||||||
BeeModes,
|
BeeModes,
|
||||||
ChainState,
|
ChainState,
|
||||||
ChequebookAddressResponse,
|
ChequebookAddressResponse,
|
||||||
Health,
|
|
||||||
LastChequesResponse,
|
LastChequesResponse,
|
||||||
NodeAddresses,
|
NodeAddresses,
|
||||||
NodeInfo,
|
NodeInfo,
|
||||||
Peer,
|
Peer,
|
||||||
Topology,
|
Topology,
|
||||||
} from '@ethersphere/bee-js'
|
} from '@ethersphere/bee-js'
|
||||||
import { createContext, ReactChild, ReactElement, useContext, useEffect, useState } from 'react'
|
import { ReactChild, ReactElement, createContext, useContext, useEffect, useState } from 'react'
|
||||||
import semver from 'semver'
|
|
||||||
import PackageJson from '../../package.json'
|
|
||||||
import { useLatestBeeRelease } from '../hooks/apiHooks'
|
import { useLatestBeeRelease } from '../hooks/apiHooks'
|
||||||
|
import { BzzToken } from '../models/BzzToken'
|
||||||
import { Token } from '../models/Token'
|
import { Token } from '../models/Token'
|
||||||
import type { Balance, ChequebookBalance, Settlements } from '../types'
|
import type { Balance, ChequebookBalance, Settlements } from '../types'
|
||||||
import { Context as SettingsContext } from './Settings'
|
import { Context as SettingsContext } from './Settings'
|
||||||
|
|
||||||
|
const LAUNCH_GRACE_PERIOD = 15_000
|
||||||
const REFRESH_WHEN_OK = 30_000
|
const REFRESH_WHEN_OK = 30_000
|
||||||
const REFRESH_WHEN_ERROR = 5_000
|
const REFRESH_WHEN_ERROR = 5_000
|
||||||
const TIMEOUT = 3_000
|
const TIMEOUT = 3_000
|
||||||
|
|
||||||
export enum CheckState {
|
export enum CheckState {
|
||||||
|
CONNECTING = 'Connecting',
|
||||||
OK = 'OK',
|
OK = 'OK',
|
||||||
WARNING = 'Warning',
|
WARNING = 'Warning',
|
||||||
ERROR = 'Error',
|
ERROR = 'Error',
|
||||||
@@ -35,31 +35,23 @@ interface StatusItem {
|
|||||||
|
|
||||||
interface Status {
|
interface Status {
|
||||||
all: CheckState
|
all: CheckState
|
||||||
version: StatusItem
|
|
||||||
blockchainConnection: StatusItem
|
|
||||||
debugApiConnection: StatusItem
|
|
||||||
apiConnection: StatusItem
|
apiConnection: StatusItem
|
||||||
topology: StatusItem
|
topology: StatusItem
|
||||||
chequebook: StatusItem
|
chequebook: StatusItem
|
||||||
}
|
}
|
||||||
|
|
||||||
interface ContextInterface {
|
interface ContextInterface {
|
||||||
|
beeVersion: string | null
|
||||||
status: Status
|
status: Status
|
||||||
latestPublishedVersion?: string
|
|
||||||
latestUserVersion?: string
|
|
||||||
latestUserVersionExact?: string
|
|
||||||
isLatestBeeVersion: boolean
|
|
||||||
latestBeeVersionUrl: string
|
|
||||||
error: Error | null
|
error: Error | null
|
||||||
apiHealth: boolean
|
apiHealth: boolean
|
||||||
debugApiHealth: Health | null
|
|
||||||
debugApiReadiness: boolean
|
|
||||||
nodeAddresses: NodeAddresses | null
|
nodeAddresses: NodeAddresses | null
|
||||||
nodeInfo: NodeInfo | null
|
nodeInfo: NodeInfo | null
|
||||||
topology: Topology | null
|
topology: Topology | null
|
||||||
chequebookAddress: ChequebookAddressResponse | null
|
chequebookAddress: ChequebookAddressResponse | null
|
||||||
peers: Peer[] | null
|
peers: Peer[] | null
|
||||||
chequebookBalance: ChequebookBalance | null
|
chequebookBalance: ChequebookBalance | null
|
||||||
|
stake: BzzToken | null
|
||||||
peerBalances: Balance[] | null
|
peerBalances: Balance[] | null
|
||||||
peerCheques: LastChequesResponse | null
|
peerCheques: LastChequesResponse | null
|
||||||
settlements: Settlements | null
|
settlements: Settlements | null
|
||||||
@@ -74,28 +66,20 @@ interface ContextInterface {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const initialValues: ContextInterface = {
|
const initialValues: ContextInterface = {
|
||||||
|
beeVersion: null,
|
||||||
status: {
|
status: {
|
||||||
all: CheckState.ERROR,
|
all: CheckState.ERROR,
|
||||||
version: { isEnabled: false, checkState: CheckState.ERROR },
|
|
||||||
blockchainConnection: { isEnabled: false, checkState: CheckState.ERROR },
|
|
||||||
debugApiConnection: { isEnabled: false, checkState: CheckState.ERROR },
|
|
||||||
apiConnection: { isEnabled: false, checkState: CheckState.ERROR },
|
apiConnection: { isEnabled: false, checkState: CheckState.ERROR },
|
||||||
topology: { isEnabled: false, checkState: CheckState.ERROR },
|
topology: { isEnabled: false, checkState: CheckState.ERROR },
|
||||||
chequebook: { isEnabled: false, checkState: CheckState.ERROR },
|
chequebook: { isEnabled: false, checkState: CheckState.ERROR },
|
||||||
},
|
},
|
||||||
latestPublishedVersion: undefined,
|
|
||||||
latestUserVersion: undefined,
|
|
||||||
latestUserVersionExact: undefined,
|
|
||||||
isLatestBeeVersion: false,
|
|
||||||
latestBeeVersionUrl: 'https://github.com/ethersphere/bee/releases/latest',
|
|
||||||
error: null,
|
error: null,
|
||||||
apiHealth: false,
|
apiHealth: false,
|
||||||
debugApiHealth: null,
|
|
||||||
debugApiReadiness: false,
|
|
||||||
nodeAddresses: null,
|
nodeAddresses: null,
|
||||||
nodeInfo: null,
|
nodeInfo: null,
|
||||||
topology: null,
|
topology: null,
|
||||||
chequebookAddress: null,
|
chequebookAddress: null,
|
||||||
|
stake: null,
|
||||||
peers: null,
|
peers: null,
|
||||||
chequebookBalance: null,
|
chequebookBalance: null,
|
||||||
peerBalances: null,
|
peerBalances: null,
|
||||||
@@ -119,36 +103,16 @@ interface Props {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function getStatus(
|
function getStatus(
|
||||||
debugApiHealth: Health | null,
|
|
||||||
debugApiReadiness: boolean,
|
|
||||||
nodeInfo: NodeInfo | null,
|
nodeInfo: NodeInfo | null,
|
||||||
apiHealth: boolean,
|
apiHealth: boolean,
|
||||||
topology: Topology | null,
|
topology: Topology | null,
|
||||||
chequebookAddress: ChequebookAddressResponse | null,
|
chequebookAddress: ChequebookAddressResponse | null,
|
||||||
chequebookBalance: ChequebookBalance | null,
|
chequebookBalance: ChequebookBalance | null,
|
||||||
error: Error | null,
|
error: Error | null,
|
||||||
isDesktop: boolean,
|
startedAt: number,
|
||||||
): Status {
|
): Status {
|
||||||
const status: Status = { ...initialValues.status }
|
const status: Status = { ...initialValues.status }
|
||||||
|
|
||||||
// Version check
|
|
||||||
status.version.isEnabled = !isDesktop
|
|
||||||
status.version.checkState =
|
|
||||||
debugApiHealth &&
|
|
||||||
semver.satisfies(debugApiHealth.version, PackageJson.engines.bee, {
|
|
||||||
includePrerelease: true,
|
|
||||||
})
|
|
||||||
? CheckState.OK
|
|
||||||
: CheckState.WARNING
|
|
||||||
|
|
||||||
// Blockchain connection check
|
|
||||||
status.blockchainConnection.isEnabled = true
|
|
||||||
status.blockchainConnection.checkState = Boolean(debugApiHealth?.status === 'ok') ? CheckState.OK : CheckState.ERROR
|
|
||||||
|
|
||||||
// Debug API connection check
|
|
||||||
status.debugApiConnection.isEnabled = true
|
|
||||||
status.debugApiConnection.checkState = Boolean(debugApiHealth?.status === 'ok') ? CheckState.OK : CheckState.ERROR
|
|
||||||
|
|
||||||
// API connection check
|
// API connection check
|
||||||
status.apiConnection.isEnabled = true
|
status.apiConnection.isEnabled = true
|
||||||
status.apiConnection.checkState = apiHealth ? CheckState.OK : CheckState.ERROR
|
status.apiConnection.checkState = apiHealth ? CheckState.OK : CheckState.ERROR
|
||||||
@@ -168,19 +132,25 @@ function getStatus(
|
|||||||
} else status.chequebook.checkState = CheckState.OK
|
} else status.chequebook.checkState = CheckState.OK
|
||||||
}
|
}
|
||||||
|
|
||||||
status.all = determineOverallStatus(debugApiHealth, debugApiReadiness, status)
|
status.all = determineOverallStatus(status, startedAt)
|
||||||
|
|
||||||
return status
|
return status
|
||||||
}
|
}
|
||||||
|
|
||||||
function determineOverallStatus(debugApiHealth: Health | null, debugApiReadiness: boolean, status: Status): CheckState {
|
function determineOverallStatus(status: Status, startedAt: number): CheckState {
|
||||||
if (debugApiHealth?.status === 'ok' && !debugApiReadiness) {
|
const hasErrors = Object.values(status).some(
|
||||||
return CheckState.STARTING
|
({ isEnabled, checkState }) => isEnabled && checkState === CheckState.ERROR,
|
||||||
} else if (Object.values(status).some(({ isEnabled, checkState }) => isEnabled && checkState === CheckState.ERROR)) {
|
)
|
||||||
|
const hasWarnings = Object.values(status).some(
|
||||||
|
({ isEnabled, checkState }) => isEnabled && checkState === CheckState.WARNING,
|
||||||
|
)
|
||||||
|
const isInGracePeriod = Date.now() - startedAt < LAUNCH_GRACE_PERIOD
|
||||||
|
|
||||||
|
if (hasErrors && isInGracePeriod) {
|
||||||
|
return CheckState.CONNECTING
|
||||||
|
} else if (hasErrors) {
|
||||||
return CheckState.ERROR
|
return CheckState.ERROR
|
||||||
} else if (
|
} else if (hasWarnings) {
|
||||||
Object.values(status).some(({ isEnabled, checkState }) => isEnabled && checkState === CheckState.WARNING)
|
|
||||||
) {
|
|
||||||
return CheckState.WARNING
|
return CheckState.WARNING
|
||||||
} else {
|
} else {
|
||||||
return CheckState.OK
|
return CheckState.OK
|
||||||
@@ -190,30 +160,27 @@ function determineOverallStatus(debugApiHealth: Health | null, debugApiReadiness
|
|||||||
// This does not need to be exposed and works much better as variable than state variable which may trigger some unnecessary re-renders
|
// This does not need to be exposed and works much better as variable than state variable which may trigger some unnecessary re-renders
|
||||||
let isRefreshing = false
|
let isRefreshing = false
|
||||||
|
|
||||||
interface InitialSettings {
|
interface Props {
|
||||||
isDesktop?: boolean
|
|
||||||
}
|
|
||||||
|
|
||||||
interface Props extends InitialSettings {
|
|
||||||
children: ReactChild
|
children: ReactChild
|
||||||
}
|
}
|
||||||
|
|
||||||
export function Provider({ children, isDesktop }: Props): ReactElement {
|
export function Provider({ children }: Props): ReactElement {
|
||||||
const { beeApi, beeDebugApi } = useContext(SettingsContext)
|
const { beeApi } = useContext(SettingsContext)
|
||||||
|
const [beeVersion, setBeeVersion] = useState<string | null>(null)
|
||||||
const [apiHealth, setApiHealth] = useState<boolean>(false)
|
const [apiHealth, setApiHealth] = useState<boolean>(false)
|
||||||
const [debugApiHealth, setDebugApiHealth] = useState<Health | null>(null)
|
|
||||||
const [debugApiReadiness, setDebugApiReadiness] = useState(false)
|
|
||||||
const [nodeAddresses, setNodeAddresses] = useState<NodeAddresses | null>(null)
|
const [nodeAddresses, setNodeAddresses] = useState<NodeAddresses | null>(null)
|
||||||
const [nodeInfo, setNodeInfo] = useState<NodeInfo | null>(null)
|
const [nodeInfo, setNodeInfo] = useState<NodeInfo | null>(null)
|
||||||
const [topology, setNodeTopology] = useState<Topology | null>(null)
|
const [topology, setNodeTopology] = useState<Topology | null>(null)
|
||||||
const [chequebookAddress, setChequebookAddress] = useState<ChequebookAddressResponse | null>(null)
|
const [chequebookAddress, setChequebookAddress] = useState<ChequebookAddressResponse | null>(null)
|
||||||
const [peers, setPeers] = useState<Peer[] | null>(null)
|
const [peers, setPeers] = useState<Peer[] | null>(null)
|
||||||
const [chequebookBalance, setChequebookBalance] = useState<ChequebookBalance | null>(null)
|
const [chequebookBalance, setChequebookBalance] = useState<ChequebookBalance | null>(null)
|
||||||
|
const [stake, setStake] = useState<BzzToken | null>(null)
|
||||||
const [peerBalances, setPeerBalances] = useState<Balance[] | null>(null)
|
const [peerBalances, setPeerBalances] = useState<Balance[] | null>(null)
|
||||||
const [peerCheques, setPeerCheques] = useState<LastChequesResponse | null>(null)
|
const [peerCheques, setPeerCheques] = useState<LastChequesResponse | null>(null)
|
||||||
const [settlements, setSettlements] = useState<Settlements | null>(null)
|
const [settlements, setSettlements] = useState<Settlements | null>(null)
|
||||||
const [chainState, setChainState] = useState<ChainState | null>(null)
|
const [chainState, setChainState] = useState<ChainState | null>(null)
|
||||||
const [chainId, setChainId] = useState<number | null>(null)
|
const [chainId, setChainId] = useState<number | null>(null)
|
||||||
|
const [startedAt] = useState(Date.now())
|
||||||
|
|
||||||
const { latestBeeRelease } = useLatestBeeRelease()
|
const { latestBeeRelease } = useLatestBeeRelease()
|
||||||
|
|
||||||
@@ -222,10 +189,6 @@ export function Provider({ children, isDesktop }: Props): ReactElement {
|
|||||||
const [lastUpdate, setLastUpdate] = useState<number | null>(initialValues.lastUpdate)
|
const [lastUpdate, setLastUpdate] = useState<number | null>(initialValues.lastUpdate)
|
||||||
const [frequency, setFrequency] = useState<number | null>(30000)
|
const [frequency, setFrequency] = useState<number | null>(30000)
|
||||||
|
|
||||||
const latestPublishedVersion = semver.coerce(latestBeeRelease?.name)?.version
|
|
||||||
const latestUserVersion = semver.coerce(debugApiHealth?.version)?.version
|
|
||||||
const latestUserVersionExact = debugApiHealth?.version
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setIsLoading(true)
|
setIsLoading(true)
|
||||||
|
|
||||||
@@ -236,8 +199,6 @@ export function Provider({ children, isDesktop }: Props): ReactElement {
|
|||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setIsLoading(true)
|
setIsLoading(true)
|
||||||
|
|
||||||
setDebugApiHealth(null)
|
|
||||||
setNodeAddresses(null)
|
setNodeAddresses(null)
|
||||||
setNodeTopology(null)
|
setNodeTopology(null)
|
||||||
setNodeInfo(null)
|
setNodeInfo(null)
|
||||||
@@ -249,15 +210,19 @@ export function Provider({ children, isDesktop }: Props): ReactElement {
|
|||||||
setSettlements(null)
|
setSettlements(null)
|
||||||
setChainState(null)
|
setChainState(null)
|
||||||
|
|
||||||
if (beeDebugApi !== null) refresh()
|
if (beeApi !== null) {
|
||||||
}, [beeDebugApi]) // eslint-disable-line react-hooks/exhaustive-deps
|
refresh()
|
||||||
|
}
|
||||||
|
}, [beeApi]) // eslint-disable-line react-hooks/exhaustive-deps
|
||||||
|
|
||||||
const refresh = async () => {
|
const refresh = async () => {
|
||||||
// Don't want to refresh when already refreshing
|
// Don't want to refresh when already refreshing
|
||||||
if (isRefreshing) return
|
if (isRefreshing) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// Not a valid bee api
|
// Not a valid bee api
|
||||||
if (!beeApi || !beeDebugApi) {
|
if (!beeApi) {
|
||||||
setIsLoading(false)
|
setIsLoading(false)
|
||||||
|
|
||||||
return
|
return
|
||||||
@@ -269,7 +234,7 @@ export function Provider({ children, isDesktop }: Props): ReactElement {
|
|||||||
|
|
||||||
// Wrap the chequebook balance call to return BZZ values as Token object
|
// Wrap the chequebook balance call to return BZZ values as Token object
|
||||||
const chequeBalanceWrapper = async () => {
|
const chequeBalanceWrapper = async () => {
|
||||||
const { totalBalance, availableBalance } = await beeDebugApi.getChequebookBalance({ timeout: TIMEOUT })
|
const { totalBalance, availableBalance } = await beeApi.getChequebookBalance({ timeout: TIMEOUT })
|
||||||
|
|
||||||
return {
|
return {
|
||||||
totalBalance: new Token(totalBalance),
|
totalBalance: new Token(totalBalance),
|
||||||
@@ -279,14 +244,14 @@ export function Provider({ children, isDesktop }: Props): ReactElement {
|
|||||||
|
|
||||||
// Wrap the balances call to return BZZ values as Token object
|
// Wrap the balances call to return BZZ values as Token object
|
||||||
const peerBalanceWrapper = async () => {
|
const peerBalanceWrapper = async () => {
|
||||||
const { balances } = await beeDebugApi.getAllBalances({ timeout: TIMEOUT })
|
const { balances } = await beeApi.getAllBalances({ timeout: TIMEOUT })
|
||||||
|
|
||||||
return balances.map(({ peer, balance }) => ({ peer, balance: new Token(balance) }))
|
return balances.map(({ peer, balance }) => ({ peer, balance: new Token(balance) }))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wrap the settlements call to return BZZ values as Token object
|
// Wrap the settlements call to return BZZ values as Token object
|
||||||
const settlementsWrapper = async () => {
|
const settlementsWrapper = async () => {
|
||||||
const { totalReceived, settlements, totalSent } = await beeDebugApi.getAllSettlements({ timeout: TIMEOUT })
|
const { totalReceived, settlements, totalSent } = await beeApi.getAllSettlements({ timeout: TIMEOUT })
|
||||||
|
|
||||||
return {
|
return {
|
||||||
totalReceived: new Token(totalReceived),
|
totalReceived: new Token(totalReceived),
|
||||||
@@ -302,66 +267,58 @@ export function Provider({ children, isDesktop }: Props): ReactElement {
|
|||||||
const promises = [
|
const promises = [
|
||||||
// API health
|
// API health
|
||||||
beeApi
|
beeApi
|
||||||
.isConnected({ timeout: TIMEOUT })
|
|
||||||
.then(setApiHealth)
|
|
||||||
.catch(() => setApiHealth(false)),
|
|
||||||
|
|
||||||
// Debug API health
|
|
||||||
beeDebugApi
|
|
||||||
.getHealth({ timeout: TIMEOUT })
|
.getHealth({ timeout: TIMEOUT })
|
||||||
.then(setDebugApiHealth)
|
.then(response => setBeeVersion(response.version))
|
||||||
.catch(() => setDebugApiHealth(null)),
|
.then(() => setApiHealth(true))
|
||||||
|
.catch(() => {
|
||||||
// Debug API readiness
|
setBeeVersion(null)
|
||||||
beeDebugApi
|
setApiHealth(false)
|
||||||
.getReadiness({ timeout: TIMEOUT })
|
}),
|
||||||
.then(setDebugApiReadiness)
|
|
||||||
.catch(() => setDebugApiReadiness(false)),
|
|
||||||
|
|
||||||
// Node Addresses
|
// Node Addresses
|
||||||
beeDebugApi
|
beeApi
|
||||||
.getNodeAddresses({ timeout: TIMEOUT })
|
.getNodeAddresses({ timeout: TIMEOUT })
|
||||||
.then(setNodeAddresses)
|
.then(setNodeAddresses)
|
||||||
.catch(() => setNodeAddresses(null)),
|
.catch(() => setNodeAddresses(null)),
|
||||||
|
|
||||||
// NodeInfo
|
// NodeInfo
|
||||||
beeDebugApi
|
beeApi
|
||||||
.getNodeInfo({ timeout: TIMEOUT })
|
.getNodeInfo({ timeout: TIMEOUT })
|
||||||
.then(setNodeInfo)
|
.then(setNodeInfo)
|
||||||
.catch(() => setNodeInfo(null)),
|
.catch(() => setNodeInfo(null)),
|
||||||
|
|
||||||
// Network Topology
|
// Network Topology
|
||||||
beeDebugApi
|
beeApi
|
||||||
.getTopology({ timeout: TIMEOUT })
|
.getTopology({ timeout: TIMEOUT })
|
||||||
.then(setNodeTopology)
|
.then(setNodeTopology)
|
||||||
.catch(() => setNodeTopology(null)),
|
.catch(() => setNodeTopology(null)),
|
||||||
|
|
||||||
// Peers
|
// Peers
|
||||||
beeDebugApi
|
beeApi
|
||||||
.getPeers({ timeout: TIMEOUT })
|
.getPeers({ timeout: TIMEOUT })
|
||||||
.then(setPeers)
|
.then(setPeers)
|
||||||
.catch(() => setPeers(null)),
|
.catch(() => setPeers(null)),
|
||||||
|
|
||||||
// Chequebook address
|
// Chequebook address
|
||||||
beeDebugApi
|
beeApi
|
||||||
.getChequebookAddress({ timeout: TIMEOUT })
|
.getChequebookAddress({ timeout: TIMEOUT })
|
||||||
.then(setChequebookAddress)
|
.then(setChequebookAddress)
|
||||||
.catch(() => setChequebookAddress(null)),
|
.catch(() => setChequebookAddress(null)),
|
||||||
|
|
||||||
// Cheques
|
// Cheques
|
||||||
beeDebugApi
|
beeApi
|
||||||
.getLastCheques({ timeout: TIMEOUT })
|
.getLastCheques({ timeout: TIMEOUT })
|
||||||
.then(setPeerCheques)
|
.then(setPeerCheques)
|
||||||
.catch(() => setPeerCheques(null)),
|
.catch(() => setPeerCheques(null)),
|
||||||
|
|
||||||
// Chain state
|
// Chain state
|
||||||
beeDebugApi
|
beeApi
|
||||||
.getChainState({ timeout: TIMEOUT })
|
.getChainState({ timeout: TIMEOUT })
|
||||||
.then(setChainState)
|
.then(setChainState)
|
||||||
.catch(() => setChainState(null)),
|
.catch(() => setChainState(null)),
|
||||||
|
|
||||||
// Wallet
|
// Wallet
|
||||||
beeDebugApi
|
beeApi
|
||||||
.getWalletBalance({ timeout: TIMEOUT })
|
.getWalletBalance({ timeout: TIMEOUT })
|
||||||
.then(({ chainID }) => setChainId(chainID))
|
.then(({ chainID }) => setChainId(chainID))
|
||||||
.catch(() => setChainId(null)),
|
.catch(() => setChainId(null)),
|
||||||
@@ -371,6 +328,11 @@ export function Provider({ children, isDesktop }: Props): ReactElement {
|
|||||||
.then(setChequebookBalance)
|
.then(setChequebookBalance)
|
||||||
.catch(() => setChequebookBalance(null)),
|
.catch(() => setChequebookBalance(null)),
|
||||||
|
|
||||||
|
beeApi
|
||||||
|
.getStake({ timeout: TIMEOUT })
|
||||||
|
.then(stake => setStake(new BzzToken(stake)))
|
||||||
|
.catch(() => setStake(null)),
|
||||||
|
|
||||||
// Peer balances
|
// Peer balances
|
||||||
peerBalanceWrapper()
|
peerBalanceWrapper()
|
||||||
.then(setPeerBalances)
|
.then(setPeerBalances)
|
||||||
@@ -398,17 +360,7 @@ export function Provider({ children, isDesktop }: Props): ReactElement {
|
|||||||
}
|
}
|
||||||
const stop = () => setFrequency(null)
|
const stop = () => setFrequency(null)
|
||||||
|
|
||||||
const status = getStatus(
|
const status = getStatus(nodeInfo, apiHealth, topology, chequebookAddress, chequebookBalance, error, startedAt)
|
||||||
debugApiHealth,
|
|
||||||
debugApiReadiness,
|
|
||||||
nodeInfo,
|
|
||||||
apiHealth,
|
|
||||||
topology,
|
|
||||||
chequebookAddress,
|
|
||||||
chequebookBalance,
|
|
||||||
error,
|
|
||||||
Boolean(isDesktop),
|
|
||||||
)
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
let newFrequency = REFRESH_WHEN_OK
|
let newFrequency = REFRESH_WHEN_OK
|
||||||
@@ -426,33 +378,22 @@ export function Provider({ children, isDesktop }: Props): ReactElement {
|
|||||||
|
|
||||||
return () => clearInterval(interval)
|
return () => clearInterval(interval)
|
||||||
}
|
}
|
||||||
}, [frequency, beeDebugApi, beeApi]) // eslint-disable-line react-hooks/exhaustive-deps
|
}, [frequency, beeApi]) // eslint-disable-line react-hooks/exhaustive-deps
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Context.Provider
|
<Context.Provider
|
||||||
value={{
|
value={{
|
||||||
|
beeVersion,
|
||||||
status,
|
status,
|
||||||
latestUserVersion,
|
|
||||||
latestUserVersionExact,
|
|
||||||
latestPublishedVersion,
|
|
||||||
isLatestBeeVersion: Boolean(
|
|
||||||
latestPublishedVersion &&
|
|
||||||
latestUserVersion &&
|
|
||||||
semver.satisfies(latestPublishedVersion, latestUserVersion, {
|
|
||||||
includePrerelease: true,
|
|
||||||
}),
|
|
||||||
),
|
|
||||||
latestBeeVersionUrl: latestBeeRelease?.html_url || 'https://github.com/ethersphere/bee/releases/latest',
|
|
||||||
error,
|
error,
|
||||||
apiHealth,
|
apiHealth,
|
||||||
debugApiHealth,
|
|
||||||
debugApiReadiness,
|
|
||||||
nodeAddresses,
|
nodeAddresses,
|
||||||
nodeInfo,
|
nodeInfo,
|
||||||
topology,
|
topology,
|
||||||
chequebookAddress,
|
chequebookAddress,
|
||||||
peers,
|
peers,
|
||||||
chequebookBalance,
|
chequebookBalance,
|
||||||
|
stake,
|
||||||
peerBalances,
|
peerBalances,
|
||||||
peerCheques,
|
peerCheques,
|
||||||
settlements,
|
settlements,
|
||||||
|
|||||||
+16
-6
@@ -41,7 +41,8 @@ export function Provider({ children }: Props): ReactElement {
|
|||||||
const [previewBlob, setPreviewBlob] = useState<Blob | undefined>(undefined)
|
const [previewBlob, setPreviewBlob] = useState<Blob | undefined>(undefined)
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setMetadata(getMetadata(files))
|
const metadata = getMetadata(files)
|
||||||
|
setMetadata(metadata)
|
||||||
|
|
||||||
if (previewUri) {
|
if (previewUri) {
|
||||||
URL.revokeObjectURL(previewUri) // Clear the preview from memory
|
URL.revokeObjectURL(previewUri) // Clear the preview from memory
|
||||||
@@ -49,12 +50,21 @@ export function Provider({ children }: Props): ReactElement {
|
|||||||
setPreviewBlob(undefined)
|
setPreviewBlob(undefined)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (files.length !== 1 || !files[0].type.startsWith('image')) return
|
if (files.length !== 1) return
|
||||||
|
|
||||||
resize(files[0], PREVIEW_DIMENSIONS.maxWidth, PREVIEW_DIMENSIONS.maxHeight).then(blob => {
|
if (metadata.isVideo) {
|
||||||
setPreviewUri(URL.createObjectURL(blob)) // NOTE: Until it is cleared with URL.revokeObjectURL, the file stays allocated in memory
|
const videoFile = files[0]
|
||||||
setPreviewBlob(blob)
|
const videoBlob = new Blob([videoFile], { type: videoFile.type })
|
||||||
})
|
setPreviewUri(URL.createObjectURL(videoBlob))
|
||||||
|
setPreviewBlob(videoBlob)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (metadata.isImage) {
|
||||||
|
resize(files[0], PREVIEW_DIMENSIONS.maxWidth, PREVIEW_DIMENSIONS.maxHeight).then(blob => {
|
||||||
|
setPreviewUri(URL.createObjectURL(blob)) // NOTE: Until it is cleared with URL.revokeObjectURL, the file stays allocated in memory
|
||||||
|
setPreviewBlob(blob)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
return () => {
|
return () => {
|
||||||
if (previewUri) {
|
if (previewUri) {
|
||||||
|
|||||||
+11
-39
@@ -1,8 +1,8 @@
|
|||||||
import { Bee, BeeDebug } from '@ethersphere/bee-js'
|
import { Bee } from '@ethersphere/bee-js'
|
||||||
import { providers } from 'ethers'
|
import { providers } from 'ethers'
|
||||||
import { createContext, ReactNode, ReactElement, useEffect, useState } from 'react'
|
import { ReactElement, ReactNode, createContext, useEffect, useState } from 'react'
|
||||||
|
import { DEFAULT_BEE_API_HOST, DEFAULT_RPC_URL } from '../constants'
|
||||||
import { useGetBeeConfig } from '../hooks/apiHooks'
|
import { useGetBeeConfig } from '../hooks/apiHooks'
|
||||||
import { DEFAULT_BEE_API_HOST, DEFAULT_BEE_DEBUG_API_HOST, DEFAULT_RPC_URL } from '../constants'
|
|
||||||
|
|
||||||
const LocalStorageKeys = {
|
const LocalStorageKeys = {
|
||||||
providerUrl: 'json-rpc-provider',
|
providerUrl: 'json-rpc-provider',
|
||||||
@@ -10,20 +10,17 @@ const LocalStorageKeys = {
|
|||||||
|
|
||||||
interface ContextInterface {
|
interface ContextInterface {
|
||||||
apiUrl: string
|
apiUrl: string
|
||||||
apiDebugUrl: string
|
|
||||||
beeApi: Bee | null
|
beeApi: Bee | null
|
||||||
beeDebugApi: BeeDebug | null
|
|
||||||
lockedApiSettings: boolean
|
lockedApiSettings: boolean
|
||||||
desktopApiKey: string
|
desktopApiKey: string
|
||||||
isDesktop: boolean
|
isDesktop: boolean
|
||||||
desktopUrl: string
|
desktopUrl: string
|
||||||
rpcProviderUrl: string
|
rpcProviderUrl: string
|
||||||
rpcProvider: providers.JsonRpcProvider
|
rpcProvider: providers.JsonRpcProvider | null
|
||||||
cors: string | null
|
cors: string | null
|
||||||
dataDir: string | null
|
dataDir: string | null
|
||||||
ensResolver: string | null
|
ensResolver: string | null
|
||||||
setApiUrl: (url: string) => void
|
setApiUrl: (url: string) => void
|
||||||
setDebugApiUrl: (url: string) => void
|
|
||||||
setAndPersistJsonRpcProvider: (url: string) => void
|
setAndPersistJsonRpcProvider: (url: string) => void
|
||||||
isLoading: boolean
|
isLoading: boolean
|
||||||
error: Error | null
|
error: Error | null
|
||||||
@@ -31,18 +28,15 @@ interface ContextInterface {
|
|||||||
|
|
||||||
const initialValues: ContextInterface = {
|
const initialValues: ContextInterface = {
|
||||||
beeApi: null,
|
beeApi: null,
|
||||||
beeDebugApi: null,
|
|
||||||
apiUrl: DEFAULT_BEE_API_HOST,
|
apiUrl: DEFAULT_BEE_API_HOST,
|
||||||
apiDebugUrl: DEFAULT_BEE_DEBUG_API_HOST,
|
|
||||||
setApiUrl: () => {}, // eslint-disable-line
|
setApiUrl: () => {}, // eslint-disable-line
|
||||||
setDebugApiUrl: () => {}, // eslint-disable-line
|
|
||||||
lockedApiSettings: false,
|
lockedApiSettings: false,
|
||||||
isDesktop: false,
|
isDesktop: false,
|
||||||
desktopApiKey: '',
|
desktopApiKey: '',
|
||||||
desktopUrl: window.location.origin,
|
desktopUrl: window.location.origin,
|
||||||
setAndPersistJsonRpcProvider: async () => {}, // eslint-disable-line
|
setAndPersistJsonRpcProvider: async () => {}, // eslint-disable-line
|
||||||
rpcProviderUrl: '',
|
rpcProviderUrl: '',
|
||||||
rpcProvider: new providers.JsonRpcProvider(''),
|
rpcProvider: null,
|
||||||
cors: null,
|
cors: null,
|
||||||
dataDir: null,
|
dataDir: null,
|
||||||
ensResolver: null,
|
ensResolver: null,
|
||||||
@@ -55,7 +49,6 @@ export const Consumer = Context.Consumer
|
|||||||
|
|
||||||
interface InitialSettings {
|
interface InitialSettings {
|
||||||
beeApiUrl?: string
|
beeApiUrl?: string
|
||||||
beeDebugApiUrl?: string
|
|
||||||
lockedApiSettings?: boolean
|
lockedApiSettings?: boolean
|
||||||
isDesktop?: boolean
|
isDesktop?: boolean
|
||||||
desktopUrl?: string
|
desktopUrl?: string
|
||||||
@@ -72,25 +65,15 @@ export function Provider({ children, ...propsSettings }: Props): ReactElement {
|
|||||||
const propsProviderUrl =
|
const propsProviderUrl =
|
||||||
localStorage.getItem(LocalStorageKeys.providerUrl) || propsSettings.defaultRpcUrl || DEFAULT_RPC_URL
|
localStorage.getItem(LocalStorageKeys.providerUrl) || propsSettings.defaultRpcUrl || DEFAULT_RPC_URL
|
||||||
|
|
||||||
const [apiUrl, setApiUrl] = useState<string>(initialValues.apiUrl)
|
const [apiUrl, setApiUrl] = useState<string>(
|
||||||
const [apiDebugUrl, setDebugApiUrl] = useState<string>(initialValues.apiDebugUrl)
|
sessionStorage.getItem('api_host') ?? propsSettings.beeApiUrl ?? initialValues.apiUrl,
|
||||||
|
)
|
||||||
const [beeApi, setBeeApi] = useState<Bee | null>(null)
|
const [beeApi, setBeeApi] = useState<Bee | null>(null)
|
||||||
const [beeDebugApi, setBeeDebugApi] = useState<BeeDebug | null>(null)
|
|
||||||
const [desktopApiKey, setDesktopApiKey] = useState<string>(initialValues.desktopApiKey)
|
const [desktopApiKey, setDesktopApiKey] = useState<string>(initialValues.desktopApiKey)
|
||||||
const [rpcProviderUrl, setRpcProviderUrl] = useState(propsProviderUrl)
|
const [rpcProviderUrl, setRpcProviderUrl] = useState(propsProviderUrl)
|
||||||
const [rpcProvider, setRpcProvider] = useState(new providers.JsonRpcProvider(propsProviderUrl))
|
const [rpcProvider, setRpcProvider] = useState(new providers.JsonRpcProvider(propsProviderUrl))
|
||||||
const { config, isLoading, error } = useGetBeeConfig(desktopUrl)
|
const { config, isLoading, error } = useGetBeeConfig(desktopUrl)
|
||||||
|
|
||||||
const url = makeHttpUrl(
|
|
||||||
config?.['api-addr'] ?? sessionStorage.getItem('api_host') ?? propsSettings.beeApiUrl ?? apiUrl,
|
|
||||||
)
|
|
||||||
const debugUrl = makeHttpUrl(
|
|
||||||
config?.['debug-api-addr'] ??
|
|
||||||
sessionStorage.getItem('debug_api_host') ??
|
|
||||||
propsSettings.beeDebugApiUrl ??
|
|
||||||
apiDebugUrl,
|
|
||||||
)
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const urlSearchParams = new URLSearchParams(window.location.search)
|
const urlSearchParams = new URLSearchParams(window.location.search)
|
||||||
const newApiKey = urlSearchParams.get('v')
|
const newApiKey = urlSearchParams.get('v')
|
||||||
@@ -103,32 +86,21 @@ export function Provider({ children, ...propsSettings }: Props): ReactElement {
|
|||||||
}, [])
|
}, [])
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
const url = makeHttpUrl(config?.['api-addr'] ?? apiUrl)
|
||||||
try {
|
try {
|
||||||
setBeeApi(new Bee(url))
|
setBeeApi(new Bee(url))
|
||||||
sessionStorage.setItem('api_host', url)
|
sessionStorage.setItem('api_host', url)
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
setBeeApi(null)
|
setBeeApi(null)
|
||||||
}
|
}
|
||||||
}, [url])
|
}, [config, apiUrl])
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
try {
|
|
||||||
setBeeDebugApi(new BeeDebug(debugUrl))
|
|
||||||
sessionStorage.setItem('debug_api_host', debugUrl)
|
|
||||||
} catch (e) {
|
|
||||||
setBeeDebugApi(null)
|
|
||||||
}
|
|
||||||
}, [debugUrl])
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Context.Provider
|
<Context.Provider
|
||||||
value={{
|
value={{
|
||||||
apiUrl: url,
|
apiUrl,
|
||||||
apiDebugUrl: debugUrl,
|
|
||||||
beeApi,
|
beeApi,
|
||||||
beeDebugApi,
|
|
||||||
setApiUrl,
|
setApiUrl,
|
||||||
setDebugApiUrl,
|
|
||||||
lockedApiSettings: Boolean(propsSettings.lockedApiSettings),
|
lockedApiSettings: Boolean(propsSettings.lockedApiSettings),
|
||||||
desktopApiKey,
|
desktopApiKey,
|
||||||
isDesktop,
|
isDesktop,
|
||||||
|
|||||||
@@ -48,7 +48,7 @@ export function enrichStamp(postageBatch: PostageBatch): EnrichedPostageBatch {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function Provider({ children }: Props): ReactElement {
|
export function Provider({ children }: Props): ReactElement {
|
||||||
const { beeDebugApi } = useContext(SettingsContext)
|
const { beeApi } = useContext(SettingsContext)
|
||||||
const [stamps, setStamps] = useState<EnrichedPostageBatch[] | null>(initialValues.stamps)
|
const [stamps, setStamps] = useState<EnrichedPostageBatch[] | null>(initialValues.stamps)
|
||||||
const [error, setError] = useState<Error | null>(initialValues.error)
|
const [error, setError] = useState<Error | null>(initialValues.error)
|
||||||
const [isLoading, setIsLoading] = useState<boolean>(initialValues.isLoading)
|
const [isLoading, setIsLoading] = useState<boolean>(initialValues.isLoading)
|
||||||
@@ -56,17 +56,21 @@ export function Provider({ children }: Props): ReactElement {
|
|||||||
const [frequency, setFrequency] = useState<number | null>(null)
|
const [frequency, setFrequency] = useState<number | null>(null)
|
||||||
|
|
||||||
const refresh = async () => {
|
const refresh = async () => {
|
||||||
// Don't want to refresh when already refreshing
|
if (isLoading) {
|
||||||
if (isLoading) return
|
return
|
||||||
|
}
|
||||||
|
|
||||||
if (!beeDebugApi) return
|
if (!beeApi) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
setIsLoading(true)
|
setIsLoading(true)
|
||||||
const stamps = await beeDebugApi.getAllPostageBatch()
|
const stamps = await beeApi.getAllPostageBatch()
|
||||||
|
|
||||||
setStamps(stamps.filter(x => x.exists).map(enrichStamp))
|
setStamps(stamps.filter(x => x.exists).map(enrichStamp))
|
||||||
setLastUpdate(Date.now())
|
setLastUpdate(Date.now())
|
||||||
|
setError(null)
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
setError(e as Error)
|
setError(e as Error)
|
||||||
} finally {
|
} finally {
|
||||||
|
|||||||
Vendored
+4
-2
@@ -6,14 +6,16 @@ interface LatestBeeRelease {
|
|||||||
}
|
}
|
||||||
|
|
||||||
interface SwarmMetadata {
|
interface SwarmMetadata {
|
||||||
size: number
|
size?: number
|
||||||
name: string
|
name: string
|
||||||
type?: string
|
type?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
interface Metadata extends SwarmMetadata {
|
interface Metadata extends SwarmMetadata {
|
||||||
type: string
|
type: string
|
||||||
isWebsite: boolean
|
isWebsite?: boolean
|
||||||
|
isVideo?: boolean
|
||||||
|
isImage?: boolean
|
||||||
count?: number
|
count?: number
|
||||||
hash?: string
|
hash?: string
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,118 @@
|
|||||||
|
import { useEffect, useState } from 'react'
|
||||||
|
import { CafeReactFsCreate } from './CafeReactFsCreate'
|
||||||
|
import { CafeReactFsItem } from './CafeReactFsItem'
|
||||||
|
import { CafeReactFsLoading } from './CafeReactFsLoading'
|
||||||
|
import { CafeReactFsPath } from './CafeReactFsPath'
|
||||||
|
import { CafeReactFsSync } from './CafeReactFsSync'
|
||||||
|
import { CafeReactFsUpload } from './CafeReactFsUpload'
|
||||||
|
import { FsItem } from './CafeReactType'
|
||||||
|
|
||||||
|
const DEFAULT_BACKGROUND_COLOR = '#f0f0f0'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
download: (path: string) => Promise<void>
|
||||||
|
list: (path: string) => Promise<FsItem[]>
|
||||||
|
onUpload: (path: string) => Promise<void>
|
||||||
|
onCreateDirectory: (path: string) => Promise<void>
|
||||||
|
onDeleteFile: (path: string) => Promise<void>
|
||||||
|
onDeleteDirectory: (path: string) => Promise<void>
|
||||||
|
onSync: () => Promise<void>
|
||||||
|
reloader: number
|
||||||
|
backgroundColor?: string
|
||||||
|
rootAlias?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function CafeReactFs({
|
||||||
|
download,
|
||||||
|
list,
|
||||||
|
onUpload,
|
||||||
|
onCreateDirectory,
|
||||||
|
onDeleteFile,
|
||||||
|
onDeleteDirectory,
|
||||||
|
onSync,
|
||||||
|
reloader,
|
||||||
|
backgroundColor,
|
||||||
|
rootAlias,
|
||||||
|
}: Props) {
|
||||||
|
const [path, setPath] = useState('/')
|
||||||
|
const [items, setItems] = useState<FsItem[]>([])
|
||||||
|
const [loading, setLoading] = useState(false)
|
||||||
|
|
||||||
|
function setItemsSorted(items: FsItem[]) {
|
||||||
|
// directories first, all alphabetically
|
||||||
|
const sortedItems = items.slice().sort((a, b) => {
|
||||||
|
if (a.$type === b.$type) {
|
||||||
|
return a.name.localeCompare(b.name)
|
||||||
|
}
|
||||||
|
|
||||||
|
return a.$type === 'directory' ? -1 : 1
|
||||||
|
})
|
||||||
|
setItems(sortedItems)
|
||||||
|
}
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setLoading(true)
|
||||||
|
list(path)
|
||||||
|
.then(setItemsSorted)
|
||||||
|
.finally(() => setLoading(false))
|
||||||
|
}, [reloader, list, path])
|
||||||
|
|
||||||
|
const pathParts = ['/', ...path.split('/').filter(x => x)]
|
||||||
|
|
||||||
|
function jumpToDirectory(fullPath: string) {
|
||||||
|
setPath(fullPath)
|
||||||
|
setLoading(true)
|
||||||
|
list(fullPath)
|
||||||
|
.then(setItemsSorted)
|
||||||
|
.finally(() => setLoading(false))
|
||||||
|
}
|
||||||
|
|
||||||
|
function enterDirectory(name: string) {
|
||||||
|
const newPath = path.endsWith('/') ? `${path}${name}` : `${path}/${name}`
|
||||||
|
setPath(newPath)
|
||||||
|
setLoading(true)
|
||||||
|
list(newPath)
|
||||||
|
.then(setItemsSorted)
|
||||||
|
.finally(() => setLoading(false))
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div style={{ display: 'flex', flexDirection: 'column', gap: '4px' }}>
|
||||||
|
<CafeReactFsPath
|
||||||
|
pathParts={pathParts}
|
||||||
|
jumpToDirectory={jumpToDirectory}
|
||||||
|
backgroundColor={backgroundColor ?? DEFAULT_BACKGROUND_COLOR}
|
||||||
|
rootAlias={rootAlias}
|
||||||
|
/>
|
||||||
|
<div style={{ display: 'flex', flexDirection: 'row', flexWrap: 'wrap', gap: '4px' }}>
|
||||||
|
{loading && <CafeReactFsLoading backgroundColor={backgroundColor ?? DEFAULT_BACKGROUND_COLOR} />}
|
||||||
|
{!loading &&
|
||||||
|
items.map(item => (
|
||||||
|
<CafeReactFsItem
|
||||||
|
key={item.id}
|
||||||
|
path={path}
|
||||||
|
item={item}
|
||||||
|
enterDirectory={enterDirectory}
|
||||||
|
onDeleteFile={onDeleteFile}
|
||||||
|
onDeleteDirectory={onDeleteDirectory}
|
||||||
|
download={download}
|
||||||
|
backgroundColor={backgroundColor ?? DEFAULT_BACKGROUND_COLOR}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
{!loading && (
|
||||||
|
<>
|
||||||
|
<CafeReactFsUpload
|
||||||
|
onUpload={() => onUpload(path)}
|
||||||
|
backgroundColor={backgroundColor ?? DEFAULT_BACKGROUND_COLOR}
|
||||||
|
/>
|
||||||
|
<CafeReactFsCreate
|
||||||
|
onCreateDirectory={() => onCreateDirectory(path)}
|
||||||
|
backgroundColor={backgroundColor ?? DEFAULT_BACKGROUND_COLOR}
|
||||||
|
/>
|
||||||
|
<CafeReactFsSync backgroundColor={backgroundColor ?? DEFAULT_BACKGROUND_COLOR} onSync={onSync} />
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,54 @@
|
|||||||
|
import { useState } from 'react'
|
||||||
|
import { CafeReactFsLoading } from './CafeReactFsLoading'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
backgroundColor: string
|
||||||
|
onCreateDirectory: () => Promise<void>
|
||||||
|
}
|
||||||
|
|
||||||
|
export function CafeReactFsCreate({ backgroundColor, onCreateDirectory }: Props) {
|
||||||
|
const [loading, setLoading] = useState(false)
|
||||||
|
|
||||||
|
function proxyUpload() {
|
||||||
|
setLoading(true)
|
||||||
|
onCreateDirectory().finally(() => setLoading(false))
|
||||||
|
}
|
||||||
|
|
||||||
|
if (loading) {
|
||||||
|
return <CafeReactFsLoading backgroundColor={backgroundColor} />
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
width: '80px',
|
||||||
|
height: '80px',
|
||||||
|
position: 'relative',
|
||||||
|
background: backgroundColor,
|
||||||
|
borderRadius: '2px',
|
||||||
|
cursor: 'pointer',
|
||||||
|
}}
|
||||||
|
onClick={proxyUpload}
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
alt="Create"
|
||||||
|
style={{ width: '64px', height: '64px', position: 'absolute', left: '8px', top: 0 }}
|
||||||
|
src="data:image/svg+xml,%3C%3Fxml%20version%3D%221.0%22%20encoding%3D%22UTF-8%22%3F%3E%3Csvg%20width%3D%22512%22%20height%3D%22512%22%20viewBox%3D%220%200%20512%20512%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20d%3D%22M%20106%20131%20l%20100%200%20l%2025%2025%20l%20175%200%20l%200%20200%20l%20-300%200%20z%22%20stroke%3D%22%231F2D3D%22%20stroke-width%3D%2220%22%20fill%3D%22none%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%20%20%2F%3E%3Cpath%20d%3D%22M%20206%20256%20l%20100%200%22%20stroke%3D%22%23B83B5E%22%20stroke-width%3D%2220%22%20fill%3D%22none%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%20%20%2F%3E%3Cpath%20d%3D%22M%20256%20206%20l%200%20100%22%20stroke%3D%22%23B83B5E%22%20stroke-width%3D%2220%22%20fill%3D%22none%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%20%20%2F%3E%3C%2Fsvg%3E"
|
||||||
|
/>
|
||||||
|
<p
|
||||||
|
style={{
|
||||||
|
margin: 0,
|
||||||
|
fontFamily: 'sans-serif',
|
||||||
|
fontSize: '12px',
|
||||||
|
textAlign: 'center',
|
||||||
|
width: '80px',
|
||||||
|
position: 'absolute',
|
||||||
|
bottom: '5px',
|
||||||
|
left: 0,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
New Folder
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,25 @@
|
|||||||
|
interface Props {
|
||||||
|
onDelete: () => Promise<void>
|
||||||
|
}
|
||||||
|
|
||||||
|
export function CafeReactFsDelete({ onDelete }: Props) {
|
||||||
|
return (
|
||||||
|
<img
|
||||||
|
alt="Delete"
|
||||||
|
style={{
|
||||||
|
position: 'absolute',
|
||||||
|
top: '2px',
|
||||||
|
right: '2px',
|
||||||
|
width: '20px',
|
||||||
|
height: '20px',
|
||||||
|
cursor: 'pointer',
|
||||||
|
zIndex: 1,
|
||||||
|
}}
|
||||||
|
src="data:image/svg+xml,%3C%3Fxml%20version%3D%221.0%22%20encoding%3D%22UTF-8%22%3F%3E%3Csvg%20width%3D%22128%22%20height%3D%22128%22%20viewBox%3D%220%200%20128%20128%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Ccircle%20cx%3D%2264%22%20cy%3D%2264%22%20r%3D%2264%22%20fill%3D%22%231F2D3D%22%20stroke%3D%22none%22%20stroke-width%3D%220%22%20%20%2F%3E%3Cpath%20d%3D%22M%2032%2064%20l%2064%200%22%20stroke%3D%22%23B83B5E%22%20stroke-width%3D%2220%22%20fill%3D%22none%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%20%20%2F%3E%3C%2Fsvg%3E"
|
||||||
|
onClick={event => {
|
||||||
|
onDelete()
|
||||||
|
event.stopPropagation()
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,51 @@
|
|||||||
|
import { useState } from 'react'
|
||||||
|
import { CafeReactFsDelete } from './CafeReactFsDelete'
|
||||||
|
import { CafeReactFsLoading } from './CafeReactFsLoading'
|
||||||
|
import { CafeReactFsName } from './CafeReactFsName'
|
||||||
|
import { VirtualDirectory } from './CafeReactType'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
directory: VirtualDirectory
|
||||||
|
enterDirectory: (name: string) => void
|
||||||
|
deleteDirectory: (name: string) => Promise<void>
|
||||||
|
backgroundColor: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function CafeReactFsDirectory({ directory, enterDirectory, deleteDirectory, backgroundColor }: Props) {
|
||||||
|
const [hovered, setHovered] = useState(false)
|
||||||
|
const [loading, setLoading] = useState(false)
|
||||||
|
|
||||||
|
function proxyDelete() {
|
||||||
|
setLoading(true)
|
||||||
|
|
||||||
|
return deleteDirectory(directory.name).finally(() => setLoading(false))
|
||||||
|
}
|
||||||
|
|
||||||
|
if (loading) {
|
||||||
|
return <CafeReactFsLoading backgroundColor={backgroundColor} />
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
width: '80px',
|
||||||
|
height: '80px',
|
||||||
|
position: 'relative',
|
||||||
|
background: backgroundColor,
|
||||||
|
borderRadius: '2px',
|
||||||
|
cursor: 'pointer',
|
||||||
|
}}
|
||||||
|
onClick={() => enterDirectory(directory.name)}
|
||||||
|
onMouseEnter={() => setHovered(true)}
|
||||||
|
onMouseLeave={() => setHovered(false)}
|
||||||
|
>
|
||||||
|
{hovered && <CafeReactFsDelete onDelete={proxyDelete} />}
|
||||||
|
<img
|
||||||
|
alt="Directory"
|
||||||
|
src="data:image/svg+xml,%3C%3Fxml%20version%3D%221.0%22%20encoding%3D%22UTF-8%22%3F%3E%3Csvg%20width%3D%22512%22%20height%3D%22512%22%20viewBox%3D%220%200%20512%20512%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20d%3D%22M%20106%20131%20l%20100%200%20l%2025%2025%20l%20175%200%20l%200%20200%20l%20-300%200%20z%22%20stroke%3D%22%231F2D3D%22%20stroke-width%3D%2220%22%20fill%3D%22none%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%20%20%2F%3E%3C%2Fsvg%3E"
|
||||||
|
style={{ width: '64px', height: '64px', position: 'absolute', left: '8px', top: 0 }}
|
||||||
|
/>
|
||||||
|
<CafeReactFsName name={directory.name} />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,53 @@
|
|||||||
|
import { useState } from 'react'
|
||||||
|
import { CafeReactFsDelete } from './CafeReactFsDelete'
|
||||||
|
import { CafeReactFsLoading } from './CafeReactFsLoading'
|
||||||
|
import { CafeReactFsName } from './CafeReactFsName'
|
||||||
|
import { VirtualFile } from './CafeReactType'
|
||||||
|
import { joinUrl } from './Utility'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
path: string
|
||||||
|
file: VirtualFile
|
||||||
|
download: (path: string) => Promise<void>
|
||||||
|
deleteFile: (path: string) => Promise<void>
|
||||||
|
backgroundColor: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function CafeReactFsFile({ path, file, download, deleteFile, backgroundColor }: Props) {
|
||||||
|
const [hovered, setHovered] = useState(false)
|
||||||
|
const [loading, setLoading] = useState(false)
|
||||||
|
|
||||||
|
if (loading) {
|
||||||
|
return <CafeReactFsLoading backgroundColor={backgroundColor} />
|
||||||
|
}
|
||||||
|
|
||||||
|
function proxyDelete() {
|
||||||
|
setLoading(true)
|
||||||
|
|
||||||
|
return deleteFile(joinUrl(path, file.name)).finally(() => setLoading(false))
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
onClick={() => download(joinUrl(path, file.name))}
|
||||||
|
onMouseEnter={() => setHovered(true)}
|
||||||
|
onMouseLeave={() => setHovered(false)}
|
||||||
|
style={{
|
||||||
|
width: '80px',
|
||||||
|
height: '80px',
|
||||||
|
position: 'relative',
|
||||||
|
background: backgroundColor,
|
||||||
|
borderRadius: '2px',
|
||||||
|
cursor: 'pointer',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{hovered && <CafeReactFsDelete onDelete={proxyDelete} />}
|
||||||
|
<img
|
||||||
|
alt="File"
|
||||||
|
style={{ width: '64px', height: '64px', position: 'absolute', left: '8px', top: 0 }}
|
||||||
|
src="data:image/svg+xml,%3C%3Fxml%20version%3D%221.0%22%20encoding%3D%22UTF-8%22%3F%3E%3Csvg%20width%3D%22512%22%20height%3D%22512%22%20viewBox%3D%220%200%20512%20512%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20d%3D%22M%20156%20131%20l%20150%200%20l%2050%2050%20l%200%20200%20l%20-200%200%20z%22%20stroke%3D%22%231F2D3D%22%20stroke-width%3D%2220%22%20fill%3D%22none%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%20%20%2F%3E%3Cpath%20d%3D%22M%20306%20131%20l%200%2050%20l%2050%200%22%20stroke%3D%22%231F2D3D%22%20stroke-width%3D%2220%22%20fill%3D%22none%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%20%20%2F%3E%3C%2Fsvg%3E"
|
||||||
|
/>
|
||||||
|
<CafeReactFsName name={file.name} />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,48 @@
|
|||||||
|
import { CafeReactFsDirectory } from './CafeReactFsDirectory'
|
||||||
|
import { CafeReactFsFile } from './CafeReactFsFile'
|
||||||
|
import { FsItem, isVirtualDirectory, isVirtualFile } from './CafeReactType'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
path: string
|
||||||
|
item: FsItem
|
||||||
|
download: (path: string) => Promise<void>
|
||||||
|
enterDirectory: (name: string) => void
|
||||||
|
onDeleteFile: (path: string) => Promise<void>
|
||||||
|
onDeleteDirectory: (path: string) => Promise<void>
|
||||||
|
backgroundColor: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function CafeReactFsItem({
|
||||||
|
path,
|
||||||
|
item,
|
||||||
|
download,
|
||||||
|
enterDirectory,
|
||||||
|
onDeleteFile,
|
||||||
|
onDeleteDirectory,
|
||||||
|
backgroundColor,
|
||||||
|
}: Props) {
|
||||||
|
if (isVirtualFile(item)) {
|
||||||
|
return (
|
||||||
|
<CafeReactFsFile
|
||||||
|
path={path}
|
||||||
|
file={item}
|
||||||
|
download={download}
|
||||||
|
deleteFile={onDeleteFile}
|
||||||
|
backgroundColor={backgroundColor}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isVirtualDirectory(item)) {
|
||||||
|
return (
|
||||||
|
<CafeReactFsDirectory
|
||||||
|
directory={item}
|
||||||
|
enterDirectory={enterDirectory}
|
||||||
|
deleteDirectory={onDeleteDirectory}
|
||||||
|
backgroundColor={backgroundColor}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return null
|
||||||
|
}
|
||||||
@@ -0,0 +1,52 @@
|
|||||||
|
import { CSSProperties } from 'react'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
backgroundColor: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function CafeReactFsLoading({ backgroundColor }: Props) {
|
||||||
|
const spinnerStyle = {
|
||||||
|
width: '80px',
|
||||||
|
height: '80px',
|
||||||
|
borderRadius: '2px',
|
||||||
|
position: 'relative',
|
||||||
|
background: backgroundColor,
|
||||||
|
} as CSSProperties
|
||||||
|
|
||||||
|
const bounceStyle = {
|
||||||
|
width: '32px',
|
||||||
|
height: '32px',
|
||||||
|
borderRadius: '50%',
|
||||||
|
backgroundColor: '#333',
|
||||||
|
top: '24px',
|
||||||
|
left: '24px',
|
||||||
|
opacity: 0.6,
|
||||||
|
position: 'absolute',
|
||||||
|
animation: 'bounce 2.0s infinite ease-in-out',
|
||||||
|
} as CSSProperties
|
||||||
|
|
||||||
|
const bounceStyle2 = {
|
||||||
|
...bounceStyle,
|
||||||
|
animationDelay: '-1.0s',
|
||||||
|
}
|
||||||
|
|
||||||
|
const keyframes = `
|
||||||
|
@keyframes bounce {
|
||||||
|
0%, 100% {
|
||||||
|
transform: scale(0.0);
|
||||||
|
} 50% {
|
||||||
|
transform: scale(1.0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<style>{keyframes}</style>
|
||||||
|
<div style={spinnerStyle}>
|
||||||
|
<div style={bounceStyle}></div>
|
||||||
|
<div style={bounceStyle2}></div>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,25 @@
|
|||||||
|
interface Props {
|
||||||
|
name: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function CafeReactFsName({ name }: Props) {
|
||||||
|
const shortName = name.length > 10 ? name.slice(0, 10) + '...' : name
|
||||||
|
|
||||||
|
return (
|
||||||
|
<p
|
||||||
|
title={name}
|
||||||
|
style={{
|
||||||
|
margin: 0,
|
||||||
|
fontFamily: 'sans-serif',
|
||||||
|
fontSize: '12px',
|
||||||
|
textAlign: 'center',
|
||||||
|
width: '80px',
|
||||||
|
position: 'absolute',
|
||||||
|
bottom: '5px',
|
||||||
|
left: 0,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{shortName}
|
||||||
|
</p>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,44 @@
|
|||||||
|
import { joinUrl } from './Utility'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
pathParts: string[]
|
||||||
|
jumpToDirectory: (fullPath: string) => void
|
||||||
|
backgroundColor: string
|
||||||
|
rootAlias?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function CafeReactFsPath({ pathParts, jumpToDirectory, backgroundColor, rootAlias }: Props) {
|
||||||
|
const absolutePaths: string[] = []
|
||||||
|
|
||||||
|
for (const pathPart of pathParts) {
|
||||||
|
if (absolutePaths.length === 0) {
|
||||||
|
absolutePaths.push(pathPart)
|
||||||
|
} else {
|
||||||
|
absolutePaths.push(joinUrl(absolutePaths[absolutePaths.length - 1], pathPart))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div style={{ display: 'flex', flexDirection: 'row', flexWrap: 'wrap', gap: '4px' }}>
|
||||||
|
{pathParts.map((part, index) => (
|
||||||
|
<button
|
||||||
|
key={index}
|
||||||
|
style={{
|
||||||
|
background: backgroundColor,
|
||||||
|
borderRadius: '2px',
|
||||||
|
cursor: 'pointer',
|
||||||
|
outline: 'none',
|
||||||
|
border: 'none',
|
||||||
|
fontSize: '20px',
|
||||||
|
padding: '4px 16px',
|
||||||
|
}}
|
||||||
|
onClick={() => {
|
||||||
|
jumpToDirectory(absolutePaths[index])
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{rootAlias && part === '/' ? rootAlias : part}
|
||||||
|
</button>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,40 @@
|
|||||||
|
interface Props {
|
||||||
|
backgroundColor: string
|
||||||
|
onSync: () => Promise<void>
|
||||||
|
}
|
||||||
|
|
||||||
|
export function CafeReactFsSync({ backgroundColor, onSync }: Props) {
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
width: '80px',
|
||||||
|
height: '80px',
|
||||||
|
position: 'relative',
|
||||||
|
background: backgroundColor,
|
||||||
|
borderRadius: '2px',
|
||||||
|
cursor: 'pointer',
|
||||||
|
}}
|
||||||
|
onClick={onSync}
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
alt="Sync"
|
||||||
|
style={{ width: '64px', height: '64px', position: 'absolute', left: '8px', top: 0 }}
|
||||||
|
src="data:image/svg+xml,%3C%3Fxml%20version%3D%221.0%22%20encoding%3D%22UTF-8%22%3F%3E%3Csvg%20width%3D%22512%22%20height%3D%22512%22%20viewBox%3D%220%200%20512%20512%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20d%3D%22M%20156%20156%20l%20250%200%20l%200%20150%22%20stroke%3D%22%231F2D3D%22%20stroke-width%3D%2220%22%20fill%3D%22none%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%20%20%2F%3E%3Cpath%20d%3D%22M%20356%20256%20l%2050%2050%20l%2050%20-50%22%20stroke%3D%22%231F2D3D%22%20stroke-width%3D%2220%22%20fill%3D%22none%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%20%20%2F%3E%3Cpath%20d%3D%22M%20356%20356%20l%20-250%200%20l%200%20-150%22%20stroke%3D%22%231F2D3D%22%20stroke-width%3D%2220%22%20fill%3D%22none%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%20%20%2F%3E%3Cpath%20d%3D%22M%20156%20256%20l%20-50%20-50%20l%20-50%2050%22%20stroke%3D%22%231F2D3D%22%20stroke-width%3D%2220%22%20fill%3D%22none%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%20%20%2F%3E%3C%2Fsvg%3E"
|
||||||
|
/>
|
||||||
|
<p
|
||||||
|
style={{
|
||||||
|
margin: 0,
|
||||||
|
fontFamily: 'sans-serif',
|
||||||
|
fontSize: '12px',
|
||||||
|
textAlign: 'center',
|
||||||
|
width: '80px',
|
||||||
|
position: 'absolute',
|
||||||
|
bottom: '5px',
|
||||||
|
left: 0,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Sync
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,54 @@
|
|||||||
|
import { useState } from 'react'
|
||||||
|
import { CafeReactFsLoading } from './CafeReactFsLoading'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
onUpload: () => Promise<void>
|
||||||
|
backgroundColor: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function CafeReactFsUpload({ onUpload, backgroundColor }: Props) {
|
||||||
|
const [uploading, setUploading] = useState(false)
|
||||||
|
|
||||||
|
function proxyUpload() {
|
||||||
|
setUploading(true)
|
||||||
|
onUpload().finally(() => setUploading(false))
|
||||||
|
}
|
||||||
|
|
||||||
|
if (uploading) {
|
||||||
|
return <CafeReactFsLoading backgroundColor={backgroundColor} />
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
width: '80px',
|
||||||
|
height: '80px',
|
||||||
|
position: 'relative',
|
||||||
|
background: backgroundColor,
|
||||||
|
borderRadius: '2px',
|
||||||
|
cursor: 'pointer',
|
||||||
|
}}
|
||||||
|
onClick={proxyUpload}
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
alt="Upload"
|
||||||
|
style={{ width: '64px', height: '64px', position: 'absolute', left: '8px', top: 0 }}
|
||||||
|
src="data:image/svg+xml,%3C%3Fxml%20version%3D%221.0%22%20encoding%3D%22UTF-8%22%3F%3E%3Csvg%20width%3D%22512%22%20height%3D%22512%22%20viewBox%3D%220%200%20512%20512%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20d%3D%22M%20156%20131%20l%20150%200%20l%2050%2050%20l%200%20200%20l%20-200%200%20z%22%20stroke%3D%22%231F2D3D%22%20stroke-width%3D%2220%22%20fill%3D%22none%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%20%20%2F%3E%3Cpath%20d%3D%22M%20306%20131%20l%200%2050%20l%2050%200%22%20stroke%3D%22%231F2D3D%22%20stroke-width%3D%2220%22%20fill%3D%22none%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%20%20%2F%3E%3Cpath%20d%3D%22M%20206%20256%20l%20100%200%22%20stroke%3D%22%23B83B5E%22%20stroke-width%3D%2220%22%20fill%3D%22none%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%20%20%2F%3E%3Cpath%20d%3D%22M%20256%20206%20l%200%20100%22%20stroke%3D%22%23B83B5E%22%20stroke-width%3D%2220%22%20fill%3D%22none%22%20stroke-linecap%3D%22round%22%20stroke-linejoin%3D%22round%22%20%20%2F%3E%3C%2Fsvg%3E"
|
||||||
|
/>
|
||||||
|
<p
|
||||||
|
style={{
|
||||||
|
margin: 0,
|
||||||
|
fontFamily: 'sans-serif',
|
||||||
|
fontSize: '12px',
|
||||||
|
textAlign: 'center',
|
||||||
|
width: '80px',
|
||||||
|
position: 'absolute',
|
||||||
|
bottom: '5px',
|
||||||
|
left: 0,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Upload
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
export enum FsItemType {
|
||||||
|
FILE = 'file',
|
||||||
|
DIRECTORY = 'directory',
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface VirtualFile {
|
||||||
|
id: string | number
|
||||||
|
name: string
|
||||||
|
$type: FsItemType.FILE
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface VirtualDirectory {
|
||||||
|
id: string | number
|
||||||
|
name: string
|
||||||
|
$type: FsItemType.DIRECTORY
|
||||||
|
}
|
||||||
|
|
||||||
|
export type FsItem = VirtualFile | VirtualDirectory
|
||||||
|
|
||||||
|
export function isVirtualFile(item: FsItem): item is VirtualFile {
|
||||||
|
return item.$type === FsItemType.FILE
|
||||||
|
}
|
||||||
|
|
||||||
|
export function isVirtualDirectory(item: FsItem): item is VirtualDirectory {
|
||||||
|
return item.$type === FsItemType.DIRECTORY
|
||||||
|
}
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
export function joinUrl(...parts: unknown[]): string {
|
||||||
|
return parts
|
||||||
|
.filter(x => x)
|
||||||
|
.join('/')
|
||||||
|
.replace(/(?<!:)\/+/g, '/')
|
||||||
|
}
|
||||||
+14
-4
@@ -2,8 +2,10 @@ import { ReactElement, useContext } from 'react'
|
|||||||
import { Route, Routes } from 'react-router-dom'
|
import { Route, Routes } from 'react-router-dom'
|
||||||
import { AccountChequebook } from './pages/account/chequebook/AccountChequebook'
|
import { AccountChequebook } from './pages/account/chequebook/AccountChequebook'
|
||||||
import { AccountFeeds } from './pages/account/feeds/AccountFeeds'
|
import { AccountFeeds } from './pages/account/feeds/AccountFeeds'
|
||||||
|
import { AccountStaking } from './pages/account/staking/AccountStaking'
|
||||||
import { AccountStamps } from './pages/account/stamps/AccountStamps'
|
import { AccountStamps } from './pages/account/stamps/AccountStamps'
|
||||||
import { AccountWallet } from './pages/account/wallet/AccountWallet'
|
import { AccountWallet } from './pages/account/wallet/AccountWallet'
|
||||||
|
import FDP from './pages/fdp'
|
||||||
import CreateNewFeed from './pages/feeds/CreateNewFeed'
|
import CreateNewFeed from './pages/feeds/CreateNewFeed'
|
||||||
import { FeedSubpage } from './pages/feeds/FeedSubpage'
|
import { FeedSubpage } from './pages/feeds/FeedSubpage'
|
||||||
import UpdateFeed from './pages/feeds/UpdateFeed'
|
import UpdateFeed from './pages/feeds/UpdateFeed'
|
||||||
@@ -14,10 +16,11 @@ import { UploadLander } from './pages/files/UploadLander'
|
|||||||
import GiftCards from './pages/gift-code'
|
import GiftCards from './pages/gift-code'
|
||||||
import Info from './pages/info'
|
import Info from './pages/info'
|
||||||
import LightModeRestart from './pages/restart/LightModeRestart'
|
import LightModeRestart from './pages/restart/LightModeRestart'
|
||||||
import TopUp from './pages/top-up'
|
|
||||||
import Settings from './pages/settings'
|
import Settings from './pages/settings'
|
||||||
import { CreatePostageStampPage } from './pages/stamps/CreatePostageStampPage'
|
import { CreatePostageStampPage } from './pages/stamps/CreatePostageStampAdvancedPage'
|
||||||
|
import { CreatePostageStampBasicPage } from './pages/stamps/CreatePostageStampStandardPage'
|
||||||
import Status from './pages/status'
|
import Status from './pages/status'
|
||||||
|
import TopUp from './pages/top-up'
|
||||||
import { BankCardTopUpIndex } from './pages/top-up/BankCardTopUpIndex'
|
import { BankCardTopUpIndex } from './pages/top-up/BankCardTopUpIndex'
|
||||||
import { CryptoTopUpIndex } from './pages/top-up/CryptoTopUpIndex'
|
import { CryptoTopUpIndex } from './pages/top-up/CryptoTopUpIndex'
|
||||||
import { GiftCardFund } from './pages/top-up/GiftCardFund'
|
import { GiftCardFund } from './pages/top-up/GiftCardFund'
|
||||||
@@ -45,12 +48,15 @@ export enum ROUTES {
|
|||||||
ACCOUNT_WALLET = '/account/wallet',
|
ACCOUNT_WALLET = '/account/wallet',
|
||||||
ACCOUNT_CHEQUEBOOK = '/account/chequebook',
|
ACCOUNT_CHEQUEBOOK = '/account/chequebook',
|
||||||
ACCOUNT_STAMPS = '/account/stamps',
|
ACCOUNT_STAMPS = '/account/stamps',
|
||||||
ACCOUNT_STAMPS_NEW = '/account/stamps/new',
|
ACCOUNT_STAMPS_NEW_STANDARD = '/account/stamps/new',
|
||||||
|
ACCOUNT_STAMPS_NEW_ADVANCED = '/account/stamps/new/advanced',
|
||||||
ACCOUNT_FEEDS = '/account/feeds',
|
ACCOUNT_FEEDS = '/account/feeds',
|
||||||
ACCOUNT_FEEDS_NEW = '/account/feeds/new',
|
ACCOUNT_FEEDS_NEW = '/account/feeds/new',
|
||||||
ACCOUNT_FEEDS_UPDATE = '/account/feeds/update/:hash',
|
ACCOUNT_FEEDS_UPDATE = '/account/feeds/update/:hash',
|
||||||
ACCOUNT_FEEDS_VIEW = '/account/feeds/:uuid',
|
ACCOUNT_FEEDS_VIEW = '/account/feeds/:uuid',
|
||||||
ACCOUNT_INVITATIONS = '/account/invitations',
|
ACCOUNT_INVITATIONS = '/account/invitations',
|
||||||
|
ACCOUNT_STAKING = '/account/staking',
|
||||||
|
FDP = '/fdp',
|
||||||
}
|
}
|
||||||
|
|
||||||
export const ACCOUNT_TABS = [
|
export const ACCOUNT_TABS = [
|
||||||
@@ -58,6 +64,7 @@ export const ACCOUNT_TABS = [
|
|||||||
ROUTES.ACCOUNT_CHEQUEBOOK,
|
ROUTES.ACCOUNT_CHEQUEBOOK,
|
||||||
ROUTES.ACCOUNT_STAMPS,
|
ROUTES.ACCOUNT_STAMPS,
|
||||||
ROUTES.ACCOUNT_FEEDS,
|
ROUTES.ACCOUNT_FEEDS,
|
||||||
|
ROUTES.ACCOUNT_STAKING,
|
||||||
]
|
]
|
||||||
|
|
||||||
const BaseRouter = (): ReactElement => {
|
const BaseRouter = (): ReactElement => {
|
||||||
@@ -83,11 +90,14 @@ const BaseRouter = (): ReactElement => {
|
|||||||
<Route path={ROUTES.ACCOUNT_WALLET} element={<AccountWallet />} />
|
<Route path={ROUTES.ACCOUNT_WALLET} element={<AccountWallet />} />
|
||||||
<Route path={ROUTES.ACCOUNT_CHEQUEBOOK} element={<AccountChequebook />} />
|
<Route path={ROUTES.ACCOUNT_CHEQUEBOOK} element={<AccountChequebook />} />
|
||||||
<Route path={ROUTES.ACCOUNT_STAMPS} element={<AccountStamps />} />
|
<Route path={ROUTES.ACCOUNT_STAMPS} element={<AccountStamps />} />
|
||||||
<Route path={ROUTES.ACCOUNT_STAMPS_NEW} element={<CreatePostageStampPage />} />
|
<Route path={ROUTES.ACCOUNT_STAMPS_NEW_STANDARD} element={<CreatePostageStampBasicPage />} />
|
||||||
|
<Route path={ROUTES.ACCOUNT_STAMPS_NEW_ADVANCED} element={<CreatePostageStampPage />} />
|
||||||
<Route path={ROUTES.ACCOUNT_FEEDS} element={<AccountFeeds />} />
|
<Route path={ROUTES.ACCOUNT_FEEDS} element={<AccountFeeds />} />
|
||||||
<Route path={ROUTES.ACCOUNT_FEEDS_NEW} element={<CreateNewFeed />} />
|
<Route path={ROUTES.ACCOUNT_FEEDS_NEW} element={<CreateNewFeed />} />
|
||||||
<Route path={ROUTES.ACCOUNT_FEEDS_UPDATE} element={<UpdateFeed />} />
|
<Route path={ROUTES.ACCOUNT_FEEDS_UPDATE} element={<UpdateFeed />} />
|
||||||
<Route path={ROUTES.ACCOUNT_FEEDS_VIEW} element={<FeedSubpage />} />
|
<Route path={ROUTES.ACCOUNT_FEEDS_VIEW} element={<FeedSubpage />} />
|
||||||
|
<Route path={ROUTES.ACCOUNT_STAKING} element={<AccountStaking />} />
|
||||||
|
<Route path={ROUTES.FDP} element={<FDP />} />
|
||||||
{isDesktop && <Route path={ROUTES.ACCOUNT_INVITATIONS} element={<GiftCards />} />}
|
{isDesktop && <Route path={ROUTES.ACCOUNT_INVITATIONS} element={<GiftCards />} />}
|
||||||
</Routes>
|
</Routes>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -159,6 +159,23 @@ const componentsOverrides = (theme: Theme) => ({
|
|||||||
backgroundColor: 'transparent',
|
backgroundColor: 'transparent',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
MuiSlider: {
|
||||||
|
root: {
|
||||||
|
'& .MuiSlider-valueLabel': {
|
||||||
|
top: '-27px',
|
||||||
|
'& span': {
|
||||||
|
height: '20px',
|
||||||
|
borderRadius: '0px',
|
||||||
|
transform: 'none',
|
||||||
|
'& span': {
|
||||||
|
display: 'flex',
|
||||||
|
alignItems: 'center',
|
||||||
|
transform: 'none',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
const propsOverrides = {
|
const propsOverrides = {
|
||||||
|
|||||||
@@ -6,8 +6,6 @@ import { getJson, postJson } from './net'
|
|||||||
|
|
||||||
export interface BeeConfig {
|
export interface BeeConfig {
|
||||||
'api-addr': string
|
'api-addr': string
|
||||||
'debug-api-addr': string
|
|
||||||
'debug-api-enable': boolean
|
|
||||||
password: string
|
password: string
|
||||||
'swap-enable': boolean
|
'swap-enable': boolean
|
||||||
'swap-initial-deposit': bigint
|
'swap-initial-deposit': bigint
|
||||||
@@ -17,7 +15,7 @@ export interface BeeConfig {
|
|||||||
'resolver-options': string
|
'resolver-options': string
|
||||||
'use-postage-snapshot': boolean
|
'use-postage-snapshot': boolean
|
||||||
'data-dir': string
|
'data-dir': string
|
||||||
'swap-endpoint'?: string
|
'blockchain-rpc-endpoint'?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function getBzzPriceAsDai(desktopUrl: string): Promise<Token> {
|
export async function getBzzPriceAsDai(desktopUrl: string): Promise<Token> {
|
||||||
@@ -29,13 +27,13 @@ export async function getBzzPriceAsDai(desktopUrl: string): Promise<Token> {
|
|||||||
export function upgradeToLightNode(desktopUrl: string, rpcProvider: string): Promise<BeeConfig> {
|
export function upgradeToLightNode(desktopUrl: string, rpcProvider: string): Promise<BeeConfig> {
|
||||||
return updateDesktopConfiguration(desktopUrl, {
|
return updateDesktopConfiguration(desktopUrl, {
|
||||||
'swap-enable': true,
|
'swap-enable': true,
|
||||||
'swap-endpoint': rpcProvider,
|
'blockchain-rpc-endpoint': rpcProvider,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function setJsonRpcInDesktop(desktopUrl: string, value: string): Promise<void> {
|
export async function setJsonRpcInDesktop(desktopUrl: string, value: string): Promise<void> {
|
||||||
await updateDesktopConfiguration(desktopUrl, {
|
await updateDesktopConfiguration(desktopUrl, {
|
||||||
'swap-endpoint': value,
|
'blockchain-rpc-endpoint': value,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user