Compare commits
26 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| dda00a0f16 | |||
| bbc6689a32 | |||
| 12345dbd2c | |||
| 63a1123ef3 | |||
| 3bfdf224d3 | |||
| 0ab0efd974 | |||
| 8b7ec0756e | |||
| 062f1c28f4 | |||
| 99fe1288f4 | |||
| 8622142c3d | |||
| 1b7c048c96 | |||
| 825791f2c7 | |||
| 83af0d4d76 | |||
| 75b7b12c46 | |||
| 6286c6e2fd | |||
| 68142e7535 | |||
| 2c4611044c | |||
| 11fe5cfbf8 | |||
| 793fac514a | |||
| 4578cda0ae | |||
| 758ba5593c | |||
| 0cc67b1c77 | |||
| ee80e7faaf | |||
| e695813d24 | |||
| a5c6b31384 | |||
| 4cbfa9c0fa |
@@ -9,12 +9,12 @@ jobs:
|
|||||||
outputs:
|
outputs:
|
||||||
matrix: ${{ steps.set-matrix.outputs.matrix }}
|
matrix: ${{ steps.set-matrix.outputs.matrix }}
|
||||||
steps:
|
steps:
|
||||||
# creates the matrix strategy from nightly_build_release_matrix.json
|
# creates the matrix strategy from nightly_build_matrix_includes.json
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v3
|
||||||
- id: set-matrix
|
- id: set-matrix
|
||||||
uses: JoshuaTheMiller/conditional-build-matrix@main
|
uses: JoshuaTheMiller/conditional-build-matrix@main
|
||||||
with:
|
with:
|
||||||
inputFile: '.github/workflows/nightly_build_release_matrix.json'
|
inputFile: '.github/workflows/nightly_build_matrix_includes.json'
|
||||||
filter: '[?runOnEvent==`${{ github.event_name }}` || runOnEvent==`always`]'
|
filter: '[?runOnEvent==`${{ github.event_name }}` || runOnEvent==`always`]'
|
||||||
get_release:
|
get_release:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
@@ -174,7 +174,7 @@ jobs:
|
|||||||
args: --manifest-path nym-wallet/Cargo.toml --workspace --all-targets -- -D warnings
|
args: --manifest-path nym-wallet/Cargo.toml --workspace --all-targets -- -D warnings
|
||||||
|
|
||||||
notification:
|
notification:
|
||||||
needs: build
|
needs: [build,get_release]
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Collect jobs status
|
- name: Collect jobs status
|
||||||
@@ -192,7 +192,7 @@ jobs:
|
|||||||
NYM_PROJECT_NAME: "Nym nightly build on latest release"
|
NYM_PROJECT_NAME: "Nym nightly build on latest release"
|
||||||
GITHUB_TOKEN: "${{ secrets.GITHUB_TOKEN }}"
|
GITHUB_TOKEN: "${{ secrets.GITHUB_TOKEN }}"
|
||||||
GIT_COMMIT_MESSAGE: "${{ github.event.head_commit.message }}"
|
GIT_COMMIT_MESSAGE: "${{ github.event.head_commit.message }}"
|
||||||
GIT_BRANCH: "${GITHUB_REF##*/}"
|
GIT_BRANCH: "https://github.com/nymtech/nym/tree/${{needs.get_release.outputs.output1}}"
|
||||||
KEYBASE_NYMBOT_USERNAME: "${{ secrets.KEYBASE_NYMBOT_USERNAME }}"
|
KEYBASE_NYMBOT_USERNAME: "${{ secrets.KEYBASE_NYMBOT_USERNAME }}"
|
||||||
KEYBASE_NYMBOT_PAPERKEY: "${{ secrets.KEYBASE_NYMBOT_PAPERKEY }}"
|
KEYBASE_NYMBOT_PAPERKEY: "${{ secrets.KEYBASE_NYMBOT_PAPERKEY }}"
|
||||||
KEYBASE_NYMBOT_TEAM: "${{ secrets.KEYBASE_NYMBOT_TEAM }}"
|
KEYBASE_NYMBOT_TEAM: "${{ secrets.KEYBASE_NYMBOT_TEAM }}"
|
||||||
|
|||||||
@@ -0,0 +1,203 @@
|
|||||||
|
name: Nightly builds on second latest release
|
||||||
|
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
- cron: '24 2 * * *'
|
||||||
|
jobs:
|
||||||
|
matrix_prep:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
outputs:
|
||||||
|
matrix: ${{ steps.set-matrix.outputs.matrix }}
|
||||||
|
steps:
|
||||||
|
# creates the matrix strategy from nightly_build_matrix_includes.json
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- id: set-matrix
|
||||||
|
uses: JoshuaTheMiller/conditional-build-matrix@main
|
||||||
|
with:
|
||||||
|
inputFile: '.github/workflows/nightly_build_matrix_includes.json'
|
||||||
|
filter: '[?runOnEvent==`${{ github.event_name }}` || runOnEvent==`always`]'
|
||||||
|
get_release:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs: matrix_prep
|
||||||
|
outputs:
|
||||||
|
output1: ${{ steps.step2.outputs.latest_release }}
|
||||||
|
steps:
|
||||||
|
- name: Check out repository code
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
- name: Fetch all branches
|
||||||
|
run: git fetch --all
|
||||||
|
- name: Set output variable to latest release branch
|
||||||
|
id: step2
|
||||||
|
run: echo "latest_release=$(git branch -r | grep -E 'release/v[0-9]+\.[0-9]+\.[0-9]+' | tail -n 2 | head -n 1 | sed 's/ origin\///')" >> $GITHUB_OUTPUT
|
||||||
|
build:
|
||||||
|
needs: [get_release,matrix_prep]
|
||||||
|
strategy:
|
||||||
|
matrix: ${{fromJson(needs.matrix_prep.outputs.matrix)}}
|
||||||
|
runs-on: ${{ matrix.os }}
|
||||||
|
continue-on-error: ${{ matrix.rust == 'nightly' || matrix.rust == 'beta' || matrix.rust == 'stable' }}
|
||||||
|
steps:
|
||||||
|
- name: Install Dependencies (Linux)
|
||||||
|
run: sudo apt-get update && sudo apt-get install libwebkit2gtk-4.0-dev build-essential curl wget libssl-dev libgtk-3-dev libudev-dev squashfs-tools
|
||||||
|
if: matrix.os == 'ubuntu-latest'
|
||||||
|
|
||||||
|
- name: Check out latest release branch
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
with:
|
||||||
|
ref: ${{needs.get_release.outputs.output1}}
|
||||||
|
|
||||||
|
- name: Install rust toolchain
|
||||||
|
uses: actions-rs/toolchain@v1
|
||||||
|
with:
|
||||||
|
profile: minimal
|
||||||
|
toolchain: ${{ matrix.rust }}
|
||||||
|
override: true
|
||||||
|
components: rustfmt, clippy
|
||||||
|
|
||||||
|
- name: Build all binaries
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
with:
|
||||||
|
command: build
|
||||||
|
args: --workspace
|
||||||
|
|
||||||
|
- name: Reclaim some disk space (because Windows is being annoying)
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
if: ${{ matrix.os == 'windows-latest' }}
|
||||||
|
with:
|
||||||
|
command: clean
|
||||||
|
|
||||||
|
- name: Run all tests
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
with:
|
||||||
|
command: test
|
||||||
|
args: --workspace
|
||||||
|
|
||||||
|
- name: Reclaim some disk space (because Windows is being annoying)
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
if: ${{ matrix.os == 'windows-latest' }}
|
||||||
|
with:
|
||||||
|
command: clean
|
||||||
|
|
||||||
|
- name: Run expensive tests
|
||||||
|
if: github.ref == 'refs/heads/develop' || github.event.pull_request.base.ref == 'develop' || github.event.pull_request.base.ref == 'master'
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
with:
|
||||||
|
command: test
|
||||||
|
args: --workspace --all-features -- --ignored
|
||||||
|
|
||||||
|
- name: Check formatting
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
with:
|
||||||
|
command: fmt
|
||||||
|
args: --all -- --check
|
||||||
|
|
||||||
|
- name: Reclaim some disk space (because Windows is being annoying)
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
if: ${{ matrix.os == 'windows-latest' }}
|
||||||
|
with:
|
||||||
|
command: clean
|
||||||
|
|
||||||
|
- uses: actions-rs/clippy-check@v1
|
||||||
|
name: Clippy checks
|
||||||
|
with:
|
||||||
|
token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
args: --all-features
|
||||||
|
|
||||||
|
- name: Run clippy
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
if: ${{ matrix.rust != 'nightly' }}
|
||||||
|
with:
|
||||||
|
command: clippy
|
||||||
|
args: --workspace --all-targets -- -D warnings
|
||||||
|
|
||||||
|
- name: Reclaim some disk space
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
if: ${{ matrix.os == 'windows-latest' || matrix.os == 'ubuntu-latest' }}
|
||||||
|
with:
|
||||||
|
command: clean
|
||||||
|
|
||||||
|
# COCONUT stuff
|
||||||
|
- name: Build all binaries with coconut enabled
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
with:
|
||||||
|
command: build
|
||||||
|
args: --workspace --features=coconut
|
||||||
|
|
||||||
|
- name: Reclaim some disk space (because Windows is being annoying)
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
if: ${{ matrix.os == 'windows-latest' }}
|
||||||
|
with:
|
||||||
|
command: clean
|
||||||
|
|
||||||
|
- name: Run all tests with coconut enabled
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
with:
|
||||||
|
command: test
|
||||||
|
args: --workspace --features=coconut
|
||||||
|
|
||||||
|
- name: Reclaim some disk space (because Windows is being annoying)
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
if: ${{ matrix.os == 'windows-latest' }}
|
||||||
|
with:
|
||||||
|
command: clean
|
||||||
|
|
||||||
|
- name: Run clippy with coconut enabled
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
if: ${{ matrix.rust != 'nightly' }}
|
||||||
|
with:
|
||||||
|
command: clippy
|
||||||
|
args: --workspace --all-targets --features=coconut -- -D warnings
|
||||||
|
|
||||||
|
# nym-wallet (the rust part)
|
||||||
|
- name: Build nym-wallet rust code
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
with:
|
||||||
|
command: build
|
||||||
|
args: --manifest-path nym-wallet/Cargo.toml --workspace
|
||||||
|
|
||||||
|
- name: Run nym-wallet tests
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
with:
|
||||||
|
command: test
|
||||||
|
args: --manifest-path nym-wallet/Cargo.toml --workspace
|
||||||
|
|
||||||
|
- name: Check nym-wallet formatting
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
with:
|
||||||
|
command: fmt
|
||||||
|
args: --manifest-path nym-wallet/Cargo.toml --all -- --check
|
||||||
|
|
||||||
|
- name: Run clippy for nym-wallet
|
||||||
|
uses: actions-rs/cargo@v1
|
||||||
|
if: ${{ matrix.rust != 'nightly' }}
|
||||||
|
with:
|
||||||
|
command: clippy
|
||||||
|
args: --manifest-path nym-wallet/Cargo.toml --workspace --all-targets -- -D warnings
|
||||||
|
|
||||||
|
notification:
|
||||||
|
needs: [build,get_release]
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Collect jobs status
|
||||||
|
uses: technote-space/workflow-conclusion-action@v2
|
||||||
|
- name: Check out repository code
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
- name: Keybase - Node Install
|
||||||
|
if: env.WORKFLOW_CONCLUSION == 'failure'
|
||||||
|
run: npm install
|
||||||
|
working-directory: .github/workflows/support-files
|
||||||
|
- name: Keybase - Send Notification
|
||||||
|
if: env.WORKFLOW_CONCLUSION == 'failure'
|
||||||
|
env:
|
||||||
|
NYM_NOTIFICATION_KIND: nightly
|
||||||
|
NYM_PROJECT_NAME: "Nym nightly build on latest release"
|
||||||
|
GITHUB_TOKEN: "${{ secrets.GITHUB_TOKEN }}"
|
||||||
|
GIT_COMMIT_MESSAGE: "${{ github.event.head_commit.message }}"
|
||||||
|
GIT_BRANCH: "https://github.com/nymtech/nym/tree/${{needs.get_release.outputs.output1}}"
|
||||||
|
KEYBASE_NYMBOT_USERNAME: "${{ secrets.KEYBASE_NYMBOT_USERNAME }}"
|
||||||
|
KEYBASE_NYMBOT_PAPERKEY: "${{ secrets.KEYBASE_NYMBOT_PAPERKEY }}"
|
||||||
|
KEYBASE_NYMBOT_TEAM: "${{ secrets.KEYBASE_NYMBOT_TEAM }}"
|
||||||
|
KEYBASE_NYM_CHANNEL: "ci-nightly-release"
|
||||||
|
IS_SUCCESS: "${{ env.WORKFLOW_CONCLUSION == 'success' }}"
|
||||||
|
uses: docker://keybaseio/client:stable-node
|
||||||
|
with:
|
||||||
|
args: .github/workflows/support-files/notifications/entry_point.sh
|
||||||
@@ -1,50 +0,0 @@
|
|||||||
[
|
|
||||||
{
|
|
||||||
"os":"ubuntu-latest",
|
|
||||||
"rust":"stable",
|
|
||||||
"runOnEvent":"workflow_dispatch"
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
"os":"windows-latest",
|
|
||||||
"rust":"stable",
|
|
||||||
"runOnEvent":"workflow_dispatch"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"os":"macos-latest",
|
|
||||||
"rust":"stable",
|
|
||||||
"runOnEvent":"workflow_dispatch"
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
"os":"ubuntu-latest",
|
|
||||||
"rust":"beta",
|
|
||||||
"runOnEvent":"workflow_dispatch"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"os":"windows-latest",
|
|
||||||
"rust":"beta",
|
|
||||||
"runOnEvent":"workflow_dispatch"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"os":"macos-latest",
|
|
||||||
"rust":"beta",
|
|
||||||
"runOnEvent":"workflow_dispatch"
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
|
||||||
"os":"ubuntu-latest",
|
|
||||||
"rust":"nightly",
|
|
||||||
"runOnEvent":"workflow_dispatch"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"os":"windows-latest",
|
|
||||||
"rust":"nightly",
|
|
||||||
"runOnEvent":"workflow_dispatch"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"os":"macos-latest",
|
|
||||||
"rust":"nightly",
|
|
||||||
"runOnEvent":"workflow_dispatch"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
@@ -41,19 +41,19 @@ jobs:
|
|||||||
- name: Keybase - Node Install
|
- name: Keybase - Node Install
|
||||||
run: npm install
|
run: npm install
|
||||||
working-directory: .github/workflows/support-files
|
working-directory: .github/workflows/support-files
|
||||||
# - name: Keybase - Send Notification
|
- name: Keybase - Send Notification
|
||||||
# env:
|
env:
|
||||||
# NYM_NOTIFICATION_KIND: nym-connect
|
NYM_NOTIFICATION_KIND: nym-connect
|
||||||
# NYM_PROJECT_NAME: "nym-connect"
|
NYM_PROJECT_NAME: "nym-connect"
|
||||||
# NYM_CI_WWW_BASE: "${{ secrets.NYM_CI_WWW_BASE }}"
|
NYM_CI_WWW_BASE: "${{ secrets.NYM_CI_WWW_BASE }}"
|
||||||
# NYM_CI_WWW_LOCATION: "nym-connect-${{ env.GITHUB_REF_SLUG }}"
|
NYM_CI_WWW_LOCATION: "nym-connect-${{ env.GITHUB_REF_SLUG }}"
|
||||||
# GIT_COMMIT_MESSAGE: "${{ github.event.head_commit.message }}"
|
GIT_COMMIT_MESSAGE: "${{ github.event.head_commit.message }}"
|
||||||
# GIT_BRANCH: "${GITHUB_REF##*/}"
|
GIT_BRANCH: "${GITHUB_REF##*/}"
|
||||||
# KEYBASE_NYMBOT_USERNAME: "${{ secrets.KEYBASE_NYMBOT_USERNAME }}"
|
KEYBASE_NYMBOT_USERNAME: "${{ secrets.KEYBASE_NYMBOT_USERNAME }}"
|
||||||
# KEYBASE_NYMBOT_PAPERKEY: "${{ secrets.KEYBASE_NYMBOT_PAPERKEY }}"
|
KEYBASE_NYMBOT_PAPERKEY: "${{ secrets.KEYBASE_NYMBOT_PAPERKEY }}"
|
||||||
# KEYBASE_NYMBOT_TEAM: "${{ secrets.KEYBASE_NYMBOT_TEAM }}"
|
KEYBASE_NYMBOT_TEAM: "${{ secrets.KEYBASE_NYMBOT_TEAM }}"
|
||||||
# KEYBASE_NYM_CHANNEL: "ci-nym-connect"
|
KEYBASE_NYM_CHANNEL: "ci-nym-connect"
|
||||||
# IS_SUCCESS: "${{ job.status == 'success' }}"
|
IS_SUCCESS: "${{ job.status == 'success' }}"
|
||||||
# uses: docker://keybaseio/client:stable-node
|
uses: docker://keybaseio/client:stable-node
|
||||||
# with:
|
with:
|
||||||
# args: .github/workflows/support-files/notifications/entry_point.sh
|
args: .github/workflows/support-files/notifications/entry_point.sh
|
||||||
|
|||||||
@@ -38,3 +38,4 @@ validator-config
|
|||||||
validator-api-config.toml
|
validator-api-config.toml
|
||||||
dist
|
dist
|
||||||
storybook-static
|
storybook-static
|
||||||
|
envs/qwerty.env
|
||||||
@@ -2,10 +2,30 @@
|
|||||||
|
|
||||||
Post 1.0.0 release, the changelog format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
Post 1.0.0 release, the changelog format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
|
## Unreleased (v1.2.0 breaking)
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
-clients: ability to use multi-reply SURBs to send arbitrarily long messages fully anonymously whilst requesting additional reply blocks whenever they're about to run out ([#1796])
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- clients: improve message logging when received message fails to get reconstructed ([#1803])
|
||||||
|
|
||||||
|
[#1796]: https://github.com/nymtech/nym/pull/1796
|
||||||
|
[#1803]: https://github.com/nymtech/nym/pull/1803
|
||||||
|
|
||||||
## Unreleased
|
## Unreleased
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
|
- socks5-client/network-requester: add support for socks4a protocol
|
||||||
|
|
||||||
|
|
||||||
|
## [v1.1.1](https://github.com/nymtech/nym/tree/v1.1.1) (2022-11-29)
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
- binaries: add `-c` shortform for `--config-env-file`
|
- binaries: add `-c` shortform for `--config-env-file`
|
||||||
- websocket-requests: add server response signalling current packet queue length in the client
|
- websocket-requests: add server response signalling current packet queue length in the client
|
||||||
- contracts: DKG contract that handles coconut key generation ([#1678][#1708][#1747])
|
- contracts: DKG contract that handles coconut key generation ([#1678][#1708][#1747])
|
||||||
@@ -17,17 +37,24 @@ Post 1.0.0 release, the changelog format is based on [Keep a Changelog](https://
|
|||||||
- clients,validator-api: take coconut signers from the chain instead of specifying them via CLI ([#1747])
|
- clients,validator-api: take coconut signers from the chain instead of specifying them via CLI ([#1747])
|
||||||
- multisig contract: add DKG contract to the list of addresses that can create proposals ([#1747])
|
- multisig contract: add DKG contract to the list of addresses that can create proposals ([#1747])
|
||||||
- socks5-client: wait closing inbound connection until data is sent, and throttle incoming data in general ([#1783])
|
- socks5-client: wait closing inbound connection until data is sent, and throttle incoming data in general ([#1783])
|
||||||
|
- nym-cli: improve error reporting/handling and changed `vesting-schedule` queries to use query client instead of signing client
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- gateway-client: fix decrypting stored messages on reconnect ([#1786])
|
- gateway-client: fix decrypting stored messages on reconnect ([#1786])
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- gateway-client: fix decrypting stored messages on reconnect ([#1786])
|
||||||
|
- socks5-client: fix shutting down all tasks if anyone of them panics or errors out ([#1805])
|
||||||
|
|
||||||
[#1678]: https://github.com/nymtech/nym/pull/1678
|
[#1678]: https://github.com/nymtech/nym/pull/1678
|
||||||
[#1708]: https://github.com/nymtech/nym/pull/1708
|
[#1708]: https://github.com/nymtech/nym/pull/1708
|
||||||
[#1720]: https://github.com/nymtech/nym/pull/1720
|
[#1720]: https://github.com/nymtech/nym/pull/1720
|
||||||
[#1747]: https://github.com/nymtech/nym/pull/1747
|
[#1747]: https://github.com/nymtech/nym/pull/1747
|
||||||
[#1783]: https://github.com/nymtech/nym/pull/1783
|
[#1783]: https://github.com/nymtech/nym/pull/1783
|
||||||
[#1786]: https://github.com/nymtech/nym/pull/1786
|
[#1786]: https://github.com/nymtech/nym/pull/1786
|
||||||
|
[#1805]: https://github.com/nymtech/nym/pull/1805
|
||||||
|
|
||||||
|
|
||||||
## [v1.1.0](https://github.com/nymtech/nym/tree/v1.1.0) (2022-11-09)
|
## [v1.1.0](https://github.com/nymtech/nym/tree/v1.1.0) (2022-11-09)
|
||||||
|
|||||||
Generated
+150
-98
@@ -131,9 +131,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "async-trait"
|
name = "async-trait"
|
||||||
version = "0.1.53"
|
version = "0.1.58"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "ed6aa3524a2dfcf9fe180c51eae2b58738348d819517ceadf95789c51fff7600"
|
checksum = "1e805d94e6b5001b651426cf4cd446b1ab5f319d27bab5c644f61de0a804360c"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
@@ -586,11 +586,13 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "client-core"
|
name = "client-core"
|
||||||
version = "1.1.0"
|
version = "1.1.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
|
"async-trait",
|
||||||
"client-connections",
|
"client-connections",
|
||||||
"config",
|
"config",
|
||||||
"crypto",
|
"crypto",
|
||||||
|
"dashmap 5.4.0",
|
||||||
"dirs",
|
"dirs",
|
||||||
"futures",
|
"futures",
|
||||||
"gateway-client",
|
"gateway-client",
|
||||||
@@ -603,18 +605,21 @@ dependencies = [
|
|||||||
"pemstore",
|
"pemstore",
|
||||||
"rand 0.7.3",
|
"rand 0.7.3",
|
||||||
"serde",
|
"serde",
|
||||||
"sled",
|
"sqlx 0.6.2",
|
||||||
"tap",
|
"tap",
|
||||||
"task",
|
"task",
|
||||||
"tempfile",
|
"tempfile",
|
||||||
"thiserror",
|
"thiserror",
|
||||||
|
"time 0.3.17",
|
||||||
"tokio",
|
"tokio",
|
||||||
|
"tokio-stream",
|
||||||
"topology",
|
"topology",
|
||||||
"url",
|
"url",
|
||||||
"validator-client",
|
"validator-client",
|
||||||
"wasm-bindgen",
|
"wasm-bindgen",
|
||||||
"wasm-bindgen-futures",
|
"wasm-bindgen-futures",
|
||||||
"wasm-timer",
|
"wasm-timer",
|
||||||
|
"wasm-utils",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -780,7 +785,7 @@ dependencies = [
|
|||||||
"rand 0.8.5",
|
"rand 0.8.5",
|
||||||
"sha2 0.10.2",
|
"sha2 0.10.2",
|
||||||
"subtle 2.4.1",
|
"subtle 2.4.1",
|
||||||
"time 0.3.14",
|
"time 0.3.17",
|
||||||
"version_check",
|
"version_check",
|
||||||
]
|
]
|
||||||
|
|
||||||
@@ -1350,6 +1355,19 @@ dependencies = [
|
|||||||
"num_cpus",
|
"num_cpus",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "dashmap"
|
||||||
|
version = "5.4.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "907076dfda823b0b36d2a1bb5f90c96660a5bbcd7729e10727f07858f22c4edc"
|
||||||
|
dependencies = [
|
||||||
|
"cfg-if 1.0.0",
|
||||||
|
"hashbrown 0.12.3",
|
||||||
|
"lock_api",
|
||||||
|
"once_cell",
|
||||||
|
"parking_lot_core 0.9.4",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "der"
|
name = "der"
|
||||||
version = "0.5.1"
|
version = "0.5.1"
|
||||||
@@ -1633,7 +1651,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "explorer-api"
|
name = "explorer-api"
|
||||||
version = "1.1.0"
|
version = "1.1.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"chrono",
|
"chrono",
|
||||||
"clap 3.2.8",
|
"clap 3.2.8",
|
||||||
@@ -2274,13 +2292,14 @@ checksum = "7ebdb29d2ea9ed0083cd8cece49bbd968021bd99b0849edb4a9a7ee0fdf6a4e0"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "hidapi"
|
name = "hidapi"
|
||||||
version = "1.4.2"
|
version = "1.5.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "9d26e1151deaab68f34fbfd16d491a2a0170cf98d69d3efa23873b567a4199e1"
|
checksum = "798154e4b6570af74899d71155fb0072d5b17e6aa12f39c8ef22c60fb8ec99e7"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"cc",
|
"cc",
|
||||||
"libc",
|
"libc",
|
||||||
"pkg-config",
|
"pkg-config",
|
||||||
|
"winapi",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -2611,9 +2630,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "js-sys"
|
name = "js-sys"
|
||||||
version = "0.3.55"
|
version = "0.3.60"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "7cc9ffccd38c451a86bf13657df244e9c3f37493cce8e5e21e940963777acc84"
|
checksum = "49409df3e3bf0856b916e2ceaca09ee28e6871cf7d9ce97a692cacfdb2a25a47"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"wasm-bindgen",
|
"wasm-bindgen",
|
||||||
]
|
]
|
||||||
@@ -2669,7 +2688,7 @@ checksum = "fe435806c197dfeaa5efcded5e623c4b8230fd28fdf1e91e7a86e40ef2acbf90"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"arrayref",
|
"arrayref",
|
||||||
"no-std-compat",
|
"no-std-compat",
|
||||||
"snafu 0.7.1",
|
"snafu 0.7.3",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -2765,9 +2784,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "lock_api"
|
name = "lock_api"
|
||||||
version = "0.4.7"
|
version = "0.4.9"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "327fa5b6a6940e4699ec49a9beae1ea4845c6bab9314e4f84ac68742139d8c53"
|
checksum = "435011366fe56583b16cf956f9df0095b405b82d76425bc8981c0e22e60ec4df"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"autocfg 1.1.0",
|
"autocfg 1.1.0",
|
||||||
"scopeguard",
|
"scopeguard",
|
||||||
@@ -2919,7 +2938,7 @@ dependencies = [
|
|||||||
"serde_json",
|
"serde_json",
|
||||||
"serde_repr",
|
"serde_repr",
|
||||||
"thiserror",
|
"thiserror",
|
||||||
"time 0.3.14",
|
"time 0.3.17",
|
||||||
"ts-rs",
|
"ts-rs",
|
||||||
]
|
]
|
||||||
|
|
||||||
@@ -3096,15 +3115,6 @@ dependencies = [
|
|||||||
"libc",
|
"libc",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "num_threads"
|
|
||||||
version = "0.1.5"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "aba1801fb138d8e85e11d0fc70baf4fe1cdfffda7c6cd34a854905df588e5ed0"
|
|
||||||
dependencies = [
|
|
||||||
"libc",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "nym-bity-integration"
|
name = "nym-bity-integration"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
@@ -3122,7 +3132,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "nym-cli"
|
name = "nym-cli"
|
||||||
version = "1.1.0"
|
version = "1.1.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"base64",
|
"base64",
|
||||||
@@ -3139,6 +3149,7 @@ dependencies = [
|
|||||||
"pretty_env_logger",
|
"pretty_env_logger",
|
||||||
"serde",
|
"serde",
|
||||||
"serde_json",
|
"serde_json",
|
||||||
|
"tap",
|
||||||
"tokio",
|
"tokio",
|
||||||
"validator-client",
|
"validator-client",
|
||||||
]
|
]
|
||||||
@@ -3164,8 +3175,9 @@ dependencies = [
|
|||||||
"rand 0.6.5",
|
"rand 0.6.5",
|
||||||
"serde",
|
"serde",
|
||||||
"serde_json",
|
"serde_json",
|
||||||
|
"tap",
|
||||||
"thiserror",
|
"thiserror",
|
||||||
"time 0.3.14",
|
"time 0.3.17",
|
||||||
"toml",
|
"toml",
|
||||||
"url",
|
"url",
|
||||||
"validator-client",
|
"validator-client",
|
||||||
@@ -3174,7 +3186,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "nym-client"
|
name = "nym-client"
|
||||||
version = "1.1.0"
|
version = "1.1.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"clap 3.2.8",
|
"clap 3.2.8",
|
||||||
"client-connections",
|
"client-connections",
|
||||||
@@ -3214,7 +3226,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "nym-gateway"
|
name = "nym-gateway"
|
||||||
version = "1.1.0"
|
version = "1.1.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"async-trait",
|
"async-trait",
|
||||||
@@ -3227,7 +3239,7 @@ dependencies = [
|
|||||||
"config",
|
"config",
|
||||||
"credentials",
|
"credentials",
|
||||||
"crypto",
|
"crypto",
|
||||||
"dashmap",
|
"dashmap 4.0.2",
|
||||||
"dirs",
|
"dirs",
|
||||||
"dotenv",
|
"dotenv",
|
||||||
"futures",
|
"futures",
|
||||||
@@ -3261,7 +3273,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "nym-mixnode"
|
name = "nym-mixnode"
|
||||||
version = "1.1.0"
|
version = "1.1.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"bs58",
|
"bs58",
|
||||||
@@ -3303,7 +3315,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "nym-network-requester"
|
name = "nym-network-requester"
|
||||||
version = "1.1.0"
|
version = "1.1.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"async-trait",
|
"async-trait",
|
||||||
"clap 3.2.8",
|
"clap 3.2.8",
|
||||||
@@ -3351,7 +3363,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "nym-socks5-client"
|
name = "nym-socks5-client"
|
||||||
version = "1.1.0"
|
version = "1.1.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"clap 3.2.8",
|
"clap 3.2.8",
|
||||||
"client-connections",
|
"client-connections",
|
||||||
@@ -3418,7 +3430,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "nym-validator-api"
|
name = "nym-validator-api"
|
||||||
version = "1.1.0"
|
version = "1.1.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"async-trait",
|
"async-trait",
|
||||||
@@ -3468,7 +3480,7 @@ dependencies = [
|
|||||||
"tap",
|
"tap",
|
||||||
"task",
|
"task",
|
||||||
"thiserror",
|
"thiserror",
|
||||||
"time 0.3.14",
|
"time 0.3.17",
|
||||||
"tokio",
|
"tokio",
|
||||||
"tokio-stream",
|
"tokio-stream",
|
||||||
"topology",
|
"topology",
|
||||||
@@ -3541,6 +3553,7 @@ dependencies = [
|
|||||||
"nymsphinx-types",
|
"nymsphinx-types",
|
||||||
"rand 0.7.3",
|
"rand 0.7.3",
|
||||||
"rand_distr",
|
"rand_distr",
|
||||||
|
"thiserror",
|
||||||
"tokio",
|
"tokio",
|
||||||
"topology",
|
"topology",
|
||||||
]
|
]
|
||||||
@@ -3566,6 +3579,7 @@ dependencies = [
|
|||||||
"nymsphinx-types",
|
"nymsphinx-types",
|
||||||
"rand 0.7.3",
|
"rand 0.7.3",
|
||||||
"serde",
|
"serde",
|
||||||
|
"thiserror",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -3579,7 +3593,9 @@ dependencies = [
|
|||||||
"nymsphinx-types",
|
"nymsphinx-types",
|
||||||
"rand 0.7.3",
|
"rand 0.7.3",
|
||||||
"serde",
|
"serde",
|
||||||
|
"thiserror",
|
||||||
"topology",
|
"topology",
|
||||||
|
"wasm-bindgen",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -3591,6 +3607,7 @@ dependencies = [
|
|||||||
"nymsphinx-params",
|
"nymsphinx-params",
|
||||||
"nymsphinx-types",
|
"nymsphinx-types",
|
||||||
"rand 0.7.3",
|
"rand 0.7.3",
|
||||||
|
"thiserror",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -3605,6 +3622,7 @@ dependencies = [
|
|||||||
"nymsphinx-params",
|
"nymsphinx-params",
|
||||||
"nymsphinx-types",
|
"nymsphinx-types",
|
||||||
"rand 0.7.3",
|
"rand 0.7.3",
|
||||||
|
"thiserror",
|
||||||
"topology",
|
"topology",
|
||||||
]
|
]
|
||||||
|
|
||||||
@@ -3656,9 +3674,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "once_cell"
|
name = "once_cell"
|
||||||
version = "1.13.0"
|
version = "1.16.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "18a6dbe30758c9f83eb00cbea4ac95966305f5a7772f3f42ebfc7fc7eddbd8e1"
|
checksum = "86f0b0d4bf799edbc74508c1e8bf170ff5f41238e5f8225603ca7caaae2b7860"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "oorandom"
|
name = "oorandom"
|
||||||
@@ -3760,7 +3778,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "87f5ec2493a61ac0506c0f4199f99070cbe83857b0337006a30f3e6719b8ef58"
|
checksum = "87f5ec2493a61ac0506c0f4199f99070cbe83857b0337006a30f3e6719b8ef58"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"lock_api",
|
"lock_api",
|
||||||
"parking_lot_core 0.9.2",
|
"parking_lot_core 0.9.4",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -3779,15 +3797,15 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "parking_lot_core"
|
name = "parking_lot_core"
|
||||||
version = "0.9.2"
|
version = "0.9.4"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "995f667a6c822200b0433ac218e05582f0e2efa1b922a3fd2fbaadc5f87bab37"
|
checksum = "4dc9e0dc2adc1c69d09143aff38d3d30c5c3f0df0dad82e6d25547af174ebec0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"cfg-if 1.0.0",
|
"cfg-if 1.0.0",
|
||||||
"libc",
|
"libc",
|
||||||
"redox_syscall",
|
"redox_syscall",
|
||||||
"smallvec 1.8.0",
|
"smallvec 1.8.0",
|
||||||
"windows-sys 0.34.0",
|
"windows-sys 0.42.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -4076,11 +4094,11 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "proc-macro2"
|
name = "proc-macro2"
|
||||||
version = "1.0.37"
|
version = "1.0.47"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "ec757218438d5fda206afc041538b2f6d889286160d649a86a24d37e1235afd1"
|
checksum = "5ea3d908b0e36316caf9e9e2c4625cdde190a7e6f440d794667ed17a1855e725"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"unicode-xid",
|
"unicode-ident",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -4639,7 +4657,7 @@ dependencies = [
|
|||||||
"serde_json",
|
"serde_json",
|
||||||
"state",
|
"state",
|
||||||
"tempfile",
|
"tempfile",
|
||||||
"time 0.3.14",
|
"time 0.3.17",
|
||||||
"tokio",
|
"tokio",
|
||||||
"tokio-stream",
|
"tokio-stream",
|
||||||
"tokio-util 0.7.3",
|
"tokio-util 0.7.3",
|
||||||
@@ -4701,7 +4719,7 @@ dependencies = [
|
|||||||
"smallvec 1.8.0",
|
"smallvec 1.8.0",
|
||||||
"stable-pattern",
|
"stable-pattern",
|
||||||
"state",
|
"state",
|
||||||
"time 0.3.14",
|
"time 0.3.17",
|
||||||
"tokio",
|
"tokio",
|
||||||
"uncased",
|
"uncased",
|
||||||
]
|
]
|
||||||
@@ -5237,12 +5255,12 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "snafu"
|
name = "snafu"
|
||||||
version = "0.7.1"
|
version = "0.7.3"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "5177903bf45656592d9eb5c0e22f408fc023aae51dbe2088889b71633ba451f2"
|
checksum = "a152ba99b054b22972ee794cf04e5ef572da1229e33b65f3c57abbff0525a454"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"doc-comment",
|
"doc-comment",
|
||||||
"snafu-derive 0.7.1",
|
"snafu-derive 0.7.3",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -5258,9 +5276,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "snafu-derive"
|
name = "snafu-derive"
|
||||||
version = "0.7.1"
|
version = "0.7.3"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "410b26ed97440d90ced3e2488c868d56a86e2064f5d7d6f417909b286afe25e5"
|
checksum = "d5e79cdebbabaebb06a9bdbaedc7f159b410461f63611d4d0e3fb0fab8fed850"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"heck 0.4.0",
|
"heck 0.4.0",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
@@ -5289,7 +5307,7 @@ dependencies = [
|
|||||||
[[package]]
|
[[package]]
|
||||||
name = "sphinx"
|
name = "sphinx"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
source = "git+https://github.com/nymtech/sphinx?rev=c494250f2a78bed33a618d470792418eee932859#c494250f2a78bed33a618d470792418eee932859"
|
source = "git+https://github.com/nymtech/sphinx?rev=e05a1992522ed0afd3c6fcac160313ffc9bb306a#e05a1992522ed0afd3c6fcac160313ffc9bb306a"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"aes 0.7.5",
|
"aes 0.7.5",
|
||||||
"arrayref",
|
"arrayref",
|
||||||
@@ -5647,13 +5665,13 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "syn"
|
name = "syn"
|
||||||
version = "1.0.91"
|
version = "1.0.104"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "b683b2b825c8eef438b77c36a06dc262294da3d5a5813fac20da149241dcd44d"
|
checksum = "4ae548ec36cf198c0ef7710d3c230987c2d6d7bd98ad6edc0274462724c585ce"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"unicode-xid",
|
"unicode-ident",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -5693,8 +5711,12 @@ checksum = "55937e1799185b12863d447f42597ed69d9928686b8d88a1df17376a097d8369"
|
|||||||
name = "task"
|
name = "task"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
|
"futures",
|
||||||
"log",
|
"log",
|
||||||
|
"thiserror",
|
||||||
"tokio",
|
"tokio",
|
||||||
|
"wasm-bindgen",
|
||||||
|
"wasm-bindgen-futures",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -5738,7 +5760,7 @@ dependencies = [
|
|||||||
"subtle 2.4.1",
|
"subtle 2.4.1",
|
||||||
"subtle-encoding",
|
"subtle-encoding",
|
||||||
"tendermint-proto",
|
"tendermint-proto",
|
||||||
"time 0.3.14",
|
"time 0.3.17",
|
||||||
"zeroize",
|
"zeroize",
|
||||||
]
|
]
|
||||||
|
|
||||||
@@ -5771,7 +5793,7 @@ dependencies = [
|
|||||||
"serde",
|
"serde",
|
||||||
"serde_bytes",
|
"serde_bytes",
|
||||||
"subtle-encoding",
|
"subtle-encoding",
|
||||||
"time 0.3.14",
|
"time 0.3.17",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -5799,7 +5821,7 @@ dependencies = [
|
|||||||
"tendermint-config",
|
"tendermint-config",
|
||||||
"tendermint-proto",
|
"tendermint-proto",
|
||||||
"thiserror",
|
"thiserror",
|
||||||
"time 0.3.14",
|
"time 0.3.17",
|
||||||
"tokio",
|
"tokio",
|
||||||
"tracing",
|
"tracing",
|
||||||
"url",
|
"url",
|
||||||
@@ -5833,18 +5855,18 @@ checksum = "b1141d4d61095b28419e22cb0bbf02755f5e54e0526f97f1e3d1d160e60885fb"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "thiserror"
|
name = "thiserror"
|
||||||
version = "1.0.35"
|
version = "1.0.37"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "c53f98874615aea268107765aa1ed8f6116782501d18e53d08b471733bea6c85"
|
checksum = "10deb33631e3c9018b9baf9dcbbc4f737320d2b576bac10f6aefa048fa407e3e"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"thiserror-impl",
|
"thiserror-impl",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "thiserror-impl"
|
name = "thiserror-impl"
|
||||||
version = "1.0.35"
|
version = "1.0.37"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "f8b463991b4eab2d801e724172285ec4195c650e8ec79b149e6c2a8e6dd3f783"
|
checksum = "982d17546b47146b28f7c22e3d08465f6b8903d0ea13c1660d9d84a6e7adcdbb"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
@@ -5873,22 +5895,31 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "time"
|
name = "time"
|
||||||
version = "0.3.14"
|
version = "0.3.17"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "3c3f9a28b618c3a6b9251b6908e9c99e04b9e5c02e6581ccbb67d59c34ef7f9b"
|
checksum = "a561bf4617eebd33bca6434b988f39ed798e527f51a1e797d0ee4f61c0a38376"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"itoa 1.0.1",
|
"itoa 1.0.1",
|
||||||
"libc",
|
"js-sys",
|
||||||
"num_threads",
|
|
||||||
"serde",
|
"serde",
|
||||||
|
"time-core",
|
||||||
"time-macros",
|
"time-macros",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "time-macros"
|
name = "time-core"
|
||||||
version = "0.2.4"
|
version = "0.1.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "42657b1a6f4d817cda8e7a0ace261fe0cc946cf3a80314390b22cc61ae080792"
|
checksum = "2e153e1f1acaef8acc537e68b44906d2db6436e2b35ac2c6b42640fff91f00fd"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "time-macros"
|
||||||
|
version = "0.2.6"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "d967f99f534ca7e495c575c62638eebc2898a8c84c119b89e250477bc4ba16b2"
|
||||||
|
dependencies = [
|
||||||
|
"time-core",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "tinytemplate"
|
name = "tinytemplate"
|
||||||
@@ -5902,9 +5933,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "tokio"
|
name = "tokio"
|
||||||
version = "1.21.2"
|
version = "1.22.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "a9e03c497dc955702ba729190dc4aac6f2a0ce97f913e5b1b5912fc5039d9099"
|
checksum = "d76ce4a75fb488c605c54bf610f221cea8b0dafb53333c1a67e8ee199dcd2ae3"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"autocfg 1.1.0",
|
"autocfg 1.1.0",
|
||||||
"bytes",
|
"bytes",
|
||||||
@@ -6116,6 +6147,7 @@ dependencies = [
|
|||||||
"nymsphinx-addressing",
|
"nymsphinx-addressing",
|
||||||
"nymsphinx-types",
|
"nymsphinx-types",
|
||||||
"rand 0.7.3",
|
"rand 0.7.3",
|
||||||
|
"thiserror",
|
||||||
"version-checker",
|
"version-checker",
|
||||||
]
|
]
|
||||||
|
|
||||||
@@ -6375,6 +6407,12 @@ version = "0.3.7"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "1a01404663e3db436ed2746d9fefef640d868edae3cceb81c3b8d5732fda678f"
|
checksum = "1a01404663e3db436ed2746d9fefef640d868edae3cceb81c3b8d5732fda678f"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "unicode-ident"
|
||||||
|
version = "1.0.5"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "6ceab39d59e4c9499d4e5a8ee0e2735b891bb7308ac83dfb4e80cad195c9f6f3"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "unicode-normalization"
|
name = "unicode-normalization"
|
||||||
version = "0.1.9"
|
version = "0.1.9"
|
||||||
@@ -6545,7 +6583,7 @@ dependencies = [
|
|||||||
"rustc_version 0.4.0",
|
"rustc_version 0.4.0",
|
||||||
"rustversion",
|
"rustversion",
|
||||||
"thiserror",
|
"thiserror",
|
||||||
"time 0.3.14",
|
"time 0.3.17",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -6818,19 +6856,6 @@ version = "0.4.0"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f"
|
checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "windows-sys"
|
|
||||||
version = "0.34.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "5acdd78cb4ba54c0045ac14f62d8f94a03d10047904ae2a40afa1e99d8f70825"
|
|
||||||
dependencies = [
|
|
||||||
"windows_aarch64_msvc 0.34.0",
|
|
||||||
"windows_i686_gnu 0.34.0",
|
|
||||||
"windows_i686_msvc 0.34.0",
|
|
||||||
"windows_x86_64_gnu 0.34.0",
|
|
||||||
"windows_x86_64_msvc 0.34.0",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "windows-sys"
|
name = "windows-sys"
|
||||||
version = "0.36.1"
|
version = "0.36.1"
|
||||||
@@ -6845,10 +6870,25 @@ dependencies = [
|
|||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "windows_aarch64_msvc"
|
name = "windows-sys"
|
||||||
version = "0.34.0"
|
version = "0.42.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "17cffbe740121affb56fad0fc0e421804adf0ae00891205213b5cecd30db881d"
|
checksum = "5a3e1820f08b8513f676f7ab6c1f99ff312fb97b553d30ff4dd86f9f15728aa7"
|
||||||
|
dependencies = [
|
||||||
|
"windows_aarch64_gnullvm",
|
||||||
|
"windows_aarch64_msvc 0.42.0",
|
||||||
|
"windows_i686_gnu 0.42.0",
|
||||||
|
"windows_i686_msvc 0.42.0",
|
||||||
|
"windows_x86_64_gnu 0.42.0",
|
||||||
|
"windows_x86_64_gnullvm",
|
||||||
|
"windows_x86_64_msvc 0.42.0",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "windows_aarch64_gnullvm"
|
||||||
|
version = "0.42.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "41d2aa71f6f0cbe00ae5167d90ef3cfe66527d6f613ca78ac8024c3ccab9a19e"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "windows_aarch64_msvc"
|
name = "windows_aarch64_msvc"
|
||||||
@@ -6857,10 +6897,10 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "9bb8c3fd39ade2d67e9874ac4f3db21f0d710bee00fe7cab16949ec184eeaa47"
|
checksum = "9bb8c3fd39ade2d67e9874ac4f3db21f0d710bee00fe7cab16949ec184eeaa47"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "windows_i686_gnu"
|
name = "windows_aarch64_msvc"
|
||||||
version = "0.34.0"
|
version = "0.42.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "2564fde759adb79129d9b4f54be42b32c89970c18ebf93124ca8870a498688ed"
|
checksum = "dd0f252f5a35cac83d6311b2e795981f5ee6e67eb1f9a7f64eb4500fbc4dcdb4"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "windows_i686_gnu"
|
name = "windows_i686_gnu"
|
||||||
@@ -6869,10 +6909,10 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "180e6ccf01daf4c426b846dfc66db1fc518f074baa793aa7d9b9aaeffad6a3b6"
|
checksum = "180e6ccf01daf4c426b846dfc66db1fc518f074baa793aa7d9b9aaeffad6a3b6"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "windows_i686_msvc"
|
name = "windows_i686_gnu"
|
||||||
version = "0.34.0"
|
version = "0.42.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "9cd9d32ba70453522332c14d38814bceeb747d80b3958676007acadd7e166956"
|
checksum = "fbeae19f6716841636c28d695375df17562ca208b2b7d0dc47635a50ae6c5de7"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "windows_i686_msvc"
|
name = "windows_i686_msvc"
|
||||||
@@ -6881,10 +6921,10 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "e2e7917148b2812d1eeafaeb22a97e4813dfa60a3f8f78ebe204bcc88f12f024"
|
checksum = "e2e7917148b2812d1eeafaeb22a97e4813dfa60a3f8f78ebe204bcc88f12f024"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "windows_x86_64_gnu"
|
name = "windows_i686_msvc"
|
||||||
version = "0.34.0"
|
version = "0.42.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "cfce6deae227ee8d356d19effc141a509cc503dfd1f850622ec4b0f84428e1f4"
|
checksum = "84c12f65daa39dd2babe6e442988fc329d6243fdce47d7d2d155b8d874862246"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "windows_x86_64_gnu"
|
name = "windows_x86_64_gnu"
|
||||||
@@ -6893,10 +6933,16 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "4dcd171b8776c41b97521e5da127a2d86ad280114807d0b2ab1e462bc764d9e1"
|
checksum = "4dcd171b8776c41b97521e5da127a2d86ad280114807d0b2ab1e462bc764d9e1"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "windows_x86_64_msvc"
|
name = "windows_x86_64_gnu"
|
||||||
version = "0.34.0"
|
version = "0.42.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "d19538ccc21819d01deaf88d6a17eae6596a12e9aafdbb97916fb49896d89de9"
|
checksum = "bf7b1b21b5362cbc318f686150e5bcea75ecedc74dd157d874d754a2ca44b0ed"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "windows_x86_64_gnullvm"
|
||||||
|
version = "0.42.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "09d525d2ba30eeb3297665bd434a54297e4170c7f1a44cad4ef58095b4cd2028"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "windows_x86_64_msvc"
|
name = "windows_x86_64_msvc"
|
||||||
@@ -6904,6 +6950,12 @@ version = "0.36.1"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "c811ca4a8c853ef420abd8592ba53ddbbac90410fab6903b3e79972a631f7680"
|
checksum = "c811ca4a8c853ef420abd8592ba53ddbbac90410fab6903b3e79972a631f7680"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "windows_x86_64_msvc"
|
||||||
|
version = "0.42.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "f40009d85759725a34da6d89a94e63d7bdc50a862acf0dbc7c8e488f1edcb6f5"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "winreg"
|
name = "winreg"
|
||||||
version = "0.10.1"
|
version = "0.10.1"
|
||||||
|
|||||||
@@ -1,22 +1,25 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "client-core"
|
name = "client-core"
|
||||||
version = "1.1.0"
|
version = "1.1.1"
|
||||||
authors = ["Dave Hrycyszyn <futurechimp@users.noreply.github.com>"]
|
authors = ["Dave Hrycyszyn <futurechimp@users.noreply.github.com>"]
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
|
|
||||||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
async-trait = { version = "0.1.58" }
|
||||||
dirs = "4.0"
|
dirs = "4.0"
|
||||||
|
dashmap = "5.4.0"
|
||||||
futures = "0.3"
|
futures = "0.3"
|
||||||
humantime-serde = "1.0"
|
humantime-serde = "1.0"
|
||||||
log = "0.4"
|
log = "0.4"
|
||||||
rand = { version = "0.7.3", features = ["wasm-bindgen"] }
|
rand = { version = "0.7.3", features = ["wasm-bindgen"] }
|
||||||
serde = { version = "1.0", features = ["derive"] }
|
serde = { version = "1.0", features = ["derive"] }
|
||||||
sled = { version = "0.34", optional = true }
|
|
||||||
tap = "1.0.1"
|
tap = "1.0.1"
|
||||||
thiserror = "1.0.34"
|
thiserror = "1.0.34"
|
||||||
url = { version ="2.2", features = ["serde"] }
|
url = { version ="2.2", features = ["serde"] }
|
||||||
|
tokio = { version = "1.21.2", features = ["macros"]}
|
||||||
|
time = "0.3.17"
|
||||||
|
|
||||||
# internal
|
# internal
|
||||||
config = { path = "../../common/config" }
|
config = { path = "../../common/config" }
|
||||||
@@ -30,8 +33,20 @@ nymsphinx = { path = "../../common/nymsphinx" }
|
|||||||
pemstore = { path = "../../common/pemstore" }
|
pemstore = { path = "../../common/pemstore" }
|
||||||
topology = { path = "../../common/topology" }
|
topology = { path = "../../common/topology" }
|
||||||
validator-client = { path = "../../common/client-libs/validator-client", default-features = false }
|
validator-client = { path = "../../common/client-libs/validator-client", default-features = false }
|
||||||
|
task = { path = "../../common/task" }
|
||||||
|
|
||||||
tokio = { version = "1.21.2", features = ["time", "macros"]}
|
[target."cfg(not(target_arch = \"wasm32\"))".dependencies.tokio-stream]
|
||||||
|
version = "0.1.9"
|
||||||
|
features = ["time"]
|
||||||
|
|
||||||
|
[target."cfg(not(target_arch = \"wasm32\"))".dependencies.tokio]
|
||||||
|
version = "1.21.2"
|
||||||
|
features = ["time"]
|
||||||
|
|
||||||
|
[target."cfg(not(target_arch = \"wasm32\"))".dependencies.sqlx]
|
||||||
|
version = "0.6.2"
|
||||||
|
features = ["runtime-tokio-rustls", "sqlite", "macros", "migrate"]
|
||||||
|
optional = true
|
||||||
|
|
||||||
[target."cfg(target_arch = \"wasm32\")".dependencies.wasm-bindgen-futures]
|
[target."cfg(target_arch = \"wasm32\")".dependencies.wasm-bindgen-futures]
|
||||||
version = "0.4"
|
version = "0.4"
|
||||||
@@ -47,15 +62,23 @@ rev = "b9d1a54ad514c2f230a026afe0dde341e98cd7b6"
|
|||||||
version = "0.2.4"
|
version = "0.2.4"
|
||||||
features = ["futures"]
|
features = ["futures"]
|
||||||
|
|
||||||
[target."cfg(not(target_arch = \"wasm32\"))".dependencies.task]
|
[target."cfg(target_arch = \"wasm32\")".dependencies.wasm-utils]
|
||||||
path = "../../common/task"
|
path = "../../common/wasm-utils"
|
||||||
|
|
||||||
|
[target."cfg(target_arch = \"wasm32\")".dependencies.time]
|
||||||
|
version = "0.3.17"
|
||||||
|
features = ["wasm-bindgen"]
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
tempfile = "3.1.0"
|
tempfile = "3.1.0"
|
||||||
|
|
||||||
|
[build-dependencies]
|
||||||
|
tokio = { version = "1.21.2", features = ["rt-multi-thread", "macros"] }
|
||||||
|
sqlx = { version = "0.6.2", features = ["runtime-tokio-rustls", "sqlite", "macros", "migrate"] }
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
default = ["reply-surb"]
|
default = []
|
||||||
|
fs-surb-storage = ["sqlx"]
|
||||||
wasm = ["gateway-client/wasm"]
|
wasm = ["gateway-client/wasm"]
|
||||||
coconut = ["gateway-client/coconut", "gateway-requests/coconut"]
|
coconut = ["gateway-client/coconut", "gateway-requests/coconut"]
|
||||||
reply-surb = ["sled"]
|
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,31 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
#[tokio::main]
|
||||||
|
async fn main() {
|
||||||
|
#[cfg(feature = "fs-surb-storage")]
|
||||||
|
{
|
||||||
|
use sqlx::{Connection, SqliteConnection};
|
||||||
|
use std::env;
|
||||||
|
|
||||||
|
let out_dir = env::var("OUT_DIR").unwrap();
|
||||||
|
let database_path = format!("{}/fs-surbs-example.sqlite", out_dir);
|
||||||
|
|
||||||
|
let mut conn = SqliteConnection::connect(&format!("sqlite://{}?mode=rwc", database_path))
|
||||||
|
.await
|
||||||
|
.expect("Failed to create SQLx database connection");
|
||||||
|
|
||||||
|
sqlx::migrate!("./fs_surbs_migrations")
|
||||||
|
.run(&mut conn)
|
||||||
|
.await
|
||||||
|
.expect("Failed to perform SQLx migrations");
|
||||||
|
|
||||||
|
#[cfg(target_family = "unix")]
|
||||||
|
println!("cargo:rustc-env=DATABASE_URL=sqlite://{}", &database_path);
|
||||||
|
|
||||||
|
#[cfg(target_family = "windows")]
|
||||||
|
// for some strange reason we need to add a leading `/` to the windows path even though it's
|
||||||
|
// not a valid windows path... but hey, it works...
|
||||||
|
println!("cargo:rustc-env=DATABASE_URL=sqlite:///{}", &database_path);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,40 @@
|
|||||||
|
CREATE TABLE status
|
||||||
|
(
|
||||||
|
flush_in_progress INTEGER NOT NULL,
|
||||||
|
previous_flush_timestamp INTEGER NOT NULL,
|
||||||
|
client_in_use INTEGER NOT NULL
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE reply_surb_storage_metadata
|
||||||
|
(
|
||||||
|
min_reply_surb_threshold INTEGER NOT NULL,
|
||||||
|
max_reply_surb_threshold INTEGER NOT NULL
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE sender_tag
|
||||||
|
(
|
||||||
|
recipient BLOB NOT NULL UNIQUE,
|
||||||
|
tag BLOB NOT NULL UNIQUE
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE reply_key
|
||||||
|
(
|
||||||
|
key_digest BLOB NOT NULL UNIQUE,
|
||||||
|
reply_key BLOB NOT NULL UNIQUE,
|
||||||
|
sent_at_timestamp INTEGER NOT NULL
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE reply_surb_sender
|
||||||
|
(
|
||||||
|
id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT,
|
||||||
|
last_sent_timestamp INTEGER NOT NULL,
|
||||||
|
tag BLOB NOT NULL UNIQUE
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE reply_surb
|
||||||
|
(
|
||||||
|
reply_surb_sender_id INTEGER NOT NULL,
|
||||||
|
reply_surb BLOB NOT NULL,
|
||||||
|
|
||||||
|
FOREIGN KEY (reply_surb_sender_id) REFERENCES reply_surb_sender (id)
|
||||||
|
);
|
||||||
@@ -0,0 +1,480 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use crate::client::cover_traffic_stream::LoopCoverTrafficStream;
|
||||||
|
use crate::client::inbound_messages::{InputMessage, InputMessageReceiver, InputMessageSender};
|
||||||
|
use crate::client::key_manager::KeyManager;
|
||||||
|
use crate::client::mix_traffic::{BatchMixMessageSender, MixTrafficController};
|
||||||
|
use crate::client::real_messages_control;
|
||||||
|
use crate::client::real_messages_control::RealMessagesController;
|
||||||
|
use crate::client::received_buffer::{
|
||||||
|
ReceivedBufferRequestReceiver, ReceivedBufferRequestSender, ReceivedMessagesBufferController,
|
||||||
|
};
|
||||||
|
use crate::client::replies::reply_controller;
|
||||||
|
use crate::client::replies::reply_controller::{ReplyControllerReceiver, ReplyControllerSender};
|
||||||
|
use crate::client::replies::reply_storage::{
|
||||||
|
CombinedReplyStorage, PersistentReplyStorage, ReplyStorageBackend, SentReplyKeys,
|
||||||
|
};
|
||||||
|
use crate::client::topology_control::{
|
||||||
|
TopologyAccessor, TopologyRefresher, TopologyRefresherConfig,
|
||||||
|
};
|
||||||
|
use crate::config::{Config, DebugConfig, GatewayEndpointConfig};
|
||||||
|
use crate::error::ClientCoreError;
|
||||||
|
use crate::spawn_future;
|
||||||
|
use client_connections::{ConnectionCommandReceiver, ConnectionCommandSender, LaneQueueLengths};
|
||||||
|
use crypto::asymmetric::{encryption, identity};
|
||||||
|
use futures::channel::mpsc;
|
||||||
|
use gateway_client::bandwidth::BandwidthController;
|
||||||
|
use gateway_client::{
|
||||||
|
AcknowledgementReceiver, AcknowledgementSender, GatewayClient, MixnetMessageReceiver,
|
||||||
|
MixnetMessageSender,
|
||||||
|
};
|
||||||
|
use log::info;
|
||||||
|
use nymsphinx::acknowledgements::AckKey;
|
||||||
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
|
use nymsphinx::addressing::nodes::NodeIdentity;
|
||||||
|
use std::sync::Arc;
|
||||||
|
use std::time::Duration;
|
||||||
|
use task::{ShutdownListener, ShutdownNotifier};
|
||||||
|
use url::Url;
|
||||||
|
|
||||||
|
#[cfg(all(not(target_arch = "wasm32"), feature = "fs-surb-storage"))]
|
||||||
|
pub mod non_wasm_helpers;
|
||||||
|
|
||||||
|
pub struct ClientInput {
|
||||||
|
pub shared_lane_queue_lengths: LaneQueueLengths,
|
||||||
|
pub connection_command_sender: ConnectionCommandSender,
|
||||||
|
pub input_sender: InputMessageSender,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct ClientOutput {
|
||||||
|
pub received_buffer_request_sender: ReceivedBufferRequestSender,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub enum ClientInputStatus {
|
||||||
|
AwaitingProducer { client_input: ClientInput },
|
||||||
|
Connected,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ClientInputStatus {
|
||||||
|
pub fn register_producer(&mut self) -> ClientInput {
|
||||||
|
match std::mem::replace(self, ClientInputStatus::Connected) {
|
||||||
|
ClientInputStatus::AwaitingProducer { client_input } => client_input,
|
||||||
|
ClientInputStatus::Connected => panic!("producer was already registered before"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub enum ClientOutputStatus {
|
||||||
|
AwaitingConsumer { client_output: ClientOutput },
|
||||||
|
Connected,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ClientOutputStatus {
|
||||||
|
pub fn register_consumer(&mut self) -> ClientOutput {
|
||||||
|
match std::mem::replace(self, ClientOutputStatus::Connected) {
|
||||||
|
ClientOutputStatus::AwaitingConsumer { client_output } => client_output,
|
||||||
|
ClientOutputStatus::Connected => panic!("consumer was already registered before"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct BaseClientBuilder<'a, B> {
|
||||||
|
// due to wasm limitations I had to split it like this : (
|
||||||
|
gateway_config: &'a GatewayEndpointConfig,
|
||||||
|
debug_config: &'a DebugConfig,
|
||||||
|
disabled_credentials: bool,
|
||||||
|
validator_api_endpoints: Vec<Url>,
|
||||||
|
reply_storage_backend: B,
|
||||||
|
|
||||||
|
bandwidth_controller: Option<BandwidthController>,
|
||||||
|
key_manager: KeyManager,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, B> BaseClientBuilder<'a, B>
|
||||||
|
where
|
||||||
|
B: ReplyStorageBackend + Send + Sync + 'static,
|
||||||
|
{
|
||||||
|
pub fn new_from_base_config<T>(
|
||||||
|
base_config: &'a Config<T>,
|
||||||
|
key_manager: KeyManager,
|
||||||
|
bandwidth_controller: Option<BandwidthController>,
|
||||||
|
reply_storage_backend: B,
|
||||||
|
) -> BaseClientBuilder<'a, B> {
|
||||||
|
BaseClientBuilder {
|
||||||
|
gateway_config: base_config.get_gateway_endpoint_config(),
|
||||||
|
debug_config: base_config.get_debug_config(),
|
||||||
|
disabled_credentials: base_config.get_disabled_credentials_mode(),
|
||||||
|
validator_api_endpoints: base_config.get_validator_api_endpoints(),
|
||||||
|
bandwidth_controller,
|
||||||
|
reply_storage_backend,
|
||||||
|
key_manager,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn new(
|
||||||
|
gateway_config: &'a GatewayEndpointConfig,
|
||||||
|
debug_config: &'a DebugConfig,
|
||||||
|
key_manager: KeyManager,
|
||||||
|
bandwidth_controller: Option<BandwidthController>,
|
||||||
|
reply_storage_backend: B,
|
||||||
|
disabled_credentials: bool,
|
||||||
|
validator_api_endpoints: Vec<Url>,
|
||||||
|
) -> BaseClientBuilder<'a, B> {
|
||||||
|
BaseClientBuilder {
|
||||||
|
gateway_config,
|
||||||
|
debug_config,
|
||||||
|
disabled_credentials,
|
||||||
|
validator_api_endpoints,
|
||||||
|
bandwidth_controller,
|
||||||
|
reply_storage_backend,
|
||||||
|
key_manager,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn as_mix_recipient(&self) -> Recipient {
|
||||||
|
Recipient::new(
|
||||||
|
*self.key_manager.identity_keypair().public_key(),
|
||||||
|
*self.key_manager.encryption_keypair().public_key(),
|
||||||
|
// TODO: below only works under assumption that gateway address == gateway id
|
||||||
|
// (which currently is true)
|
||||||
|
NodeIdentity::from_base58_string(&self.gateway_config.gateway_id).unwrap(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// future constantly pumping loop cover traffic at some specified average rate
|
||||||
|
// the pumped traffic goes to the MixTrafficController
|
||||||
|
fn start_cover_traffic_stream(
|
||||||
|
debug_config: &DebugConfig,
|
||||||
|
ack_key: Arc<AckKey>,
|
||||||
|
self_address: Recipient,
|
||||||
|
topology_accessor: TopologyAccessor,
|
||||||
|
mix_tx: BatchMixMessageSender,
|
||||||
|
shutdown: ShutdownListener,
|
||||||
|
) {
|
||||||
|
info!("Starting loop cover traffic stream...");
|
||||||
|
|
||||||
|
let mut stream = LoopCoverTrafficStream::new(
|
||||||
|
ack_key,
|
||||||
|
debug_config.average_ack_delay,
|
||||||
|
debug_config.average_packet_delay,
|
||||||
|
debug_config.loop_cover_traffic_average_delay,
|
||||||
|
mix_tx,
|
||||||
|
self_address,
|
||||||
|
topology_accessor,
|
||||||
|
);
|
||||||
|
|
||||||
|
if let Some(size) = debug_config.use_extended_packet_size {
|
||||||
|
log::debug!("Setting extended packet size: {:?}", size);
|
||||||
|
stream.set_custom_packet_size(size.into());
|
||||||
|
}
|
||||||
|
|
||||||
|
stream.start_with_shutdown(shutdown);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(clippy::too_many_arguments)]
|
||||||
|
fn start_real_traffic_controller(
|
||||||
|
controller_config: real_messages_control::Config,
|
||||||
|
topology_accessor: TopologyAccessor,
|
||||||
|
ack_receiver: AcknowledgementReceiver,
|
||||||
|
input_receiver: InputMessageReceiver,
|
||||||
|
mix_sender: BatchMixMessageSender,
|
||||||
|
reply_storage: CombinedReplyStorage,
|
||||||
|
reply_controller_sender: ReplyControllerSender,
|
||||||
|
reply_controller_receiver: ReplyControllerReceiver,
|
||||||
|
lane_queue_lengths: LaneQueueLengths,
|
||||||
|
client_connection_rx: ConnectionCommandReceiver,
|
||||||
|
shutdown: ShutdownListener,
|
||||||
|
) {
|
||||||
|
info!("Starting real traffic stream...");
|
||||||
|
|
||||||
|
RealMessagesController::new(
|
||||||
|
controller_config,
|
||||||
|
ack_receiver,
|
||||||
|
input_receiver,
|
||||||
|
mix_sender,
|
||||||
|
topology_accessor,
|
||||||
|
reply_storage,
|
||||||
|
reply_controller_sender,
|
||||||
|
reply_controller_receiver,
|
||||||
|
lane_queue_lengths,
|
||||||
|
client_connection_rx,
|
||||||
|
)
|
||||||
|
.start_with_shutdown(shutdown);
|
||||||
|
}
|
||||||
|
|
||||||
|
// buffer controlling all messages fetched from provider
|
||||||
|
// required so that other components would be able to use them (say the websocket)
|
||||||
|
fn start_received_messages_buffer_controller(
|
||||||
|
local_encryption_keypair: Arc<encryption::KeyPair>,
|
||||||
|
query_receiver: ReceivedBufferRequestReceiver,
|
||||||
|
mixnet_receiver: MixnetMessageReceiver,
|
||||||
|
reply_key_storage: SentReplyKeys,
|
||||||
|
reply_controller_sender: ReplyControllerSender,
|
||||||
|
shutdown: ShutdownListener,
|
||||||
|
) {
|
||||||
|
info!("Starting received messages buffer controller...");
|
||||||
|
ReceivedMessagesBufferController::new(
|
||||||
|
local_encryption_keypair,
|
||||||
|
query_receiver,
|
||||||
|
mixnet_receiver,
|
||||||
|
reply_key_storage,
|
||||||
|
reply_controller_sender,
|
||||||
|
)
|
||||||
|
.start_with_shutdown(shutdown)
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn start_gateway_client(
|
||||||
|
&mut self,
|
||||||
|
mixnet_message_sender: MixnetMessageSender,
|
||||||
|
ack_sender: AcknowledgementSender,
|
||||||
|
shutdown: ShutdownListener,
|
||||||
|
) -> GatewayClient {
|
||||||
|
let gateway_id = self.gateway_config.gateway_id.clone();
|
||||||
|
if gateway_id.is_empty() {
|
||||||
|
panic!("The identity of the gateway is unknown - did you run `nym-client` init?")
|
||||||
|
}
|
||||||
|
let gateway_owner = self.gateway_config.gateway_owner.clone();
|
||||||
|
if gateway_owner.is_empty() {
|
||||||
|
panic!("The owner of the gateway is unknown - did you run `nym-client` init?")
|
||||||
|
}
|
||||||
|
let gateway_address = self.gateway_config.gateway_listener.clone();
|
||||||
|
if gateway_address.is_empty() {
|
||||||
|
panic!("The address of the gateway is unknown - did you run `nym-client` init?")
|
||||||
|
}
|
||||||
|
|
||||||
|
let gateway_identity = identity::PublicKey::from_base58_string(gateway_id)
|
||||||
|
.expect("provided gateway id is invalid!");
|
||||||
|
|
||||||
|
// disgusting wasm workaround since there's no key persistence there (nor `client init`)
|
||||||
|
let shared_key = if self.key_manager.gateway_key_set() {
|
||||||
|
Some(self.key_manager.gateway_shared_key())
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut gateway_client = GatewayClient::new(
|
||||||
|
gateway_address,
|
||||||
|
self.key_manager.identity_keypair(),
|
||||||
|
gateway_identity,
|
||||||
|
gateway_owner,
|
||||||
|
shared_key,
|
||||||
|
mixnet_message_sender,
|
||||||
|
ack_sender,
|
||||||
|
self.debug_config.gateway_response_timeout,
|
||||||
|
self.bandwidth_controller.take(),
|
||||||
|
shutdown,
|
||||||
|
);
|
||||||
|
|
||||||
|
gateway_client.set_disabled_credentials_mode(self.disabled_credentials);
|
||||||
|
|
||||||
|
gateway_client
|
||||||
|
.authenticate_and_start()
|
||||||
|
.await
|
||||||
|
.expect("could not authenticate and start up the gateway connection");
|
||||||
|
|
||||||
|
gateway_client
|
||||||
|
}
|
||||||
|
|
||||||
|
// future responsible for periodically polling directory server and updating
|
||||||
|
// the current global view of topology
|
||||||
|
async fn start_topology_refresher(
|
||||||
|
validator_api_urls: Vec<Url>,
|
||||||
|
refresh_rate: Duration,
|
||||||
|
topology_accessor: TopologyAccessor,
|
||||||
|
shutdown: ShutdownListener,
|
||||||
|
) -> Result<(), ClientCoreError<B>> {
|
||||||
|
let topology_refresher_config = TopologyRefresherConfig::new(
|
||||||
|
validator_api_urls,
|
||||||
|
refresh_rate,
|
||||||
|
env!("CARGO_PKG_VERSION").to_string(),
|
||||||
|
);
|
||||||
|
let mut topology_refresher =
|
||||||
|
TopologyRefresher::new(topology_refresher_config, topology_accessor);
|
||||||
|
// before returning, block entire runtime to refresh the current network view so that any
|
||||||
|
// components depending on topology would see a non-empty view
|
||||||
|
info!("Obtaining initial network topology");
|
||||||
|
topology_refresher.refresh().await;
|
||||||
|
|
||||||
|
if let Err(err) = topology_refresher.ensure_topology_is_routable().await {
|
||||||
|
log::error!(
|
||||||
|
"The current network topology seem to be insufficient to route any packets through \
|
||||||
|
- check if enough nodes and a gateway are online - source: {err}"
|
||||||
|
);
|
||||||
|
return Err(ClientCoreError::InsufficientNetworkTopology(err));
|
||||||
|
}
|
||||||
|
|
||||||
|
info!("Starting topology refresher...");
|
||||||
|
topology_refresher.start_with_shutdown(shutdown);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
// controller for sending sphinx packets to mixnet (either real traffic or cover traffic)
|
||||||
|
// TODO: if we want to send control messages to gateway_client, this CAN'T take the ownership
|
||||||
|
// over it. Perhaps GatewayClient needs to be thread-shareable or have some channel for
|
||||||
|
// requests?
|
||||||
|
fn start_mix_traffic_controller(
|
||||||
|
gateway_client: GatewayClient,
|
||||||
|
shutdown: ShutdownListener,
|
||||||
|
) -> BatchMixMessageSender {
|
||||||
|
info!("Starting mix traffic controller...");
|
||||||
|
let (mix_traffic_controller, mix_tx) = MixTrafficController::new(gateway_client);
|
||||||
|
mix_traffic_controller.start_with_shutdown(shutdown);
|
||||||
|
mix_tx
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn setup_persistent_reply_storage(
|
||||||
|
backend: B,
|
||||||
|
shutdown: ShutdownListener,
|
||||||
|
) -> Result<CombinedReplyStorage, ClientCoreError<B>> {
|
||||||
|
let persistent_storage = PersistentReplyStorage::new(backend);
|
||||||
|
let mem_store = persistent_storage
|
||||||
|
.load_state_from_backend()
|
||||||
|
.await
|
||||||
|
.map_err(|err| ClientCoreError::SurbStorageError { source: err })?;
|
||||||
|
|
||||||
|
let store_clone = mem_store.clone();
|
||||||
|
spawn_future(async move {
|
||||||
|
persistent_storage
|
||||||
|
.flush_on_shutdown(store_clone, shutdown)
|
||||||
|
.await
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok(mem_store)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn start_base(mut self) -> Result<BaseClient, ClientCoreError<B>> {
|
||||||
|
info!("Starting nym client");
|
||||||
|
// channels for inter-component communication
|
||||||
|
// TODO: make the channels be internally created by the relevant components
|
||||||
|
// rather than creating them here, so say for example the buffer controller would create the request channels
|
||||||
|
// and would allow anyone to clone the sender channel
|
||||||
|
|
||||||
|
// unwrapped_sphinx_sender is the transmitter of mixnet messages received from the gateway
|
||||||
|
// unwrapped_sphinx_receiver is the receiver for said messages - used by ReceivedMessagesBuffer
|
||||||
|
let (mixnet_messages_sender, mixnet_messages_receiver) = mpsc::unbounded();
|
||||||
|
|
||||||
|
// used for announcing connection or disconnection of a channel for pushing re-assembled messages to
|
||||||
|
let (received_buffer_request_sender, received_buffer_request_receiver) = mpsc::unbounded();
|
||||||
|
|
||||||
|
// channels responsible for controlling real messages
|
||||||
|
let (input_sender, input_receiver) = tokio::sync::mpsc::channel::<InputMessage>(1);
|
||||||
|
|
||||||
|
// channels responsible for controlling ack messages
|
||||||
|
let (ack_sender, ack_receiver) = mpsc::unbounded();
|
||||||
|
let shared_topology_accessor = TopologyAccessor::new();
|
||||||
|
|
||||||
|
// Shutdown notifier for signalling tasks to stop
|
||||||
|
let shutdown = ShutdownNotifier::default();
|
||||||
|
|
||||||
|
// channels responsible for dealing with reply-related fun
|
||||||
|
let (reply_controller_sender, reply_controller_receiver) =
|
||||||
|
reply_controller::new_control_channels();
|
||||||
|
|
||||||
|
let self_address = self.as_mix_recipient();
|
||||||
|
|
||||||
|
// the components are started in very specific order. Unless you know what you are doing,
|
||||||
|
// do not change that.
|
||||||
|
let gateway_client = self
|
||||||
|
.start_gateway_client(mixnet_messages_sender, ack_sender, shutdown.subscribe())
|
||||||
|
.await;
|
||||||
|
|
||||||
|
let reply_storage =
|
||||||
|
Self::setup_persistent_reply_storage(self.reply_storage_backend, shutdown.subscribe())
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
Self::start_topology_refresher(
|
||||||
|
self.validator_api_endpoints.clone(),
|
||||||
|
self.debug_config.topology_refresh_rate,
|
||||||
|
shared_topology_accessor.clone(),
|
||||||
|
shutdown.subscribe(),
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
Self::start_received_messages_buffer_controller(
|
||||||
|
self.key_manager.encryption_keypair(),
|
||||||
|
received_buffer_request_receiver,
|
||||||
|
mixnet_messages_receiver,
|
||||||
|
reply_storage.key_storage(),
|
||||||
|
reply_controller_sender.clone(),
|
||||||
|
shutdown.subscribe(),
|
||||||
|
);
|
||||||
|
|
||||||
|
// The sphinx_message_sender is the transmitter for any component generating sphinx packets
|
||||||
|
// that are to be sent to the mixnet. They are used by cover traffic stream and real
|
||||||
|
// traffic stream.
|
||||||
|
// The MixTrafficController then sends the actual traffic
|
||||||
|
let sphinx_message_sender =
|
||||||
|
Self::start_mix_traffic_controller(gateway_client, shutdown.subscribe());
|
||||||
|
|
||||||
|
// Channels that the websocket listener can use to signal downstream to the real traffic
|
||||||
|
// controller that connections are closed.
|
||||||
|
let (client_connection_tx, client_connection_rx) = mpsc::unbounded();
|
||||||
|
|
||||||
|
// Shared queue length data. Published by the `OutQueueController` in the client, and used
|
||||||
|
// primarily to throttle incoming connections (e.g socks5 for attached network-requesters)
|
||||||
|
let shared_lane_queue_lengths = LaneQueueLengths::new();
|
||||||
|
|
||||||
|
let mut controller_config = real_messages_control::Config::new(
|
||||||
|
self.debug_config,
|
||||||
|
self.key_manager.ack_key(),
|
||||||
|
self_address,
|
||||||
|
);
|
||||||
|
|
||||||
|
if let Some(size) = self.debug_config.use_extended_packet_size {
|
||||||
|
log::debug!("Setting extended packet size: {:?}", size);
|
||||||
|
controller_config.set_custom_packet_size(size.into());
|
||||||
|
}
|
||||||
|
|
||||||
|
Self::start_real_traffic_controller(
|
||||||
|
controller_config,
|
||||||
|
shared_topology_accessor.clone(),
|
||||||
|
ack_receiver,
|
||||||
|
input_receiver,
|
||||||
|
sphinx_message_sender.clone(),
|
||||||
|
reply_storage,
|
||||||
|
reply_controller_sender,
|
||||||
|
reply_controller_receiver,
|
||||||
|
shared_lane_queue_lengths.clone(),
|
||||||
|
client_connection_rx,
|
||||||
|
shutdown.subscribe(),
|
||||||
|
);
|
||||||
|
|
||||||
|
if !self.debug_config.disable_loop_cover_traffic_stream {
|
||||||
|
Self::start_cover_traffic_stream(
|
||||||
|
self.debug_config,
|
||||||
|
self.key_manager.ack_key(),
|
||||||
|
self_address,
|
||||||
|
shared_topology_accessor,
|
||||||
|
sphinx_message_sender,
|
||||||
|
shutdown.subscribe(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
info!("Client startup finished!");
|
||||||
|
info!("The address of this client is: {self_address}");
|
||||||
|
|
||||||
|
Ok(BaseClient {
|
||||||
|
client_input: ClientInputStatus::AwaitingProducer {
|
||||||
|
client_input: ClientInput {
|
||||||
|
shared_lane_queue_lengths,
|
||||||
|
connection_command_sender: client_connection_tx,
|
||||||
|
input_sender,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
client_output: ClientOutputStatus::AwaitingConsumer {
|
||||||
|
client_output: ClientOutput {
|
||||||
|
received_buffer_request_sender,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
shutdown_notifier: shutdown,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct BaseClient {
|
||||||
|
pub client_input: ClientInputStatus,
|
||||||
|
pub client_output: ClientOutputStatus,
|
||||||
|
|
||||||
|
pub shutdown_notifier: ShutdownNotifier,
|
||||||
|
}
|
||||||
@@ -0,0 +1,51 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use crate::client::replies::reply_storage::{
|
||||||
|
fs_backend, CombinedReplyStorage, ReplyStorageBackend,
|
||||||
|
};
|
||||||
|
use crate::config::DebugConfig;
|
||||||
|
use crate::error::ClientCoreError;
|
||||||
|
use log::{error, info};
|
||||||
|
use std::path::Path;
|
||||||
|
|
||||||
|
pub async fn setup_fs_reply_surb_backend<P: AsRef<Path>>(
|
||||||
|
db_path: P,
|
||||||
|
debug_config: &DebugConfig,
|
||||||
|
) -> Result<fs_backend::Backend, ClientCoreError<fs_backend::Backend>> {
|
||||||
|
// if the database file doesnt exist, initialise fresh storage, otherwise attempt to load the existing one
|
||||||
|
let db_path = db_path.as_ref();
|
||||||
|
if db_path.exists() {
|
||||||
|
info!("loading existing surb database");
|
||||||
|
match fs_backend::Backend::try_load(db_path).await {
|
||||||
|
Ok(backend) => Ok(backend),
|
||||||
|
Err(err) => {
|
||||||
|
error!("failed to setup persistent storage backend for our reply needs: {err}");
|
||||||
|
Err(ClientCoreError::SurbStorageError { source: err })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
info!("creating fresh surb database");
|
||||||
|
let mut storage_backend = match fs_backend::Backend::init(db_path).await {
|
||||||
|
Ok(backend) => backend,
|
||||||
|
Err(err) => {
|
||||||
|
error!("failed to setup persistent storage backend for our reply needs: {err}");
|
||||||
|
return Err(ClientCoreError::SurbStorageError { source: err });
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// while I kinda hate that we're going to be creating `CombinedReplyStorage` twice,
|
||||||
|
// it will only be happening on the very first run and in practice won't incur huge
|
||||||
|
// costs since the storage is going to be empty
|
||||||
|
let mem_store = CombinedReplyStorage::new(
|
||||||
|
debug_config.minimum_reply_surb_storage_threshold,
|
||||||
|
debug_config.maximum_reply_surb_storage_threshold,
|
||||||
|
);
|
||||||
|
storage_backend
|
||||||
|
.init_fresh(&mem_store)
|
||||||
|
.await
|
||||||
|
.map_err(|err| ClientCoreError::SurbStorageError { source: err })?;
|
||||||
|
|
||||||
|
Ok(storage_backend)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -143,6 +143,16 @@ impl LoopCoverTrafficStream<OsRng> {
|
|||||||
self.packet_size = packet_size;
|
self.packet_size = packet_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn set_next_delay(&mut self, amount: Duration) {
|
||||||
|
#[cfg(not(target_arch = "wasm32"))]
|
||||||
|
let next_delay = Box::pin(time::sleep(amount));
|
||||||
|
|
||||||
|
#[cfg(target_arch = "wasm32")]
|
||||||
|
let next_delay = Box::pin(wasm_timer::Delay::new(amount));
|
||||||
|
|
||||||
|
self.next_delay = next_delay;
|
||||||
|
}
|
||||||
|
|
||||||
async fn on_new_message(&mut self) {
|
async fn on_new_message(&mut self) {
|
||||||
trace!("next cover message!");
|
trace!("next cover message!");
|
||||||
|
|
||||||
@@ -151,15 +161,16 @@ impl LoopCoverTrafficStream<OsRng> {
|
|||||||
// poisson delay, but is it really a problem?
|
// poisson delay, but is it really a problem?
|
||||||
let topology_permit = self.topology_access.get_read_permit().await;
|
let topology_permit = self.topology_access.get_read_permit().await;
|
||||||
// the ack is sent back to ourselves (and then ignored)
|
// the ack is sent back to ourselves (and then ignored)
|
||||||
let topology_ref_option = topology_permit.try_get_valid_topology_ref(
|
let topology_ref = match topology_permit.try_get_valid_topology_ref(
|
||||||
&self.our_full_destination,
|
&self.our_full_destination,
|
||||||
Some(&self.our_full_destination),
|
Some(&self.our_full_destination),
|
||||||
);
|
) {
|
||||||
if topology_ref_option.is_none() {
|
Ok(topology) => topology,
|
||||||
warn!("No valid topology detected - won't send any loop cover message this time");
|
Err(err) => {
|
||||||
return;
|
warn!("We're not going to send any loop cover message this time, as the current topology seem to be invalid - {err}");
|
||||||
}
|
return;
|
||||||
let topology_ref = topology_ref_option.unwrap();
|
}
|
||||||
|
};
|
||||||
|
|
||||||
let cover_message = generate_loop_cover_packet(
|
let cover_message = generate_loop_cover_packet(
|
||||||
&mut self.rng,
|
&mut self.rng,
|
||||||
@@ -202,12 +213,11 @@ impl LoopCoverTrafficStream<OsRng> {
|
|||||||
tokio::task::yield_now().await;
|
tokio::task::yield_now().await;
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
pub fn start_with_shutdown(mut self, mut shutdown: task::ShutdownListener) {
|
pub fn start_with_shutdown(mut self, mut shutdown: task::ShutdownListener) {
|
||||||
// we should set initial delay only when we actually start the stream
|
// we should set initial delay only when we actually start the stream
|
||||||
let sampled =
|
let sampled =
|
||||||
sample_poisson_duration(&mut self.rng, self.average_cover_message_sending_delay);
|
sample_poisson_duration(&mut self.rng, self.average_cover_message_sending_delay);
|
||||||
self.next_delay = Box::pin(time::sleep(sampled));
|
self.set_next_delay(sampled);
|
||||||
|
|
||||||
spawn_future(async move {
|
spawn_future(async move {
|
||||||
debug!("Started LoopCoverTrafficStream with graceful shutdown support");
|
debug!("Started LoopCoverTrafficStream with graceful shutdown support");
|
||||||
@@ -228,24 +238,8 @@ impl LoopCoverTrafficStream<OsRng> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
assert!(shutdown.is_shutdown_poll());
|
shutdown.recv_timeout().await;
|
||||||
log::debug!("LoopCoverTrafficStream: Exiting");
|
log::debug!("LoopCoverTrafficStream: Exiting");
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
pub fn start(mut self) {
|
|
||||||
// we should set initial delay only when we actually start the stream
|
|
||||||
let sampled =
|
|
||||||
sample_poisson_duration(&mut self.rng, self.average_cover_message_sending_delay);
|
|
||||||
self.next_delay = Box::pin(wasm_timer::Delay::new(sampled));
|
|
||||||
|
|
||||||
spawn_future(async move {
|
|
||||||
debug!("Started LoopCoverTrafficStream without graceful shutdown support");
|
|
||||||
|
|
||||||
while self.next().await.is_some() {
|
|
||||||
self.on_new_message().await;
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,40 +1,80 @@
|
|||||||
use client_connections::TransmissionLane;
|
use client_connections::TransmissionLane;
|
||||||
use nymsphinx::addressing::clients::Recipient;
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
use nymsphinx::anonymous_replies::ReplySurb;
|
use nymsphinx::anonymous_replies::requests::AnonymousSenderTag;
|
||||||
|
|
||||||
pub type InputMessageSender = tokio::sync::mpsc::Sender<InputMessage>;
|
pub type InputMessageSender = tokio::sync::mpsc::Sender<InputMessage>;
|
||||||
pub type InputMessageReceiver = tokio::sync::mpsc::Receiver<InputMessage>;
|
pub type InputMessageReceiver = tokio::sync::mpsc::Receiver<InputMessage>;
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum InputMessage {
|
pub enum InputMessage {
|
||||||
Fresh {
|
/// The simplest message variant where no additional information is attached.
|
||||||
|
/// You're simply sending your `data` to specified `recipient` without any tagging.
|
||||||
|
///
|
||||||
|
/// Ends up with `NymMessage::Plain` variant
|
||||||
|
Regular {
|
||||||
recipient: Recipient,
|
recipient: Recipient,
|
||||||
data: Vec<u8>,
|
data: Vec<u8>,
|
||||||
with_reply_surb: bool,
|
|
||||||
lane: TransmissionLane,
|
lane: TransmissionLane,
|
||||||
},
|
},
|
||||||
Reply {
|
|
||||||
reply_surb: ReplySurb,
|
/// Creates a message used for a duplex anonymous communication where the recipient
|
||||||
|
/// will never learn of our true identity. This is achieved by carefully sending `reply_surbs`.
|
||||||
|
///
|
||||||
|
/// Note that if reply_surbs is set to zero then
|
||||||
|
/// this variant requires the client having sent some reply_surbs in the past
|
||||||
|
/// (and thus the recipient also knowing our sender tag).
|
||||||
|
///
|
||||||
|
/// Ends up with `NymMessage::Repliable` variant
|
||||||
|
Anonymous {
|
||||||
|
recipient: Recipient,
|
||||||
data: Vec<u8>,
|
data: Vec<u8>,
|
||||||
|
reply_surbs: u32,
|
||||||
|
lane: TransmissionLane,
|
||||||
|
},
|
||||||
|
|
||||||
|
/// Attempt to use our internally received and stored `ReplySurb` to send the message back
|
||||||
|
/// to specified recipient whilst not knowing its full identity (or even gateway).
|
||||||
|
///
|
||||||
|
/// Ends up with `NymMessage::Reply` variant
|
||||||
|
Reply {
|
||||||
|
recipient_tag: AnonymousSenderTag,
|
||||||
|
data: Vec<u8>,
|
||||||
|
lane: TransmissionLane,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
impl InputMessage {
|
impl InputMessage {
|
||||||
pub fn new_fresh(
|
pub fn new_regular(recipient: Recipient, data: Vec<u8>, lane: TransmissionLane) -> Self {
|
||||||
recipient: Recipient,
|
InputMessage::Regular {
|
||||||
data: Vec<u8>,
|
|
||||||
with_reply_surb: bool,
|
|
||||||
lane: TransmissionLane,
|
|
||||||
) -> Self {
|
|
||||||
InputMessage::Fresh {
|
|
||||||
recipient,
|
recipient,
|
||||||
data,
|
data,
|
||||||
with_reply_surb,
|
|
||||||
lane,
|
lane,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn new_reply(reply_surb: ReplySurb, data: Vec<u8>) -> Self {
|
pub fn new_anonymous(
|
||||||
InputMessage::Reply { reply_surb, data }
|
recipient: Recipient,
|
||||||
|
data: Vec<u8>,
|
||||||
|
reply_surbs: u32,
|
||||||
|
lane: TransmissionLane,
|
||||||
|
) -> Self {
|
||||||
|
InputMessage::Anonymous {
|
||||||
|
recipient,
|
||||||
|
data,
|
||||||
|
reply_surbs,
|
||||||
|
lane,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn new_reply(
|
||||||
|
recipient_tag: AnonymousSenderTag,
|
||||||
|
data: Vec<u8>,
|
||||||
|
lane: TransmissionLane,
|
||||||
|
) -> Self {
|
||||||
|
InputMessage::Reply {
|
||||||
|
recipient_tag,
|
||||||
|
data,
|
||||||
|
lane,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -149,6 +149,10 @@ impl KeyManager {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn gateway_key_set(&self) -> bool {
|
||||||
|
self.gateway_shared_key.is_some()
|
||||||
|
}
|
||||||
|
|
||||||
/// Gets an atomically reference counted pointer to [`AckKey`].
|
/// Gets an atomically reference counted pointer to [`AckKey`].
|
||||||
pub fn ack_key(&self) -> Arc<AckKey> {
|
pub fn ack_key(&self) -> Arc<AckKey> {
|
||||||
Arc::clone(&self.ack_key)
|
Arc::clone(&self.ack_key)
|
||||||
|
|||||||
@@ -41,6 +41,16 @@ impl MixTrafficController {
|
|||||||
async fn on_messages(&mut self, mut mix_packets: Vec<MixPacket>) {
|
async fn on_messages(&mut self, mut mix_packets: Vec<MixPacket>) {
|
||||||
debug_assert!(!mix_packets.is_empty());
|
debug_assert!(!mix_packets.is_empty());
|
||||||
|
|
||||||
|
// // simulate some dropped packets
|
||||||
|
// use rand::rngs::OsRng;
|
||||||
|
// use rand::Rng;
|
||||||
|
// let mut rng = OsRng;
|
||||||
|
// let number = rng.gen_range(0, 100);
|
||||||
|
// if number > 95 {
|
||||||
|
// error!("simulating dropped packet");
|
||||||
|
// return;
|
||||||
|
// }
|
||||||
|
|
||||||
let result = if mix_packets.len() == 1 {
|
let result = if mix_packets.len() == 1 {
|
||||||
let mix_packet = mix_packets.pop().unwrap();
|
let mix_packet = mix_packets.pop().unwrap();
|
||||||
self.gateway_client.send_mix_packet(mix_packet).await
|
self.gateway_client.send_mix_packet(mix_packet).await
|
||||||
@@ -67,7 +77,6 @@ impl MixTrafficController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
pub fn start_with_shutdown(mut self, mut shutdown: task::ShutdownListener) {
|
pub fn start_with_shutdown(mut self, mut shutdown: task::ShutdownListener) {
|
||||||
spawn_future(async move {
|
spawn_future(async move {
|
||||||
debug!("Started MixTrafficController with graceful shutdown support");
|
debug!("Started MixTrafficController with graceful shutdown support");
|
||||||
@@ -88,19 +97,8 @@ impl MixTrafficController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
assert!(shutdown.is_shutdown_poll());
|
shutdown.recv_timeout().await;
|
||||||
log::debug!("MixTrafficController: Exiting");
|
log::debug!("MixTrafficController: Exiting");
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
pub fn start(mut self) {
|
|
||||||
spawn_future(async move {
|
|
||||||
debug!("Started MixTrafficController without graceful shutdown support");
|
|
||||||
|
|
||||||
while let Some(mix_packets) = self.mix_rx.recv().await {
|
|
||||||
self.on_messages(mix_packets).await;
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,18 +1,12 @@
|
|||||||
use std::sync::atomic::AtomicBool;
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
pub mod base_client;
|
||||||
pub mod cover_traffic_stream;
|
pub mod cover_traffic_stream;
|
||||||
pub mod inbound_messages;
|
pub mod inbound_messages;
|
||||||
pub mod key_manager;
|
pub mod key_manager;
|
||||||
pub mod mix_traffic;
|
pub mod mix_traffic;
|
||||||
pub mod real_messages_control;
|
pub mod real_messages_control;
|
||||||
pub mod received_buffer;
|
pub mod received_buffer;
|
||||||
#[cfg(feature = "reply-surb")]
|
pub mod replies;
|
||||||
pub mod reply_key_storage;
|
|
||||||
pub mod topology_control;
|
pub mod topology_control;
|
||||||
|
|
||||||
// This is *NOT* used to signal shutdown.
|
|
||||||
// It's critical that we don't have any tasks finishing early, this is an additional safety check
|
|
||||||
// that tasks exiting are doing so because shutdown has been signalled, and no other reason.
|
|
||||||
// In particular for tasks that rely on their associated channel being closed to signal shutdown,
|
|
||||||
// and don't have access to a shutdown listener channel.
|
|
||||||
pub static SHUTDOWN_HAS_BEEN_SIGNALLED: AtomicBool = AtomicBool::new(false);
|
|
||||||
|
|||||||
+4
-19
@@ -1,7 +1,7 @@
|
|||||||
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use super::action_controller::{Action, ActionSender};
|
use super::action_controller::{AckActionSender, Action};
|
||||||
use futures::StreamExt;
|
use futures::StreamExt;
|
||||||
use gateway_client::AcknowledgementReceiver;
|
use gateway_client::AcknowledgementReceiver;
|
||||||
use log::*;
|
use log::*;
|
||||||
@@ -16,14 +16,14 @@ use std::sync::Arc;
|
|||||||
pub(super) struct AcknowledgementListener {
|
pub(super) struct AcknowledgementListener {
|
||||||
ack_key: Arc<AckKey>,
|
ack_key: Arc<AckKey>,
|
||||||
ack_receiver: AcknowledgementReceiver,
|
ack_receiver: AcknowledgementReceiver,
|
||||||
action_sender: ActionSender,
|
action_sender: AckActionSender,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AcknowledgementListener {
|
impl AcknowledgementListener {
|
||||||
pub(super) fn new(
|
pub(super) fn new(
|
||||||
ack_key: Arc<AckKey>,
|
ack_key: Arc<AckKey>,
|
||||||
ack_receiver: AcknowledgementReceiver,
|
ack_receiver: AcknowledgementReceiver,
|
||||||
action_sender: ActionSender,
|
action_sender: AckActionSender,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
AcknowledgementListener {
|
AcknowledgementListener {
|
||||||
ack_key,
|
ack_key,
|
||||||
@@ -49,11 +49,6 @@ impl AcknowledgementListener {
|
|||||||
if frag_id == COVER_FRAG_ID {
|
if frag_id == COVER_FRAG_ID {
|
||||||
trace!("Received an ack for a cover message - no need to do anything");
|
trace!("Received an ack for a cover message - no need to do anything");
|
||||||
return;
|
return;
|
||||||
} else if frag_id.is_reply() {
|
|
||||||
info!("Received an ack for a reply message - no need to do anything! (don't know what to do!)");
|
|
||||||
// TODO: probably there will need to be some extra procedure here, something to notify
|
|
||||||
// user that his reply reached the recipient (since we got an ack)
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
trace!("Received {} from the mix network", frag_id);
|
trace!("Received {} from the mix network", frag_id);
|
||||||
@@ -70,7 +65,6 @@ impl AcknowledgementListener {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
pub(super) async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
pub(super) async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
||||||
debug!("Started AcknowledgementListener with graceful shutdown support");
|
debug!("Started AcknowledgementListener with graceful shutdown support");
|
||||||
|
|
||||||
@@ -88,16 +82,7 @@ impl AcknowledgementListener {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
assert!(shutdown.is_shutdown_poll());
|
shutdown.recv_timeout().await;
|
||||||
log::debug!("AcknowledgementListener: Exiting");
|
log::debug!("AcknowledgementListener: Exiting");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
pub(super) async fn run(&mut self) {
|
|
||||||
debug!("Started AcknowledgementListener without graceful shutdown support");
|
|
||||||
|
|
||||||
while let Some(acks) = self.ack_receiver.next().await {
|
|
||||||
self.handle_ack_receiver_item(acks).await
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
+31
-37
@@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
use super::PendingAcknowledgement;
|
use super::PendingAcknowledgement;
|
||||||
use crate::client::real_messages_control::acknowledgement_control::RetransmissionRequestSender;
|
use crate::client::real_messages_control::acknowledgement_control::RetransmissionRequestSender;
|
||||||
use futures::channel::mpsc::{self, UnboundedReceiver, UnboundedSender};
|
use futures::channel::mpsc;
|
||||||
use futures::StreamExt;
|
use futures::StreamExt;
|
||||||
use log::*;
|
use log::*;
|
||||||
use nonexhaustive_delayqueue::{Expired, NonExhaustiveDelayQueue, QueueKey};
|
use nonexhaustive_delayqueue::{Expired, NonExhaustiveDelayQueue, QueueKey};
|
||||||
@@ -13,7 +13,8 @@ use std::collections::HashMap;
|
|||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
|
||||||
pub(crate) type ActionSender = UnboundedSender<Action>;
|
pub(crate) type AckActionSender = mpsc::UnboundedSender<Action>;
|
||||||
|
pub(crate) type AckActionReceiver = mpsc::UnboundedReceiver<Action>;
|
||||||
|
|
||||||
// The actual data being sent off as well as potential key to the delay queue
|
// The actual data being sent off as well as potential key to the delay queue
|
||||||
type PendingAckEntry = (Arc<PendingAcknowledgement>, Option<QueueKey>);
|
type PendingAckEntry = (Arc<PendingAcknowledgement>, Option<QueueKey>);
|
||||||
@@ -95,7 +96,7 @@ pub(super) struct ActionController {
|
|||||||
pending_acks_timers: NonExhaustiveDelayQueue<FragmentIdentifier>,
|
pending_acks_timers: NonExhaustiveDelayQueue<FragmentIdentifier>,
|
||||||
|
|
||||||
/// Channel for receiving `Action`s from other modules.
|
/// Channel for receiving `Action`s from other modules.
|
||||||
incoming_actions: UnboundedReceiver<Action>,
|
incoming_actions: AckActionReceiver,
|
||||||
|
|
||||||
/// Channel for notifying `RetransmissionRequestListener` about expired acknowledgements.
|
/// Channel for notifying `RetransmissionRequestListener` about expired acknowledgements.
|
||||||
retransmission_sender: RetransmissionRequestSender,
|
retransmission_sender: RetransmissionRequestSender,
|
||||||
@@ -105,18 +106,15 @@ impl ActionController {
|
|||||||
pub(super) fn new(
|
pub(super) fn new(
|
||||||
config: Config,
|
config: Config,
|
||||||
retransmission_sender: RetransmissionRequestSender,
|
retransmission_sender: RetransmissionRequestSender,
|
||||||
) -> (Self, ActionSender) {
|
incoming_actions: AckActionReceiver,
|
||||||
let (sender, receiver) = mpsc::unbounded();
|
) -> Self {
|
||||||
(
|
ActionController {
|
||||||
ActionController {
|
config,
|
||||||
config,
|
pending_acks_data: HashMap::new(),
|
||||||
pending_acks_data: HashMap::new(),
|
pending_acks_timers: NonExhaustiveDelayQueue::new(),
|
||||||
pending_acks_timers: NonExhaustiveDelayQueue::new(),
|
incoming_actions,
|
||||||
incoming_actions: receiver,
|
retransmission_sender,
|
||||||
retransmission_sender,
|
}
|
||||||
},
|
|
||||||
sender,
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn handle_insert(&mut self, pending_acks: Vec<PendingAcknowledgement>) {
|
fn handle_insert(&mut self, pending_acks: Vec<PendingAcknowledgement>) {
|
||||||
@@ -138,13 +136,18 @@ impl ActionController {
|
|||||||
trace!("{} is starting its timer", frag_id);
|
trace!("{} is starting its timer", frag_id);
|
||||||
|
|
||||||
if let Some((pending_ack_data, queue_key)) = self.pending_acks_data.get_mut(&frag_id) {
|
if let Some((pending_ack_data, queue_key)) = self.pending_acks_data.get_mut(&frag_id) {
|
||||||
if queue_key.is_some() {
|
// the fact that this branch is now POSSIBLE is a sign of a need to refactor this whole
|
||||||
// this branch should be IMPOSSIBLE under ANY condition. It would imply starting
|
// retransmission procedure
|
||||||
// timer TWICE for the SAME PendingAcknowledgement
|
//
|
||||||
panic!("Tried to start an already started ack timer!")
|
// (it can happen as timer is started when ack expires to make sure it's not stuck in memory
|
||||||
}
|
// and the second instance can be fired when we finally get reply surbs for data we failed to retransmit)
|
||||||
let timeout = (pending_ack_data.delay.clone() * self.config.ack_wait_multiplier)
|
|
||||||
.to_duration()
|
// if queue_key.is_some() {
|
||||||
|
// // this branch should be IMPOSSIBLE under ANY condition. It would imply starting
|
||||||
|
// // timer TWICE for the SAME PendingAcknowledgement
|
||||||
|
// panic!("Tried to start an already started ack timer!")
|
||||||
|
// }
|
||||||
|
let timeout = (pending_ack_data.delay * self.config.ack_wait_multiplier).to_duration()
|
||||||
+ self.config.ack_wait_addition;
|
+ self.config.ack_wait_addition;
|
||||||
|
|
||||||
let new_queue_key = self.pending_acks_timers.insert(frag_id, timeout);
|
let new_queue_key = self.pending_acks_timers.insert(frag_id, timeout);
|
||||||
@@ -192,7 +195,8 @@ impl ActionController {
|
|||||||
trace!("{} is updating its delay", frag_id);
|
trace!("{} is updating its delay", frag_id);
|
||||||
// TODO: is it possible to solve this without either locking or temporarily removing the value?
|
// TODO: is it possible to solve this without either locking or temporarily removing the value?
|
||||||
if let Some((pending_ack_data, queue_key)) = self.pending_acks_data.remove(&frag_id) {
|
if let Some((pending_ack_data, queue_key)) = self.pending_acks_data.remove(&frag_id) {
|
||||||
// this Action is triggered by `RetransmissionRequestListener` which held the other potential
|
// this Action is triggered by `RetransmissionRequestListener` (for 'normal' packets)
|
||||||
|
// or `ReplyController` (for 'reply' packets) which held the other potential
|
||||||
// reference to this Arc. HOWEVER, before the Action was pushed onto the queue, the reference
|
// reference to this Arc. HOWEVER, before the Action was pushed onto the queue, the reference
|
||||||
// was dropped hence this unwrap is safe.
|
// was dropped hence this unwrap is safe.
|
||||||
let mut inner_data = Arc::try_unwrap(pending_ack_data).unwrap();
|
let mut inner_data = Arc::try_unwrap(pending_ack_data).unwrap();
|
||||||
@@ -245,7 +249,6 @@ impl ActionController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
pub(super) async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
pub(super) async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
||||||
debug!("Started ActionController with graceful shutdown support");
|
debug!("Started ActionController with graceful shutdown support");
|
||||||
|
|
||||||
@@ -272,19 +275,10 @@ impl ActionController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
assert!(shutdown.is_shutdown_poll());
|
#[cfg(not(target_arch = "wasm32"))]
|
||||||
|
tokio::time::timeout(Duration::from_secs(5), shutdown.recv())
|
||||||
|
.await
|
||||||
|
.expect("Task stopped without shutdown called");
|
||||||
log::debug!("ActionController: Exiting");
|
log::debug!("ActionController: Exiting");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
pub(super) async fn run(&mut self) {
|
|
||||||
debug!("Started ActionController without graceful shutdown support");
|
|
||||||
|
|
||||||
loop {
|
|
||||||
tokio::select! {
|
|
||||||
action = self.incoming_actions.next() => self.process_action(action.unwrap()),
|
|
||||||
expired_ack = self.pending_acks_timers.next() => self.handle_expired_ack_timer(expired_ack.unwrap())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
+61
-155
@@ -1,23 +1,14 @@
|
|||||||
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
// Copyright 2021-2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use super::action_controller::{Action, ActionSender};
|
use crate::client::inbound_messages::{InputMessage, InputMessageReceiver};
|
||||||
use super::PendingAcknowledgement;
|
use crate::client::real_messages_control::message_handler::MessageHandler;
|
||||||
use crate::client::{
|
use crate::client::replies::reply_controller::ReplyControllerSender;
|
||||||
inbound_messages::{InputMessage, InputMessageReceiver},
|
|
||||||
real_messages_control::real_traffic_stream::{BatchRealMessageSender, RealMessage},
|
|
||||||
topology_control::TopologyAccessor,
|
|
||||||
};
|
|
||||||
use client_connections::TransmissionLane;
|
use client_connections::TransmissionLane;
|
||||||
use log::*;
|
use log::*;
|
||||||
use nymsphinx::anonymous_replies::ReplySurb;
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
use nymsphinx::preparer::MessagePreparer;
|
use nymsphinx::anonymous_replies::requests::AnonymousSenderTag;
|
||||||
use nymsphinx::{acknowledgements::AckKey, addressing::clients::Recipient};
|
|
||||||
use rand::{CryptoRng, Rng};
|
use rand::{CryptoRng, Rng};
|
||||||
use std::sync::Arc;
|
|
||||||
|
|
||||||
#[cfg(feature = "reply-surb")]
|
|
||||||
use crate::client::reply_key_storage::ReplyKeyStorage;
|
|
||||||
|
|
||||||
/// Module responsible for dealing with the received messages: splitting them, creating acknowledgements,
|
/// Module responsible for dealing with the received messages: splitting them, creating acknowledgements,
|
||||||
/// putting everything into sphinx packets, etc.
|
/// putting everything into sphinx packets, etc.
|
||||||
@@ -26,15 +17,9 @@ pub(super) struct InputMessageListener<R>
|
|||||||
where
|
where
|
||||||
R: CryptoRng + Rng,
|
R: CryptoRng + Rng,
|
||||||
{
|
{
|
||||||
ack_key: Arc<AckKey>,
|
|
||||||
ack_recipient: Recipient,
|
|
||||||
input_receiver: InputMessageReceiver,
|
input_receiver: InputMessageReceiver,
|
||||||
message_preparer: MessagePreparer<R>,
|
message_handler: MessageHandler<R>,
|
||||||
action_sender: ActionSender,
|
reply_controller_sender: ReplyControllerSender,
|
||||||
real_message_sender: BatchRealMessageSender,
|
|
||||||
topology_access: TopologyAccessor,
|
|
||||||
#[cfg(feature = "reply-surb")]
|
|
||||||
reply_key_storage: ReplyKeyStorage,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<R> InputMessageListener<R>
|
impl<R> InputMessageListener<R>
|
||||||
@@ -45,156 +30,85 @@ where
|
|||||||
// some considerable refactoring
|
// some considerable refactoring
|
||||||
#[allow(clippy::too_many_arguments)]
|
#[allow(clippy::too_many_arguments)]
|
||||||
pub(super) fn new(
|
pub(super) fn new(
|
||||||
ack_key: Arc<AckKey>,
|
|
||||||
ack_recipient: Recipient,
|
|
||||||
input_receiver: InputMessageReceiver,
|
input_receiver: InputMessageReceiver,
|
||||||
message_preparer: MessagePreparer<R>,
|
message_handler: MessageHandler<R>,
|
||||||
action_sender: ActionSender,
|
reply_controller_sender: ReplyControllerSender,
|
||||||
real_message_sender: BatchRealMessageSender,
|
|
||||||
topology_access: TopologyAccessor,
|
|
||||||
#[cfg(feature = "reply-surb")] reply_key_storage: ReplyKeyStorage,
|
|
||||||
) -> Self {
|
) -> Self {
|
||||||
InputMessageListener {
|
InputMessageListener {
|
||||||
ack_key,
|
|
||||||
ack_recipient,
|
|
||||||
input_receiver,
|
input_receiver,
|
||||||
message_preparer,
|
message_handler,
|
||||||
action_sender,
|
reply_controller_sender,
|
||||||
real_message_sender,
|
|
||||||
topology_access,
|
|
||||||
#[cfg(feature = "reply-surb")]
|
|
||||||
reply_key_storage,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// we require topology for replies to generate surb_acks
|
async fn handle_reply(
|
||||||
async fn handle_reply(&mut self, reply_surb: ReplySurb, data: Vec<u8>) -> Option<RealMessage> {
|
&mut self,
|
||||||
let topology_permit = self.topology_access.get_read_permit().await;
|
recipient_tag: AnonymousSenderTag,
|
||||||
let topology = match topology_permit.try_get_valid_topology_ref(&self.ack_recipient, None) {
|
data: Vec<u8>,
|
||||||
Some(topology_ref) => topology_ref,
|
lane: TransmissionLane,
|
||||||
None => {
|
) {
|
||||||
warn!("Could not process the message - the network topology is invalid");
|
// offload reply handling to the dedicated task
|
||||||
return None;
|
self.reply_controller_sender
|
||||||
}
|
.send_reply(recipient_tag, data, lane)
|
||||||
};
|
|
||||||
|
|
||||||
match self
|
|
||||||
.message_preparer
|
|
||||||
.prepare_reply_for_use(data, reply_surb, topology, &self.ack_key)
|
|
||||||
.await
|
|
||||||
{
|
|
||||||
Ok((mix_packet, reply_id)) => {
|
|
||||||
// TODO: later probably write pending ack here
|
|
||||||
// and deal with them....
|
|
||||||
// ... somehow
|
|
||||||
Some(RealMessage::new(mix_packet, reply_id))
|
|
||||||
}
|
|
||||||
Err(err) => {
|
|
||||||
// TODO: should we have some mechanism to indicate to the user that the `reply_surb`
|
|
||||||
// could be reused since technically it wasn't used up here?
|
|
||||||
warn!("failed to deal with received reply surb - {:?}", err);
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn handle_fresh_message(
|
async fn handle_plain_message(
|
||||||
&mut self,
|
&mut self,
|
||||||
recipient: Recipient,
|
recipient: Recipient,
|
||||||
content: Vec<u8>,
|
content: Vec<u8>,
|
||||||
with_reply_surb: bool,
|
lane: TransmissionLane,
|
||||||
) -> Option<Vec<RealMessage>> {
|
) {
|
||||||
log::trace!("handling msg size: {}", content.len());
|
if let Err(err) = self
|
||||||
let topology_permit = self.topology_access.get_read_permit().await;
|
.message_handler
|
||||||
let topology = match topology_permit
|
.try_send_plain_message(recipient, content, lane)
|
||||||
.try_get_valid_topology_ref(&self.ack_recipient, Some(&recipient))
|
.await
|
||||||
{
|
{
|
||||||
Some(topology_ref) => topology_ref,
|
warn!("failed to send a plain message - {err}")
|
||||||
None => {
|
|
||||||
warn!("Could not process the message - the network topology is invalid");
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// split the message, attach optional reply surb
|
|
||||||
let (split_message, reply_key) = self
|
|
||||||
.message_preparer
|
|
||||||
.prepare_and_split_message(content, with_reply_surb, topology)
|
|
||||||
.expect("somehow the topology was invalid after all!");
|
|
||||||
|
|
||||||
#[cfg(feature = "reply-surb")]
|
|
||||||
if let Some(reply_key) = reply_key {
|
|
||||||
self.reply_key_storage
|
|
||||||
.insert_encryption_key(reply_key)
|
|
||||||
.expect("Failed to insert surb reply key to the store!")
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(not(feature = "reply-surb"))]
|
async fn handle_repliable_message(
|
||||||
let _reply_key = reply_key;
|
&mut self,
|
||||||
|
recipient: Recipient,
|
||||||
// encrypt chunks, put them inside sphinx packets and generate acks
|
content: Vec<u8>,
|
||||||
let mut pending_acks = Vec::with_capacity(split_message.len());
|
reply_surbs: u32,
|
||||||
let mut real_messages = Vec::with_capacity(split_message.len());
|
lane: TransmissionLane,
|
||||||
for message_chunk in split_message {
|
) {
|
||||||
// we need to clone it because we need to keep it in memory in case we had to retransmit
|
if let Err(err) = self
|
||||||
// it. And then we'd need to recreate entire ACK again.
|
.message_handler
|
||||||
let chunk_clone = message_chunk.clone();
|
.try_send_message_with_reply_surbs(recipient, content, reply_surbs, lane)
|
||||||
let prepared_fragment = self
|
.await
|
||||||
.message_preparer
|
{
|
||||||
.prepare_chunk_for_sending(chunk_clone, topology, &self.ack_key, &recipient)
|
warn!("failed to send a repliable message - {err}")
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
real_messages.push(RealMessage::new(
|
|
||||||
prepared_fragment.mix_packet,
|
|
||||||
message_chunk.fragment_identifier(),
|
|
||||||
));
|
|
||||||
|
|
||||||
pending_acks.push(PendingAcknowledgement::new(
|
|
||||||
message_chunk,
|
|
||||||
prepared_fragment.total_delay,
|
|
||||||
recipient,
|
|
||||||
));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// tells the controller to put this into the hashmap
|
|
||||||
self.action_sender
|
|
||||||
.unbounded_send(Action::new_insert(pending_acks))
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
Some(real_messages)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn on_input_message(&mut self, msg: InputMessage) {
|
async fn on_input_message(&mut self, msg: InputMessage) {
|
||||||
let (real_messages, lane) = match msg {
|
match msg {
|
||||||
InputMessage::Fresh {
|
InputMessage::Regular {
|
||||||
recipient,
|
recipient,
|
||||||
data,
|
data,
|
||||||
with_reply_surb,
|
|
||||||
lane,
|
lane,
|
||||||
} => (
|
} => self.handle_plain_message(recipient, data, lane).await,
|
||||||
self.handle_fresh_message(recipient, data, with_reply_surb)
|
InputMessage::Anonymous {
|
||||||
.await,
|
recipient,
|
||||||
|
data,
|
||||||
|
reply_surbs,
|
||||||
lane,
|
lane,
|
||||||
),
|
} => {
|
||||||
InputMessage::Reply { reply_surb, data } => (
|
self.handle_repliable_message(recipient, data, reply_surbs, lane)
|
||||||
self.handle_reply(reply_surb, data)
|
|
||||||
.await
|
.await
|
||||||
.map(|message| vec![message]),
|
}
|
||||||
TransmissionLane::Reply,
|
InputMessage::Reply {
|
||||||
),
|
recipient_tag,
|
||||||
|
data,
|
||||||
|
lane,
|
||||||
|
} => {
|
||||||
|
self.handle_reply(recipient_tag, data, lane).await;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// there's no point in trying to send nothing
|
|
||||||
if let Some(real_messages) = real_messages {
|
|
||||||
// tells real message sender (with the poisson timer) to send this to the mix network
|
|
||||||
self.real_message_sender
|
|
||||||
.send((real_messages, lane))
|
|
||||||
.await
|
|
||||||
.expect("BatchRealMessageReceiver has stopped receiving!");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
pub(super) async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
pub(super) async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
||||||
debug!("Started InputMessageListener with graceful shutdown support");
|
debug!("Started InputMessageListener with graceful shutdown support");
|
||||||
|
|
||||||
@@ -214,15 +128,7 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
assert!(shutdown.is_shutdown_poll());
|
shutdown.recv_timeout().await;
|
||||||
log::debug!("InputMessageListener: Exiting");
|
log::debug!("InputMessageListener: Exiting");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
pub(super) async fn run(&mut self) {
|
|
||||||
debug!("Started InputMessageListener without graceful shutdown support");
|
|
||||||
while let Some(input_msg) = self.input_receiver.recv().await {
|
|
||||||
self.on_input_message(input_msg).await;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
+75
-91
@@ -7,18 +7,20 @@ use self::{
|
|||||||
retransmission_request_listener::RetransmissionRequestListener,
|
retransmission_request_listener::RetransmissionRequestListener,
|
||||||
sent_notification_listener::SentNotificationListener,
|
sent_notification_listener::SentNotificationListener,
|
||||||
};
|
};
|
||||||
use super::real_traffic_stream::BatchRealMessageSender;
|
use crate::client::inbound_messages::InputMessageReceiver;
|
||||||
use crate::client::{inbound_messages::InputMessageReceiver, topology_control::TopologyAccessor};
|
use crate::client::real_messages_control::message_handler::MessageHandler;
|
||||||
|
use crate::client::replies::reply_controller::ReplyControllerSender;
|
||||||
use crate::spawn_future;
|
use crate::spawn_future;
|
||||||
|
use action_controller::AckActionReceiver;
|
||||||
use futures::channel::mpsc;
|
use futures::channel::mpsc;
|
||||||
use gateway_client::AcknowledgementReceiver;
|
use gateway_client::AcknowledgementReceiver;
|
||||||
use log::*;
|
use log::*;
|
||||||
|
use nymsphinx::anonymous_replies::requests::AnonymousSenderTag;
|
||||||
use nymsphinx::params::PacketSize;
|
use nymsphinx::params::PacketSize;
|
||||||
use nymsphinx::{
|
use nymsphinx::{
|
||||||
acknowledgements::AckKey,
|
acknowledgements::AckKey,
|
||||||
addressing::clients::Recipient,
|
addressing::clients::Recipient,
|
||||||
chunking::fragment::{Fragment, FragmentIdentifier},
|
chunking::fragment::{Fragment, FragmentIdentifier},
|
||||||
preparer::MessagePreparer,
|
|
||||||
Delay as SphinxDelay,
|
Delay as SphinxDelay,
|
||||||
};
|
};
|
||||||
use rand::{CryptoRng, Rng};
|
use rand::{CryptoRng, Rng};
|
||||||
@@ -27,8 +29,7 @@ use std::{
|
|||||||
time::Duration,
|
time::Duration,
|
||||||
};
|
};
|
||||||
|
|
||||||
#[cfg(feature = "reply-surb")]
|
pub(crate) use action_controller::{AckActionSender, Action};
|
||||||
use crate::client::reply_key_storage::ReplyKeyStorage;
|
|
||||||
|
|
||||||
mod acknowledgement_listener;
|
mod acknowledgement_listener;
|
||||||
mod action_controller;
|
mod action_controller;
|
||||||
@@ -50,24 +51,64 @@ pub(super) type SentPacketNotificationSender = mpsc::UnboundedSender<FragmentIde
|
|||||||
/// that it is about to be sent to the mix network and its timeout timer should be started.
|
/// that it is about to be sent to the mix network and its timeout timer should be started.
|
||||||
type SentPacketNotificationReceiver = mpsc::UnboundedReceiver<FragmentIdentifier>;
|
type SentPacketNotificationReceiver = mpsc::UnboundedReceiver<FragmentIdentifier>;
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub(crate) enum PacketDestination {
|
||||||
|
Anonymous {
|
||||||
|
recipient_tag: AnonymousSenderTag,
|
||||||
|
// special flag to indicate whether this was an ack for requesting additional surbs,
|
||||||
|
// in that case we have to do everything we can to get it through, even if it means going
|
||||||
|
// below our stored reply surb threshold
|
||||||
|
extra_surb_request: bool,
|
||||||
|
},
|
||||||
|
KnownRecipient(Box<Recipient>),
|
||||||
|
}
|
||||||
|
|
||||||
/// Structure representing a data `Fragment` that is on-route to the specified `Recipient`
|
/// Structure representing a data `Fragment` that is on-route to the specified `Recipient`
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub(crate) struct PendingAcknowledgement {
|
pub(crate) struct PendingAcknowledgement {
|
||||||
message_chunk: Fragment,
|
message_chunk: Fragment,
|
||||||
delay: SphinxDelay,
|
delay: SphinxDelay,
|
||||||
recipient: Recipient,
|
destination: PacketDestination,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl PendingAcknowledgement {
|
impl PendingAcknowledgement {
|
||||||
/// Creates new instance of `PendingAcknowledgement` using the provided data.
|
/// Creates new instance of `PendingAcknowledgement` using the provided data.
|
||||||
fn new(message_chunk: Fragment, delay: SphinxDelay, recipient: Recipient) -> Self {
|
pub(crate) fn new_known(
|
||||||
|
message_chunk: Fragment,
|
||||||
|
delay: SphinxDelay,
|
||||||
|
recipient: Recipient,
|
||||||
|
) -> Self {
|
||||||
PendingAcknowledgement {
|
PendingAcknowledgement {
|
||||||
message_chunk,
|
message_chunk,
|
||||||
delay,
|
delay,
|
||||||
recipient,
|
destination: PacketDestination::KnownRecipient(recipient.into()),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) fn new_anonymous(
|
||||||
|
message_chunk: Fragment,
|
||||||
|
delay: SphinxDelay,
|
||||||
|
recipient_tag: AnonymousSenderTag,
|
||||||
|
extra_surb_request: bool,
|
||||||
|
) -> Self {
|
||||||
|
PendingAcknowledgement {
|
||||||
|
message_chunk,
|
||||||
|
delay,
|
||||||
|
destination: PacketDestination::Anonymous {
|
||||||
|
recipient_tag,
|
||||||
|
extra_surb_request,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn inner_fragment_identifier(&self) -> FragmentIdentifier {
|
||||||
|
self.message_chunk.fragment_identifier()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn fragment_data(&self) -> Fragment {
|
||||||
|
self.message_chunk.clone()
|
||||||
|
}
|
||||||
|
|
||||||
fn update_delay(&mut self, new_delay: SphinxDelay) {
|
fn update_delay(&mut self, new_delay: SphinxDelay) {
|
||||||
self.delay = new_delay;
|
self.delay = new_delay;
|
||||||
}
|
}
|
||||||
@@ -76,10 +117,6 @@ impl PendingAcknowledgement {
|
|||||||
/// AcknowledgementControllerConnectors represents set of channels for communication with
|
/// AcknowledgementControllerConnectors represents set of channels for communication with
|
||||||
/// other parts of the system in order to support acknowledgements and retransmission.
|
/// other parts of the system in order to support acknowledgements and retransmission.
|
||||||
pub(super) struct AcknowledgementControllerConnectors {
|
pub(super) struct AcknowledgementControllerConnectors {
|
||||||
/// Channel used for forwarding prepared sphinx messages into the poisson sender
|
|
||||||
/// to be sent to the mix network.
|
|
||||||
real_message_sender: BatchRealMessageSender,
|
|
||||||
|
|
||||||
/// Channel used for receiving raw messages from a client. The messages need to be put
|
/// Channel used for receiving raw messages from a client. The messages need to be put
|
||||||
/// into sphinx packets first.
|
/// into sphinx packets first.
|
||||||
input_receiver: InputMessageReceiver,
|
input_receiver: InputMessageReceiver,
|
||||||
@@ -91,20 +128,28 @@ pub(super) struct AcknowledgementControllerConnectors {
|
|||||||
|
|
||||||
/// Channel used for receiving acknowledgements from the mix network.
|
/// Channel used for receiving acknowledgements from the mix network.
|
||||||
ack_receiver: AcknowledgementReceiver,
|
ack_receiver: AcknowledgementReceiver,
|
||||||
|
|
||||||
|
/// Channel used for sending request to `ActionController` to deal with anything ack-related,
|
||||||
|
ack_action_sender: AckActionSender,
|
||||||
|
|
||||||
|
/// Channel used for receiving request by `ActionController` to deal with anything ack-related,
|
||||||
|
ack_action_receiver: AckActionReceiver,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AcknowledgementControllerConnectors {
|
impl AcknowledgementControllerConnectors {
|
||||||
pub(super) fn new(
|
pub(super) fn new(
|
||||||
real_message_sender: BatchRealMessageSender,
|
|
||||||
input_receiver: InputMessageReceiver,
|
input_receiver: InputMessageReceiver,
|
||||||
sent_notifier: SentPacketNotificationReceiver,
|
sent_notifier: SentPacketNotificationReceiver,
|
||||||
ack_receiver: AcknowledgementReceiver,
|
ack_receiver: AcknowledgementReceiver,
|
||||||
|
ack_action_sender: AckActionSender,
|
||||||
|
ack_action_receiver: AckActionReceiver,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
AcknowledgementControllerConnectors {
|
AcknowledgementControllerConnectors {
|
||||||
real_message_sender,
|
|
||||||
input_receiver,
|
input_receiver,
|
||||||
sent_notifier,
|
sent_notifier,
|
||||||
ack_receiver,
|
ack_receiver,
|
||||||
|
ack_action_sender,
|
||||||
|
ack_action_receiver,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -117,28 +162,15 @@ pub(super) struct Config {
|
|||||||
/// Given ack timeout in the form a * BASE_DELAY + b, it specifies the multiplier `a`
|
/// Given ack timeout in the form a * BASE_DELAY + b, it specifies the multiplier `a`
|
||||||
ack_wait_multiplier: f64,
|
ack_wait_multiplier: f64,
|
||||||
|
|
||||||
/// Average delay an acknowledgement packet is going to get delayed at a single mixnode.
|
|
||||||
average_ack_delay: Duration,
|
|
||||||
|
|
||||||
/// Average delay a data packet is going to get delayed at a single mixnode.
|
|
||||||
average_packet_delay: Duration,
|
|
||||||
|
|
||||||
/// Predefined packet size used for the encapsulated messages.
|
/// Predefined packet size used for the encapsulated messages.
|
||||||
packet_size: PacketSize,
|
packet_size: PacketSize,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Config {
|
impl Config {
|
||||||
pub(super) fn new(
|
pub(super) fn new(ack_wait_addition: Duration, ack_wait_multiplier: f64) -> Self {
|
||||||
ack_wait_addition: Duration,
|
|
||||||
ack_wait_multiplier: f64,
|
|
||||||
average_ack_delay: Duration,
|
|
||||||
average_packet_delay: Duration,
|
|
||||||
) -> Self {
|
|
||||||
Config {
|
Config {
|
||||||
ack_wait_addition,
|
ack_wait_addition,
|
||||||
ack_wait_multiplier,
|
ack_wait_multiplier,
|
||||||
average_ack_delay,
|
|
||||||
average_packet_delay,
|
|
||||||
packet_size: Default::default(),
|
packet_size: Default::default(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -162,68 +194,51 @@ where
|
|||||||
|
|
||||||
impl<R> AcknowledgementController<R>
|
impl<R> AcknowledgementController<R>
|
||||||
where
|
where
|
||||||
R: 'static + CryptoRng + Rng + Clone + Send,
|
R: 'static + CryptoRng + Rng + Clone + Send + Sync,
|
||||||
{
|
{
|
||||||
#[allow(clippy::too_many_arguments)]
|
|
||||||
pub(super) fn new(
|
pub(super) fn new(
|
||||||
config: Config,
|
config: Config,
|
||||||
rng: R,
|
|
||||||
topology_access: TopologyAccessor,
|
|
||||||
ack_key: Arc<AckKey>,
|
ack_key: Arc<AckKey>,
|
||||||
ack_recipient: Recipient,
|
|
||||||
connectors: AcknowledgementControllerConnectors,
|
connectors: AcknowledgementControllerConnectors,
|
||||||
#[cfg(feature = "reply-surb")] reply_key_storage: ReplyKeyStorage,
|
message_handler: MessageHandler<R>,
|
||||||
|
reply_controller_sender: ReplyControllerSender,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
let (retransmission_tx, retransmission_rx) = mpsc::unbounded();
|
let (retransmission_tx, retransmission_rx) = mpsc::unbounded();
|
||||||
|
|
||||||
let action_config =
|
let action_config =
|
||||||
action_controller::Config::new(config.ack_wait_addition, config.ack_wait_multiplier);
|
action_controller::Config::new(config.ack_wait_addition, config.ack_wait_multiplier);
|
||||||
let (action_controller, action_sender) =
|
let action_controller = ActionController::new(
|
||||||
ActionController::new(action_config, retransmission_tx);
|
action_config,
|
||||||
|
retransmission_tx,
|
||||||
let message_preparer = MessagePreparer::new(
|
connectors.ack_action_receiver,
|
||||||
rng,
|
);
|
||||||
ack_recipient,
|
|
||||||
config.average_packet_delay,
|
|
||||||
config.average_ack_delay,
|
|
||||||
)
|
|
||||||
.with_custom_real_message_packet_size(config.packet_size);
|
|
||||||
|
|
||||||
// will listen for any acks coming from the network
|
// will listen for any acks coming from the network
|
||||||
let acknowledgement_listener = AcknowledgementListener::new(
|
let acknowledgement_listener = AcknowledgementListener::new(
|
||||||
Arc::clone(&ack_key),
|
Arc::clone(&ack_key),
|
||||||
connectors.ack_receiver,
|
connectors.ack_receiver,
|
||||||
action_sender.clone(),
|
connectors.ack_action_sender.clone(),
|
||||||
);
|
);
|
||||||
|
|
||||||
// will listen for any new messages from the client
|
// will listen for any new messages from the client
|
||||||
let input_message_listener = InputMessageListener::new(
|
let input_message_listener = InputMessageListener::new(
|
||||||
Arc::clone(&ack_key),
|
|
||||||
ack_recipient,
|
|
||||||
connectors.input_receiver,
|
connectors.input_receiver,
|
||||||
message_preparer.clone(),
|
message_handler.clone(),
|
||||||
action_sender.clone(),
|
reply_controller_sender.clone(),
|
||||||
connectors.real_message_sender.clone(),
|
|
||||||
topology_access.clone(),
|
|
||||||
#[cfg(feature = "reply-surb")]
|
|
||||||
reply_key_storage,
|
|
||||||
);
|
);
|
||||||
|
|
||||||
// will listen for any ack timeouts and trigger retransmission
|
// will listen for any ack timeouts and trigger retransmission
|
||||||
let retransmission_request_listener = RetransmissionRequestListener::new(
|
let retransmission_request_listener = RetransmissionRequestListener::new(
|
||||||
Arc::clone(&ack_key),
|
connectors.ack_action_sender.clone(),
|
||||||
ack_recipient,
|
message_handler,
|
||||||
message_preparer,
|
|
||||||
action_sender.clone(),
|
|
||||||
connectors.real_message_sender,
|
|
||||||
retransmission_rx,
|
retransmission_rx,
|
||||||
topology_access,
|
reply_controller_sender,
|
||||||
);
|
);
|
||||||
|
|
||||||
// will listen for events indicating the packet was sent through the network so that
|
// will listen for events indicating the packet was sent through the network so that
|
||||||
// the retransmission timer should be started.
|
// the retransmission timer should be started.
|
||||||
let sent_notification_listener =
|
let sent_notification_listener =
|
||||||
SentNotificationListener::new(connectors.sent_notifier, action_sender);
|
SentNotificationListener::new(connectors.sent_notifier, connectors.ack_action_sender);
|
||||||
|
|
||||||
AcknowledgementController {
|
AcknowledgementController {
|
||||||
acknowledgement_listener,
|
acknowledgement_listener,
|
||||||
@@ -234,7 +249,6 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
pub(super) fn start_with_shutdown(self, shutdown: task::ShutdownListener) {
|
pub(super) fn start_with_shutdown(self, shutdown: task::ShutdownListener) {
|
||||||
let mut acknowledgement_listener = self.acknowledgement_listener;
|
let mut acknowledgement_listener = self.acknowledgement_listener;
|
||||||
let mut input_message_listener = self.input_message_listener;
|
let mut input_message_listener = self.input_message_listener;
|
||||||
@@ -279,34 +293,4 @@ where
|
|||||||
debug!("The controller has finished execution!");
|
debug!("The controller has finished execution!");
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
pub(super) fn start(self) {
|
|
||||||
let mut acknowledgement_listener = self.acknowledgement_listener;
|
|
||||||
let mut input_message_listener = self.input_message_listener;
|
|
||||||
let mut retransmission_request_listener = self.retransmission_request_listener;
|
|
||||||
let mut sent_notification_listener = self.sent_notification_listener;
|
|
||||||
let mut action_controller = self.action_controller;
|
|
||||||
|
|
||||||
spawn_future(async move {
|
|
||||||
acknowledgement_listener.run().await;
|
|
||||||
error!("The acknowledgement listener has finished execution!");
|
|
||||||
});
|
|
||||||
spawn_future(async move {
|
|
||||||
input_message_listener.run().await;
|
|
||||||
error!("The input listener has finished execution!");
|
|
||||||
});
|
|
||||||
spawn_future(async move {
|
|
||||||
retransmission_request_listener.run().await;
|
|
||||||
error!("The retransmission request listener has finished execution!");
|
|
||||||
});
|
|
||||||
spawn_future(async move {
|
|
||||||
sent_notification_listener.run().await;
|
|
||||||
error!("The sent notification listener has finished execution!");
|
|
||||||
});
|
|
||||||
spawn_future(async move {
|
|
||||||
action_controller.run().await;
|
|
||||||
error!("The controller has finished execution!");
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
+67
-65
@@ -1,82 +1,101 @@
|
|||||||
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
// Copyright 2021-2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
action_controller::{Action, ActionSender},
|
action_controller::{AckActionSender, Action},
|
||||||
PendingAcknowledgement, RetransmissionRequestReceiver,
|
PendingAcknowledgement, RetransmissionRequestReceiver,
|
||||||
};
|
};
|
||||||
use crate::client::{
|
use crate::client::real_messages_control::acknowledgement_control::PacketDestination;
|
||||||
real_messages_control::real_traffic_stream::{BatchRealMessageSender, RealMessage},
|
use crate::client::real_messages_control::message_handler::{MessageHandler, PreparationError};
|
||||||
topology_control::TopologyAccessor,
|
use crate::client::real_messages_control::real_traffic_stream::RealMessage;
|
||||||
};
|
use crate::client::replies::reply_controller::ReplyControllerSender;
|
||||||
|
|
||||||
use client_connections::TransmissionLane;
|
use client_connections::TransmissionLane;
|
||||||
use futures::StreamExt;
|
use futures::StreamExt;
|
||||||
use log::*;
|
use log::*;
|
||||||
use nymsphinx::{
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
acknowledgements::AckKey, addressing::clients::Recipient, preparer::MessagePreparer,
|
use nymsphinx::chunking::fragment::Fragment;
|
||||||
};
|
use nymsphinx::preparer::PreparedFragment;
|
||||||
use rand::{CryptoRng, Rng};
|
use rand::{CryptoRng, Rng};
|
||||||
use std::sync::{Arc, Weak};
|
use std::sync::{Arc, Weak};
|
||||||
|
|
||||||
// responsible for packet retransmission upon fired timer
|
// responsible for packet retransmission upon fired timer
|
||||||
pub(super) struct RetransmissionRequestListener<R>
|
pub(super) struct RetransmissionRequestListener<R> {
|
||||||
where
|
action_sender: AckActionSender,
|
||||||
R: CryptoRng + Rng,
|
message_handler: MessageHandler<R>,
|
||||||
{
|
|
||||||
ack_key: Arc<AckKey>,
|
|
||||||
ack_recipient: Recipient,
|
|
||||||
message_preparer: MessagePreparer<R>,
|
|
||||||
action_sender: ActionSender,
|
|
||||||
real_message_sender: BatchRealMessageSender,
|
|
||||||
request_receiver: RetransmissionRequestReceiver,
|
request_receiver: RetransmissionRequestReceiver,
|
||||||
topology_access: TopologyAccessor,
|
reply_controller_sender: ReplyControllerSender,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<R> RetransmissionRequestListener<R>
|
impl<R> RetransmissionRequestListener<R>
|
||||||
where
|
where
|
||||||
R: CryptoRng + Rng,
|
R: CryptoRng + Rng,
|
||||||
{
|
{
|
||||||
#[allow(clippy::too_many_arguments)]
|
|
||||||
pub(super) fn new(
|
pub(super) fn new(
|
||||||
ack_key: Arc<AckKey>,
|
action_sender: AckActionSender,
|
||||||
ack_recipient: Recipient,
|
message_handler: MessageHandler<R>,
|
||||||
message_preparer: MessagePreparer<R>,
|
|
||||||
action_sender: ActionSender,
|
|
||||||
real_message_sender: BatchRealMessageSender,
|
|
||||||
request_receiver: RetransmissionRequestReceiver,
|
request_receiver: RetransmissionRequestReceiver,
|
||||||
topology_access: TopologyAccessor,
|
reply_controller_sender: ReplyControllerSender,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
RetransmissionRequestListener {
|
RetransmissionRequestListener {
|
||||||
ack_key,
|
|
||||||
ack_recipient,
|
|
||||||
message_preparer,
|
|
||||||
action_sender,
|
action_sender,
|
||||||
real_message_sender,
|
message_handler,
|
||||||
request_receiver,
|
request_receiver,
|
||||||
topology_access,
|
reply_controller_sender,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn on_retransmission_request(&mut self, timed_out_ack: Weak<PendingAcknowledgement>) {
|
async fn prepare_normal_retransmission_chunk(
|
||||||
let timed_out_ack = match timed_out_ack.upgrade() {
|
&mut self,
|
||||||
|
packet_recipient: Recipient,
|
||||||
|
chunk_data: Fragment,
|
||||||
|
) -> Result<PreparedFragment, PreparationError> {
|
||||||
|
debug!("retransmitting normal packet...");
|
||||||
|
|
||||||
|
self.message_handler
|
||||||
|
.try_prepare_single_chunk_for_sending(packet_recipient, chunk_data)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn on_retransmission_request(
|
||||||
|
&mut self,
|
||||||
|
weak_timed_out_ack: Weak<PendingAcknowledgement>,
|
||||||
|
) {
|
||||||
|
let timed_out_ack = match weak_timed_out_ack.upgrade() {
|
||||||
Some(timed_out_ack) => timed_out_ack,
|
Some(timed_out_ack) => timed_out_ack,
|
||||||
None => {
|
None => {
|
||||||
debug!("We received an ack JUST as we were about to retransmit [1]");
|
debug!("We received an ack JUST as we were about to retransmit [1]");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
let packet_recipient = &timed_out_ack.recipient;
|
|
||||||
let chunk_clone = timed_out_ack.message_chunk.clone();
|
|
||||||
let frag_id = chunk_clone.fragment_identifier();
|
|
||||||
|
|
||||||
let topology_permit = self.topology_access.get_read_permit().await;
|
let maybe_prepared_fragment = match &timed_out_ack.destination {
|
||||||
let topology_ref = match topology_permit
|
PacketDestination::Anonymous {
|
||||||
.try_get_valid_topology_ref(&self.ack_recipient, Some(packet_recipient))
|
recipient_tag,
|
||||||
{
|
extra_surb_request,
|
||||||
Some(topology_ref) => topology_ref,
|
} => {
|
||||||
None => {
|
// if this is retransmission for reply, offload it to the dedicated task
|
||||||
warn!("Could not retransmit the packet - the network topology is invalid");
|
// that deals with all the surbs
|
||||||
|
return self.reply_controller_sender.send_retransmission_data(
|
||||||
|
*recipient_tag,
|
||||||
|
weak_timed_out_ack,
|
||||||
|
*extra_surb_request,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
PacketDestination::KnownRecipient(recipient) => {
|
||||||
|
self.prepare_normal_retransmission_chunk(
|
||||||
|
**recipient,
|
||||||
|
timed_out_ack.message_chunk.clone(),
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let frag_id = timed_out_ack.message_chunk.fragment_identifier();
|
||||||
|
|
||||||
|
let prepared_fragment = match maybe_prepared_fragment {
|
||||||
|
Ok(prepared_fragment) => prepared_fragment,
|
||||||
|
Err(err) => {
|
||||||
|
warn!("Could not retransmit the packet - {err}");
|
||||||
// we NEED to start timer here otherwise we will have this guy permanently stuck in memory
|
// we NEED to start timer here otherwise we will have this guy permanently stuck in memory
|
||||||
self.action_sender
|
self.action_sender
|
||||||
.unbounded_send(Action::new_start_timer(frag_id))
|
.unbounded_send(Action::new_start_timer(frag_id))
|
||||||
@@ -85,11 +104,6 @@ where
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
let prepared_fragment = self
|
|
||||||
.message_preparer
|
|
||||||
.prepare_chunk_for_sending(chunk_clone, topology_ref, &self.ack_key, packet_recipient)
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
// if we have the ONLY strong reference to the ack data, it means it was removed from the
|
// if we have the ONLY strong reference to the ack data, it means it was removed from the
|
||||||
// pending acks
|
// pending acks
|
||||||
if Arc::strong_count(&timed_out_ack) == 1 {
|
if Arc::strong_count(&timed_out_ack) == 1 {
|
||||||
@@ -101,7 +115,6 @@ where
|
|||||||
// we no longer need the reference - let's drop it so that if somehow `UpdateTimer` action
|
// we no longer need the reference - let's drop it so that if somehow `UpdateTimer` action
|
||||||
// reached the controller before this function terminated, the controller would not panic.
|
// reached the controller before this function terminated, the controller would not panic.
|
||||||
drop(timed_out_ack);
|
drop(timed_out_ack);
|
||||||
|
|
||||||
let new_delay = prepared_fragment.total_delay;
|
let new_delay = prepared_fragment.total_delay;
|
||||||
|
|
||||||
// We know this update will be reflected by the `StartTimer` Action performed when this
|
// We know this update will be reflected by the `StartTimer` Action performed when this
|
||||||
@@ -116,16 +129,14 @@ where
|
|||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
// send to `OutQueueControl` to eventually send to the mix network
|
// send to `OutQueueControl` to eventually send to the mix network
|
||||||
self.real_message_sender
|
self.message_handler
|
||||||
.send((
|
.forward_messages(
|
||||||
vec![RealMessage::new(prepared_fragment.mix_packet, frag_id)],
|
vec![RealMessage::new(prepared_fragment.mix_packet, frag_id)],
|
||||||
TransmissionLane::Retransmission,
|
TransmissionLane::Retransmission,
|
||||||
))
|
)
|
||||||
.await
|
.await
|
||||||
.expect("BatchRealMessageReceiver has stopped receiving!");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
pub(super) async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
pub(super) async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
||||||
debug!("Started RetransmissionRequestListener with graceful shutdown support");
|
debug!("Started RetransmissionRequestListener with graceful shutdown support");
|
||||||
|
|
||||||
@@ -143,16 +154,7 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
assert!(shutdown.is_shutdown_poll());
|
shutdown.recv_timeout().await;
|
||||||
log::debug!("RetransmissionRequestListener: Exiting");
|
log::debug!("RetransmissionRequestListener: Exiting");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
pub(super) async fn run(&mut self) {
|
|
||||||
debug!("Started RetransmissionRequestListener without graceful shutdown support");
|
|
||||||
|
|
||||||
while let Some(timed_out_ack) = self.request_receiver.next().await {
|
|
||||||
self.on_retransmission_request(timed_out_ack).await;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
+3
-18
@@ -1,7 +1,7 @@
|
|||||||
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use super::action_controller::{Action, ActionSender};
|
use super::action_controller::{AckActionSender, Action};
|
||||||
use super::SentPacketNotificationReceiver;
|
use super::SentPacketNotificationReceiver;
|
||||||
use futures::StreamExt;
|
use futures::StreamExt;
|
||||||
use log::*;
|
use log::*;
|
||||||
@@ -13,13 +13,13 @@ use nymsphinx::chunking::fragment::{FragmentIdentifier, COVER_FRAG_ID};
|
|||||||
/// accidentally fire retransmission way quicker than we should have.
|
/// accidentally fire retransmission way quicker than we should have.
|
||||||
pub(super) struct SentNotificationListener {
|
pub(super) struct SentNotificationListener {
|
||||||
sent_notifier: SentPacketNotificationReceiver,
|
sent_notifier: SentPacketNotificationReceiver,
|
||||||
action_sender: ActionSender,
|
action_sender: AckActionSender,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SentNotificationListener {
|
impl SentNotificationListener {
|
||||||
pub(super) fn new(
|
pub(super) fn new(
|
||||||
sent_notifier: SentPacketNotificationReceiver,
|
sent_notifier: SentPacketNotificationReceiver,
|
||||||
action_sender: ActionSender,
|
action_sender: AckActionSender,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
SentNotificationListener {
|
SentNotificationListener {
|
||||||
sent_notifier,
|
sent_notifier,
|
||||||
@@ -31,18 +31,12 @@ impl SentNotificationListener {
|
|||||||
if frag_id == COVER_FRAG_ID {
|
if frag_id == COVER_FRAG_ID {
|
||||||
trace!("sent off a cover message - no need to start retransmission timer!");
|
trace!("sent off a cover message - no need to start retransmission timer!");
|
||||||
return;
|
return;
|
||||||
} else if frag_id.is_reply() {
|
|
||||||
debug!("sent off a reply message - no need to start retransmission timer!");
|
|
||||||
// TODO: probably there will need to be some extra procedure here, like it would
|
|
||||||
// be nice to know that our reply actually reached the recipient (i.e. we got the ack)
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
self.action_sender
|
self.action_sender
|
||||||
.unbounded_send(Action::new_start_timer(frag_id))
|
.unbounded_send(Action::new_start_timer(frag_id))
|
||||||
.unwrap();
|
.unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
pub(super) async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
pub(super) async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
||||||
debug!("Started SentNotificationListener with graceful shutdown support");
|
debug!("Started SentNotificationListener with graceful shutdown support");
|
||||||
|
|
||||||
@@ -65,13 +59,4 @@ impl SentNotificationListener {
|
|||||||
assert!(shutdown.is_shutdown_poll());
|
assert!(shutdown.is_shutdown_poll());
|
||||||
log::debug!("SentNotificationListener: Exiting");
|
log::debug!("SentNotificationListener: Exiting");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
pub(super) async fn run(&mut self) {
|
|
||||||
debug!("Started SentNotificationListener without graceful shutdown support");
|
|
||||||
|
|
||||||
while let Some(frag_id) = self.sent_notifier.next().await {
|
|
||||||
self.on_sent_message(frag_id).await;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,548 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use crate::client::real_messages_control::acknowledgement_control::PendingAcknowledgement;
|
||||||
|
use crate::client::real_messages_control::real_traffic_stream::{
|
||||||
|
BatchRealMessageSender, RealMessage,
|
||||||
|
};
|
||||||
|
use crate::client::real_messages_control::{AckActionSender, Action};
|
||||||
|
use crate::client::replies::reply_storage::{ReceivedReplySurbsMap, SentReplyKeys, UsedSenderTags};
|
||||||
|
use crate::client::topology_control::{TopologyAccessor, TopologyReadPermit};
|
||||||
|
use client_connections::TransmissionLane;
|
||||||
|
use log::{debug, error, info, trace, warn};
|
||||||
|
use nymsphinx::acknowledgements::AckKey;
|
||||||
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
|
use nymsphinx::anonymous_replies::requests::{AnonymousSenderTag, RepliableMessage, ReplyMessage};
|
||||||
|
use nymsphinx::anonymous_replies::{ReplySurb, SurbEncryptionKey};
|
||||||
|
use nymsphinx::chunking::fragment::{Fragment, FragmentIdentifier};
|
||||||
|
use nymsphinx::message::NymMessage;
|
||||||
|
use nymsphinx::params::{PacketSize, DEFAULT_NUM_MIX_HOPS};
|
||||||
|
use nymsphinx::preparer::{MessagePreparer, PreparedFragment};
|
||||||
|
use nymsphinx::Delay;
|
||||||
|
use rand::{CryptoRng, Rng};
|
||||||
|
use std::sync::Arc;
|
||||||
|
use std::time::Duration;
|
||||||
|
use thiserror::Error;
|
||||||
|
use topology::{NymTopology, NymTopologyError};
|
||||||
|
|
||||||
|
// TODO: move that error elsewhere since it seems to be contaminating different files
|
||||||
|
#[derive(Debug, Clone, Error)]
|
||||||
|
pub enum PreparationError {
|
||||||
|
#[error(transparent)]
|
||||||
|
NymTopologyError(#[from] NymTopologyError),
|
||||||
|
|
||||||
|
#[error("The received message cannot be sent using a single reply surb. It ended up getting split into {fragments} fragments.")]
|
||||||
|
MessageTooLongForSingleSurb { fragments: usize },
|
||||||
|
|
||||||
|
#[error("Not enough reply SURBs to send the message. We have {available} available and require at least {required}.")]
|
||||||
|
NotEnoughSurbs { available: usize, required: usize },
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PreparationError {
|
||||||
|
fn return_surbs(self, returned_surbs: Vec<ReplySurb>) -> SurbWrappedPreparationError {
|
||||||
|
SurbWrappedPreparationError {
|
||||||
|
source: self,
|
||||||
|
returned_surbs: Some(returned_surbs),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Error)]
|
||||||
|
#[error("Failed to prepare packets - {source}. {} reply surbs will be returned", .returned_surbs.as_ref().map(|s| s.len()).unwrap_or_default())]
|
||||||
|
pub struct SurbWrappedPreparationError {
|
||||||
|
#[source]
|
||||||
|
source: PreparationError,
|
||||||
|
|
||||||
|
returned_surbs: Option<Vec<ReplySurb>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> From<T> for SurbWrappedPreparationError
|
||||||
|
where
|
||||||
|
T: Into<PreparationError>,
|
||||||
|
{
|
||||||
|
fn from(err: T) -> Self {
|
||||||
|
SurbWrappedPreparationError {
|
||||||
|
source: err.into(),
|
||||||
|
returned_surbs: None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SurbWrappedPreparationError {
|
||||||
|
pub(crate) fn return_unused_surbs(
|
||||||
|
self,
|
||||||
|
surb_storage: &ReceivedReplySurbsMap,
|
||||||
|
target: &AnonymousSenderTag,
|
||||||
|
) -> PreparationError {
|
||||||
|
if let Some(reply_surbs) = self.returned_surbs {
|
||||||
|
surb_storage.insert_surbs(target, reply_surbs)
|
||||||
|
}
|
||||||
|
self.source
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub(crate) struct Config {
|
||||||
|
/// Key used to decrypt contents of received SURBAcks
|
||||||
|
ack_key: Arc<AckKey>,
|
||||||
|
|
||||||
|
/// Address of this client which also represent an address to which all acknowledgements
|
||||||
|
/// and surb-based are going to be sent.
|
||||||
|
sender_address: Recipient,
|
||||||
|
|
||||||
|
/// Average delay a data packet is going to get delay at a single mixnode.
|
||||||
|
average_packet_delay: Duration,
|
||||||
|
|
||||||
|
/// Average delay an acknowledgement packet is going to get delay at a single mixnode.
|
||||||
|
average_ack_delay: Duration,
|
||||||
|
|
||||||
|
/// Number of mix hops each packet ('real' message, ack, reply) is expected to take.
|
||||||
|
/// Note that it does not include gateway hops.
|
||||||
|
num_mix_hops: u8,
|
||||||
|
|
||||||
|
/// Predefined packet size used for the encapsulated messages.
|
||||||
|
packet_size: PacketSize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Config {
|
||||||
|
pub fn new(
|
||||||
|
ack_key: Arc<AckKey>,
|
||||||
|
sender_address: Recipient,
|
||||||
|
average_packet_delay: Duration,
|
||||||
|
average_ack_delay: Duration,
|
||||||
|
) -> Self {
|
||||||
|
Config {
|
||||||
|
ack_key,
|
||||||
|
sender_address,
|
||||||
|
average_packet_delay,
|
||||||
|
average_ack_delay,
|
||||||
|
num_mix_hops: DEFAULT_NUM_MIX_HOPS,
|
||||||
|
packet_size: PacketSize::default(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Allows setting non-default number of expected mix hops in the network.
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub fn with_mix_hops(mut self, hops: u8) -> Self {
|
||||||
|
self.num_mix_hops = hops;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Allows setting non-default size of the sphinx packets sent out.
|
||||||
|
pub fn with_custom_packet_size(mut self, packet_size: PacketSize) -> Self {
|
||||||
|
self.packet_size = packet_size;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub(crate) struct MessageHandler<R> {
|
||||||
|
config: Config,
|
||||||
|
rng: R,
|
||||||
|
message_preparer: MessagePreparer<R>,
|
||||||
|
action_sender: AckActionSender,
|
||||||
|
real_message_sender: BatchRealMessageSender,
|
||||||
|
topology_access: TopologyAccessor,
|
||||||
|
reply_key_storage: SentReplyKeys,
|
||||||
|
tag_storage: UsedSenderTags,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<R> MessageHandler<R>
|
||||||
|
where
|
||||||
|
R: CryptoRng + Rng,
|
||||||
|
{
|
||||||
|
pub(crate) fn new(
|
||||||
|
config: Config,
|
||||||
|
rng: R,
|
||||||
|
action_sender: AckActionSender,
|
||||||
|
real_message_sender: BatchRealMessageSender,
|
||||||
|
topology_access: TopologyAccessor,
|
||||||
|
reply_key_storage: SentReplyKeys,
|
||||||
|
tag_storage: UsedSenderTags,
|
||||||
|
) -> Self
|
||||||
|
where
|
||||||
|
R: Copy,
|
||||||
|
{
|
||||||
|
let message_preparer = MessagePreparer::new(
|
||||||
|
rng,
|
||||||
|
config.sender_address,
|
||||||
|
config.average_packet_delay,
|
||||||
|
config.average_ack_delay,
|
||||||
|
)
|
||||||
|
.with_custom_real_message_packet_size(config.packet_size)
|
||||||
|
.with_mix_hops(config.num_mix_hops);
|
||||||
|
|
||||||
|
MessageHandler {
|
||||||
|
config,
|
||||||
|
rng,
|
||||||
|
message_preparer,
|
||||||
|
action_sender,
|
||||||
|
real_message_sender,
|
||||||
|
topology_access,
|
||||||
|
reply_key_storage,
|
||||||
|
tag_storage,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_or_create_sender_tag(&mut self, recipient: &Recipient) -> AnonymousSenderTag {
|
||||||
|
if let Some(existing) = self.tag_storage.try_get_existing(recipient) {
|
||||||
|
trace!("we already had sender tag for {recipient}");
|
||||||
|
existing
|
||||||
|
} else {
|
||||||
|
info!("creating new sender tag for {recipient}");
|
||||||
|
let new_tag = AnonymousSenderTag::new_random(&mut self.rng);
|
||||||
|
self.tag_storage.insert_new(recipient, new_tag);
|
||||||
|
info!("we'll be using {new_tag} for all anonymous messages sent to {recipient}");
|
||||||
|
new_tag
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_topology<'a>(
|
||||||
|
&self,
|
||||||
|
permit: &'a TopologyReadPermit<'a>,
|
||||||
|
) -> Result<&'a NymTopology, PreparationError> {
|
||||||
|
match permit.try_get_valid_topology_ref(&self.config.sender_address, None) {
|
||||||
|
Ok(topology_ref) => Ok(topology_ref),
|
||||||
|
Err(err) => {
|
||||||
|
warn!("Could not process the packet - the network topology is invalid - {err}");
|
||||||
|
Err(err.into())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn generate_reply_surbs_with_keys(
|
||||||
|
&mut self,
|
||||||
|
amount: usize,
|
||||||
|
) -> Result<(Vec<ReplySurb>, Vec<SurbEncryptionKey>), PreparationError> {
|
||||||
|
let topology_permit = self.topology_access.get_read_permit().await;
|
||||||
|
let topology = self.get_topology(&topology_permit)?;
|
||||||
|
|
||||||
|
let reply_surbs = self
|
||||||
|
.message_preparer
|
||||||
|
.generate_reply_surbs(amount, topology)?;
|
||||||
|
|
||||||
|
let reply_keys = reply_surbs
|
||||||
|
.iter()
|
||||||
|
.map(|s| *s.encryption_key())
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
Ok((reply_surbs, reply_keys))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn try_send_single_surb_message(
|
||||||
|
&mut self,
|
||||||
|
target: AnonymousSenderTag,
|
||||||
|
message: ReplyMessage,
|
||||||
|
reply_surb: ReplySurb,
|
||||||
|
is_extra_surb_request: bool,
|
||||||
|
) -> Result<(), SurbWrappedPreparationError> {
|
||||||
|
let mut fragment = self
|
||||||
|
.message_preparer
|
||||||
|
.pad_and_split_message(NymMessage::new_reply(message));
|
||||||
|
if fragment.len() > 1 {
|
||||||
|
// well, it's not a single surb message
|
||||||
|
return Err(SurbWrappedPreparationError {
|
||||||
|
source: PreparationError::MessageTooLongForSingleSurb {
|
||||||
|
fragments: fragment.len(),
|
||||||
|
},
|
||||||
|
returned_surbs: Some(vec![reply_surb]),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
let chunk = fragment.pop().unwrap();
|
||||||
|
let chunk_clone = chunk.clone();
|
||||||
|
let prepared_fragment = self
|
||||||
|
.try_prepare_single_reply_chunk_for_sending(reply_surb, chunk_clone)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
let real_messages =
|
||||||
|
RealMessage::new(prepared_fragment.mix_packet, chunk.fragment_identifier());
|
||||||
|
let delay = prepared_fragment.total_delay;
|
||||||
|
let pending_ack =
|
||||||
|
PendingAcknowledgement::new_anonymous(chunk, delay, target, is_extra_surb_request);
|
||||||
|
|
||||||
|
let lane = if is_extra_surb_request {
|
||||||
|
TransmissionLane::ReplySurbRequest
|
||||||
|
} else {
|
||||||
|
TransmissionLane::General
|
||||||
|
};
|
||||||
|
|
||||||
|
self.forward_messages(vec![real_messages], lane).await;
|
||||||
|
self.insert_pending_acks(vec![pending_ack]);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn try_request_additional_reply_surbs(
|
||||||
|
&mut self,
|
||||||
|
from: AnonymousSenderTag,
|
||||||
|
reply_surb: ReplySurb,
|
||||||
|
amount: u32,
|
||||||
|
) -> Result<(), SurbWrappedPreparationError> {
|
||||||
|
debug!("requesting {amount} reply SURBs from {from:?}");
|
||||||
|
|
||||||
|
let surbs_request =
|
||||||
|
ReplyMessage::new_surb_request_message(self.config.sender_address, amount);
|
||||||
|
self.try_send_single_surb_message(from, surbs_request, reply_surb, true)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
// // TODO: this will require additional argument to make it use different variant of `ReplyMessage`
|
||||||
|
pub(crate) fn split_reply_message(&mut self, message: Vec<u8>) -> Vec<Fragment> {
|
||||||
|
self.message_preparer
|
||||||
|
.pad_and_split_message(NymMessage::new_reply(ReplyMessage::new_data_message(
|
||||||
|
message,
|
||||||
|
)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// the only difference between this method and `try_send_reply_chunks` is that
|
||||||
|
// here we are not creating acks as acks are already in memory waiting to get cleared.
|
||||||
|
// we are only updating their existing delays
|
||||||
|
pub(crate) async fn try_send_retransmission_reply_chunks(
|
||||||
|
&mut self,
|
||||||
|
fragments: Vec<Fragment>,
|
||||||
|
reply_surbs: Vec<ReplySurb>,
|
||||||
|
lane: TransmissionLane,
|
||||||
|
) -> Result<(), SurbWrappedPreparationError> {
|
||||||
|
let prepared_fragments = self
|
||||||
|
.prepare_reply_chunks_for_sending(fragments.clone(), reply_surbs)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
let mut real_messages = Vec::with_capacity(prepared_fragments.len());
|
||||||
|
|
||||||
|
for prepared in prepared_fragments {
|
||||||
|
self.update_ack_delay(prepared.fragment_identifier, prepared.total_delay);
|
||||||
|
real_messages.push(prepared.into())
|
||||||
|
}
|
||||||
|
|
||||||
|
self.forward_messages(real_messages, lane).await;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn try_send_reply_chunks(
|
||||||
|
&mut self,
|
||||||
|
target: AnonymousSenderTag,
|
||||||
|
fragments: Vec<Fragment>,
|
||||||
|
reply_surbs: Vec<ReplySurb>,
|
||||||
|
lane: TransmissionLane,
|
||||||
|
) -> Result<(), SurbWrappedPreparationError> {
|
||||||
|
let prepared_fragments = self
|
||||||
|
.prepare_reply_chunks_for_sending(fragments.clone(), reply_surbs)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
let mut pending_acks = Vec::with_capacity(fragments.len());
|
||||||
|
let mut real_messages = Vec::with_capacity(fragments.len());
|
||||||
|
|
||||||
|
for (raw, prepared) in fragments.into_iter().zip(prepared_fragments.into_iter()) {
|
||||||
|
let real_message = RealMessage::new(prepared.mix_packet, prepared.fragment_identifier);
|
||||||
|
let delay = prepared.total_delay;
|
||||||
|
let pending_ack = PendingAcknowledgement::new_anonymous(raw, delay, target, false);
|
||||||
|
|
||||||
|
real_messages.push(real_message);
|
||||||
|
pending_acks.push(pending_ack);
|
||||||
|
}
|
||||||
|
|
||||||
|
self.forward_messages(real_messages, lane).await;
|
||||||
|
self.insert_pending_acks(pending_acks);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn try_send_plain_message(
|
||||||
|
&mut self,
|
||||||
|
recipient: Recipient,
|
||||||
|
message: Vec<u8>,
|
||||||
|
lane: TransmissionLane,
|
||||||
|
) -> Result<(), PreparationError> {
|
||||||
|
let message = NymMessage::new_plain(message);
|
||||||
|
self.try_split_and_send_non_reply_message(message, recipient, lane)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn try_split_and_send_non_reply_message(
|
||||||
|
&mut self,
|
||||||
|
message: NymMessage,
|
||||||
|
recipient: Recipient,
|
||||||
|
lane: TransmissionLane,
|
||||||
|
) -> Result<(), PreparationError> {
|
||||||
|
// TODO: I really dislike existence of this assertion, it implies code has to be re-organised
|
||||||
|
debug_assert!(!matches!(message, NymMessage::Reply(_)));
|
||||||
|
|
||||||
|
// TODO2: it's really annoying we have to get topology permit again here due to borrow-checker
|
||||||
|
let topology_permit = self.topology_access.get_read_permit().await;
|
||||||
|
let topology = self.get_topology(&topology_permit)?;
|
||||||
|
|
||||||
|
let fragments = self.message_preparer.pad_and_split_message(message);
|
||||||
|
|
||||||
|
let mut pending_acks = Vec::with_capacity(fragments.len());
|
||||||
|
let mut real_messages = Vec::with_capacity(fragments.len());
|
||||||
|
for fragment in fragments {
|
||||||
|
// we need to clone it because we need to keep it in memory in case we had to retransmit
|
||||||
|
// it. And then we'd need to recreate entire ACK again.
|
||||||
|
let chunk_clone = fragment.clone();
|
||||||
|
let prepared_fragment = self.message_preparer.prepare_chunk_for_sending(
|
||||||
|
chunk_clone,
|
||||||
|
topology,
|
||||||
|
&self.config.ack_key,
|
||||||
|
&recipient,
|
||||||
|
)?;
|
||||||
|
|
||||||
|
let real_message =
|
||||||
|
RealMessage::new(prepared_fragment.mix_packet, fragment.fragment_identifier());
|
||||||
|
let delay = prepared_fragment.total_delay;
|
||||||
|
let pending_ack = PendingAcknowledgement::new_known(fragment, delay, recipient);
|
||||||
|
|
||||||
|
real_messages.push(real_message);
|
||||||
|
pending_acks.push(pending_ack);
|
||||||
|
}
|
||||||
|
|
||||||
|
self.insert_pending_acks(pending_acks);
|
||||||
|
self.forward_messages(real_messages, lane).await;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn try_send_additional_reply_surbs(
|
||||||
|
&mut self,
|
||||||
|
recipient: Recipient,
|
||||||
|
amount: u32,
|
||||||
|
) -> Result<(), PreparationError> {
|
||||||
|
let sender_tag = self.get_or_create_sender_tag(&recipient);
|
||||||
|
let (reply_surbs, reply_keys) =
|
||||||
|
self.generate_reply_surbs_with_keys(amount as usize).await?;
|
||||||
|
|
||||||
|
let message = NymMessage::new_repliable(RepliableMessage::new_additional_surbs(
|
||||||
|
sender_tag,
|
||||||
|
reply_surbs,
|
||||||
|
));
|
||||||
|
|
||||||
|
self.try_split_and_send_non_reply_message(
|
||||||
|
message,
|
||||||
|
recipient,
|
||||||
|
TransmissionLane::AdditionalReplySurbs,
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
log::trace!("storing {} reply keys", reply_keys.len());
|
||||||
|
self.reply_key_storage.insert_multiple(reply_keys);
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn try_send_message_with_reply_surbs(
|
||||||
|
&mut self,
|
||||||
|
recipient: Recipient,
|
||||||
|
message: Vec<u8>,
|
||||||
|
num_reply_surbs: u32,
|
||||||
|
lane: TransmissionLane,
|
||||||
|
) -> Result<(), SurbWrappedPreparationError> {
|
||||||
|
let sender_tag = self.get_or_create_sender_tag(&recipient);
|
||||||
|
let (reply_surbs, reply_keys) = self
|
||||||
|
.generate_reply_surbs_with_keys(num_reply_surbs as usize)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
let message =
|
||||||
|
NymMessage::new_repliable(RepliableMessage::new_data(message, sender_tag, reply_surbs));
|
||||||
|
|
||||||
|
self.try_split_and_send_non_reply_message(message, recipient, lane)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
log::trace!("storing {} reply keys", reply_keys.len());
|
||||||
|
self.reply_key_storage.insert_multiple(reply_keys);
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn try_prepare_single_chunk_for_sending(
|
||||||
|
&mut self,
|
||||||
|
recipient: Recipient,
|
||||||
|
chunk: Fragment,
|
||||||
|
) -> Result<PreparedFragment, PreparationError> {
|
||||||
|
let topology_permit = self.topology_access.get_read_permit().await;
|
||||||
|
let topology = self.get_topology(&topology_permit)?;
|
||||||
|
|
||||||
|
let prepared_fragment = self
|
||||||
|
.message_preparer
|
||||||
|
.prepare_chunk_for_sending(chunk, topology, &self.config.ack_key, &recipient)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
Ok(prepared_fragment)
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn prepare_reply_chunks_for_sending(
|
||||||
|
&mut self,
|
||||||
|
fragments: Vec<Fragment>,
|
||||||
|
reply_surbs: Vec<ReplySurb>,
|
||||||
|
) -> Result<Vec<PreparedFragment>, SurbWrappedPreparationError> {
|
||||||
|
debug_assert_ne!(
|
||||||
|
fragments.len(),
|
||||||
|
reply_surbs.len(),
|
||||||
|
"attempted to send {} fragments with {} reply surbs",
|
||||||
|
fragments.len(),
|
||||||
|
reply_surbs.len()
|
||||||
|
);
|
||||||
|
|
||||||
|
let topology_permit = self.topology_access.get_read_permit().await;
|
||||||
|
let topology = match self.get_topology(&topology_permit) {
|
||||||
|
Ok(topology) => topology,
|
||||||
|
Err(err) => return Err(err.return_surbs(reply_surbs)),
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(fragments
|
||||||
|
.into_iter()
|
||||||
|
.zip(reply_surbs.into_iter())
|
||||||
|
.map(|(fragment, reply_surb)| {
|
||||||
|
// unwrap here is fine as we know we have a valid topology
|
||||||
|
self.message_preparer
|
||||||
|
.prepare_reply_chunk_for_sending(
|
||||||
|
fragment,
|
||||||
|
topology,
|
||||||
|
&self.config.ack_key,
|
||||||
|
reply_surb,
|
||||||
|
)
|
||||||
|
.unwrap()
|
||||||
|
})
|
||||||
|
.collect())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn try_prepare_single_reply_chunk_for_sending(
|
||||||
|
&mut self,
|
||||||
|
reply_surb: ReplySurb,
|
||||||
|
chunk: Fragment,
|
||||||
|
) -> Result<PreparedFragment, SurbWrappedPreparationError> {
|
||||||
|
let topology_permit = self.topology_access.get_read_permit().await;
|
||||||
|
let topology = match self.get_topology(&topology_permit) {
|
||||||
|
Ok(topology) => topology,
|
||||||
|
Err(err) => return Err(err.return_surbs(vec![reply_surb])),
|
||||||
|
};
|
||||||
|
|
||||||
|
let prepared_fragment = self
|
||||||
|
.message_preparer
|
||||||
|
.prepare_reply_chunk_for_sending(chunk, topology, &self.config.ack_key, reply_surb)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
Ok(prepared_fragment)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn update_ack_delay(&self, id: FragmentIdentifier, new_delay: Delay) {
|
||||||
|
self.action_sender
|
||||||
|
.unbounded_send(Action::UpdateDelay(id, new_delay))
|
||||||
|
.expect("action control task has died")
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn insert_pending_acks(&self, pending_acks: Vec<PendingAcknowledgement>) {
|
||||||
|
self.action_sender
|
||||||
|
.unbounded_send(Action::new_insert(pending_acks))
|
||||||
|
.expect("action control task has died")
|
||||||
|
}
|
||||||
|
|
||||||
|
// tells real message sender (with the poisson timer) to send this to the mix network
|
||||||
|
pub(crate) async fn forward_messages(
|
||||||
|
&self,
|
||||||
|
messages: Vec<RealMessage>,
|
||||||
|
transmission_lane: TransmissionLane,
|
||||||
|
) {
|
||||||
|
self.real_message_sender
|
||||||
|
.send((messages, transmission_lane))
|
||||||
|
.await
|
||||||
|
.expect("real message receiver task (OutQueueControl) has died");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -8,6 +8,11 @@
|
|||||||
use self::{
|
use self::{
|
||||||
acknowledgement_control::AcknowledgementController, real_traffic_stream::OutQueueControl,
|
acknowledgement_control::AcknowledgementController, real_traffic_stream::OutQueueControl,
|
||||||
};
|
};
|
||||||
|
use crate::client::real_messages_control::message_handler::MessageHandler;
|
||||||
|
use crate::client::replies::reply_controller::{
|
||||||
|
ReplyController, ReplyControllerReceiver, ReplyControllerSender,
|
||||||
|
};
|
||||||
|
use crate::client::replies::reply_storage::CombinedReplyStorage;
|
||||||
use crate::{
|
use crate::{
|
||||||
client::{
|
client::{
|
||||||
inbound_messages::InputMessageReceiver, mix_traffic::BatchMixMessageSender,
|
inbound_messages::InputMessageReceiver, mix_traffic::BatchMixMessageSender,
|
||||||
@@ -27,11 +32,13 @@ use rand::{rngs::OsRng, CryptoRng, Rng};
|
|||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
|
||||||
#[cfg(feature = "reply-surb")]
|
use crate::client::replies::reply_controller;
|
||||||
use crate::client::reply_key_storage::ReplyKeyStorage;
|
use crate::config;
|
||||||
|
pub(crate) use acknowledgement_control::{AckActionSender, Action};
|
||||||
|
|
||||||
mod acknowledgement_control;
|
pub(crate) mod acknowledgement_control;
|
||||||
mod real_traffic_stream;
|
pub(crate) mod message_handler;
|
||||||
|
pub(crate) mod real_traffic_stream;
|
||||||
|
|
||||||
// TODO: ack_key and self_recipient shouldn't really be part of this config
|
// TODO: ack_key and self_recipient shouldn't really be part of this config
|
||||||
pub struct Config {
|
pub struct Config {
|
||||||
@@ -62,31 +69,102 @@ pub struct Config {
|
|||||||
|
|
||||||
/// Predefined packet size used for the encapsulated messages.
|
/// Predefined packet size used for the encapsulated messages.
|
||||||
packet_size: PacketSize,
|
packet_size: PacketSize,
|
||||||
|
|
||||||
|
/// Defines the minimum number of reply surbs the client would request.
|
||||||
|
minimum_reply_surb_request_size: u32,
|
||||||
|
|
||||||
|
/// Defines the maximum number of reply surbs the client would request.
|
||||||
|
maximum_reply_surb_request_size: u32,
|
||||||
|
|
||||||
|
/// Defines the maximum number of reply surbs a remote party is allowed to request from this client at once.
|
||||||
|
maximum_allowed_reply_surb_request_size: u32,
|
||||||
|
|
||||||
|
/// Defines maximum amount of time the client is going to wait for reply surbs before explicitly asking
|
||||||
|
/// for more even though in theory they wouldn't need to.
|
||||||
|
maximum_reply_surb_waiting_period: Duration,
|
||||||
|
|
||||||
|
/// Defines maximum amount of time given reply surb is going to be valid for.
|
||||||
|
/// This is going to be superseded by key rotation once implemented.
|
||||||
|
maximum_reply_surb_age: Duration,
|
||||||
|
|
||||||
|
/// Defines maximum amount of time given reply key is going to be valid for.
|
||||||
|
/// This is going to be superseded by key rotation once implemented.
|
||||||
|
maximum_reply_key_age: Duration,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> From<&'a Config> for acknowledgement_control::Config {
|
||||||
|
fn from(cfg: &'a Config) -> Self {
|
||||||
|
acknowledgement_control::Config::new(cfg.ack_wait_addition, cfg.ack_wait_multiplier)
|
||||||
|
.with_custom_packet_size(cfg.packet_size)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> From<&'a Config> for real_traffic_stream::Config {
|
||||||
|
fn from(cfg: &'a Config) -> Self {
|
||||||
|
real_traffic_stream::Config::new(
|
||||||
|
Arc::clone(&cfg.ack_key),
|
||||||
|
cfg.self_recipient,
|
||||||
|
cfg.average_ack_delay_duration,
|
||||||
|
cfg.average_packet_delay_duration,
|
||||||
|
cfg.average_message_sending_delay,
|
||||||
|
cfg.disable_main_poisson_packet_distribution,
|
||||||
|
)
|
||||||
|
.with_custom_cover_packet_size(cfg.packet_size)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> From<&'a Config> for reply_controller::Config {
|
||||||
|
fn from(cfg: &'a Config) -> Self {
|
||||||
|
reply_controller::Config::new(
|
||||||
|
cfg.minimum_reply_surb_request_size,
|
||||||
|
cfg.maximum_reply_surb_request_size,
|
||||||
|
cfg.maximum_allowed_reply_surb_request_size,
|
||||||
|
cfg.maximum_reply_surb_waiting_period,
|
||||||
|
cfg.maximum_reply_surb_age,
|
||||||
|
cfg.maximum_reply_key_age,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> From<&'a Config> for message_handler::Config {
|
||||||
|
fn from(cfg: &'a Config) -> Self {
|
||||||
|
message_handler::Config::new(
|
||||||
|
Arc::clone(&cfg.ack_key),
|
||||||
|
cfg.self_recipient,
|
||||||
|
cfg.average_packet_delay_duration,
|
||||||
|
cfg.average_ack_delay_duration,
|
||||||
|
)
|
||||||
|
.with_custom_packet_size(cfg.packet_size)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Config {
|
impl Config {
|
||||||
// TODO: change the config into a builder
|
|
||||||
#[allow(clippy::too_many_arguments)]
|
|
||||||
pub fn new(
|
pub fn new(
|
||||||
|
base_client_debug_config: &config::DebugConfig,
|
||||||
ack_key: Arc<AckKey>,
|
ack_key: Arc<AckKey>,
|
||||||
ack_wait_multiplier: f64,
|
|
||||||
ack_wait_addition: Duration,
|
|
||||||
average_ack_delay_duration: Duration,
|
|
||||||
average_message_sending_delay: Duration,
|
|
||||||
average_packet_delay_duration: Duration,
|
|
||||||
disable_main_poisson_packet_distribution: bool,
|
|
||||||
self_recipient: Recipient,
|
self_recipient: Recipient,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
Config {
|
Config {
|
||||||
ack_key,
|
ack_key,
|
||||||
ack_wait_addition,
|
|
||||||
ack_wait_multiplier,
|
|
||||||
self_recipient,
|
self_recipient,
|
||||||
average_message_sending_delay,
|
|
||||||
average_packet_delay_duration,
|
|
||||||
average_ack_delay_duration,
|
|
||||||
disable_main_poisson_packet_distribution,
|
|
||||||
packet_size: Default::default(),
|
packet_size: Default::default(),
|
||||||
|
ack_wait_addition: base_client_debug_config.ack_wait_addition,
|
||||||
|
ack_wait_multiplier: base_client_debug_config.ack_wait_multiplier,
|
||||||
|
average_message_sending_delay: base_client_debug_config.message_sending_average_delay,
|
||||||
|
average_packet_delay_duration: base_client_debug_config.average_packet_delay,
|
||||||
|
average_ack_delay_duration: base_client_debug_config.average_ack_delay,
|
||||||
|
disable_main_poisson_packet_distribution: base_client_debug_config
|
||||||
|
.disable_main_poisson_packet_distribution,
|
||||||
|
minimum_reply_surb_request_size: base_client_debug_config
|
||||||
|
.minimum_reply_surb_request_size,
|
||||||
|
maximum_reply_surb_request_size: base_client_debug_config
|
||||||
|
.maximum_reply_surb_request_size,
|
||||||
|
maximum_allowed_reply_surb_request_size: base_client_debug_config
|
||||||
|
.maximum_allowed_reply_surb_request_size,
|
||||||
|
maximum_reply_surb_waiting_period: base_client_debug_config
|
||||||
|
.maximum_reply_surb_waiting_period,
|
||||||
|
maximum_reply_surb_age: base_client_debug_config.maximum_reply_surb_age,
|
||||||
|
maximum_reply_key_age: base_client_debug_config.maximum_reply_key_age,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -95,75 +173,84 @@ impl Config {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct RealMessagesController<R>
|
pub(crate) struct RealMessagesController<R>
|
||||||
where
|
where
|
||||||
R: CryptoRng + Rng,
|
R: CryptoRng + Rng,
|
||||||
{
|
{
|
||||||
out_queue_control: OutQueueControl<R>,
|
out_queue_control: OutQueueControl<R>,
|
||||||
ack_control: AcknowledgementController<R>,
|
ack_control: AcknowledgementController<R>,
|
||||||
|
reply_control: ReplyController<R>,
|
||||||
}
|
}
|
||||||
|
|
||||||
// obviously when we finally make shared rng that is on 'higher' level, this should become
|
// obviously when we finally make shared rng that is on 'higher' level, this should become
|
||||||
// generic `R`
|
// generic `R`
|
||||||
impl RealMessagesController<OsRng> {
|
impl RealMessagesController<OsRng> {
|
||||||
#[allow(clippy::too_many_arguments)]
|
#[allow(clippy::too_many_arguments)]
|
||||||
pub fn new(
|
pub(crate) fn new(
|
||||||
config: Config,
|
config: Config,
|
||||||
ack_receiver: AcknowledgementReceiver,
|
ack_receiver: AcknowledgementReceiver,
|
||||||
input_receiver: InputMessageReceiver,
|
input_receiver: InputMessageReceiver,
|
||||||
mix_sender: BatchMixMessageSender,
|
mix_sender: BatchMixMessageSender,
|
||||||
topology_access: TopologyAccessor,
|
topology_access: TopologyAccessor,
|
||||||
#[cfg(feature = "reply-surb")] reply_key_storage: ReplyKeyStorage,
|
reply_storage: CombinedReplyStorage,
|
||||||
|
// so much refactoring needed, but this is temporary just to test things out
|
||||||
|
reply_controller_sender: ReplyControllerSender,
|
||||||
|
reply_controller_receiver: ReplyControllerReceiver,
|
||||||
lane_queue_lengths: LaneQueueLengths,
|
lane_queue_lengths: LaneQueueLengths,
|
||||||
client_connection_rx: ConnectionCommandReceiver,
|
client_connection_rx: ConnectionCommandReceiver,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
let rng = OsRng;
|
let rng = OsRng;
|
||||||
|
|
||||||
|
// create channels for inter-task communication
|
||||||
let (real_message_sender, real_message_receiver) = tokio::sync::mpsc::channel(1);
|
let (real_message_sender, real_message_receiver) = tokio::sync::mpsc::channel(1);
|
||||||
let (sent_notifier_tx, sent_notifier_rx) = mpsc::unbounded();
|
let (sent_notifier_tx, sent_notifier_rx) = mpsc::unbounded();
|
||||||
|
let (ack_action_tx, ack_action_rx) = mpsc::unbounded();
|
||||||
let ack_controller_connectors = AcknowledgementControllerConnectors::new(
|
let ack_controller_connectors = AcknowledgementControllerConnectors::new(
|
||||||
real_message_sender,
|
|
||||||
input_receiver,
|
input_receiver,
|
||||||
sent_notifier_rx,
|
sent_notifier_rx,
|
||||||
ack_receiver,
|
ack_receiver,
|
||||||
|
ack_action_tx.clone(),
|
||||||
|
ack_action_rx,
|
||||||
);
|
);
|
||||||
|
|
||||||
let ack_control_config = acknowledgement_control::Config::new(
|
// create all configs for the components
|
||||||
config.ack_wait_addition,
|
let ack_control_config = (&config).into();
|
||||||
config.ack_wait_multiplier,
|
let out_queue_config = (&config).into();
|
||||||
config.average_ack_delay_duration,
|
let reply_controller_config = (&config).into();
|
||||||
config.average_packet_delay_duration,
|
let message_handler_config = (&config).into();
|
||||||
)
|
|
||||||
.with_custom_packet_size(config.packet_size);
|
// create the actual components
|
||||||
|
let message_handler = MessageHandler::new(
|
||||||
|
message_handler_config,
|
||||||
|
rng,
|
||||||
|
ack_action_tx,
|
||||||
|
real_message_sender,
|
||||||
|
topology_access.clone(),
|
||||||
|
reply_storage.key_storage(),
|
||||||
|
reply_storage.tags_storage(),
|
||||||
|
);
|
||||||
|
|
||||||
let ack_control = AcknowledgementController::new(
|
let ack_control = AcknowledgementController::new(
|
||||||
ack_control_config,
|
ack_control_config,
|
||||||
rng,
|
|
||||||
topology_access.clone(),
|
|
||||||
Arc::clone(&config.ack_key),
|
Arc::clone(&config.ack_key),
|
||||||
config.self_recipient,
|
|
||||||
ack_controller_connectors,
|
ack_controller_connectors,
|
||||||
#[cfg(feature = "reply-surb")]
|
message_handler.clone(),
|
||||||
reply_key_storage,
|
reply_controller_sender,
|
||||||
);
|
);
|
||||||
|
|
||||||
let out_queue_config = real_traffic_stream::Config::new(
|
let reply_control = ReplyController::new(
|
||||||
config.average_ack_delay_duration,
|
reply_controller_config,
|
||||||
config.average_packet_delay_duration,
|
message_handler,
|
||||||
config.average_message_sending_delay,
|
reply_storage,
|
||||||
config.disable_main_poisson_packet_distribution,
|
reply_controller_receiver,
|
||||||
)
|
);
|
||||||
.with_custom_cover_packet_size(config.packet_size);
|
|
||||||
|
|
||||||
let out_queue_control = OutQueueControl::new(
|
let out_queue_control = OutQueueControl::new(
|
||||||
out_queue_config,
|
out_queue_config,
|
||||||
Arc::clone(&config.ack_key),
|
rng,
|
||||||
sent_notifier_tx,
|
sent_notifier_tx,
|
||||||
mix_sender,
|
mix_sender,
|
||||||
real_message_receiver,
|
real_message_receiver,
|
||||||
rng,
|
|
||||||
config.self_recipient,
|
|
||||||
topology_access,
|
topology_access,
|
||||||
lane_queue_lengths,
|
lane_queue_lengths,
|
||||||
client_connection_rx,
|
client_connection_rx,
|
||||||
@@ -172,31 +259,26 @@ impl RealMessagesController<OsRng> {
|
|||||||
RealMessagesController {
|
RealMessagesController {
|
||||||
out_queue_control,
|
out_queue_control,
|
||||||
ack_control,
|
ack_control,
|
||||||
|
reply_control,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
pub fn start_with_shutdown(self, shutdown: task::ShutdownListener) {
|
pub fn start_with_shutdown(self, shutdown: task::ShutdownListener) {
|
||||||
let mut out_queue_control = self.out_queue_control;
|
let mut out_queue_control = self.out_queue_control;
|
||||||
let ack_control = self.ack_control;
|
let ack_control = self.ack_control;
|
||||||
|
let mut reply_control = self.reply_control;
|
||||||
|
|
||||||
let shutdown_handle = shutdown.clone();
|
let shutdown_handle = shutdown.clone();
|
||||||
spawn_future(async move {
|
spawn_future(async move {
|
||||||
out_queue_control.run_with_shutdown(shutdown_handle).await;
|
out_queue_control.run_with_shutdown(shutdown_handle).await;
|
||||||
debug!("The out queue controller has finished execution!");
|
debug!("The out queue controller has finished execution!");
|
||||||
});
|
});
|
||||||
|
let shutdown_handle = shutdown.clone();
|
||||||
|
spawn_future(async move {
|
||||||
|
reply_control.run_with_shutdown(shutdown_handle).await;
|
||||||
|
debug!("The reply controller has finished execution!");
|
||||||
|
});
|
||||||
|
|
||||||
ack_control.start_with_shutdown(shutdown);
|
ack_control.start_with_shutdown(shutdown);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
pub fn start(self) {
|
|
||||||
let mut out_queue_control = self.out_queue_control;
|
|
||||||
let ack_control = self.ack_control;
|
|
||||||
|
|
||||||
spawn_future(async move {
|
|
||||||
out_queue_control.run().await;
|
|
||||||
debug!("The out queue controller has finished execution!");
|
|
||||||
});
|
|
||||||
ack_control.start();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ use nymsphinx::chunking::fragment::FragmentIdentifier;
|
|||||||
use nymsphinx::cover::generate_loop_cover_packet;
|
use nymsphinx::cover::generate_loop_cover_packet;
|
||||||
use nymsphinx::forwarding::packet::MixPacket;
|
use nymsphinx::forwarding::packet::MixPacket;
|
||||||
use nymsphinx::params::PacketSize;
|
use nymsphinx::params::PacketSize;
|
||||||
|
use nymsphinx::preparer::PreparedFragment;
|
||||||
use nymsphinx::utils::sample_poisson_duration;
|
use nymsphinx::utils::sample_poisson_duration;
|
||||||
use rand::{CryptoRng, Rng};
|
use rand::{CryptoRng, Rng};
|
||||||
use std::pin::Pin;
|
use std::pin::Pin;
|
||||||
@@ -47,6 +48,12 @@ fn get_time_now() -> wasm_timer::Instant {
|
|||||||
|
|
||||||
/// Configurable parameters of the `OutQueueControl`
|
/// Configurable parameters of the `OutQueueControl`
|
||||||
pub(crate) struct Config {
|
pub(crate) struct Config {
|
||||||
|
/// Key used to encrypt and decrypt content of an ACK packet.
|
||||||
|
ack_key: Arc<AckKey>,
|
||||||
|
|
||||||
|
/// Represents full address of this client.
|
||||||
|
our_full_destination: Recipient,
|
||||||
|
|
||||||
/// Average delay an acknowledgement packet is going to get delay at a single mixnode.
|
/// Average delay an acknowledgement packet is going to get delay at a single mixnode.
|
||||||
average_ack_delay: Duration,
|
average_ack_delay: Duration,
|
||||||
|
|
||||||
@@ -66,12 +73,16 @@ pub(crate) struct Config {
|
|||||||
|
|
||||||
impl Config {
|
impl Config {
|
||||||
pub(crate) fn new(
|
pub(crate) fn new(
|
||||||
|
ack_key: Arc<AckKey>,
|
||||||
|
our_full_destination: Recipient,
|
||||||
average_ack_delay: Duration,
|
average_ack_delay: Duration,
|
||||||
average_packet_delay: Duration,
|
average_packet_delay: Duration,
|
||||||
average_message_sending_delay: Duration,
|
average_message_sending_delay: Duration,
|
||||||
disable_poisson_packet_distribution: bool,
|
disable_poisson_packet_distribution: bool,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
Config {
|
Config {
|
||||||
|
ack_key,
|
||||||
|
our_full_destination,
|
||||||
average_ack_delay,
|
average_ack_delay,
|
||||||
average_packet_delay,
|
average_packet_delay,
|
||||||
average_message_sending_delay,
|
average_message_sending_delay,
|
||||||
@@ -93,9 +104,6 @@ where
|
|||||||
/// Configurable parameters of the `ActionController`
|
/// Configurable parameters of the `ActionController`
|
||||||
config: Config,
|
config: Config,
|
||||||
|
|
||||||
/// Key used to encrypt and decrypt content of an ACK packet.
|
|
||||||
ack_key: Arc<AckKey>,
|
|
||||||
|
|
||||||
/// Channel used for notifying of a real packet being sent out. Used to start up retransmission timer.
|
/// Channel used for notifying of a real packet being sent out. Used to start up retransmission timer.
|
||||||
sent_notifier: SentPacketNotificationSender,
|
sent_notifier: SentPacketNotificationSender,
|
||||||
|
|
||||||
@@ -119,9 +127,6 @@ where
|
|||||||
/// before being sent out into the network.
|
/// before being sent out into the network.
|
||||||
real_receiver: BatchRealMessageReceiver,
|
real_receiver: BatchRealMessageReceiver,
|
||||||
|
|
||||||
/// Represents full address of this client.
|
|
||||||
our_full_destination: Recipient,
|
|
||||||
|
|
||||||
/// Instance of a cryptographically secure random number generator.
|
/// Instance of a cryptographically secure random number generator.
|
||||||
rng: R,
|
rng: R,
|
||||||
|
|
||||||
@@ -144,6 +149,16 @@ where
|
|||||||
pub(crate) struct RealMessage {
|
pub(crate) struct RealMessage {
|
||||||
mix_packet: MixPacket,
|
mix_packet: MixPacket,
|
||||||
fragment_id: FragmentIdentifier,
|
fragment_id: FragmentIdentifier,
|
||||||
|
// TODO: add info about it being constructed with reply-surb
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<PreparedFragment> for RealMessage {
|
||||||
|
fn from(fragment: PreparedFragment) -> Self {
|
||||||
|
RealMessage {
|
||||||
|
mix_packet: fragment.mix_packet,
|
||||||
|
fragment_id: fragment.fragment_identifier,
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl RealMessage {
|
impl RealMessage {
|
||||||
@@ -175,25 +190,21 @@ where
|
|||||||
#[allow(clippy::too_many_arguments)]
|
#[allow(clippy::too_many_arguments)]
|
||||||
pub(crate) fn new(
|
pub(crate) fn new(
|
||||||
config: Config,
|
config: Config,
|
||||||
ack_key: Arc<AckKey>,
|
rng: R,
|
||||||
sent_notifier: SentPacketNotificationSender,
|
sent_notifier: SentPacketNotificationSender,
|
||||||
mix_tx: BatchMixMessageSender,
|
mix_tx: BatchMixMessageSender,
|
||||||
real_receiver: BatchRealMessageReceiver,
|
real_receiver: BatchRealMessageReceiver,
|
||||||
rng: R,
|
|
||||||
our_full_destination: Recipient,
|
|
||||||
topology_access: TopologyAccessor,
|
topology_access: TopologyAccessor,
|
||||||
lane_queue_lengths: LaneQueueLengths,
|
lane_queue_lengths: LaneQueueLengths,
|
||||||
client_connection_rx: ConnectionCommandReceiver,
|
client_connection_rx: ConnectionCommandReceiver,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
OutQueueControl {
|
OutQueueControl {
|
||||||
config,
|
config,
|
||||||
ack_key,
|
|
||||||
sent_notifier,
|
sent_notifier,
|
||||||
next_delay: None,
|
next_delay: None,
|
||||||
sending_delay_controller: Default::default(),
|
sending_delay_controller: Default::default(),
|
||||||
mix_tx,
|
mix_tx,
|
||||||
real_receiver,
|
real_receiver,
|
||||||
our_full_destination,
|
|
||||||
rng,
|
rng,
|
||||||
topology_access,
|
topology_access,
|
||||||
transmission_buffer: Default::default(),
|
transmission_buffer: Default::default(),
|
||||||
@@ -220,24 +231,23 @@ where
|
|||||||
// poisson delay, but is it really a problem?
|
// poisson delay, but is it really a problem?
|
||||||
let topology_permit = self.topology_access.get_read_permit().await;
|
let topology_permit = self.topology_access.get_read_permit().await;
|
||||||
// the ack is sent back to ourselves (and then ignored)
|
// the ack is sent back to ourselves (and then ignored)
|
||||||
let topology_ref_option = topology_permit.try_get_valid_topology_ref(
|
let topology_ref = match topology_permit.try_get_valid_topology_ref(
|
||||||
&self.our_full_destination,
|
&self.config.our_full_destination,
|
||||||
Some(&self.our_full_destination),
|
Some(&self.config.our_full_destination),
|
||||||
);
|
) {
|
||||||
if topology_ref_option.is_none() {
|
Ok(topology) => topology,
|
||||||
warn!(
|
Err(err) => {
|
||||||
"No valid topology detected - won't send any loop cover message this time"
|
warn!("We're not going to send any loop cover message this time, as the current topology seem to be invalid - {err}");
|
||||||
);
|
return;
|
||||||
return;
|
}
|
||||||
}
|
};
|
||||||
let topology_ref = topology_ref_option.unwrap();
|
|
||||||
|
|
||||||
(
|
(
|
||||||
generate_loop_cover_packet(
|
generate_loop_cover_packet(
|
||||||
&mut self.rng,
|
&mut self.rng,
|
||||||
topology_ref,
|
topology_ref,
|
||||||
&self.ack_key,
|
&self.config.ack_key,
|
||||||
&self.our_full_destination,
|
&self.config.our_full_destination,
|
||||||
self.config.average_ack_delay,
|
self.config.average_ack_delay,
|
||||||
self.config.average_packet_delay,
|
self.config.average_packet_delay,
|
||||||
self.config.cover_packet_size,
|
self.config.cover_packet_size,
|
||||||
@@ -483,6 +493,10 @@ where
|
|||||||
} else {
|
} else {
|
||||||
log::debug!("{status_str}");
|
log::debug!("{status_str}");
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(not(target_arch = "wasm32"))]
|
||||||
|
fn log_status_infrequent(&self) {
|
||||||
if self.sending_delay_controller.current_multiplier() > 1 {
|
if self.sending_delay_controller.current_multiplier() > 1 {
|
||||||
log::warn!(
|
log::warn!(
|
||||||
"Unable to send packets fast enough - sending delay multiplier set to: {}",
|
"Unable to send packets fast enough - sending delay multiplier set to: {}",
|
||||||
@@ -491,41 +505,58 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
pub(super) async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
pub(super) async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
||||||
debug!("Started OutQueueControl with graceful shutdown support");
|
debug!("Started OutQueueControl with graceful shutdown support");
|
||||||
|
|
||||||
let mut status_timer = tokio::time::interval(Duration::from_secs(1));
|
#[cfg(not(target_arch = "wasm32"))]
|
||||||
|
{
|
||||||
|
let mut status_timer = tokio::time::interval(Duration::from_secs(5));
|
||||||
|
let mut infrequent_status_timer = tokio::time::interval(Duration::from_secs(60));
|
||||||
|
|
||||||
while !shutdown.is_shutdown() {
|
while !shutdown.is_shutdown() {
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
biased;
|
biased;
|
||||||
_ = shutdown.recv() => {
|
_ = shutdown.recv() => {
|
||||||
log::trace!("OutQueueControl: Received shutdown");
|
log::trace!("OutQueueControl: Received shutdown");
|
||||||
|
}
|
||||||
|
_ = status_timer.tick() => {
|
||||||
|
self.log_status();
|
||||||
|
}
|
||||||
|
_ = infrequent_status_timer.tick() => {
|
||||||
|
self.log_status_infrequent();
|
||||||
|
}
|
||||||
|
next_message = self.next() => if let Some(next_message) = next_message {
|
||||||
|
self.on_message(next_message).await;
|
||||||
|
} else {
|
||||||
|
log::trace!("OutQueueControl: Stopping since channel closed");
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
_ = status_timer.tick() => {
|
}
|
||||||
self.log_status();
|
tokio::time::timeout(Duration::from_secs(5), shutdown.recv())
|
||||||
}
|
.await
|
||||||
next_message = self.next() => if let Some(next_message) = next_message {
|
.expect("Task stopped without shutdown called");
|
||||||
self.on_message(next_message).await;
|
}
|
||||||
} else {
|
|
||||||
log::trace!("OutQueueControl: Stopping since channel closed");
|
#[cfg(target_arch = "wasm32")]
|
||||||
break;
|
{
|
||||||
|
while !shutdown.is_shutdown() {
|
||||||
|
tokio::select! {
|
||||||
|
biased;
|
||||||
|
_ = shutdown.recv() => {
|
||||||
|
log::trace!("OutQueueControl: Received shutdown");
|
||||||
|
}
|
||||||
|
next_message = self.next() => if let Some(next_message) = next_message {
|
||||||
|
self.on_message(next_message).await;
|
||||||
|
} else {
|
||||||
|
log::trace!("OutQueueControl: Stopping since channel closed");
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
assert!(shutdown.is_shutdown_poll());
|
|
||||||
log::debug!("OutQueueControl: Exiting");
|
log::debug!("OutQueueControl: Exiting");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
pub(super) async fn run(&mut self) {
|
|
||||||
debug!("Started OutQueueControl without graceful shutdown support");
|
|
||||||
|
|
||||||
while let Some(next_message) = self.next().await {
|
|
||||||
self.on_message(next_message).await;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<R> Stream for OutQueueControl<R>
|
impl<R> Stream for OutQueueControl<R>
|
||||||
|
|||||||
+1
-1
@@ -82,7 +82,7 @@ impl SendingDelayController {
|
|||||||
self.current_multiplier =
|
self.current_multiplier =
|
||||||
(self.current_multiplier + 1).clamp(self.lower_bound, self.upper_bound);
|
(self.current_multiplier + 1).clamp(self.lower_bound, self.upper_bound);
|
||||||
self.time_when_changed = get_time_now();
|
self.time_when_changed = get_time_now();
|
||||||
log::debug!(
|
log::warn!(
|
||||||
"Increasing sending delay multiplier to: {}",
|
"Increasing sending delay multiplier to: {}",
|
||||||
self.current_multiplier
|
self.current_multiplier
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,26 +1,26 @@
|
|||||||
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use crate::client::replies::reply_controller::ReplyControllerSender;
|
||||||
|
use crate::client::replies::reply_storage::SentReplyKeys;
|
||||||
use crate::spawn_future;
|
use crate::spawn_future;
|
||||||
use crypto::asymmetric::encryption;
|
use crypto::asymmetric::encryption;
|
||||||
|
use crypto::Digest;
|
||||||
use futures::channel::mpsc;
|
use futures::channel::mpsc;
|
||||||
use futures::lock::Mutex;
|
use futures::lock::Mutex;
|
||||||
use futures::StreamExt;
|
use futures::StreamExt;
|
||||||
use gateway_client::MixnetMessageReceiver;
|
use gateway_client::MixnetMessageReceiver;
|
||||||
use log::*;
|
use log::*;
|
||||||
|
use nymsphinx::anonymous_replies::requests::{
|
||||||
|
RepliableMessage, RepliableMessageContent, ReplyMessage, ReplyMessageContent,
|
||||||
|
};
|
||||||
|
use nymsphinx::anonymous_replies::{encryption_key::EncryptionKeyDigest, SurbEncryptionKey};
|
||||||
|
use nymsphinx::message::{NymMessage, PlainMessage};
|
||||||
|
use nymsphinx::params::ReplySurbKeyDigestAlgorithm;
|
||||||
use nymsphinx::receiver::{MessageReceiver, MessageRecoveryError, ReconstructedMessage};
|
use nymsphinx::receiver::{MessageReceiver, MessageRecoveryError, ReconstructedMessage};
|
||||||
use std::collections::HashSet;
|
use std::collections::HashSet;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
#[cfg(feature = "reply-surb")]
|
|
||||||
use crate::client::reply_key_storage::ReplyKeyStorage;
|
|
||||||
#[cfg(feature = "reply-surb")]
|
|
||||||
use crypto::{symmetric::stream_cipher, Digest};
|
|
||||||
#[cfg(feature = "reply-surb")]
|
|
||||||
use nymsphinx::anonymous_replies::{encryption_key::EncryptionKeyDigest, SurbEncryptionKey};
|
|
||||||
#[cfg(feature = "reply-surb")]
|
|
||||||
use nymsphinx::params::{ReplySurbEncryptionAlgorithm, ReplySurbKeyDigestAlgorithm};
|
|
||||||
|
|
||||||
// Buffer Requests to say "hey, send any reconstructed messages to this channel"
|
// Buffer Requests to say "hey, send any reconstructed messages to this channel"
|
||||||
// or to say "hey, I'm going offline, don't send anything more to me. Just buffer them instead"
|
// or to say "hey, I'm going offline, don't send anything more to me. Just buffer them instead"
|
||||||
pub type ReceivedBufferRequestSender = mpsc::UnboundedSender<ReceivedBufferMessage>;
|
pub type ReceivedBufferRequestSender = mpsc::UnboundedSender<ReceivedBufferMessage>;
|
||||||
@@ -46,26 +46,15 @@ struct ReceivedMessagesBufferInner {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl ReceivedMessagesBufferInner {
|
impl ReceivedMessagesBufferInner {
|
||||||
fn process_received_fragment(&mut self, raw_fragment: Vec<u8>) -> Option<ReconstructedMessage> {
|
fn recover_from_fragment(&mut self, fragment_data: &[u8]) -> Option<NymMessage> {
|
||||||
let fragment_data = match self
|
if nymsphinx::cover::is_cover(fragment_data) {
|
||||||
.message_receiver
|
|
||||||
.recover_plaintext(self.local_encryption_keypair.private_key(), raw_fragment)
|
|
||||||
{
|
|
||||||
Err(e) => {
|
|
||||||
warn!("failed to recover fragment data: {:?}. The whole underlying message might be corrupted and unrecoverable!", e);
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
Ok(frag_data) => frag_data,
|
|
||||||
};
|
|
||||||
|
|
||||||
if nymsphinx::cover::is_cover(&fragment_data) {
|
|
||||||
trace!("The message was a loop cover message! Skipping it");
|
trace!("The message was a loop cover message! Skipping it");
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
|
|
||||||
let fragment = match self.message_receiver.recover_fragment(&fragment_data) {
|
let fragment = match self.message_receiver.recover_fragment(fragment_data) {
|
||||||
Err(e) => {
|
Err(err) => {
|
||||||
warn!("failed to recover fragment from raw data: {:?}. The whole underlying message might be corrupted and unrecoverable!", e);
|
warn!("failed to recover fragment from raw data: {err}. The whole underlying message might be corrupted and unrecoverable!");
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
Ok(frag) => frag,
|
Ok(frag) => frag,
|
||||||
@@ -79,9 +68,10 @@ impl ReceivedMessagesBufferInner {
|
|||||||
// if we returned an error the underlying message is malformed in some way
|
// if we returned an error the underlying message is malformed in some way
|
||||||
match self.message_receiver.insert_new_fragment(fragment) {
|
match self.message_receiver.insert_new_fragment(fragment) {
|
||||||
Err(err) => match err {
|
Err(err) => match err {
|
||||||
MessageRecoveryError::MalformedReconstructedMessage(message_sets) => {
|
MessageRecoveryError::MalformedReconstructedMessage { source, used_sets } => {
|
||||||
|
error!("message reconstruction failed - {source}. Attempting to re-use the message sets...");
|
||||||
// TODO: should we really insert reconstructed sets? could this be abused for some attack?
|
// TODO: should we really insert reconstructed sets? could this be abused for some attack?
|
||||||
for set_id in message_sets {
|
for set_id in used_sets {
|
||||||
if !self.recently_reconstructed.insert(set_id) {
|
if !self.recently_reconstructed.insert(set_id) {
|
||||||
// or perhaps we should even panic at this point?
|
// or perhaps we should even panic at this point?
|
||||||
error!("Reconstructed another message containing already used set id!")
|
error!("Reconstructed another message containing already used set id!")
|
||||||
@@ -107,6 +97,34 @@ impl ReceivedMessagesBufferInner {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn process_received_reply(
|
||||||
|
&mut self,
|
||||||
|
reply_ciphertext: &mut [u8],
|
||||||
|
reply_key: SurbEncryptionKey,
|
||||||
|
) -> Option<NymMessage> {
|
||||||
|
// note: this performs decryption IN PLACE without extra allocation
|
||||||
|
self.message_receiver
|
||||||
|
.recover_plaintext_from_reply(reply_ciphertext, reply_key);
|
||||||
|
let fragment_data = reply_ciphertext;
|
||||||
|
|
||||||
|
self.recover_from_fragment(fragment_data)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn process_received_regular_packet(&mut self, mut raw_fragment: Vec<u8>) -> Option<NymMessage> {
|
||||||
|
let fragment_data = match self.message_receiver.recover_plaintext_from_regular_packet(
|
||||||
|
self.local_encryption_keypair.private_key(),
|
||||||
|
&mut raw_fragment,
|
||||||
|
) {
|
||||||
|
Err(err) => {
|
||||||
|
warn!("failed to recover fragment data: {err}. The whole underlying message might be corrupted and unrecoverable!");
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
Ok(frag_data) => frag_data,
|
||||||
|
};
|
||||||
|
|
||||||
|
self.recover_from_fragment(fragment_data)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
@@ -114,17 +132,15 @@ impl ReceivedMessagesBufferInner {
|
|||||||
// You should always use .clone() to create additional instances
|
// You should always use .clone() to create additional instances
|
||||||
struct ReceivedMessagesBuffer {
|
struct ReceivedMessagesBuffer {
|
||||||
inner: Arc<Mutex<ReceivedMessagesBufferInner>>,
|
inner: Arc<Mutex<ReceivedMessagesBufferInner>>,
|
||||||
|
reply_key_storage: SentReplyKeys,
|
||||||
/// Storage containing keys to all [`ReplySURB`]s ever sent out that we did not receive back.
|
reply_controller_sender: ReplyControllerSender,
|
||||||
// There's no need to put it behind a Mutex since it's already properly concurrent
|
|
||||||
#[cfg(feature = "reply-surb")]
|
|
||||||
reply_key_storage: ReplyKeyStorage,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ReceivedMessagesBuffer {
|
impl ReceivedMessagesBuffer {
|
||||||
fn new(
|
fn new(
|
||||||
local_encryption_keypair: Arc<encryption::KeyPair>,
|
local_encryption_keypair: Arc<encryption::KeyPair>,
|
||||||
#[cfg(feature = "reply-surb")] reply_key_storage: ReplyKeyStorage,
|
reply_key_storage: SentReplyKeys,
|
||||||
|
reply_controller_sender: ReplyControllerSender,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
ReceivedMessagesBuffer {
|
ReceivedMessagesBuffer {
|
||||||
inner: Arc::new(Mutex::new(ReceivedMessagesBufferInner {
|
inner: Arc::new(Mutex::new(ReceivedMessagesBufferInner {
|
||||||
@@ -134,8 +150,8 @@ impl ReceivedMessagesBuffer {
|
|||||||
message_sender: None,
|
message_sender: None,
|
||||||
recently_reconstructed: HashSet::new(),
|
recently_reconstructed: HashSet::new(),
|
||||||
})),
|
})),
|
||||||
#[cfg(feature = "reply-surb")]
|
|
||||||
reply_key_storage,
|
reply_key_storage,
|
||||||
|
reply_controller_sender,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -177,34 +193,139 @@ impl ReceivedMessagesBuffer {
|
|||||||
guard.message_sender = Some(sender);
|
guard.message_sender = Some(sender);
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn add_reconstructed_messages(&mut self, msgs: Vec<ReconstructedMessage>) {
|
fn handle_reconstructed_plain_messages(
|
||||||
debug!("Adding {:?} new messages to the buffer!", msgs.len());
|
&mut self,
|
||||||
trace!("Adding new messages to the buffer! {:?}", msgs);
|
msgs: Vec<PlainMessage>,
|
||||||
self.inner.lock().await.messages.extend(msgs)
|
) -> Vec<ReconstructedMessage> {
|
||||||
|
msgs.into_iter().map(Into::into).collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "reply-surb")]
|
fn handle_reconstructed_repliable_messages(
|
||||||
fn process_received_reply(
|
&mut self,
|
||||||
reply_ciphertext: &[u8],
|
msgs: Vec<RepliableMessage>,
|
||||||
reply_key: SurbEncryptionKey,
|
) -> Vec<ReconstructedMessage> {
|
||||||
) -> Option<ReconstructedMessage> {
|
let mut reconstructed = Vec::new();
|
||||||
let zero_iv = stream_cipher::zero_iv::<ReplySurbEncryptionAlgorithm>();
|
for msg in msgs {
|
||||||
|
let (reply_surbs, from_surb_request) = match msg.content {
|
||||||
|
RepliableMessageContent::Data {
|
||||||
|
message,
|
||||||
|
reply_surbs,
|
||||||
|
} => {
|
||||||
|
trace!(
|
||||||
|
"received message that also contained additional {} reply surbs from {:?}!",
|
||||||
|
reply_surbs.len(),
|
||||||
|
msg.sender_tag
|
||||||
|
);
|
||||||
|
|
||||||
let mut reply_msg = stream_cipher::decrypt::<ReplySurbEncryptionAlgorithm>(
|
reconstructed.push(ReconstructedMessage::new(message, msg.sender_tag));
|
||||||
reply_key.inner(),
|
|
||||||
&zero_iv,
|
(reply_surbs, false)
|
||||||
reply_ciphertext,
|
}
|
||||||
);
|
RepliableMessageContent::AdditionalSurbs { reply_surbs } => {
|
||||||
if let Err(err) = MessageReceiver::remove_padding(&mut reply_msg) {
|
trace!(
|
||||||
warn!("Received reply had malformed padding! - {:?}", err);
|
"received additional {} reply surbs from {:?}!",
|
||||||
None
|
reply_surbs.len(),
|
||||||
} else {
|
msg.sender_tag
|
||||||
// TODO: perhaps having to say it doesn't have a surb an indication the type should be changed?
|
);
|
||||||
Some(ReconstructedMessage {
|
(reply_surbs, true)
|
||||||
message: reply_msg,
|
}
|
||||||
reply_surb: None,
|
RepliableMessageContent::Heartbeat {
|
||||||
})
|
additional_reply_surbs,
|
||||||
|
} => {
|
||||||
|
error!("received a repliable heartbeat message - we don't know how to handle it yet (and we won't know until future PRs)");
|
||||||
|
(additional_reply_surbs, false)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
self.reply_controller_sender.send_additional_surbs(
|
||||||
|
msg.sender_tag,
|
||||||
|
reply_surbs,
|
||||||
|
from_surb_request,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
reconstructed
|
||||||
|
}
|
||||||
|
|
||||||
|
fn handle_reconstructed_reply_messages(
|
||||||
|
&mut self,
|
||||||
|
msgs: Vec<ReplyMessage>,
|
||||||
|
) -> Vec<ReconstructedMessage> {
|
||||||
|
let mut reconstructed = Vec::new();
|
||||||
|
for msg in msgs {
|
||||||
|
match msg.content {
|
||||||
|
ReplyMessageContent::Data { message } => reconstructed.push(message.into()),
|
||||||
|
ReplyMessageContent::SurbRequest { recipient, amount } => {
|
||||||
|
debug!("received request for {amount} additional reply SURBs from {recipient}");
|
||||||
|
self.reply_controller_sender
|
||||||
|
.send_additional_surbs_request(*recipient, amount);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
reconstructed
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn handle_reconstructed_messages(&mut self, msgs: Vec<NymMessage>) {
|
||||||
|
if msgs.is_empty() {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut plain_messages = Vec::new();
|
||||||
|
let mut repliable_messages = Vec::new();
|
||||||
|
let mut reply_messages = Vec::new();
|
||||||
|
|
||||||
|
for msg in msgs {
|
||||||
|
match msg {
|
||||||
|
NymMessage::Plain(plain) => plain_messages.push(plain),
|
||||||
|
NymMessage::Repliable(repliable) => repliable_messages.push(repliable),
|
||||||
|
NymMessage::Reply(reply) => reply_messages.push(reply),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut reconstructed_messages = self.handle_reconstructed_plain_messages(plain_messages);
|
||||||
|
reconstructed_messages
|
||||||
|
.append(&mut self.handle_reconstructed_repliable_messages(repliable_messages));
|
||||||
|
reconstructed_messages
|
||||||
|
.append(&mut self.handle_reconstructed_reply_messages(reply_messages));
|
||||||
|
|
||||||
|
let mut inner_guard = self.inner.lock().await;
|
||||||
|
debug!(
|
||||||
|
"Adding {:?} new messages to the buffer!",
|
||||||
|
reconstructed_messages.len()
|
||||||
|
);
|
||||||
|
|
||||||
|
if let Some(sender) = &inner_guard.message_sender {
|
||||||
|
trace!("Sending reconstructed messages to announced sender");
|
||||||
|
if let Err(err) = sender.unbounded_send(reconstructed_messages) {
|
||||||
|
warn!("The reconstructed message receiver went offline without explicit notification (relevant error: - {err})");
|
||||||
|
inner_guard.message_sender = None;
|
||||||
|
inner_guard.messages.extend(err.into_inner());
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
trace!("No sender available - buffering reconstructed messages");
|
||||||
|
inner_guard.messages.extend(reconstructed_messages)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// this function doesn't really belong here...
|
||||||
|
fn get_reply_key<'a>(
|
||||||
|
&self,
|
||||||
|
raw_message: &'a mut [u8],
|
||||||
|
) -> Option<(SurbEncryptionKey, &'a mut [u8])> {
|
||||||
|
let reply_surb_digest_size = ReplySurbKeyDigestAlgorithm::output_size();
|
||||||
|
if raw_message.len() < reply_surb_digest_size {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
let possible_key_digest =
|
||||||
|
EncryptionKeyDigest::clone_from_slice(&raw_message[..reply_surb_digest_size]);
|
||||||
|
self.reply_key_storage
|
||||||
|
.try_pop(possible_key_digest)
|
||||||
|
.map(|reply_encryption_key| {
|
||||||
|
(
|
||||||
|
*reply_encryption_key,
|
||||||
|
&mut raw_message[reply_surb_digest_size..],
|
||||||
|
)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn handle_new_received(&mut self, msgs: Vec<Vec<u8>>) {
|
async fn handle_new_received(&mut self, msgs: Vec<Vec<u8>>) {
|
||||||
@@ -217,69 +338,27 @@ impl ReceivedMessagesBuffer {
|
|||||||
let mut inner_guard = self.inner.lock().await;
|
let mut inner_guard = self.inner.lock().await;
|
||||||
|
|
||||||
// first check if this is a reply or a chunked message
|
// first check if this is a reply or a chunked message
|
||||||
// TODO: verify with @AP if this way of doing it is safe or whether it could
|
// note: there's a possible information leakage associated with this check https://github.com/nymtech/nym/issues/296
|
||||||
// cause some attacks due to, I don't know, stupid edge case collisions?
|
for mut msg in msgs {
|
||||||
// Update: this DOES introduce a possible leakage: https://github.com/nymtech/nym/issues/296
|
// check first `HasherOutputSize` bytes if they correspond to known encryption key
|
||||||
for msg in msgs {
|
// if yes - this is a reply message
|
||||||
// TODO:
|
let completed_message =
|
||||||
// 1. make it nicer
|
if let Some((reply_key, reply_message)) = self.get_reply_key(&mut msg) {
|
||||||
// 2. make it not feature-locked
|
inner_guard.process_received_reply(reply_message, reply_key)
|
||||||
|
|
||||||
#[cfg(feature = "reply-surb")]
|
|
||||||
{
|
|
||||||
let reply_surb_digest_size = ReplySurbKeyDigestAlgorithm::output_size();
|
|
||||||
|
|
||||||
let possible_key_digest =
|
|
||||||
EncryptionKeyDigest::clone_from_slice(&msg[..reply_surb_digest_size]);
|
|
||||||
|
|
||||||
// check first `HasherOutputSize` bytes if they correspond to known encryption key
|
|
||||||
// if yes - this is a reply message
|
|
||||||
|
|
||||||
// TODO: this might be a bottleneck - since the keys are stored on disk we, presumably,
|
|
||||||
// are doing a disk operation every single received fragment
|
|
||||||
if let Some(reply_encryption_key) = self
|
|
||||||
.reply_key_storage
|
|
||||||
.get_and_remove_encryption_key(possible_key_digest)
|
|
||||||
.expect("storage operation failed!")
|
|
||||||
{
|
|
||||||
if let Some(completed_message) = Self::process_received_reply(
|
|
||||||
&msg[reply_surb_digest_size..],
|
|
||||||
reply_encryption_key,
|
|
||||||
) {
|
|
||||||
completed_messages.push(completed_message)
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
// otherwise - it's a 'normal' message
|
inner_guard.process_received_regular_packet(msg)
|
||||||
if let Some(completed_message) = inner_guard.process_received_fragment(msg) {
|
};
|
||||||
completed_messages.push(completed_message)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(not(feature = "reply-surb"))]
|
if let Some(completed) = completed_message {
|
||||||
if let Some(completed_message) = inner_guard.process_received_fragment(msg) {
|
info!("received {completed}");
|
||||||
completed_messages.push(completed_message)
|
completed_messages.push(completed)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
drop(inner_guard);
|
||||||
|
|
||||||
if !completed_messages.is_empty() {
|
if !completed_messages.is_empty() {
|
||||||
if let Some(sender) = &inner_guard.message_sender {
|
self.handle_reconstructed_messages(completed_messages).await
|
||||||
trace!("Sending reconstructed messages to announced sender");
|
|
||||||
if let Err(err) = sender.unbounded_send(completed_messages) {
|
|
||||||
warn!("The reconstructed message receiver went offline without explicit notification (relevant error: - {:?})", err);
|
|
||||||
// make sure to drop the lock to not deadlock
|
|
||||||
// (it is required by `add_reconstructed_messages`)
|
|
||||||
inner_guard.message_sender = None;
|
|
||||||
drop(inner_guard);
|
|
||||||
self.add_reconstructed_messages(err.into_inner()).await;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// make sure to drop the lock to not deadlock
|
|
||||||
// (it is required by `add_reconstructed_messages`)
|
|
||||||
drop(inner_guard);
|
|
||||||
trace!("No sender available - buffering reconstructed messages");
|
|
||||||
self.add_reconstructed_messages(completed_messages).await;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -320,7 +399,6 @@ impl RequestReceiver {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
||||||
debug!("Started RequestReceiver with graceful shutdown support");
|
debug!("Started RequestReceiver with graceful shutdown support");
|
||||||
while !shutdown.is_shutdown() {
|
while !shutdown.is_shutdown() {
|
||||||
@@ -340,18 +418,9 @@ impl RequestReceiver {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
assert!(shutdown.is_shutdown_poll());
|
shutdown.recv_timeout().await;
|
||||||
log::debug!("RequestReceiver: Exiting");
|
log::debug!("RequestReceiver: Exiting");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
async fn run(&mut self) {
|
|
||||||
debug!("Started RequestReceiver without graceful shutdown support");
|
|
||||||
|
|
||||||
while let Some(message) = self.query_receiver.next().await {
|
|
||||||
self.handle_message(message).await
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
struct FragmentedMessageReceiver {
|
struct FragmentedMessageReceiver {
|
||||||
@@ -370,7 +439,6 @@ impl FragmentedMessageReceiver {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
||||||
debug!("Started FragmentedMessageReceiver with graceful shutdown support");
|
debug!("Started FragmentedMessageReceiver with graceful shutdown support");
|
||||||
while !shutdown.is_shutdown() {
|
while !shutdown.is_shutdown() {
|
||||||
@@ -389,36 +457,28 @@ impl FragmentedMessageReceiver {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
assert!(shutdown.is_shutdown_poll());
|
shutdown.recv_timeout().await;
|
||||||
log::debug!("FragmentedMessageReceiver: Exiting");
|
log::debug!("FragmentedMessageReceiver: Exiting");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
async fn run(&mut self) {
|
|
||||||
debug!("Started FragmentedMessageReceiver without graceful shutdown support");
|
|
||||||
|
|
||||||
while let Some(new_messages) = self.mixnet_packet_receiver.next().await {
|
|
||||||
self.received_buffer.handle_new_received(new_messages).await;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct ReceivedMessagesBufferController {
|
pub(crate) struct ReceivedMessagesBufferController {
|
||||||
fragmented_message_receiver: FragmentedMessageReceiver,
|
fragmented_message_receiver: FragmentedMessageReceiver,
|
||||||
request_receiver: RequestReceiver,
|
request_receiver: RequestReceiver,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ReceivedMessagesBufferController {
|
impl ReceivedMessagesBufferController {
|
||||||
pub fn new(
|
pub(crate) fn new(
|
||||||
local_encryption_keypair: Arc<encryption::KeyPair>,
|
local_encryption_keypair: Arc<encryption::KeyPair>,
|
||||||
query_receiver: ReceivedBufferRequestReceiver,
|
query_receiver: ReceivedBufferRequestReceiver,
|
||||||
mixnet_packet_receiver: MixnetMessageReceiver,
|
mixnet_packet_receiver: MixnetMessageReceiver,
|
||||||
#[cfg(feature = "reply-surb")] reply_key_storage: ReplyKeyStorage,
|
reply_key_storage: SentReplyKeys,
|
||||||
|
reply_controller_sender: ReplyControllerSender,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
let received_buffer = ReceivedMessagesBuffer::new(
|
let received_buffer = ReceivedMessagesBuffer::new(
|
||||||
local_encryption_keypair,
|
local_encryption_keypair,
|
||||||
#[cfg(feature = "reply-surb")]
|
|
||||||
reply_key_storage,
|
reply_key_storage,
|
||||||
|
reply_controller_sender,
|
||||||
);
|
);
|
||||||
|
|
||||||
ReceivedMessagesBufferController {
|
ReceivedMessagesBufferController {
|
||||||
@@ -430,7 +490,6 @@ impl ReceivedMessagesBufferController {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
pub fn start_with_shutdown(self, shutdown: task::ShutdownListener) {
|
pub fn start_with_shutdown(self, shutdown: task::ShutdownListener) {
|
||||||
let mut fragmented_message_receiver = self.fragmented_message_receiver;
|
let mut fragmented_message_receiver = self.fragmented_message_receiver;
|
||||||
let mut request_receiver = self.request_receiver;
|
let mut request_receiver = self.request_receiver;
|
||||||
@@ -445,16 +504,4 @@ impl ReceivedMessagesBufferController {
|
|||||||
request_receiver.run_with_shutdown(shutdown).await;
|
request_receiver.run_with_shutdown(shutdown).await;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
pub fn start(self) {
|
|
||||||
let mut fragmented_message_receiver = self.fragmented_message_receiver;
|
|
||||||
let mut request_receiver = self.request_receiver;
|
|
||||||
spawn_future(async move {
|
|
||||||
fragmented_message_receiver.run().await;
|
|
||||||
});
|
|
||||||
spawn_future(async move {
|
|
||||||
request_receiver.run().await;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,5 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
pub mod reply_controller;
|
||||||
|
pub mod reply_storage;
|
||||||
@@ -0,0 +1,928 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use crate::client::real_messages_control::acknowledgement_control::PendingAcknowledgement;
|
||||||
|
use crate::client::real_messages_control::message_handler::{MessageHandler, PreparationError};
|
||||||
|
use crate::client::replies::reply_storage::CombinedReplyStorage;
|
||||||
|
use client_connections::TransmissionLane;
|
||||||
|
use futures::channel::mpsc;
|
||||||
|
use futures::StreamExt;
|
||||||
|
use log::{debug, error, info, trace, warn};
|
||||||
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
|
use nymsphinx::anonymous_replies::requests::AnonymousSenderTag;
|
||||||
|
use nymsphinx::anonymous_replies::ReplySurb;
|
||||||
|
use nymsphinx::chunking::fragment::{Fragment, FragmentIdentifier};
|
||||||
|
use rand::{CryptoRng, Rng};
|
||||||
|
use std::cmp::{max, min};
|
||||||
|
use std::collections::btree_map::Entry;
|
||||||
|
use std::collections::{BTreeMap, HashMap, VecDeque};
|
||||||
|
use std::sync::{Arc, Weak};
|
||||||
|
use std::time::Duration;
|
||||||
|
use time::OffsetDateTime;
|
||||||
|
|
||||||
|
#[cfg(not(target_arch = "wasm32"))]
|
||||||
|
type IntervalStream = tokio_stream::wrappers::IntervalStream;
|
||||||
|
|
||||||
|
#[cfg(target_arch = "wasm32")]
|
||||||
|
type IntervalStream = gloo_timers::future::IntervalStream;
|
||||||
|
|
||||||
|
pub(crate) fn new_control_channels() -> (ReplyControllerSender, ReplyControllerReceiver) {
|
||||||
|
let (tx, rx) = mpsc::unbounded();
|
||||||
|
(tx.into(), rx)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub(crate) struct ReplyControllerSender(mpsc::UnboundedSender<ReplyControllerMessage>);
|
||||||
|
|
||||||
|
impl From<mpsc::UnboundedSender<ReplyControllerMessage>> for ReplyControllerSender {
|
||||||
|
fn from(inner: mpsc::UnboundedSender<ReplyControllerMessage>) -> Self {
|
||||||
|
ReplyControllerSender(inner)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ReplyControllerSender {
|
||||||
|
pub(crate) fn send_retransmission_data(
|
||||||
|
&self,
|
||||||
|
recipient: AnonymousSenderTag,
|
||||||
|
timed_out_ack: Weak<PendingAcknowledgement>,
|
||||||
|
extra_surb_request: bool,
|
||||||
|
) {
|
||||||
|
self.0
|
||||||
|
.unbounded_send(ReplyControllerMessage::RetransmitReply {
|
||||||
|
recipient,
|
||||||
|
timed_out_ack,
|
||||||
|
extra_surb_request,
|
||||||
|
})
|
||||||
|
.expect("ReplyControllerReceiver has died!")
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn send_reply(
|
||||||
|
&self,
|
||||||
|
recipient: AnonymousSenderTag,
|
||||||
|
message: Vec<u8>,
|
||||||
|
lane: TransmissionLane,
|
||||||
|
) {
|
||||||
|
self.0
|
||||||
|
.unbounded_send(ReplyControllerMessage::SendReply {
|
||||||
|
recipient,
|
||||||
|
message,
|
||||||
|
lane,
|
||||||
|
})
|
||||||
|
.expect("ReplyControllerReceiver has died!")
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn send_additional_surbs(
|
||||||
|
&self,
|
||||||
|
sender_tag: AnonymousSenderTag,
|
||||||
|
reply_surbs: Vec<ReplySurb>,
|
||||||
|
from_surb_request: bool,
|
||||||
|
) {
|
||||||
|
self.0
|
||||||
|
.unbounded_send(ReplyControllerMessage::AdditionalSurbs {
|
||||||
|
sender_tag,
|
||||||
|
reply_surbs,
|
||||||
|
from_surb_request,
|
||||||
|
})
|
||||||
|
.expect("ReplyControllerReceiver has died!")
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn send_additional_surbs_request(&self, recipient: Recipient, amount: u32) {
|
||||||
|
self.0
|
||||||
|
.unbounded_send(ReplyControllerMessage::AdditionalSurbsRequest {
|
||||||
|
recipient: Box::new(recipient),
|
||||||
|
amount,
|
||||||
|
})
|
||||||
|
.expect("ReplyControllerReceiver has died!")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) type ReplyControllerReceiver = mpsc::UnboundedReceiver<ReplyControllerMessage>;
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub(crate) enum ReplyControllerMessage {
|
||||||
|
RetransmitReply {
|
||||||
|
recipient: AnonymousSenderTag,
|
||||||
|
timed_out_ack: Weak<PendingAcknowledgement>,
|
||||||
|
extra_surb_request: bool,
|
||||||
|
},
|
||||||
|
|
||||||
|
SendReply {
|
||||||
|
recipient: AnonymousSenderTag,
|
||||||
|
message: Vec<u8>,
|
||||||
|
lane: TransmissionLane,
|
||||||
|
},
|
||||||
|
|
||||||
|
AdditionalSurbs {
|
||||||
|
sender_tag: AnonymousSenderTag,
|
||||||
|
reply_surbs: Vec<ReplySurb>,
|
||||||
|
from_surb_request: bool,
|
||||||
|
},
|
||||||
|
|
||||||
|
// Should this also be handled in here? it's technically a completely different side of the pipe
|
||||||
|
// let's see how it works when combined, might split it before creating PR
|
||||||
|
AdditionalSurbsRequest {
|
||||||
|
recipient: Box<Recipient>,
|
||||||
|
amount: u32,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct Config {
|
||||||
|
min_surb_request_size: u32,
|
||||||
|
max_surb_request_size: u32,
|
||||||
|
maximum_allowed_reply_surb_request_size: u32,
|
||||||
|
max_surb_waiting_period: Duration,
|
||||||
|
max_reply_surb_age: Duration,
|
||||||
|
max_reply_key_age: Duration,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Config {
|
||||||
|
pub(crate) fn new(
|
||||||
|
min_surb_request_size: u32,
|
||||||
|
max_surb_request_size: u32,
|
||||||
|
maximum_allowed_reply_surb_request_size: u32,
|
||||||
|
max_surb_waiting_period: Duration,
|
||||||
|
max_reply_surb_age: Duration,
|
||||||
|
max_reply_key_age: Duration,
|
||||||
|
) -> Self {
|
||||||
|
Self {
|
||||||
|
min_surb_request_size,
|
||||||
|
max_surb_request_size,
|
||||||
|
maximum_allowed_reply_surb_request_size,
|
||||||
|
max_surb_waiting_period,
|
||||||
|
max_reply_surb_age,
|
||||||
|
max_reply_key_age,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// the purpose of this task:
|
||||||
|
// - buffers split messages from input message listener if there were insufficient surbs to send them
|
||||||
|
// - upon getting extra surbs, resends them
|
||||||
|
// - so I guess it will handle all 'RepliableMessage' and requests from 'ReplyMessage'
|
||||||
|
// - replies to "give additional surbs" requests
|
||||||
|
// - will reply to future heartbeats
|
||||||
|
|
||||||
|
// TODO: this should be split into ingress and egress controllers
|
||||||
|
// because currently its trying to perform two distinct jobs
|
||||||
|
pub struct ReplyController<R> {
|
||||||
|
config: Config,
|
||||||
|
|
||||||
|
// TODO: incorporate that field at some point
|
||||||
|
// and use binomial distribution to determine the expected required number
|
||||||
|
// of surbs required to send the message through
|
||||||
|
// expected_reliability: f32,
|
||||||
|
request_receiver: ReplyControllerReceiver,
|
||||||
|
pending_replies: HashMap<AnonymousSenderTag, VecDeque<Fragment>>,
|
||||||
|
|
||||||
|
/// Retransmission packets that have already timed out and are waiting for additional reply SURBs
|
||||||
|
/// so that they could be sent back to the network. Once we receive more SURBs, we should send them ASAP.
|
||||||
|
// TODO: when purging stale entries, we must take extra care to also purge all pending ACK data!!
|
||||||
|
pending_retransmissions:
|
||||||
|
HashMap<AnonymousSenderTag, BTreeMap<FragmentIdentifier, Weak<PendingAcknowledgement>>>,
|
||||||
|
|
||||||
|
message_handler: MessageHandler<R>,
|
||||||
|
full_reply_storage: CombinedReplyStorage,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<R> ReplyController<R>
|
||||||
|
where
|
||||||
|
R: CryptoRng + Rng,
|
||||||
|
{
|
||||||
|
pub(crate) fn new(
|
||||||
|
config: Config,
|
||||||
|
message_handler: MessageHandler<R>,
|
||||||
|
full_reply_storage: CombinedReplyStorage,
|
||||||
|
request_receiver: ReplyControllerReceiver,
|
||||||
|
) -> Self {
|
||||||
|
ReplyController {
|
||||||
|
config,
|
||||||
|
request_receiver,
|
||||||
|
pending_replies: HashMap::new(),
|
||||||
|
pending_retransmissions: HashMap::new(),
|
||||||
|
message_handler,
|
||||||
|
full_reply_storage,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Inserts the pending replies into the BACK of the queue fn insert_pending_replies<V: Into<VecDeque<Fragment>>>(
|
||||||
|
fn insert_pending_replies<V: Into<VecDeque<Fragment>>>(
|
||||||
|
&mut self,
|
||||||
|
recipient: &AnonymousSenderTag,
|
||||||
|
fragments: V,
|
||||||
|
) {
|
||||||
|
if let Some(existing) = self.pending_replies.get_mut(recipient) {
|
||||||
|
existing.append(&mut fragments.into())
|
||||||
|
} else {
|
||||||
|
self.pending_replies.insert(*recipient, fragments.into());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn re_insert_pending_retransmission(
|
||||||
|
&mut self,
|
||||||
|
recipient: &AnonymousSenderTag,
|
||||||
|
data: Vec<Arc<PendingAcknowledgement>>,
|
||||||
|
) {
|
||||||
|
// the underlying entry MUST exist as we've just got data from there
|
||||||
|
let map_entry = self
|
||||||
|
.pending_retransmissions
|
||||||
|
.get_mut(recipient)
|
||||||
|
.expect("our pending retransmission entry is somehow gone!");
|
||||||
|
|
||||||
|
for pending in data {
|
||||||
|
// if it's 0, we don't need to do anything - we just got that ack!
|
||||||
|
if Arc::strong_count(&pending) > 1 {
|
||||||
|
let id = pending.inner_fragment_identifier();
|
||||||
|
let downgraded = Arc::downgrade(&pending);
|
||||||
|
map_entry.insert(id, downgraded);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn should_request_more_surbs(&self, target: &AnonymousSenderTag) -> bool {
|
||||||
|
trace!("checking if we should request more surbs from {:?}", target);
|
||||||
|
|
||||||
|
let pending_queue_size = self
|
||||||
|
.pending_replies
|
||||||
|
.get(target)
|
||||||
|
.map(|pending_queue| pending_queue.len())
|
||||||
|
.unwrap_or_default();
|
||||||
|
|
||||||
|
let retransmission_queue = self
|
||||||
|
.pending_retransmissions
|
||||||
|
.get(target)
|
||||||
|
.map(|pending_queue| pending_queue.len())
|
||||||
|
.unwrap_or_default();
|
||||||
|
|
||||||
|
let total_queue = pending_queue_size + retransmission_queue;
|
||||||
|
|
||||||
|
// simple as that - there's absolutely nothing to retransmit
|
||||||
|
if total_queue == 0 {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
let available_surbs = self
|
||||||
|
.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.available_surbs(target);
|
||||||
|
let pending_surbs = self
|
||||||
|
.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.pending_reception(target) as usize;
|
||||||
|
let min_surbs_threshold = self
|
||||||
|
.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.min_surb_threshold();
|
||||||
|
let max_surbs_threshold = self
|
||||||
|
.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.max_surb_threshold();
|
||||||
|
|
||||||
|
debug!("total queue size: {total_queue} = pending data {pending_queue_size} + pending retransmission {retransmission_queue}, available surbs: {available_surbs} pending surbs: {pending_surbs} threshold range: {min_surbs_threshold}..{max_surbs_threshold}");
|
||||||
|
|
||||||
|
(pending_surbs + available_surbs) < max_surbs_threshold
|
||||||
|
&& (pending_surbs + available_surbs) < (total_queue + min_surbs_threshold)
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn handle_send_reply(
|
||||||
|
&mut self,
|
||||||
|
recipient_tag: AnonymousSenderTag,
|
||||||
|
data: Vec<u8>,
|
||||||
|
lane: TransmissionLane,
|
||||||
|
) {
|
||||||
|
if !self
|
||||||
|
.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.contains_surbs_for(&recipient_tag)
|
||||||
|
{
|
||||||
|
warn!("received reply request for {:?} but we don't have any surbs stored for that recipient!", recipient_tag);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
trace!("handling reply to {:?}", recipient_tag);
|
||||||
|
let fragments = self.message_handler.split_reply_message(data);
|
||||||
|
|
||||||
|
let required_surbs = fragments.len();
|
||||||
|
trace!("This reply requires {:?} SURBs", required_surbs);
|
||||||
|
|
||||||
|
// TODO: edge case:
|
||||||
|
// we're making a lot of requests and have to request a lot of surbs
|
||||||
|
// (but at some point we run out of surbs for surb requests)
|
||||||
|
|
||||||
|
let (surbs, _surbs_left) = self
|
||||||
|
.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.get_reply_surbs(&recipient_tag, required_surbs);
|
||||||
|
|
||||||
|
if let Some(reply_surbs) = surbs {
|
||||||
|
if let Err(err) = self
|
||||||
|
.message_handler
|
||||||
|
.try_send_reply_chunks(recipient_tag, fragments, reply_surbs, lane)
|
||||||
|
.await
|
||||||
|
{
|
||||||
|
let err = err.return_unused_surbs(
|
||||||
|
self.full_reply_storage.surbs_storage_ref(),
|
||||||
|
&recipient_tag,
|
||||||
|
);
|
||||||
|
warn!("failed to send reply to {:?} - {err}", recipient_tag);
|
||||||
|
|
||||||
|
// TODO: should we buffer that data to try again?
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// we don't have enough surbs for this reply
|
||||||
|
self.insert_pending_replies(&recipient_tag, fragments);
|
||||||
|
|
||||||
|
if self.should_request_more_surbs(&recipient_tag) {
|
||||||
|
self.request_reply_surbs_for_queue_clearing(recipient_tag)
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn request_additional_reply_surbs(
|
||||||
|
&mut self,
|
||||||
|
target: AnonymousSenderTag,
|
||||||
|
amount: u32,
|
||||||
|
) -> Result<(), PreparationError> {
|
||||||
|
let reply_surb = self
|
||||||
|
.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.get_reply_surb_ignoring_threshold(&target)
|
||||||
|
.and_then(|(reply_surb, _)| reply_surb)
|
||||||
|
.ok_or(PreparationError::NotEnoughSurbs {
|
||||||
|
available: 0,
|
||||||
|
required: 1,
|
||||||
|
})?;
|
||||||
|
|
||||||
|
if let Err(err) = self
|
||||||
|
.message_handler
|
||||||
|
.try_request_additional_reply_surbs(target, reply_surb, amount)
|
||||||
|
.await
|
||||||
|
{
|
||||||
|
let err = err.return_unused_surbs(self.full_reply_storage.surbs_storage_ref(), &target);
|
||||||
|
warn!(
|
||||||
|
"failed to request additional surbs from {:?} - {err}",
|
||||||
|
target
|
||||||
|
);
|
||||||
|
return Err(err);
|
||||||
|
} else {
|
||||||
|
self.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.increment_pending_reception(&target, amount);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn try_clear_pending_retransmission(&mut self, target: AnonymousSenderTag) {
|
||||||
|
trace!("trying to clear pending retransmission queue");
|
||||||
|
let available_surbs = self
|
||||||
|
.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.available_surbs(&target);
|
||||||
|
let min_surbs_threshold = self
|
||||||
|
.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.min_surb_threshold();
|
||||||
|
|
||||||
|
let max_to_clear = if available_surbs > min_surbs_threshold {
|
||||||
|
available_surbs - min_surbs_threshold
|
||||||
|
} else {
|
||||||
|
trace!("we don't have enough surbs for retransmission queue clearing...");
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
trace!("we can clear up to {max_to_clear} entries");
|
||||||
|
|
||||||
|
let Some(pending) = self.pending_retransmissions.get_mut(&target) else {
|
||||||
|
trace!("there are no pending retransmissions for {target}!");
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut to_take = Vec::new();
|
||||||
|
let mut to_remove = Vec::new();
|
||||||
|
|
||||||
|
// TODO: once rust 1.66.0 is stabilised on 15.12.22, just change it to
|
||||||
|
// `.pop_front()` to directly take ownership
|
||||||
|
for (k, data) in pending.iter() {
|
||||||
|
let upgraded = match data.upgrade() {
|
||||||
|
Some(upgraded) => upgraded,
|
||||||
|
None => {
|
||||||
|
// we got the ack while the data was waiting in the queue
|
||||||
|
to_remove.push(*k);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
to_take.push(upgraded);
|
||||||
|
|
||||||
|
// we have taken as many entries as we could have
|
||||||
|
if to_take.len() >= max_to_clear {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
// TODO: use if upgraded.is_extra_surb_request() to bypass the limit
|
||||||
|
}
|
||||||
|
|
||||||
|
for ack in &to_take {
|
||||||
|
pending.remove(&ack.inner_fragment_identifier());
|
||||||
|
}
|
||||||
|
|
||||||
|
for id in to_remove {
|
||||||
|
pending.remove(&id);
|
||||||
|
}
|
||||||
|
|
||||||
|
if to_take.is_empty() {
|
||||||
|
// no need to do anything
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let (surbs_for_reply, _) = self
|
||||||
|
.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.get_reply_surbs(&target, to_take.len());
|
||||||
|
|
||||||
|
let Some(surbs_for_reply) = surbs_for_reply else {
|
||||||
|
error!("somehow different task has stolen our reply surbs! - this should have been impossible");
|
||||||
|
self.re_insert_pending_retransmission(&target, to_take);
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
let to_send_vec = to_take.iter().map(|ack| ack.fragment_data()).collect();
|
||||||
|
|
||||||
|
if let Err(err) = self
|
||||||
|
.message_handler
|
||||||
|
.try_send_retransmission_reply_chunks(
|
||||||
|
to_send_vec,
|
||||||
|
surbs_for_reply,
|
||||||
|
TransmissionLane::Retransmission,
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
{
|
||||||
|
let err = err.return_unused_surbs(self.full_reply_storage.surbs_storage_ref(), &target);
|
||||||
|
self.re_insert_pending_retransmission(&target, to_take);
|
||||||
|
|
||||||
|
warn!(
|
||||||
|
"failed to clear pending retransmission queue for {:?} - {err}",
|
||||||
|
target
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn pop_at_most_pending_replies(
|
||||||
|
&mut self,
|
||||||
|
from: &AnonymousSenderTag,
|
||||||
|
amount: usize,
|
||||||
|
) -> Option<VecDeque<Fragment>> {
|
||||||
|
// if possible, pop all pending replies, if not, pop only entries for which we'd have a reply surb
|
||||||
|
let total = self.pending_replies.get(from)?.len();
|
||||||
|
trace!("pending queue has {total} elements");
|
||||||
|
if total == 0 {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
if total < amount {
|
||||||
|
self.pending_replies.remove(from)
|
||||||
|
} else {
|
||||||
|
Some(
|
||||||
|
self.pending_replies
|
||||||
|
.get_mut(from)?
|
||||||
|
.drain(..amount)
|
||||||
|
.collect(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn try_clear_pending_queue(&mut self, target: AnonymousSenderTag) {
|
||||||
|
trace!("trying to clear pending queue");
|
||||||
|
let available_surbs = self
|
||||||
|
.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.available_surbs(&target);
|
||||||
|
let min_surbs_threshold = self
|
||||||
|
.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.min_surb_threshold();
|
||||||
|
|
||||||
|
let max_to_clear = if available_surbs > min_surbs_threshold {
|
||||||
|
available_surbs - min_surbs_threshold
|
||||||
|
} else {
|
||||||
|
trace!("we don't have enough surbs for queue clearing...");
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
trace!("we can clear up to {max_to_clear} entries");
|
||||||
|
|
||||||
|
// we're guaranteed to not get more entries than we have reply surbs for
|
||||||
|
if let Some(to_send) = self.pop_at_most_pending_replies(&target, max_to_clear) {
|
||||||
|
let to_send_vec = to_send.iter().cloned().collect::<Vec<_>>();
|
||||||
|
|
||||||
|
if to_send_vec.is_empty() {
|
||||||
|
panic!(
|
||||||
|
"please let the devs know if you ever see this message (reply_controller.rs)"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
let (surbs_for_reply, _) = self
|
||||||
|
.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.get_reply_surbs(&target, to_send_vec.len());
|
||||||
|
|
||||||
|
let Some(surbs_for_reply) = surbs_for_reply else {
|
||||||
|
error!("somehow different task has stolen our reply surbs! - this should have been impossible");
|
||||||
|
self.insert_pending_replies(&target, to_send);
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Err(err) = self
|
||||||
|
.message_handler
|
||||||
|
.try_send_reply_chunks(
|
||||||
|
target,
|
||||||
|
to_send_vec,
|
||||||
|
surbs_for_reply,
|
||||||
|
TransmissionLane::General,
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
{
|
||||||
|
let err =
|
||||||
|
err.return_unused_surbs(self.full_reply_storage.surbs_storage_ref(), &target);
|
||||||
|
self.insert_pending_replies(&target, to_send);
|
||||||
|
warn!("failed to clear pending queue for {:?} - {err}", target);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
trace!("the pending queue is empty");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn handle_received_surbs(
|
||||||
|
&mut self,
|
||||||
|
from: AnonymousSenderTag,
|
||||||
|
reply_surbs: Vec<ReplySurb>,
|
||||||
|
from_surb_request: bool,
|
||||||
|
) {
|
||||||
|
trace!("handling received surbs");
|
||||||
|
|
||||||
|
// clear the requesting flag since we should have been asking for surbs
|
||||||
|
self.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.reset_surbs_last_received_at(&from);
|
||||||
|
if from_surb_request {
|
||||||
|
self.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.decrement_pending_reception(&from, reply_surbs.len() as u32);
|
||||||
|
}
|
||||||
|
|
||||||
|
// store received surbs
|
||||||
|
self.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.insert_surbs(&from, reply_surbs);
|
||||||
|
|
||||||
|
// use as many as we can for clearing pending retransmission queue
|
||||||
|
self.try_clear_pending_retransmission(from).await;
|
||||||
|
|
||||||
|
// use as many as we can for clearing pending 'normal' queue
|
||||||
|
self.try_clear_pending_queue(from).await;
|
||||||
|
|
||||||
|
// if we have to, request more
|
||||||
|
if self.should_request_more_surbs(&from) {
|
||||||
|
self.request_reply_surbs_for_queue_clearing(from).await;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn handle_surb_request(&mut self, recipient: Recipient, mut amount: u32) {
|
||||||
|
// 1. check whether we sent any surbs in the past to this recipient, otherwise
|
||||||
|
// they have no business in asking for more
|
||||||
|
if !self
|
||||||
|
.full_reply_storage
|
||||||
|
.tags_storage_ref()
|
||||||
|
.exists(&recipient)
|
||||||
|
{
|
||||||
|
warn!("{recipient} asked us for reply SURBs even though we never sent them any anonymous messages before!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2. check whether the requested amount is within sane range
|
||||||
|
if amount > self.config.maximum_allowed_reply_surb_request_size {
|
||||||
|
warn!("The requested reply surb amount is larger than our maximum allowed ({amount} > {}). Lowering it to a more sane value...", self.config.maximum_allowed_reply_surb_request_size);
|
||||||
|
amount = self.config.maximum_allowed_reply_surb_request_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 3. construct and send the surbs away
|
||||||
|
// (send them in smaller batches to make the experience a bit smoother
|
||||||
|
let mut remaining = amount;
|
||||||
|
while remaining > 0 {
|
||||||
|
let to_send = min(remaining, 100);
|
||||||
|
if let Err(err) = self
|
||||||
|
.message_handler
|
||||||
|
.try_send_additional_reply_surbs(recipient, to_send)
|
||||||
|
.await
|
||||||
|
{
|
||||||
|
warn!("failed to send additional surbs to {recipient} - {err}");
|
||||||
|
} else {
|
||||||
|
trace!("sent {to_send} reply SURBs to {recipient}");
|
||||||
|
}
|
||||||
|
|
||||||
|
remaining -= to_send;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn buffer_pending_ack(
|
||||||
|
&mut self,
|
||||||
|
recipient: AnonymousSenderTag,
|
||||||
|
ack_ref: Arc<PendingAcknowledgement>,
|
||||||
|
weak_ack_ref: Weak<PendingAcknowledgement>,
|
||||||
|
) {
|
||||||
|
let frag_id = ack_ref.inner_fragment_identifier();
|
||||||
|
if let Some(existing) = self.pending_retransmissions.get_mut(&recipient) {
|
||||||
|
if let Entry::Vacant(e) = existing.entry(frag_id) {
|
||||||
|
e.insert(weak_ack_ref);
|
||||||
|
} else {
|
||||||
|
warn!("we're already trying to retransmit {frag_id}. We must be really behind in surbs!");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
let mut inner = BTreeMap::new();
|
||||||
|
inner.insert(frag_id, weak_ack_ref);
|
||||||
|
self.pending_retransmissions.insert(recipient, inner);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn handle_reply_retransmission(
|
||||||
|
&mut self,
|
||||||
|
recipient_tag: AnonymousSenderTag,
|
||||||
|
timed_out_ack: Weak<PendingAcknowledgement>,
|
||||||
|
extra_surbs_request: bool,
|
||||||
|
) {
|
||||||
|
// seems we got the ack in the end
|
||||||
|
let ack_ref = match timed_out_ack.upgrade() {
|
||||||
|
Some(ack) => ack,
|
||||||
|
None => {
|
||||||
|
debug!("we received the ack for one of the reply packets as we were putting it in the retransmission queue");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// if this is retransmission for obtaining additional reply surbs,
|
||||||
|
// we can dip below the storage threshold
|
||||||
|
let (maybe_reply_surb, _) = if extra_surbs_request {
|
||||||
|
self.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.get_reply_surb_ignoring_threshold(&recipient_tag)
|
||||||
|
} else {
|
||||||
|
self.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.get_reply_surb(&recipient_tag)
|
||||||
|
}
|
||||||
|
.expect("attempted to retransmit a packet to an unknown recipient - we shouldn't have sent the original packet in the first place!");
|
||||||
|
|
||||||
|
if let Some(reply_surb) = maybe_reply_surb {
|
||||||
|
match self
|
||||||
|
.message_handler
|
||||||
|
.try_prepare_single_reply_chunk_for_sending(reply_surb, ack_ref.fragment_data())
|
||||||
|
.await
|
||||||
|
{
|
||||||
|
Ok(prepared) => {
|
||||||
|
// drop the ack ref so that controller would not panic on `UpdateTimer` if that task
|
||||||
|
// got to handle the action before this function terminated (which is very much
|
||||||
|
// possible if `forward_messages` takes a while)
|
||||||
|
drop(ack_ref);
|
||||||
|
|
||||||
|
self.message_handler
|
||||||
|
.update_ack_delay(prepared.fragment_identifier, prepared.total_delay);
|
||||||
|
self.message_handler
|
||||||
|
.forward_messages(vec![prepared.into()], TransmissionLane::Retransmission)
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
Err(err) => {
|
||||||
|
let err = err.return_unused_surbs(
|
||||||
|
self.full_reply_storage.surbs_storage_ref(),
|
||||||
|
&recipient_tag,
|
||||||
|
);
|
||||||
|
warn!("failed to prepare message for retransmission - {err}");
|
||||||
|
// we buffer that packet and to try another day
|
||||||
|
self.buffer_pending_ack(recipient_tag, ack_ref, timed_out_ack);
|
||||||
|
|
||||||
|
if self.should_request_more_surbs(&recipient_tag) {
|
||||||
|
self.request_reply_surbs_for_queue_clearing(recipient_tag)
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
self.buffer_pending_ack(recipient_tag, ack_ref, timed_out_ack);
|
||||||
|
|
||||||
|
if self.should_request_more_surbs(&recipient_tag) {
|
||||||
|
self.request_reply_surbs_for_queue_clearing(recipient_tag)
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn handle_request(&mut self, request: ReplyControllerMessage) {
|
||||||
|
match request {
|
||||||
|
ReplyControllerMessage::RetransmitReply {
|
||||||
|
recipient,
|
||||||
|
timed_out_ack,
|
||||||
|
extra_surb_request,
|
||||||
|
} => {
|
||||||
|
self.handle_reply_retransmission(recipient, timed_out_ack, extra_surb_request)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
ReplyControllerMessage::SendReply {
|
||||||
|
recipient,
|
||||||
|
message,
|
||||||
|
lane,
|
||||||
|
} => self.handle_send_reply(recipient, message, lane).await,
|
||||||
|
ReplyControllerMessage::AdditionalSurbs {
|
||||||
|
sender_tag,
|
||||||
|
reply_surbs,
|
||||||
|
from_surb_request,
|
||||||
|
} => {
|
||||||
|
self.handle_received_surbs(sender_tag, reply_surbs, from_surb_request)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
ReplyControllerMessage::AdditionalSurbsRequest { recipient, amount } => {
|
||||||
|
self.handle_surb_request(*recipient, amount).await
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn request_reply_surbs_for_queue_clearing(&mut self, target: AnonymousSenderTag) {
|
||||||
|
trace!("requesting surbs for queues clearing");
|
||||||
|
|
||||||
|
let pending_queue_size = self
|
||||||
|
.pending_replies
|
||||||
|
.get(&target)
|
||||||
|
.map(|pending_queue| pending_queue.len())
|
||||||
|
.unwrap_or_default();
|
||||||
|
|
||||||
|
let retransmission_queue = self
|
||||||
|
.pending_retransmissions
|
||||||
|
.get(&target)
|
||||||
|
.map(|pending_queue| pending_queue.len())
|
||||||
|
.unwrap_or_default();
|
||||||
|
|
||||||
|
let total_queue = (pending_queue_size + retransmission_queue) as u32;
|
||||||
|
|
||||||
|
if total_queue == 0 {
|
||||||
|
trace!("the pending queues for {:?} are already empty", target);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let request_size = min(
|
||||||
|
self.config.max_surb_request_size,
|
||||||
|
max(total_queue, self.config.min_surb_request_size),
|
||||||
|
);
|
||||||
|
|
||||||
|
if let Err(err) = self
|
||||||
|
.request_additional_reply_surbs(target, request_size)
|
||||||
|
.await
|
||||||
|
{
|
||||||
|
warn!("failed to request additional surbs... - {err}")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn inspect_stale_entries(&mut self) {
|
||||||
|
let mut to_request = Vec::new();
|
||||||
|
let mut to_remove = Vec::new();
|
||||||
|
|
||||||
|
let now = OffsetDateTime::now_utc();
|
||||||
|
for (pending_reply_target, vals) in &self.pending_replies {
|
||||||
|
if vals.is_empty() {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let Some(last_received) = self.full_reply_storage.surbs_storage_ref().surbs_last_received_at(pending_reply_target) else {
|
||||||
|
error!("we have {} pending replies for {pending_reply_target}, but we somehow never received any reply surbs from them!", vals.len());
|
||||||
|
to_remove.push(*pending_reply_target);
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
|
||||||
|
// this should never ever happen (famous last words, eh?), but in case it DOES happen eventually
|
||||||
|
// purge that malformed data
|
||||||
|
let Ok(last_received_time) = OffsetDateTime::from_unix_timestamp(last_received) else {
|
||||||
|
error!("somehow our stored timestamp ({last_received}) for surbs from {pending_reply_target} is corrupted!. Going to remove all the associated entries");
|
||||||
|
to_remove.push(*pending_reply_target);
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
|
||||||
|
let diff = now - last_received_time;
|
||||||
|
|
||||||
|
if diff > self.config.max_surb_waiting_period {
|
||||||
|
warn!("We haven't received any surbs in {:?} from {pending_reply_target}. Going to explicitly ask for more", diff);
|
||||||
|
to_request.push(*pending_reply_target);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for pending_reply_target in to_request {
|
||||||
|
self.request_reply_surbs_for_queue_clearing(pending_reply_target)
|
||||||
|
.await;
|
||||||
|
self.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.reset_pending_reception(&pending_reply_target)
|
||||||
|
}
|
||||||
|
for to_remove in to_remove {
|
||||||
|
self.pending_replies.remove(&to_remove);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn invalidate_old_data(&self) {
|
||||||
|
let now = OffsetDateTime::now_utc();
|
||||||
|
|
||||||
|
let mut to_remove_surbs = Vec::new();
|
||||||
|
let mut to_remove_keys = Vec::new();
|
||||||
|
for map_ref in self.full_reply_storage.surbs_storage_ref().as_raw_iter() {
|
||||||
|
let (sender, received) = map_ref.pair();
|
||||||
|
// TODO: handle the following edge case:
|
||||||
|
// there's a malicious client sending us exactly one reply surb just before we should have invalidated
|
||||||
|
// the data thus making us keep everything in memory
|
||||||
|
// possible solution: keep timestamp PER reply surb (but that seems like an overkill)
|
||||||
|
// but I doubt this is ever going to be a problem...
|
||||||
|
// ...
|
||||||
|
// However, if you're reading this message, it probably became a legit problem,
|
||||||
|
// so I guess add timestamp per surb then? chop-chop.
|
||||||
|
|
||||||
|
let last_received = received.surbs_last_received_at();
|
||||||
|
// this should never ever happen (famous last words, eh?), but in case it DOES happen eventually
|
||||||
|
// purge that malformed data
|
||||||
|
let Ok(last_received_time) = OffsetDateTime::from_unix_timestamp(last_received) else {
|
||||||
|
error!("somehow our stored timestamp ({last_received}) for surbs from {sender} is corrupted!. Going to remove all the associated entries");
|
||||||
|
to_remove_surbs.push(*sender);
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
let diff = now - last_received_time;
|
||||||
|
|
||||||
|
if diff > self.config.max_reply_surb_age {
|
||||||
|
info!("it's been {diff:?} since we last received any reply surb from {sender}. Going to remove all stored entries...");
|
||||||
|
|
||||||
|
to_remove_surbs.push(*sender);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for map_ref in self.full_reply_storage.key_storage_ref().as_raw_iter() {
|
||||||
|
let (digest, reply_key) = map_ref.pair();
|
||||||
|
|
||||||
|
// this should never ever happen (famous last words, eh?), but in case it DOES happen eventually
|
||||||
|
// purge that malformed data
|
||||||
|
let Ok(sent_at) = OffsetDateTime::from_unix_timestamp(reply_key.sent_at_timestamp) else {
|
||||||
|
error!("somehow our stored timestamp ({}) for one of our reply key is corrupted!. Going to remove all the entry", reply_key.sent_at_timestamp);
|
||||||
|
to_remove_keys.push(*digest);
|
||||||
|
continue;
|
||||||
|
};
|
||||||
|
|
||||||
|
let diff = now - sent_at;
|
||||||
|
|
||||||
|
if diff > self.config.max_reply_key_age {
|
||||||
|
debug!("it's been {diff:?} since we created this reply key. it's probably never going to get used, so we're going to purge it...");
|
||||||
|
to_remove_keys.push(*digest);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for to_remove in to_remove_surbs {
|
||||||
|
self.full_reply_storage
|
||||||
|
.surbs_storage_ref()
|
||||||
|
.remove(&to_remove);
|
||||||
|
}
|
||||||
|
|
||||||
|
for to_remove in to_remove_keys {
|
||||||
|
self.full_reply_storage.key_storage().remove(to_remove)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn create_interval_stream(polling_rate: Duration) -> IntervalStream {
|
||||||
|
#[cfg(not(target_arch = "wasm32"))]
|
||||||
|
return tokio_stream::wrappers::IntervalStream::new(tokio::time::interval(polling_rate));
|
||||||
|
|
||||||
|
#[cfg(target_arch = "wasm32")]
|
||||||
|
return gloo_timers::future::IntervalStream::new(polling_rate.as_millis() as u32);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn run_with_shutdown(&mut self, mut shutdown: task::ShutdownListener) {
|
||||||
|
debug!("Started ReplyController with graceful shutdown support");
|
||||||
|
|
||||||
|
let polling_rate = Duration::from_secs(5);
|
||||||
|
let mut stale_inspection = Self::create_interval_stream(polling_rate);
|
||||||
|
|
||||||
|
// this is in the order of hours/days so we don't have to poll it that often
|
||||||
|
let polling_rate = Duration::from_secs(self.config.max_reply_surb_age.as_secs() / 10);
|
||||||
|
let mut invalidation_inspection = Self::create_interval_stream(polling_rate);
|
||||||
|
|
||||||
|
while !shutdown.is_shutdown() {
|
||||||
|
tokio::select! {
|
||||||
|
biased;
|
||||||
|
_ = shutdown.recv() => {
|
||||||
|
log::trace!("ReplyController: Received shutdown");
|
||||||
|
},
|
||||||
|
req = self.request_receiver.next() => match req {
|
||||||
|
Some(req) => self.handle_request(req).await,
|
||||||
|
None => {
|
||||||
|
log::trace!("ReplyController: Stopping since channel closed");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
_ = stale_inspection.next() => {
|
||||||
|
self.inspect_stale_entries().await
|
||||||
|
},
|
||||||
|
_ = invalidation_inspection.next() => {
|
||||||
|
self.invalidate_old_data().await
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
assert!(shutdown.is_shutdown_poll());
|
||||||
|
log::debug!("ReplyController: Exiting");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,43 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use crate::client::replies::reply_storage::backend::Empty;
|
||||||
|
use crate::client::replies::reply_storage::{CombinedReplyStorage, ReplyStorageBackend};
|
||||||
|
use async_trait::async_trait;
|
||||||
|
|
||||||
|
// well, right now we don't have the browser storage : (
|
||||||
|
// so we keep everything in memory
|
||||||
|
pub struct Backend {
|
||||||
|
empty: Empty,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Backend {
|
||||||
|
pub fn new(min_surb_threshold: usize, max_surb_threshold: usize) -> Self {
|
||||||
|
Backend {
|
||||||
|
empty: Empty {
|
||||||
|
min_surb_threshold,
|
||||||
|
max_surb_threshold,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
impl ReplyStorageBackend for Backend {
|
||||||
|
type StorageError = <Empty as ReplyStorageBackend>::StorageError;
|
||||||
|
|
||||||
|
async fn flush_surb_storage(
|
||||||
|
&mut self,
|
||||||
|
storage: &CombinedReplyStorage,
|
||||||
|
) -> Result<(), Self::StorageError> {
|
||||||
|
self.empty.flush_surb_storage(storage).await
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn init_fresh(&mut self, fresh: &CombinedReplyStorage) -> Result<(), Self::StorageError> {
|
||||||
|
self.empty.init_fresh(fresh).await
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn load_surb_storage(&self) -> Result<CombinedReplyStorage, Self::StorageError> {
|
||||||
|
self.empty.load_surb_storage().await
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,53 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use std::io;
|
||||||
|
use std::path::PathBuf;
|
||||||
|
use thiserror::Error;
|
||||||
|
|
||||||
|
#[derive(Debug, Error)]
|
||||||
|
pub enum StorageError {
|
||||||
|
#[error("the provided database path doesn't have a filename defined")]
|
||||||
|
DatabasePathWithoutFilename { provided_path: PathBuf },
|
||||||
|
|
||||||
|
#[error("failed to rename our databse file - {source}")]
|
||||||
|
DatabaseRenameError {
|
||||||
|
#[source]
|
||||||
|
source: io::Error,
|
||||||
|
},
|
||||||
|
|
||||||
|
#[error("failed to rename our old databse file - {source}")]
|
||||||
|
DatabaseOldFileRemoveError {
|
||||||
|
#[source]
|
||||||
|
source: io::Error,
|
||||||
|
},
|
||||||
|
|
||||||
|
#[error("failed to perform sqlx migration: {source}")]
|
||||||
|
MigrationError {
|
||||||
|
#[source]
|
||||||
|
#[from]
|
||||||
|
source: sqlx::migrate::MigrateError,
|
||||||
|
},
|
||||||
|
|
||||||
|
#[error("failed to connect to the underlying connection pool: {source}")]
|
||||||
|
DatabaseConnectionError {
|
||||||
|
#[source]
|
||||||
|
source: sqlx::error::Error,
|
||||||
|
},
|
||||||
|
|
||||||
|
#[error("failed to run the SQL query: {source}")]
|
||||||
|
QueryError {
|
||||||
|
#[source]
|
||||||
|
#[from]
|
||||||
|
source: sqlx::error::Error,
|
||||||
|
},
|
||||||
|
|
||||||
|
#[error("The loaded data is inconsistent - it seems that on the last shutdown the client hasn't finished the data flush. You may have to remove the entire storage manually")]
|
||||||
|
IncompleteDataFlush,
|
||||||
|
|
||||||
|
#[error("data retrieved from the underlying storage is corrupted: {details}")]
|
||||||
|
CorruptedData {
|
||||||
|
details: String,
|
||||||
|
// err: Option<Box<dyn std::error::Error>>
|
||||||
|
},
|
||||||
|
}
|
||||||
@@ -0,0 +1,257 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use crate::client::replies::reply_storage::backend::fs_backend::error::StorageError;
|
||||||
|
use crate::client::replies::reply_storage::backend::fs_backend::models::{
|
||||||
|
ReplySurbStorageMetadata, StoredReplyKey, StoredReplySurb, StoredSenderTag, StoredSurbSender,
|
||||||
|
};
|
||||||
|
use log::{error, info};
|
||||||
|
use sqlx::ConnectOptions;
|
||||||
|
use std::path::Path;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub(crate) struct StorageManager {
|
||||||
|
pub(crate) connection_pool: sqlx::SqlitePool,
|
||||||
|
}
|
||||||
|
|
||||||
|
// all SQL goes here
|
||||||
|
impl StorageManager {
|
||||||
|
pub(crate) async fn init<P: AsRef<Path>>(
|
||||||
|
database_path: P,
|
||||||
|
fresh: bool,
|
||||||
|
) -> Result<Self, StorageError> {
|
||||||
|
let mut opts = sqlx::sqlite::SqliteConnectOptions::new()
|
||||||
|
.filename(database_path)
|
||||||
|
.create_if_missing(fresh);
|
||||||
|
|
||||||
|
opts.disable_statement_logging();
|
||||||
|
|
||||||
|
let connection_pool = match sqlx::SqlitePool::connect_with(opts).await {
|
||||||
|
Ok(pool) => pool,
|
||||||
|
Err(err) => {
|
||||||
|
error!("Failed to connect to SQLx database: {err}");
|
||||||
|
return Err(StorageError::DatabaseConnectionError { source: err });
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Err(err) = sqlx::migrate!("./fs_surbs_migrations")
|
||||||
|
.run(&connection_pool)
|
||||||
|
.await
|
||||||
|
{
|
||||||
|
error!("Failed to initialize SQLx database: {err}");
|
||||||
|
return Err(err.into());
|
||||||
|
}
|
||||||
|
|
||||||
|
info!("Database migration finished!");
|
||||||
|
Ok(StorageManager { connection_pool })
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub(crate) async fn status_table_exists(&self) -> Result<bool, sqlx::Error> {
|
||||||
|
sqlx::query!("SELECT name FROM sqlite_master WHERE type='table' AND name='status'")
|
||||||
|
.fetch_optional(&self.connection_pool)
|
||||||
|
.await
|
||||||
|
.map(|r| r.is_some())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn create_status_table(&self) -> Result<(), sqlx::Error> {
|
||||||
|
sqlx::query!("INSERT INTO status(flush_in_progress, previous_flush_timestamp, client_in_use) VALUES (0, 0, 1)")
|
||||||
|
.execute(&self.connection_pool)
|
||||||
|
.await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn get_flush_status(&self) -> Result<bool, sqlx::Error> {
|
||||||
|
sqlx::query!("SELECT flush_in_progress FROM status;")
|
||||||
|
.fetch_one(&self.connection_pool)
|
||||||
|
.await
|
||||||
|
.map(|r| r.flush_in_progress > 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn set_previous_flush_timestamp(
|
||||||
|
&self,
|
||||||
|
timestamp: i64,
|
||||||
|
) -> Result<(), sqlx::Error> {
|
||||||
|
sqlx::query!("UPDATE status SET previous_flush_timestamp = ?", timestamp)
|
||||||
|
.execute(&self.connection_pool)
|
||||||
|
.await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn get_previous_flush_timestamp(&self) -> Result<i64, sqlx::Error> {
|
||||||
|
sqlx::query!("SELECT previous_flush_timestamp FROM status;")
|
||||||
|
.fetch_one(&self.connection_pool)
|
||||||
|
.await
|
||||||
|
.map(|r| r.previous_flush_timestamp)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn set_flush_status(&self, in_progress: bool) -> Result<(), sqlx::Error> {
|
||||||
|
let in_progress_int = i64::from(in_progress);
|
||||||
|
sqlx::query!("UPDATE status SET flush_in_progress = ?", in_progress_int)
|
||||||
|
.execute(&self.connection_pool)
|
||||||
|
.await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn get_client_in_use_status(&self) -> Result<bool, sqlx::Error> {
|
||||||
|
sqlx::query!("SELECT client_in_use FROM status;")
|
||||||
|
.fetch_one(&self.connection_pool)
|
||||||
|
.await
|
||||||
|
.map(|r| r.client_in_use > 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn set_client_in_use_status(&self, in_use: bool) -> Result<(), sqlx::Error> {
|
||||||
|
let in_use_int = i64::from(in_use);
|
||||||
|
sqlx::query!("UPDATE status SET client_in_use = ?", in_use_int)
|
||||||
|
.execute(&self.connection_pool)
|
||||||
|
.await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn delete_all_tags(&self) -> Result<(), sqlx::Error> {
|
||||||
|
sqlx::query!("DELETE FROM sender_tag;")
|
||||||
|
.execute(&self.connection_pool)
|
||||||
|
.await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn get_tags(&self) -> Result<Vec<StoredSenderTag>, sqlx::Error> {
|
||||||
|
sqlx::query_as!(StoredSenderTag, "SELECT * FROM sender_tag;",)
|
||||||
|
.fetch_all(&self.connection_pool)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn insert_tag(&self, stored_tag: StoredSenderTag) -> Result<(), sqlx::Error> {
|
||||||
|
sqlx::query!(
|
||||||
|
r#"
|
||||||
|
INSERT INTO sender_tag(recipient, tag) VALUES (?, ?);
|
||||||
|
"#,
|
||||||
|
stored_tag.recipient,
|
||||||
|
stored_tag.tag
|
||||||
|
)
|
||||||
|
.execute(&self.connection_pool)
|
||||||
|
.await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn delete_all_reply_keys(&self) -> Result<(), sqlx::Error> {
|
||||||
|
sqlx::query!("DELETE FROM reply_key;")
|
||||||
|
.execute(&self.connection_pool)
|
||||||
|
.await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn get_reply_keys(&self) -> Result<Vec<StoredReplyKey>, sqlx::Error> {
|
||||||
|
sqlx::query_as!(StoredReplyKey, "SELECT * FROM reply_key;",)
|
||||||
|
.fetch_all(&self.connection_pool)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn insert_reply_key(
|
||||||
|
&self,
|
||||||
|
stored_reply_key: StoredReplyKey,
|
||||||
|
) -> Result<(), sqlx::Error> {
|
||||||
|
sqlx::query!(
|
||||||
|
r#"
|
||||||
|
INSERT INTO reply_key(key_digest, reply_key, sent_at_timestamp) VALUES (?, ?, ?);
|
||||||
|
"#,
|
||||||
|
stored_reply_key.key_digest,
|
||||||
|
stored_reply_key.reply_key,
|
||||||
|
stored_reply_key.sent_at_timestamp
|
||||||
|
)
|
||||||
|
.execute(&self.connection_pool)
|
||||||
|
.await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn get_surb_senders(&self) -> Result<Vec<StoredSurbSender>, sqlx::Error> {
|
||||||
|
sqlx::query_as!(StoredSurbSender, "SELECT * FROM reply_surb_sender;",)
|
||||||
|
.fetch_all(&self.connection_pool)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn insert_surb_sender(
|
||||||
|
&self,
|
||||||
|
stored_surb_sender: StoredSurbSender,
|
||||||
|
) -> Result<i64, sqlx::Error> {
|
||||||
|
let id = sqlx::query!(
|
||||||
|
r#"
|
||||||
|
INSERT INTO reply_surb_sender(tag, last_sent_timestamp) VALUES (?, ?);
|
||||||
|
"#,
|
||||||
|
stored_surb_sender.tag,
|
||||||
|
stored_surb_sender.last_sent_timestamp
|
||||||
|
)
|
||||||
|
.execute(&self.connection_pool)
|
||||||
|
.await?
|
||||||
|
.last_insert_rowid();
|
||||||
|
Ok(id)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn get_reply_surbs(
|
||||||
|
&self,
|
||||||
|
sender_id: i64,
|
||||||
|
) -> Result<Vec<StoredReplySurb>, sqlx::Error> {
|
||||||
|
sqlx::query_as!(
|
||||||
|
StoredReplySurb,
|
||||||
|
"SELECT * FROM reply_surb WHERE reply_surb_sender_id = ?",
|
||||||
|
sender_id
|
||||||
|
)
|
||||||
|
.fetch_all(&self.connection_pool)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn delete_all_reply_surb_data(&self) -> Result<(), sqlx::Error> {
|
||||||
|
sqlx::query!("DELETE FROM reply_surb;")
|
||||||
|
.execute(&self.connection_pool)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
sqlx::query!("DELETE FROM reply_surb_sender;")
|
||||||
|
.execute(&self.connection_pool)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn insert_reply_surb(
|
||||||
|
&self,
|
||||||
|
stored_reply_surb: StoredReplySurb,
|
||||||
|
) -> Result<(), sqlx::Error> {
|
||||||
|
sqlx::query!(
|
||||||
|
r#"
|
||||||
|
INSERT INTO reply_surb(reply_surb_sender_id, reply_surb) VALUES (?, ?);
|
||||||
|
"#,
|
||||||
|
stored_reply_surb.reply_surb_sender_id,
|
||||||
|
stored_reply_surb.reply_surb
|
||||||
|
)
|
||||||
|
.execute(&self.connection_pool)
|
||||||
|
.await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn get_reply_surb_storage_metadata(
|
||||||
|
&self,
|
||||||
|
) -> Result<ReplySurbStorageMetadata, sqlx::Error> {
|
||||||
|
sqlx::query_as!(
|
||||||
|
ReplySurbStorageMetadata,
|
||||||
|
r#"
|
||||||
|
SELECT min_reply_surb_threshold as "min_reply_surb_threshold: u32", max_reply_surb_threshold as "max_reply_surb_threshold: u32" FROM reply_surb_storage_metadata;
|
||||||
|
"#,
|
||||||
|
)
|
||||||
|
.fetch_one(&self.connection_pool)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn insert_reply_surb_storage_metadata(
|
||||||
|
&self,
|
||||||
|
metadata: ReplySurbStorageMetadata,
|
||||||
|
) -> Result<(), sqlx::Error> {
|
||||||
|
sqlx::query!(r#"
|
||||||
|
INSERT INTO reply_surb_storage_metadata(min_reply_surb_threshold, max_reply_surb_threshold)
|
||||||
|
VALUES (?, ?);
|
||||||
|
"#,
|
||||||
|
metadata.min_reply_surb_threshold,
|
||||||
|
metadata.max_reply_surb_threshold,
|
||||||
|
).execute(&self.connection_pool).await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,336 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
pub use self::error::StorageError;
|
||||||
|
use crate::client::replies::reply_storage::backend::fs_backend::manager::StorageManager;
|
||||||
|
use crate::client::replies::reply_storage::backend::fs_backend::models::{
|
||||||
|
ReplySurbStorageMetadata, StoredReplyKey, StoredReplySurb, StoredSenderTag, StoredSurbSender,
|
||||||
|
};
|
||||||
|
use crate::client::replies::reply_storage::surb_storage::ReceivedReplySurbs;
|
||||||
|
use crate::client::replies::reply_storage::{
|
||||||
|
CombinedReplyStorage, ReceivedReplySurbsMap, ReplyStorageBackend, SentReplyKeys, UsedSenderTags,
|
||||||
|
};
|
||||||
|
use async_trait::async_trait;
|
||||||
|
use log::{error, info, warn};
|
||||||
|
use nymsphinx::anonymous_replies::requests::AnonymousSenderTag;
|
||||||
|
use std::fs;
|
||||||
|
use std::path::{Path, PathBuf};
|
||||||
|
use time::OffsetDateTime;
|
||||||
|
|
||||||
|
mod error;
|
||||||
|
mod manager;
|
||||||
|
mod models;
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct Backend {
|
||||||
|
temporary_old_path: Option<PathBuf>,
|
||||||
|
database_path: PathBuf,
|
||||||
|
manager: StorageManager,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Backend {
|
||||||
|
const OLD_EXTENSION: &'static str = "old";
|
||||||
|
|
||||||
|
pub async fn init<P: AsRef<Path>>(database_path: P) -> Result<Self, StorageError> {
|
||||||
|
let owned_path: PathBuf = database_path.as_ref().into();
|
||||||
|
if owned_path.file_name().is_none() {
|
||||||
|
return Err(StorageError::DatabasePathWithoutFilename {
|
||||||
|
provided_path: owned_path,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
let backend = Backend {
|
||||||
|
temporary_old_path: None,
|
||||||
|
database_path: owned_path,
|
||||||
|
manager: StorageManager::init(database_path, true).await?,
|
||||||
|
};
|
||||||
|
|
||||||
|
backend.manager.create_status_table().await?;
|
||||||
|
|
||||||
|
Ok(backend)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn try_load<P: AsRef<Path>>(database_path: P) -> Result<Self, StorageError> {
|
||||||
|
let owned_path: PathBuf = database_path.as_ref().into();
|
||||||
|
if owned_path.file_name().is_none() {
|
||||||
|
return Err(StorageError::DatabasePathWithoutFilename {
|
||||||
|
provided_path: owned_path,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
let manager = StorageManager::init(database_path, false).await?;
|
||||||
|
|
||||||
|
// the database flush wasn't fully finished and thus the data is in inconsistent state
|
||||||
|
// (we don't really know what's properly saved or what's not)
|
||||||
|
if manager.get_flush_status().await? {
|
||||||
|
return Err(StorageError::IncompleteDataFlush);
|
||||||
|
}
|
||||||
|
|
||||||
|
// the process has gone down without full graceful shutdown,
|
||||||
|
// meaning the database doesn't contain valid data anymore
|
||||||
|
// so we have to purge it
|
||||||
|
if manager.get_client_in_use_status().await? {
|
||||||
|
error!("the client hasn't undergone through graceful shutdown the last time it's gone down - we can't trust its reply surbs or stored encryption keys. They shall get purged");
|
||||||
|
manager.delete_all_reply_surb_data().await?;
|
||||||
|
manager.delete_all_reply_keys().await?;
|
||||||
|
}
|
||||||
|
|
||||||
|
let last_flush_timestamp = manager.get_previous_flush_timestamp().await?;
|
||||||
|
let last_flush = match OffsetDateTime::from_unix_timestamp(last_flush_timestamp) {
|
||||||
|
Ok(last_flush) => last_flush,
|
||||||
|
Err(err) => {
|
||||||
|
return Err(StorageError::CorruptedData {
|
||||||
|
details: format!("failed to parse stored timestamp - {err}"),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// in theory clients can use our reply surbs whenever they want, even a year in the future
|
||||||
|
// (assuming no key rotation has happened)
|
||||||
|
// but the way it's currently coded, everyone will purge old data
|
||||||
|
let since_last_flush = OffsetDateTime::now_utc() - last_flush;
|
||||||
|
if since_last_flush.whole_days() > 0 {
|
||||||
|
info!("it's been over {} days and {} hours since we last used our data store. our reply surbs are already outdated - we're going to purge them now.", since_last_flush.whole_days(), since_last_flush.whole_hours());
|
||||||
|
manager.delete_all_reply_surb_data().await?;
|
||||||
|
}
|
||||||
|
|
||||||
|
if since_last_flush.whole_days() > 1 {
|
||||||
|
info!("it's been over {} days and {} hours since we last used our data store. our reply keys are already outdated - we're going to purge them now.", since_last_flush.whole_days(), since_last_flush.whole_hours());
|
||||||
|
manager.delete_all_reply_keys().await?;
|
||||||
|
}
|
||||||
|
|
||||||
|
if since_last_flush.whole_days() > 2 {
|
||||||
|
info!("it's been over {} days and {} hours since we last used our data store. our used sender tags are already outdated - we're going to purge them now.", since_last_flush.whole_days(), since_last_flush.whole_hours());
|
||||||
|
manager.delete_all_tags().await?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(Backend {
|
||||||
|
temporary_old_path: None,
|
||||||
|
database_path: owned_path,
|
||||||
|
manager,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn close_pool(&mut self) {
|
||||||
|
self.manager.connection_pool.close().await;
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn rotate(&mut self) -> Result<(), StorageError> {
|
||||||
|
self.close_pool().await;
|
||||||
|
|
||||||
|
let new_extension = if let Some(existing_extension) =
|
||||||
|
self.database_path.extension().and_then(|ext| ext.to_str())
|
||||||
|
{
|
||||||
|
format!("{existing_extension}.{}", Self::OLD_EXTENSION)
|
||||||
|
} else {
|
||||||
|
Self::OLD_EXTENSION.to_string()
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut temp_old = self.database_path.clone();
|
||||||
|
temp_old.set_extension(new_extension);
|
||||||
|
|
||||||
|
fs::rename(&self.database_path, &temp_old)
|
||||||
|
.map_err(|err| StorageError::DatabaseRenameError { source: err })?;
|
||||||
|
self.manager = StorageManager::init(&self.database_path, true).await?;
|
||||||
|
self.manager.create_status_table().await?;
|
||||||
|
|
||||||
|
self.temporary_old_path = Some(temp_old);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn remove_old(&mut self) -> Result<(), StorageError> {
|
||||||
|
if let Some(old_path) = self.temporary_old_path.take() {
|
||||||
|
fs::remove_file(old_path)
|
||||||
|
.map_err(|err| StorageError::DatabaseOldFileRemoveError { source: err })
|
||||||
|
} else {
|
||||||
|
warn!("the old database file doesn't seem to exist!");
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn start_storage_flush(&self) -> Result<(), StorageError> {
|
||||||
|
Ok(self.manager.set_flush_status(true).await?)
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn end_storage_flush(&self) -> Result<(), StorageError> {
|
||||||
|
self.manager
|
||||||
|
.set_previous_flush_timestamp(OffsetDateTime::now_utc().unix_timestamp())
|
||||||
|
.await?;
|
||||||
|
Ok(self.manager.set_flush_status(false).await?)
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn start_client_use(&self) -> Result<(), StorageError> {
|
||||||
|
Ok(self.manager.set_client_in_use_status(true).await?)
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn stop_client_use(&self) -> Result<(), StorageError> {
|
||||||
|
Ok(self.manager.set_client_in_use_status(false).await?)
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn get_stored_tags(&self) -> Result<UsedSenderTags, StorageError> {
|
||||||
|
let stored = self.manager.get_tags().await?;
|
||||||
|
|
||||||
|
// stop at the first instance of corruption. if even a single entry is malformed,
|
||||||
|
// something weird has happened and we can't trust the rest of the data
|
||||||
|
let raw = stored
|
||||||
|
.into_iter()
|
||||||
|
.map(TryInto::try_into)
|
||||||
|
.collect::<Result<_, _>>()?;
|
||||||
|
|
||||||
|
Ok(UsedSenderTags::from_raw(raw))
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn dump_sender_tags(&self, tags: &UsedSenderTags) -> Result<(), StorageError> {
|
||||||
|
for map_ref in tags.as_raw_iter() {
|
||||||
|
let (recipient, tag) = map_ref.pair();
|
||||||
|
self.manager
|
||||||
|
.insert_tag(StoredSenderTag::new(*recipient, *tag))
|
||||||
|
.await?;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn get_stored_reply_keys(&self) -> Result<SentReplyKeys, StorageError> {
|
||||||
|
let stored = self.manager.get_reply_keys().await?;
|
||||||
|
|
||||||
|
// stop at the first instance of corruption. if even a single entry is malformed,
|
||||||
|
// something weird has happened and we can't trust the rest of the data
|
||||||
|
let raw = stored
|
||||||
|
.into_iter()
|
||||||
|
.map(TryInto::try_into)
|
||||||
|
.collect::<Result<_, _>>()?;
|
||||||
|
|
||||||
|
Ok(SentReplyKeys::from_raw(raw))
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn dump_sender_reply_keys(&self, reply_keys: &SentReplyKeys) -> Result<(), StorageError> {
|
||||||
|
for map_ref in reply_keys.as_raw_iter() {
|
||||||
|
let (digest, key) = map_ref.pair();
|
||||||
|
self.manager
|
||||||
|
.insert_reply_key(StoredReplyKey::new(*digest, *key))
|
||||||
|
.await?;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn get_stored_reply_surbs(&self) -> Result<ReceivedReplySurbsMap, StorageError> {
|
||||||
|
let surb_senders = self.manager.get_surb_senders().await?;
|
||||||
|
|
||||||
|
let metadata = self.get_reply_surb_storage_metadata().await?;
|
||||||
|
let mut received_surbs = Vec::with_capacity(surb_senders.len());
|
||||||
|
for sender in surb_senders {
|
||||||
|
let sender_id = sender.id;
|
||||||
|
let (sender_tag, surbs_last_received_at_timestamp): (AnonymousSenderTag, i64) =
|
||||||
|
sender.try_into()?;
|
||||||
|
let stored_surbs = self
|
||||||
|
.manager
|
||||||
|
.get_reply_surbs(sender_id)
|
||||||
|
.await?
|
||||||
|
.into_iter()
|
||||||
|
.map(|raw| raw.try_into())
|
||||||
|
.collect::<Result<_, _>>()?;
|
||||||
|
|
||||||
|
received_surbs.push((
|
||||||
|
sender_tag,
|
||||||
|
ReceivedReplySurbs::new_retrieved(stored_surbs, surbs_last_received_at_timestamp),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(ReceivedReplySurbsMap::from_raw(
|
||||||
|
metadata.min_reply_surb_threshold as usize,
|
||||||
|
metadata.max_reply_surb_threshold as usize,
|
||||||
|
received_surbs,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn dump_reply_surbs(
|
||||||
|
&self,
|
||||||
|
reply_surbs: &ReceivedReplySurbsMap,
|
||||||
|
) -> Result<(), StorageError> {
|
||||||
|
for map_ref in reply_surbs.as_raw_iter() {
|
||||||
|
let (tag, received_surbs) = map_ref.pair();
|
||||||
|
let sender_id = self
|
||||||
|
.manager
|
||||||
|
.insert_surb_sender(StoredSurbSender::new(
|
||||||
|
*tag,
|
||||||
|
received_surbs.surbs_last_received_at(),
|
||||||
|
))
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
for reply_surb in received_surbs.surbs_ref() {
|
||||||
|
self.manager
|
||||||
|
.insert_reply_surb(StoredReplySurb::new(sender_id, reply_surb))
|
||||||
|
.await?
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn get_reply_surb_storage_metadata(
|
||||||
|
&self,
|
||||||
|
) -> Result<ReplySurbStorageMetadata, StorageError> {
|
||||||
|
self.manager
|
||||||
|
.get_reply_surb_storage_metadata()
|
||||||
|
.await
|
||||||
|
.map_err(Into::into)
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn dump_reply_surb_storage_metadata(
|
||||||
|
&self,
|
||||||
|
reply_surbs: &ReceivedReplySurbsMap,
|
||||||
|
) -> Result<(), StorageError> {
|
||||||
|
self.manager
|
||||||
|
.insert_reply_surb_storage_metadata(ReplySurbStorageMetadata::new(
|
||||||
|
reply_surbs.min_surb_threshold(),
|
||||||
|
reply_surbs.max_surb_threshold(),
|
||||||
|
))
|
||||||
|
.await
|
||||||
|
.map_err(Into::into)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
impl ReplyStorageBackend for Backend {
|
||||||
|
type StorageError = error::StorageError;
|
||||||
|
|
||||||
|
async fn start_storage_session(&self) -> Result<(), Self::StorageError> {
|
||||||
|
self.start_client_use().await
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn flush_surb_storage(
|
||||||
|
&mut self,
|
||||||
|
storage: &CombinedReplyStorage,
|
||||||
|
) -> Result<(), Self::StorageError> {
|
||||||
|
// close all connections (there should be none! and rename the file to contain .old extension)
|
||||||
|
self.rotate().await?;
|
||||||
|
self.start_storage_flush().await?;
|
||||||
|
|
||||||
|
self.dump_sender_tags(storage.tags_storage_ref()).await?;
|
||||||
|
self.dump_sender_reply_keys(storage.key_storage_ref())
|
||||||
|
.await?;
|
||||||
|
let surbs_ref = storage.surbs_storage_ref();
|
||||||
|
self.dump_reply_surb_storage_metadata(surbs_ref).await?;
|
||||||
|
self.dump_reply_surbs(surbs_ref).await?;
|
||||||
|
|
||||||
|
self.remove_old()?;
|
||||||
|
self.end_storage_flush().await
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn init_fresh(&mut self, fresh: &CombinedReplyStorage) -> Result<(), Self::StorageError> {
|
||||||
|
// for now nothing more to do apart from dumping the metadata
|
||||||
|
self.dump_reply_surb_storage_metadata(fresh.surbs_storage_ref())
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn load_surb_storage(&self) -> Result<CombinedReplyStorage, Self::StorageError> {
|
||||||
|
let reply_keys = self.get_stored_reply_keys().await?;
|
||||||
|
let tags = self.get_stored_tags().await?;
|
||||||
|
let reply_surbs = self.get_stored_reply_surbs().await?;
|
||||||
|
|
||||||
|
Ok(CombinedReplyStorage::load(reply_keys, reply_surbs, tags))
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn stop_storage_session(self) -> Result<(), Self::StorageError> {
|
||||||
|
self.stop_client_use().await
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,185 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use crate::client::replies::reply_storage::backend::fs_backend::error::StorageError;
|
||||||
|
use crate::client::replies::reply_storage::key_storage::UsedReplyKey;
|
||||||
|
use crypto::generic_array::typenum::Unsigned;
|
||||||
|
use crypto::Digest;
|
||||||
|
use nymsphinx::addressing::clients::{Recipient, RecipientBytes};
|
||||||
|
use nymsphinx::anonymous_replies::encryption_key::EncryptionKeyDigest;
|
||||||
|
use nymsphinx::anonymous_replies::requests::{AnonymousSenderTag, SENDER_TAG_SIZE};
|
||||||
|
use nymsphinx::anonymous_replies::{ReplySurb, SurbEncryptionKey, SurbEncryptionKeySize};
|
||||||
|
use nymsphinx::params::ReplySurbKeyDigestAlgorithm;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub(crate) struct StoredSenderTag {
|
||||||
|
pub(crate) recipient: Vec<u8>,
|
||||||
|
pub(crate) tag: Vec<u8>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl StoredSenderTag {
|
||||||
|
pub(crate) fn new(recipient: RecipientBytes, tag: AnonymousSenderTag) -> StoredSenderTag {
|
||||||
|
StoredSenderTag {
|
||||||
|
recipient: recipient.to_vec(),
|
||||||
|
tag: tag.to_bytes().to_vec(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TryFrom<StoredSenderTag> for (RecipientBytes, AnonymousSenderTag) {
|
||||||
|
type Error = StorageError;
|
||||||
|
|
||||||
|
fn try_from(value: StoredSenderTag) -> Result<Self, Self::Error> {
|
||||||
|
let recipient_len = value.recipient.len();
|
||||||
|
let Ok(recipient_bytes) = value.recipient.try_into() else {
|
||||||
|
return Err(StorageError::CorruptedData {
|
||||||
|
details: format!(
|
||||||
|
"the retrieved recipient has length of {recipient_len} while {} was expected",
|
||||||
|
Recipient::LEN
|
||||||
|
),
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
let tag_len = value.tag.len();
|
||||||
|
let Ok(sender_tag_bytes) = value.tag.try_into() else {
|
||||||
|
return Err(StorageError::CorruptedData {
|
||||||
|
details: format!(
|
||||||
|
"the retrieved sender tag has length of {tag_len} while {} was expected",
|
||||||
|
SENDER_TAG_SIZE
|
||||||
|
),
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok((
|
||||||
|
recipient_bytes,
|
||||||
|
AnonymousSenderTag::from_bytes(sender_tag_bytes),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub(crate) struct StoredReplyKey {
|
||||||
|
pub(crate) key_digest: Vec<u8>,
|
||||||
|
pub(crate) reply_key: Vec<u8>,
|
||||||
|
pub(crate) sent_at_timestamp: i64,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl StoredReplyKey {
|
||||||
|
pub(crate) fn new(key_digest: EncryptionKeyDigest, reply_key: UsedReplyKey) -> StoredReplyKey {
|
||||||
|
StoredReplyKey {
|
||||||
|
key_digest: key_digest.to_vec(),
|
||||||
|
reply_key: (*reply_key).to_bytes(),
|
||||||
|
sent_at_timestamp: reply_key.sent_at_timestamp,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TryFrom<StoredReplyKey> for (EncryptionKeyDigest, UsedReplyKey) {
|
||||||
|
type Error = StorageError;
|
||||||
|
|
||||||
|
fn try_from(value: StoredReplyKey) -> Result<Self, Self::Error> {
|
||||||
|
let expected_reply_key_digest_size = ReplySurbKeyDigestAlgorithm::output_size();
|
||||||
|
let reply_key_digest_size = value.key_digest.len();
|
||||||
|
|
||||||
|
let Some(digest) = EncryptionKeyDigest::from_exact_iter(value.key_digest) else {
|
||||||
|
return Err(StorageError::CorruptedData {
|
||||||
|
details: format!(
|
||||||
|
"the reply surb digest has length of {reply_key_digest_size} while {expected_reply_key_digest_size} was expected",
|
||||||
|
),
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
let reply_key_len = value.reply_key.len();
|
||||||
|
let Ok(reply_key) = SurbEncryptionKey::try_from_bytes(&value.reply_key) else {
|
||||||
|
return Err(StorageError::CorruptedData {
|
||||||
|
details: format!(
|
||||||
|
"the reply key has length of {reply_key_len} while {} was expected",
|
||||||
|
SurbEncryptionKeySize::USIZE
|
||||||
|
),
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok((
|
||||||
|
digest,
|
||||||
|
UsedReplyKey::new(reply_key, value.sent_at_timestamp),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) struct StoredSurbSender {
|
||||||
|
pub(crate) id: i64,
|
||||||
|
pub(crate) tag: Vec<u8>,
|
||||||
|
pub(crate) last_sent_timestamp: i64,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl StoredSurbSender {
|
||||||
|
pub(crate) fn new(tag: AnonymousSenderTag, last_sent_timestamp: i64) -> Self {
|
||||||
|
StoredSurbSender {
|
||||||
|
// for the purposes of STORING data,
|
||||||
|
// we ignore that field anyway
|
||||||
|
id: 0,
|
||||||
|
tag: tag.to_bytes().to_vec(),
|
||||||
|
last_sent_timestamp,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TryFrom<StoredSurbSender> for (AnonymousSenderTag, i64) {
|
||||||
|
type Error = StorageError;
|
||||||
|
|
||||||
|
fn try_from(value: StoredSurbSender) -> Result<Self, Self::Error> {
|
||||||
|
let tag_len = value.tag.len();
|
||||||
|
let Ok(sender_tag_bytes) = value.tag.try_into() else {
|
||||||
|
return Err(StorageError::CorruptedData {
|
||||||
|
details: format!(
|
||||||
|
"the retrieved sender tag has length of {tag_len} while {} was expected",
|
||||||
|
SENDER_TAG_SIZE
|
||||||
|
),
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok((
|
||||||
|
AnonymousSenderTag::from_bytes(sender_tag_bytes),
|
||||||
|
value.last_sent_timestamp,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) struct StoredReplySurb {
|
||||||
|
pub(crate) reply_surb_sender_id: i64,
|
||||||
|
pub(crate) reply_surb: Vec<u8>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl StoredReplySurb {
|
||||||
|
pub(crate) fn new(reply_surb_sender_id: i64, reply_surb: &ReplySurb) -> Self {
|
||||||
|
StoredReplySurb {
|
||||||
|
reply_surb_sender_id,
|
||||||
|
reply_surb: reply_surb.to_bytes(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TryFrom<StoredReplySurb> for ReplySurb {
|
||||||
|
type Error = StorageError;
|
||||||
|
|
||||||
|
fn try_from(value: StoredReplySurb) -> Result<Self, Self::Error> {
|
||||||
|
ReplySurb::from_bytes(&value.reply_surb).map_err(|err| StorageError::CorruptedData {
|
||||||
|
details: format!("failed to recover the reply surb: {err}"),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Copy, Clone)]
|
||||||
|
pub(crate) struct ReplySurbStorageMetadata {
|
||||||
|
pub(crate) min_reply_surb_threshold: u32,
|
||||||
|
pub(crate) max_reply_surb_threshold: u32,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ReplySurbStorageMetadata {
|
||||||
|
pub(crate) fn new(min_reply_surb_threshold: usize, max_reply_surb_threshold: usize) -> Self {
|
||||||
|
Self {
|
||||||
|
min_reply_surb_threshold: min_reply_surb_threshold as u32,
|
||||||
|
max_reply_surb_threshold: max_reply_surb_threshold as u32,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,78 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use crate::client::replies::reply_storage::CombinedReplyStorage;
|
||||||
|
use async_trait::async_trait;
|
||||||
|
use std::error::Error;
|
||||||
|
use thiserror::Error;
|
||||||
|
|
||||||
|
#[cfg(target_arch = "wasm32")]
|
||||||
|
pub mod browser_backend;
|
||||||
|
|
||||||
|
#[cfg(all(not(target_arch = "wasm32"), feature = "fs-surb-storage"))]
|
||||||
|
pub mod fs_backend;
|
||||||
|
|
||||||
|
// #[cfg(all(test, feature = "std"))]
|
||||||
|
// third case: node with actual filesystem
|
||||||
|
|
||||||
|
#[derive(Debug, Error)]
|
||||||
|
#[error("no information provided")]
|
||||||
|
pub struct UndefinedError;
|
||||||
|
|
||||||
|
pub struct Empty {
|
||||||
|
// we need to keep 'basic' metadata here to "load" the CombinedReplyStorage
|
||||||
|
min_surb_threshold: usize,
|
||||||
|
max_surb_threshold: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
impl ReplyStorageBackend for Empty {
|
||||||
|
type StorageError = UndefinedError;
|
||||||
|
|
||||||
|
async fn flush_surb_storage(
|
||||||
|
&mut self,
|
||||||
|
_storage: &CombinedReplyStorage,
|
||||||
|
) -> Result<(), Self::StorageError> {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn init_fresh(
|
||||||
|
&mut self,
|
||||||
|
_fresh: &CombinedReplyStorage,
|
||||||
|
) -> Result<(), Self::StorageError> {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn load_surb_storage(&self) -> Result<CombinedReplyStorage, Self::StorageError> {
|
||||||
|
Ok(CombinedReplyStorage::new(
|
||||||
|
self.min_surb_threshold,
|
||||||
|
self.max_surb_threshold,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
pub trait ReplyStorageBackend: Sized {
|
||||||
|
type StorageError: Error + 'static;
|
||||||
|
|
||||||
|
async fn start_storage_session(&self) -> Result<(), Self::StorageError> {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
// reply keys and surbs would need additional field set when data is loaded
|
||||||
|
// so if there's some failure, we'd trash it all
|
||||||
|
async fn flush_surb_storage(
|
||||||
|
&mut self,
|
||||||
|
storage: &CombinedReplyStorage,
|
||||||
|
) -> Result<(), Self::StorageError>;
|
||||||
|
|
||||||
|
/// The purpose of this call is to save any metadata that might be present.
|
||||||
|
/// (such as surb thresholds)
|
||||||
|
async fn init_fresh(&mut self, fresh: &CombinedReplyStorage) -> Result<(), Self::StorageError>;
|
||||||
|
|
||||||
|
async fn load_surb_storage(&self) -> Result<CombinedReplyStorage, Self::StorageError>;
|
||||||
|
|
||||||
|
async fn stop_storage_session(self) -> Result<(), Self::StorageError> {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,60 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use crate::client::replies::reply_storage::{ReceivedReplySurbsMap, SentReplyKeys, UsedSenderTags};
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct CombinedReplyStorage {
|
||||||
|
sent_reply_keys: SentReplyKeys,
|
||||||
|
received_reply_surbs: ReceivedReplySurbsMap,
|
||||||
|
used_tags: UsedSenderTags,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl CombinedReplyStorage {
|
||||||
|
pub fn new(min_surb_threshold: usize, max_surb_threshold: usize) -> CombinedReplyStorage {
|
||||||
|
CombinedReplyStorage {
|
||||||
|
sent_reply_keys: SentReplyKeys::new(),
|
||||||
|
received_reply_surbs: ReceivedReplySurbsMap::new(
|
||||||
|
min_surb_threshold,
|
||||||
|
max_surb_threshold,
|
||||||
|
),
|
||||||
|
used_tags: UsedSenderTags::new(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn load(
|
||||||
|
sent_reply_keys: SentReplyKeys,
|
||||||
|
received_reply_surbs: ReceivedReplySurbsMap,
|
||||||
|
used_tags: UsedSenderTags,
|
||||||
|
) -> Self {
|
||||||
|
CombinedReplyStorage {
|
||||||
|
sent_reply_keys,
|
||||||
|
received_reply_surbs,
|
||||||
|
used_tags,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn key_storage(&self) -> SentReplyKeys {
|
||||||
|
self.sent_reply_keys.clone()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn surbs_storage(&self) -> ReceivedReplySurbsMap {
|
||||||
|
self.received_reply_surbs.clone()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn tags_storage(&self) -> UsedSenderTags {
|
||||||
|
self.used_tags.clone()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn key_storage_ref(&self) -> &SentReplyKeys {
|
||||||
|
&self.sent_reply_keys
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn surbs_storage_ref(&self) -> &ReceivedReplySurbsMap {
|
||||||
|
&self.received_reply_surbs
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn tags_storage_ref(&self) -> &UsedSenderTags {
|
||||||
|
&self.used_tags
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,86 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use dashmap::iter::Iter;
|
||||||
|
use dashmap::DashMap;
|
||||||
|
use nymsphinx::anonymous_replies::encryption_key::EncryptionKeyDigest;
|
||||||
|
use nymsphinx::anonymous_replies::SurbEncryptionKey;
|
||||||
|
use std::ops::Deref;
|
||||||
|
use std::sync::Arc;
|
||||||
|
use time::OffsetDateTime;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct SentReplyKeys {
|
||||||
|
inner: Arc<SentReplyKeysInner>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
struct SentReplyKeysInner {
|
||||||
|
data: DashMap<EncryptionKeyDigest, UsedReplyKey>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SentReplyKeys {
|
||||||
|
pub(crate) fn new() -> SentReplyKeys {
|
||||||
|
SentReplyKeys {
|
||||||
|
inner: Arc::new(SentReplyKeysInner {
|
||||||
|
data: DashMap::new(),
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(all(not(target_arch = "wasm32"), feature = "fs-surb-storage"))]
|
||||||
|
pub(crate) fn from_raw(raw: Vec<(EncryptionKeyDigest, UsedReplyKey)>) -> SentReplyKeys {
|
||||||
|
SentReplyKeys {
|
||||||
|
inner: Arc::new(SentReplyKeysInner {
|
||||||
|
data: raw.into_iter().collect(),
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn as_raw_iter(&self) -> Iter<'_, EncryptionKeyDigest, UsedReplyKey> {
|
||||||
|
self.inner.data.iter()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn insert_multiple(&self, keys: Vec<SurbEncryptionKey>) {
|
||||||
|
let now = OffsetDateTime::now_utc().unix_timestamp();
|
||||||
|
for key in keys {
|
||||||
|
self.insert(UsedReplyKey::new(key, now))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn insert(&self, key: UsedReplyKey) {
|
||||||
|
self.inner.data.insert(key.compute_digest(), key);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn try_pop(&self, digest: EncryptionKeyDigest) -> Option<UsedReplyKey> {
|
||||||
|
self.inner.data.remove(&digest).map(|(_k, v)| v)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn remove(&self, digest: EncryptionKeyDigest) {
|
||||||
|
self.inner.data.remove(&digest);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Copy, Clone)]
|
||||||
|
pub(crate) struct UsedReplyKey {
|
||||||
|
key: SurbEncryptionKey,
|
||||||
|
// the purpose of this field is to perform invalidation at relatively very long intervals
|
||||||
|
pub(crate) sent_at_timestamp: i64,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl UsedReplyKey {
|
||||||
|
pub(crate) fn new(key: SurbEncryptionKey, sent_at_timestamp: i64) -> Self {
|
||||||
|
UsedReplyKey {
|
||||||
|
key,
|
||||||
|
sent_at_timestamp,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Deref for UsedReplyKey {
|
||||||
|
type Target = SurbEncryptionKey;
|
||||||
|
|
||||||
|
fn deref(&self) -> &Self::Target {
|
||||||
|
&self.key
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,64 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
pub use crate::client::replies::reply_storage::combined::CombinedReplyStorage;
|
||||||
|
pub use crate::client::replies::reply_storage::key_storage::SentReplyKeys;
|
||||||
|
pub use crate::client::replies::reply_storage::surb_storage::ReceivedReplySurbsMap;
|
||||||
|
pub use crate::client::replies::reply_storage::tag_storage::UsedSenderTags;
|
||||||
|
pub use backend::*;
|
||||||
|
|
||||||
|
mod backend;
|
||||||
|
mod combined;
|
||||||
|
mod key_storage;
|
||||||
|
mod surb_storage;
|
||||||
|
mod tag_storage;
|
||||||
|
|
||||||
|
// only really exists to get information about shutdown and save data to the backing storage
|
||||||
|
pub struct PersistentReplyStorage<T = backend::Empty>
|
||||||
|
where
|
||||||
|
T: ReplyStorageBackend,
|
||||||
|
{
|
||||||
|
backend: T,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> PersistentReplyStorage<T>
|
||||||
|
where
|
||||||
|
T: ReplyStorageBackend + Send + Sync,
|
||||||
|
{
|
||||||
|
pub fn new(backend: T) -> Self {
|
||||||
|
PersistentReplyStorage { backend }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn load_state_from_backend(&self) -> Result<CombinedReplyStorage, T::StorageError> {
|
||||||
|
self.backend.load_surb_storage().await
|
||||||
|
}
|
||||||
|
|
||||||
|
// this will have to get enabled after merging develop
|
||||||
|
pub async fn flush_on_shutdown(
|
||||||
|
mut self,
|
||||||
|
mem_state: CombinedReplyStorage,
|
||||||
|
mut shutdown: task::ShutdownListener,
|
||||||
|
) {
|
||||||
|
use log::{debug, error, info, warn};
|
||||||
|
|
||||||
|
debug!("Started PersistentReplyStorage");
|
||||||
|
if let Err(err) = self.backend.start_storage_session().await {
|
||||||
|
error!("failed to start the storage session - {err}");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
shutdown.recv().await;
|
||||||
|
|
||||||
|
info!("PersistentReplyStorage is flushing all reply-related data to underlying storage");
|
||||||
|
warn!("you MUST NOT forcefully shutdown now or you risk data corruption!");
|
||||||
|
if let Err(err) = self.backend.flush_surb_storage(&mem_state).await {
|
||||||
|
error!("failed to flush our reply-related data to the persistent storage: {err}")
|
||||||
|
} else {
|
||||||
|
info!("Data flush is complete")
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Err(err) = self.backend.stop_storage_session().await {
|
||||||
|
error!("failed to properly stop the storage session - {err}. We might not be able to smoothly restore it")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,278 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use dashmap::iter::Iter;
|
||||||
|
use dashmap::DashMap;
|
||||||
|
use log::trace;
|
||||||
|
use nymsphinx::anonymous_replies::requests::AnonymousSenderTag;
|
||||||
|
use nymsphinx::anonymous_replies::ReplySurb;
|
||||||
|
use std::collections::VecDeque;
|
||||||
|
use std::sync::atomic::{AtomicUsize, Ordering};
|
||||||
|
use std::sync::Arc;
|
||||||
|
use time::OffsetDateTime;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct ReceivedReplySurbsMap {
|
||||||
|
inner: Arc<ReceivedReplySurbsMapInner>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
struct ReceivedReplySurbsMapInner {
|
||||||
|
data: DashMap<AnonymousSenderTag, ReceivedReplySurbs>,
|
||||||
|
|
||||||
|
// the minimum amount of surbs that have to be kept in storage for requests for more surbs
|
||||||
|
min_surb_threshold: AtomicUsize,
|
||||||
|
|
||||||
|
// the maximum amount of surbs that we want to keep in storage so that we don't over-request them
|
||||||
|
max_surb_threshold: AtomicUsize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ReceivedReplySurbsMap {
|
||||||
|
pub(crate) fn new(
|
||||||
|
min_surb_threshold: usize,
|
||||||
|
max_surb_threshold: usize,
|
||||||
|
) -> ReceivedReplySurbsMap {
|
||||||
|
ReceivedReplySurbsMap {
|
||||||
|
inner: Arc::new(ReceivedReplySurbsMapInner {
|
||||||
|
data: DashMap::new(),
|
||||||
|
min_surb_threshold: AtomicUsize::new(min_surb_threshold),
|
||||||
|
max_surb_threshold: AtomicUsize::new(max_surb_threshold),
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(all(not(target_arch = "wasm32"), feature = "fs-surb-storage"))]
|
||||||
|
pub(crate) fn from_raw(
|
||||||
|
min_surb_threshold: usize,
|
||||||
|
max_surb_threshold: usize,
|
||||||
|
raw: Vec<(AnonymousSenderTag, ReceivedReplySurbs)>,
|
||||||
|
) -> ReceivedReplySurbsMap {
|
||||||
|
ReceivedReplySurbsMap {
|
||||||
|
inner: Arc::new(ReceivedReplySurbsMapInner {
|
||||||
|
data: raw.into_iter().collect(),
|
||||||
|
min_surb_threshold: AtomicUsize::new(min_surb_threshold),
|
||||||
|
max_surb_threshold: AtomicUsize::new(max_surb_threshold),
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn as_raw_iter(&self) -> Iter<'_, AnonymousSenderTag, ReceivedReplySurbs> {
|
||||||
|
self.inner.data.iter()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn remove(&self, target: &AnonymousSenderTag) {
|
||||||
|
self.inner.data.remove(target);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn reset_surbs_last_received_at(&self, target: &AnonymousSenderTag) {
|
||||||
|
if let Some(mut entry) = self.inner.data.get_mut(target) {
|
||||||
|
entry.surbs_last_received_at_timestamp = OffsetDateTime::now_utc().unix_timestamp();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn surbs_last_received_at(&self, target: &AnonymousSenderTag) -> Option<i64> {
|
||||||
|
self.inner
|
||||||
|
.data
|
||||||
|
.get(target)
|
||||||
|
.map(|e| e.surbs_last_received_at())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn pending_reception(&self, target: &AnonymousSenderTag) -> u32 {
|
||||||
|
self.inner
|
||||||
|
.data
|
||||||
|
.get(target)
|
||||||
|
.map(|e| e.pending_reception())
|
||||||
|
.unwrap_or_default()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn increment_pending_reception(
|
||||||
|
&self,
|
||||||
|
target: &AnonymousSenderTag,
|
||||||
|
amount: u32,
|
||||||
|
) -> Option<u32> {
|
||||||
|
self.inner
|
||||||
|
.data
|
||||||
|
.get_mut(target)
|
||||||
|
.map(|mut e| e.increment_pending_reception(amount))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn decrement_pending_reception(
|
||||||
|
&self,
|
||||||
|
target: &AnonymousSenderTag,
|
||||||
|
amount: u32,
|
||||||
|
) -> Option<u32> {
|
||||||
|
self.inner
|
||||||
|
.data
|
||||||
|
.get_mut(target)
|
||||||
|
.map(|mut e| e.decrement_pending_reception(amount))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn reset_pending_reception(&self, target: &AnonymousSenderTag) {
|
||||||
|
if let Some(mut e) = self.inner.data.get_mut(target) {
|
||||||
|
e.reset_pending_reception()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn min_surb_threshold(&self) -> usize {
|
||||||
|
self.inner.min_surb_threshold.load(Ordering::Relaxed)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn max_surb_threshold(&self) -> usize {
|
||||||
|
self.inner.max_surb_threshold.load(Ordering::Relaxed)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn available_surbs(&self, target: &AnonymousSenderTag) -> usize {
|
||||||
|
self.inner
|
||||||
|
.data
|
||||||
|
.get(target)
|
||||||
|
.map(|entry| entry.items_left())
|
||||||
|
.unwrap_or_default()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn contains_surbs_for(&self, target: &AnonymousSenderTag) -> bool {
|
||||||
|
self.inner.data.contains_key(target)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn get_reply_surbs(
|
||||||
|
&self,
|
||||||
|
target: &AnonymousSenderTag,
|
||||||
|
amount: usize,
|
||||||
|
) -> (Option<Vec<ReplySurb>>, usize) {
|
||||||
|
if let Some(mut entry) = self.inner.data.get_mut(target) {
|
||||||
|
let surbs_left = entry.items_left();
|
||||||
|
if surbs_left < self.min_surb_threshold() + amount {
|
||||||
|
(None, surbs_left)
|
||||||
|
} else {
|
||||||
|
entry.get_reply_surbs(amount)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
(None, 0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn get_reply_surb_ignoring_threshold(
|
||||||
|
&self,
|
||||||
|
target: &AnonymousSenderTag,
|
||||||
|
) -> Option<(Option<ReplySurb>, usize)> {
|
||||||
|
self.inner
|
||||||
|
.data
|
||||||
|
.get_mut(target)
|
||||||
|
.map(|mut s| s.get_reply_surb())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn get_reply_surb(
|
||||||
|
&self,
|
||||||
|
target: &AnonymousSenderTag,
|
||||||
|
) -> Option<(Option<ReplySurb>, usize)> {
|
||||||
|
self.inner.data.get_mut(target).map(|mut entry| {
|
||||||
|
let surbs_left = entry.items_left();
|
||||||
|
if surbs_left < self.min_surb_threshold() {
|
||||||
|
(None, surbs_left)
|
||||||
|
} else {
|
||||||
|
entry.get_reply_surb()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn insert_surbs<I: IntoIterator<Item = ReplySurb>>(
|
||||||
|
&self,
|
||||||
|
target: &AnonymousSenderTag,
|
||||||
|
surbs: I,
|
||||||
|
) {
|
||||||
|
if let Some(mut existing_data) = self.inner.data.get_mut(target) {
|
||||||
|
existing_data.insert_reply_surbs(surbs)
|
||||||
|
} else {
|
||||||
|
let new_entry = ReceivedReplySurbs::new(surbs.into_iter().collect());
|
||||||
|
self.inner.data.insert(*target, new_entry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub(crate) struct ReceivedReplySurbs {
|
||||||
|
// in the future we'd probably want to put extra data here to indicate when the SURBs got received
|
||||||
|
// so we could invalidate entries from the previous key rotations
|
||||||
|
data: VecDeque<ReplySurb>,
|
||||||
|
|
||||||
|
pending_reception: u32,
|
||||||
|
surbs_last_received_at_timestamp: i64,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ReceivedReplySurbs {
|
||||||
|
fn new(initial_surbs: VecDeque<ReplySurb>) -> Self {
|
||||||
|
ReceivedReplySurbs {
|
||||||
|
data: initial_surbs,
|
||||||
|
pending_reception: 0,
|
||||||
|
surbs_last_received_at_timestamp: OffsetDateTime::now_utc().unix_timestamp(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(all(not(target_arch = "wasm32"), feature = "fs-surb-storage"))]
|
||||||
|
pub(crate) fn new_retrieved(
|
||||||
|
surbs: Vec<ReplySurb>,
|
||||||
|
surbs_last_received_at_timestamp: i64,
|
||||||
|
) -> ReceivedReplySurbs {
|
||||||
|
ReceivedReplySurbs {
|
||||||
|
data: surbs.into(),
|
||||||
|
pending_reception: 0,
|
||||||
|
surbs_last_received_at_timestamp,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(all(not(target_arch = "wasm32"), feature = "fs-surb-storage"))]
|
||||||
|
pub(crate) fn surbs_ref(&self) -> &VecDeque<ReplySurb> {
|
||||||
|
&self.data
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn surbs_last_received_at(&self) -> i64 {
|
||||||
|
self.surbs_last_received_at_timestamp
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn pending_reception(&self) -> u32 {
|
||||||
|
self.pending_reception
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn increment_pending_reception(&mut self, amount: u32) -> u32 {
|
||||||
|
self.pending_reception += amount;
|
||||||
|
self.pending_reception
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn decrement_pending_reception(&mut self, amount: u32) -> u32 {
|
||||||
|
self.pending_reception = self.pending_reception.saturating_sub(amount);
|
||||||
|
self.pending_reception
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn reset_pending_reception(&mut self) {
|
||||||
|
self.pending_reception = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn get_reply_surbs(&mut self, amount: usize) -> (Option<Vec<ReplySurb>>, usize) {
|
||||||
|
if self.items_left() < amount {
|
||||||
|
(None, self.items_left())
|
||||||
|
} else {
|
||||||
|
let surbs = self.data.drain(..amount).collect();
|
||||||
|
(Some(surbs), self.items_left())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn get_reply_surb(&mut self) -> (Option<ReplySurb>, usize) {
|
||||||
|
(self.pop_surb(), self.items_left())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn pop_surb(&mut self) -> Option<ReplySurb> {
|
||||||
|
self.data.pop_front()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn items_left(&self) -> usize {
|
||||||
|
self.data.len()
|
||||||
|
}
|
||||||
|
|
||||||
|
// realistically we're always going to be getting multiple surbs at once
|
||||||
|
pub(crate) fn insert_reply_surbs<I: IntoIterator<Item = ReplySurb>>(&mut self, surbs: I) {
|
||||||
|
let mut v = surbs.into_iter().collect::<VecDeque<_>>();
|
||||||
|
trace!("storing {} surbs in the storage", v.len());
|
||||||
|
self.data.append(&mut v);
|
||||||
|
self.surbs_last_received_at_timestamp = OffsetDateTime::now_utc().unix_timestamp();
|
||||||
|
trace!("we now have {} surbs!", self.data.len());
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,59 @@
|
|||||||
|
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use dashmap::DashMap;
|
||||||
|
use nymsphinx::addressing::clients::{Recipient, RecipientBytes};
|
||||||
|
use nymsphinx::anonymous_replies::requests::AnonymousSenderTag;
|
||||||
|
use std::sync::Arc;
|
||||||
|
|
||||||
|
#[cfg(all(not(target_arch = "wasm32"), feature = "fs-surb-storage"))]
|
||||||
|
use dashmap::iter::Iter;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct UsedSenderTags {
|
||||||
|
inner: Arc<UsedSenderTagsInner>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
struct UsedSenderTagsInner {
|
||||||
|
data: DashMap<RecipientBytes, AnonymousSenderTag>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl UsedSenderTags {
|
||||||
|
pub(crate) fn new() -> UsedSenderTags {
|
||||||
|
UsedSenderTags {
|
||||||
|
inner: Arc::new(UsedSenderTagsInner {
|
||||||
|
data: DashMap::new(),
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(all(not(target_arch = "wasm32"), feature = "fs-surb-storage"))]
|
||||||
|
pub(crate) fn from_raw(raw: Vec<(RecipientBytes, AnonymousSenderTag)>) -> UsedSenderTags {
|
||||||
|
UsedSenderTags {
|
||||||
|
inner: Arc::new(UsedSenderTagsInner {
|
||||||
|
data: raw.into_iter().collect(),
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(all(not(target_arch = "wasm32"), feature = "fs-surb-storage"))]
|
||||||
|
pub(crate) fn as_raw_iter(&self) -> Iter<'_, RecipientBytes, AnonymousSenderTag> {
|
||||||
|
self.inner.data.iter()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn insert_new(&self, recipient: &Recipient, tag: AnonymousSenderTag) {
|
||||||
|
self.inner.data.insert(recipient.to_bytes(), tag);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn try_get_existing(&self, recipient: &Recipient) -> Option<AnonymousSenderTag> {
|
||||||
|
self.inner
|
||||||
|
.data
|
||||||
|
.get(&recipient.to_bytes())
|
||||||
|
.map(|r| *r.value())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn exists(&self, recipient: &Recipient) -> bool {
|
||||||
|
self.inner.data.contains_key(&recipient.to_bytes())
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,97 +0,0 @@
|
|||||||
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
|
||||||
|
|
||||||
use crypto::generic_array::typenum::Unsigned;
|
|
||||||
use log::*;
|
|
||||||
use nymsphinx::anonymous_replies::{
|
|
||||||
encryption_key::EncryptionKeyDigest, SurbEncryptionKey, SurbEncryptionKeySize,
|
|
||||||
};
|
|
||||||
use std::path::Path;
|
|
||||||
|
|
||||||
#[derive(Debug, thiserror::Error)]
|
|
||||||
pub enum ReplyKeyStorageError {
|
|
||||||
#[error("DB Read Error: {0}")]
|
|
||||||
DbReadError(sled::Error),
|
|
||||||
#[error("DB Write Error: {0}")]
|
|
||||||
DbWriteError(sled::Error),
|
|
||||||
#[error("DB Open Error: {0}")]
|
|
||||||
DbOpenError(sled::Error),
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Permanent storage for keys in all sent [`ReplySURB`]
|
|
||||||
///
|
|
||||||
/// Each sent out [`ReplySURB`] has a new key associated with it that is going to be used for
|
|
||||||
/// payload encryption. In order to -decrypt whatever reply we receive, we need to know which
|
|
||||||
/// key to use for that purpose. We do it based on received `H(t)` which has to be included
|
|
||||||
/// with each reply.
|
|
||||||
/// Moreover, there is no restriction when the [`ReplySURB`] might get used so we need to
|
|
||||||
/// have a permanent storage for all the keys that we might ever see in the future.
|
|
||||||
#[derive(Debug, Clone)]
|
|
||||||
pub struct ReplyKeyStorage {
|
|
||||||
db: sled::Db,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ReplyKeyStorage {
|
|
||||||
pub fn load<P: AsRef<Path>>(path: P) -> Result<Self, ReplyKeyStorageError> {
|
|
||||||
let db = match sled::open(path) {
|
|
||||||
Err(e) => return Err(ReplyKeyStorageError::DbOpenError(e)),
|
|
||||||
Ok(db) => db,
|
|
||||||
};
|
|
||||||
|
|
||||||
Ok(ReplyKeyStorage { db })
|
|
||||||
}
|
|
||||||
|
|
||||||
fn read_encryption_key(&self, raw_key: sled::IVec) -> SurbEncryptionKey {
|
|
||||||
let key_bytes_ref = raw_key.as_ref();
|
|
||||||
// if this fails it means we have some database corruption and we
|
|
||||||
// absolutely can't continue
|
|
||||||
|
|
||||||
if key_bytes_ref.len() != SurbEncryptionKeySize::USIZE {
|
|
||||||
error!("REPLY KEY STORAGE DATA CORRUPTION - ENCRYPTION KEY HAS INVALID LENGTH");
|
|
||||||
panic!("REPLY KEY STORAGE DATA CORRUPTION - ENCRYPTION KEY HAS INVALID LENGTH");
|
|
||||||
}
|
|
||||||
|
|
||||||
// this can only fail if the bytes have invalid length but we already asserted it
|
|
||||||
SurbEncryptionKey::try_from_bytes(key_bytes_ref).unwrap()
|
|
||||||
}
|
|
||||||
|
|
||||||
// TOOD: perhaps we could also store some part of original message here too?
|
|
||||||
pub fn insert_encryption_key(
|
|
||||||
&mut self,
|
|
||||||
encryption_key: SurbEncryptionKey,
|
|
||||||
) -> Result<(), ReplyKeyStorageError> {
|
|
||||||
let digest = encryption_key.compute_digest();
|
|
||||||
|
|
||||||
let insertion_result = match self.db.insert(digest, encryption_key.to_bytes()) {
|
|
||||||
Err(e) => Err(ReplyKeyStorageError::DbWriteError(e)),
|
|
||||||
Ok(existing_key) => {
|
|
||||||
if existing_key.is_some() {
|
|
||||||
panic!("HASH COLLISION DETECTED")
|
|
||||||
};
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// TODO: perhaps we could implement some batching mechanism to avoid frequent flushes?
|
|
||||||
self.db.flush().unwrap();
|
|
||||||
insertion_result
|
|
||||||
}
|
|
||||||
|
|
||||||
// Once we use key once, we do not expect to use it again
|
|
||||||
pub fn get_and_remove_encryption_key(
|
|
||||||
&self,
|
|
||||||
key_digest: EncryptionKeyDigest,
|
|
||||||
) -> Result<Option<SurbEncryptionKey>, ReplyKeyStorageError> {
|
|
||||||
let removal_result = match self.db.remove(key_digest) {
|
|
||||||
Err(e) => Err(ReplyKeyStorageError::DbReadError(e)),
|
|
||||||
Ok(existing_key) => {
|
|
||||||
Ok(existing_key.map(|existing_key| self.read_encryption_key(existing_key)))
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// TODO: not sure how to feel about flushing it every single time here...
|
|
||||||
// same with insertion
|
|
||||||
self.db.flush().unwrap();
|
|
||||||
removal_result
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,7 +1,8 @@
|
|||||||
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
// Copyright 2021-2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use crate::spawn_future;
|
use crate::spawn_future;
|
||||||
|
use futures::StreamExt;
|
||||||
use log::*;
|
use log::*;
|
||||||
use nymsphinx::addressing::clients::Recipient;
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
use nymsphinx::params::DEFAULT_NUM_MIX_HOPS;
|
use nymsphinx::params::DEFAULT_NUM_MIX_HOPS;
|
||||||
@@ -9,10 +10,9 @@ use rand::seq::SliceRandom;
|
|||||||
use rand::thread_rng;
|
use rand::thread_rng;
|
||||||
use std::ops::Deref;
|
use std::ops::Deref;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::time;
|
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
use tokio::sync::{RwLock, RwLockReadGuard};
|
use tokio::sync::{RwLock, RwLockReadGuard};
|
||||||
use topology::{nym_topology_from_detailed, NymTopology};
|
use topology::{nym_topology_from_detailed, NymTopology, NymTopologyError};
|
||||||
use url::Url;
|
use url::Url;
|
||||||
|
|
||||||
// I'm extremely curious why compiler NEVER complained about lack of Debug here before
|
// I'm extremely curious why compiler NEVER complained about lack of Debug here before
|
||||||
@@ -54,18 +54,36 @@ impl<'a> TopologyReadPermit<'a> {
|
|||||||
&'a self,
|
&'a self,
|
||||||
ack_recipient: &Recipient,
|
ack_recipient: &Recipient,
|
||||||
packet_recipient: Option<&Recipient>,
|
packet_recipient: Option<&Recipient>,
|
||||||
) -> Option<&'a NymTopology> {
|
) -> Result<&'a NymTopology, NymTopologyError> {
|
||||||
// Note: implicit deref with Deref for TopologyReadPermit is happening here
|
// 1. Have we managed to get anything from the refresher, i.e. have the nym-api queries gone through?
|
||||||
let topology_ref_option = self.permit.as_ref();
|
let topology = self
|
||||||
topology_ref_option.as_ref().filter(|topology_ref| {
|
.permit
|
||||||
!(!topology_ref.can_construct_path_through(DEFAULT_NUM_MIX_HOPS)
|
.as_ref()
|
||||||
|| !topology_ref.gateway_exists(ack_recipient.gateway())
|
.as_ref()
|
||||||
|| if let Some(packet_recipient) = packet_recipient {
|
.ok_or(NymTopologyError::EmptyNetworkTopology)?;
|
||||||
!topology_ref.gateway_exists(packet_recipient.gateway())
|
|
||||||
} else {
|
// 2. does it have any mixnode at all?
|
||||||
false
|
// 3. does it have any gateways at all?
|
||||||
})
|
// 4. does it have a mixnode on each layer?
|
||||||
})
|
topology.ensure_can_construct_path_through(DEFAULT_NUM_MIX_HOPS)?;
|
||||||
|
|
||||||
|
// 5. does it contain OUR gateway (so that we could create an ack packet)?
|
||||||
|
if !topology.gateway_exists(ack_recipient.gateway()) {
|
||||||
|
return Err(NymTopologyError::NonExistentGatewayError {
|
||||||
|
identity_key: ack_recipient.gateway().to_base58_string(),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 6. for our target recipient, does it contain THEIR gateway (so that we could create
|
||||||
|
if let Some(recipient) = packet_recipient {
|
||||||
|
if !topology.gateway_exists(recipient.gateway()) {
|
||||||
|
return Err(NymTopologyError::NonExistentGatewayError {
|
||||||
|
identity_key: recipient.gateway().to_base58_string(),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(topology)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -103,10 +121,10 @@ impl TopologyAccessor {
|
|||||||
|
|
||||||
// only used by the client at startup to get a slightly more reasonable error message
|
// only used by the client at startup to get a slightly more reasonable error message
|
||||||
// (currently displays as unused because health checker is disabled due to required changes)
|
// (currently displays as unused because health checker is disabled due to required changes)
|
||||||
pub async fn is_routable(&self) -> bool {
|
pub async fn ensure_is_routable(&self) -> Result<(), NymTopologyError> {
|
||||||
match &self.inner.read().await.0 {
|
match &self.inner.read().await.0 {
|
||||||
None => false,
|
None => Err(NymTopologyError::EmptyNetworkTopology),
|
||||||
Some(ref topology) => topology.can_construct_path_through(DEFAULT_NUM_MIX_HOPS),
|
Some(ref topology) => topology.ensure_can_construct_path_through(DEFAULT_NUM_MIX_HOPS),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -119,14 +137,14 @@ impl Default for TopologyAccessor {
|
|||||||
|
|
||||||
pub struct TopologyRefresherConfig {
|
pub struct TopologyRefresherConfig {
|
||||||
validator_api_urls: Vec<Url>,
|
validator_api_urls: Vec<Url>,
|
||||||
refresh_rate: time::Duration,
|
refresh_rate: Duration,
|
||||||
client_version: String,
|
client_version: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TopologyRefresherConfig {
|
impl TopologyRefresherConfig {
|
||||||
pub fn new(
|
pub fn new(
|
||||||
validator_api_urls: Vec<Url>,
|
validator_api_urls: Vec<Url>,
|
||||||
refresh_rate: time::Duration,
|
refresh_rate: Duration,
|
||||||
client_version: String,
|
client_version: String,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
TopologyRefresherConfig {
|
TopologyRefresherConfig {
|
||||||
@@ -243,7 +261,7 @@ impl TopologyRefresher {
|
|||||||
|
|
||||||
let mixnodes = match self.validator_client.get_cached_active_mixnodes().await {
|
let mixnodes = match self.validator_client.get_cached_active_mixnodes().await {
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
error!("failed to get network mixnodes - {}", err);
|
error!("failed to get network mixnodes - {err}");
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
Ok(mixes) => mixes,
|
Ok(mixes) => mixes,
|
||||||
@@ -251,7 +269,7 @@ impl TopologyRefresher {
|
|||||||
|
|
||||||
let gateways = match self.validator_client.get_cached_gateways().await {
|
let gateways = match self.validator_client.get_cached_gateways().await {
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
error!("failed to get network gateways - {}", err);
|
error!("failed to get network gateways - {err}");
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
Ok(gateways) => gateways,
|
Ok(gateways) => gateways,
|
||||||
@@ -292,18 +310,26 @@ impl TopologyRefresher {
|
|||||||
.await;
|
.await;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn is_topology_routable(&self) -> bool {
|
pub async fn ensure_topology_is_routable(&self) -> Result<(), NymTopologyError> {
|
||||||
self.topology_accessor.is_routable().await
|
self.topology_accessor.ensure_is_routable().await
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
pub fn start_with_shutdown(mut self, mut shutdown: task::ShutdownListener) {
|
pub fn start_with_shutdown(mut self, mut shutdown: task::ShutdownListener) {
|
||||||
spawn_future(async move {
|
spawn_future(async move {
|
||||||
debug!("Started TopologyRefresher with graceful shutdown support");
|
debug!("Started TopologyRefresher with graceful shutdown support");
|
||||||
|
|
||||||
|
#[cfg(not(target_arch = "wasm32"))]
|
||||||
|
let mut interval = tokio_stream::wrappers::IntervalStream::new(tokio::time::interval(
|
||||||
|
self.refresh_rate,
|
||||||
|
));
|
||||||
|
|
||||||
|
#[cfg(target_arch = "wasm32")]
|
||||||
|
let mut interval =
|
||||||
|
gloo_timers::future::IntervalStream::new(self.refresh_rate.as_millis() as u32);
|
||||||
|
|
||||||
while !shutdown.is_shutdown() {
|
while !shutdown.is_shutdown() {
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
_ = tokio::time::sleep(self.refresh_rate) => {
|
_ = interval.next() => {
|
||||||
self.refresh().await;
|
self.refresh().await;
|
||||||
},
|
},
|
||||||
_ = shutdown.recv() => {
|
_ = shutdown.recv() => {
|
||||||
@@ -311,21 +337,8 @@ impl TopologyRefresher {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
assert!(shutdown.is_shutdown_poll());
|
shutdown.recv_timeout().await;
|
||||||
log::debug!("TopologyRefresher: Exiting");
|
log::debug!("TopologyRefresher: Exiting");
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
pub fn start(mut self) {
|
|
||||||
use futures::StreamExt;
|
|
||||||
|
|
||||||
spawn_future(async move {
|
|
||||||
let mut interval =
|
|
||||||
gloo_timers::future::IntervalStream::new(self.refresh_rate.as_millis() as u32);
|
|
||||||
while let Some(_) = interval.next().await {
|
|
||||||
self.refresh().await;
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -30,11 +30,33 @@ const DEFAULT_TOPOLOGY_RESOLUTION_TIMEOUT: Duration = Duration::from_millis(5_00
|
|||||||
// bandwidth bridging protocol, we can come back to a smaller timeout value
|
// bandwidth bridging protocol, we can come back to a smaller timeout value
|
||||||
const DEFAULT_GATEWAY_RESPONSE_TIMEOUT: Duration = Duration::from_secs(5 * 60);
|
const DEFAULT_GATEWAY_RESPONSE_TIMEOUT: Duration = Duration::from_secs(5 * 60);
|
||||||
|
|
||||||
|
// reply-surbs related:
|
||||||
|
|
||||||
|
// define when to request
|
||||||
|
// clients/client-core/src/client/replies/reply_storage/surb_storage.rs
|
||||||
|
const DEFAULT_MINIMUM_REPLY_SURB_STORAGE_THRESHOLD: usize = 10;
|
||||||
|
const DEFAULT_MAXIMUM_REPLY_SURB_STORAGE_THRESHOLD: usize = 200;
|
||||||
|
|
||||||
|
// define how much to request at once
|
||||||
|
// clients/client-core/src/client/replies/reply_controller.rs
|
||||||
|
const DEFAULT_MINIMUM_REPLY_SURB_REQUEST_SIZE: u32 = 10;
|
||||||
|
const DEFAULT_MAXIMUM_REPLY_SURB_REQUEST_SIZE: u32 = 100;
|
||||||
|
|
||||||
|
const DEFAULT_MAXIMUM_ALLOWED_SURB_REQUEST_SIZE: u32 = 500;
|
||||||
|
|
||||||
|
const DEFAULT_MAXIMUM_REPLY_SURB_WAITING_PERIOD: Duration = Duration::from_secs(10);
|
||||||
|
|
||||||
|
// 12 hours
|
||||||
|
const DEFAULT_MAXIMUM_REPLY_SURB_AGE: Duration = Duration::from_secs(12 * 60 * 60);
|
||||||
|
|
||||||
|
// 24 hours
|
||||||
|
const DEFAULT_MAXIMUM_REPLY_KEY_AGE: Duration = Duration::from_secs(24 * 60 * 60);
|
||||||
|
|
||||||
pub fn missing_string_value() -> String {
|
pub fn missing_string_value() -> String {
|
||||||
MISSING_VALUE.to_string()
|
MISSING_VALUE.to_string()
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Deserialize, PartialEq, Serialize)]
|
#[derive(Debug, Clone, Deserialize, PartialEq, Serialize)]
|
||||||
#[serde(deny_unknown_fields)]
|
#[serde(deny_unknown_fields)]
|
||||||
pub struct Config<T> {
|
pub struct Config<T> {
|
||||||
client: Client<T>,
|
client: Client<T>,
|
||||||
@@ -42,17 +64,21 @@ pub struct Config<T> {
|
|||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
logging: Logging,
|
logging: Logging,
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
debug: Debug,
|
debug: DebugConfig,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: NymConfig> Config<T> {
|
impl<T> Config<T> {
|
||||||
pub fn new<S: Into<String>>(id: S) -> Self {
|
pub fn new<S: Into<String>>(id: S) -> Self
|
||||||
let mut cfg = Config::default();
|
where
|
||||||
cfg.with_id(id);
|
T: NymConfig,
|
||||||
cfg
|
{
|
||||||
|
Config::default().with_id(id)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn with_id<S: Into<String>>(&mut self, id: S) {
|
pub fn with_id<S: Into<String>>(mut self, id: S) -> Self
|
||||||
|
where
|
||||||
|
T: NymConfig,
|
||||||
|
{
|
||||||
let id = id.into();
|
let id = id.into();
|
||||||
|
|
||||||
// identity key setting
|
// identity key setting
|
||||||
@@ -96,14 +122,9 @@ impl<T: NymConfig> Config<T> {
|
|||||||
self.client.ack_key_file = self::Client::<T>::default_ack_key_file(&id);
|
self.client.ack_key_file = self::Client::<T>::default_ack_key_file(&id);
|
||||||
}
|
}
|
||||||
|
|
||||||
if self
|
if self.client.reply_surb_database_path.as_os_str().is_empty() {
|
||||||
.client
|
self.client.reply_surb_database_path =
|
||||||
.reply_encryption_key_store_path
|
self::Client::<T>::default_reply_surb_database_path(&id);
|
||||||
.as_os_str()
|
|
||||||
.is_empty()
|
|
||||||
{
|
|
||||||
self.client.reply_encryption_key_store_path =
|
|
||||||
self::Client::<T>::default_reply_encryption_key_store_path(&id);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if self.client.database_path.as_os_str().is_empty() {
|
if self.client.database_path.as_os_str().is_empty() {
|
||||||
@@ -111,13 +132,14 @@ impl<T: NymConfig> Config<T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
self.client.id = id;
|
self.client.id = id;
|
||||||
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn with_disabled_credentials(&mut self, disabled_credentials_mode: bool) {
|
pub fn with_disabled_credentials(&mut self, disabled_credentials_mode: bool) {
|
||||||
self.client.disabled_credentials_mode = disabled_credentials_mode;
|
self.client.disabled_credentials_mode = disabled_credentials_mode;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn with_gateway_endpoint(&mut self, gateway_endpoint: GatewayEndpoint) {
|
pub fn with_gateway_endpoint(&mut self, gateway_endpoint: GatewayEndpointConfig) {
|
||||||
self.client.gateway_endpoint = gateway_endpoint;
|
self.client.gateway_endpoint = gateway_endpoint;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -135,8 +157,15 @@ impl<T: NymConfig> Config<T> {
|
|||||||
|
|
||||||
pub fn set_high_default_traffic_volume(&mut self) {
|
pub fn set_high_default_traffic_volume(&mut self) {
|
||||||
self.debug.average_packet_delay = Duration::from_millis(10);
|
self.debug.average_packet_delay = Duration::from_millis(10);
|
||||||
self.debug.loop_cover_traffic_average_delay = Duration::from_millis(2_000_000); // basically don't really send cover messages
|
// basically don't really send cover messages
|
||||||
self.debug.message_sending_average_delay = Duration::from_millis(4); // 250 "real" messages / s
|
self.debug.loop_cover_traffic_average_delay = Duration::from_millis(2_000_000);
|
||||||
|
// 250 "real" messages / s
|
||||||
|
self.debug.message_sending_average_delay = Duration::from_millis(4);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_no_cover_traffic(&mut self) {
|
||||||
|
self.debug.disable_loop_cover_traffic_stream = true;
|
||||||
|
self.debug.disable_main_poisson_packet_distribution = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_custom_version(&mut self, version: &str) {
|
pub fn set_custom_version(&mut self, version: &str) {
|
||||||
@@ -175,10 +204,6 @@ impl<T: NymConfig> Config<T> {
|
|||||||
self.client.gateway_shared_key_file.clone()
|
self.client.gateway_shared_key_file.clone()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_reply_encryption_key_store_path(&self) -> PathBuf {
|
|
||||||
self.client.reply_encryption_key_store_path.clone()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn get_ack_key_file(&self) -> PathBuf {
|
pub fn get_ack_key_file(&self) -> PathBuf {
|
||||||
self.client.ack_key_file.clone()
|
self.client.ack_key_file.clone()
|
||||||
}
|
}
|
||||||
@@ -203,7 +228,11 @@ impl<T: NymConfig> Config<T> {
|
|||||||
self.client.gateway_endpoint.gateway_listener.clone()
|
self.client.gateway_endpoint.gateway_listener.clone()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_gateway_endpoint(&self) -> &GatewayEndpoint {
|
pub fn get_gateway_endpoint_config(&self) -> &GatewayEndpointConfig {
|
||||||
|
&self.client.gateway_endpoint
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_gateway_endpoint(&self) -> &GatewayEndpointConfig {
|
||||||
&self.client.gateway_endpoint
|
&self.client.gateway_endpoint
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -211,7 +240,19 @@ impl<T: NymConfig> Config<T> {
|
|||||||
self.client.database_path.clone()
|
self.client.database_path.clone()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn get_reply_surb_database_path(&self) -> PathBuf {
|
||||||
|
self.client.reply_surb_database_path.clone()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_version(&self) -> &str {
|
||||||
|
&self.client.version
|
||||||
|
}
|
||||||
|
|
||||||
// Debug getters
|
// Debug getters
|
||||||
|
pub fn get_debug_config(&self) -> &DebugConfig {
|
||||||
|
&self.debug
|
||||||
|
}
|
||||||
|
|
||||||
pub fn get_average_packet_delay(&self) -> Duration {
|
pub fn get_average_packet_delay(&self) -> Duration {
|
||||||
self.debug.average_packet_delay
|
self.debug.average_packet_delay
|
||||||
}
|
}
|
||||||
@@ -257,11 +298,39 @@ impl<T: NymConfig> Config<T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_use_extended_packet_size(&self) -> Option<ExtendedPacketSize> {
|
pub fn get_use_extended_packet_size(&self) -> Option<ExtendedPacketSize> {
|
||||||
self.debug.use_extended_packet_size.clone()
|
self.debug.use_extended_packet_size
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_version(&self) -> &str {
|
pub fn get_minimum_reply_surb_storage_threshold(&self) -> usize {
|
||||||
&self.client.version
|
self.debug.minimum_reply_surb_storage_threshold
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_maximum_reply_surb_storage_threshold(&self) -> usize {
|
||||||
|
self.debug.maximum_reply_surb_storage_threshold
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_minimum_reply_surb_request_size(&self) -> u32 {
|
||||||
|
self.debug.minimum_reply_surb_request_size
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_maximum_reply_surb_request_size(&self) -> u32 {
|
||||||
|
self.debug.maximum_reply_surb_request_size
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_maximum_allowed_reply_surb_request_size(&self) -> u32 {
|
||||||
|
self.debug.maximum_allowed_reply_surb_request_size
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_maximum_reply_surb_waiting_period(&self) -> Duration {
|
||||||
|
self.debug.maximum_reply_surb_waiting_period
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_maximum_reply_surb_age(&self) -> Duration {
|
||||||
|
self.debug.maximum_reply_surb_age
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_maximum_reply_key_age(&self) -> Duration {
|
||||||
|
self.debug.maximum_reply_key_age
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -277,7 +346,7 @@ impl<T: NymConfig> Default for Config<T> {
|
|||||||
|
|
||||||
#[derive(Clone, Debug, Default, Deserialize, PartialEq, Eq, Serialize)]
|
#[derive(Clone, Debug, Default, Deserialize, PartialEq, Eq, Serialize)]
|
||||||
#[cfg_attr(target_arch = "wasm32", wasm_bindgen(getter_with_clone))]
|
#[cfg_attr(target_arch = "wasm32", wasm_bindgen(getter_with_clone))]
|
||||||
pub struct GatewayEndpoint {
|
pub struct GatewayEndpointConfig {
|
||||||
/// gateway_id specifies ID of the gateway to which the client should send messages.
|
/// gateway_id specifies ID of the gateway to which the client should send messages.
|
||||||
/// If initially omitted, a random gateway will be chosen from the available topology.
|
/// If initially omitted, a random gateway will be chosen from the available topology.
|
||||||
pub gateway_id: String,
|
pub gateway_id: String,
|
||||||
@@ -289,10 +358,26 @@ pub struct GatewayEndpoint {
|
|||||||
pub gateway_listener: String,
|
pub gateway_listener: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<topology::gateway::Node> for GatewayEndpoint {
|
#[cfg_attr(target_arch = "wasm32", wasm_bindgen)]
|
||||||
fn from(node: topology::gateway::Node) -> GatewayEndpoint {
|
impl GatewayEndpointConfig {
|
||||||
|
#[cfg_attr(target_arch = "wasm32", wasm_bindgen(constructor))]
|
||||||
|
pub fn new(
|
||||||
|
gateway_id: String,
|
||||||
|
gateway_owner: String,
|
||||||
|
gateway_listener: String,
|
||||||
|
) -> GatewayEndpointConfig {
|
||||||
|
GatewayEndpointConfig {
|
||||||
|
gateway_id,
|
||||||
|
gateway_owner,
|
||||||
|
gateway_listener,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<topology::gateway::Node> for GatewayEndpointConfig {
|
||||||
|
fn from(node: topology::gateway::Node) -> GatewayEndpointConfig {
|
||||||
let gateway_listener = node.clients_address();
|
let gateway_listener = node.clients_address();
|
||||||
GatewayEndpoint {
|
GatewayEndpointConfig {
|
||||||
gateway_id: node.identity_key.to_base58_string(),
|
gateway_id: node.identity_key.to_base58_string(),
|
||||||
gateway_owner: node.owner,
|
gateway_owner: node.owner,
|
||||||
gateway_listener,
|
gateway_listener,
|
||||||
@@ -300,7 +385,7 @@ impl From<topology::gateway::Node> for GatewayEndpoint {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Deserialize, PartialEq, Eq, Serialize)]
|
#[derive(Debug, Clone, Deserialize, PartialEq, Eq, Serialize)]
|
||||||
pub struct Client<T> {
|
pub struct Client<T> {
|
||||||
/// Version of the client for which this configuration was created.
|
/// Version of the client for which this configuration was created.
|
||||||
#[serde(default = "missing_string_value")]
|
#[serde(default = "missing_string_value")]
|
||||||
@@ -315,6 +400,7 @@ pub struct Client<T> {
|
|||||||
disabled_credentials_mode: bool,
|
disabled_credentials_mode: bool,
|
||||||
|
|
||||||
/// Addresses to nymd validators via which the client can communicate with the chain.
|
/// Addresses to nymd validators via which the client can communicate with the chain.
|
||||||
|
#[serde(default)]
|
||||||
validator_urls: Vec<Url>,
|
validator_urls: Vec<Url>,
|
||||||
|
|
||||||
/// Addresses to APIs running on validator from which the client gets the view of the network.
|
/// Addresses to APIs running on validator from which the client gets the view of the network.
|
||||||
@@ -340,16 +426,15 @@ pub struct Client<T> {
|
|||||||
/// acknowledgement so that nobody besides the client knows which packet it refers to.
|
/// acknowledgement so that nobody besides the client knows which packet it refers to.
|
||||||
ack_key_file: PathBuf,
|
ack_key_file: PathBuf,
|
||||||
|
|
||||||
/// Full path to file containing reply encryption keys of all reply-SURBs we have ever
|
|
||||||
/// sent but not received back.
|
|
||||||
reply_encryption_key_store_path: PathBuf,
|
|
||||||
|
|
||||||
/// Information regarding how the client should send data to gateway.
|
/// Information regarding how the client should send data to gateway.
|
||||||
gateway_endpoint: GatewayEndpoint,
|
gateway_endpoint: GatewayEndpointConfig,
|
||||||
|
|
||||||
/// Path to the database containing bandwidth credentials of this client.
|
/// Path to the database containing bandwidth credentials of this client.
|
||||||
database_path: PathBuf,
|
database_path: PathBuf,
|
||||||
|
|
||||||
|
/// Path to the persistent store for received reply surbs, unused encryption keys and used sender tags.
|
||||||
|
reply_surb_database_path: PathBuf,
|
||||||
|
|
||||||
/// nym_home_directory specifies absolute path to the home nym Clients directory.
|
/// nym_home_directory specifies absolute path to the home nym Clients directory.
|
||||||
/// It is expected to use default value and hence .toml file should not redefine this field.
|
/// It is expected to use default value and hence .toml file should not redefine this field.
|
||||||
nym_root_directory: PathBuf,
|
nym_root_directory: PathBuf,
|
||||||
@@ -373,9 +458,9 @@ impl<T: NymConfig> Default for Client<T> {
|
|||||||
public_encryption_key_file: Default::default(),
|
public_encryption_key_file: Default::default(),
|
||||||
gateway_shared_key_file: Default::default(),
|
gateway_shared_key_file: Default::default(),
|
||||||
ack_key_file: Default::default(),
|
ack_key_file: Default::default(),
|
||||||
reply_encryption_key_store_path: Default::default(),
|
|
||||||
gateway_endpoint: Default::default(),
|
gateway_endpoint: Default::default(),
|
||||||
database_path: Default::default(),
|
database_path: Default::default(),
|
||||||
|
reply_surb_database_path: Default::default(),
|
||||||
nym_root_directory: T::default_root_directory(),
|
nym_root_directory: T::default_root_directory(),
|
||||||
super_struct: Default::default(),
|
super_struct: Default::default(),
|
||||||
}
|
}
|
||||||
@@ -407,21 +492,22 @@ impl<T: NymConfig> Client<T> {
|
|||||||
T::default_data_directory(Some(id)).join("ack_key.pem")
|
T::default_data_directory(Some(id)).join("ack_key.pem")
|
||||||
}
|
}
|
||||||
|
|
||||||
fn default_reply_encryption_key_store_path(id: &str) -> PathBuf {
|
fn default_reply_surb_database_path(id: &str) -> PathBuf {
|
||||||
T::default_data_directory(Some(id)).join("reply_key_store")
|
T::default_data_directory(Some(id)).join("persistent_reply_store.sqlite")
|
||||||
}
|
}
|
||||||
|
|
||||||
fn default_database_path(id: &str) -> PathBuf {
|
fn default_database_path(id: &str) -> PathBuf {
|
||||||
T::default_data_directory(Some(id)).join("db.sqlite")
|
T::default_data_directory(Some(id)).join("db.sqlite")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Default, Deserialize, PartialEq, Eq, Serialize)]
|
#[derive(Debug, Clone, Default, Deserialize, PartialEq, Eq, Serialize)]
|
||||||
#[serde(deny_unknown_fields)]
|
#[serde(deny_unknown_fields)]
|
||||||
pub struct Logging {}
|
pub struct Logging {}
|
||||||
|
|
||||||
#[derive(Debug, Deserialize, PartialEq, Serialize)]
|
#[derive(Debug, Clone, Deserialize, PartialEq, Serialize)]
|
||||||
#[serde(default, deny_unknown_fields)]
|
#[serde(default, deny_unknown_fields)]
|
||||||
pub struct Debug {
|
pub struct DebugConfig {
|
||||||
/// The parameter of Poisson distribution determining how long, on average,
|
/// The parameter of Poisson distribution determining how long, on average,
|
||||||
/// sent packet is going to be delayed at any given mix node.
|
/// sent packet is going to be delayed at any given mix node.
|
||||||
/// So for a packet going through three mix nodes, on average, it will take three times this value
|
/// So for a packet going through three mix nodes, on average, it will take three times this value
|
||||||
@@ -485,9 +571,40 @@ pub struct Debug {
|
|||||||
|
|
||||||
/// Controls whether the sent sphinx packet use a NON-DEFAULT bigger size.
|
/// Controls whether the sent sphinx packet use a NON-DEFAULT bigger size.
|
||||||
pub use_extended_packet_size: Option<ExtendedPacketSize>,
|
pub use_extended_packet_size: Option<ExtendedPacketSize>,
|
||||||
|
|
||||||
|
/// Defines the minimum number of reply surbs the client wants to keep in its storage at all times.
|
||||||
|
/// It can only allow to go below that value if its to request additional reply surbs.
|
||||||
|
pub minimum_reply_surb_storage_threshold: usize,
|
||||||
|
|
||||||
|
/// Defines the maximum number of reply surbs the client wants to keep in its storage at any times.
|
||||||
|
pub maximum_reply_surb_storage_threshold: usize,
|
||||||
|
|
||||||
|
/// Defines the minimum number of reply surbs the client would request.
|
||||||
|
pub minimum_reply_surb_request_size: u32,
|
||||||
|
|
||||||
|
/// Defines the maximum number of reply surbs the client would request.
|
||||||
|
pub maximum_reply_surb_request_size: u32,
|
||||||
|
|
||||||
|
/// Defines the maximum number of reply surbs a remote party is allowed to request from this client at once.
|
||||||
|
pub maximum_allowed_reply_surb_request_size: u32,
|
||||||
|
|
||||||
|
/// Defines maximum amount of time the client is going to wait for reply surbs before explicitly asking
|
||||||
|
/// for more even though in theory they wouldn't need to.
|
||||||
|
#[serde(with = "humantime_serde")]
|
||||||
|
pub maximum_reply_surb_waiting_period: Duration,
|
||||||
|
|
||||||
|
/// Defines maximum amount of time given reply surb is going to be valid for.
|
||||||
|
/// This is going to be superseded by key rotation once implemented.
|
||||||
|
#[serde(with = "humantime_serde")]
|
||||||
|
pub maximum_reply_surb_age: Duration,
|
||||||
|
|
||||||
|
/// Defines maximum amount of time given reply key is going to be valid for.
|
||||||
|
/// This is going to be superseded by key rotation once implemented.
|
||||||
|
#[serde(with = "humantime_serde")]
|
||||||
|
pub maximum_reply_key_age: Duration,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug, Serialize, Deserialize, PartialEq, Eq)]
|
#[derive(Clone, Copy, Debug, Serialize, Deserialize, PartialEq, Eq)]
|
||||||
#[serde(rename_all = "lowercase")]
|
#[serde(rename_all = "lowercase")]
|
||||||
pub enum ExtendedPacketSize {
|
pub enum ExtendedPacketSize {
|
||||||
Extended8,
|
Extended8,
|
||||||
@@ -495,9 +612,9 @@ pub enum ExtendedPacketSize {
|
|||||||
Extended32,
|
Extended32,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for Debug {
|
impl Default for DebugConfig {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
Debug {
|
DebugConfig {
|
||||||
average_packet_delay: DEFAULT_AVERAGE_PACKET_DELAY,
|
average_packet_delay: DEFAULT_AVERAGE_PACKET_DELAY,
|
||||||
average_ack_delay: DEFAULT_AVERAGE_PACKET_DELAY,
|
average_ack_delay: DEFAULT_AVERAGE_PACKET_DELAY,
|
||||||
ack_wait_multiplier: DEFAULT_ACK_WAIT_MULTIPLIER,
|
ack_wait_multiplier: DEFAULT_ACK_WAIT_MULTIPLIER,
|
||||||
@@ -510,6 +627,14 @@ impl Default for Debug {
|
|||||||
disable_loop_cover_traffic_stream: false,
|
disable_loop_cover_traffic_stream: false,
|
||||||
disable_main_poisson_packet_distribution: false,
|
disable_main_poisson_packet_distribution: false,
|
||||||
use_extended_packet_size: None,
|
use_extended_packet_size: None,
|
||||||
|
minimum_reply_surb_storage_threshold: DEFAULT_MINIMUM_REPLY_SURB_STORAGE_THRESHOLD,
|
||||||
|
maximum_reply_surb_storage_threshold: DEFAULT_MAXIMUM_REPLY_SURB_STORAGE_THRESHOLD,
|
||||||
|
minimum_reply_surb_request_size: DEFAULT_MINIMUM_REPLY_SURB_REQUEST_SIZE,
|
||||||
|
maximum_reply_surb_request_size: DEFAULT_MAXIMUM_REPLY_SURB_REQUEST_SIZE,
|
||||||
|
maximum_allowed_reply_surb_request_size: DEFAULT_MAXIMUM_ALLOWED_SURB_REQUEST_SIZE,
|
||||||
|
maximum_reply_surb_waiting_period: DEFAULT_MAXIMUM_REPLY_SURB_WAITING_PERIOD,
|
||||||
|
maximum_reply_surb_age: DEFAULT_MAXIMUM_REPLY_SURB_AGE,
|
||||||
|
maximum_reply_key_age: DEFAULT_MAXIMUM_REPLY_KEY_AGE,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,12 +1,14 @@
|
|||||||
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use crate::client::replies::reply_storage::ReplyStorageBackend;
|
||||||
use crypto::asymmetric::identity::Ed25519RecoveryError;
|
use crypto::asymmetric::identity::Ed25519RecoveryError;
|
||||||
use gateway_client::error::GatewayClientError;
|
use gateway_client::error::GatewayClientError;
|
||||||
|
use topology::NymTopologyError;
|
||||||
use validator_client::ValidatorClientError;
|
use validator_client::ValidatorClientError;
|
||||||
|
|
||||||
#[derive(thiserror::Error, Debug)]
|
#[derive(thiserror::Error, Debug)]
|
||||||
pub enum ClientCoreError {
|
pub enum ClientCoreError<B: ReplyStorageBackend> {
|
||||||
#[error("I/O error: {0}")]
|
#[error("I/O error: {0}")]
|
||||||
IoError(#[from] std::io::Error),
|
IoError(#[from] std::io::Error),
|
||||||
#[error("Gateway client error: {0}")]
|
#[error("Gateway client error: {0}")]
|
||||||
@@ -25,5 +27,11 @@ pub enum ClientCoreError {
|
|||||||
#[error("Could not load existing gateway configuration: {0}")]
|
#[error("Could not load existing gateway configuration: {0}")]
|
||||||
CouldNotLoadExistingGatewayConfiguration(std::io::Error),
|
CouldNotLoadExistingGatewayConfiguration(std::io::Error),
|
||||||
#[error("The current network topology seem to be insufficient to route any packets through")]
|
#[error("The current network topology seem to be insufficient to route any packets through")]
|
||||||
InsufficientNetworkTopology,
|
InsufficientNetworkTopology(#[from] NymTopologyError),
|
||||||
|
|
||||||
|
#[error("Unexpected exit")]
|
||||||
|
UnexpectedExit,
|
||||||
|
|
||||||
|
#[error("experienced a failure with our reply surb persistent storage: {source}")]
|
||||||
|
SurbStorageError { source: B::StorageError },
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,16 +18,20 @@ use tap::TapFallible;
|
|||||||
use topology::{filter::VersionFilterable, gateway};
|
use topology::{filter::VersionFilterable, gateway};
|
||||||
use url::Url;
|
use url::Url;
|
||||||
|
|
||||||
|
use crate::client::replies::reply_storage::ReplyStorageBackend;
|
||||||
use crate::{
|
use crate::{
|
||||||
client::key_manager::KeyManager,
|
client::key_manager::KeyManager,
|
||||||
config::{persistence::key_pathfinder::ClientKeyPathfinder, Config},
|
config::{persistence::key_pathfinder::ClientKeyPathfinder, Config},
|
||||||
error::ClientCoreError,
|
error::ClientCoreError,
|
||||||
};
|
};
|
||||||
|
|
||||||
pub async fn query_gateway_details(
|
pub async fn query_gateway_details<B>(
|
||||||
validator_servers: Vec<Url>,
|
validator_servers: Vec<Url>,
|
||||||
chosen_gateway_id: Option<&str>,
|
chosen_gateway_id: Option<&str>,
|
||||||
) -> Result<gateway::Node, ClientCoreError> {
|
) -> Result<gateway::Node, ClientCoreError<B>>
|
||||||
|
where
|
||||||
|
B: ReplyStorageBackend,
|
||||||
|
{
|
||||||
let validator_api = validator_servers
|
let validator_api = validator_servers
|
||||||
.choose(&mut thread_rng())
|
.choose(&mut thread_rng())
|
||||||
.ok_or(ClientCoreError::ListOfValidatorApisIsEmpty)?;
|
.ok_or(ClientCoreError::ListOfValidatorApisIsEmpty)?;
|
||||||
@@ -59,12 +63,13 @@ pub async fn query_gateway_details(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn register_with_gateway_and_store_keys<T>(
|
pub async fn register_with_gateway_and_store_keys<T, B>(
|
||||||
gateway_details: gateway::Node,
|
gateway_details: gateway::Node,
|
||||||
config: &Config<T>,
|
config: &Config<T>,
|
||||||
) -> Result<(), ClientCoreError>
|
) -> Result<(), ClientCoreError<B>>
|
||||||
where
|
where
|
||||||
T: NymConfig,
|
T: NymConfig,
|
||||||
|
B: ReplyStorageBackend,
|
||||||
{
|
{
|
||||||
let mut rng = OsRng;
|
let mut rng = OsRng;
|
||||||
let mut key_manager = KeyManager::new(&mut rng);
|
let mut key_manager = KeyManager::new(&mut rng);
|
||||||
@@ -79,10 +84,13 @@ where
|
|||||||
.tap_err(|err| log::error!("Failed to generate keys: {err}"))?)
|
.tap_err(|err| log::error!("Failed to generate keys: {err}"))?)
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn register_with_gateway(
|
async fn register_with_gateway<B>(
|
||||||
gateway: &gateway::Node,
|
gateway: &gateway::Node,
|
||||||
our_identity: Arc<identity::KeyPair>,
|
our_identity: Arc<identity::KeyPair>,
|
||||||
) -> Result<Arc<SharedKeys>, ClientCoreError> {
|
) -> Result<Arc<SharedKeys>, ClientCoreError<B>>
|
||||||
|
where
|
||||||
|
B: ReplyStorageBackend,
|
||||||
|
{
|
||||||
let timeout = Duration::from_millis(1500);
|
let timeout = Duration::from_millis(1500);
|
||||||
let mut gateway_client = GatewayClient::new_init(
|
let mut gateway_client = GatewayClient::new_init(
|
||||||
gateway.clients_address(),
|
gateway.clients_address(),
|
||||||
@@ -90,8 +98,6 @@ async fn register_with_gateway(
|
|||||||
gateway.owner.clone(),
|
gateway.owner.clone(),
|
||||||
our_identity.clone(),
|
our_identity.clone(),
|
||||||
timeout,
|
timeout,
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
None,
|
|
||||||
);
|
);
|
||||||
gateway_client
|
gateway_client
|
||||||
.establish_connection()
|
.establish_connection()
|
||||||
@@ -104,13 +110,17 @@ async fn register_with_gateway(
|
|||||||
Ok(shared_keys)
|
Ok(shared_keys)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn show_address<T>(config: &Config<T>) -> Result<(), ClientCoreError>
|
pub fn show_address<T, B>(config: &Config<T>) -> Result<(), ClientCoreError<B>>
|
||||||
where
|
where
|
||||||
T: config::NymConfig,
|
T: config::NymConfig,
|
||||||
|
B: ReplyStorageBackend,
|
||||||
{
|
{
|
||||||
fn load_identity_keys(
|
fn load_identity_keys<B>(
|
||||||
pathfinder: &ClientKeyPathfinder,
|
pathfinder: &ClientKeyPathfinder,
|
||||||
) -> Result<identity::KeyPair, ClientCoreError> {
|
) -> Result<identity::KeyPair, ClientCoreError<B>>
|
||||||
|
where
|
||||||
|
B: ReplyStorageBackend,
|
||||||
|
{
|
||||||
let identity_keypair: identity::KeyPair =
|
let identity_keypair: identity::KeyPair =
|
||||||
pemstore::load_keypair(&pemstore::KeyPairPath::new(
|
pemstore::load_keypair(&pemstore::KeyPairPath::new(
|
||||||
pathfinder.private_identity_key().to_owned(),
|
pathfinder.private_identity_key().to_owned(),
|
||||||
@@ -120,9 +130,12 @@ where
|
|||||||
Ok(identity_keypair)
|
Ok(identity_keypair)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn load_sphinx_keys(
|
fn load_sphinx_keys<B>(
|
||||||
pathfinder: &ClientKeyPathfinder,
|
pathfinder: &ClientKeyPathfinder,
|
||||||
) -> Result<encryption::KeyPair, ClientCoreError> {
|
) -> Result<encryption::KeyPair, ClientCoreError<B>>
|
||||||
|
where
|
||||||
|
B: ReplyStorageBackend,
|
||||||
|
{
|
||||||
let sphinx_keypair: encryption::KeyPair =
|
let sphinx_keypair: encryption::KeyPair =
|
||||||
pemstore::load_keypair(&pemstore::KeyPairPath::new(
|
pemstore::load_keypair(&pemstore::KeyPairPath::new(
|
||||||
pathfinder.private_encryption_key().to_owned(),
|
pathfinder.private_encryption_key().to_owned(),
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "nym-client"
|
name = "nym-client"
|
||||||
version = "1.1.0"
|
version = "1.1.1"
|
||||||
authors = ["Dave Hrycyszyn <futurechimp@users.noreply.github.com>", "Jędrzej Stuczyński <andrew@nymtech.net>"]
|
authors = ["Dave Hrycyszyn <futurechimp@users.noreply.github.com>", "Jędrzej Stuczyński <andrew@nymtech.net>"]
|
||||||
description = "Implementation of the Nym Client"
|
description = "Implementation of the Nym Client"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
@@ -32,7 +32,7 @@ tokio = { version = "1.21.2", features = ["rt-multi-thread", "net", "signal"] }
|
|||||||
tokio-tungstenite = "0.14" # websocket
|
tokio-tungstenite = "0.14" # websocket
|
||||||
|
|
||||||
## internal
|
## internal
|
||||||
client-core = { path = "../client-core" }
|
client-core = { path = "../client-core", features = ["fs-surb-storage"] }
|
||||||
client-connections = { path = "../../common/client-connections" }
|
client-connections = { path = "../../common/client-connections" }
|
||||||
coconut-interface = { path = "../../common/coconut-interface", optional = true }
|
coconut-interface = { path = "../../common/coconut-interface", optional = true }
|
||||||
config = { path = "../../common/config" }
|
config = { path = "../../common/config" }
|
||||||
|
|||||||
@@ -25,58 +25,59 @@ async fn get_self_address(ws_stream: &mut WebSocketStream<MaybeTlsStream<TcpStre
|
|||||||
let response = send_message_and_get_response(ws_stream, self_address_request).await;
|
let response = send_message_and_get_response(ws_stream, self_address_request).await;
|
||||||
|
|
||||||
match response {
|
match response {
|
||||||
ServerResponse::SelfAddress(recipient) => recipient,
|
ServerResponse::SelfAddress(recipient) => *recipient,
|
||||||
_ => panic!("received an unexpected response!"),
|
_ => panic!("received an unexpected response!"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn send_file_with_reply() {
|
async fn send_file_with_reply() {
|
||||||
let uri = "ws://localhost:1977";
|
todo!("reimplement surb usage here : )")
|
||||||
let (mut ws_stream, _) = connect_async(uri).await.unwrap();
|
// let uri = "ws://localhost:1977";
|
||||||
|
// let (mut ws_stream, _) = connect_async(uri).await.unwrap();
|
||||||
let recipient = get_self_address(&mut ws_stream).await;
|
//
|
||||||
println!("our full address is: {}", recipient);
|
// let recipient = get_self_address(&mut ws_stream).await;
|
||||||
|
// println!("our full address is: {}", recipient);
|
||||||
let read_data = std::fs::read("examples/dummy_file").unwrap();
|
//
|
||||||
|
// let read_data = std::fs::read("examples/dummy_file").unwrap();
|
||||||
let send_request = ClientRequest::Send {
|
//
|
||||||
recipient,
|
// let send_request = ClientRequest::Send {
|
||||||
message: read_data,
|
// recipient,
|
||||||
with_reply_surb: true,
|
// message: read_data,
|
||||||
connection_id: Some(0),
|
// with_reply_surb: true,
|
||||||
};
|
// connection_id: Some(0),
|
||||||
|
// };
|
||||||
println!("sending content of 'dummy_file' over the mix network...");
|
//
|
||||||
let response = send_message_and_get_response(&mut ws_stream, send_request.serialize()).await;
|
// println!("sending content of 'dummy_file' over the mix network...");
|
||||||
|
// let response = send_message_and_get_response(&mut ws_stream, send_request.serialize()).await;
|
||||||
let received = match response {
|
//
|
||||||
ServerResponse::Received(received) => received,
|
// let received = match response {
|
||||||
_ => panic!("received an unexpected response!"),
|
// ServerResponse::Received(received) => received,
|
||||||
};
|
// _ => panic!("received an unexpected response!"),
|
||||||
|
// };
|
||||||
println!("writing the file back to the disk!");
|
//
|
||||||
std::fs::write("examples/received_file_withreply", received.message).unwrap();
|
// println!("writing the file back to the disk!");
|
||||||
|
// std::fs::write("examples/received_file_withreply", received.message).unwrap();
|
||||||
let reply_message = b"hello from reply SURB! - thanks for sending me the file!".to_vec();
|
//
|
||||||
let reply_request = ClientRequest::Reply {
|
// let reply_message = b"hello from reply SURB! - thanks for sending me the file!".to_vec();
|
||||||
message: reply_message.clone(),
|
// let reply_request = ClientRequest::Reply {
|
||||||
reply_surb: received.reply_surb.unwrap(),
|
// message: reply_message.clone(),
|
||||||
};
|
// reply_surb: received.reply_surb.unwrap(),
|
||||||
|
// };
|
||||||
println!(
|
//
|
||||||
"sending {:?} (using reply SURB!) over the mix network...",
|
// println!(
|
||||||
String::from_utf8(reply_message).unwrap()
|
// "sending {:?} (using reply SURB!) over the mix network...",
|
||||||
);
|
// String::from_utf8(reply_message).unwrap()
|
||||||
let response = send_message_and_get_response(&mut ws_stream, reply_request.serialize()).await;
|
// );
|
||||||
let received = match response {
|
// let response = send_message_and_get_response(&mut ws_stream, reply_request.serialize()).await;
|
||||||
ServerResponse::Received(received) => received,
|
// let received = match response {
|
||||||
_ => panic!("received an unexpected response!"),
|
// ServerResponse::Received(received) => received,
|
||||||
};
|
// _ => panic!("received an unexpected response!"),
|
||||||
|
// };
|
||||||
println!(
|
//
|
||||||
"received {:#?} from the mix network!",
|
// println!(
|
||||||
String::from_utf8(received.message).unwrap()
|
// "received {:#?} from the mix network!",
|
||||||
);
|
// String::from_utf8(received.message).unwrap()
|
||||||
|
// );
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn send_file_without_reply() {
|
async fn send_file_without_reply() {
|
||||||
@@ -91,7 +92,6 @@ async fn send_file_without_reply() {
|
|||||||
let send_request = ClientRequest::Send {
|
let send_request = ClientRequest::Send {
|
||||||
recipient,
|
recipient,
|
||||||
message: read_data,
|
message: read_data,
|
||||||
with_reply_surb: false,
|
|
||||||
connection_id: Some(0),
|
connection_id: Some(0),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -2,8 +2,8 @@
|
|||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use crate::client::config::template::config_template;
|
use crate::client::config::template::config_template;
|
||||||
use client_core::config::Config as BaseConfig;
|
|
||||||
pub use client_core::config::MISSING_VALUE;
|
pub use client_core::config::MISSING_VALUE;
|
||||||
|
use client_core::config::{Config as BaseConfig, DebugConfig};
|
||||||
use config::defaults::DEFAULT_WEBSOCKET_LISTENING_PORT;
|
use config::defaults::DEFAULT_WEBSOCKET_LISTENING_PORT;
|
||||||
use config::NymConfig;
|
use config::NymConfig;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
@@ -27,6 +27,10 @@ impl SocketType {
|
|||||||
_ => SocketType::None,
|
_ => SocketType::None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn is_websocket(&self) -> bool {
|
||||||
|
matches!(self, SocketType::WebSocket)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Default, Deserialize, PartialEq, Serialize)]
|
#[derive(Debug, Default, Deserialize, PartialEq, Serialize)]
|
||||||
@@ -100,6 +104,10 @@ impl Config {
|
|||||||
&mut self.base
|
&mut self.base
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn get_debug_settings(&self) -> &DebugConfig {
|
||||||
|
self.get_base().get_debug_config()
|
||||||
|
}
|
||||||
|
|
||||||
pub fn get_socket_type(&self) -> SocketType {
|
pub fn get_socket_type(&self) -> SocketType {
|
||||||
self.socket.socket_type
|
self.socket.socket_type
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -49,13 +49,12 @@ private_encryption_key_file = '{{ client.private_encryption_key_file }}'
|
|||||||
# Path to file containing public encryption key.
|
# Path to file containing public encryption key.
|
||||||
public_encryption_key_file = '{{ client.public_encryption_key_file }}'
|
public_encryption_key_file = '{{ client.public_encryption_key_file }}'
|
||||||
|
|
||||||
# Full path to file containing reply encryption keys of all reply-SURBs we have ever
|
|
||||||
# sent but not received back.
|
|
||||||
reply_encryption_key_store_path = '{{ client.reply_encryption_key_store_path }}'
|
|
||||||
|
|
||||||
# Path to the database containing bandwidth credentials
|
# Path to the database containing bandwidth credentials
|
||||||
database_path = '{{ client.database_path }}'
|
database_path = '{{ client.database_path }}'
|
||||||
|
|
||||||
|
# Path to the persistent store for received reply surbs, unused encryption keys and used sender tags.
|
||||||
|
reply_surb_database_path = '{{ client.reply_surb_database_path }}'
|
||||||
|
|
||||||
##### additional client config options #####
|
##### additional client config options #####
|
||||||
|
|
||||||
# A gateway specific, optional, base58 stringified shared key used for
|
# A gateway specific, optional, base58 stringified shared key used for
|
||||||
|
|||||||
+169
-416
@@ -1,205 +1,48 @@
|
|||||||
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
// Copyright 2021-2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use client_connections::{
|
use crate::client::config::Config;
|
||||||
ConnectionCommandReceiver, ConnectionCommandSender, LaneQueueLengths, TransmissionLane,
|
|
||||||
};
|
|
||||||
use client_core::client::cover_traffic_stream::LoopCoverTrafficStream;
|
|
||||||
use client_core::client::inbound_messages::{
|
|
||||||
InputMessage, InputMessageReceiver, InputMessageSender,
|
|
||||||
};
|
|
||||||
use client_core::client::key_manager::KeyManager;
|
|
||||||
use client_core::client::mix_traffic::{BatchMixMessageSender, MixTrafficController};
|
|
||||||
use client_core::client::real_messages_control;
|
|
||||||
use client_core::client::real_messages_control::RealMessagesController;
|
|
||||||
use client_core::client::received_buffer::{
|
|
||||||
ReceivedBufferMessage, ReceivedBufferRequestReceiver, ReceivedBufferRequestSender,
|
|
||||||
ReceivedMessagesBufferController, ReconstructedMessagesReceiver,
|
|
||||||
};
|
|
||||||
use client_core::client::reply_key_storage::ReplyKeyStorage;
|
|
||||||
use client_core::client::topology_control::{
|
|
||||||
TopologyAccessor, TopologyRefresher, TopologyRefresherConfig,
|
|
||||||
};
|
|
||||||
use client_core::config::persistence::key_pathfinder::ClientKeyPathfinder;
|
|
||||||
use client_core::error::ClientCoreError;
|
|
||||||
use crypto::asymmetric::identity;
|
|
||||||
use futures::channel::mpsc;
|
|
||||||
use gateway_client::bandwidth::BandwidthController;
|
|
||||||
use gateway_client::{
|
|
||||||
AcknowledgementReceiver, AcknowledgementSender, GatewayClient, MixnetMessageReceiver,
|
|
||||||
MixnetMessageSender,
|
|
||||||
};
|
|
||||||
use log::*;
|
|
||||||
use nymsphinx::addressing::clients::Recipient;
|
|
||||||
use nymsphinx::addressing::nodes::NodeIdentity;
|
|
||||||
use nymsphinx::anonymous_replies::ReplySurb;
|
|
||||||
use nymsphinx::receiver::ReconstructedMessage;
|
|
||||||
use tap::TapFallible;
|
|
||||||
use task::{wait_for_signal, ShutdownListener, ShutdownNotifier};
|
|
||||||
|
|
||||||
use crate::client::config::{Config, SocketType};
|
|
||||||
use crate::error::ClientError;
|
use crate::error::ClientError;
|
||||||
use crate::websocket;
|
use crate::websocket;
|
||||||
|
use client_connections::TransmissionLane;
|
||||||
|
use client_core::client::base_client::{
|
||||||
|
non_wasm_helpers, BaseClientBuilder, ClientInput, ClientOutput,
|
||||||
|
};
|
||||||
|
use client_core::client::inbound_messages::InputMessage;
|
||||||
|
use client_core::client::key_manager::KeyManager;
|
||||||
|
use client_core::client::received_buffer::{ReceivedBufferMessage, ReconstructedMessagesReceiver};
|
||||||
|
use client_core::config::persistence::key_pathfinder::ClientKeyPathfinder;
|
||||||
|
use futures::channel::mpsc;
|
||||||
|
use gateway_client::bandwidth::BandwidthController;
|
||||||
|
use log::*;
|
||||||
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
|
use nymsphinx::anonymous_replies::requests::AnonymousSenderTag;
|
||||||
|
use nymsphinx::receiver::ReconstructedMessage;
|
||||||
|
use task::{wait_for_signal, ShutdownNotifier};
|
||||||
|
|
||||||
pub(crate) mod config;
|
pub(crate) mod config;
|
||||||
|
|
||||||
pub struct NymClient {
|
pub struct SocketClient {
|
||||||
/// Client configuration options, including, among other things, packet sending rates,
|
/// Client configuration options, including, among other things, packet sending rates,
|
||||||
/// key filepaths, etc.
|
/// key filepaths, etc.
|
||||||
config: Config,
|
config: Config,
|
||||||
|
|
||||||
/// KeyManager object containing smart pointers to all relevant keys used by the client.
|
/// KeyManager object containing smart pointers to all relevant keys used by the client.
|
||||||
key_manager: KeyManager,
|
key_manager: KeyManager,
|
||||||
|
|
||||||
/// Channel used for transforming 'raw' messages into sphinx packets and sending them
|
|
||||||
/// through the mix network.
|
|
||||||
/// It is only available if the client started with the websocket listener disabled.
|
|
||||||
input_tx: Option<InputMessageSender>,
|
|
||||||
|
|
||||||
/// Channel used for obtaining reconstructed messages received from the mix network.
|
|
||||||
/// It is only available if the client started with the websocket listener disabled.
|
|
||||||
receive_tx: Option<ReconstructedMessagesReceiver>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl NymClient {
|
impl SocketClient {
|
||||||
pub fn new(config: Config) -> Self {
|
pub fn new(config: Config) -> Self {
|
||||||
let pathfinder = ClientKeyPathfinder::new_from_config(config.get_base());
|
let pathfinder = ClientKeyPathfinder::new_from_config(config.get_base());
|
||||||
let key_manager = KeyManager::load_keys(&pathfinder).expect("failed to load stored keys");
|
let key_manager = KeyManager::load_keys(&pathfinder).expect("failed to load stored keys");
|
||||||
|
|
||||||
NymClient {
|
SocketClient {
|
||||||
config,
|
config,
|
||||||
key_manager,
|
key_manager,
|
||||||
input_tx: None,
|
|
||||||
receive_tx: None,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn as_mix_recipient(&self) -> Recipient {
|
async fn create_bandwidth_controller(config: &Config) -> BandwidthController {
|
||||||
Recipient::new(
|
|
||||||
*self.key_manager.identity_keypair().public_key(),
|
|
||||||
*self.key_manager.encryption_keypair().public_key(),
|
|
||||||
// TODO: below only works under assumption that gateway address == gateway id
|
|
||||||
// (which currently is true)
|
|
||||||
NodeIdentity::from_base58_string(self.config.get_base().get_gateway_id()).unwrap(),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
// future constantly pumping loop cover traffic at some specified average rate
|
|
||||||
// the pumped traffic goes to the MixTrafficController
|
|
||||||
fn start_cover_traffic_stream(
|
|
||||||
&self,
|
|
||||||
topology_accessor: TopologyAccessor,
|
|
||||||
mix_tx: BatchMixMessageSender,
|
|
||||||
shutdown: ShutdownListener,
|
|
||||||
) {
|
|
||||||
info!("Starting loop cover traffic stream...");
|
|
||||||
|
|
||||||
let mut stream = LoopCoverTrafficStream::new(
|
|
||||||
self.key_manager.ack_key(),
|
|
||||||
self.config.get_base().get_average_ack_delay(),
|
|
||||||
self.config.get_base().get_average_packet_delay(),
|
|
||||||
self.config
|
|
||||||
.get_base()
|
|
||||||
.get_loop_cover_traffic_average_delay(),
|
|
||||||
mix_tx,
|
|
||||||
self.as_mix_recipient(),
|
|
||||||
topology_accessor,
|
|
||||||
);
|
|
||||||
|
|
||||||
if let Some(size) = self.config.get_base().get_use_extended_packet_size() {
|
|
||||||
log::debug!("Setting extended packet size: {:?}", size);
|
|
||||||
stream.set_custom_packet_size(size.into());
|
|
||||||
}
|
|
||||||
|
|
||||||
stream.start_with_shutdown(shutdown);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[allow(clippy::too_many_arguments)]
|
|
||||||
fn start_real_traffic_controller(
|
|
||||||
&self,
|
|
||||||
topology_accessor: TopologyAccessor,
|
|
||||||
reply_key_storage: ReplyKeyStorage,
|
|
||||||
ack_receiver: AcknowledgementReceiver,
|
|
||||||
input_receiver: InputMessageReceiver,
|
|
||||||
mix_sender: BatchMixMessageSender,
|
|
||||||
lane_queue_lengths: LaneQueueLengths,
|
|
||||||
client_connection_rx: ConnectionCommandReceiver,
|
|
||||||
shutdown: ShutdownListener,
|
|
||||||
) {
|
|
||||||
let mut controller_config = real_messages_control::Config::new(
|
|
||||||
self.key_manager.ack_key(),
|
|
||||||
self.config.get_base().get_ack_wait_multiplier(),
|
|
||||||
self.config.get_base().get_ack_wait_addition(),
|
|
||||||
self.config.get_base().get_average_ack_delay(),
|
|
||||||
self.config.get_base().get_message_sending_average_delay(),
|
|
||||||
self.config.get_base().get_average_packet_delay(),
|
|
||||||
self.config
|
|
||||||
.get_base()
|
|
||||||
.get_disabled_main_poisson_packet_distribution(),
|
|
||||||
self.as_mix_recipient(),
|
|
||||||
);
|
|
||||||
|
|
||||||
if let Some(size) = self.config.get_base().get_use_extended_packet_size() {
|
|
||||||
log::debug!("Setting extended packet size: {:?}", size);
|
|
||||||
controller_config.set_custom_packet_size(size.into());
|
|
||||||
}
|
|
||||||
|
|
||||||
info!("Starting real traffic stream...");
|
|
||||||
|
|
||||||
RealMessagesController::new(
|
|
||||||
controller_config,
|
|
||||||
ack_receiver,
|
|
||||||
input_receiver,
|
|
||||||
mix_sender,
|
|
||||||
topology_accessor,
|
|
||||||
reply_key_storage,
|
|
||||||
lane_queue_lengths,
|
|
||||||
client_connection_rx,
|
|
||||||
)
|
|
||||||
.start_with_shutdown(shutdown);
|
|
||||||
}
|
|
||||||
|
|
||||||
// buffer controlling all messages fetched from provider
|
|
||||||
// required so that other components would be able to use them (say the websocket)
|
|
||||||
fn start_received_messages_buffer_controller(
|
|
||||||
&self,
|
|
||||||
query_receiver: ReceivedBufferRequestReceiver,
|
|
||||||
mixnet_receiver: MixnetMessageReceiver,
|
|
||||||
reply_key_storage: ReplyKeyStorage,
|
|
||||||
shutdown: ShutdownListener,
|
|
||||||
) {
|
|
||||||
info!("Starting received messages buffer controller...");
|
|
||||||
ReceivedMessagesBufferController::new(
|
|
||||||
self.key_manager.encryption_keypair(),
|
|
||||||
query_receiver,
|
|
||||||
mixnet_receiver,
|
|
||||||
reply_key_storage,
|
|
||||||
)
|
|
||||||
.start_with_shutdown(shutdown)
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn start_gateway_client(
|
|
||||||
&mut self,
|
|
||||||
mixnet_message_sender: MixnetMessageSender,
|
|
||||||
ack_sender: AcknowledgementSender,
|
|
||||||
shutdown: ShutdownListener,
|
|
||||||
) -> GatewayClient {
|
|
||||||
let gateway_id = self.config.get_base().get_gateway_id();
|
|
||||||
if gateway_id.is_empty() {
|
|
||||||
panic!("The identity of the gateway is unknown - did you run `nym-client` init?")
|
|
||||||
}
|
|
||||||
let gateway_owner = self.config.get_base().get_gateway_owner();
|
|
||||||
if gateway_owner.is_empty() {
|
|
||||||
panic!("The owner of the gateway is unknown - did you run `nym-client` init?")
|
|
||||||
}
|
|
||||||
let gateway_address = self.config.get_base().get_gateway_listener();
|
|
||||||
if gateway_address.is_empty() {
|
|
||||||
panic!("The address of the gateway is unknown - did you run `nym-client` init?")
|
|
||||||
}
|
|
||||||
|
|
||||||
let gateway_identity = identity::PublicKey::from_base58_string(gateway_id)
|
|
||||||
.expect("provided gateway id is invalid!");
|
|
||||||
|
|
||||||
#[cfg(feature = "coconut")]
|
#[cfg(feature = "coconut")]
|
||||||
let bandwidth_controller = {
|
let bandwidth_controller = {
|
||||||
let details = network_defaults::NymNetworkDetails::new_from_env();
|
let details = network_defaults::NymNetworkDetails::new_from_env();
|
||||||
@@ -212,124 +55,153 @@ impl NymClient {
|
|||||||
.await
|
.await
|
||||||
.expect("Could not query api clients");
|
.expect("Could not query api clients");
|
||||||
BandwidthController::new(
|
BandwidthController::new(
|
||||||
credential_storage::initialise_storage(self.config.get_base().get_database_path())
|
credential_storage::initialise_storage(config.get_base().get_database_path()).await,
|
||||||
.await,
|
|
||||||
coconut_api_clients,
|
coconut_api_clients,
|
||||||
)
|
)
|
||||||
};
|
};
|
||||||
#[cfg(not(feature = "coconut"))]
|
#[cfg(not(feature = "coconut"))]
|
||||||
let bandwidth_controller = BandwidthController::new(
|
let bandwidth_controller = BandwidthController::new(
|
||||||
credential_storage::initialise_storage(self.config.get_base().get_database_path())
|
credential_storage::initialise_storage(config.get_base().get_database_path()).await,
|
||||||
.await,
|
|
||||||
)
|
)
|
||||||
.expect("Could not create bandwidth controller");
|
.expect("Could not create bandwidth controller");
|
||||||
|
bandwidth_controller
|
||||||
let mut gateway_client = GatewayClient::new(
|
|
||||||
gateway_address,
|
|
||||||
self.key_manager.identity_keypair(),
|
|
||||||
gateway_identity,
|
|
||||||
gateway_owner,
|
|
||||||
Some(self.key_manager.gateway_shared_key()),
|
|
||||||
mixnet_message_sender,
|
|
||||||
ack_sender,
|
|
||||||
self.config.get_base().get_gateway_response_timeout(),
|
|
||||||
Some(bandwidth_controller),
|
|
||||||
Some(shutdown),
|
|
||||||
);
|
|
||||||
|
|
||||||
gateway_client
|
|
||||||
.set_disabled_credentials_mode(self.config.get_base().get_disabled_credentials_mode());
|
|
||||||
|
|
||||||
gateway_client
|
|
||||||
.authenticate_and_start()
|
|
||||||
.await
|
|
||||||
.expect("could not authenticate and start up the gateway connection");
|
|
||||||
|
|
||||||
gateway_client
|
|
||||||
}
|
|
||||||
|
|
||||||
// future responsible for periodically polling directory server and updating
|
|
||||||
// the current global view of topology
|
|
||||||
async fn start_topology_refresher(
|
|
||||||
&mut self,
|
|
||||||
topology_accessor: TopologyAccessor,
|
|
||||||
shutdown: ShutdownListener,
|
|
||||||
) -> Result<(), ClientError> {
|
|
||||||
let topology_refresher_config = TopologyRefresherConfig::new(
|
|
||||||
self.config.get_base().get_validator_api_endpoints(),
|
|
||||||
self.config.get_base().get_topology_refresh_rate(),
|
|
||||||
env!("CARGO_PKG_VERSION").to_string(),
|
|
||||||
);
|
|
||||||
let mut topology_refresher =
|
|
||||||
TopologyRefresher::new(topology_refresher_config, topology_accessor);
|
|
||||||
// before returning, block entire runtime to refresh the current network view so that any
|
|
||||||
// components depending on topology would see a non-empty view
|
|
||||||
info!("Obtaining initial network topology");
|
|
||||||
topology_refresher.refresh().await;
|
|
||||||
|
|
||||||
// TODO: a slightly more graceful termination here
|
|
||||||
if !topology_refresher.is_topology_routable().await {
|
|
||||||
log::error!(
|
|
||||||
"The current network topology seem to be insufficient to route any packets through \
|
|
||||||
- check if enough nodes and a gateway are online"
|
|
||||||
);
|
|
||||||
return Err(ClientCoreError::InsufficientNetworkTopology.into());
|
|
||||||
}
|
|
||||||
|
|
||||||
info!("Starting topology refresher...");
|
|
||||||
topology_refresher.start_with_shutdown(shutdown);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
// controller for sending sphinx packets to mixnet (either real traffic or cover traffic)
|
|
||||||
// TODO: if we want to send control messages to gateway_client, this CAN'T take the ownership
|
|
||||||
// over it. Perhaps GatewayClient needs to be thread-shareable or have some channel for
|
|
||||||
// requests?
|
|
||||||
fn start_mix_traffic_controller(
|
|
||||||
gateway_client: GatewayClient,
|
|
||||||
shutdown: ShutdownListener,
|
|
||||||
) -> BatchMixMessageSender {
|
|
||||||
info!("Starting mix traffic controller...");
|
|
||||||
let (mix_traffic_controller, mix_tx) = MixTrafficController::new(gateway_client);
|
|
||||||
mix_traffic_controller.start_with_shutdown(shutdown);
|
|
||||||
mix_tx
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn start_websocket_listener(
|
fn start_websocket_listener(
|
||||||
&self,
|
config: &Config,
|
||||||
buffer_requester: ReceivedBufferRequestSender,
|
client_input: ClientInput,
|
||||||
msg_input: InputMessageSender,
|
client_output: ClientOutput,
|
||||||
shared_lane_queue_lengths: LaneQueueLengths,
|
self_address: Recipient,
|
||||||
client_connection_tx: ConnectionCommandSender,
|
|
||||||
) {
|
) {
|
||||||
info!("Starting websocket listener...");
|
info!("Starting websocket listener...");
|
||||||
|
|
||||||
|
let ClientInput {
|
||||||
|
shared_lane_queue_lengths,
|
||||||
|
connection_command_sender,
|
||||||
|
input_sender,
|
||||||
|
} = client_input;
|
||||||
|
|
||||||
|
let received_buffer_request_sender = client_output.received_buffer_request_sender;
|
||||||
|
|
||||||
let websocket_handler = websocket::Handler::new(
|
let websocket_handler = websocket::Handler::new(
|
||||||
msg_input,
|
input_sender,
|
||||||
client_connection_tx,
|
connection_command_sender,
|
||||||
buffer_requester,
|
received_buffer_request_sender,
|
||||||
&self.as_mix_recipient(),
|
self_address,
|
||||||
shared_lane_queue_lengths,
|
shared_lane_queue_lengths,
|
||||||
);
|
);
|
||||||
|
|
||||||
websocket::Listener::new(self.config.get_listening_port()).start(websocket_handler);
|
websocket::Listener::new(config.get_listening_port()).start(websocket_handler);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// blocking version of `start_socket` method. Will run forever (or until SIGINT is sent)
|
||||||
|
pub async fn run_socket_forever(self) -> Result<(), ClientError> {
|
||||||
|
let mut shutdown = self.start_socket().await?;
|
||||||
|
wait_for_signal().await;
|
||||||
|
|
||||||
|
println!(
|
||||||
|
"Received signal - the client will terminate now (threads are not yet nicely stopped, if you see stack traces that's alright)."
|
||||||
|
);
|
||||||
|
|
||||||
|
log::info!("Sending shutdown");
|
||||||
|
shutdown.signal_shutdown().ok();
|
||||||
|
|
||||||
|
// Some of these components have shutdown signalling implemented as part of socks5 work,
|
||||||
|
// but since it's not fully implemented (yet) for all the components of the native client,
|
||||||
|
// we don't try to wait and instead just stop immediately.
|
||||||
|
log::info!("Waiting for tasks to finish... (Press ctrl-c to force)");
|
||||||
|
shutdown.wait_for_shutdown().await;
|
||||||
|
|
||||||
|
log::info!("Stopping nym-client");
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn start_socket(self) -> Result<ShutdownNotifier, ClientError> {
|
||||||
|
if !self.config.get_socket_type().is_websocket() {
|
||||||
|
return Err(ClientError::InvalidSocketMode);
|
||||||
|
}
|
||||||
|
|
||||||
|
let base_builder = BaseClientBuilder::new_from_base_config(
|
||||||
|
self.config.get_base(),
|
||||||
|
self.key_manager,
|
||||||
|
Some(Self::create_bandwidth_controller(&self.config).await),
|
||||||
|
non_wasm_helpers::setup_fs_reply_surb_backend(
|
||||||
|
self.config.get_base().get_reply_surb_database_path(),
|
||||||
|
self.config.get_debug_settings(),
|
||||||
|
)
|
||||||
|
.await?,
|
||||||
|
);
|
||||||
|
|
||||||
|
let self_address = base_builder.as_mix_recipient();
|
||||||
|
let mut started_client = base_builder.start_base().await?;
|
||||||
|
let client_input = started_client.client_input.register_producer();
|
||||||
|
let client_output = started_client.client_output.register_consumer();
|
||||||
|
|
||||||
|
Self::start_websocket_listener(&self.config, client_input, client_output, self_address);
|
||||||
|
|
||||||
|
info!("Client startup finished!");
|
||||||
|
info!("The address of this client is: {}", self_address);
|
||||||
|
|
||||||
|
Ok(started_client.shutdown_notifier)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn start_direct(self) -> Result<DirectClient, ClientError> {
|
||||||
|
if self.config.get_socket_type().is_websocket() {
|
||||||
|
return Err(ClientError::InvalidSocketMode);
|
||||||
|
}
|
||||||
|
|
||||||
|
let base_client = BaseClientBuilder::new_from_base_config(
|
||||||
|
self.config.get_base(),
|
||||||
|
self.key_manager,
|
||||||
|
Some(Self::create_bandwidth_controller(&self.config).await),
|
||||||
|
non_wasm_helpers::setup_fs_reply_surb_backend(
|
||||||
|
self.config.get_base().get_reply_surb_database_path(),
|
||||||
|
self.config.get_debug_settings(),
|
||||||
|
)
|
||||||
|
.await?,
|
||||||
|
);
|
||||||
|
|
||||||
|
let mut started_client = base_client.start_base().await?;
|
||||||
|
let client_input = started_client.client_input.register_producer();
|
||||||
|
let client_output = started_client.client_output.register_consumer();
|
||||||
|
|
||||||
|
// register our receiver
|
||||||
|
let (reconstructed_sender, reconstructed_receiver) = mpsc::unbounded();
|
||||||
|
|
||||||
|
// tell the buffer to start sending stuff to us
|
||||||
|
client_output
|
||||||
|
.received_buffer_request_sender
|
||||||
|
.unbounded_send(ReceivedBufferMessage::ReceiverAnnounce(
|
||||||
|
reconstructed_sender,
|
||||||
|
))
|
||||||
|
.expect("the buffer request failed!");
|
||||||
|
|
||||||
|
Ok(DirectClient {
|
||||||
|
client_input,
|
||||||
|
reconstructed_receiver,
|
||||||
|
_shutdown_notifier: started_client.shutdown_notifier,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct DirectClient {
|
||||||
|
client_input: ClientInput,
|
||||||
|
reconstructed_receiver: ReconstructedMessagesReceiver,
|
||||||
|
|
||||||
|
// we need to keep reference to this guy otherwise things will start dropping
|
||||||
|
_shutdown_notifier: ShutdownNotifier,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl DirectClient {
|
||||||
/// EXPERIMENTAL DIRECT RUST API
|
/// EXPERIMENTAL DIRECT RUST API
|
||||||
/// It's untested and there are absolutely no guarantees about it (but seems to have worked
|
/// It's untested and there are absolutely no guarantees about it (but seems to have worked
|
||||||
/// well enough in local tests)
|
/// well enough in local tests)
|
||||||
pub async fn send_message(
|
pub async fn send_regular_message(&mut self, recipient: Recipient, message: Vec<u8>) {
|
||||||
&mut self,
|
|
||||||
recipient: Recipient,
|
|
||||||
message: Vec<u8>,
|
|
||||||
with_reply_surb: bool,
|
|
||||||
) {
|
|
||||||
let lane = TransmissionLane::General;
|
let lane = TransmissionLane::General;
|
||||||
let input_msg = InputMessage::new_fresh(recipient, message, with_reply_surb, lane);
|
let input_msg = InputMessage::new_regular(recipient, message, lane);
|
||||||
|
|
||||||
self.input_tx
|
self.client_input
|
||||||
.as_ref()
|
.input_sender
|
||||||
.expect("start method was not called before!")
|
|
||||||
.send(input_msg)
|
.send(input_msg)
|
||||||
.await
|
.await
|
||||||
.expect("InputMessageReceiver has stopped receiving!");
|
.expect("InputMessageReceiver has stopped receiving!");
|
||||||
@@ -338,12 +210,31 @@ impl NymClient {
|
|||||||
/// EXPERIMENTAL DIRECT RUST API
|
/// EXPERIMENTAL DIRECT RUST API
|
||||||
/// It's untested and there are absolutely no guarantees about it (but seems to have worked
|
/// It's untested and there are absolutely no guarantees about it (but seems to have worked
|
||||||
/// well enough in local tests)
|
/// well enough in local tests)
|
||||||
pub async fn send_reply(&mut self, reply_surb: ReplySurb, message: Vec<u8>) {
|
pub async fn send_anonymous_message(
|
||||||
let input_msg = InputMessage::new_reply(reply_surb, message);
|
&mut self,
|
||||||
|
recipient: Recipient,
|
||||||
|
message: Vec<u8>,
|
||||||
|
reply_surbs: u32,
|
||||||
|
) {
|
||||||
|
let lane = TransmissionLane::General;
|
||||||
|
let input_msg = InputMessage::new_anonymous(recipient, message, reply_surbs, lane);
|
||||||
|
|
||||||
self.input_tx
|
self.client_input
|
||||||
.as_ref()
|
.input_sender
|
||||||
.expect("start method was not called before!")
|
.send(input_msg)
|
||||||
|
.await
|
||||||
|
.expect("InputMessageReceiver has stopped receiving!");
|
||||||
|
}
|
||||||
|
|
||||||
|
/// EXPERIMENTAL DIRECT RUST API
|
||||||
|
/// It's untested and there are absolutely no guarantees about it (but seems to have worked
|
||||||
|
/// well enough in local tests)
|
||||||
|
pub async fn send_reply(&mut self, recipient_tag: AnonymousSenderTag, message: Vec<u8>) {
|
||||||
|
let lane = TransmissionLane::General;
|
||||||
|
let input_msg = InputMessage::new_reply(recipient_tag, message, lane);
|
||||||
|
|
||||||
|
self.client_input
|
||||||
|
.input_sender
|
||||||
.send(input_msg)
|
.send(input_msg)
|
||||||
.await
|
.await
|
||||||
.expect("InputMessageReceiver has stopped receiving!");
|
.expect("InputMessageReceiver has stopped receiving!");
|
||||||
@@ -358,147 +249,9 @@ impl NymClient {
|
|||||||
pub async fn wait_for_messages(&mut self) -> Vec<ReconstructedMessage> {
|
pub async fn wait_for_messages(&mut self) -> Vec<ReconstructedMessage> {
|
||||||
use futures::StreamExt;
|
use futures::StreamExt;
|
||||||
|
|
||||||
self.receive_tx
|
self.reconstructed_receiver
|
||||||
.as_mut()
|
|
||||||
.expect("start method was not called before!")
|
|
||||||
.next()
|
.next()
|
||||||
.await
|
.await
|
||||||
.expect("buffer controller seems to have somehow died!")
|
.expect("buffer controller seems to have somehow died!")
|
||||||
}
|
}
|
||||||
|
|
||||||
/// blocking version of `start` method. Will run forever (or until SIGINT is sent)
|
|
||||||
pub async fn run_forever(&mut self) -> Result<(), ClientError> {
|
|
||||||
let shutdown = self.start().await?;
|
|
||||||
wait_for_signal().await;
|
|
||||||
|
|
||||||
println!(
|
|
||||||
"Received signal - the client will terminate now (threads are not yet nicely stopped, if you see stack traces that's alright)."
|
|
||||||
);
|
|
||||||
|
|
||||||
log::info!("Sending shutdown");
|
|
||||||
shutdown.signal_shutdown().ok();
|
|
||||||
|
|
||||||
// Some of these components have shutdown signalling implemented as part of socks5 work,
|
|
||||||
// but since it's not fully implemented (yet) for all the components of the native client,
|
|
||||||
// we don't try to wait and instead just stop immediately.
|
|
||||||
//log::info!("Waiting for tasks to finish... (Press ctrl-c to force)");
|
|
||||||
//shutdown.wait_for_shutdown().await;
|
|
||||||
|
|
||||||
log::info!("Stopping nym-client");
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn start(&mut self) -> Result<ShutdownNotifier, ClientError> {
|
|
||||||
info!("Starting nym client");
|
|
||||||
// channels for inter-component communication
|
|
||||||
// TODO: make the channels be internally created by the relevant components
|
|
||||||
// rather than creating them here, so say for example the buffer controller would create the request channels
|
|
||||||
// and would allow anyone to clone the sender channel
|
|
||||||
|
|
||||||
// unwrapped_sphinx_sender is the transmitter of mixnet messages received from the gateway
|
|
||||||
// unwrapped_sphinx_receiver is the receiver for said messages - used by ReceivedMessagesBuffer
|
|
||||||
let (mixnet_messages_sender, mixnet_messages_receiver) = mpsc::unbounded();
|
|
||||||
|
|
||||||
// used for announcing connection or disconnection of a channel for pushing re-assembled messages to
|
|
||||||
let (received_buffer_request_sender, received_buffer_request_receiver) = mpsc::unbounded();
|
|
||||||
|
|
||||||
// channels responsible for controlling real messages
|
|
||||||
let (input_sender, input_receiver) = tokio::sync::mpsc::channel::<InputMessage>(1);
|
|
||||||
|
|
||||||
// channels responsible for controlling ack messages
|
|
||||||
let (ack_sender, ack_receiver) = mpsc::unbounded();
|
|
||||||
let shared_topology_accessor = TopologyAccessor::new();
|
|
||||||
|
|
||||||
let reply_key_storage =
|
|
||||||
ReplyKeyStorage::load(self.config.get_base().get_reply_encryption_key_store_path())
|
|
||||||
.tap_err(|err| {
|
|
||||||
log::error!("Failed to load reply key storage - is it perhaps already in use?");
|
|
||||||
log::error!("{}", err);
|
|
||||||
})?;
|
|
||||||
|
|
||||||
// Shutdown notifier for signalling tasks to stop
|
|
||||||
let shutdown = ShutdownNotifier::default();
|
|
||||||
|
|
||||||
// the components are started in very specific order. Unless you know what you are doing,
|
|
||||||
// do not change that.
|
|
||||||
self.start_topology_refresher(shared_topology_accessor.clone(), shutdown.subscribe())
|
|
||||||
.await?;
|
|
||||||
self.start_received_messages_buffer_controller(
|
|
||||||
received_buffer_request_receiver,
|
|
||||||
mixnet_messages_receiver,
|
|
||||||
reply_key_storage.clone(),
|
|
||||||
shutdown.subscribe(),
|
|
||||||
);
|
|
||||||
|
|
||||||
let gateway_client = self
|
|
||||||
.start_gateway_client(mixnet_messages_sender, ack_sender, shutdown.subscribe())
|
|
||||||
.await;
|
|
||||||
|
|
||||||
// The sphinx_message_sender is the transmitter for any component generating sphinx packets
|
|
||||||
// that are to be sent to the mixnet. They are used by cover traffic stream and real
|
|
||||||
// traffic stream.
|
|
||||||
// The MixTrafficController then sends the actual traffic
|
|
||||||
let sphinx_message_sender =
|
|
||||||
Self::start_mix_traffic_controller(gateway_client, shutdown.subscribe());
|
|
||||||
|
|
||||||
// Channels that the websocket listener can use to signal downstream to the real traffic
|
|
||||||
// controller that connections are closed.
|
|
||||||
let (client_connection_tx, client_connection_rx) = mpsc::unbounded();
|
|
||||||
|
|
||||||
// Shared queue length data. Published by the `OutQueueController` in the client, and used
|
|
||||||
// primarily to throttle incoming connections (e.g socks5 for attached network-requesters)
|
|
||||||
let shared_lane_queue_lengths = LaneQueueLengths::new();
|
|
||||||
|
|
||||||
self.start_real_traffic_controller(
|
|
||||||
shared_topology_accessor.clone(),
|
|
||||||
reply_key_storage,
|
|
||||||
ack_receiver,
|
|
||||||
input_receiver,
|
|
||||||
sphinx_message_sender.clone(),
|
|
||||||
shared_lane_queue_lengths.clone(),
|
|
||||||
client_connection_rx,
|
|
||||||
shutdown.subscribe(),
|
|
||||||
);
|
|
||||||
|
|
||||||
if !self
|
|
||||||
.config
|
|
||||||
.get_base()
|
|
||||||
.get_disabled_loop_cover_traffic_stream()
|
|
||||||
{
|
|
||||||
self.start_cover_traffic_stream(
|
|
||||||
shared_topology_accessor,
|
|
||||||
sphinx_message_sender,
|
|
||||||
shutdown.subscribe(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
match self.config.get_socket_type() {
|
|
||||||
SocketType::WebSocket => self.start_websocket_listener(
|
|
||||||
received_buffer_request_sender,
|
|
||||||
input_sender,
|
|
||||||
shared_lane_queue_lengths,
|
|
||||||
client_connection_tx,
|
|
||||||
),
|
|
||||||
SocketType::None => {
|
|
||||||
// if we did not start the socket, it means we're running (supposedly) in the native mode
|
|
||||||
// and hence we should announce 'ourselves' to the buffer
|
|
||||||
let (reconstructed_sender, reconstructed_receiver) = mpsc::unbounded();
|
|
||||||
|
|
||||||
// tell the buffer to start sending stuff to us
|
|
||||||
received_buffer_request_sender
|
|
||||||
.unbounded_send(ReceivedBufferMessage::ReceiverAnnounce(
|
|
||||||
reconstructed_sender,
|
|
||||||
))
|
|
||||||
.expect("the buffer request failed!");
|
|
||||||
|
|
||||||
self.receive_tx = Some(reconstructed_receiver);
|
|
||||||
self.input_tx = Some(input_sender);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
info!("Client startup finished!");
|
|
||||||
info!("The address of this client is: {}", self.as_mix_recipient());
|
|
||||||
|
|
||||||
Ok(shutdown)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,7 +2,8 @@
|
|||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use clap::Args;
|
use clap::Args;
|
||||||
use client_core::{config::GatewayEndpoint, error::ClientCoreError};
|
use client_core::client::replies::reply_storage::fs_backend;
|
||||||
|
use client_core::{config::GatewayEndpointConfig, error::ClientCoreError};
|
||||||
use config::NymConfig;
|
use config::NymConfig;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
@@ -46,6 +47,10 @@ pub(crate) struct Init {
|
|||||||
#[clap(long, hidden = true)]
|
#[clap(long, hidden = true)]
|
||||||
fastmode: bool,
|
fastmode: bool,
|
||||||
|
|
||||||
|
/// Disable loop cover traffic and the Poisson rate limiter (for debugging only)
|
||||||
|
#[clap(long, hidden = true)]
|
||||||
|
no_cover: bool,
|
||||||
|
|
||||||
/// Set this client to work in a enabled credentials mode that would attempt to use gateway
|
/// Set this client to work in a enabled credentials mode that would attempt to use gateway
|
||||||
/// with bandwidth credential requirement.
|
/// with bandwidth credential requirement.
|
||||||
#[cfg(feature = "coconut")]
|
#[cfg(feature = "coconut")]
|
||||||
@@ -61,6 +66,7 @@ impl From<Init> for OverrideConfig {
|
|||||||
disable_socket: init_config.disable_socket,
|
disable_socket: init_config.disable_socket,
|
||||||
port: init_config.port,
|
port: init_config.port,
|
||||||
fastmode: init_config.fastmode,
|
fastmode: init_config.fastmode,
|
||||||
|
no_cover: init_config.no_cover,
|
||||||
|
|
||||||
#[cfg(feature = "coconut")]
|
#[cfg(feature = "coconut")]
|
||||||
enabled_credentials_mode: init_config.enabled_credentials_mode,
|
enabled_credentials_mode: init_config.enabled_credentials_mode,
|
||||||
@@ -121,10 +127,12 @@ pub(crate) async fn execute(args: &Init) {
|
|||||||
);
|
);
|
||||||
println!("Client configuration completed.");
|
println!("Client configuration completed.");
|
||||||
|
|
||||||
client_core::init::show_address(config.get_base()).unwrap_or_else(|err| {
|
client_core::init::show_address::<_, fs_backend::Backend>(config.get_base()).unwrap_or_else(
|
||||||
eprintln!("Failed to show address\nError: {err}");
|
|err| {
|
||||||
std::process::exit(1)
|
eprintln!("Failed to show address\nError: {err}");
|
||||||
});
|
std::process::exit(1)
|
||||||
|
},
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn setup_gateway(
|
async fn setup_gateway(
|
||||||
@@ -132,7 +140,7 @@ async fn setup_gateway(
|
|||||||
register: bool,
|
register: bool,
|
||||||
user_chosen_gateway_id: Option<&str>,
|
user_chosen_gateway_id: Option<&str>,
|
||||||
config: &Config,
|
config: &Config,
|
||||||
) -> Result<GatewayEndpoint, ClientCoreError> {
|
) -> Result<GatewayEndpointConfig, ClientCoreError<fs_backend::Backend>> {
|
||||||
if register {
|
if register {
|
||||||
// Get the gateway details by querying the validator-api. Either pick one at random or use
|
// Get the gateway details by querying the validator-api. Either pick one at random or use
|
||||||
// the chosen one if it's among the available ones.
|
// the chosen one if it's among the available ones.
|
||||||
|
|||||||
@@ -80,6 +80,7 @@ pub(crate) struct OverrideConfig {
|
|||||||
disable_socket: bool,
|
disable_socket: bool,
|
||||||
port: Option<u16>,
|
port: Option<u16>,
|
||||||
fastmode: bool,
|
fastmode: bool,
|
||||||
|
no_cover: bool,
|
||||||
|
|
||||||
#[cfg(feature = "coconut")]
|
#[cfg(feature = "coconut")]
|
||||||
enabled_credentials_mode: bool,
|
enabled_credentials_mode: bool,
|
||||||
@@ -141,6 +142,10 @@ pub(crate) fn override_config(mut config: Config, args: OverrideConfig) -> Confi
|
|||||||
config.get_base_mut().set_high_default_traffic_volume();
|
config.get_base_mut().set_high_default_traffic_volume();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if args.no_cover {
|
||||||
|
config.get_base_mut().set_no_cover_traffic();
|
||||||
|
}
|
||||||
|
|
||||||
config
|
config
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
client::{config::Config, NymClient},
|
client::{config::Config, SocketClient},
|
||||||
commands::{override_config, OverrideConfig},
|
commands::{override_config, OverrideConfig},
|
||||||
error::ClientError,
|
error::ClientError,
|
||||||
};
|
};
|
||||||
@@ -39,6 +39,15 @@ pub(crate) struct Run {
|
|||||||
#[clap(short, long)]
|
#[clap(short, long)]
|
||||||
port: Option<u16>,
|
port: Option<u16>,
|
||||||
|
|
||||||
|
/// Mostly debug-related option to increase default traffic rate so that you would not need to
|
||||||
|
/// modify config post init
|
||||||
|
#[clap(long, hidden = true)]
|
||||||
|
fastmode: bool,
|
||||||
|
|
||||||
|
/// Disable loop cover traffic and the Poisson rate limiter (for debugging only)
|
||||||
|
#[clap(long, hidden = true)]
|
||||||
|
no_cover: bool,
|
||||||
|
|
||||||
/// Set this client to work in a enabled credentials mode that would attempt to use gateway
|
/// Set this client to work in a enabled credentials mode that would attempt to use gateway
|
||||||
/// with bandwidth credential requirement.
|
/// with bandwidth credential requirement.
|
||||||
#[cfg(feature = "coconut")]
|
#[cfg(feature = "coconut")]
|
||||||
@@ -53,7 +62,8 @@ impl From<Run> for OverrideConfig {
|
|||||||
api_validators: run_config.api_validators,
|
api_validators: run_config.api_validators,
|
||||||
disable_socket: run_config.disable_socket,
|
disable_socket: run_config.disable_socket,
|
||||||
port: run_config.port,
|
port: run_config.port,
|
||||||
fastmode: false,
|
fastmode: run_config.fastmode,
|
||||||
|
no_cover: run_config.no_cover,
|
||||||
#[cfg(feature = "coconut")]
|
#[cfg(feature = "coconut")]
|
||||||
enabled_credentials_mode: run_config.enabled_credentials_mode,
|
enabled_credentials_mode: run_config.enabled_credentials_mode,
|
||||||
}
|
}
|
||||||
@@ -98,5 +108,5 @@ pub(crate) async fn execute(args: &Run) -> Result<(), ClientError> {
|
|||||||
return Err(ClientError::FailedLocalVersionCheck);
|
return Err(ClientError::FailedLocalVersionCheck);
|
||||||
}
|
}
|
||||||
|
|
||||||
NymClient::new(config).run_forever().await
|
SocketClient::new(config).run_socket_forever().await
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,25 +1,17 @@
|
|||||||
use client_core::{client::reply_key_storage::ReplyKeyStorageError, error::ClientCoreError};
|
use client_core::client::replies::reply_storage::fs_backend;
|
||||||
use crypto::asymmetric::identity::Ed25519RecoveryError;
|
use client_core::error::ClientCoreError;
|
||||||
use gateway_client::error::GatewayClientError;
|
|
||||||
use validator_client::ValidatorClientError;
|
|
||||||
|
|
||||||
#[derive(thiserror::Error, Debug)]
|
#[derive(thiserror::Error, Debug)]
|
||||||
pub enum ClientError {
|
pub enum ClientError {
|
||||||
#[error("I/O error: {0}")]
|
|
||||||
IoError(#[from] std::io::Error),
|
|
||||||
#[error("Gateway client error: {0}")]
|
|
||||||
GatewayClientError(#[from] GatewayClientError),
|
|
||||||
#[error("Ed25519 error: {0}")]
|
|
||||||
Ed25519RecoveryError(#[from] Ed25519RecoveryError),
|
|
||||||
#[error("Validator client error: {0}")]
|
|
||||||
ValidatorClientError(#[from] ValidatorClientError),
|
|
||||||
#[error("client-core error: {0}")]
|
#[error("client-core error: {0}")]
|
||||||
ClientCoreError(#[from] ClientCoreError),
|
ClientCoreError(#[from] ClientCoreError<fs_backend::Backend>),
|
||||||
#[error("Reply key storage error: {0}")]
|
|
||||||
ReplyKeyStorageError(#[from] ReplyKeyStorageError),
|
|
||||||
|
|
||||||
#[error("Failed to load config for: {0}")]
|
#[error("Failed to load config for: {0}")]
|
||||||
FailedToLoadConfig(String),
|
FailedToLoadConfig(String),
|
||||||
|
|
||||||
#[error("Failed local version check, client and config mismatch")]
|
#[error("Failed local version check, client and config mismatch")]
|
||||||
FailedLocalVersionCheck,
|
FailedLocalVersionCheck,
|
||||||
|
|
||||||
|
#[error("Attempted to start the client in invalid socket mode")]
|
||||||
|
InvalidSocketMode,
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ use futures::channel::mpsc;
|
|||||||
use futures::{SinkExt, StreamExt};
|
use futures::{SinkExt, StreamExt};
|
||||||
use log::*;
|
use log::*;
|
||||||
use nymsphinx::addressing::clients::Recipient;
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
use nymsphinx::anonymous_replies::ReplySurb;
|
use nymsphinx::anonymous_replies::requests::AnonymousSenderTag;
|
||||||
use nymsphinx::receiver::ReconstructedMessage;
|
use nymsphinx::receiver::ReconstructedMessage;
|
||||||
use tokio::net::TcpStream;
|
use tokio::net::TcpStream;
|
||||||
use tokio_tungstenite::{
|
use tokio_tungstenite::{
|
||||||
@@ -62,9 +62,13 @@ impl Clone for Handler {
|
|||||||
|
|
||||||
impl Drop for Handler {
|
impl Drop for Handler {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
self.buffer_requester
|
if self
|
||||||
|
.buffer_requester
|
||||||
.unbounded_send(ReceivedBufferMessage::ReceiverDisconnect)
|
.unbounded_send(ReceivedBufferMessage::ReceiverDisconnect)
|
||||||
.expect("the buffer request failed!")
|
.is_err()
|
||||||
|
{
|
||||||
|
error!("we failed to disconnect the receiver from the buffer! presumably the shutdown procedure has been initiated!")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -73,14 +77,14 @@ impl Handler {
|
|||||||
msg_input: InputMessageSender,
|
msg_input: InputMessageSender,
|
||||||
client_connection_tx: ConnectionCommandSender,
|
client_connection_tx: ConnectionCommandSender,
|
||||||
buffer_requester: ReceivedBufferRequestSender,
|
buffer_requester: ReceivedBufferRequestSender,
|
||||||
self_full_address: &Recipient,
|
self_full_address: Recipient,
|
||||||
lane_queue_lengths: LaneQueueLengths,
|
lane_queue_lengths: LaneQueueLengths,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
Handler {
|
Handler {
|
||||||
msg_input,
|
msg_input,
|
||||||
client_connection_tx,
|
client_connection_tx,
|
||||||
buffer_requester,
|
buffer_requester,
|
||||||
self_full_address: *self_full_address,
|
self_full_address,
|
||||||
socket: None,
|
socket: None,
|
||||||
received_response_type: Default::default(),
|
received_response_type: Default::default(),
|
||||||
lane_queue_lengths,
|
lane_queue_lengths,
|
||||||
@@ -89,18 +93,22 @@ impl Handler {
|
|||||||
|
|
||||||
async fn handle_send(
|
async fn handle_send(
|
||||||
&mut self,
|
&mut self,
|
||||||
recipient: &Recipient,
|
recipient: Recipient,
|
||||||
message: Vec<u8>,
|
message: Vec<u8>,
|
||||||
with_reply_surb: bool,
|
|
||||||
connection_id: Option<u64>,
|
connection_id: Option<u64>,
|
||||||
) -> Option<ServerResponse> {
|
) -> Option<ServerResponse> {
|
||||||
|
info!(
|
||||||
|
"Attempting to send {:.2} kiB message to {recipient} on connection_id {connection_id:?}",
|
||||||
|
message.len() as f64 / 1024.0
|
||||||
|
);
|
||||||
|
|
||||||
// We map the absence of a connection id as going into the general lane.
|
// We map the absence of a connection id as going into the general lane.
|
||||||
let lane = connection_id.map_or(TransmissionLane::General, |id| {
|
let lane = connection_id.map_or(TransmissionLane::General, |id| {
|
||||||
TransmissionLane::ConnectionId(id)
|
TransmissionLane::ConnectionId(id)
|
||||||
});
|
});
|
||||||
|
|
||||||
// the ack control is now responsible for chunking, etc.
|
// the ack control is now responsible for chunking, etc.
|
||||||
let input_msg = InputMessage::new_fresh(*recipient, message, with_reply_surb, lane);
|
let input_msg = InputMessage::new_regular(recipient, message, lane);
|
||||||
self.msg_input
|
self.msg_input
|
||||||
.send(input_msg)
|
.send(input_msg)
|
||||||
.await
|
.await
|
||||||
@@ -109,53 +117,119 @@ impl Handler {
|
|||||||
// Only reply back with a `LaneQueueLength` if the sender providided a connection id
|
// Only reply back with a `LaneQueueLength` if the sender providided a connection id
|
||||||
let connection_id = match lane {
|
let connection_id = match lane {
|
||||||
TransmissionLane::General
|
TransmissionLane::General
|
||||||
| TransmissionLane::Reply
|
| TransmissionLane::ReplySurbRequest
|
||||||
| TransmissionLane::Retransmission
|
| TransmissionLane::Retransmission
|
||||||
| TransmissionLane::Control => return None,
|
| TransmissionLane::AdditionalReplySurbs => return None,
|
||||||
TransmissionLane::ConnectionId(id) => id,
|
TransmissionLane::ConnectionId(id) => id,
|
||||||
};
|
};
|
||||||
|
|
||||||
// on receiving a send, we reply back the current lane queue length for that connection id.
|
// on receiving a send, we reply back the current lane queue length for that connection id.
|
||||||
// Note that this does _NOT_ take into account the packets that have been received but not
|
// Note that this does _NOT_ take into account the packets that have been received but not
|
||||||
// yet reach `OutQueueControl`, so it might be a tad low.
|
// yet reach `OutQueueControl`, so it might be a tad low.
|
||||||
let Ok(lane_queue_lengths) = self.lane_queue_lengths.lock() else {
|
if let Ok(lane_queue_lengths) = self.lane_queue_lengths.lock() {
|
||||||
log::warn!(
|
let queue_length = lane_queue_lengths.get(&lane).unwrap_or(0);
|
||||||
"Failed to get the lane queue length lock, \
|
return Some(ServerResponse::LaneQueueLength {
|
||||||
not responding back with the current queue length"
|
lane: connection_id,
|
||||||
);
|
queue_length,
|
||||||
return None;
|
});
|
||||||
};
|
|
||||||
|
|
||||||
let queue_length = lane_queue_lengths.get(&lane).unwrap_or(0);
|
|
||||||
Some(ServerResponse::LaneQueueLength(connection_id, queue_length))
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn handle_reply(
|
|
||||||
&mut self,
|
|
||||||
reply_surb: ReplySurb,
|
|
||||||
message: Vec<u8>,
|
|
||||||
) -> Option<ServerResponse> {
|
|
||||||
if message.len() > ReplySurb::max_msg_len(Default::default()) {
|
|
||||||
return Some(
|
|
||||||
ServerResponse::new_error(
|
|
||||||
format!(
|
|
||||||
"too long message to put inside a reply SURB. Received: {} bytes and maximum is {} bytes",
|
|
||||||
message.len(), ReplySurb::max_msg_len(Default::default()))
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let input_msg = InputMessage::new_reply(reply_surb, message);
|
log::warn!("Failed to get the lane queue length lock, not responding back with the current queue length");
|
||||||
|
None
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn handle_send_anonymous(
|
||||||
|
&mut self,
|
||||||
|
recipient: Recipient,
|
||||||
|
message: Vec<u8>,
|
||||||
|
reply_surbs: u32,
|
||||||
|
connection_id: Option<u64>,
|
||||||
|
) -> Option<ServerResponse> {
|
||||||
|
info!(
|
||||||
|
"Attempting to anonymously send {:.2} kiB message to {recipient} on connection_id {connection_id:?} while attaching {reply_surbs} replySURBs.",
|
||||||
|
message.len() as f64 / 1024.0
|
||||||
|
);
|
||||||
|
|
||||||
|
// We map the absence of a connection id as going into the general lane.
|
||||||
|
let lane = connection_id.map_or(TransmissionLane::General, |id| {
|
||||||
|
TransmissionLane::ConnectionId(id)
|
||||||
|
});
|
||||||
|
|
||||||
|
let input_msg = InputMessage::new_anonymous(recipient, message, reply_surbs, lane);
|
||||||
self.msg_input
|
self.msg_input
|
||||||
.send(input_msg)
|
.send(input_msg)
|
||||||
.await
|
.await
|
||||||
.expect("InputMessageReceiver has stopped receiving!");
|
.expect("InputMessageReceiver has stopped receiving!");
|
||||||
|
|
||||||
|
// Only reply back with a `LaneQueueLength` if the sender providided a connection id
|
||||||
|
let connection_id = match lane {
|
||||||
|
TransmissionLane::General
|
||||||
|
| TransmissionLane::ReplySurbRequest
|
||||||
|
| TransmissionLane::Retransmission
|
||||||
|
| TransmissionLane::AdditionalReplySurbs => return None,
|
||||||
|
TransmissionLane::ConnectionId(id) => id,
|
||||||
|
};
|
||||||
|
|
||||||
|
// on receiving a send, we reply back the current lane queue length for that connection id.
|
||||||
|
// Note that this does _NOT_ take into account the packets that have been received but not
|
||||||
|
// yet reach `OutQueueControl`, so it might be a tad low.
|
||||||
|
if let Ok(lane_queue_lengths) = self.lane_queue_lengths.lock() {
|
||||||
|
let queue_length = lane_queue_lengths.get(&lane).unwrap_or(0);
|
||||||
|
return Some(ServerResponse::LaneQueueLength {
|
||||||
|
lane: connection_id,
|
||||||
|
queue_length,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
log::warn!("Failed to get the lane queue length lock, not responding back with the current queue length");
|
||||||
|
None
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn handle_reply(
|
||||||
|
&mut self,
|
||||||
|
recipient_tag: AnonymousSenderTag,
|
||||||
|
message: Vec<u8>,
|
||||||
|
connection_id: Option<u64>,
|
||||||
|
) -> Option<ServerResponse> {
|
||||||
|
info!("Attempting to send {:.2} kiB reply message to {recipient_tag} on connection_id {connection_id:?}", message.len() as f64 / 1024.0);
|
||||||
|
|
||||||
|
// We map the absence of a connection id as going into the general lane.
|
||||||
|
let lane = connection_id.map_or(TransmissionLane::General, |id| {
|
||||||
|
TransmissionLane::ConnectionId(id)
|
||||||
|
});
|
||||||
|
|
||||||
|
let input_msg = InputMessage::new_reply(recipient_tag, message, lane);
|
||||||
|
self.msg_input
|
||||||
|
.send(input_msg)
|
||||||
|
.await
|
||||||
|
.expect("InputMessageReceiver has stopped receiving!");
|
||||||
|
|
||||||
|
// Only reply back with a `LaneQueueLength` if the sender providided a connection id
|
||||||
|
let connection_id = match lane {
|
||||||
|
TransmissionLane::General
|
||||||
|
| TransmissionLane::ReplySurbRequest
|
||||||
|
| TransmissionLane::Retransmission
|
||||||
|
| TransmissionLane::AdditionalReplySurbs => return None,
|
||||||
|
TransmissionLane::ConnectionId(id) => id,
|
||||||
|
};
|
||||||
|
|
||||||
|
// on receiving a send, we reply back the current lane queue length for that connection id.
|
||||||
|
// Note that this does _NOT_ take into account the packets that have been received but not
|
||||||
|
// yet reach `OutQueueControl`, so it might be a tad low.
|
||||||
|
if let Ok(lane_queue_lengths) = self.lane_queue_lengths.lock() {
|
||||||
|
let queue_length = lane_queue_lengths.get(&lane).unwrap_or(0);
|
||||||
|
return Some(ServerResponse::LaneQueueLength {
|
||||||
|
lane: connection_id,
|
||||||
|
queue_length,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
log::warn!("Failed to get the lane queue length lock, not responding back with the current queue length");
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
|
|
||||||
fn handle_self_address(&self) -> ServerResponse {
|
fn handle_self_address(&self) -> ServerResponse {
|
||||||
ServerResponse::SelfAddress(self.self_full_address)
|
ServerResponse::SelfAddress(Box::new(self.self_full_address))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn handle_closed_connection(&self, connection_id: u64) -> Option<ServerResponse> {
|
fn handle_closed_connection(&self, connection_id: u64) -> Option<ServerResponse> {
|
||||||
@@ -175,7 +249,10 @@ impl Handler {
|
|||||||
|
|
||||||
let lane = TransmissionLane::ConnectionId(connection_id);
|
let lane = TransmissionLane::ConnectionId(connection_id);
|
||||||
let queue_length = lane_queue_lengths.get(&lane).unwrap_or(0);
|
let queue_length = lane_queue_lengths.get(&lane).unwrap_or(0);
|
||||||
Some(ServerResponse::LaneQueueLength(connection_id, queue_length))
|
Some(ServerResponse::LaneQueueLength {
|
||||||
|
lane: connection_id,
|
||||||
|
queue_length,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn handle_request(&mut self, request: ClientRequest) -> Option<ServerResponse> {
|
async fn handle_request(&mut self, request: ClientRequest) -> Option<ServerResponse> {
|
||||||
@@ -183,16 +260,25 @@ impl Handler {
|
|||||||
ClientRequest::Send {
|
ClientRequest::Send {
|
||||||
recipient,
|
recipient,
|
||||||
message,
|
message,
|
||||||
with_reply_surb,
|
connection_id,
|
||||||
|
} => self.handle_send(recipient, message, connection_id).await,
|
||||||
|
|
||||||
|
ClientRequest::SendAnonymous {
|
||||||
|
recipient,
|
||||||
|
message,
|
||||||
|
reply_surbs,
|
||||||
connection_id,
|
connection_id,
|
||||||
} => {
|
} => {
|
||||||
self.handle_send(&recipient, message, with_reply_surb, connection_id)
|
self.handle_send_anonymous(recipient, message, reply_surbs, connection_id)
|
||||||
.await
|
.await
|
||||||
}
|
}
|
||||||
|
|
||||||
ClientRequest::Reply {
|
ClientRequest::Reply {
|
||||||
message,
|
message,
|
||||||
reply_surb,
|
sender_tag,
|
||||||
} => self.handle_reply(reply_surb, message).await,
|
connection_id,
|
||||||
|
} => self.handle_reply(sender_tag, message, connection_id).await,
|
||||||
|
|
||||||
ClientRequest::SelfAddress => Some(self.handle_self_address()),
|
ClientRequest::SelfAddress => Some(self.handle_self_address()),
|
||||||
ClientRequest::ClosedConnection(id) => self.handle_closed_connection(id),
|
ClientRequest::ClosedConnection(id) => self.handle_closed_connection(id),
|
||||||
ClientRequest::GetLaneQueueLength(id) => self.handle_get_lane_queue_length(id),
|
ClientRequest::GetLaneQueueLength(id) => self.handle_get_lane_queue_length(id),
|
||||||
@@ -299,8 +385,7 @@ impl Handler {
|
|||||||
if let Some(response) = self.handle_ws_request(socket_msg).await {
|
if let Some(response) = self.handle_ws_request(socket_msg).await {
|
||||||
if let Err(err) = self.send_websocket_response(response).await {
|
if let Err(err) = self.send_websocket_response(response).await {
|
||||||
warn!(
|
warn!(
|
||||||
"Failed to send message over websocket: {}. Assuming the connection is dead.",
|
"Failed to send message over websocket: {err}. Assuming the connection is dead.",
|
||||||
err
|
|
||||||
);
|
);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -308,9 +393,10 @@ impl Handler {
|
|||||||
}
|
}
|
||||||
// or a reconstructed mix message that we need to push back to the client
|
// or a reconstructed mix message that we need to push back to the client
|
||||||
mix_messages = msg_receiver.next() => {
|
mix_messages = msg_receiver.next() => {
|
||||||
let mix_messages = mix_messages.expect(
|
let Some(mix_messages) = mix_messages else {
|
||||||
"mix messages sender was unexpectedly closed! this shouldn't have ever happened!",
|
error!("mix messages sender was unexpectedly closed! this shouldn't have ever happened! (unless we're shutting down - TODO: implement proper graceful shutdown handler)");
|
||||||
);
|
return
|
||||||
|
};
|
||||||
if let Err(e) = self.push_websocket_received_plaintexts(mix_messages).await {
|
if let Err(e) = self.push_websocket_received_plaintexts(mix_messages).await {
|
||||||
warn!("failed to send sphinx packets back to the client - {:?}, assuming the connection is dead", e);
|
warn!("failed to send sphinx packets back to the client - {:?}, assuming the connection is dead", e);
|
||||||
break;
|
break;
|
||||||
|
|||||||
@@ -54,9 +54,8 @@ impl Listener {
|
|||||||
Ok((mut socket, remote_addr)) => {
|
Ok((mut socket, remote_addr)) => {
|
||||||
debug!("Received connection from {:?}", remote_addr);
|
debug!("Received connection from {:?}", remote_addr);
|
||||||
if self.state.is_connected() {
|
if self.state.is_connected() {
|
||||||
warn!("tried to duplicate!");
|
warn!("Tried to open a duplicate websocket connection. The request came from {}", remote_addr);
|
||||||
// if we've already got a connection, don't allow another one
|
// if we've already got a connection, don't allow another one
|
||||||
debug!("but there was already a connection present!");
|
|
||||||
// while we only ever want to accept a single connection, we don't want
|
// while we only ever want to accept a single connection, we don't want
|
||||||
// to leave clients hanging (and also allow for reconnection if it somehow
|
// to leave clients hanging (and also allow for reconnection if it somehow
|
||||||
// was dropped)
|
// was dropped)
|
||||||
|
|||||||
@@ -24,8 +24,11 @@ impl fmt::Debug for Error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl Error {
|
impl Error {
|
||||||
pub fn new(kind: ErrorKind, message: String) -> Self {
|
pub fn new<S: Into<String>>(kind: ErrorKind, message: S) -> Self {
|
||||||
Error { kind, message }
|
Error {
|
||||||
|
kind,
|
||||||
|
message: message.into(),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -62,6 +65,31 @@ pub enum ErrorKind {
|
|||||||
Other = 0xFF,
|
Other = 0xFF,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl TryFrom<u8> for ErrorKind {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn try_from(value: u8) -> Result<Self, Self::Error> {
|
||||||
|
match value {
|
||||||
|
_ if value == (ErrorKind::EmptyRequest as u8) => Ok(ErrorKind::EmptyRequest),
|
||||||
|
_ if value == (ErrorKind::TooShortRequest as u8) => Ok(ErrorKind::TooShortRequest),
|
||||||
|
_ if value == (ErrorKind::UnknownRequest as u8) => Ok(ErrorKind::UnknownRequest),
|
||||||
|
_ if value == (ErrorKind::MalformedRequest as u8) => Ok(ErrorKind::MalformedRequest),
|
||||||
|
|
||||||
|
_ if value == (ErrorKind::EmptyResponse as u8) => Ok(ErrorKind::EmptyResponse),
|
||||||
|
_ if value == (ErrorKind::TooShortResponse as u8) => Ok(ErrorKind::TooShortResponse),
|
||||||
|
_ if value == (ErrorKind::UnknownResponse as u8) => Ok(ErrorKind::UnknownResponse),
|
||||||
|
_ if value == (ErrorKind::MalformedResponse as u8) => Ok(ErrorKind::MalformedResponse),
|
||||||
|
|
||||||
|
_ if value == (ErrorKind::Other as u8) => Ok(ErrorKind::Other),
|
||||||
|
|
||||||
|
n => Err(Error::new(
|
||||||
|
ErrorKind::MalformedResponse,
|
||||||
|
format!("invalid error code {}", n),
|
||||||
|
)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl ErrorKind {
|
impl ErrorKind {
|
||||||
pub(crate) fn as_str(&self) -> &'static str {
|
pub(crate) fn as_str(&self) -> &'static str {
|
||||||
match *self {
|
match *self {
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
// Copyright 2021-2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
// all variable size data is always prefixed with u64 length
|
// all variable size data is always prefixed with u64 length
|
||||||
@@ -7,69 +7,115 @@
|
|||||||
use crate::error::{self, ErrorKind};
|
use crate::error::{self, ErrorKind};
|
||||||
use crate::text::ClientRequestText;
|
use crate::text::ClientRequestText;
|
||||||
use nymsphinx::addressing::clients::Recipient;
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
use nymsphinx::anonymous_replies::ReplySurb;
|
use nymsphinx::anonymous_replies::requests::{AnonymousSenderTag, SENDER_TAG_SIZE};
|
||||||
use std::convert::{TryFrom, TryInto};
|
use std::convert::{TryFrom, TryInto};
|
||||||
use std::mem::size_of;
|
use std::mem::size_of;
|
||||||
|
|
||||||
/// Value tag representing [`Send`] variant of the [`ClientRequest`]
|
#[repr(u8)]
|
||||||
pub const SEND_REQUEST_TAG: u8 = 0x00;
|
enum ClientRequestTag {
|
||||||
|
/// Value tag representing [`Send`] variant of the [`ClientRequest`]
|
||||||
|
Send = 0x00,
|
||||||
|
|
||||||
/// Value tag representing [`Reply`] variant of the [`ClientRequest`]
|
/// Value tag representing [`SendAnonymous`] variant of the [`ClientRequest`]
|
||||||
pub const REPLY_REQUEST_TAG: u8 = 0x01;
|
SendAnonymous = 0x01,
|
||||||
|
|
||||||
/// Value tag representing [`SelfAddress`] variant of the [`ClientRequest`]
|
/// Value tag representing [`Reply`] variant of the [`ClientRequest`]
|
||||||
pub const SELF_ADDRESS_REQUEST_TAG: u8 = 0x02;
|
Reply = 0x02,
|
||||||
|
|
||||||
/// Value tag representing [`ClosedConnection`] variant of the [`ClientRequest`]
|
/// Value tag representing [`SelfAddress`] variant of the [`ClientRequest`]
|
||||||
pub const CLOSED_CONNECTION_REQUEST_TAG: u8 = 0x03;
|
SelfAddress = 0x03,
|
||||||
|
|
||||||
/// Value tag representing [`GetLaneQueueLength`] variant of the [`ClientRequest`]
|
/// Value tag representing [`ClosedConnection`] variant of the [`ClientRequest`]
|
||||||
pub const GET_LANE_QUEUE_LENGHT_TAG: u8 = 0x04;
|
ClosedConnection = 0x04,
|
||||||
|
|
||||||
|
/// Value tag representing [`GetLaneQueueLength`] variant of the [`ClientRequest`]
|
||||||
|
GetLaneQueueLength = 0x05,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TryFrom<u8> for ClientRequestTag {
|
||||||
|
type Error = error::Error;
|
||||||
|
|
||||||
|
fn try_from(value: u8) -> Result<Self, Self::Error> {
|
||||||
|
match value {
|
||||||
|
_ if value == (Self::Send as u8) => Ok(Self::Send),
|
||||||
|
_ if value == (Self::SendAnonymous as u8) => Ok(Self::SendAnonymous),
|
||||||
|
_ if value == (Self::Reply as u8) => Ok(Self::Reply),
|
||||||
|
_ if value == (Self::SelfAddress as u8) => Ok(Self::SelfAddress),
|
||||||
|
_ if value == (Self::ClosedConnection as u8) => Ok(Self::ClosedConnection),
|
||||||
|
_ if value == (Self::GetLaneQueueLength as u8) => Ok(Self::GetLaneQueueLength),
|
||||||
|
n => Err(error::Error::new(
|
||||||
|
ErrorKind::UnknownRequest,
|
||||||
|
format!("{n} does not correspond to any valid request tag"),
|
||||||
|
)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[allow(non_snake_case)]
|
#[allow(non_snake_case)]
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum ClientRequest {
|
pub enum ClientRequest {
|
||||||
|
/// The simplest message variant where no additional information is attached.
|
||||||
|
/// You're simply sending your `data` to specified `recipient` without any tagging.
|
||||||
|
///
|
||||||
|
/// Ends up with `NymMessage::Plain` variant
|
||||||
Send {
|
Send {
|
||||||
recipient: Recipient,
|
recipient: Recipient,
|
||||||
message: Vec<u8>,
|
message: Vec<u8>,
|
||||||
// Perhaps we could change it to a number to indicate how many reply_SURBs we want to include?
|
|
||||||
with_reply_surb: bool,
|
|
||||||
connection_id: Option<u64>,
|
connection_id: Option<u64>,
|
||||||
},
|
},
|
||||||
Reply {
|
|
||||||
|
/// Create a message used for a duplex anonymous communication where the recipient
|
||||||
|
/// will never learn of our true identity. This is achieved by carefully sending `reply_surbs`.
|
||||||
|
///
|
||||||
|
/// Note that if reply_surbs is set to zero then
|
||||||
|
/// this variant requires the client having sent some reply_surbs in the past
|
||||||
|
/// (and thus the recipient also knowing our sender tag).
|
||||||
|
///
|
||||||
|
/// Ends up with `NymMessage::Repliable` variant
|
||||||
|
SendAnonymous {
|
||||||
|
recipient: Recipient,
|
||||||
message: Vec<u8>,
|
message: Vec<u8>,
|
||||||
reply_surb: ReplySurb,
|
reply_surbs: u32,
|
||||||
|
connection_id: Option<u64>,
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/// Attempt to use our internally received and stored `ReplySurb` to send the message back
|
||||||
|
/// to specified recipient whilst not knowing its full identity (or even gateway).
|
||||||
|
///
|
||||||
|
/// Ends up with `NymMessage::Reply` variant
|
||||||
|
Reply {
|
||||||
|
sender_tag: AnonymousSenderTag,
|
||||||
|
message: Vec<u8>,
|
||||||
|
connection_id: Option<u64>,
|
||||||
|
},
|
||||||
|
|
||||||
SelfAddress,
|
SelfAddress,
|
||||||
|
|
||||||
ClosedConnection(u64),
|
ClosedConnection(u64),
|
||||||
|
|
||||||
GetLaneQueueLength(u64),
|
GetLaneQueueLength(u64),
|
||||||
}
|
}
|
||||||
|
|
||||||
// we could have been parsing it directly TryFrom<WsMessage>, but we want to retain
|
// we could have been parsing it directly TryFrom<WsMessage>, but we want to retain
|
||||||
// information about whether it came from binary or text to send appropriate response back
|
// information about whether it came from binary or text to send appropriate response back
|
||||||
impl ClientRequest {
|
impl ClientRequest {
|
||||||
// SEND_REQUEST_TAG || with_surb || recipient || conn_id || data_len || data
|
// SEND_REQUEST_TAG || recipient || conn_id || data_len || data
|
||||||
fn serialize_send(
|
fn serialize_send(recipient: Recipient, data: Vec<u8>, connection_id: Option<u64>) -> Vec<u8> {
|
||||||
recipient: Recipient,
|
|
||||||
data: Vec<u8>,
|
|
||||||
with_reply_surb: bool,
|
|
||||||
connection_id: Option<u64>,
|
|
||||||
) -> Vec<u8> {
|
|
||||||
let data_len_bytes = (data.len() as u64).to_be_bytes();
|
let data_len_bytes = (data.len() as u64).to_be_bytes();
|
||||||
let conn_id_bytes = connection_id.unwrap_or(0).to_be_bytes();
|
let conn_id_bytes = connection_id.unwrap_or(0).to_be_bytes();
|
||||||
std::iter::once(SEND_REQUEST_TAG)
|
|
||||||
.chain(std::iter::once(with_reply_surb as u8))
|
std::iter::once(ClientRequestTag::Send as u8)
|
||||||
.chain(recipient.to_bytes().iter().cloned()) // will not be length prefixed because the length is constant
|
.chain(recipient.to_bytes().into_iter()) // will not be length prefixed because the length is constant
|
||||||
.chain(conn_id_bytes.iter().cloned())
|
.chain(conn_id_bytes.into_iter())
|
||||||
.chain(data_len_bytes.iter().cloned())
|
.chain(data_len_bytes.into_iter())
|
||||||
.chain(data.into_iter())
|
.chain(data.into_iter())
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
// SEND_REQUEST_TAG || with_reply || recipient || conn_id || data_len || data
|
// SEND_REQUEST_TAG || recipient || conn_id || data_len || data
|
||||||
fn deserialize_send(b: &[u8]) -> Result<Self, error::Error> {
|
fn deserialize_send(b: &[u8]) -> Result<Self, error::Error> {
|
||||||
// we need to have at least 1 (tag) + 1 (reply flag) + Recipient::LEN + 2*sizeof<u64> bytes
|
// we need to have at least 1 (tag) + Recipient::LEN + 2*sizeof<u64> bytes
|
||||||
if b.len() < 2 + Recipient::LEN + 2 * size_of::<u64>() {
|
if b.len() < 1 + Recipient::LEN + 2 * size_of::<u64>() {
|
||||||
return Err(error::Error::new(
|
return Err(error::Error::new(
|
||||||
ErrorKind::TooShortRequest,
|
ErrorKind::TooShortRequest,
|
||||||
"not enough data provided to recover 'send'".to_string(),
|
"not enough data provided to recover 'send'".to_string(),
|
||||||
@@ -77,21 +123,10 @@ impl ClientRequest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// this MUST match because it was called by 'deserialize'
|
// this MUST match because it was called by 'deserialize'
|
||||||
debug_assert_eq!(b[0], SEND_REQUEST_TAG);
|
debug_assert_eq!(b[0], ClientRequestTag::Send as u8);
|
||||||
|
|
||||||
let with_reply_surb = match b[1] {
|
|
||||||
0 => false,
|
|
||||||
1 => true,
|
|
||||||
n => {
|
|
||||||
return Err(error::Error::new(
|
|
||||||
ErrorKind::MalformedRequest,
|
|
||||||
format!("invalid reply surb flag {}", n),
|
|
||||||
))
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut recipient_bytes = [0u8; Recipient::LEN];
|
let mut recipient_bytes = [0u8; Recipient::LEN];
|
||||||
recipient_bytes.copy_from_slice(&b[2..2 + Recipient::LEN]);
|
recipient_bytes.copy_from_slice(&b[1..1 + Recipient::LEN]);
|
||||||
let recipient = match Recipient::try_from_bytes(recipient_bytes) {
|
let recipient = match Recipient::try_from_bytes(recipient_bytes) {
|
||||||
Ok(recipient) => recipient,
|
Ok(recipient) => recipient,
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
@@ -104,7 +139,7 @@ impl ClientRequest {
|
|||||||
|
|
||||||
let mut connection_id_bytes = [0u8; size_of::<u64>()];
|
let mut connection_id_bytes = [0u8; size_of::<u64>()];
|
||||||
connection_id_bytes
|
connection_id_bytes
|
||||||
.copy_from_slice(&b[2 + Recipient::LEN..2 + Recipient::LEN + size_of::<u64>()]);
|
.copy_from_slice(&b[1 + Recipient::LEN..1 + Recipient::LEN + size_of::<u64>()]);
|
||||||
let connection_id = u64::from_be_bytes(connection_id_bytes);
|
let connection_id = u64::from_be_bytes(connection_id_bytes);
|
||||||
let connection_id = if connection_id == 0 {
|
let connection_id = if connection_id == 0 {
|
||||||
None
|
None
|
||||||
@@ -113,9 +148,9 @@ impl ClientRequest {
|
|||||||
};
|
};
|
||||||
|
|
||||||
let data_len_bytes =
|
let data_len_bytes =
|
||||||
&b[2 + Recipient::LEN + size_of::<u64>()..2 + Recipient::LEN + 2 * size_of::<u64>()];
|
&b[1 + Recipient::LEN + size_of::<u64>()..1 + Recipient::LEN + 2 * size_of::<u64>()];
|
||||||
let data_len = u64::from_be_bytes(data_len_bytes.try_into().unwrap());
|
let data_len = u64::from_be_bytes(data_len_bytes.try_into().unwrap());
|
||||||
let data = &b[2 + Recipient::LEN + 2 * size_of::<u64>()..];
|
let data = &b[1 + Recipient::LEN + 2 * size_of::<u64>()..];
|
||||||
if data.len() as u64 != data_len {
|
if data.len() as u64 != data_len {
|
||||||
return Err(error::Error::new(
|
return Err(error::Error::new(
|
||||||
ErrorKind::MalformedRequest,
|
ErrorKind::MalformedRequest,
|
||||||
@@ -128,33 +163,111 @@ impl ClientRequest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Ok(ClientRequest::Send {
|
Ok(ClientRequest::Send {
|
||||||
with_reply_surb,
|
|
||||||
recipient,
|
recipient,
|
||||||
message: data.to_vec(),
|
message: data.to_vec(),
|
||||||
connection_id,
|
connection_id,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// REPLY_REQUEST_TAG || surb_len || surb || message_len || message
|
// SEND_ANONYMOUS_REQUEST_TAG || reply_surbs || recipient || conn_id || data_len || data
|
||||||
fn serialize_reply(message: Vec<u8>, reply_surb: &ReplySurb) -> Vec<u8> {
|
fn serialize_send_anonymous(
|
||||||
let reply_surb_bytes = reply_surb.to_bytes();
|
recipient: Recipient,
|
||||||
let surb_len_bytes = (reply_surb_bytes.len() as u64).to_be_bytes();
|
data: Vec<u8>,
|
||||||
let message_len_bytes = (message.len() as u64).to_be_bytes();
|
reply_surbs: u32,
|
||||||
|
connection_id: Option<u64>,
|
||||||
|
) -> Vec<u8> {
|
||||||
|
let data_len_bytes = (data.len() as u64).to_be_bytes();
|
||||||
|
let conn_id_bytes = connection_id.unwrap_or(0).to_be_bytes();
|
||||||
|
|
||||||
std::iter::once(REPLY_REQUEST_TAG)
|
std::iter::once(ClientRequestTag::SendAnonymous as u8)
|
||||||
.chain(surb_len_bytes.iter().cloned())
|
.chain(reply_surbs.to_be_bytes().into_iter())
|
||||||
.chain(reply_surb_bytes.into_iter())
|
.chain(recipient.to_bytes().into_iter()) // will not be length prefixed because the length is constant
|
||||||
.chain(message_len_bytes.iter().cloned())
|
.chain(conn_id_bytes.into_iter())
|
||||||
|
.chain(data_len_bytes.into_iter())
|
||||||
|
.chain(data.into_iter())
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
// SEND_ANONYMOUS_REQUEST_TAG || reply_surbs || recipient || data_len || data
|
||||||
|
fn deserialize_send_anonymous(b: &[u8]) -> Result<Self, error::Error> {
|
||||||
|
// we need to have at least 1 (tag) + sizeof<u32> (num surbs) + Recipient::LEN + 2 *sizeof<u64> bytes
|
||||||
|
if b.len() < 1 + size_of::<u32>() + Recipient::LEN + 2 * size_of::<u64>() {
|
||||||
|
return Err(error::Error::new(
|
||||||
|
ErrorKind::TooShortRequest,
|
||||||
|
"not enough data provided to recover 'send_anonymous'".to_string(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
// this MUST match because it was called by 'deserialize'
|
||||||
|
debug_assert_eq!(b[0], ClientRequestTag::SendAnonymous as u8);
|
||||||
|
|
||||||
|
let reply_surbs = u32::from_be_bytes([b[1], b[2], b[3], b[4]]);
|
||||||
|
|
||||||
|
let mut recipient_bytes = [0u8; Recipient::LEN];
|
||||||
|
recipient_bytes.copy_from_slice(&b[5..5 + Recipient::LEN]);
|
||||||
|
let recipient = match Recipient::try_from_bytes(recipient_bytes) {
|
||||||
|
Ok(recipient) => recipient,
|
||||||
|
Err(err) => {
|
||||||
|
return Err(error::Error::new(
|
||||||
|
ErrorKind::MalformedRequest,
|
||||||
|
format!("malformed recipient: {:?}", err),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut connection_id_bytes = [0u8; size_of::<u64>()];
|
||||||
|
connection_id_bytes
|
||||||
|
.copy_from_slice(&b[5 + Recipient::LEN..5 + Recipient::LEN + size_of::<u64>()]);
|
||||||
|
let connection_id = u64::from_be_bytes(connection_id_bytes);
|
||||||
|
let connection_id = if connection_id == 0 {
|
||||||
|
None
|
||||||
|
} else {
|
||||||
|
Some(connection_id)
|
||||||
|
};
|
||||||
|
|
||||||
|
let data_len_bytes =
|
||||||
|
&b[5 + Recipient::LEN + size_of::<u64>()..5 + Recipient::LEN + 2 * size_of::<u64>()];
|
||||||
|
let data_len = u64::from_be_bytes(data_len_bytes.try_into().unwrap());
|
||||||
|
let data = &b[5 + Recipient::LEN + 2 * size_of::<u64>()..];
|
||||||
|
if data.len() as u64 != data_len {
|
||||||
|
return Err(error::Error::new(
|
||||||
|
ErrorKind::MalformedRequest,
|
||||||
|
format!(
|
||||||
|
"data len has inconsistent length. specified: {} got: {}",
|
||||||
|
data_len,
|
||||||
|
data.len()
|
||||||
|
),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(ClientRequest::SendAnonymous {
|
||||||
|
reply_surbs,
|
||||||
|
recipient,
|
||||||
|
message: data.to_vec(),
|
||||||
|
connection_id,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// REPLY_REQUEST_TAG || SENDER_TAG || conn_id || message_len || message
|
||||||
|
fn serialize_reply(
|
||||||
|
message: Vec<u8>,
|
||||||
|
sender_tag: AnonymousSenderTag,
|
||||||
|
connection_id: Option<u64>,
|
||||||
|
) -> Vec<u8> {
|
||||||
|
let message_len_bytes = (message.len() as u64).to_be_bytes();
|
||||||
|
let conn_id_bytes = connection_id.unwrap_or(0).to_be_bytes();
|
||||||
|
|
||||||
|
std::iter::once(ClientRequestTag::Reply as u8)
|
||||||
|
.chain(sender_tag.to_bytes().into_iter())
|
||||||
|
.chain(conn_id_bytes.into_iter())
|
||||||
|
.chain(message_len_bytes.into_iter())
|
||||||
.chain(message.into_iter())
|
.chain(message.into_iter())
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
// REPLY_REQUEST_TAG || surb_len || surb || message_len || message
|
// REPLY_REQUEST_TAG || SENDER_TAG || conn_id || message_len || message
|
||||||
fn deserialize_reply(b: &[u8]) -> Result<Self, error::Error> {
|
fn deserialize_reply(b: &[u8]) -> Result<Self, error::Error> {
|
||||||
// we need to have at the very least 2 * sizeof<u64> bytes (in case, for some peculiar reason
|
if b.len() < 1 + SENDER_TAG_SIZE + 2 * size_of::<u64>() {
|
||||||
// message and reply surb were 0 len - the request would still be malformed, but would in theory
|
|
||||||
// be parse'able)
|
|
||||||
if b.len() < 1 + 2 * size_of::<u64>() {
|
|
||||||
return Err(error::Error::new(
|
return Err(error::Error::new(
|
||||||
ErrorKind::TooShortRequest,
|
ErrorKind::TooShortRequest,
|
||||||
"not enough data provided to recover 'reply'".to_string(),
|
"not enough data provided to recover 'reply'".to_string(),
|
||||||
@@ -162,42 +275,28 @@ impl ClientRequest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// this MUST match because it was called by 'deserialize'
|
// this MUST match because it was called by 'deserialize'
|
||||||
debug_assert_eq!(b[0], REPLY_REQUEST_TAG);
|
debug_assert_eq!(b[0], ClientRequestTag::Reply as u8);
|
||||||
|
|
||||||
let reply_surb_len =
|
// the unwrap here is fine as we're definitely using exactly SENDER_TAG_SIZE bytes
|
||||||
u64::from_be_bytes(b[1..1 + size_of::<u64>()].as_ref().try_into().unwrap());
|
let sender_tag =
|
||||||
|
AnonymousSenderTag::from_bytes(b[1..1 + SENDER_TAG_SIZE].try_into().unwrap());
|
||||||
|
|
||||||
// make sure we won't go out of bounds here
|
let mut connection_id_bytes = [0u8; size_of::<u64>()];
|
||||||
if reply_surb_len > (b.len() - 1 + 2 * size_of::<u64>()) as u64 {
|
connection_id_bytes
|
||||||
return Err(error::Error::new(
|
.copy_from_slice(&b[1 + SENDER_TAG_SIZE..1 + SENDER_TAG_SIZE + size_of::<u64>()]);
|
||||||
ErrorKind::MalformedRequest,
|
let connection_id = u64::from_be_bytes(connection_id_bytes);
|
||||||
format!(
|
let connection_id = if connection_id == 0 {
|
||||||
"not enough data to recover reply surb with specified length {}",
|
None
|
||||||
reply_surb_len
|
} else {
|
||||||
),
|
Some(connection_id)
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
let surb_bound = 1 + size_of::<u64>() + reply_surb_len as usize;
|
|
||||||
|
|
||||||
let reply_surb_bytes = &b[1 + size_of::<u64>()..surb_bound];
|
|
||||||
let reply_surb = match ReplySurb::from_bytes(reply_surb_bytes) {
|
|
||||||
Ok(reply_surb) => reply_surb,
|
|
||||||
Err(err) => {
|
|
||||||
return Err(error::Error::new(
|
|
||||||
ErrorKind::MalformedRequest,
|
|
||||||
format!("malformed reply surb: {:?}", err),
|
|
||||||
))
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
let message_len = u64::from_be_bytes(
|
let message_len = u64::from_be_bytes(
|
||||||
b[surb_bound..surb_bound + size_of::<u64>()]
|
b[1 + SENDER_TAG_SIZE + size_of::<u64>()..1 + SENDER_TAG_SIZE + 2 * size_of::<u64>()]
|
||||||
.as_ref()
|
|
||||||
.try_into()
|
.try_into()
|
||||||
.unwrap(),
|
.unwrap(),
|
||||||
);
|
);
|
||||||
let message = &b[surb_bound + size_of::<u64>()..];
|
let message = &b[1 + SENDER_TAG_SIZE + 2 * size_of::<u64>()..];
|
||||||
if message.len() as u64 != message_len {
|
if message.len() as u64 != message_len {
|
||||||
return Err(error::Error::new(
|
return Err(error::Error::new(
|
||||||
ErrorKind::MalformedRequest,
|
ErrorKind::MalformedRequest,
|
||||||
@@ -208,33 +307,32 @@ impl ClientRequest {
|
|||||||
),
|
),
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
// TODO: should this blow HERE, i.e. during deserialization that the data you're trying
|
|
||||||
// to send via reply is too long?
|
|
||||||
|
|
||||||
Ok(ClientRequest::Reply {
|
Ok(ClientRequest::Reply {
|
||||||
reply_surb,
|
|
||||||
message: message.to_vec(),
|
message: message.to_vec(),
|
||||||
|
sender_tag,
|
||||||
|
connection_id,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// SELF_ADDRESS_REQUEST_TAG
|
// SELF_ADDRESS_REQUEST_TAG
|
||||||
fn serialize_self_address() -> Vec<u8> {
|
fn serialize_self_address() -> Vec<u8> {
|
||||||
std::iter::once(SELF_ADDRESS_REQUEST_TAG).collect()
|
vec![ClientRequestTag::SelfAddress as u8]
|
||||||
}
|
}
|
||||||
|
|
||||||
// SELF_ADDRESS_REQUEST_TAG
|
// SELF_ADDRESS_REQUEST_TAG
|
||||||
fn deserialize_self_address(b: &[u8]) -> Self {
|
fn deserialize_self_address(b: &[u8]) -> Result<Self, error::Error> {
|
||||||
// this MUST match because it was called by 'deserialize'
|
// this MUST match because it was called by 'deserialize'
|
||||||
debug_assert_eq!(b[0], SELF_ADDRESS_REQUEST_TAG);
|
debug_assert_eq!(b[0], ClientRequestTag::SelfAddress as u8);
|
||||||
|
|
||||||
ClientRequest::SelfAddress
|
Ok(ClientRequest::SelfAddress)
|
||||||
}
|
}
|
||||||
|
|
||||||
// CLOSED_CONNECTION_REQUEST_TAG
|
// CLOSED_CONNECTION_REQUEST_TAG
|
||||||
fn serialize_closed_connection(connection_id: u64) -> Vec<u8> {
|
fn serialize_closed_connection(connection_id: u64) -> Vec<u8> {
|
||||||
let conn_id_bytes = connection_id.to_be_bytes();
|
let conn_id_bytes = connection_id.to_be_bytes();
|
||||||
std::iter::once(CLOSED_CONNECTION_REQUEST_TAG)
|
std::iter::once(ClientRequestTag::ClosedConnection as u8)
|
||||||
.chain(conn_id_bytes.iter().copied())
|
.chain(conn_id_bytes.into_iter())
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -243,12 +341,12 @@ impl ClientRequest {
|
|||||||
if b.len() != 1 + size_of::<u64>() {
|
if b.len() != 1 + size_of::<u64>() {
|
||||||
return Err(error::Error::new(
|
return Err(error::Error::new(
|
||||||
ErrorKind::MalformedRequest,
|
ErrorKind::MalformedRequest,
|
||||||
"the received closed connection has invalid length".to_string(),
|
"The received closed connection has invalid length",
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
// this MUST match because it was called by 'deserialize'
|
// this MUST match because it was called by 'deserialize'
|
||||||
debug_assert_eq!(b[0], CLOSED_CONNECTION_REQUEST_TAG);
|
debug_assert_eq!(b[0], ClientRequestTag::ClosedConnection as u8);
|
||||||
|
|
||||||
let mut connection_id_bytes = [0u8; size_of::<u64>()];
|
let mut connection_id_bytes = [0u8; size_of::<u64>()];
|
||||||
connection_id_bytes.copy_from_slice(&b[1..=size_of::<u64>()]);
|
connection_id_bytes.copy_from_slice(&b[1..=size_of::<u64>()]);
|
||||||
@@ -260,8 +358,8 @@ impl ClientRequest {
|
|||||||
// GET_LANE_QUEUE_LENGHT_TAG
|
// GET_LANE_QUEUE_LENGHT_TAG
|
||||||
fn serialize_get_lane_queue_lengths(connection_id: u64) -> Vec<u8> {
|
fn serialize_get_lane_queue_lengths(connection_id: u64) -> Vec<u8> {
|
||||||
let conn_id_bytes = connection_id.to_be_bytes();
|
let conn_id_bytes = connection_id.to_be_bytes();
|
||||||
std::iter::once(GET_LANE_QUEUE_LENGHT_TAG)
|
std::iter::once(ClientRequestTag::GetLaneQueueLength as u8)
|
||||||
.chain(conn_id_bytes.iter().copied())
|
.chain(conn_id_bytes.into_iter())
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -270,12 +368,12 @@ impl ClientRequest {
|
|||||||
if b.len() != 1 + size_of::<u64>() {
|
if b.len() != 1 + size_of::<u64>() {
|
||||||
return Err(error::Error::new(
|
return Err(error::Error::new(
|
||||||
ErrorKind::MalformedRequest,
|
ErrorKind::MalformedRequest,
|
||||||
"the received get lane queue length has invalid length".to_string(),
|
"The received get lane queue lengths has invalid length",
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
// this MUST match because it was called by 'deserialize'
|
// this MUST match because it was called by 'deserialize'
|
||||||
debug_assert_eq!(b[0], GET_LANE_QUEUE_LENGHT_TAG);
|
debug_assert_eq!(b[0], ClientRequestTag::GetLaneQueueLength as u8);
|
||||||
|
|
||||||
let mut connection_id_bytes = [0u8; size_of::<u64>()];
|
let mut connection_id_bytes = [0u8; size_of::<u64>()];
|
||||||
connection_id_bytes.copy_from_slice(&b[1..=size_of::<u64>()]);
|
connection_id_bytes.copy_from_slice(&b[1..=size_of::<u64>()]);
|
||||||
@@ -289,14 +387,21 @@ impl ClientRequest {
|
|||||||
ClientRequest::Send {
|
ClientRequest::Send {
|
||||||
recipient,
|
recipient,
|
||||||
message,
|
message,
|
||||||
with_reply_surb,
|
|
||||||
connection_id,
|
connection_id,
|
||||||
} => Self::serialize_send(recipient, message, with_reply_surb, connection_id),
|
} => Self::serialize_send(recipient, message, connection_id),
|
||||||
|
|
||||||
|
ClientRequest::SendAnonymous {
|
||||||
|
recipient,
|
||||||
|
message,
|
||||||
|
reply_surbs,
|
||||||
|
connection_id,
|
||||||
|
} => Self::serialize_send_anonymous(recipient, message, reply_surbs, connection_id),
|
||||||
|
|
||||||
ClientRequest::Reply {
|
ClientRequest::Reply {
|
||||||
message,
|
message,
|
||||||
reply_surb,
|
sender_tag,
|
||||||
} => Self::serialize_reply(message, &reply_surb),
|
connection_id,
|
||||||
|
} => Self::serialize_reply(message, sender_tag, connection_id),
|
||||||
|
|
||||||
ClientRequest::SelfAddress => Self::serialize_self_address(),
|
ClientRequest::SelfAddress => Self::serialize_self_address(),
|
||||||
|
|
||||||
@@ -316,28 +421,16 @@ impl ClientRequest {
|
|||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
if b.len() < size_of::<u8>() {
|
let request_tag = ClientRequestTag::try_from(b[0])?;
|
||||||
return Err(error::Error::new(
|
|
||||||
ErrorKind::TooShortRequest,
|
|
||||||
format!(
|
|
||||||
"not enough data provided to recover request tag. Provided only {} bytes",
|
|
||||||
b.len()
|
|
||||||
),
|
|
||||||
));
|
|
||||||
}
|
|
||||||
let request_tag = b[0];
|
|
||||||
|
|
||||||
// determine what kind of request that is and try to deserialize it
|
// determine what kind of request that is and try to deserialize it
|
||||||
match request_tag {
|
match request_tag {
|
||||||
SEND_REQUEST_TAG => Self::deserialize_send(b),
|
ClientRequestTag::Send => Self::deserialize_send(b),
|
||||||
REPLY_REQUEST_TAG => Self::deserialize_reply(b),
|
ClientRequestTag::SendAnonymous => Self::deserialize_send_anonymous(b),
|
||||||
SELF_ADDRESS_REQUEST_TAG => Ok(Self::deserialize_self_address(b)),
|
ClientRequestTag::Reply => Self::deserialize_reply(b),
|
||||||
CLOSED_CONNECTION_REQUEST_TAG => Self::deserialize_closed_connection(b),
|
ClientRequestTag::SelfAddress => Self::deserialize_self_address(b),
|
||||||
GET_LANE_QUEUE_LENGHT_TAG => Self::deserialize_get_lane_queue_length(b),
|
ClientRequestTag::ClosedConnection => Self::deserialize_closed_connection(b),
|
||||||
n => Err(error::Error::new(
|
ClientRequestTag::GetLaneQueueLength => Self::deserialize_get_lane_queue_length(b),
|
||||||
ErrorKind::UnknownRequest,
|
|
||||||
format!("type {n}"),
|
|
||||||
)),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -365,50 +458,52 @@ mod tests {
|
|||||||
let recipient = Recipient::try_from_base58_string("CytBseW6yFXUMzz4SGAKdNLGR7q3sJLLYxyBGvutNEQV.4QXYyEVc5fUDjmmi8PrHN9tdUFV4PCvSJE1278cHyvoe@4sBbL1ngf1vtNqykydQKTFh26sQCw888GpUqvPvyNB4f").unwrap();
|
let recipient = Recipient::try_from_base58_string("CytBseW6yFXUMzz4SGAKdNLGR7q3sJLLYxyBGvutNEQV.4QXYyEVc5fUDjmmi8PrHN9tdUFV4PCvSJE1278cHyvoe@4sBbL1ngf1vtNqykydQKTFh26sQCw888GpUqvPvyNB4f").unwrap();
|
||||||
let recipient_string = recipient.to_string();
|
let recipient_string = recipient.to_string();
|
||||||
|
|
||||||
let send_request_no_surb = ClientRequest::Send {
|
let send_request = ClientRequest::Send {
|
||||||
recipient,
|
recipient,
|
||||||
message: b"foomp".to_vec(),
|
message: b"foomp".to_vec(),
|
||||||
with_reply_surb: false,
|
|
||||||
connection_id: Some(42),
|
connection_id: Some(42),
|
||||||
};
|
};
|
||||||
|
|
||||||
let bytes = send_request_no_surb.serialize();
|
let bytes = send_request.serialize();
|
||||||
let recovered = ClientRequest::deserialize(&bytes).unwrap();
|
let recovered = ClientRequest::deserialize(&bytes).unwrap();
|
||||||
match recovered {
|
match recovered {
|
||||||
ClientRequest::Send {
|
ClientRequest::Send {
|
||||||
recipient,
|
recipient,
|
||||||
message,
|
message,
|
||||||
with_reply_surb,
|
|
||||||
connection_id,
|
connection_id,
|
||||||
} => {
|
} => {
|
||||||
assert_eq!(recipient.to_string(), recipient_string);
|
assert_eq!(recipient.to_string(), recipient_string);
|
||||||
assert_eq!(message, b"foomp".to_vec());
|
assert_eq!(message, b"foomp".to_vec());
|
||||||
assert!(!with_reply_surb);
|
|
||||||
assert_eq!(connection_id, Some(42))
|
assert_eq!(connection_id, Some(42))
|
||||||
}
|
}
|
||||||
_ => unreachable!(),
|
_ => unreachable!(),
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let send_request_surb = ClientRequest::Send {
|
#[test]
|
||||||
recipient,
|
fn send_anonymous_request_serialization_works() {
|
||||||
|
let original_recipient = Recipient::try_from_base58_string("CytBseW6yFXUMzz4SGAKdNLGR7q3sJLLYxyBGvutNEQV.4QXYyEVc5fUDjmmi8PrHN9tdUFV4PCvSJE1278cHyvoe@4sBbL1ngf1vtNqykydQKTFh26sQCw888GpUqvPvyNB4f").unwrap();
|
||||||
|
|
||||||
|
let send_anonymous_request = ClientRequest::SendAnonymous {
|
||||||
|
recipient: original_recipient,
|
||||||
message: b"foomp".to_vec(),
|
message: b"foomp".to_vec(),
|
||||||
with_reply_surb: true,
|
reply_surbs: 666,
|
||||||
connection_id: None,
|
connection_id: Some(42),
|
||||||
};
|
};
|
||||||
|
|
||||||
let bytes = send_request_surb.serialize();
|
let bytes = send_anonymous_request.serialize();
|
||||||
let recovered = ClientRequest::deserialize(&bytes).unwrap();
|
let recovered = ClientRequest::deserialize(&bytes).unwrap();
|
||||||
match recovered {
|
match recovered {
|
||||||
ClientRequest::Send {
|
ClientRequest::SendAnonymous {
|
||||||
recipient,
|
recipient,
|
||||||
message,
|
message,
|
||||||
with_reply_surb,
|
reply_surbs,
|
||||||
connection_id,
|
connection_id,
|
||||||
} => {
|
} => {
|
||||||
assert_eq!(recipient.to_string(), recipient_string);
|
assert_eq!(recipient, original_recipient);
|
||||||
assert_eq!(message, b"foomp".to_vec());
|
assert_eq!(message, b"foomp".to_vec());
|
||||||
assert!(with_reply_surb);
|
assert_eq!(connection_id, Some(42));
|
||||||
assert_eq!(connection_id, None)
|
assert_eq!(reply_surbs, 666)
|
||||||
}
|
}
|
||||||
_ => unreachable!(),
|
_ => unreachable!(),
|
||||||
}
|
}
|
||||||
@@ -416,22 +511,23 @@ mod tests {
|
|||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn reply_request_serialization_works() {
|
fn reply_request_serialization_works() {
|
||||||
let reply_surb_string = "CjfVbHbfAjbC3W1BvNHGXmM8KNAnDNYGaHMLqVDxRYeo352csAihstup9bvqXam4dTWgfHak6KYwL9STaxWJ47E8XFZbSEvs7hEsfCkxr6K9WJuSBPK84GDDEvad8ZAuMCoaXsAd5S2Lj9a5eYyzG4SL1jHzhSMni55LyJwumxo1ZTGZNXggxw1RREosvyzNrW9Rsi3owyPqLCwXpiei2tHZty8w8midVvg8vDa7ZEJD842CLv8D4ohynSG7gDpqTrhkRaqYAuz7dzqNbMXLJRM7v823Jn16fA1L7YQxmcaUdUigyRSgTdb4i9ebiLGSyJ1iDe6Acz613PQZh6Ua3bZ2zVKq3dSycpDm9ngarRK4zJrAaUxRkdih8YzW3BY4nL9eqkfKA4N1TWCLaRU7zpSaf8yMEwrAZReU3d5zLV8c5KBfa2w8R5anhQeBojduZEGEad8kkHuKU52Zg93FeWHvH1qgZaEJMHH4nN7gKXz9mvWDhYwyF4vt3Uy2NhCHC3N5pL1gMme27YcoPcTEia1fxKZtnt6rtEozzTrAgCJGswigkFbkafiV5QaJwLKTUxtzhkZ57eEuLPte9UvJHzhhXUQ2CV7R2BUkJjYZy3Zsx6YYvdYWiAFFkWUwNEGA4QpShUHciBfsQVHQ7pN41YcyYUhbywQDFnTVgEmdUZ1XCBi3gyK5U3tDQmFzP1u9m3mWrUA8qB9mRDE7ptNDm5c3c1458L6uXLUth7sdMaa1Was5LCmCdmNDtvNpCDAEt1in6q6mrZFR85aCSU9b1baNGwZoCqPpPvydkVe63gXWoi8ebvdyxARrqACFrSB3ZdY3uJBw8CTMNkKK6MvcefMkSVVsbLd36TQAtYSCqrpiMc5dQuKcEu5QfciwvWYXYx8WFNAgKwP2mv49KCTvfozNDUCbjzDwSx92Zv5zjG8HbFpB13bY9UZGeyTPvv7gGxCzjGjJGbW6FRAheRQaaje5fUgCNM95Tv7wBmAMRHHFgWafeK1sdFH7dtCX9u898HucGTaboSKLsVh8J78gbbkHErwjMh7y9YRkceq5TTYS5da4kHnyNKYWSbxgZrmFg44XGKoeYcqoHB3XTZrdsf7F5fFeNwnihkmADvhAcaxXUmVqq4rQFZH84a1iC3WBWXYcqiZH2L7ujGWV7mMDT4HBEerDYjc8rNY4xGTPfivCrBCJW1i14aqW8xRdsdgTM88eTksvC3WPJLJ7iMzfKXeL7fMW1Ek6QGyQtLBW98vEESpdcDg6DeZ5rMz6VqjTGGqcCaFGfHoqtfxMDaBAEsyQ8h7XDX6dg1wq9wH6j4Tw7Tj1MEv1b8uj5NJkozZdzVdYA2QyE2Dp8vuurQG6uVdTDNww2d88RBQ8sVgjxN8gR45y4woJLhFAaNTAtrY6wDTxyXST13ni6oyqdYxjFVk9Am4v3DzH7Y2K8iRVSHfTk4FRbPULyaeK6wt2anvMJH1XdvVRgc14h67MnBxMgMD1UFk8AErN7CDj26fppe3c5G6KozJe4cSqQUGbBjVzBnrHCruqrfZBn5hNZHTV37bQiomqhRQXohxhuKEnNrGbAe1xNvJr9X";
|
|
||||||
let reply_surb = ReplySurb::from_base58_string(reply_surb_string).unwrap();
|
|
||||||
let reply_request = ClientRequest::Reply {
|
let reply_request = ClientRequest::Reply {
|
||||||
|
sender_tag: [8u8; SENDER_TAG_SIZE].into(),
|
||||||
message: b"foomp".to_vec(),
|
message: b"foomp".to_vec(),
|
||||||
reply_surb,
|
connection_id: Some(42),
|
||||||
};
|
};
|
||||||
|
|
||||||
let bytes = reply_request.serialize();
|
let bytes = reply_request.serialize();
|
||||||
let recovered = ClientRequest::deserialize(&bytes).unwrap();
|
let recovered = ClientRequest::deserialize(&bytes).unwrap();
|
||||||
match recovered {
|
match recovered {
|
||||||
ClientRequest::Reply {
|
ClientRequest::Reply {
|
||||||
reply_surb,
|
sender_tag,
|
||||||
message,
|
message,
|
||||||
|
connection_id,
|
||||||
} => {
|
} => {
|
||||||
assert_eq!(reply_surb.to_base58_string(), reply_surb_string);
|
assert_eq!(sender_tag, [8u8; SENDER_TAG_SIZE].into());
|
||||||
assert_eq!(message, b"foomp".to_vec());
|
assert_eq!(message, b"foomp".to_vec());
|
||||||
|
assert_eq!(connection_id, Some(42));
|
||||||
}
|
}
|
||||||
_ => unreachable!(),
|
_ => unreachable!(),
|
||||||
}
|
}
|
||||||
@@ -458,4 +554,15 @@ mod tests {
|
|||||||
_ => unreachable!(),
|
_ => unreachable!(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn get_lane_queue_length_request_serialization_works() {
|
||||||
|
let close_connection_request = ClientRequest::GetLaneQueueLength(42);
|
||||||
|
let bytes = close_connection_request.serialize();
|
||||||
|
let recovered = ClientRequest::deserialize(&bytes).unwrap();
|
||||||
|
match recovered {
|
||||||
|
ClientRequest::GetLaneQueueLength(id) => assert_eq!(id, 42),
|
||||||
|
_ => unreachable!(),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,36 +1,54 @@
|
|||||||
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
// Copyright 2021-2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
// all variable size data is always prefixed with u64 length
|
// all variable size data is always prefixed with u64 length
|
||||||
// tags are u8
|
// tags are u8
|
||||||
|
|
||||||
#![allow(unknown_lints)] // due to using `clippy::branches_sharing_code` which does not exist on `stable` just yet
|
|
||||||
|
|
||||||
use crate::error::{self, ErrorKind};
|
use crate::error::{self, ErrorKind};
|
||||||
use crate::text::ServerResponseText;
|
use crate::text::ServerResponseText;
|
||||||
use nymsphinx::addressing::clients::Recipient;
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
use nymsphinx::anonymous_replies::ReplySurb;
|
use nymsphinx::anonymous_replies::requests::{AnonymousSenderTag, SENDER_TAG_SIZE};
|
||||||
use nymsphinx::receiver::ReconstructedMessage;
|
use nymsphinx::receiver::ReconstructedMessage;
|
||||||
use std::convert::TryInto;
|
use std::convert::TryInto;
|
||||||
use std::mem::size_of;
|
use std::mem::size_of;
|
||||||
|
|
||||||
/// Value tag representing [`Error`] variant of the [`ServerResponse`]
|
#[repr(u8)]
|
||||||
pub const ERROR_RESPONSE_TAG: u8 = 0x00;
|
enum ServerResponseTag {
|
||||||
|
/// Value tag representing [`Error`] variant of the [`ServerResponse`]
|
||||||
|
Error = 0x00,
|
||||||
|
|
||||||
/// Value tag representing [`Received`] variant of the [`ServerResponse`]
|
/// Value tag representing [`Received`] variant of the [`ServerResponse`]
|
||||||
pub const RECEIVED_RESPONSE_TAG: u8 = 0x01;
|
Received = 0x01,
|
||||||
|
|
||||||
/// Value tag representing [`SelfAddress`] variant of the [`ServerResponse`]
|
/// Value tag representing [`SelfAddress`] variant of the [`ServerResponse`]
|
||||||
pub const SELF_ADDRESS_RESPONSE_TAG: u8 = 0x02;
|
SelfAddress = 0x02,
|
||||||
|
|
||||||
/// Value tag representing [`LaneQueueLength`] variant of the [`ServerResponse`]
|
/// Value tag representing [`LaneQueueLength`] variant of the [`ServerResponse`]
|
||||||
pub const LANE_QUEUE_LENGTH_RESPONSE_TAG: u8 = 0x03;
|
LaneQueueLength = 0x03,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TryFrom<u8> for ServerResponseTag {
|
||||||
|
type Error = error::Error;
|
||||||
|
|
||||||
|
fn try_from(value: u8) -> Result<Self, error::Error> {
|
||||||
|
match value {
|
||||||
|
_ if value == (Self::Error as u8) => Ok(Self::Error),
|
||||||
|
_ if value == (Self::Received as u8) => Ok(Self::Received),
|
||||||
|
_ if value == (Self::SelfAddress as u8) => Ok(Self::SelfAddress),
|
||||||
|
_ if value == (Self::LaneQueueLength as u8) => Ok(Self::LaneQueueLength),
|
||||||
|
n => Err(error::Error::new(
|
||||||
|
ErrorKind::UnknownResponse,
|
||||||
|
format!("{n} does not correspond to any valid response tag"),
|
||||||
|
)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum ServerResponse {
|
pub enum ServerResponse {
|
||||||
Received(ReconstructedMessage),
|
Received(ReconstructedMessage),
|
||||||
SelfAddress(Recipient),
|
SelfAddress(Box<Recipient>),
|
||||||
LaneQueueLength(u64, usize),
|
LaneQueueLength { lane: u64, queue_length: usize },
|
||||||
Error(error::Error),
|
Error(error::Error),
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -42,24 +60,19 @@ impl ServerResponse {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// RECEIVED_RESPONSE_TAG || with_reply || (surb_len || surb) || msg_len || msg
|
// RECEIVED_RESPONSE_TAG || 1 | 0 indicating sender_tag || Option<sender_tag> || msg_len || msg
|
||||||
fn serialize_received(reconstructed_message: ReconstructedMessage) -> Vec<u8> {
|
fn serialize_received(reconstructed_message: ReconstructedMessage) -> Vec<u8> {
|
||||||
let message_len_bytes = (reconstructed_message.message.len() as u64).to_be_bytes();
|
let message_len_bytes = (reconstructed_message.message.len() as u64).to_be_bytes();
|
||||||
if let Some(reply_surb) = reconstructed_message.reply_surb {
|
|
||||||
let reply_surb_bytes = reply_surb.to_bytes();
|
|
||||||
let surb_len_bytes = (reply_surb_bytes.len() as u64).to_be_bytes();
|
|
||||||
|
|
||||||
// with_reply || surb_len || surb || msg_len || msg
|
if let Some(sender_tag) = reconstructed_message.sender_tag {
|
||||||
std::iter::once(RECEIVED_RESPONSE_TAG)
|
std::iter::once(ServerResponseTag::Received as u8)
|
||||||
.chain(std::iter::once(true as u8))
|
.chain(std::iter::once(true as u8))
|
||||||
.chain(surb_len_bytes.iter().cloned())
|
.chain(sender_tag.to_bytes().into_iter())
|
||||||
.chain(reply_surb_bytes.iter().cloned())
|
|
||||||
.chain(message_len_bytes.iter().cloned())
|
.chain(message_len_bytes.iter().cloned())
|
||||||
.chain(reconstructed_message.message.into_iter())
|
.chain(reconstructed_message.message.into_iter())
|
||||||
.collect()
|
.collect()
|
||||||
} else {
|
} else {
|
||||||
// without_reply || msg_len || msg
|
std::iter::once(ServerResponseTag::Received as u8)
|
||||||
std::iter::once(RECEIVED_RESPONSE_TAG)
|
|
||||||
.chain(std::iter::once(false as u8))
|
.chain(std::iter::once(false as u8))
|
||||||
.chain(message_len_bytes.iter().cloned())
|
.chain(message_len_bytes.iter().cloned())
|
||||||
.chain(reconstructed_message.message.into_iter())
|
.chain(reconstructed_message.message.into_iter())
|
||||||
@@ -67,10 +80,9 @@ impl ServerResponse {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// RECEIVED_RESPONSE_TAG || with_reply || (surb_len || surb) || msg_len || msg
|
// RECEIVED_RESPONSE_TAG || 1 | 0 indicating sender_tag || Option<sender_tag> || msg_len || msg
|
||||||
fn deserialize_received(b: &[u8]) -> Result<Self, error::Error> {
|
fn deserialize_received(b: &[u8]) -> Result<Self, error::Error> {
|
||||||
// this MUST match because it was called by 'deserialize'
|
// this MUST match because it was called by 'deserialize'
|
||||||
debug_assert_eq!(b[0], RECEIVED_RESPONSE_TAG);
|
|
||||||
|
|
||||||
// we must be able to read at the very least if it has a reply_surb and length of some field
|
// we must be able to read at the very least if it has a reply_surb and length of some field
|
||||||
if b.len() < 2 + size_of::<u64>() {
|
if b.len() < 2 + size_of::<u64>() {
|
||||||
@@ -79,101 +91,70 @@ impl ServerResponse {
|
|||||||
"not enough data provided to recover 'received'".to_string(),
|
"not enough data provided to recover 'received'".to_string(),
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
debug_assert_eq!(b[0], ServerResponseTag::Received as u8);
|
||||||
|
|
||||||
let with_reply_surb = match b[1] {
|
let has_sender_tag = match b[1] {
|
||||||
0 => false,
|
0 => false,
|
||||||
1 => true,
|
1 => true,
|
||||||
n => {
|
n => {
|
||||||
return Err(error::Error::new(
|
return Err(error::Error::new(
|
||||||
ErrorKind::MalformedResponse,
|
ErrorKind::MalformedResponse,
|
||||||
format!("invalid reply flag {}", n),
|
format!("invalid sender tag flag {n}"),
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// this is a false positive as even though the code is the same, it refers to different things
|
let mut i = 2;
|
||||||
#[allow(clippy::branches_sharing_code)]
|
let sender_tag = if has_sender_tag {
|
||||||
if with_reply_surb {
|
if b[2..].len() < SENDER_TAG_SIZE {
|
||||||
let reply_surb_len =
|
|
||||||
u64::from_be_bytes(b[2..2 + size_of::<u64>()].as_ref().try_into().unwrap());
|
|
||||||
|
|
||||||
// make sure we won't go out of bounds here
|
|
||||||
if reply_surb_len > (b.len() - 2 + 2 * size_of::<u64>()) as u64 {
|
|
||||||
return Err(error::Error::new(
|
return Err(error::Error::new(
|
||||||
ErrorKind::MalformedResponse,
|
ErrorKind::TooShortResponse,
|
||||||
"not enough bytes to read reply_surb bytes!".to_string(),
|
"not enough data provided to recover 'received'".to_string(),
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
i += SENDER_TAG_SIZE;
|
||||||
let surb_bound = 2 + size_of::<u64>() + reply_surb_len as usize;
|
Some(AnonymousSenderTag::from_bytes(
|
||||||
|
b[2..2 + SENDER_TAG_SIZE].try_into().unwrap(),
|
||||||
let reply_surb_bytes = &b[2 + size_of::<u64>()..surb_bound];
|
))
|
||||||
let reply_surb = match ReplySurb::from_bytes(reply_surb_bytes) {
|
|
||||||
Ok(reply_surb) => reply_surb,
|
|
||||||
Err(err) => {
|
|
||||||
return Err(error::Error::new(
|
|
||||||
ErrorKind::MalformedResponse,
|
|
||||||
format!("malformed reply SURB: {:?}", err),
|
|
||||||
))
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let message_len = u64::from_be_bytes(
|
|
||||||
b[surb_bound..surb_bound + size_of::<u64>()]
|
|
||||||
.as_ref()
|
|
||||||
.try_into()
|
|
||||||
.unwrap(),
|
|
||||||
);
|
|
||||||
let message = &b[surb_bound + size_of::<u64>()..];
|
|
||||||
if message.len() as u64 != message_len {
|
|
||||||
return Err(error::Error::new(
|
|
||||||
ErrorKind::MalformedResponse,
|
|
||||||
format!(
|
|
||||||
"message len has inconsistent length. specified: {} got: {}",
|
|
||||||
message_len,
|
|
||||||
message.len()
|
|
||||||
),
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(ServerResponse::Received(ReconstructedMessage {
|
|
||||||
message: message.to_vec(),
|
|
||||||
reply_surb: Some(reply_surb),
|
|
||||||
}))
|
|
||||||
} else {
|
} else {
|
||||||
let message_len =
|
None
|
||||||
u64::from_be_bytes(b[2..2 + size_of::<u64>()].as_ref().try_into().unwrap());
|
};
|
||||||
let message = &b[2 + size_of::<u64>()..];
|
|
||||||
if message.len() as u64 != message_len {
|
|
||||||
return Err(error::Error::new(
|
|
||||||
ErrorKind::MalformedResponse,
|
|
||||||
format!(
|
|
||||||
"message len has inconsistent length. specified: {} got: {}",
|
|
||||||
message_len,
|
|
||||||
message.len()
|
|
||||||
),
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(ServerResponse::Received(ReconstructedMessage {
|
if b[i..].len() < size_of::<u64>() {
|
||||||
message: message.to_vec(),
|
return Err(error::Error::new(
|
||||||
reply_surb: None,
|
ErrorKind::TooShortResponse,
|
||||||
}))
|
"not enough data provided to recover 'received'".to_string(),
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let message_len = u64::from_be_bytes(b[i..i + size_of::<u64>()].try_into().unwrap());
|
||||||
|
let message = &b[i + size_of::<u64>()..];
|
||||||
|
if message.len() as u64 != message_len {
|
||||||
|
return Err(error::Error::new(
|
||||||
|
ErrorKind::MalformedResponse,
|
||||||
|
format!(
|
||||||
|
"message len has inconsistent length. specified: {} got: {}",
|
||||||
|
message_len,
|
||||||
|
message.len()
|
||||||
|
),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(ServerResponse::Received(ReconstructedMessage {
|
||||||
|
message: message.to_vec(),
|
||||||
|
sender_tag,
|
||||||
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
// SELF_ADDRESS_RESPONSE_TAG || self_address
|
// SELF_ADDRESS_RESPONSE_TAG || self_address
|
||||||
fn serialize_self_address(address: Recipient) -> Vec<u8> {
|
fn serialize_self_address(address: Recipient) -> Vec<u8> {
|
||||||
std::iter::once(SELF_ADDRESS_RESPONSE_TAG)
|
std::iter::once(ServerResponseTag::SelfAddress as u8)
|
||||||
.chain(address.to_bytes().iter().cloned())
|
.chain(address.to_bytes().into_iter())
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
// SELF_ADDRESS_RESPONSE_TAG || self_address
|
// SELF_ADDRESS_RESPONSE_TAG || self_address
|
||||||
fn deserialize_self_address(b: &[u8]) -> Result<Self, error::Error> {
|
fn deserialize_self_address(b: &[u8]) -> Result<Self, error::Error> {
|
||||||
// this MUST match because it was called by 'deserialize'
|
|
||||||
debug_assert_eq!(b[0], SELF_ADDRESS_RESPONSE_TAG);
|
|
||||||
|
|
||||||
if b.len() != 1 + Recipient::LEN {
|
if b.len() != 1 + Recipient::LEN {
|
||||||
return Err(error::Error::new(
|
return Err(error::Error::new(
|
||||||
ErrorKind::TooShortResponse,
|
ErrorKind::TooShortResponse,
|
||||||
@@ -181,6 +162,9 @@ impl ServerResponse {
|
|||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// this MUST match because it was called by 'deserialize'
|
||||||
|
debug_assert_eq!(b[0], ServerResponseTag::SelfAddress as u8);
|
||||||
|
|
||||||
let mut recipient_bytes = [0u8; Recipient::LEN];
|
let mut recipient_bytes = [0u8; Recipient::LEN];
|
||||||
recipient_bytes.copy_from_slice(&b[1..1 + Recipient::LEN]);
|
recipient_bytes.copy_from_slice(&b[1..1 + Recipient::LEN]);
|
||||||
|
|
||||||
@@ -194,12 +178,12 @@ impl ServerResponse {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(ServerResponse::SelfAddress(recipient))
|
Ok(ServerResponse::SelfAddress(Box::new(recipient)))
|
||||||
}
|
}
|
||||||
|
|
||||||
// LANE_QUEUE_LENGTH_RESPONSE_TAG || lane || queue_length
|
// LANE_QUEUE_LENGTH_RESPONSE_TAG || lane || queue_length
|
||||||
fn serialize_lane_queue_length(lane: u64, queue_length: usize) -> Vec<u8> {
|
fn serialize_lane_queue_length(lane: u64, queue_length: usize) -> Vec<u8> {
|
||||||
std::iter::once(LANE_QUEUE_LENGTH_RESPONSE_TAG)
|
std::iter::once(ServerResponseTag::LaneQueueLength as u8)
|
||||||
.chain(lane.to_be_bytes().iter().cloned())
|
.chain(lane.to_be_bytes().iter().cloned())
|
||||||
.chain(queue_length.to_be_bytes().iter().cloned())
|
.chain(queue_length.to_be_bytes().iter().cloned())
|
||||||
.collect()
|
.collect()
|
||||||
@@ -208,7 +192,7 @@ impl ServerResponse {
|
|||||||
// LANE_QUEUE_LENGTH_RESPONSE_TAG || lane || queue_length
|
// LANE_QUEUE_LENGTH_RESPONSE_TAG || lane || queue_length
|
||||||
fn deserialize_lane_queue_length(b: &[u8]) -> Result<Self, error::Error> {
|
fn deserialize_lane_queue_length(b: &[u8]) -> Result<Self, error::Error> {
|
||||||
// this MUST match because it was called by 'deserialize'
|
// this MUST match because it was called by 'deserialize'
|
||||||
debug_assert_eq!(b[0], LANE_QUEUE_LENGTH_RESPONSE_TAG);
|
debug_assert_eq!(b[0], ServerResponseTag::LaneQueueLength as u8);
|
||||||
|
|
||||||
let mut lane_bytes = [0u8; size_of::<u64>()];
|
let mut lane_bytes = [0u8; size_of::<u64>()];
|
||||||
lane_bytes.copy_from_slice(&b[1..=size_of::<u64>()]);
|
lane_bytes.copy_from_slice(&b[1..=size_of::<u64>()]);
|
||||||
@@ -219,15 +203,15 @@ impl ServerResponse {
|
|||||||
.copy_from_slice(&b[1 + size_of::<u64>()..1 + size_of::<u64>() + size_of::<usize>()]);
|
.copy_from_slice(&b[1 + size_of::<u64>()..1 + size_of::<u64>() + size_of::<usize>()]);
|
||||||
let queue_length = usize::from_be_bytes(queue_length_bytes);
|
let queue_length = usize::from_be_bytes(queue_length_bytes);
|
||||||
|
|
||||||
Ok(ServerResponse::LaneQueueLength(lane, queue_length))
|
Ok(ServerResponse::LaneQueueLength { lane, queue_length })
|
||||||
}
|
}
|
||||||
|
|
||||||
// ERROR_RESPONSE_TAG || err_code || msg_len || msg
|
// ERROR_RESPONSE_TAG || err_code || msg_len || msg
|
||||||
fn serialize_error(error: error::Error) -> Vec<u8> {
|
fn serialize_error(error: error::Error) -> Vec<u8> {
|
||||||
let message_len_bytes = (error.message.len() as u64).to_be_bytes();
|
let message_len_bytes = (error.message.len() as u64).to_be_bytes();
|
||||||
std::iter::once(ERROR_RESPONSE_TAG)
|
std::iter::once(ServerResponseTag::Error as u8)
|
||||||
.chain(std::iter::once(error.kind as u8))
|
.chain(std::iter::once(error.kind as u8))
|
||||||
.chain(message_len_bytes.iter().cloned())
|
.chain(message_len_bytes.into_iter())
|
||||||
.chain(error.message.into_bytes().into_iter())
|
.chain(error.message.into_bytes().into_iter())
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
@@ -235,7 +219,7 @@ impl ServerResponse {
|
|||||||
// ERROR_RESPONSE_TAG || err_code || msg_len || msg
|
// ERROR_RESPONSE_TAG || err_code || msg_len || msg
|
||||||
fn deserialize_error(b: &[u8]) -> Result<Self, error::Error> {
|
fn deserialize_error(b: &[u8]) -> Result<Self, error::Error> {
|
||||||
// this MUST match because it was called by 'deserialize'
|
// this MUST match because it was called by 'deserialize'
|
||||||
debug_assert_eq!(b[0], ERROR_RESPONSE_TAG);
|
debug_assert_eq!(b[0], ServerResponseTag::Error as u8);
|
||||||
|
|
||||||
if b.len() < size_of::<u8>() + size_of::<u64>() {
|
if b.len() < size_of::<u8>() + size_of::<u64>() {
|
||||||
return Err(error::Error::new(
|
return Err(error::Error::new(
|
||||||
@@ -244,26 +228,7 @@ impl ServerResponse {
|
|||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
let error_kind = match b[1] {
|
let error_kind = ErrorKind::try_from(b[1])?;
|
||||||
_ if b[1] == (ErrorKind::EmptyRequest as u8) => ErrorKind::EmptyRequest,
|
|
||||||
_ if b[1] == (ErrorKind::TooShortRequest as u8) => ErrorKind::TooShortRequest,
|
|
||||||
_ if b[1] == (ErrorKind::UnknownRequest as u8) => ErrorKind::UnknownRequest,
|
|
||||||
_ if b[1] == (ErrorKind::MalformedRequest as u8) => ErrorKind::MalformedRequest,
|
|
||||||
|
|
||||||
_ if b[1] == (ErrorKind::EmptyResponse as u8) => ErrorKind::EmptyResponse,
|
|
||||||
_ if b[1] == (ErrorKind::TooShortResponse as u8) => ErrorKind::TooShortResponse,
|
|
||||||
_ if b[1] == (ErrorKind::UnknownResponse as u8) => ErrorKind::UnknownResponse,
|
|
||||||
_ if b[1] == (ErrorKind::MalformedResponse as u8) => ErrorKind::MalformedResponse,
|
|
||||||
|
|
||||||
_ if b[1] == (ErrorKind::Other as u8) => ErrorKind::Other,
|
|
||||||
|
|
||||||
n => {
|
|
||||||
return Err(error::Error::new(
|
|
||||||
ErrorKind::MalformedResponse,
|
|
||||||
format!("invalid error code {}", n),
|
|
||||||
))
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let message_len =
|
let message_len =
|
||||||
u64::from_be_bytes(b[2..2 + size_of::<u64>()].as_ref().try_into().unwrap());
|
u64::from_be_bytes(b[2..2 + size_of::<u64>()].as_ref().try_into().unwrap());
|
||||||
@@ -300,8 +265,8 @@ impl ServerResponse {
|
|||||||
ServerResponse::Received(reconstructed_message) => {
|
ServerResponse::Received(reconstructed_message) => {
|
||||||
Self::serialize_received(reconstructed_message)
|
Self::serialize_received(reconstructed_message)
|
||||||
}
|
}
|
||||||
ServerResponse::SelfAddress(address) => Self::serialize_self_address(address),
|
ServerResponse::SelfAddress(address) => Self::serialize_self_address(*address),
|
||||||
ServerResponse::LaneQueueLength(lane, queue_length) => {
|
ServerResponse::LaneQueueLength { lane, queue_length } => {
|
||||||
Self::serialize_lane_queue_length(lane, queue_length)
|
Self::serialize_lane_queue_length(lane, queue_length)
|
||||||
}
|
}
|
||||||
ServerResponse::Error(err) => Self::serialize_error(err),
|
ServerResponse::Error(err) => Self::serialize_error(err),
|
||||||
@@ -328,18 +293,14 @@ impl ServerResponse {
|
|||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
let response_tag = b[0];
|
let response_tag = ServerResponseTag::try_from(b[0])?;
|
||||||
|
|
||||||
// determine what kind of response that is and try to deserialize it
|
// determine what kind of response that is and try to deserialize it
|
||||||
match response_tag {
|
match response_tag {
|
||||||
RECEIVED_RESPONSE_TAG => Self::deserialize_received(b),
|
ServerResponseTag::Received => Self::deserialize_received(b),
|
||||||
SELF_ADDRESS_RESPONSE_TAG => Self::deserialize_self_address(b),
|
ServerResponseTag::SelfAddress => Self::deserialize_self_address(b),
|
||||||
LANE_QUEUE_LENGTH_RESPONSE_TAG => Self::deserialize_lane_queue_length(b),
|
ServerResponseTag::LaneQueueLength => Self::deserialize_lane_queue_length(b),
|
||||||
ERROR_RESPONSE_TAG => Self::deserialize_error(b),
|
ServerResponseTag::Error => Self::deserialize_error(b),
|
||||||
n => Err(error::Error::new(
|
|
||||||
ErrorKind::UnknownResponse,
|
|
||||||
format!("type {}", n),
|
|
||||||
)),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -361,35 +322,33 @@ mod tests {
|
|||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn received_response_serialization_works() {
|
fn received_response_serialization_works() {
|
||||||
let reply_surb_string = "CjfVbHbfAjbC3W1BvNHGXmM8KNAnDNYGaHMLqVDxRYeo352csAihstup9bvqXam4dTWgfHak6KYwL9STaxWJ47E8XFZbSEvs7hEsfCkxr6K9WJuSBPK84GDDEvad8ZAuMCoaXsAd5S2Lj9a5eYyzG4SL1jHzhSMni55LyJwumxo1ZTGZNXggxw1RREosvyzNrW9Rsi3owyPqLCwXpiei2tHZty8w8midVvg8vDa7ZEJD842CLv8D4ohynSG7gDpqTrhkRaqYAuz7dzqNbMXLJRM7v823Jn16fA1L7YQxmcaUdUigyRSgTdb4i9ebiLGSyJ1iDe6Acz613PQZh6Ua3bZ2zVKq3dSycpDm9ngarRK4zJrAaUxRkdih8YzW3BY4nL9eqkfKA4N1TWCLaRU7zpSaf8yMEwrAZReU3d5zLV8c5KBfa2w8R5anhQeBojduZEGEad8kkHuKU52Zg93FeWHvH1qgZaEJMHH4nN7gKXz9mvWDhYwyF4vt3Uy2NhCHC3N5pL1gMme27YcoPcTEia1fxKZtnt6rtEozzTrAgCJGswigkFbkafiV5QaJwLKTUxtzhkZ57eEuLPte9UvJHzhhXUQ2CV7R2BUkJjYZy3Zsx6YYvdYWiAFFkWUwNEGA4QpShUHciBfsQVHQ7pN41YcyYUhbywQDFnTVgEmdUZ1XCBi3gyK5U3tDQmFzP1u9m3mWrUA8qB9mRDE7ptNDm5c3c1458L6uXLUth7sdMaa1Was5LCmCdmNDtvNpCDAEt1in6q6mrZFR85aCSU9b1baNGwZoCqPpPvydkVe63gXWoi8ebvdyxARrqACFrSB3ZdY3uJBw8CTMNkKK6MvcefMkSVVsbLd36TQAtYSCqrpiMc5dQuKcEu5QfciwvWYXYx8WFNAgKwP2mv49KCTvfozNDUCbjzDwSx92Zv5zjG8HbFpB13bY9UZGeyTPvv7gGxCzjGjJGbW6FRAheRQaaje5fUgCNM95Tv7wBmAMRHHFgWafeK1sdFH7dtCX9u898HucGTaboSKLsVh8J78gbbkHErwjMh7y9YRkceq5TTYS5da4kHnyNKYWSbxgZrmFg44XGKoeYcqoHB3XTZrdsf7F5fFeNwnihkmADvhAcaxXUmVqq4rQFZH84a1iC3WBWXYcqiZH2L7ujGWV7mMDT4HBEerDYjc8rNY4xGTPfivCrBCJW1i14aqW8xRdsdgTM88eTksvC3WPJLJ7iMzfKXeL7fMW1Ek6QGyQtLBW98vEESpdcDg6DeZ5rMz6VqjTGGqcCaFGfHoqtfxMDaBAEsyQ8h7XDX6dg1wq9wH6j4Tw7Tj1MEv1b8uj5NJkozZdzVdYA2QyE2Dp8vuurQG6uVdTDNww2d88RBQ8sVgjxN8gR45y4woJLhFAaNTAtrY6wDTxyXST13ni6oyqdYxjFVk9Am4v3DzH7Y2K8iRVSHfTk4FRbPULyaeK6wt2anvMJH1XdvVRgc14h67MnBxMgMD1UFk8AErN7CDj26fppe3c5G6KozJe4cSqQUGbBjVzBnrHCruqrfZBn5hNZHTV37bQiomqhRQXohxhuKEnNrGbAe1xNvJr9X";
|
let received_with_sender_tag = ServerResponse::Received(ReconstructedMessage {
|
||||||
|
|
||||||
let received_with_surb = ServerResponse::Received(ReconstructedMessage {
|
|
||||||
message: b"foomp".to_vec(),
|
message: b"foomp".to_vec(),
|
||||||
reply_surb: Some(ReplySurb::from_base58_string(reply_surb_string).unwrap()),
|
sender_tag: Some([42u8; SENDER_TAG_SIZE].into()),
|
||||||
});
|
});
|
||||||
let bytes = received_with_surb.serialize();
|
let bytes = received_with_sender_tag.serialize();
|
||||||
let recovered = ServerResponse::deserialize(&bytes).unwrap();
|
let recovered = ServerResponse::deserialize(&bytes).unwrap();
|
||||||
match recovered {
|
match recovered {
|
||||||
ServerResponse::Received(reconstructed) => {
|
ServerResponse::Received(reconstructed) => {
|
||||||
assert_eq!(reconstructed.message, b"foomp".to_vec());
|
assert_eq!(reconstructed.message, b"foomp".to_vec());
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
reconstructed.reply_surb.unwrap().to_base58_string(),
|
reconstructed.sender_tag,
|
||||||
reply_surb_string
|
Some([42u8; SENDER_TAG_SIZE].into())
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
_ => unreachable!(),
|
_ => unreachable!(),
|
||||||
}
|
}
|
||||||
|
|
||||||
let received_without_surb = ServerResponse::Received(ReconstructedMessage {
|
let received_without_sender_tag = ServerResponse::Received(ReconstructedMessage {
|
||||||
message: b"foomp".to_vec(),
|
message: b"foomp".to_vec(),
|
||||||
reply_surb: None,
|
sender_tag: None,
|
||||||
});
|
});
|
||||||
let bytes = received_without_surb.serialize();
|
let bytes = received_without_sender_tag.serialize();
|
||||||
let recovered = ServerResponse::deserialize(&bytes).unwrap();
|
let recovered = ServerResponse::deserialize(&bytes).unwrap();
|
||||||
match recovered {
|
match recovered {
|
||||||
ServerResponse::Received(reconstructed) => {
|
ServerResponse::Received(reconstructed) => {
|
||||||
assert_eq!(reconstructed.message, b"foomp".to_vec());
|
assert_eq!(reconstructed.message, b"foomp".to_vec());
|
||||||
assert!(reconstructed.reply_surb.is_none())
|
assert!(reconstructed.sender_tag.is_none())
|
||||||
}
|
}
|
||||||
_ => unreachable!(),
|
_ => unreachable!(),
|
||||||
}
|
}
|
||||||
@@ -400,7 +359,7 @@ mod tests {
|
|||||||
let recipient = Recipient::try_from_base58_string("CytBseW6yFXUMzz4SGAKdNLGR7q3sJLLYxyBGvutNEQV.4QXYyEVc5fUDjmmi8PrHN9tdUFV4PCvSJE1278cHyvoe@4sBbL1ngf1vtNqykydQKTFh26sQCw888GpUqvPvyNB4f").unwrap();
|
let recipient = Recipient::try_from_base58_string("CytBseW6yFXUMzz4SGAKdNLGR7q3sJLLYxyBGvutNEQV.4QXYyEVc5fUDjmmi8PrHN9tdUFV4PCvSJE1278cHyvoe@4sBbL1ngf1vtNqykydQKTFh26sQCw888GpUqvPvyNB4f").unwrap();
|
||||||
let recipient_string = recipient.to_string();
|
let recipient_string = recipient.to_string();
|
||||||
|
|
||||||
let self_address_response = ServerResponse::SelfAddress(recipient);
|
let self_address_response = ServerResponse::SelfAddress(Box::new(recipient));
|
||||||
let bytes = self_address_response.serialize();
|
let bytes = self_address_response.serialize();
|
||||||
let recovered = ServerResponse::deserialize(&bytes).unwrap();
|
let recovered = ServerResponse::deserialize(&bytes).unwrap();
|
||||||
match recovered {
|
match recovered {
|
||||||
@@ -413,11 +372,14 @@ mod tests {
|
|||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn lane_queue_length_response_serialization_works() {
|
fn lane_queue_length_response_serialization_works() {
|
||||||
let lane_queue_length_response = ServerResponse::LaneQueueLength(13, 42);
|
let lane_queue_length_response = ServerResponse::LaneQueueLength {
|
||||||
|
lane: 13,
|
||||||
|
queue_length: 42,
|
||||||
|
};
|
||||||
let bytes = lane_queue_length_response.serialize();
|
let bytes = lane_queue_length_response.serialize();
|
||||||
let recovered = ServerResponse::deserialize(&bytes).unwrap();
|
let recovered = ServerResponse::deserialize(&bytes).unwrap();
|
||||||
match recovered {
|
match recovered {
|
||||||
ServerResponse::LaneQueueLength(lane, queue_length) => {
|
ServerResponse::LaneQueueLength { lane, queue_length } => {
|
||||||
assert_eq!(lane, 13);
|
assert_eq!(lane, 13);
|
||||||
assert_eq!(queue_length, 42)
|
assert_eq!(queue_length, 42)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
// Copyright 2021-2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use crate::error::ErrorKind;
|
use crate::error::ErrorKind;
|
||||||
use crate::requests::ClientRequest;
|
use crate::requests::ClientRequest;
|
||||||
use crate::responses::ServerResponse;
|
use crate::responses::ServerResponse;
|
||||||
use nymsphinx::addressing::clients::Recipient;
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
use nymsphinx::anonymous_replies::ReplySurb;
|
use nymsphinx::anonymous_replies::requests::AnonymousSenderTag;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use std::convert::{TryFrom, TryInto};
|
use std::convert::{TryFrom, TryInto};
|
||||||
|
|
||||||
@@ -19,15 +19,22 @@ pub(super) enum ClientRequestText {
|
|||||||
Send {
|
Send {
|
||||||
message: String,
|
message: String,
|
||||||
recipient: String,
|
recipient: String,
|
||||||
with_reply_surb: bool,
|
connection_id: Option<u64>,
|
||||||
|
},
|
||||||
|
#[serde(rename_all = "camelCase")]
|
||||||
|
SendAnonymous {
|
||||||
|
recipient: String,
|
||||||
|
message: String,
|
||||||
|
reply_surbs: u32,
|
||||||
|
connection_id: Option<u64>,
|
||||||
|
},
|
||||||
|
#[serde(rename_all = "camelCase")]
|
||||||
|
Reply {
|
||||||
|
sender_tag: String,
|
||||||
|
message: String,
|
||||||
connection_id: Option<u64>,
|
connection_id: Option<u64>,
|
||||||
},
|
},
|
||||||
SelfAddress,
|
SelfAddress,
|
||||||
#[serde(rename_all = "camelCase")]
|
|
||||||
Reply {
|
|
||||||
message: String,
|
|
||||||
reply_surb: String,
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TryFrom<String> for ClientRequestText {
|
impl TryFrom<String> for ClientRequestText {
|
||||||
@@ -46,7 +53,6 @@ impl TryInto<ClientRequest> for ClientRequestText {
|
|||||||
ClientRequestText::Send {
|
ClientRequestText::Send {
|
||||||
message,
|
message,
|
||||||
recipient,
|
recipient,
|
||||||
with_reply_surb,
|
|
||||||
connection_id,
|
connection_id,
|
||||||
} => {
|
} => {
|
||||||
let message_bytes = message.into_bytes();
|
let message_bytes = message.into_bytes();
|
||||||
@@ -57,23 +63,42 @@ impl TryInto<ClientRequest> for ClientRequestText {
|
|||||||
Ok(ClientRequest::Send {
|
Ok(ClientRequest::Send {
|
||||||
message: message_bytes,
|
message: message_bytes,
|
||||||
recipient,
|
recipient,
|
||||||
with_reply_surb,
|
connection_id,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
ClientRequestText::SendAnonymous {
|
||||||
|
recipient,
|
||||||
|
message,
|
||||||
|
reply_surbs,
|
||||||
|
connection_id,
|
||||||
|
} => {
|
||||||
|
let message_bytes = message.into_bytes();
|
||||||
|
let recipient = Recipient::try_from_base58_string(recipient).map_err(|err| {
|
||||||
|
Self::Error::new(ErrorKind::MalformedRequest, err.to_string())
|
||||||
|
})?;
|
||||||
|
Ok(ClientRequest::SendAnonymous {
|
||||||
|
recipient,
|
||||||
|
message: message_bytes,
|
||||||
|
reply_surbs,
|
||||||
connection_id,
|
connection_id,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
ClientRequestText::SelfAddress => Ok(ClientRequest::SelfAddress),
|
ClientRequestText::SelfAddress => Ok(ClientRequest::SelfAddress),
|
||||||
ClientRequestText::Reply {
|
ClientRequestText::Reply {
|
||||||
|
sender_tag,
|
||||||
message,
|
message,
|
||||||
reply_surb,
|
connection_id,
|
||||||
} => {
|
} => {
|
||||||
let message_bytes = message.into_bytes();
|
let message_bytes = message.into_bytes();
|
||||||
let reply_surb = ReplySurb::from_base58_string(reply_surb).map_err(|err| {
|
let sender_tag =
|
||||||
Self::Error::new(ErrorKind::MalformedRequest, err.to_string())
|
AnonymousSenderTag::try_from_base58_string(sender_tag).map_err(|err| {
|
||||||
})?;
|
Self::Error::new(ErrorKind::MalformedRequest, err.to_string())
|
||||||
|
})?;
|
||||||
|
|
||||||
Ok(ClientRequest::Reply {
|
Ok(ClientRequest::Reply {
|
||||||
|
sender_tag,
|
||||||
message: message_bytes,
|
message: message_bytes,
|
||||||
reply_surb,
|
connection_id,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -89,7 +114,7 @@ pub(super) enum ServerResponseText {
|
|||||||
#[serde(rename_all = "camelCase")]
|
#[serde(rename_all = "camelCase")]
|
||||||
Received {
|
Received {
|
||||||
message: String,
|
message: String,
|
||||||
reply_surb: Option<String>,
|
sender_tag: Option<String>,
|
||||||
},
|
},
|
||||||
SelfAddress {
|
SelfAddress {
|
||||||
address: String,
|
address: String,
|
||||||
@@ -131,15 +156,13 @@ impl From<ServerResponse> for ServerResponseText {
|
|||||||
// TODO: ask DH what is more appropriate, lossy utf8 conversion or returning error and then
|
// TODO: ask DH what is more appropriate, lossy utf8 conversion or returning error and then
|
||||||
// pure binary later
|
// pure binary later
|
||||||
message: String::from_utf8_lossy(&reconstructed.message).into_owned(),
|
message: String::from_utf8_lossy(&reconstructed.message).into_owned(),
|
||||||
reply_surb: reconstructed
|
sender_tag: reconstructed.sender_tag.map(|tag| tag.to_base58_string()),
|
||||||
.reply_surb
|
|
||||||
.map(|reply_surb| reply_surb.to_base58_string()),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ServerResponse::SelfAddress(recipient) => ServerResponseText::SelfAddress {
|
ServerResponse::SelfAddress(recipient) => ServerResponseText::SelfAddress {
|
||||||
address: recipient.to_string(),
|
address: recipient.to_string(),
|
||||||
},
|
},
|
||||||
ServerResponse::LaneQueueLength(lane, queue_length) => {
|
ServerResponse::LaneQueueLength { lane, queue_length } => {
|
||||||
ServerResponseText::LaneQueueLength { lane, queue_length }
|
ServerResponseText::LaneQueueLength { lane, queue_length }
|
||||||
}
|
}
|
||||||
ServerResponse::Error(err) => ServerResponseText::Error {
|
ServerResponse::Error(err) => ServerResponseText::Error {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "nym-socks5-client"
|
name = "nym-socks5-client"
|
||||||
version = "1.1.0"
|
version = "1.1.1"
|
||||||
authors = ["Dave Hrycyszyn <futurechimp@users.noreply.github.com>"]
|
authors = ["Dave Hrycyszyn <futurechimp@users.noreply.github.com>"]
|
||||||
description = "A SOCKS5 localhost proxy that converts incoming messages to Sphinx and sends them to a Nym address"
|
description = "A SOCKS5 localhost proxy that converts incoming messages to Sphinx and sends them to a Nym address"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
@@ -25,7 +25,7 @@ tokio = { version = "1.21.2", features = ["rt-multi-thread", "net", "signal"] }
|
|||||||
url = "2.2"
|
url = "2.2"
|
||||||
|
|
||||||
# internal
|
# internal
|
||||||
client-core = { path = "../client-core" }
|
client-core = { path = "../client-core", features = ["fs-surb-storage"] }
|
||||||
client-connections = { path = "../../common/client-connections" }
|
client-connections = { path = "../../common/client-connections" }
|
||||||
coconut-interface = { path = "../../common/coconut-interface", optional = true }
|
coconut-interface = { path = "../../common/coconut-interface", optional = true }
|
||||||
config = { path = "../../common/config" }
|
config = { path = "../../common/config" }
|
||||||
|
|||||||
@@ -2,8 +2,8 @@
|
|||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use crate::client::config::template::config_template;
|
use crate::client::config::template::config_template;
|
||||||
use client_core::config::Config as BaseConfig;
|
|
||||||
pub use client_core::config::MISSING_VALUE;
|
pub use client_core::config::MISSING_VALUE;
|
||||||
|
use client_core::config::{Config as BaseConfig, DebugConfig};
|
||||||
use config::defaults::DEFAULT_SOCKS5_LISTENING_PORT;
|
use config::defaults::DEFAULT_SOCKS5_LISTENING_PORT;
|
||||||
use config::NymConfig;
|
use config::NymConfig;
|
||||||
use nymsphinx::addressing::clients::Recipient;
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
@@ -12,6 +12,9 @@ use std::path::PathBuf;
|
|||||||
|
|
||||||
mod template;
|
mod template;
|
||||||
|
|
||||||
|
const DEFAULT_CONNECTION_START_SURBS: u32 = 20;
|
||||||
|
const DEFAULT_PER_REQUEST_SURBS: u32 = 3;
|
||||||
|
|
||||||
#[derive(Debug, Default, Deserialize, PartialEq, Serialize)]
|
#[derive(Debug, Default, Deserialize, PartialEq, Serialize)]
|
||||||
#[serde(deny_unknown_fields)]
|
#[serde(deny_unknown_fields)]
|
||||||
pub struct Config {
|
pub struct Config {
|
||||||
@@ -19,6 +22,9 @@ pub struct Config {
|
|||||||
base: BaseConfig<Config>,
|
base: BaseConfig<Config>,
|
||||||
|
|
||||||
socks5: Socks5,
|
socks5: Socks5,
|
||||||
|
|
||||||
|
#[serde(default)]
|
||||||
|
socks5_debug: Socks5Debug,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl NymConfig for Config {
|
impl NymConfig for Config {
|
||||||
@@ -57,6 +63,7 @@ impl Config {
|
|||||||
Config {
|
Config {
|
||||||
base: BaseConfig::new(id),
|
base: BaseConfig::new(id),
|
||||||
socks5: Socks5::new(provider_mix_address),
|
socks5: Socks5::new(provider_mix_address),
|
||||||
|
socks5_debug: Socks5Debug::default(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -70,7 +77,24 @@ impl Config {
|
|||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn with_anonymous_replies(mut self, anonymous_replies: bool) -> Self {
|
||||||
|
self.socks5.send_anonymously = anonymous_replies;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
// getters
|
// getters
|
||||||
|
pub fn get_base(&self) -> &BaseConfig<Self> {
|
||||||
|
&self.base
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_base_mut(&mut self) -> &mut BaseConfig<Self> {
|
||||||
|
&mut self.base
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_debug_settings(&self) -> &DebugConfig {
|
||||||
|
self.get_base().get_debug_config()
|
||||||
|
}
|
||||||
|
|
||||||
pub fn get_config_file_save_location(&self) -> PathBuf {
|
pub fn get_config_file_save_location(&self) -> PathBuf {
|
||||||
self.config_directory().join(Self::config_file_name())
|
self.config_directory().join(Self::config_file_name())
|
||||||
}
|
}
|
||||||
@@ -80,17 +104,21 @@ impl Config {
|
|||||||
.expect("malformed provider address")
|
.expect("malformed provider address")
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_base(&self) -> &BaseConfig<Self> {
|
pub fn get_send_anonymously(&self) -> bool {
|
||||||
&self.base
|
self.socks5.send_anonymously
|
||||||
}
|
|
||||||
|
|
||||||
pub fn get_base_mut(&mut self) -> &mut BaseConfig<Self> {
|
|
||||||
&mut self.base
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_listening_port(&self) -> u16 {
|
pub fn get_listening_port(&self) -> u16 {
|
||||||
self.socks5.listening_port
|
self.socks5.listening_port
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn get_connection_start_surbs(&self) -> u32 {
|
||||||
|
self.socks5_debug.connection_start_surbs
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_per_request_surbs(&self) -> u32 {
|
||||||
|
self.socks5_debug.per_request_surbs
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Deserialize, PartialEq, Eq, Serialize)]
|
#[derive(Debug, Deserialize, PartialEq, Eq, Serialize)]
|
||||||
@@ -101,6 +129,13 @@ pub struct Socks5 {
|
|||||||
|
|
||||||
/// The mix address of the provider to which all requests are going to be sent.
|
/// The mix address of the provider to which all requests are going to be sent.
|
||||||
provider_mix_address: String,
|
provider_mix_address: String,
|
||||||
|
|
||||||
|
/// Specifies whether this client is going to use an anonymous sender tag for communication with the service provider.
|
||||||
|
/// While this is going to hide its actual address information, it will make the actual communication
|
||||||
|
/// slower and consume nearly double the bandwidth as it will require sending reply SURBs.
|
||||||
|
///
|
||||||
|
/// Note that some service providers might not support this.
|
||||||
|
send_anonymously: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Socks5 {
|
impl Socks5 {
|
||||||
@@ -108,6 +143,7 @@ impl Socks5 {
|
|||||||
Socks5 {
|
Socks5 {
|
||||||
listening_port: DEFAULT_SOCKS5_LISTENING_PORT,
|
listening_port: DEFAULT_SOCKS5_LISTENING_PORT,
|
||||||
provider_mix_address: provider_mix_address.into(),
|
provider_mix_address: provider_mix_address.into(),
|
||||||
|
send_anonymously: false,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -117,6 +153,26 @@ impl Default for Socks5 {
|
|||||||
Socks5 {
|
Socks5 {
|
||||||
listening_port: DEFAULT_SOCKS5_LISTENING_PORT,
|
listening_port: DEFAULT_SOCKS5_LISTENING_PORT,
|
||||||
provider_mix_address: "".into(),
|
provider_mix_address: "".into(),
|
||||||
|
send_anonymously: false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Deserialize, PartialEq, Eq, Serialize)]
|
||||||
|
#[serde(deny_unknown_fields)]
|
||||||
|
pub struct Socks5Debug {
|
||||||
|
/// Number of reply SURBs attached to each `Request::Connect` message.
|
||||||
|
connection_start_surbs: u32,
|
||||||
|
|
||||||
|
/// Number of reply SURBs attached to each `Request::Send` message.
|
||||||
|
per_request_surbs: u32,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for Socks5Debug {
|
||||||
|
fn default() -> Self {
|
||||||
|
Socks5Debug {
|
||||||
|
connection_start_surbs: DEFAULT_CONNECTION_START_SURBS,
|
||||||
|
per_request_surbs: DEFAULT_PER_REQUEST_SURBS,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -49,13 +49,12 @@ private_encryption_key_file = '{{ client.private_encryption_key_file }}'
|
|||||||
# Path to file containing public encryption key.
|
# Path to file containing public encryption key.
|
||||||
public_encryption_key_file = '{{ client.public_encryption_key_file }}'
|
public_encryption_key_file = '{{ client.public_encryption_key_file }}'
|
||||||
|
|
||||||
# Full path to file containing reply encryption keys of all reply-SURBs we have ever
|
|
||||||
# sent but not received back.
|
|
||||||
reply_encryption_key_store_path = '{{ client.reply_encryption_key_store_path }}'
|
|
||||||
|
|
||||||
# Path to the database containing bandwidth credentials
|
# Path to the database containing bandwidth credentials
|
||||||
database_path = '{{ client.database_path }}'
|
database_path = '{{ client.database_path }}'
|
||||||
|
|
||||||
|
# Path to the persistent store for received reply surbs, unused encryption keys and used sender tags.
|
||||||
|
reply_surb_database_path = '{{ client.reply_surb_database_path }}'
|
||||||
|
|
||||||
##### additional client config options #####
|
##### additional client config options #####
|
||||||
|
|
||||||
# A gateway specific, optional, base58 stringified shared key used for
|
# A gateway specific, optional, base58 stringified shared key used for
|
||||||
@@ -92,6 +91,12 @@ provider_mix_address = '{{ socks5.provider_mix_address }}'
|
|||||||
# The port on which the client will be listening for incoming requests
|
# The port on which the client will be listening for incoming requests
|
||||||
listening_port = {{ socks5.listening_port }}
|
listening_port = {{ socks5.listening_port }}
|
||||||
|
|
||||||
|
# Specifies whether this client is going to use an anonymous sender tag for communication with the service provider.
|
||||||
|
# While this is going to hide its actual address information, it will make the actual communication
|
||||||
|
# slower and consume nearly double the bandwidth as it will require sending reply SURBs.
|
||||||
|
#
|
||||||
|
# Note that some service providers might not support this.
|
||||||
|
send_anonymously = {{ socks5.send_anonymously }}
|
||||||
|
|
||||||
##### logging configuration options #####
|
##### logging configuration options #####
|
||||||
|
|
||||||
@@ -104,6 +109,9 @@ listening_port = {{ socks5.listening_port }}
|
|||||||
# The following options should not be modified unless you know EXACTLY what you are doing
|
# The following options should not be modified unless you know EXACTLY what you are doing
|
||||||
# as if set incorrectly, they may impact your anonymity.
|
# as if set incorrectly, they may impact your anonymity.
|
||||||
|
|
||||||
|
# [socks5_debug]
|
||||||
|
|
||||||
|
|
||||||
[debug]
|
[debug]
|
||||||
|
|
||||||
average_packet_delay = '{{ debug.average_packet_delay }}'
|
average_packet_delay = '{{ debug.average_packet_delay }}'
|
||||||
|
|||||||
@@ -1,44 +1,25 @@
|
|||||||
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use std::sync::atomic::Ordering;
|
|
||||||
|
|
||||||
use crate::client::config::Config;
|
use crate::client::config::Config;
|
||||||
use crate::error::Socks5ClientError;
|
use crate::error::Socks5ClientError;
|
||||||
|
use crate::socks;
|
||||||
use crate::socks::{
|
use crate::socks::{
|
||||||
authentication::{AuthenticationMethods, Authenticator, User},
|
authentication::{AuthenticationMethods, Authenticator, User},
|
||||||
server::SphinxSocksServer,
|
server::SphinxSocksServer,
|
||||||
};
|
};
|
||||||
use client_connections::{ConnectionCommandReceiver, ConnectionCommandSender, LaneQueueLengths};
|
use client_core::client::base_client::{
|
||||||
use client_core::client::cover_traffic_stream::LoopCoverTrafficStream;
|
non_wasm_helpers, BaseClientBuilder, ClientInput, ClientOutput,
|
||||||
use client_core::client::inbound_messages::{
|
|
||||||
InputMessage, InputMessageReceiver, InputMessageSender,
|
|
||||||
};
|
};
|
||||||
use client_core::client::key_manager::KeyManager;
|
use client_core::client::key_manager::KeyManager;
|
||||||
use client_core::client::mix_traffic::{BatchMixMessageSender, MixTrafficController};
|
|
||||||
use client_core::client::real_messages_control::RealMessagesController;
|
|
||||||
use client_core::client::received_buffer::{
|
|
||||||
ReceivedBufferRequestReceiver, ReceivedBufferRequestSender, ReceivedMessagesBufferController,
|
|
||||||
};
|
|
||||||
use client_core::client::reply_key_storage::ReplyKeyStorage;
|
|
||||||
use client_core::client::topology_control::{
|
|
||||||
TopologyAccessor, TopologyRefresher, TopologyRefresherConfig,
|
|
||||||
};
|
|
||||||
use client_core::config::persistence::key_pathfinder::ClientKeyPathfinder;
|
use client_core::config::persistence::key_pathfinder::ClientKeyPathfinder;
|
||||||
use client_core::error::ClientCoreError;
|
|
||||||
use crypto::asymmetric::identity;
|
|
||||||
use futures::channel::mpsc;
|
use futures::channel::mpsc;
|
||||||
use futures::StreamExt;
|
use futures::StreamExt;
|
||||||
use gateway_client::bandwidth::BandwidthController;
|
use gateway_client::bandwidth::BandwidthController;
|
||||||
use gateway_client::{
|
|
||||||
AcknowledgementReceiver, AcknowledgementSender, GatewayClient, MixnetMessageReceiver,
|
|
||||||
MixnetMessageSender,
|
|
||||||
};
|
|
||||||
use log::*;
|
use log::*;
|
||||||
use nymsphinx::addressing::clients::Recipient;
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
use nymsphinx::addressing::nodes::NodeIdentity;
|
use std::error::Error;
|
||||||
use tap::TapFallible;
|
use task::{wait_for_signal_and_error, ShutdownListener, ShutdownNotifier};
|
||||||
use task::{wait_for_signal, ShutdownListener, ShutdownNotifier};
|
|
||||||
|
|
||||||
pub mod config;
|
pub mod config;
|
||||||
|
|
||||||
@@ -72,132 +53,7 @@ impl NymClient {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn as_mix_recipient(&self) -> Recipient {
|
async fn create_bandwidth_controller(config: &Config) -> BandwidthController {
|
||||||
Recipient::new(
|
|
||||||
*self.key_manager.identity_keypair().public_key(),
|
|
||||||
*self.key_manager.encryption_keypair().public_key(),
|
|
||||||
// TODO: below only works under assumption that gateway address == gateway id
|
|
||||||
// (which currently is true)
|
|
||||||
NodeIdentity::from_base58_string(self.config.get_base().get_gateway_id()).unwrap(),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
// future constantly pumping loop cover traffic at some specified average rate
|
|
||||||
// the pumped traffic goes to the MixTrafficController
|
|
||||||
fn start_cover_traffic_stream(
|
|
||||||
&self,
|
|
||||||
topology_accessor: TopologyAccessor,
|
|
||||||
mix_tx: BatchMixMessageSender,
|
|
||||||
shutdown: ShutdownListener,
|
|
||||||
) {
|
|
||||||
info!("Starting loop cover traffic stream...");
|
|
||||||
|
|
||||||
let mut stream = LoopCoverTrafficStream::new(
|
|
||||||
self.key_manager.ack_key(),
|
|
||||||
self.config.get_base().get_average_ack_delay(),
|
|
||||||
self.config.get_base().get_average_packet_delay(),
|
|
||||||
self.config
|
|
||||||
.get_base()
|
|
||||||
.get_loop_cover_traffic_average_delay(),
|
|
||||||
mix_tx,
|
|
||||||
self.as_mix_recipient(),
|
|
||||||
topology_accessor,
|
|
||||||
);
|
|
||||||
|
|
||||||
if let Some(size) = self.config.get_base().get_use_extended_packet_size() {
|
|
||||||
log::debug!("Setting extended packet size: {:?}", size);
|
|
||||||
stream.set_custom_packet_size(size.into());
|
|
||||||
}
|
|
||||||
|
|
||||||
stream.start_with_shutdown(shutdown);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[allow(clippy::too_many_arguments)]
|
|
||||||
fn start_real_traffic_controller(
|
|
||||||
&self,
|
|
||||||
topology_accessor: TopologyAccessor,
|
|
||||||
reply_key_storage: ReplyKeyStorage,
|
|
||||||
ack_receiver: AcknowledgementReceiver,
|
|
||||||
input_receiver: InputMessageReceiver,
|
|
||||||
mix_sender: BatchMixMessageSender,
|
|
||||||
client_connection_rx: ConnectionCommandReceiver,
|
|
||||||
lane_queue_lengths: LaneQueueLengths,
|
|
||||||
shutdown: ShutdownListener,
|
|
||||||
) {
|
|
||||||
let mut controller_config = client_core::client::real_messages_control::Config::new(
|
|
||||||
self.key_manager.ack_key(),
|
|
||||||
self.config.get_base().get_ack_wait_multiplier(),
|
|
||||||
self.config.get_base().get_ack_wait_addition(),
|
|
||||||
self.config.get_base().get_average_ack_delay(),
|
|
||||||
self.config.get_base().get_message_sending_average_delay(),
|
|
||||||
self.config.get_base().get_average_packet_delay(),
|
|
||||||
self.config
|
|
||||||
.get_base()
|
|
||||||
.get_disabled_main_poisson_packet_distribution(),
|
|
||||||
self.as_mix_recipient(),
|
|
||||||
);
|
|
||||||
|
|
||||||
if let Some(size) = self.config.get_base().get_use_extended_packet_size() {
|
|
||||||
log::debug!("Setting extended packet size: {:?}", size);
|
|
||||||
controller_config.set_custom_packet_size(size.into());
|
|
||||||
}
|
|
||||||
|
|
||||||
info!("Starting real traffic stream...");
|
|
||||||
|
|
||||||
RealMessagesController::new(
|
|
||||||
controller_config,
|
|
||||||
ack_receiver,
|
|
||||||
input_receiver,
|
|
||||||
mix_sender,
|
|
||||||
topology_accessor,
|
|
||||||
reply_key_storage,
|
|
||||||
lane_queue_lengths,
|
|
||||||
client_connection_rx,
|
|
||||||
)
|
|
||||||
.start_with_shutdown(shutdown);
|
|
||||||
}
|
|
||||||
|
|
||||||
// buffer controlling all messages fetched from provider
|
|
||||||
// required so that other components would be able to use them (say the websocket)
|
|
||||||
fn start_received_messages_buffer_controller(
|
|
||||||
&self,
|
|
||||||
query_receiver: ReceivedBufferRequestReceiver,
|
|
||||||
mixnet_receiver: MixnetMessageReceiver,
|
|
||||||
reply_key_storage: ReplyKeyStorage,
|
|
||||||
shutdown: ShutdownListener,
|
|
||||||
) {
|
|
||||||
info!("Starting received messages buffer controller...");
|
|
||||||
ReceivedMessagesBufferController::new(
|
|
||||||
self.key_manager.encryption_keypair(),
|
|
||||||
query_receiver,
|
|
||||||
mixnet_receiver,
|
|
||||||
reply_key_storage,
|
|
||||||
)
|
|
||||||
.start_with_shutdown(shutdown);
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn start_gateway_client(
|
|
||||||
&mut self,
|
|
||||||
mixnet_message_sender: MixnetMessageSender,
|
|
||||||
ack_sender: AcknowledgementSender,
|
|
||||||
shutdown: ShutdownListener,
|
|
||||||
) -> GatewayClient {
|
|
||||||
let gateway_id = self.config.get_base().get_gateway_id();
|
|
||||||
if gateway_id.is_empty() {
|
|
||||||
panic!("The identity of the gateway is unknown - did you run `nym-client` init?")
|
|
||||||
}
|
|
||||||
let gateway_owner = self.config.get_base().get_gateway_owner();
|
|
||||||
if gateway_owner.is_empty() {
|
|
||||||
panic!("The owner of the gateway is unknown - did you run `nym-client` init?")
|
|
||||||
}
|
|
||||||
let gateway_address = self.config.get_base().get_gateway_listener();
|
|
||||||
if gateway_address.is_empty() {
|
|
||||||
panic!("The address of the gateway is unknown - did you run `nym-client` init?")
|
|
||||||
}
|
|
||||||
|
|
||||||
let gateway_identity = identity::PublicKey::from_base58_string(gateway_id)
|
|
||||||
.expect("provided gateway id is invalid!");
|
|
||||||
|
|
||||||
#[cfg(feature = "coconut")]
|
#[cfg(feature = "coconut")]
|
||||||
let bandwidth_controller = {
|
let bandwidth_controller = {
|
||||||
let details = network_defaults::NymNetworkDetails::new_from_env();
|
let details = network_defaults::NymNetworkDetails::new_from_env();
|
||||||
@@ -210,140 +66,97 @@ impl NymClient {
|
|||||||
.await
|
.await
|
||||||
.expect("Could not query api clients");
|
.expect("Could not query api clients");
|
||||||
BandwidthController::new(
|
BandwidthController::new(
|
||||||
credential_storage::initialise_storage(self.config.get_base().get_database_path())
|
credential_storage::initialise_storage(config.get_base().get_database_path()).await,
|
||||||
.await,
|
|
||||||
coconut_api_clients,
|
coconut_api_clients,
|
||||||
)
|
)
|
||||||
};
|
};
|
||||||
#[cfg(not(feature = "coconut"))]
|
#[cfg(not(feature = "coconut"))]
|
||||||
let bandwidth_controller = BandwidthController::new(
|
let bandwidth_controller = BandwidthController::new(
|
||||||
credential_storage::initialise_storage(self.config.get_base().get_database_path())
|
credential_storage::initialise_storage(config.get_base().get_database_path()).await,
|
||||||
.await,
|
|
||||||
)
|
)
|
||||||
.expect("Could not create bandwidth controller");
|
.expect("Could not create bandwidth controller");
|
||||||
|
bandwidth_controller
|
||||||
let mut gateway_client = GatewayClient::new(
|
|
||||||
gateway_address,
|
|
||||||
self.key_manager.identity_keypair(),
|
|
||||||
gateway_identity,
|
|
||||||
gateway_owner,
|
|
||||||
Some(self.key_manager.gateway_shared_key()),
|
|
||||||
mixnet_message_sender,
|
|
||||||
ack_sender,
|
|
||||||
self.config.get_base().get_gateway_response_timeout(),
|
|
||||||
Some(bandwidth_controller),
|
|
||||||
Some(shutdown),
|
|
||||||
);
|
|
||||||
|
|
||||||
gateway_client
|
|
||||||
.set_disabled_credentials_mode(self.config.get_base().get_disabled_credentials_mode());
|
|
||||||
|
|
||||||
gateway_client
|
|
||||||
.authenticate_and_start()
|
|
||||||
.await
|
|
||||||
.expect("could not authenticate and start up the gateway connection");
|
|
||||||
|
|
||||||
gateway_client
|
|
||||||
}
|
|
||||||
|
|
||||||
// future responsible for periodically polling directory server and updating
|
|
||||||
// the current global view of topology
|
|
||||||
async fn start_topology_refresher(
|
|
||||||
&mut self,
|
|
||||||
topology_accessor: TopologyAccessor,
|
|
||||||
shutdown: ShutdownListener,
|
|
||||||
) -> Result<(), Socks5ClientError> {
|
|
||||||
let topology_refresher_config = TopologyRefresherConfig::new(
|
|
||||||
self.config.get_base().get_validator_api_endpoints(),
|
|
||||||
self.config.get_base().get_topology_refresh_rate(),
|
|
||||||
env!("CARGO_PKG_VERSION").to_string(),
|
|
||||||
);
|
|
||||||
let mut topology_refresher =
|
|
||||||
TopologyRefresher::new(topology_refresher_config, topology_accessor);
|
|
||||||
// before returning, block entire runtime to refresh the current network view so that any
|
|
||||||
// components depending on topology would see a non-empty view
|
|
||||||
info!("Obtaining initial network topology");
|
|
||||||
topology_refresher.refresh().await;
|
|
||||||
|
|
||||||
// TODO: a slightly more graceful termination here
|
|
||||||
if !topology_refresher.is_topology_routable().await {
|
|
||||||
log::error!(
|
|
||||||
"The current network topology seem to be insufficient to route any packets through \
|
|
||||||
- check if enough nodes and a gateway are online"
|
|
||||||
);
|
|
||||||
return Err(ClientCoreError::InsufficientNetworkTopology.into());
|
|
||||||
}
|
|
||||||
|
|
||||||
info!("Starting topology refresher...");
|
|
||||||
topology_refresher.start_with_shutdown(shutdown);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
// controller for sending sphinx packets to mixnet (either real traffic or cover traffic)
|
|
||||||
// TODO: if we want to send control messages to gateway_client, this CAN'T take the ownership
|
|
||||||
// over it. Perhaps GatewayClient needs to be thread-shareable or have some channel for
|
|
||||||
// requests?
|
|
||||||
fn start_mix_traffic_controller(
|
|
||||||
gateway_client: GatewayClient,
|
|
||||||
shutdown: ShutdownListener,
|
|
||||||
) -> BatchMixMessageSender {
|
|
||||||
info!("Starting mix traffic controller...");
|
|
||||||
let (mix_traffic_controller, mix_tx) = MixTrafficController::new(gateway_client);
|
|
||||||
mix_traffic_controller.start_with_shutdown(shutdown);
|
|
||||||
mix_tx
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn start_socks5_listener(
|
fn start_socks5_listener(
|
||||||
&self,
|
config: &Config,
|
||||||
buffer_requester: ReceivedBufferRequestSender,
|
client_input: ClientInput,
|
||||||
msg_input: InputMessageSender,
|
client_output: ClientOutput,
|
||||||
client_connection_tx: ConnectionCommandSender,
|
self_address: Recipient,
|
||||||
lane_queue_lengths: LaneQueueLengths,
|
|
||||||
shutdown: ShutdownListener,
|
shutdown: ShutdownListener,
|
||||||
) {
|
) {
|
||||||
info!("Starting socks5 listener...");
|
info!("Starting socks5 listener...");
|
||||||
let auth_methods = vec![AuthenticationMethods::NoAuth as u8];
|
let auth_methods = vec![AuthenticationMethods::NoAuth as u8];
|
||||||
let allowed_users: Vec<User> = Vec::new();
|
let allowed_users: Vec<User> = Vec::new();
|
||||||
|
|
||||||
|
let ClientInput {
|
||||||
|
shared_lane_queue_lengths,
|
||||||
|
connection_command_sender,
|
||||||
|
input_sender,
|
||||||
|
} = client_input;
|
||||||
|
|
||||||
|
let received_buffer_request_sender = client_output.received_buffer_request_sender;
|
||||||
|
|
||||||
let authenticator = Authenticator::new(auth_methods, allowed_users);
|
let authenticator = Authenticator::new(auth_methods, allowed_users);
|
||||||
let mut sphinx_socks = SphinxSocksServer::new(
|
let mut sphinx_socks = SphinxSocksServer::new(
|
||||||
self.config.get_listening_port(),
|
config.get_listening_port(),
|
||||||
authenticator,
|
authenticator,
|
||||||
self.config.get_provider_mix_address(),
|
config.get_provider_mix_address(),
|
||||||
self.as_mix_recipient(),
|
self_address,
|
||||||
lane_queue_lengths,
|
shared_lane_queue_lengths,
|
||||||
|
socks::client::Config::new(
|
||||||
|
config.get_send_anonymously(),
|
||||||
|
config.get_connection_start_surbs(),
|
||||||
|
config.get_per_request_surbs(),
|
||||||
|
),
|
||||||
|
shutdown.clone(),
|
||||||
|
);
|
||||||
|
task::spawn_with_report_error(
|
||||||
|
async move {
|
||||||
|
sphinx_socks
|
||||||
|
.serve(
|
||||||
|
input_sender,
|
||||||
|
received_buffer_request_sender,
|
||||||
|
connection_command_sender,
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
},
|
||||||
shutdown,
|
shutdown,
|
||||||
);
|
);
|
||||||
tokio::spawn(async move {
|
|
||||||
sphinx_socks
|
|
||||||
.serve(msg_input, buffer_requester, client_connection_tx)
|
|
||||||
.await
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// blocking version of `start` method. Will run forever (or until SIGINT is sent)
|
/// blocking version of `start` method. Will run forever (or until SIGINT is sent)
|
||||||
pub async fn run_forever(&mut self) -> Result<(), Socks5ClientError> {
|
pub async fn run_forever(self) -> Result<(), Box<dyn Error + Send>> {
|
||||||
let mut shutdown = self.start().await?;
|
let mut shutdown = self
|
||||||
wait_for_signal().await;
|
.start()
|
||||||
|
.await
|
||||||
|
.map_err(|err| Box::new(err) as Box<dyn Error + Send>)?;
|
||||||
|
|
||||||
|
let res = wait_for_signal_and_error(&mut shutdown).await;
|
||||||
|
|
||||||
log::info!("Sending shutdown");
|
log::info!("Sending shutdown");
|
||||||
client_core::client::SHUTDOWN_HAS_BEEN_SIGNALLED.store(true, Ordering::Relaxed);
|
|
||||||
shutdown.signal_shutdown().ok();
|
shutdown.signal_shutdown().ok();
|
||||||
|
|
||||||
log::info!("Waiting for tasks to finish... (Press ctrl-c to force)");
|
log::info!("Waiting for tasks to finish... (Press ctrl-c to force)");
|
||||||
shutdown.wait_for_shutdown().await;
|
shutdown.wait_for_shutdown().await;
|
||||||
|
|
||||||
log::info!("Stopping nym-socks5-client");
|
log::info!("Stopping nym-socks5-client");
|
||||||
Ok(())
|
res
|
||||||
}
|
}
|
||||||
|
|
||||||
// Variant of `run_forever` that listends for remote control messages
|
// Variant of `run_forever` that listends for remote control messages
|
||||||
pub async fn run_and_listen(
|
pub async fn run_and_listen(
|
||||||
&mut self,
|
self,
|
||||||
mut receiver: Socks5ControlMessageReceiver,
|
mut receiver: Socks5ControlMessageReceiver,
|
||||||
) -> Result<(), Socks5ClientError> {
|
) -> Result<(), Box<dyn Error + Send>> {
|
||||||
let mut shutdown = self.start().await?;
|
// Start the main task
|
||||||
tokio::select! {
|
let mut shutdown = self
|
||||||
|
.start()
|
||||||
|
.await
|
||||||
|
.map_err(|err| Box::new(err) as Box<dyn Error + Send>)?;
|
||||||
|
|
||||||
|
let res = tokio::select! {
|
||||||
|
biased;
|
||||||
message = receiver.next() => {
|
message = receiver.next() => {
|
||||||
log::debug!("Received message: {:?}", message);
|
log::debug!("Received message: {:?}", message);
|
||||||
match message {
|
match message {
|
||||||
@@ -354,118 +167,56 @@ impl NymClient {
|
|||||||
log::info!("Channel closed, stopping");
|
log::info!("Channel closed, stopping");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
Some(msg) = shutdown.wait_for_error() => {
|
||||||
|
log::info!("Task error: {:?}", msg);
|
||||||
|
Err(msg)
|
||||||
}
|
}
|
||||||
_ = tokio::signal::ctrl_c() => {
|
_ = tokio::signal::ctrl_c() => {
|
||||||
log::info!("Received SIGINT");
|
log::info!("Received SIGINT");
|
||||||
|
Ok(())
|
||||||
},
|
},
|
||||||
}
|
};
|
||||||
|
|
||||||
log::info!("Sending shutdown");
|
log::info!("Sending shutdown");
|
||||||
client_core::client::SHUTDOWN_HAS_BEEN_SIGNALLED.store(true, Ordering::Relaxed);
|
|
||||||
shutdown.signal_shutdown().ok();
|
shutdown.signal_shutdown().ok();
|
||||||
|
|
||||||
log::info!("Waiting for tasks to finish... (Press ctrl-c to force)");
|
log::info!("Waiting for tasks to finish... (Press ctrl-c to force)");
|
||||||
shutdown.wait_for_shutdown().await;
|
shutdown.wait_for_shutdown().await;
|
||||||
|
|
||||||
log::info!("Stopping nym-socks5-client");
|
log::info!("Stopping nym-socks5-client");
|
||||||
Ok(())
|
res
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn start(&mut self) -> Result<ShutdownNotifier, Socks5ClientError> {
|
pub async fn start(self) -> Result<ShutdownNotifier, Socks5ClientError> {
|
||||||
info!("Starting nym client");
|
let base_builder = BaseClientBuilder::new_from_base_config(
|
||||||
// channels for inter-component communication
|
self.config.get_base(),
|
||||||
// TODO: make the channels be internally created by the relevant components
|
self.key_manager,
|
||||||
// rather than creating them here, so say for example the buffer controller would create the request channels
|
Some(Self::create_bandwidth_controller(&self.config).await),
|
||||||
// and would allow anyone to clone the sender channel
|
non_wasm_helpers::setup_fs_reply_surb_backend(
|
||||||
|
self.config.get_base().get_reply_surb_database_path(),
|
||||||
// unwrapped_sphinx_sender is the transmitter of mixnet messages received from the gateway
|
self.config.get_debug_settings(),
|
||||||
// unwrapped_sphinx_receiver is the receiver for said messages - used by ReceivedMessagesBuffer
|
)
|
||||||
let (mixnet_messages_sender, mixnet_messages_receiver) = mpsc::unbounded();
|
.await?,
|
||||||
|
|
||||||
// used for announcing connection or disconnection of a channel for pushing re-assembled messages to
|
|
||||||
let (received_buffer_request_sender, received_buffer_request_receiver) = mpsc::unbounded();
|
|
||||||
|
|
||||||
// channels responsible for controlling real messages
|
|
||||||
let (input_sender, input_receiver) = tokio::sync::mpsc::channel::<InputMessage>(1);
|
|
||||||
|
|
||||||
// channels responsible for controlling ack messages
|
|
||||||
let (ack_sender, ack_receiver) = mpsc::unbounded();
|
|
||||||
let shared_topology_accessor = TopologyAccessor::new();
|
|
||||||
|
|
||||||
let reply_key_storage =
|
|
||||||
ReplyKeyStorage::load(self.config.get_base().get_reply_encryption_key_store_path())
|
|
||||||
.tap_err(|err| {
|
|
||||||
log::error!("Failed to load reply key storage - is it perhaps already in use?");
|
|
||||||
log::error!("{}", err);
|
|
||||||
})?;
|
|
||||||
|
|
||||||
// Shutdown notifier for signalling tasks to stop
|
|
||||||
let shutdown = ShutdownNotifier::default();
|
|
||||||
|
|
||||||
// the components are started in very specific order. Unless you know what you are doing,
|
|
||||||
// do not change that.
|
|
||||||
self.start_topology_refresher(shared_topology_accessor.clone(), shutdown.subscribe())
|
|
||||||
.await?;
|
|
||||||
self.start_received_messages_buffer_controller(
|
|
||||||
received_buffer_request_receiver,
|
|
||||||
mixnet_messages_receiver,
|
|
||||||
reply_key_storage.clone(),
|
|
||||||
shutdown.subscribe(),
|
|
||||||
);
|
);
|
||||||
|
|
||||||
let gateway_client = self
|
let self_address = base_builder.as_mix_recipient();
|
||||||
.start_gateway_client(mixnet_messages_sender, ack_sender, shutdown.subscribe())
|
let mut started_client = base_builder.start_base().await?;
|
||||||
.await;
|
let client_input = started_client.client_input.register_producer();
|
||||||
|
let client_output = started_client.client_output.register_consumer();
|
||||||
|
|
||||||
// The sphinx_message_sender is the transmitter for any component generating sphinx packets
|
Self::start_socks5_listener(
|
||||||
// that are to be sent to the mixnet. They are used by cover traffic stream and real
|
&self.config,
|
||||||
// traffic stream.
|
client_input,
|
||||||
// The MixTrafficController then sends the actual traffic
|
client_output,
|
||||||
let sphinx_message_sender =
|
self_address,
|
||||||
Self::start_mix_traffic_controller(gateway_client, shutdown.subscribe());
|
started_client.shutdown_notifier.subscribe(),
|
||||||
|
|
||||||
// Channel for announcing closed (socks5) connections by the controller.
|
|
||||||
// This will be forwarded to `OutQueueControl`
|
|
||||||
let (client_connection_tx, client_connection_rx) = mpsc::unbounded();
|
|
||||||
|
|
||||||
// Shared queue length data. Published by the `OutQueueController` in the client, and used
|
|
||||||
// primarily to throttle incoming connections
|
|
||||||
let shared_lane_queue_lengths = LaneQueueLengths::new();
|
|
||||||
|
|
||||||
self.start_real_traffic_controller(
|
|
||||||
shared_topology_accessor.clone(),
|
|
||||||
reply_key_storage,
|
|
||||||
ack_receiver,
|
|
||||||
input_receiver,
|
|
||||||
sphinx_message_sender.clone(),
|
|
||||||
client_connection_rx,
|
|
||||||
shared_lane_queue_lengths.clone(),
|
|
||||||
shutdown.subscribe(),
|
|
||||||
);
|
|
||||||
|
|
||||||
if !self
|
|
||||||
.config
|
|
||||||
.get_base()
|
|
||||||
.get_disabled_loop_cover_traffic_stream()
|
|
||||||
{
|
|
||||||
self.start_cover_traffic_stream(
|
|
||||||
shared_topology_accessor,
|
|
||||||
sphinx_message_sender,
|
|
||||||
shutdown.subscribe(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
self.start_socks5_listener(
|
|
||||||
received_buffer_request_sender,
|
|
||||||
input_sender,
|
|
||||||
client_connection_tx,
|
|
||||||
shared_lane_queue_lengths,
|
|
||||||
shutdown.subscribe(),
|
|
||||||
);
|
);
|
||||||
|
|
||||||
info!("Client startup finished!");
|
info!("Client startup finished!");
|
||||||
info!("The address of this client is: {}", self.as_mix_recipient());
|
info!("The address of this client is: {}", self_address);
|
||||||
|
|
||||||
Ok(shutdown)
|
Ok(started_client.shutdown_notifier)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,7 +2,8 @@
|
|||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use clap::Args;
|
use clap::Args;
|
||||||
use client_core::{config::GatewayEndpoint, error::ClientCoreError};
|
use client_core::client::replies::reply_storage::fs_backend;
|
||||||
|
use client_core::{config::GatewayEndpointConfig, error::ClientCoreError};
|
||||||
use config::NymConfig;
|
use config::NymConfig;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
@@ -20,6 +21,14 @@ pub(crate) struct Init {
|
|||||||
#[clap(long)]
|
#[clap(long)]
|
||||||
provider: String,
|
provider: String,
|
||||||
|
|
||||||
|
/// Specifies whether this client is going to use an anonymous sender tag for communication with the service provider.
|
||||||
|
/// While this is going to hide its actual address information, it will make the actual communication
|
||||||
|
/// slower and consume nearly double the bandwidth as it will require sending reply SURBs.
|
||||||
|
///
|
||||||
|
/// Note that some service providers might not support this.
|
||||||
|
#[clap(long)]
|
||||||
|
use_anonymous_sender_tag: bool,
|
||||||
|
|
||||||
/// Id of the gateway we are going to connect to.
|
/// Id of the gateway we are going to connect to.
|
||||||
#[clap(long)]
|
#[clap(long)]
|
||||||
gateway: Option<String>,
|
gateway: Option<String>,
|
||||||
@@ -46,6 +55,10 @@ pub(crate) struct Init {
|
|||||||
#[clap(long, hidden = true)]
|
#[clap(long, hidden = true)]
|
||||||
fastmode: bool,
|
fastmode: bool,
|
||||||
|
|
||||||
|
/// Disable loop cover traffic and the Poisson rate limiter (for debugging only)
|
||||||
|
#[clap(long, hidden = true)]
|
||||||
|
no_cover: bool,
|
||||||
|
|
||||||
/// Set this client to work in a enabled credentials mode that would attempt to use gateway
|
/// Set this client to work in a enabled credentials mode that would attempt to use gateway
|
||||||
/// with bandwidth credential requirement.
|
/// with bandwidth credential requirement.
|
||||||
#[cfg(feature = "coconut")]
|
#[cfg(feature = "coconut")]
|
||||||
@@ -59,7 +72,9 @@ impl From<Init> for OverrideConfig {
|
|||||||
nymd_validators: init_config.nymd_validators,
|
nymd_validators: init_config.nymd_validators,
|
||||||
api_validators: init_config.api_validators,
|
api_validators: init_config.api_validators,
|
||||||
port: init_config.port,
|
port: init_config.port,
|
||||||
|
use_anonymous_sender_tag: init_config.use_anonymous_sender_tag,
|
||||||
fastmode: init_config.fastmode,
|
fastmode: init_config.fastmode,
|
||||||
|
no_cover: init_config.no_cover,
|
||||||
#[cfg(feature = "coconut")]
|
#[cfg(feature = "coconut")]
|
||||||
enabled_credentials_mode: init_config.enabled_credentials_mode,
|
enabled_credentials_mode: init_config.enabled_credentials_mode,
|
||||||
}
|
}
|
||||||
@@ -120,10 +135,12 @@ pub(crate) async fn execute(args: &Init) {
|
|||||||
);
|
);
|
||||||
println!("Client configuration completed.");
|
println!("Client configuration completed.");
|
||||||
|
|
||||||
client_core::init::show_address(config.get_base()).unwrap_or_else(|err| {
|
client_core::init::show_address::<_, fs_backend::Backend>(config.get_base()).unwrap_or_else(
|
||||||
eprintln!("Failed to show address\nError: {err}");
|
|err| {
|
||||||
std::process::exit(1)
|
eprintln!("Failed to show address\nError: {err}");
|
||||||
});
|
std::process::exit(1)
|
||||||
|
},
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn setup_gateway(
|
async fn setup_gateway(
|
||||||
@@ -131,7 +148,7 @@ async fn setup_gateway(
|
|||||||
register: bool,
|
register: bool,
|
||||||
user_chosen_gateway_id: Option<&str>,
|
user_chosen_gateway_id: Option<&str>,
|
||||||
config: &Config,
|
config: &Config,
|
||||||
) -> Result<GatewayEndpoint, ClientCoreError> {
|
) -> Result<GatewayEndpointConfig, ClientCoreError<fs_backend::Backend>> {
|
||||||
if register {
|
if register {
|
||||||
// Get the gateway details by querying the validator-api. Either pick one at random or use
|
// Get the gateway details by querying the validator-api. Either pick one at random or use
|
||||||
// the chosen one if it's among the available ones.
|
// the chosen one if it's among the available ones.
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use std::error::Error;
|
||||||
|
|
||||||
use crate::client::config::Config;
|
use crate::client::config::Config;
|
||||||
use crate::error::Socks5ClientError;
|
|
||||||
use clap::CommandFactory;
|
use clap::CommandFactory;
|
||||||
use clap::{Parser, Subcommand};
|
use clap::{Parser, Subcommand};
|
||||||
use completions::{fig_generate, ArgShell};
|
use completions::{fig_generate, ArgShell};
|
||||||
@@ -81,13 +82,15 @@ pub(crate) struct OverrideConfig {
|
|||||||
nymd_validators: Option<String>,
|
nymd_validators: Option<String>,
|
||||||
api_validators: Option<String>,
|
api_validators: Option<String>,
|
||||||
port: Option<u16>,
|
port: Option<u16>,
|
||||||
|
use_anonymous_sender_tag: bool,
|
||||||
fastmode: bool,
|
fastmode: bool,
|
||||||
|
no_cover: bool,
|
||||||
|
|
||||||
#[cfg(feature = "coconut")]
|
#[cfg(feature = "coconut")]
|
||||||
enabled_credentials_mode: bool,
|
enabled_credentials_mode: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) async fn execute(args: &Cli) -> Result<(), Socks5ClientError> {
|
pub(crate) async fn execute(args: &Cli) -> Result<(), Box<dyn Error + Send>> {
|
||||||
let bin_name = "nym-socks5-client";
|
let bin_name = "nym-socks5-client";
|
||||||
|
|
||||||
match &args.command {
|
match &args.command {
|
||||||
@@ -120,6 +123,10 @@ pub(crate) fn override_config(mut config: Config, args: OverrideConfig) -> Confi
|
|||||||
.set_custom_validator_apis(parse_validators(&raw_validators));
|
.set_custom_validator_apis(parse_validators(&raw_validators));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if args.use_anonymous_sender_tag {
|
||||||
|
config = config.with_anonymous_replies(true)
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(port) = args.port {
|
if let Some(port) = args.port {
|
||||||
config = config.with_port(port);
|
config = config.with_port(port);
|
||||||
}
|
}
|
||||||
@@ -135,6 +142,10 @@ pub(crate) fn override_config(mut config: Config, args: OverrideConfig) -> Confi
|
|||||||
config.get_base_mut().set_high_default_traffic_volume();
|
config.get_base_mut().set_high_default_traffic_volume();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if args.no_cover {
|
||||||
|
config.get_base_mut().set_no_cover_traffic();
|
||||||
|
}
|
||||||
|
|
||||||
config
|
config
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -22,6 +22,14 @@ pub(crate) struct Run {
|
|||||||
#[clap(long)]
|
#[clap(long)]
|
||||||
config: Option<String>,
|
config: Option<String>,
|
||||||
|
|
||||||
|
/// Specifies whether this client is going to use an anonymous sender tag for communication with the service provider.
|
||||||
|
/// While this is going to hide its actual address information, it will make the actual communication
|
||||||
|
/// slower and consume nearly double the bandwidth as it will require sending reply SURBs.
|
||||||
|
///
|
||||||
|
/// Note that some service providers might not support this.
|
||||||
|
#[clap(long)]
|
||||||
|
use_anonymous_sender_tag: bool,
|
||||||
|
|
||||||
/// Address of the socks5 provider to send messages to.
|
/// Address of the socks5 provider to send messages to.
|
||||||
#[clap(long)]
|
#[clap(long)]
|
||||||
provider: Option<String>,
|
provider: Option<String>,
|
||||||
@@ -43,6 +51,15 @@ pub(crate) struct Run {
|
|||||||
#[clap(short, long)]
|
#[clap(short, long)]
|
||||||
port: Option<u16>,
|
port: Option<u16>,
|
||||||
|
|
||||||
|
/// Mostly debug-related option to increase default traffic rate so that you would not need to
|
||||||
|
/// modify config post init
|
||||||
|
#[clap(long, hidden = true)]
|
||||||
|
fastmode: bool,
|
||||||
|
|
||||||
|
/// Disable loop cover traffic and the Poisson rate limiter (for debugging only)
|
||||||
|
#[clap(long, hidden = true)]
|
||||||
|
no_cover: bool,
|
||||||
|
|
||||||
/// Set this client to work in a enabled credentials mode that would attempt to use gateway
|
/// Set this client to work in a enabled credentials mode that would attempt to use gateway
|
||||||
/// with bandwidth credential requirement.
|
/// with bandwidth credential requirement.
|
||||||
#[cfg(feature = "coconut")]
|
#[cfg(feature = "coconut")]
|
||||||
@@ -56,8 +73,9 @@ impl From<Run> for OverrideConfig {
|
|||||||
nymd_validators: run_config.nymd_validators,
|
nymd_validators: run_config.nymd_validators,
|
||||||
api_validators: run_config.api_validators,
|
api_validators: run_config.api_validators,
|
||||||
port: run_config.port,
|
port: run_config.port,
|
||||||
fastmode: false,
|
use_anonymous_sender_tag: run_config.use_anonymous_sender_tag,
|
||||||
|
fastmode: run_config.fastmode,
|
||||||
|
no_cover: run_config.no_cover,
|
||||||
#[cfg(feature = "coconut")]
|
#[cfg(feature = "coconut")]
|
||||||
enabled_credentials_mode: run_config.enabled_credentials_mode,
|
enabled_credentials_mode: run_config.enabled_credentials_mode,
|
||||||
}
|
}
|
||||||
@@ -86,14 +104,16 @@ fn version_check(cfg: &Config) -> bool {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) async fn execute(args: &Run) -> Result<(), Socks5ClientError> {
|
pub(crate) async fn execute(args: &Run) -> Result<(), Box<dyn std::error::Error + Send>> {
|
||||||
let id = &args.id;
|
let id = &args.id;
|
||||||
|
|
||||||
let mut config = match Config::load_from_file(Some(id)) {
|
let mut config = match Config::load_from_file(Some(id)) {
|
||||||
Ok(cfg) => cfg,
|
Ok(cfg) => cfg,
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
error!("Failed to load config for {}. Are you sure you have run `init` before? (Error was: {})", id, err);
|
error!("Failed to load config for {}. Are you sure you have run `init` before? (Error was: {})", id, err);
|
||||||
return Err(Socks5ClientError::FailedToLoadConfig(id.to_string()));
|
return Err(Box::new(Socks5ClientError::FailedToLoadConfig(
|
||||||
|
id.to_string(),
|
||||||
|
)));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -102,7 +122,7 @@ pub(crate) async fn execute(args: &Run) -> Result<(), Socks5ClientError> {
|
|||||||
|
|
||||||
if !version_check(&config) {
|
if !version_check(&config) {
|
||||||
error!("failed the local version check");
|
error!("failed the local version check");
|
||||||
return Err(Socks5ClientError::FailedLocalVersionCheck);
|
return Err(Box::new(Socks5ClientError::FailedLocalVersionCheck));
|
||||||
}
|
}
|
||||||
|
|
||||||
NymClient::new(config).run_forever().await
|
NymClient::new(config).run_forever().await
|
||||||
|
|||||||
+12
-13
@@ -1,25 +1,24 @@
|
|||||||
use client_core::{client::reply_key_storage::ReplyKeyStorageError, error::ClientCoreError};
|
use crate::socks::types::SocksProxyError;
|
||||||
use crypto::asymmetric::identity::Ed25519RecoveryError;
|
use client_core::client::replies::reply_storage::fs_backend;
|
||||||
use gateway_client::error::GatewayClientError;
|
use client_core::error::ClientCoreError;
|
||||||
use validator_client::ValidatorClientError;
|
|
||||||
|
|
||||||
#[derive(thiserror::Error, Debug)]
|
#[derive(thiserror::Error, Debug)]
|
||||||
pub enum Socks5ClientError {
|
pub enum Socks5ClientError {
|
||||||
#[error("I/O error: {0}")]
|
#[error("I/O error: {0}")]
|
||||||
IoError(#[from] std::io::Error),
|
IoError(#[from] std::io::Error),
|
||||||
#[error("Gateway client error: {0}")]
|
|
||||||
GatewayClientError(#[from] GatewayClientError),
|
|
||||||
#[error("Ed25519 error: {0}")]
|
|
||||||
Ed25519RecoveryError(#[from] Ed25519RecoveryError),
|
|
||||||
#[error("Validator client error: {0}")]
|
|
||||||
ValidatorClientError(#[from] ValidatorClientError),
|
|
||||||
#[error("client-core error: {0}")]
|
#[error("client-core error: {0}")]
|
||||||
ClientCoreError(#[from] ClientCoreError),
|
ClientCoreError(#[from] ClientCoreError<fs_backend::Backend>),
|
||||||
#[error("Reply key storage error: {0}")]
|
|
||||||
ReplyKeyStorageError(#[from] ReplyKeyStorageError),
|
#[error("SOCKS proxy error")]
|
||||||
|
SocksProxyError(SocksProxyError),
|
||||||
|
|
||||||
#[error("Failed to load config for: {0}")]
|
#[error("Failed to load config for: {0}")]
|
||||||
FailedToLoadConfig(String),
|
FailedToLoadConfig(String),
|
||||||
|
|
||||||
#[error("Failed local version check, client and config mismatch")]
|
#[error("Failed local version check, client and config mismatch")]
|
||||||
FailedLocalVersionCheck,
|
FailedLocalVersionCheck,
|
||||||
|
|
||||||
|
#[error("Fail to bind address")]
|
||||||
|
FailToBindAddress,
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
// Copyright 2021 - Nym Technologies SA <contact@nymtech.net>
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use std::error::Error;
|
||||||
|
|
||||||
use clap::{crate_version, Parser};
|
use clap::{crate_version, Parser};
|
||||||
use error::Socks5ClientError;
|
|
||||||
use logging::setup_logging;
|
use logging::setup_logging;
|
||||||
use network_defaults::setup_env;
|
use network_defaults::setup_env;
|
||||||
|
|
||||||
@@ -12,7 +13,7 @@ pub mod error;
|
|||||||
pub mod socks;
|
pub mod socks;
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() -> Result<(), Socks5ClientError> {
|
async fn main() -> Result<(), Box<dyn Error + Send>> {
|
||||||
setup_logging();
|
setup_logging();
|
||||||
println!("{}", banner());
|
println!("{}", banner());
|
||||||
|
|
||||||
|
|||||||
@@ -2,8 +2,8 @@
|
|||||||
|
|
||||||
use super::authentication::{AuthenticationMethods, Authenticator, User};
|
use super::authentication::{AuthenticationMethods, Authenticator, User};
|
||||||
use super::request::{SocksCommand, SocksRequest};
|
use super::request::{SocksCommand, SocksRequest};
|
||||||
use super::types::{ResponseCode, SocksProxyError};
|
use super::types::{ResponseCodeV4, ResponseCodeV5, SocksProxyError};
|
||||||
use super::{RESERVED, SOCKS_VERSION};
|
use super::{SocksVersion, RESERVED, SOCKS4_VERSION, SOCKS5_VERSION};
|
||||||
use client_connections::{LaneQueueLengths, TransmissionLane};
|
use client_connections::{LaneQueueLengths, TransmissionLane};
|
||||||
use client_core::client::inbound_messages::{InputMessage, InputMessageSender};
|
use client_core::client::inbound_messages::{InputMessage, InputMessageSender};
|
||||||
use futures::channel::mpsc;
|
use futures::channel::mpsc;
|
||||||
@@ -126,16 +126,38 @@ impl AsyncWrite for StreamState {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Copy, Clone)]
|
||||||
|
pub(crate) struct Config {
|
||||||
|
use_surbs_for_responses: bool,
|
||||||
|
connection_start_surbs: u32,
|
||||||
|
per_request_surbs: u32,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Config {
|
||||||
|
pub(crate) fn new(
|
||||||
|
use_surbs_for_responses: bool,
|
||||||
|
connection_start_surbs: u32,
|
||||||
|
per_request_surbs: u32,
|
||||||
|
) -> Self {
|
||||||
|
Self {
|
||||||
|
use_surbs_for_responses,
|
||||||
|
connection_start_surbs,
|
||||||
|
per_request_surbs,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// A client connecting to the Socks proxy server, because
|
/// A client connecting to the Socks proxy server, because
|
||||||
/// it wants to make a Nym-protected outbound request. Typically, this is
|
/// it wants to make a Nym-protected outbound request. Typically, this is
|
||||||
/// something like e.g. a wallet app running on your laptop connecting to
|
/// something like e.g. a wallet app running on your laptop connecting to
|
||||||
/// SphinxSocksServer.
|
/// `SphinxSocksServer`.
|
||||||
pub(crate) struct SocksClient {
|
pub(crate) struct SocksClient {
|
||||||
|
config: Config,
|
||||||
controller_sender: ControllerSender,
|
controller_sender: ControllerSender,
|
||||||
stream: StreamState,
|
stream: StreamState,
|
||||||
auth_nmethods: u8,
|
auth_nmethods: u8,
|
||||||
authenticator: Authenticator,
|
authenticator: Authenticator,
|
||||||
socks_version: u8,
|
socks_version: Option<SocksVersion>,
|
||||||
input_sender: InputMessageSender,
|
input_sender: InputMessageSender,
|
||||||
connection_id: ConnectionId,
|
connection_id: ConnectionId,
|
||||||
service_provider: Recipient,
|
service_provider: Recipient,
|
||||||
@@ -158,29 +180,34 @@ impl Drop for SocksClient {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl SocksClient {
|
impl SocksClient {
|
||||||
/// Create a new SOCKClient
|
|
||||||
#[allow(clippy::too_many_arguments)]
|
#[allow(clippy::too_many_arguments)]
|
||||||
pub fn new(
|
pub fn new(
|
||||||
|
config: Config,
|
||||||
stream: TcpStream,
|
stream: TcpStream,
|
||||||
authenticator: Authenticator,
|
authenticator: Authenticator,
|
||||||
input_sender: InputMessageSender,
|
input_sender: InputMessageSender,
|
||||||
service_provider: Recipient,
|
service_provider: &Recipient,
|
||||||
controller_sender: ControllerSender,
|
controller_sender: ControllerSender,
|
||||||
self_address: Recipient,
|
self_address: &Recipient,
|
||||||
lane_queue_lengths: LaneQueueLengths,
|
lane_queue_lengths: LaneQueueLengths,
|
||||||
shutdown_listener: ShutdownListener,
|
mut shutdown_listener: ShutdownListener,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
|
// If this task fails and exits, we don't want to send shutdown signal
|
||||||
|
shutdown_listener.mark_as_success();
|
||||||
|
|
||||||
let connection_id = Self::generate_random();
|
let connection_id = Self::generate_random();
|
||||||
|
|
||||||
SocksClient {
|
SocksClient {
|
||||||
|
config,
|
||||||
controller_sender,
|
controller_sender,
|
||||||
connection_id,
|
connection_id,
|
||||||
stream: StreamState::Available(stream),
|
stream: StreamState::Available(stream),
|
||||||
auth_nmethods: 0,
|
auth_nmethods: 0,
|
||||||
socks_version: 0,
|
socks_version: None,
|
||||||
authenticator,
|
authenticator,
|
||||||
input_sender,
|
input_sender,
|
||||||
service_provider,
|
service_provider: *service_provider,
|
||||||
self_address,
|
self_address: *self_address,
|
||||||
started_proxy: false,
|
started_proxy: false,
|
||||||
lane_queue_lengths,
|
lane_queue_lengths,
|
||||||
shutdown_listener,
|
shutdown_listener,
|
||||||
@@ -192,16 +219,49 @@ impl SocksClient {
|
|||||||
rng.next_u64()
|
rng.next_u64()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub async fn send_error(&mut self, err: SocksProxyError) -> Result<(), SocksProxyError> {
|
||||||
|
let error_text = format!("{}", err);
|
||||||
|
let Some(ref version) = self.socks_version else {
|
||||||
|
log::error!("Trying to send error without knowing the version");
|
||||||
|
return Ok(());
|
||||||
|
};
|
||||||
|
|
||||||
|
match version {
|
||||||
|
SocksVersion::V4 => {
|
||||||
|
let response = ResponseCodeV4::RequestRejected;
|
||||||
|
self.send_error_v4(response).await
|
||||||
|
}
|
||||||
|
SocksVersion::V5 => {
|
||||||
|
let response = if error_text.contains("Host") {
|
||||||
|
ResponseCodeV5::HostUnreachable
|
||||||
|
} else if error_text.contains("Network") {
|
||||||
|
ResponseCodeV5::NetworkUnreachable
|
||||||
|
} else if error_text.contains("ttl") {
|
||||||
|
ResponseCodeV5::TtlExpired
|
||||||
|
} else {
|
||||||
|
ResponseCodeV5::Failure
|
||||||
|
};
|
||||||
|
self.send_error_v5(response).await
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Send an error back to the client
|
// Send an error back to the client
|
||||||
pub async fn error(&mut self, r: ResponseCode) -> Result<(), SocksProxyError> {
|
pub async fn send_error_v4(&mut self, r: ResponseCodeV4) -> Result<(), SocksProxyError> {
|
||||||
self.stream.write_all(&[5, r as u8]).await?;
|
self.stream.write_all(&[SOCKS4_VERSION, r as u8]).await?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Shutdown the TcpStream to the client and end the session
|
pub async fn send_error_v5(&mut self, r: ResponseCodeV5) -> Result<(), SocksProxyError> {
|
||||||
|
self.stream.write_all(&[SOCKS5_VERSION, r as u8]).await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Shutdown the `TcpStream` to the client and end the session
|
||||||
pub async fn shutdown(&mut self) -> Result<(), SocksProxyError> {
|
pub async fn shutdown(&mut self) -> Result<(), SocksProxyError> {
|
||||||
info!("client is shutting down its TCP stream");
|
info!("client is shutting down its TCP stream");
|
||||||
self.stream.shutdown().await?;
|
self.stream.shutdown().await?;
|
||||||
|
self.shutdown_listener.mark_as_success();
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -209,35 +269,37 @@ impl SocksClient {
|
|||||||
/// is in use and that the client is authenticated, then runs the request.
|
/// is in use and that the client is authenticated, then runs the request.
|
||||||
pub async fn run(&mut self) -> Result<(), SocksProxyError> {
|
pub async fn run(&mut self) -> Result<(), SocksProxyError> {
|
||||||
debug!("New connection from: {}", self.stream.peer_addr()?.ip());
|
debug!("New connection from: {}", self.stream.peer_addr()?.ip());
|
||||||
let mut header = [0u8; 2];
|
|
||||||
// Read a byte from the stream and determine the version being requested
|
// Read a byte from the stream and determine the version being requested
|
||||||
|
let mut header = [0u8];
|
||||||
self.stream.read_exact(&mut header).await?;
|
self.stream.read_exact(&mut header).await?;
|
||||||
|
|
||||||
self.socks_version = header[0];
|
self.socks_version = match SocksVersion::try_from(header[0]) {
|
||||||
self.auth_nmethods = header[1];
|
Ok(version) => Some(version),
|
||||||
|
Err(_err) => {
|
||||||
|
warn!("Init: Unsupported version: SOCKS{}", header[0]);
|
||||||
|
return self.shutdown().await;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
// Handle SOCKS4 requests
|
if self.socks_version == Some(SocksVersion::V5) {
|
||||||
if header[0] != SOCKS_VERSION {
|
let mut auth = [0u8];
|
||||||
warn!("Init: Unsupported version: SOCKS{}", self.socks_version);
|
self.stream.read_exact(&mut auth).await?;
|
||||||
self.shutdown().await
|
self.auth_nmethods = auth[0];
|
||||||
}
|
self.authenticate_socks5().await?;
|
||||||
// Valid SOCKS5
|
|
||||||
else {
|
|
||||||
// Authenticate w/ client
|
|
||||||
self.authenticate().await?;
|
|
||||||
// Handle requests
|
|
||||||
self.handle_request().await
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
self.handle_request().await
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn send_connect_to_mixnet(&mut self, remote_address: RemoteAddress) {
|
async fn send_anonymous_connect_to_mixnet(&mut self, remote_address: RemoteAddress) {
|
||||||
let req = Request::new_connect(self.connection_id, remote_address, self.self_address);
|
let req = Request::new_connect(self.connection_id, remote_address, None);
|
||||||
let msg = Message::Request(req);
|
let msg = Message::Request(req);
|
||||||
|
|
||||||
let input_message = InputMessage::new_fresh(
|
let input_message = InputMessage::new_anonymous(
|
||||||
self.service_provider,
|
self.service_provider,
|
||||||
msg.into_bytes(),
|
msg.into_bytes(),
|
||||||
false,
|
self.config.connection_start_surbs,
|
||||||
TransmissionLane::ConnectionId(self.connection_id),
|
TransmissionLane::ConnectionId(self.connection_id),
|
||||||
);
|
);
|
||||||
self.input_sender
|
self.input_sender
|
||||||
@@ -246,17 +308,48 @@ impl SocksClient {
|
|||||||
.expect("InputMessageReceiver has stopped receiving!");
|
.expect("InputMessageReceiver has stopped receiving!");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async fn send_connect_to_mixnet_with_return_address(&mut self, remote_address: RemoteAddress) {
|
||||||
|
let req = Request::new_connect(self.connection_id, remote_address, Some(self.self_address));
|
||||||
|
let msg = Message::Request(req);
|
||||||
|
|
||||||
|
let input_message = InputMessage::new_regular(
|
||||||
|
self.service_provider,
|
||||||
|
msg.into_bytes(),
|
||||||
|
TransmissionLane::ConnectionId(self.connection_id),
|
||||||
|
);
|
||||||
|
self.input_sender
|
||||||
|
.send(input_message)
|
||||||
|
.await
|
||||||
|
.expect("InputMessageReceiver has stopped receiving!");
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn send_connect_to_mixnet(&mut self, remote_address: RemoteAddress) {
|
||||||
|
if self.config.use_surbs_for_responses {
|
||||||
|
self.send_anonymous_connect_to_mixnet(remote_address).await
|
||||||
|
} else {
|
||||||
|
self.send_connect_to_mixnet_with_return_address(remote_address)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
async fn run_proxy(&mut self, conn_receiver: ConnectionReceiver, remote_proxy_target: String) {
|
async fn run_proxy(&mut self, conn_receiver: ConnectionReceiver, remote_proxy_target: String) {
|
||||||
self.send_connect_to_mixnet(remote_proxy_target.clone())
|
self.send_connect_to_mixnet(remote_proxy_target.clone())
|
||||||
.await;
|
.await;
|
||||||
|
|
||||||
let stream = self.stream.run_proxy();
|
let stream = self.stream.run_proxy();
|
||||||
let local_stream_remote = stream
|
let peer_addr = match stream.peer_addr() {
|
||||||
.peer_addr()
|
Ok(peer_addr) => peer_addr,
|
||||||
.expect("failed to extract peer address")
|
Err(err) => {
|
||||||
.to_string();
|
log::error!("Unable to extract the remote peer address: {err}");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
let local_stream_remote = peer_addr.to_string();
|
||||||
|
|
||||||
let connection_id = self.connection_id;
|
let connection_id = self.connection_id;
|
||||||
let input_sender = self.input_sender.clone();
|
let input_sender = self.input_sender.clone();
|
||||||
|
let anonymous = self.config.use_surbs_for_responses;
|
||||||
|
let per_request_surbs = self.config.per_request_surbs;
|
||||||
|
|
||||||
let recipient = self.service_provider;
|
let recipient = self.service_provider;
|
||||||
let (stream, _) = ProxyRunner::new(
|
let (stream, _) = ProxyRunner::new(
|
||||||
@@ -273,7 +366,16 @@ impl SocksClient {
|
|||||||
let provider_request = Request::new_send(conn_id, read_data, socket_closed);
|
let provider_request = Request::new_send(conn_id, read_data, socket_closed);
|
||||||
let provider_message = Message::Request(provider_request);
|
let provider_message = Message::Request(provider_request);
|
||||||
let lane = TransmissionLane::ConnectionId(conn_id);
|
let lane = TransmissionLane::ConnectionId(conn_id);
|
||||||
InputMessage::new_fresh(recipient, provider_message.into_bytes(), false, lane)
|
if anonymous {
|
||||||
|
InputMessage::new_anonymous(
|
||||||
|
recipient,
|
||||||
|
provider_message.into_bytes(),
|
||||||
|
per_request_surbs,
|
||||||
|
lane,
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
InputMessage::new_regular(recipient, provider_message.into_bytes(), lane)
|
||||||
|
}
|
||||||
})
|
})
|
||||||
.await
|
.await
|
||||||
.into_inner();
|
.into_inner();
|
||||||
@@ -285,8 +387,17 @@ impl SocksClient {
|
|||||||
async fn handle_request(&mut self) -> Result<(), SocksProxyError> {
|
async fn handle_request(&mut self) -> Result<(), SocksProxyError> {
|
||||||
debug!("Handling CONNECT Command");
|
debug!("Handling CONNECT Command");
|
||||||
|
|
||||||
let request = SocksRequest::from_stream(&mut self.stream).await?;
|
let version = self
|
||||||
let remote_address = request.to_string();
|
.socks_version
|
||||||
|
.as_ref()
|
||||||
|
.expect("Must read version before parsing request");
|
||||||
|
|
||||||
|
let request = match version {
|
||||||
|
SocksVersion::V4 => SocksRequest::from_stream_socks4(&mut self.stream).await?,
|
||||||
|
SocksVersion::V5 => SocksRequest::from_stream_socks5(&mut self.stream).await?,
|
||||||
|
};
|
||||||
|
|
||||||
|
let remote_address = request.address_string();
|
||||||
|
|
||||||
// setup for receiving from the mixnet
|
// setup for receiving from the mixnet
|
||||||
let (mix_sender, mix_receiver) = mpsc::unbounded();
|
let (mix_sender, mix_receiver) = mpsc::unbounded();
|
||||||
@@ -295,7 +406,10 @@ impl SocksClient {
|
|||||||
// Use the Proxy to connect to the specified addr/port
|
// Use the Proxy to connect to the specified addr/port
|
||||||
SocksCommand::Connect => {
|
SocksCommand::Connect => {
|
||||||
trace!("Connecting to: {:?}", remote_address.clone());
|
trace!("Connecting to: {:?}", remote_address.clone());
|
||||||
self.acknowledge_socks5().await;
|
match version {
|
||||||
|
SocksVersion::V4 => self.acknowledge_socks4().await,
|
||||||
|
SocksVersion::V5 => self.acknowledge_socks5().await,
|
||||||
|
}
|
||||||
|
|
||||||
self.started_proxy = true;
|
self.started_proxy = true;
|
||||||
self.controller_sender
|
self.controller_sender
|
||||||
@@ -326,8 +440,8 @@ impl SocksClient {
|
|||||||
async fn acknowledge_socks5(&mut self) {
|
async fn acknowledge_socks5(&mut self) {
|
||||||
self.stream
|
self.stream
|
||||||
.write_all(&[
|
.write_all(&[
|
||||||
SOCKS_VERSION,
|
SOCKS5_VERSION,
|
||||||
ResponseCode::Success as u8,
|
ResponseCodeV5::Success as u8,
|
||||||
RESERVED,
|
RESERVED,
|
||||||
1,
|
1,
|
||||||
127,
|
127,
|
||||||
@@ -341,13 +455,30 @@ impl SocksClient {
|
|||||||
.unwrap();
|
.unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Writes a Socks4 header back to the requesting client's TCP stream,
|
||||||
|
async fn acknowledge_socks4(&mut self) {
|
||||||
|
self.stream
|
||||||
|
.write_all(&[
|
||||||
|
0, //SOCKS4_VERSION,
|
||||||
|
ResponseCodeV4::Granted as u8,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
127,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
1,
|
||||||
|
])
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
/// Authenticate the incoming request. Each request is checked for its
|
/// Authenticate the incoming request. Each request is checked for its
|
||||||
/// authentication method. A user/password request will extract the
|
/// authentication method. A user/password request will extract the
|
||||||
/// username and password from the stream, then check with the Authenticator
|
/// username and password from the stream, then check with the Authenticator
|
||||||
/// to see if the resulting user is allowed.
|
/// to see if the resulting user is allowed.
|
||||||
///
|
///
|
||||||
/// A lot of this could probably be put into the `SocksRequest::from_stream()`
|
/// A lot of this could probably be put into the `SocksRequest::from_stream()`
|
||||||
/// constructor, and/or cleaned up with tokio::codec. It's mostly just
|
/// constructor, and/or cleaned up with `tokio::codec`. It's mostly just
|
||||||
/// read-a-byte-or-two. The bytes being extracted look like this:
|
/// read-a-byte-or-two. The bytes being extracted look like this:
|
||||||
///
|
///
|
||||||
/// +----+------+----------+------+------------+
|
/// +----+------+----------+------+------------+
|
||||||
@@ -359,7 +490,7 @@ impl SocksClient {
|
|||||||
/// Pulling out the stream code into its own home, and moving the if/else logic
|
/// Pulling out the stream code into its own home, and moving the if/else logic
|
||||||
/// into the Authenticator (where it'll be more easily testable)
|
/// into the Authenticator (where it'll be more easily testable)
|
||||||
/// would be a good next step.
|
/// would be a good next step.
|
||||||
async fn authenticate(&mut self) -> Result<(), SocksProxyError> {
|
async fn authenticate_socks5(&mut self) -> Result<(), SocksProxyError> {
|
||||||
debug!("Authenticating w/ {}", self.stream.peer_addr()?.ip());
|
debug!("Authenticating w/ {}", self.stream.peer_addr()?.ip());
|
||||||
// Get valid auth methods
|
// Get valid auth methods
|
||||||
let methods = self.get_available_methods().await?;
|
let methods = self.get_available_methods().await?;
|
||||||
@@ -368,7 +499,7 @@ impl SocksClient {
|
|||||||
let mut response = [0u8; 2];
|
let mut response = [0u8; 2];
|
||||||
|
|
||||||
// Set the version in the response
|
// Set the version in the response
|
||||||
response[0] = SOCKS_VERSION;
|
response[0] = SOCKS5_VERSION;
|
||||||
if methods.contains(&(AuthenticationMethods::UserPass as u8)) {
|
if methods.contains(&(AuthenticationMethods::UserPass as u8)) {
|
||||||
// Set the default auth method (NO AUTH)
|
// Set the default auth method (NO AUTH)
|
||||||
response[1] = AuthenticationMethods::UserPass as u8;
|
response[1] = AuthenticationMethods::UserPass as u8;
|
||||||
@@ -404,11 +535,11 @@ impl SocksClient {
|
|||||||
// Authenticate passwords
|
// Authenticate passwords
|
||||||
if self.authenticator.is_allowed(&user) {
|
if self.authenticator.is_allowed(&user) {
|
||||||
debug!("Access Granted. User: {}", user.username);
|
debug!("Access Granted. User: {}", user.username);
|
||||||
let response = [1, ResponseCode::Success as u8];
|
let response = [1, ResponseCodeV5::Success as u8];
|
||||||
self.stream.write_all(&response).await?;
|
self.stream.write_all(&response).await?;
|
||||||
} else {
|
} else {
|
||||||
debug!("Access Denied. User: {}", user.username);
|
debug!("Access Denied. User: {}", user.username);
|
||||||
let response = [1, ResponseCode::Failure as u8];
|
let response = [1, ResponseCodeV5::Failure as u8];
|
||||||
self.stream.write_all(&response).await?;
|
self.stream.write_all(&response).await?;
|
||||||
|
|
||||||
// Shutdown
|
// Shutdown
|
||||||
@@ -427,7 +558,7 @@ impl SocksClient {
|
|||||||
response[1] = AuthenticationMethods::NoMethods as u8;
|
response[1] = AuthenticationMethods::NoMethods as u8;
|
||||||
self.stream.write_all(&response).await?;
|
self.stream.write_all(&response).await?;
|
||||||
self.shutdown().await?;
|
self.shutdown().await?;
|
||||||
Err(ResponseCode::Failure.into())
|
Err(ResponseCodeV5::Failure.into())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
use std::time::Duration;
|
||||||
|
|
||||||
use futures::channel::mpsc;
|
use futures::channel::mpsc;
|
||||||
use futures::StreamExt;
|
use futures::StreamExt;
|
||||||
use log::*;
|
use log::*;
|
||||||
@@ -52,8 +54,8 @@ impl MixnetResponseListener {
|
|||||||
|
|
||||||
async fn on_message(&self, reconstructed_message: ReconstructedMessage) {
|
async fn on_message(&self, reconstructed_message: ReconstructedMessage) {
|
||||||
let raw_message = reconstructed_message.message;
|
let raw_message = reconstructed_message.message;
|
||||||
if reconstructed_message.reply_surb.is_some() {
|
if reconstructed_message.sender_tag.is_some() {
|
||||||
warn!("this message had a surb - we didn't do anything with it");
|
warn!("this message was sent anonymously - it couldn't have come from the service provider");
|
||||||
}
|
}
|
||||||
|
|
||||||
let response = match Message::try_from_bytes(&raw_message) {
|
let response = match Message::try_from_bytes(&raw_message) {
|
||||||
@@ -103,7 +105,10 @@ impl MixnetResponseListener {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
assert!(self.shutdown.is_shutdown_poll());
|
#[cfg(not(target_arch = "wasm32"))]
|
||||||
|
tokio::time::timeout(Duration::from_secs(5), self.shutdown.recv())
|
||||||
|
.await
|
||||||
|
.expect("Task stopped without shutdown called");
|
||||||
log::debug!("MixnetResponseListener: Exiting");
|
log::debug!("MixnetResponseListener: Exiting");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,11 @@
|
|||||||
#![forbid(unsafe_code)]
|
#![forbid(unsafe_code)]
|
||||||
|
|
||||||
|
use std::convert::TryFrom;
|
||||||
|
|
||||||
|
use self::types::SocksProxyError;
|
||||||
|
|
||||||
pub mod authentication;
|
pub mod authentication;
|
||||||
mod client;
|
pub(crate) mod client;
|
||||||
pub(crate) mod mixnet_responses;
|
pub(crate) mod mixnet_responses;
|
||||||
mod request;
|
mod request;
|
||||||
pub mod server;
|
pub mod server;
|
||||||
@@ -9,6 +13,27 @@ pub mod types;
|
|||||||
pub mod utils;
|
pub mod utils;
|
||||||
|
|
||||||
/// Version of socks
|
/// Version of socks
|
||||||
const SOCKS_VERSION: u8 = 0x05;
|
const SOCKS4_VERSION: u8 = 0x04;
|
||||||
|
const SOCKS5_VERSION: u8 = 0x05;
|
||||||
|
|
||||||
const RESERVED: u8 = 0x00;
|
const RESERVED: u8 = 0x00;
|
||||||
|
|
||||||
|
#[derive(Clone, PartialEq, Eq)]
|
||||||
|
pub enum SocksVersion {
|
||||||
|
V4 = 0x04,
|
||||||
|
V5 = 0x05,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct InvalidSocksVersion;
|
||||||
|
|
||||||
|
impl TryFrom<u8> for SocksVersion {
|
||||||
|
type Error = SocksProxyError;
|
||||||
|
|
||||||
|
fn try_from(version: u8) -> Result<Self, Self::Error> {
|
||||||
|
match version {
|
||||||
|
SOCKS4_VERSION => Ok(Self::V4),
|
||||||
|
SOCKS5_VERSION => Ok(Self::V5),
|
||||||
|
_ => Err(SocksProxyError::UnsupportedProxyVersion(version)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
use super::types::{AddrType, ResponseCode, SocksProxyError};
|
use crate::socks::SOCKS4_VERSION;
|
||||||
use super::{utils as socks_utils, SOCKS_VERSION};
|
|
||||||
|
use super::types::{AddrType, ResponseCodeV5, SocksProxyError};
|
||||||
|
use super::{utils as socks_utils, SOCKS5_VERSION};
|
||||||
use log::*;
|
use log::*;
|
||||||
use std::fmt::{self, Display};
|
use std::fmt::{self, Display};
|
||||||
use tokio::io::{AsyncRead, AsyncReadExt};
|
use tokio::io::{AsyncRead, AsyncReadExt};
|
||||||
@@ -15,80 +17,114 @@ pub(crate) struct SocksRequest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl SocksRequest {
|
impl SocksRequest {
|
||||||
/// Parse a SOCKS5 request from a TcpStream
|
/// Parse a SOCKS4 request from a `TcpStream`
|
||||||
pub async fn from_stream<R>(stream: &mut R) -> Result<Self, SocksProxyError>
|
/// From documents at:
|
||||||
|
/// - SOCKS4: https://www.openssh.com/txt/socks4.protocol
|
||||||
|
/// - SOCKS4a: https://www.openssh.com/txt/socks4a.protocol
|
||||||
|
pub async fn from_stream_socks4<R>(stream: &mut R) -> Result<Self, SocksProxyError>
|
||||||
where
|
where
|
||||||
R: AsyncRead + Unpin,
|
R: AsyncRead + Unpin,
|
||||||
{
|
{
|
||||||
|
log::trace!("read from stream socks4");
|
||||||
|
|
||||||
|
let mut packet = [0u8; 3];
|
||||||
|
stream.read_exact(&mut packet).await?;
|
||||||
|
|
||||||
|
// CD (command)
|
||||||
|
let Some(command) = SocksCommand::from(packet[0] as usize) else {
|
||||||
|
log::warn!("Invalid Command");
|
||||||
|
return Err(ResponseCodeV5::CommandNotSupported.into());
|
||||||
|
};
|
||||||
|
|
||||||
|
// DSTPORT
|
||||||
|
let mut port = [0u8; 2];
|
||||||
|
port.copy_from_slice(&packet[1..]);
|
||||||
|
let port = merge_u8_into_u16(port[0], port[1]);
|
||||||
|
|
||||||
|
// DSTIP
|
||||||
|
let mut ip = [0u8; 4];
|
||||||
|
stream.read_exact(&mut ip).await?;
|
||||||
|
|
||||||
|
// USERID
|
||||||
|
let _userid = read_until_zero(stream).await;
|
||||||
|
|
||||||
|
// SOCKS4a extension
|
||||||
|
// https://www.openssh.com/txt/socks4a.protocol
|
||||||
|
// If the IP is 0.0.0.x with x nonzero, read the domain name
|
||||||
|
let (addr, addr_type) = if ip[..3] == [0, 0, 0] && ip[3] != 0 {
|
||||||
|
(read_until_zero(stream).await?, AddrType::Domain)
|
||||||
|
} else {
|
||||||
|
(ip.to_vec(), AddrType::V4)
|
||||||
|
};
|
||||||
|
|
||||||
|
// Return parsed request
|
||||||
|
Ok(SocksRequest {
|
||||||
|
version: SOCKS4_VERSION,
|
||||||
|
command,
|
||||||
|
addr_type,
|
||||||
|
addr,
|
||||||
|
port,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
/// Parse a SOCKS5 request from a `TcpStream`
|
||||||
|
/// From: https://www.rfc-editor.org/rfc/rfc1928
|
||||||
|
pub async fn from_stream_socks5<R>(stream: &mut R) -> Result<Self, SocksProxyError>
|
||||||
|
where
|
||||||
|
R: AsyncRead + Unpin,
|
||||||
|
{
|
||||||
|
log::info!("read from stream socks5");
|
||||||
|
|
||||||
let mut packet = [0u8; 4];
|
let mut packet = [0u8; 4];
|
||||||
// Read a byte from the stream and determine the version being requested
|
// Read a byte from the stream and determine the version being requested
|
||||||
stream.read_exact(&mut packet).await?;
|
stream.read_exact(&mut packet).await?;
|
||||||
|
|
||||||
if packet[0] != SOCKS_VERSION {
|
// VER
|
||||||
warn!("from_stream Unsupported version: SOCKS{}", packet[0]);
|
if packet[0] != SOCKS5_VERSION {
|
||||||
|
warn!("Unsupported version: SOCKS{}", packet[0]);
|
||||||
return Err(SocksProxyError::UnsupportedProxyVersion(packet[0]));
|
return Err(SocksProxyError::UnsupportedProxyVersion(packet[0]));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get command
|
// CMD
|
||||||
let mut command: SocksCommand = SocksCommand::Connect;
|
let Some(command) = SocksCommand::from(packet[1] as usize) else {
|
||||||
match SocksCommand::from(packet[1] as usize) {
|
warn!("Invalid Command");
|
||||||
Some(com) => {
|
return Err(ResponseCodeV5::CommandNotSupported.into());
|
||||||
command = com;
|
};
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
None => {
|
|
||||||
warn!("Invalid Command");
|
|
||||||
Err(ResponseCode::CommandNotSupported)
|
|
||||||
}
|
|
||||||
}?;
|
|
||||||
|
|
||||||
// DST.address
|
// RSV
|
||||||
|
// packet[2] is reserved
|
||||||
|
|
||||||
let mut addr_type: AddrType = AddrType::V6;
|
// ATYP
|
||||||
match AddrType::from(packet[3] as usize) {
|
let Some(addr_type) = AddrType::from(packet[3] as usize) else {
|
||||||
Some(addr) => {
|
error!("No Addr");
|
||||||
addr_type = addr;
|
return Err(ResponseCodeV5::AddrTypeNotSupported.into())
|
||||||
Ok(())
|
};
|
||||||
}
|
|
||||||
None => {
|
|
||||||
error!("No Addr");
|
|
||||||
Err(ResponseCode::AddrTypeNotSupported)
|
|
||||||
}
|
|
||||||
}?;
|
|
||||||
|
|
||||||
trace!("Getting Addr");
|
// DST.ADDR
|
||||||
// Get Addr from addr_type and stream
|
let addr = match addr_type {
|
||||||
let addr: Result<Vec<u8>, SocksProxyError> = match addr_type {
|
|
||||||
AddrType::Domain => {
|
AddrType::Domain => {
|
||||||
let mut domain_length = [0u8; 1];
|
let mut domain_length = [0u8];
|
||||||
stream.read_exact(&mut domain_length).await?;
|
stream.read_exact(&mut domain_length).await?;
|
||||||
|
|
||||||
let mut domain = vec![0u8; domain_length[0] as usize];
|
let mut domain = vec![0u8; domain_length[0] as usize];
|
||||||
stream.read_exact(&mut domain).await?;
|
stream.read_exact(&mut domain).await?;
|
||||||
|
domain
|
||||||
Ok(domain)
|
|
||||||
}
|
}
|
||||||
AddrType::V4 => {
|
AddrType::V4 => {
|
||||||
let mut addr = [0u8; 4];
|
let mut addr = [0u8; 4];
|
||||||
stream.read_exact(&mut addr).await?;
|
stream.read_exact(&mut addr).await?;
|
||||||
Ok(addr.to_vec())
|
addr.to_vec()
|
||||||
}
|
}
|
||||||
AddrType::V6 => {
|
AddrType::V6 => {
|
||||||
let mut addr = [0u8; 16];
|
let mut addr = [0u8; 16];
|
||||||
stream.read_exact(&mut addr).await?;
|
stream.read_exact(&mut addr).await?;
|
||||||
Ok(addr.to_vec())
|
addr.to_vec()
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
let addr = addr?;
|
// DST.PORT
|
||||||
|
|
||||||
// read DST.port
|
|
||||||
let mut port = [0u8; 2];
|
let mut port = [0u8; 2];
|
||||||
stream.read_exact(&mut port).await?;
|
stream.read_exact(&mut port).await?;
|
||||||
// Merge two u8s into u16
|
let port = merge_u8_into_u16(port[0], port[1]);
|
||||||
let port = (u16::from(port[0]) << 8) | u16::from(port[1]);
|
|
||||||
|
|
||||||
// Return parsed request
|
|
||||||
Ok(SocksRequest {
|
Ok(SocksRequest {
|
||||||
version: packet[0],
|
version: packet[0],
|
||||||
command,
|
command,
|
||||||
@@ -97,14 +133,18 @@ impl SocksRequest {
|
|||||||
port,
|
port,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Print out the address and port to a String.
|
||||||
|
/// This might return domain:port, ipv6:port, or ipv4:port.
|
||||||
|
pub fn address_string(&self) -> String {
|
||||||
|
let address = socks_utils::pretty_print_addr(&self.addr_type, &self.addr);
|
||||||
|
format!("{}:{}", address, self.port)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Display for SocksRequest {
|
impl Display for SocksRequest {
|
||||||
/// Print out the address and port to a String.
|
|
||||||
/// This might return domain:port, ipv6:port, or ipv4:port.
|
|
||||||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
let address = socks_utils::pretty_print_addr(&self.addr_type, &self.addr);
|
write!(f, "{}", self.address_string())
|
||||||
write!(f, "{}:{}", address, self.port)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -127,3 +167,23 @@ impl SocksCommand {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn merge_u8_into_u16(a: u8, b: u8) -> u16 {
|
||||||
|
(u16::from(a) << 8) | u16::from(b)
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn read_until_zero<R>(stream: &mut R) -> Result<Vec<u8>, SocksProxyError>
|
||||||
|
where
|
||||||
|
R: AsyncRead + Unpin,
|
||||||
|
{
|
||||||
|
let mut result = Vec::new();
|
||||||
|
let mut char = [0u8];
|
||||||
|
loop {
|
||||||
|
stream.read_exact(&mut char).await?;
|
||||||
|
if char[0] == 0 {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
result.push(char[0]);
|
||||||
|
}
|
||||||
|
Ok(result)
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
use super::authentication::Authenticator;
|
use crate::error::Socks5ClientError;
|
||||||
use super::client::SocksClient;
|
|
||||||
use super::{
|
use super::{
|
||||||
mixnet_responses::MixnetResponseListener,
|
authentication::Authenticator, client::SocksClient, mixnet_responses::MixnetResponseListener,
|
||||||
types::{ResponseCode, SocksProxyError},
|
|
||||||
};
|
};
|
||||||
|
use crate::socks::client;
|
||||||
use client_connections::{ConnectionCommandSender, LaneQueueLengths};
|
use client_connections::{ConnectionCommandSender, LaneQueueLengths};
|
||||||
use client_core::client::{
|
use client_core::client::{
|
||||||
inbound_messages::InputMessageSender, received_buffer::ReceivedBufferRequestSender,
|
inbound_messages::InputMessageSender, received_buffer::ReceivedBufferRequestSender,
|
||||||
@@ -12,6 +12,7 @@ use log::*;
|
|||||||
use nymsphinx::addressing::clients::Recipient;
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
use proxy_helpers::connection_controller::{BroadcastActiveConnections, Controller};
|
use proxy_helpers::connection_controller::{BroadcastActiveConnections, Controller};
|
||||||
use std::net::SocketAddr;
|
use std::net::SocketAddr;
|
||||||
|
use tap::TapFallible;
|
||||||
use task::ShutdownListener;
|
use task::ShutdownListener;
|
||||||
use tokio::net::TcpListener;
|
use tokio::net::TcpListener;
|
||||||
|
|
||||||
@@ -21,6 +22,7 @@ pub struct SphinxSocksServer {
|
|||||||
listening_address: SocketAddr,
|
listening_address: SocketAddr,
|
||||||
service_provider: Recipient,
|
service_provider: Recipient,
|
||||||
self_address: Recipient,
|
self_address: Recipient,
|
||||||
|
client_config: client::Config,
|
||||||
lane_queue_lengths: LaneQueueLengths,
|
lane_queue_lengths: LaneQueueLengths,
|
||||||
shutdown: ShutdownListener,
|
shutdown: ShutdownListener,
|
||||||
}
|
}
|
||||||
@@ -33,6 +35,7 @@ impl SphinxSocksServer {
|
|||||||
service_provider: Recipient,
|
service_provider: Recipient,
|
||||||
self_address: Recipient,
|
self_address: Recipient,
|
||||||
lane_queue_lengths: LaneQueueLengths,
|
lane_queue_lengths: LaneQueueLengths,
|
||||||
|
client_config: client::Config,
|
||||||
shutdown: ShutdownListener,
|
shutdown: ShutdownListener,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
// hardcode ip as we (presumably) ONLY want to listen locally. If we change it, we can
|
// hardcode ip as we (presumably) ONLY want to listen locally. If we change it, we can
|
||||||
@@ -44,6 +47,7 @@ impl SphinxSocksServer {
|
|||||||
listening_address: format!("{}:{}", ip, port).parse().unwrap(),
|
listening_address: format!("{}:{}", ip, port).parse().unwrap(),
|
||||||
service_provider,
|
service_provider,
|
||||||
self_address,
|
self_address,
|
||||||
|
client_config,
|
||||||
lane_queue_lengths,
|
lane_queue_lengths,
|
||||||
shutdown,
|
shutdown,
|
||||||
}
|
}
|
||||||
@@ -56,8 +60,10 @@ impl SphinxSocksServer {
|
|||||||
input_sender: InputMessageSender,
|
input_sender: InputMessageSender,
|
||||||
buffer_requester: ReceivedBufferRequestSender,
|
buffer_requester: ReceivedBufferRequestSender,
|
||||||
client_connection_tx: ConnectionCommandSender,
|
client_connection_tx: ConnectionCommandSender,
|
||||||
) -> Result<(), SocksProxyError> {
|
) -> Result<(), Socks5ClientError> {
|
||||||
let listener = TcpListener::bind(self.listening_address).await.unwrap();
|
let listener = TcpListener::bind(self.listening_address)
|
||||||
|
.await
|
||||||
|
.tap_err(|err| log::error!("Failed to bind to address: {err}"))?;
|
||||||
info!("Serving Connections...");
|
info!("Serving Connections...");
|
||||||
|
|
||||||
// controller for managing all active connections
|
// controller for managing all active connections
|
||||||
@@ -83,47 +89,27 @@ impl SphinxSocksServer {
|
|||||||
loop {
|
loop {
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
Ok((stream, _remote)) = listener.accept() => {
|
Ok((stream, _remote)) = listener.accept() => {
|
||||||
// TODO Optimize this
|
|
||||||
let mut client = SocksClient::new(
|
let mut client = SocksClient::new(
|
||||||
|
self.client_config,
|
||||||
stream,
|
stream,
|
||||||
self.authenticator.clone(),
|
self.authenticator.clone(),
|
||||||
input_sender.clone(),
|
input_sender.clone(),
|
||||||
self.service_provider,
|
&self.service_provider,
|
||||||
controller_sender.clone(),
|
controller_sender.clone(),
|
||||||
self.self_address,
|
&self.self_address,
|
||||||
self.lane_queue_lengths.clone(),
|
self.lane_queue_lengths.clone(),
|
||||||
self.shutdown.clone(),
|
self.shutdown.clone(),
|
||||||
);
|
);
|
||||||
|
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
{
|
if let Err(err) = client.run().await {
|
||||||
match client.run().await {
|
error!("Error! {}", err);
|
||||||
Ok(_) => {}
|
if client.send_error(err).await.is_err() {
|
||||||
Err(error) => {
|
warn!("Failed to send error code");
|
||||||
error!("Error! {}", error);
|
};
|
||||||
let error_text = format!("{}", error);
|
if client.shutdown().await.is_err() {
|
||||||
|
warn!("Failed to shutdown TcpStream");
|
||||||
let response: ResponseCode;
|
|
||||||
|
|
||||||
if error_text.contains("Host") {
|
|
||||||
response = ResponseCode::HostUnreachable;
|
|
||||||
} else if error_text.contains("Network") {
|
|
||||||
response = ResponseCode::NetworkUnreachable;
|
|
||||||
} else if error_text.contains("ttl") {
|
|
||||||
response = ResponseCode::TtlExpired
|
|
||||||
} else {
|
|
||||||
response = ResponseCode::Failure
|
|
||||||
}
|
|
||||||
|
|
||||||
if client.error(response).await.is_err() {
|
|
||||||
warn!("Failed to send error code");
|
|
||||||
};
|
|
||||||
if client.shutdown().await.is_err() {
|
|
||||||
warn!("Failed to shutdown TcpStream");
|
|
||||||
};
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
// client gets dropped here
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,7 +1,17 @@
|
|||||||
use snafu::Snafu;
|
use snafu::Snafu;
|
||||||
#[derive(Debug, Snafu)]
|
|
||||||
|
/// SOCKS4 Response codes
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub(crate) enum ResponseCodeV4 {
|
||||||
|
Granted = 0x5a,
|
||||||
|
RequestRejected = 0x5b,
|
||||||
|
CannotConnectToIdent = 0x5c,
|
||||||
|
DifferentUserId = 0x5d,
|
||||||
|
}
|
||||||
|
|
||||||
/// Possible SOCKS5 Response Codes
|
/// Possible SOCKS5 Response Codes
|
||||||
pub(crate) enum ResponseCode {
|
#[derive(Debug, Snafu)]
|
||||||
|
pub(crate) enum ResponseCodeV5 {
|
||||||
Success = 0x00,
|
Success = 0x00,
|
||||||
#[snafu(display("SOCKS5 Server Failure"))]
|
#[snafu(display("SOCKS5 Server Failure"))]
|
||||||
Failure = 0x01,
|
Failure = 0x01,
|
||||||
@@ -48,7 +58,7 @@ where
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// DST.addr variant types
|
/// DST.addr variant types
|
||||||
#[derive(PartialEq)]
|
#[derive(Debug, PartialEq)]
|
||||||
pub(crate) enum AddrType {
|
pub(crate) enum AddrType {
|
||||||
V4 = 0x01,
|
V4 = 0x01,
|
||||||
Domain = 0x03,
|
Domain = 0x03,
|
||||||
|
|||||||
@@ -39,7 +39,7 @@ topology = { path = "../../common/topology" }
|
|||||||
gateway-client = { path = "../../common/client-libs/gateway-client", default-features = false, features = ["wasm", "coconut"] }
|
gateway-client = { path = "../../common/client-libs/gateway-client", default-features = false, features = ["wasm", "coconut"] }
|
||||||
validator-client = { path = "../../common/client-libs/validator-client", default-features = false }
|
validator-client = { path = "../../common/client-libs/validator-client", default-features = false }
|
||||||
wasm-utils = { path = "../../common/wasm-utils" }
|
wasm-utils = { path = "../../common/wasm-utils" }
|
||||||
|
task = { path = "../../common/task" }
|
||||||
# The `console_error_panic_hook` crate provides better debugging of panics by
|
# The `console_error_panic_hook` crate provides better debugging of panics by
|
||||||
# logging them with `console.error`. This is great for development, but requires
|
# logging them with `console.error`. This is great for development, but requires
|
||||||
# all the `std::fmt` and `std::panicking` infrastructure, so isn't great for
|
# all the `std::fmt` and `std::panicking` infrastructure, so isn't great for
|
||||||
|
|||||||
@@ -65,7 +65,6 @@ async function main() {
|
|||||||
*
|
*
|
||||||
* Message and recipient are taken from the values in the user interface.
|
* Message and recipient are taken from the values in the user interface.
|
||||||
*
|
*
|
||||||
* @param {Client} nymClient the nym client to use for message sending
|
|
||||||
*/
|
*/
|
||||||
async function sendMessageTo() {
|
async function sendMessageTo() {
|
||||||
const message = document.getElementById('message').value;
|
const message = document.getElementById('message').value;
|
||||||
@@ -96,10 +95,13 @@ function displaySend(message) {
|
|||||||
/**
|
/**
|
||||||
* Display received text messages in the browser. Colour them green.
|
* Display received text messages in the browser. Colour them green.
|
||||||
*
|
*
|
||||||
* @param {string} message
|
* @param {Uint8Array} raw
|
||||||
*/
|
*/
|
||||||
function displayReceived(message) {
|
function displayReceived(raw, sender_tag) {
|
||||||
const content = message;
|
const content = new TextDecoder().decode(raw);
|
||||||
|
if (sender_tag !== undefined) {
|
||||||
|
console.log("this message also contained some surbs from", sender_tag)
|
||||||
|
}
|
||||||
|
|
||||||
let timestamp = new Date().toISOString().substr(11, 12);
|
let timestamp = new Date().toISOString().substr(11, 12);
|
||||||
let receivedDiv = document.createElement('div');
|
let receivedDiv = document.createElement('div');
|
||||||
@@ -116,7 +118,7 @@ function displayReceived(message) {
|
|||||||
/**
|
/**
|
||||||
* Display the nymClient's sender address in the user interface
|
* Display the nymClient's sender address in the user interface
|
||||||
*
|
*
|
||||||
* @param {Client} nymClient
|
* @param {String} address
|
||||||
*/
|
*/
|
||||||
function displaySenderAddress(address) {
|
function displaySenderAddress(address) {
|
||||||
document.getElementById('sender').value = address;
|
document.getElementById('sender').value = address;
|
||||||
|
|||||||
+1
-1
@@ -24,7 +24,7 @@
|
|||||||
},
|
},
|
||||||
"../pkg": {
|
"../pkg": {
|
||||||
"name": "@nymproject/nym-client-wasm",
|
"name": "@nymproject/nym-client-wasm",
|
||||||
"version": "1.0.0",
|
"version": "1.1.0",
|
||||||
"license": "Apache-2.0"
|
"license": "Apache-2.0"
|
||||||
},
|
},
|
||||||
"node_modules/@discoveryjs/json-ext": {
|
"node_modules/@discoveryjs/json-ext": {
|
||||||
|
|||||||
@@ -17,38 +17,7 @@ importScripts('nym_client_wasm.js');
|
|||||||
console.log('Initializing worker');
|
console.log('Initializing worker');
|
||||||
|
|
||||||
// wasm_bindgen creates a global variable (with the exports attached) that is in scope after `importScripts`
|
// wasm_bindgen creates a global variable (with the exports attached) that is in scope after `importScripts`
|
||||||
const { default_debug, get_gateway, NymClient, set_panic_hook, Config } = wasm_bindgen;
|
const { default_debug, NymClientBuilder, set_panic_hook, Config, GatewayEndpointConfig } = wasm_bindgen;
|
||||||
|
|
||||||
class ClientWrapper {
|
|
||||||
constructor(config, onMessageHandler) {
|
|
||||||
this.rustClient = new NymClient(config);
|
|
||||||
this.rustClient.set_on_message(onMessageHandler);
|
|
||||||
this.rustClient.set_on_gateway_connect(this.onConnect);
|
|
||||||
}
|
|
||||||
|
|
||||||
selfAddress = () => {
|
|
||||||
return this.rustClient.self_address();
|
|
||||||
};
|
|
||||||
|
|
||||||
onConnect = () => {
|
|
||||||
console.log('Established (and authenticated) gateway connection!');
|
|
||||||
};
|
|
||||||
|
|
||||||
start = async () => {
|
|
||||||
// this is current limitation of wasm in rust - for async methods you can't take self by reference...
|
|
||||||
// I'm trying to figure out if I can somehow hack my way around it, but for time being you have to re-assign
|
|
||||||
// the object (it's the same one)
|
|
||||||
this.rustClient = await this.rustClient.start();
|
|
||||||
};
|
|
||||||
|
|
||||||
sendMessage = async (recipient, message) => {
|
|
||||||
this.rustClient = await this.rustClient.send_message(recipient, message);
|
|
||||||
};
|
|
||||||
|
|
||||||
sendBinaryMessage = async (recipient, message) => {
|
|
||||||
this.rustClient = await this.rustClient.send_binary_message(recipient, message);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
let client = null;
|
let client = null;
|
||||||
|
|
||||||
@@ -61,19 +30,13 @@ async function main() {
|
|||||||
// sets up better stack traces in case of in-rust panics
|
// sets up better stack traces in case of in-rust panics
|
||||||
set_panic_hook();
|
set_panic_hook();
|
||||||
|
|
||||||
console.error("the current mainnet is not compatible with v2! - either use the pre-merge branch or explicitly set the client to use one of V2 QA networks")
|
|
||||||
return
|
|
||||||
|
|
||||||
// validator server we will use to get topology from
|
// validator server we will use to get topology from
|
||||||
// MAINNET (V1):
|
const validator = 'https://qwerty-validator-api.qa.nymte.ch/api';
|
||||||
const validator = 'https://validator.nymtech.net/api'; //"http://localhost:8081";
|
|
||||||
const preferredGateway = 'E3mvZTHQCdBvhfr178Swx9g4QG3kkRUun7YnToLMcMbM';
|
|
||||||
// QA (V2):
|
|
||||||
// const validator = 'https://qa-validator-api.nymtech.net/api'; //"http://localhost:8081";
|
|
||||||
// const preferredGateway = 'CgQrYP8etksSBf4nALNqp93SHPpgFwEUyTsjBNNLj5WM';
|
|
||||||
|
|
||||||
const gatewayEndpoint = await get_gateway(validator, preferredGateway);
|
const gatewayId = 'EVupP2tRUeZo5Y6RpBHAbm8kSntpgNyZNL6yCr7BDEoG';
|
||||||
gatewayEndpoint.gateway_listener = "wss://gateway1.nymtech.net:443"; // this is needed if we want it to work on the web. However this gateway is a v1 gateway, we will need to change for v2 once we get there
|
const gatewayOwner = 'n1rmlew3euapuq7rs4s4j9apv00whrsazr764kl7';
|
||||||
|
const gatewayListener = 'ws://176.58.120.72:9000';
|
||||||
|
const gatewayEndpoint = new GatewayEndpointConfig(gatewayId, gatewayOwner, gatewayListener)
|
||||||
|
|
||||||
// only really useful if you want to adjust some settings like traffic rate
|
// only really useful if you want to adjust some settings like traffic rate
|
||||||
// (if not needed you can just pass a null)
|
// (if not needed you can just pass a null)
|
||||||
@@ -101,12 +64,17 @@ async function main() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
console.log('Instantiating WASM client...');
|
console.log('Instantiating WASM client...');
|
||||||
client = new ClientWrapper(config, onMessageHandler);
|
|
||||||
|
let clientBuilder = new NymClientBuilder(config, onMessageHandler)
|
||||||
console.log('Web worker creating WASM client...');
|
console.log('Web worker creating WASM client...');
|
||||||
await client.start();
|
let local_client = await clientBuilder.start_client();
|
||||||
console.log('WASM client running!');
|
console.log('WASM client running!');
|
||||||
|
|
||||||
const selfAddress = client.rustClient.self_address();
|
const selfAddress = local_client.self_address();
|
||||||
|
|
||||||
|
// set the global (I guess we don't have to anymore?)
|
||||||
|
client = local_client;
|
||||||
|
|
||||||
console.log(`Client address is ${selfAddress}`);
|
console.log(`Client address is ${selfAddress}`);
|
||||||
self.postMessage({
|
self.postMessage({
|
||||||
kind: 'Ready',
|
kind: 'Ready',
|
||||||
@@ -121,7 +89,8 @@ async function main() {
|
|||||||
switch (event.data.kind) {
|
switch (event.data.kind) {
|
||||||
case 'SendMessage': {
|
case 'SendMessage': {
|
||||||
const { message, recipient } = event.data.args;
|
const { message, recipient } = event.data.args;
|
||||||
await client.sendMessage(message, recipient);
|
let uint8Array = new TextEncoder().encode(message);
|
||||||
|
await client.send_regular_message(uint8Array, recipient);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,12 +4,15 @@
|
|||||||
// due to expansion of #[wasm_bindgen] macro on `Debug` Config struct
|
// due to expansion of #[wasm_bindgen] macro on `Debug` Config struct
|
||||||
#![allow(clippy::drop_non_drop)]
|
#![allow(clippy::drop_non_drop)]
|
||||||
|
|
||||||
use client_core::config::{Debug as ConfigDebug, ExtendedPacketSize, GatewayEndpoint};
|
use client_core::config::{DebugConfig as ConfigDebug, ExtendedPacketSize, GatewayEndpointConfig};
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
use url::Url;
|
use url::Url;
|
||||||
use wasm_bindgen::prelude::*;
|
use wasm_bindgen::prelude::*;
|
||||||
|
|
||||||
#[wasm_bindgen]
|
#[wasm_bindgen]
|
||||||
|
#[derive(Debug, Deserialize, PartialEq, Serialize)]
|
||||||
|
#[serde(deny_unknown_fields)]
|
||||||
pub struct Config {
|
pub struct Config {
|
||||||
/// ID specifies the human readable ID of this particular client.
|
/// ID specifies the human readable ID of this particular client.
|
||||||
pub(crate) id: String,
|
pub(crate) id: String,
|
||||||
@@ -19,7 +22,7 @@ pub struct Config {
|
|||||||
pub(crate) disabled_credentials_mode: bool,
|
pub(crate) disabled_credentials_mode: bool,
|
||||||
|
|
||||||
/// Information regarding how the client should send data to gateway.
|
/// Information regarding how the client should send data to gateway.
|
||||||
pub(crate) gateway_endpoint: GatewayEndpoint,
|
pub(crate) gateway_endpoint: GatewayEndpointConfig,
|
||||||
|
|
||||||
pub(crate) debug: ConfigDebug,
|
pub(crate) debug: ConfigDebug,
|
||||||
}
|
}
|
||||||
@@ -30,7 +33,7 @@ impl Config {
|
|||||||
pub fn new(
|
pub fn new(
|
||||||
id: String,
|
id: String,
|
||||||
validator_server: String,
|
validator_server: String,
|
||||||
gateway_endpoint: GatewayEndpoint,
|
gateway_endpoint: GatewayEndpointConfig,
|
||||||
debug: Option<Debug>,
|
debug: Option<Debug>,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
Config {
|
Config {
|
||||||
@@ -103,6 +106,34 @@ pub struct Debug {
|
|||||||
|
|
||||||
/// Controls whether the sent sphinx packet use the NON-DEFAULT bigger size.
|
/// Controls whether the sent sphinx packet use the NON-DEFAULT bigger size.
|
||||||
pub use_extended_packet_size: bool,
|
pub use_extended_packet_size: bool,
|
||||||
|
|
||||||
|
/// Defines the minimum number of reply surbs the client wants to keep in its storage at all times.
|
||||||
|
/// It can only allow to go below that value if its to request additional reply surbs.
|
||||||
|
pub minimum_reply_surb_storage_threshold: usize,
|
||||||
|
|
||||||
|
/// Defines the maximum number of reply surbs the client wants to keep in its storage at any times.
|
||||||
|
pub maximum_reply_surb_storage_threshold: usize,
|
||||||
|
|
||||||
|
/// Defines the minimum number of reply surbs the client would request.
|
||||||
|
pub minimum_reply_surb_request_size: u32,
|
||||||
|
|
||||||
|
/// Defines the maximum number of reply surbs the client would request.
|
||||||
|
pub maximum_reply_surb_request_size: u32,
|
||||||
|
|
||||||
|
/// Defines the maximum number of reply surbs a remote party is allowed to request from this client at once.
|
||||||
|
pub maximum_allowed_reply_surb_request_size: u32,
|
||||||
|
|
||||||
|
/// Defines maximum amount of time the client is going to wait for reply surbs before explicitly asking
|
||||||
|
/// for more even though in theory they wouldn't need to.
|
||||||
|
pub maximum_reply_surb_waiting_period_ms: u64,
|
||||||
|
|
||||||
|
/// Defines maximum amount of time given reply surb is going to be valid for.
|
||||||
|
/// This is going to be superseded by key rotation once implemented.
|
||||||
|
pub maximum_reply_surb_age_ms: u64,
|
||||||
|
|
||||||
|
/// Defines maximum amount of time given reply key is going to be valid for.
|
||||||
|
/// This is going to be superseded by key rotation once implemented.
|
||||||
|
pub maximum_reply_key_age_ms: u64,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<Debug> for ConfigDebug {
|
impl From<Debug> for ConfigDebug {
|
||||||
@@ -132,6 +163,16 @@ impl From<Debug> for ConfigDebug {
|
|||||||
disable_main_poisson_packet_distribution: debug
|
disable_main_poisson_packet_distribution: debug
|
||||||
.disable_main_poisson_packet_distribution,
|
.disable_main_poisson_packet_distribution,
|
||||||
use_extended_packet_size,
|
use_extended_packet_size,
|
||||||
|
minimum_reply_surb_storage_threshold: debug.minimum_reply_surb_storage_threshold,
|
||||||
|
maximum_reply_surb_storage_threshold: debug.maximum_reply_surb_storage_threshold,
|
||||||
|
minimum_reply_surb_request_size: debug.minimum_reply_surb_request_size,
|
||||||
|
maximum_reply_surb_request_size: debug.maximum_reply_surb_request_size,
|
||||||
|
maximum_allowed_reply_surb_request_size: debug.maximum_allowed_reply_surb_request_size,
|
||||||
|
maximum_reply_surb_waiting_period: Duration::from_millis(
|
||||||
|
debug.maximum_reply_surb_waiting_period_ms,
|
||||||
|
),
|
||||||
|
maximum_reply_surb_age: Duration::from_millis(debug.maximum_reply_surb_age_ms),
|
||||||
|
maximum_reply_key_age: Duration::from_millis(debug.maximum_reply_key_age_ms),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -154,6 +195,16 @@ impl From<ConfigDebug> for Debug {
|
|||||||
disable_main_poisson_packet_distribution: debug
|
disable_main_poisson_packet_distribution: debug
|
||||||
.disable_main_poisson_packet_distribution,
|
.disable_main_poisson_packet_distribution,
|
||||||
use_extended_packet_size: debug.use_extended_packet_size.is_some(),
|
use_extended_packet_size: debug.use_extended_packet_size.is_some(),
|
||||||
|
minimum_reply_surb_storage_threshold: debug.minimum_reply_surb_storage_threshold,
|
||||||
|
maximum_reply_surb_storage_threshold: debug.maximum_reply_surb_storage_threshold,
|
||||||
|
minimum_reply_surb_request_size: debug.minimum_reply_surb_request_size,
|
||||||
|
maximum_reply_surb_request_size: debug.maximum_reply_surb_request_size,
|
||||||
|
maximum_allowed_reply_surb_request_size: debug.maximum_allowed_reply_surb_request_size,
|
||||||
|
maximum_reply_surb_waiting_period_ms: debug
|
||||||
|
.maximum_reply_surb_waiting_period
|
||||||
|
.as_millis() as u64,
|
||||||
|
maximum_reply_surb_age_ms: debug.maximum_reply_surb_age.as_millis() as u64,
|
||||||
|
maximum_reply_key_age_ms: debug.maximum_reply_key_age.as_millis() as u64,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,30 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use client_core::client::base_client::ClientInput;
|
||||||
|
use client_core::client::inbound_messages::InputMessage;
|
||||||
|
use js_sys::Promise;
|
||||||
|
use std::sync::Arc;
|
||||||
|
use wasm_bindgen::JsValue;
|
||||||
|
use wasm_bindgen_futures::future_to_promise;
|
||||||
|
|
||||||
|
// defining helper trait as we could directly call the method on the wrapper
|
||||||
|
pub(crate) trait InputSender {
|
||||||
|
fn send_message(&self, message: InputMessage) -> Promise;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl InputSender for Arc<ClientInput> {
|
||||||
|
fn send_message(&self, message: InputMessage) -> Promise {
|
||||||
|
let this = Arc::clone(self);
|
||||||
|
future_to_promise(async move {
|
||||||
|
match this.input_sender.send(message).await {
|
||||||
|
Ok(_) => Ok(JsValue::null()),
|
||||||
|
Err(_) => {
|
||||||
|
let js_error =
|
||||||
|
js_sys::Error::new("InputMessageReceiver has stopped receiving!");
|
||||||
|
Err(JsValue::from(js_error))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -2,67 +2,71 @@
|
|||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use self::config::Config;
|
use self::config::Config;
|
||||||
use client_connections::{ConnectionCommandReceiver, LaneQueueLengths, TransmissionLane};
|
use crate::client::helpers::InputSender;
|
||||||
use client_core::client::{
|
use crate::client::response_pusher::ResponsePusher;
|
||||||
cover_traffic_stream::LoopCoverTrafficStream,
|
use client_connections::TransmissionLane;
|
||||||
inbound_messages::{InputMessage, InputMessageReceiver, InputMessageSender},
|
use client_core::client::base_client::{BaseClientBuilder, ClientInput, ClientOutput};
|
||||||
key_manager::KeyManager,
|
use client_core::client::replies::reply_storage::browser_backend;
|
||||||
mix_traffic::{BatchMixMessageSender, MixTrafficController},
|
use client_core::client::{inbound_messages::InputMessage, key_manager::KeyManager};
|
||||||
real_messages_control::{self, RealMessagesController},
|
use gateway_client::bandwidth::BandwidthController;
|
||||||
received_buffer::{
|
use js_sys::Promise;
|
||||||
ReceivedBufferMessage, ReceivedBufferRequestReceiver, ReceivedBufferRequestSender,
|
|
||||||
ReceivedMessagesBufferController,
|
|
||||||
},
|
|
||||||
topology_control::{TopologyAccessor, TopologyRefresher, TopologyRefresherConfig},
|
|
||||||
};
|
|
||||||
use crypto::asymmetric::identity;
|
|
||||||
use futures::channel::mpsc;
|
|
||||||
use futures::StreamExt;
|
|
||||||
use gateway_client::{
|
|
||||||
AcknowledgementReceiver, AcknowledgementSender, GatewayClient, MixnetMessageReceiver,
|
|
||||||
MixnetMessageSender,
|
|
||||||
};
|
|
||||||
use nymsphinx::addressing::clients::Recipient;
|
use nymsphinx::addressing::clients::Recipient;
|
||||||
|
use nymsphinx::anonymous_replies::requests::AnonymousSenderTag;
|
||||||
use rand::rngs::OsRng;
|
use rand::rngs::OsRng;
|
||||||
|
use std::sync::Arc;
|
||||||
|
use task::ShutdownNotifier;
|
||||||
use wasm_bindgen::prelude::*;
|
use wasm_bindgen::prelude::*;
|
||||||
use wasm_bindgen_futures::spawn_local;
|
use wasm_bindgen_futures::future_to_promise;
|
||||||
use wasm_utils::console_log;
|
use wasm_utils::{console_error, console_log};
|
||||||
|
|
||||||
pub mod config;
|
pub mod config;
|
||||||
|
mod helpers;
|
||||||
|
mod response_pusher;
|
||||||
|
|
||||||
#[wasm_bindgen]
|
#[wasm_bindgen]
|
||||||
pub struct NymClient {
|
pub struct NymClient {
|
||||||
|
self_address: String,
|
||||||
|
client_input: Arc<ClientInput>,
|
||||||
|
|
||||||
|
// even though we don't use graceful shutdowns, other components rely on existence of this struct
|
||||||
|
// and if it's dropped, everything will start going offline
|
||||||
|
_shutdown: ShutdownNotifier,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
pub struct NymClientBuilder {
|
||||||
config: Config,
|
config: Config,
|
||||||
|
|
||||||
/// KeyManager object containing smart pointers to all relevant keys used by the client.
|
/// KeyManager object containing smart pointers to all relevant keys used by the client.
|
||||||
key_manager: KeyManager,
|
key_manager: KeyManager,
|
||||||
|
|
||||||
// TODO: this should be stored somewhere persistently
|
reply_surb_storage_backend: browser_backend::Backend,
|
||||||
// received_keys: HashSet<SURBEncryptionKey>,
|
|
||||||
/// Channel used for transforming 'raw' messages into sphinx packets and sending them
|
|
||||||
/// through the mix network.
|
|
||||||
input_tx: Option<InputMessageSender>,
|
|
||||||
|
|
||||||
// callbacks
|
on_message: js_sys::Function,
|
||||||
on_message: Option<js_sys::Function>,
|
|
||||||
on_binary_message: Option<js_sys::Function>,
|
// unimplemented:
|
||||||
on_gateway_connect: Option<js_sys::Function>,
|
bandwidth_controller: Option<BandwidthController>,
|
||||||
|
disabled_credentials: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[wasm_bindgen]
|
#[wasm_bindgen]
|
||||||
impl NymClient {
|
impl NymClientBuilder {
|
||||||
#[wasm_bindgen(constructor)]
|
#[wasm_bindgen(constructor)]
|
||||||
pub fn new(config: Config) -> Self {
|
pub fn new(config: Config, on_message: js_sys::Function) -> Self {
|
||||||
Self {
|
//, key_manager: Option<KeyManager>) {
|
||||||
|
NymClientBuilder {
|
||||||
|
reply_surb_storage_backend: Self::setup_reply_surb_storage_backend(&config),
|
||||||
config,
|
config,
|
||||||
key_manager: Self::setup_key_manager(),
|
key_manager: Self::setup_key_manager(),
|
||||||
on_message: None,
|
on_message,
|
||||||
on_binary_message: None,
|
bandwidth_controller: None,
|
||||||
on_gateway_connect: None,
|
disabled_credentials: true,
|
||||||
input_tx: None,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: once we make keys persistent, we'll require some kind of `init` method to generate
|
||||||
|
// a prior shared keypair between the client and the gateway
|
||||||
|
|
||||||
// perhaps this should be public?
|
// perhaps this should be public?
|
||||||
fn setup_key_manager() -> KeyManager {
|
fn setup_key_manager() -> KeyManager {
|
||||||
let mut rng = OsRng;
|
let mut rng = OsRng;
|
||||||
@@ -71,337 +75,154 @@ impl NymClient {
|
|||||||
KeyManager::new(&mut rng)
|
KeyManager::new(&mut rng)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_on_message(&mut self, on_message: js_sys::Function) {
|
// don't get too excited about the name, under the hood it's just a big fat placeholder
|
||||||
self.on_message = Some(on_message);
|
// with no persistence
|
||||||
}
|
fn setup_reply_surb_storage_backend(config: &Config) -> browser_backend::Backend {
|
||||||
|
browser_backend::Backend::new(
|
||||||
pub fn set_on_binary_message(&mut self, on_binary_message: js_sys::Function) {
|
config.debug.minimum_reply_surb_storage_threshold,
|
||||||
self.on_binary_message = Some(on_binary_message);
|
config.debug.maximum_reply_surb_storage_threshold,
|
||||||
}
|
|
||||||
|
|
||||||
pub fn set_on_gateway_connect(&mut self, on_connect: js_sys::Function) {
|
|
||||||
self.on_gateway_connect = Some(on_connect)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn as_mix_recipient(&self) -> Recipient {
|
|
||||||
Recipient::new(
|
|
||||||
*self.key_manager.identity_keypair().public_key(),
|
|
||||||
*self.key_manager.encryption_keypair().public_key(),
|
|
||||||
identity::PublicKey::from_base58_string(&self.config.gateway_endpoint.gateway_id)
|
|
||||||
.expect("no gateway has been selected"),
|
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn self_address(&self) -> String {
|
fn start_reconstructed_pusher(client_output: ClientOutput, on_message: js_sys::Function) {
|
||||||
self.as_mix_recipient().to_string()
|
ResponsePusher::new(client_output, on_message).start()
|
||||||
}
|
}
|
||||||
|
|
||||||
// future constantly pumping loop cover traffic at some specified average rate
|
pub async fn start_client(self) -> Promise {
|
||||||
// the pumped traffic goes to the MixTrafficController
|
future_to_promise(async move {
|
||||||
fn start_cover_traffic_stream(
|
console_log!("Starting the wasm client");
|
||||||
&self,
|
|
||||||
topology_accessor: TopologyAccessor,
|
|
||||||
mix_tx: BatchMixMessageSender,
|
|
||||||
) {
|
|
||||||
console_log!("Starting loop cover traffic stream...");
|
|
||||||
|
|
||||||
let mut stream = LoopCoverTrafficStream::new(
|
let base_builder = BaseClientBuilder::new(
|
||||||
self.key_manager.ack_key(),
|
&self.config.gateway_endpoint,
|
||||||
self.config.debug.average_ack_delay,
|
&self.config.debug,
|
||||||
self.config.debug.average_packet_delay,
|
self.key_manager,
|
||||||
self.config.debug.loop_cover_traffic_average_delay,
|
self.bandwidth_controller,
|
||||||
mix_tx,
|
self.reply_surb_storage_backend,
|
||||||
self.as_mix_recipient(),
|
self.disabled_credentials,
|
||||||
topology_accessor,
|
vec![self.config.validator_api_url.clone()],
|
||||||
);
|
|
||||||
|
|
||||||
if let Some(size) = &self.config.debug.use_extended_packet_size {
|
|
||||||
stream.set_custom_packet_size(size.clone().into());
|
|
||||||
}
|
|
||||||
|
|
||||||
stream.start();
|
|
||||||
}
|
|
||||||
|
|
||||||
fn start_real_traffic_controller(
|
|
||||||
&self,
|
|
||||||
topology_accessor: TopologyAccessor,
|
|
||||||
ack_receiver: AcknowledgementReceiver,
|
|
||||||
input_receiver: InputMessageReceiver,
|
|
||||||
mix_sender: BatchMixMessageSender,
|
|
||||||
client_connection_rx: ConnectionCommandReceiver,
|
|
||||||
lane_queue_lengths: LaneQueueLengths,
|
|
||||||
) {
|
|
||||||
let mut controller_config = real_messages_control::Config::new(
|
|
||||||
self.key_manager.ack_key(),
|
|
||||||
self.config.debug.ack_wait_multiplier,
|
|
||||||
self.config.debug.ack_wait_addition,
|
|
||||||
self.config.debug.average_ack_delay,
|
|
||||||
self.config.debug.message_sending_average_delay,
|
|
||||||
self.config.debug.average_packet_delay,
|
|
||||||
self.config.debug.disable_main_poisson_packet_distribution,
|
|
||||||
self.as_mix_recipient(),
|
|
||||||
);
|
|
||||||
|
|
||||||
if let Some(size) = &self.config.debug.use_extended_packet_size {
|
|
||||||
controller_config.set_custom_packet_size(size.clone().into());
|
|
||||||
}
|
|
||||||
|
|
||||||
console_log!("Starting real traffic stream...");
|
|
||||||
|
|
||||||
RealMessagesController::new(
|
|
||||||
controller_config,
|
|
||||||
ack_receiver,
|
|
||||||
input_receiver,
|
|
||||||
mix_sender,
|
|
||||||
topology_accessor,
|
|
||||||
lane_queue_lengths,
|
|
||||||
client_connection_rx,
|
|
||||||
)
|
|
||||||
.start();
|
|
||||||
}
|
|
||||||
|
|
||||||
// buffer controlling all messages fetched from provider
|
|
||||||
// required so that other components would be able to use them (say the websocket)
|
|
||||||
fn start_received_messages_buffer_controller(
|
|
||||||
&self,
|
|
||||||
query_receiver: ReceivedBufferRequestReceiver,
|
|
||||||
mixnet_receiver: MixnetMessageReceiver,
|
|
||||||
) {
|
|
||||||
console_log!("Starting received messages buffer controller...");
|
|
||||||
ReceivedMessagesBufferController::new(
|
|
||||||
self.key_manager.encryption_keypair(),
|
|
||||||
query_receiver,
|
|
||||||
mixnet_receiver,
|
|
||||||
)
|
|
||||||
.start()
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn start_gateway_client(
|
|
||||||
&mut self,
|
|
||||||
mixnet_message_sender: MixnetMessageSender,
|
|
||||||
ack_sender: AcknowledgementSender,
|
|
||||||
) -> GatewayClient {
|
|
||||||
let gateway_id = self.config.gateway_endpoint.gateway_id.clone();
|
|
||||||
if gateway_id.is_empty() {
|
|
||||||
panic!("The identity of the gateway is unknown - did you run `get_gateway()`?")
|
|
||||||
}
|
|
||||||
let gateway_owner = self.config.gateway_endpoint.gateway_owner.clone();
|
|
||||||
if gateway_owner.is_empty() {
|
|
||||||
panic!("The owner of the gateway is unknown - did you run `get_gateway()`?")
|
|
||||||
}
|
|
||||||
let gateway_address = self.config.gateway_endpoint.gateway_listener.clone();
|
|
||||||
if gateway_address.is_empty() {
|
|
||||||
panic!("The address of the gateway is unknown - did you run `get_gateway()`?")
|
|
||||||
}
|
|
||||||
|
|
||||||
let gateway_identity = identity::PublicKey::from_base58_string(gateway_id)
|
|
||||||
.expect("provided gateway id is invalid!");
|
|
||||||
|
|
||||||
let mut gateway_client = GatewayClient::new(
|
|
||||||
gateway_address,
|
|
||||||
self.key_manager.identity_keypair(),
|
|
||||||
gateway_identity,
|
|
||||||
gateway_owner,
|
|
||||||
None,
|
|
||||||
mixnet_message_sender,
|
|
||||||
ack_sender,
|
|
||||||
self.config.debug.gateway_response_timeout,
|
|
||||||
None,
|
|
||||||
);
|
|
||||||
|
|
||||||
gateway_client.set_disabled_credentials_mode(self.config.disabled_credentials_mode);
|
|
||||||
|
|
||||||
let shared_keys = gateway_client
|
|
||||||
.authenticate_and_start()
|
|
||||||
.await
|
|
||||||
.expect("could not authenticate and start up the gateway connection");
|
|
||||||
self.key_manager.insert_gateway_shared_key(shared_keys);
|
|
||||||
|
|
||||||
match self.on_gateway_connect.as_ref() {
|
|
||||||
Some(callback) => {
|
|
||||||
callback
|
|
||||||
.call0(&JsValue::null())
|
|
||||||
.expect("on connect callback failed!");
|
|
||||||
}
|
|
||||||
None => console_log!("Gateway connection established - no callback specified"),
|
|
||||||
};
|
|
||||||
|
|
||||||
gateway_client
|
|
||||||
}
|
|
||||||
|
|
||||||
// future responsible for periodically polling directory server and updating
|
|
||||||
// the current global view of topology
|
|
||||||
async fn start_topology_refresher(&mut self, topology_accessor: TopologyAccessor) {
|
|
||||||
let topology_refresher_config = TopologyRefresherConfig::new(
|
|
||||||
vec![self.config.validator_api_url.clone()],
|
|
||||||
self.config.debug.topology_refresh_rate,
|
|
||||||
env!("CARGO_PKG_VERSION").to_string(),
|
|
||||||
);
|
|
||||||
let mut topology_refresher =
|
|
||||||
TopologyRefresher::new(topology_refresher_config, topology_accessor);
|
|
||||||
// before returning, block entire runtime to refresh the current network view so that any
|
|
||||||
// components depending on topology would see a non-empty view
|
|
||||||
console_log!("Obtaining initial network topology");
|
|
||||||
topology_refresher.refresh().await;
|
|
||||||
|
|
||||||
// TODO: a slightly more graceful termination here
|
|
||||||
if !topology_refresher.is_topology_routable().await {
|
|
||||||
panic!(
|
|
||||||
"The current network topology seem to be insufficient to route any packets through\
|
|
||||||
- check if enough nodes and a gateway are online"
|
|
||||||
);
|
);
|
||||||
}
|
|
||||||
|
|
||||||
console_log!("Starting topology refresher...");
|
let self_address = base_builder.as_mix_recipient().to_string();
|
||||||
|
let mut started_client = match base_builder.start_base().await {
|
||||||
// TODO: re-enable
|
Ok(base_client) => base_client,
|
||||||
topology_refresher.start();
|
Err(err) => {
|
||||||
}
|
let error_msg = format!("failed to start the base client components - {err}");
|
||||||
|
console_error!("{}", error_msg);
|
||||||
// controller for sending sphinx packets to mixnet (either real traffic or cover traffic)
|
let js_error = js_sys::Error::new(&error_msg);
|
||||||
// TODO: if we want to send control messages to gateway_client, this CAN'T take the ownership
|
return Err(JsValue::from(js_error));
|
||||||
// over it. Perhaps GatewayClient needs to be thread-shareable or have some channel for
|
|
||||||
// requests?
|
|
||||||
fn start_mix_traffic_controller(gateway_client: GatewayClient) -> BatchMixMessageSender {
|
|
||||||
console_log!("Starting mix traffic controller...");
|
|
||||||
let (mix_traffic_controller, mix_tx) = MixTrafficController::new(gateway_client);
|
|
||||||
mix_traffic_controller.start();
|
|
||||||
mix_tx
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: this procedure is extremely overcomplicated, because it's based off native client's behaviour
|
|
||||||
// which doesn't fully apply in this case
|
|
||||||
fn start_reconstructed_pusher(
|
|
||||||
&mut self,
|
|
||||||
received_buffer_request_sender: ReceivedBufferRequestSender,
|
|
||||||
) {
|
|
||||||
let on_message = self.on_message.take();
|
|
||||||
let on_binary_message = self.on_binary_message.take();
|
|
||||||
|
|
||||||
spawn_local(async move {
|
|
||||||
let (reconstructed_sender, mut reconstructed_receiver) = mpsc::unbounded();
|
|
||||||
|
|
||||||
// tell the buffer to start sending stuff to us
|
|
||||||
received_buffer_request_sender
|
|
||||||
.unbounded_send(ReceivedBufferMessage::ReceiverAnnounce(
|
|
||||||
reconstructed_sender,
|
|
||||||
))
|
|
||||||
.expect("the buffer request failed!");
|
|
||||||
|
|
||||||
let this = JsValue::null();
|
|
||||||
|
|
||||||
while let Some(reconstructed) = reconstructed_receiver.next().await {
|
|
||||||
for msg in reconstructed {
|
|
||||||
if let Some(ref callback_binary) = on_binary_message {
|
|
||||||
let arg1 = serde_wasm_bindgen::to_value(&msg.message).unwrap();
|
|
||||||
callback_binary
|
|
||||||
.call1(&this, &arg1)
|
|
||||||
.expect("on binary message failed!");
|
|
||||||
}
|
|
||||||
if let Some(ref callback) = on_message {
|
|
||||||
if msg.reply_surb.is_some() {
|
|
||||||
console_log!("the received message contained a reply-surb that we do not know how to handle (yet)")
|
|
||||||
}
|
|
||||||
let stringified = String::from_utf8_lossy(&msg.message).into_owned();
|
|
||||||
let arg1 = serde_wasm_bindgen::to_value(&stringified).unwrap();
|
|
||||||
callback.call1(&this, &arg1).expect("on message failed!");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn start(mut self) -> NymClient {
|
let client_input = started_client.client_input.register_producer();
|
||||||
console_log!("Starting wasm client '{}'", self.config.id);
|
let client_output = started_client.client_output.register_consumer();
|
||||||
// channels for inter-component communication
|
|
||||||
// TODO: make the channels be internally created by the relevant components
|
|
||||||
// rather than creating them here, so say for example the buffer controller would create the request channels
|
|
||||||
// and would allow anyone to clone the sender channel
|
|
||||||
|
|
||||||
// unwrapped_sphinx_sender is the transmitter of mixnet messages received from the gateway
|
Self::start_reconstructed_pusher(client_output, self.on_message);
|
||||||
// unwrapped_sphinx_receiver is the receiver for said messages - used by ReceivedMessagesBuffer
|
|
||||||
let (mixnet_messages_sender, mixnet_messages_receiver) = mpsc::unbounded();
|
|
||||||
|
|
||||||
// used for announcing connection or disconnection of a channel for pushing re-assembled messages to
|
Ok(JsValue::from(NymClient {
|
||||||
let (received_buffer_request_sender, received_buffer_request_receiver) = mpsc::unbounded();
|
self_address,
|
||||||
|
client_input: Arc::new(client_input),
|
||||||
// channels responsible for controlling real messages
|
_shutdown: started_client.shutdown_notifier,
|
||||||
let (input_sender, input_receiver) = tokio::sync::mpsc::channel::<InputMessage>(1);
|
}))
|
||||||
|
})
|
||||||
// channels responsible for controlling ack messages
|
}
|
||||||
let (ack_sender, ack_receiver) = mpsc::unbounded();
|
}
|
||||||
let shared_topology_accessor = TopologyAccessor::new();
|
|
||||||
|
#[wasm_bindgen]
|
||||||
// Channel that the real traffix controller can listed to for closing connections.
|
impl NymClient {
|
||||||
// Currently unused in the wasm client.
|
pub fn self_address(&self) -> String {
|
||||||
let (_client_connection_tx, client_connection_rx) = mpsc::unbounded();
|
self.self_address.clone()
|
||||||
|
}
|
||||||
// the components are started in very specific order. Unless you know what you are doing,
|
|
||||||
// do not change that.
|
fn parse_recipient(recipient: &str) -> Result<Recipient, JsValue> {
|
||||||
self.start_topology_refresher(shared_topology_accessor.clone())
|
match Recipient::try_from_base58_string(recipient) {
|
||||||
.await;
|
Ok(recipient) => Ok(recipient),
|
||||||
self.start_received_messages_buffer_controller(
|
Err(err) => {
|
||||||
received_buffer_request_receiver,
|
let error_msg = format!("{recipient} is not a valid Nym network recipient - {err}");
|
||||||
mixnet_messages_receiver,
|
console_error!("{}", error_msg);
|
||||||
);
|
let js_error = js_sys::Error::new(&error_msg);
|
||||||
|
Err(JsValue::from(js_error))
|
||||||
let gateway_client = self
|
}
|
||||||
.start_gateway_client(mixnet_messages_sender, ack_sender)
|
}
|
||||||
.await;
|
}
|
||||||
|
|
||||||
// The sphinx_message_sender is the transmitter for any component generating sphinx packets
|
fn parse_sender_tag(tag: &str) -> Result<AnonymousSenderTag, JsValue> {
|
||||||
// that are to be sent to the mixnet. They are used by cover traffic stream and real
|
match AnonymousSenderTag::try_from_base58_string(tag) {
|
||||||
// traffic stream.
|
Ok(tag) => Ok(tag),
|
||||||
// The MixTrafficController then sends the actual traffic
|
Err(err) => {
|
||||||
let sphinx_message_sender = Self::start_mix_traffic_controller(gateway_client);
|
let error_msg = format!("{tag} is not a valid Nym AnonymousSenderTag - {err}");
|
||||||
|
console_error!("{}", error_msg);
|
||||||
// Shared queue length data. Published by the `OutQueueController` in the client, and used
|
let js_error = js_sys::Error::new(&error_msg);
|
||||||
// primarily to throttle incoming connections
|
Err(JsValue::from(js_error))
|
||||||
let shared_lane_queue_lengths = LaneQueueLengths::new();
|
}
|
||||||
|
}
|
||||||
self.start_real_traffic_controller(
|
}
|
||||||
shared_topology_accessor.clone(),
|
|
||||||
ack_receiver,
|
/// The simplest message variant where no additional information is attached.
|
||||||
input_receiver,
|
/// You're simply sending your `data` to specified `recipient` without any tagging.
|
||||||
sphinx_message_sender.clone(),
|
///
|
||||||
client_connection_rx,
|
/// Ends up with `NymMessage::Plain` variant
|
||||||
shared_lane_queue_lengths,
|
pub fn send_regular_message(&self, message: Vec<u8>, recipient: String) -> Promise {
|
||||||
);
|
console_log!(
|
||||||
|
"Attempting to send {:.2} kiB message to {recipient}",
|
||||||
if !self.config.debug.disable_loop_cover_traffic_stream {
|
message.len() as f64 / 1024.0
|
||||||
self.start_cover_traffic_stream(shared_topology_accessor, sphinx_message_sender);
|
);
|
||||||
}
|
|
||||||
|
let recipient = match Self::parse_recipient(&recipient) {
|
||||||
self.start_reconstructed_pusher(received_buffer_request_sender);
|
Ok(recipient) => recipient,
|
||||||
self.input_tx = Some(input_sender);
|
Err(err) => return Promise::reject(&err),
|
||||||
|
};
|
||||||
self
|
let lane = TransmissionLane::General;
|
||||||
}
|
|
||||||
|
let input_msg = InputMessage::new_regular(recipient, message, lane);
|
||||||
// Right now it's impossible to have async exported functions to take `&mut self` rather than mut self
|
self.client_input.send_message(input_msg)
|
||||||
// TODO: try Rc<RefCell<Self>> approach?
|
}
|
||||||
pub async fn send_message(self, message: String, recipient: String) -> Self {
|
|
||||||
console_log!("Sending {} to {}", message, recipient);
|
/// Creates a message used for a duplex anonymous communication where the recipient
|
||||||
|
/// will never learn of our true identity. This is achieved by carefully sending `reply_surbs`.
|
||||||
let message_bytes = message.into_bytes();
|
///
|
||||||
self.send_binary_message(message_bytes, recipient).await
|
/// Note that if reply_surbs is set to zero then
|
||||||
}
|
/// this variant requires the client having sent some reply_surbs in the past
|
||||||
|
/// (and thus the recipient also knowing our sender tag).
|
||||||
pub async fn send_binary_message(self, message: Vec<u8>, recipient: String) -> Self {
|
///
|
||||||
console_log!("Sending {} bytes to {}", message.len(), recipient);
|
/// Ends up with `NymMessage::Repliable` variant
|
||||||
|
pub fn send_anonymous_message(
|
||||||
let recipient = Recipient::try_from_base58_string(recipient).unwrap();
|
&self,
|
||||||
let lane = TransmissionLane::General;
|
message: Vec<u8>,
|
||||||
|
recipient: String,
|
||||||
let input_msg = InputMessage::new_fresh(recipient, message, false, lane);
|
reply_surbs: u32,
|
||||||
|
) -> Promise {
|
||||||
self.input_tx
|
console_log!(
|
||||||
.as_ref()
|
"Attempting to anonymously send {:.2} kiB message to {recipient} while attaching {reply_surbs} replySURBs.",
|
||||||
.expect("start method was not called before!")
|
message.len() as f64 / 1024.0
|
||||||
.send(input_msg)
|
);
|
||||||
.await
|
|
||||||
.expect("InputMessageReceiver has stopped receiving!");
|
let recipient = match Self::parse_recipient(&recipient) {
|
||||||
|
Ok(recipient) => recipient,
|
||||||
self
|
Err(err) => return Promise::reject(&err),
|
||||||
|
};
|
||||||
|
let lane = TransmissionLane::General;
|
||||||
|
|
||||||
|
let input_msg = InputMessage::new_anonymous(recipient, message, reply_surbs, lane);
|
||||||
|
self.client_input.send_message(input_msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Attempt to use our internally received and stored `ReplySurb` to send the message back
|
||||||
|
/// to specified recipient whilst not knowing its full identity (or even gateway).
|
||||||
|
///
|
||||||
|
/// Ends up with `NymMessage::Reply` variant
|
||||||
|
pub fn send_reply(&self, message: Vec<u8>, recipient_tag: String) -> Promise {
|
||||||
|
console_log!(
|
||||||
|
"Attempting to send {:.2} kiB reply message to {recipient_tag}",
|
||||||
|
message.len() as f64 / 1024.0
|
||||||
|
);
|
||||||
|
|
||||||
|
let sender_tag = match Self::parse_sender_tag(&recipient_tag) {
|
||||||
|
Ok(recipient) => recipient,
|
||||||
|
Err(err) => return Promise::reject(&err),
|
||||||
|
};
|
||||||
|
let lane = TransmissionLane::General;
|
||||||
|
|
||||||
|
let input_msg = InputMessage::new_reply(sender_tag, message, lane);
|
||||||
|
self.client_input.send_message(input_msg)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,58 @@
|
|||||||
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
use client_core::client::base_client::ClientOutput;
|
||||||
|
use client_core::client::received_buffer::{ReceivedBufferMessage, ReconstructedMessagesReceiver};
|
||||||
|
use futures::channel::mpsc;
|
||||||
|
use futures::StreamExt;
|
||||||
|
use js_sys::Uint8Array;
|
||||||
|
use wasm_bindgen::JsValue;
|
||||||
|
use wasm_bindgen_futures::spawn_local;
|
||||||
|
use wasm_utils::console_error;
|
||||||
|
|
||||||
|
pub(crate) struct ResponsePusher {
|
||||||
|
reconstructed_receiver: ReconstructedMessagesReceiver,
|
||||||
|
on_message: js_sys::Function,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ResponsePusher {
|
||||||
|
pub(crate) fn new(client_output: ClientOutput, on_message: js_sys::Function) -> Self {
|
||||||
|
// register our output
|
||||||
|
let (reconstructed_sender, reconstructed_receiver) = mpsc::unbounded();
|
||||||
|
|
||||||
|
// tell the buffer to start sending stuff to us
|
||||||
|
client_output
|
||||||
|
.received_buffer_request_sender
|
||||||
|
.unbounded_send(ReceivedBufferMessage::ReceiverAnnounce(
|
||||||
|
reconstructed_sender,
|
||||||
|
))
|
||||||
|
.expect("the buffer request failed!");
|
||||||
|
|
||||||
|
ResponsePusher {
|
||||||
|
reconstructed_receiver,
|
||||||
|
on_message,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn start(mut self) {
|
||||||
|
spawn_local(async move {
|
||||||
|
let this = JsValue::null();
|
||||||
|
|
||||||
|
while let Some(reconstructed) = self.reconstructed_receiver.next().await {
|
||||||
|
for reconstructed_msg in reconstructed {
|
||||||
|
let (msg, tag) = reconstructed_msg.into_inner();
|
||||||
|
|
||||||
|
let msg_slice: &[u8] = &msg;
|
||||||
|
let array = Uint8Array::from(msg_slice);
|
||||||
|
let arg1 = JsValue::from(array);
|
||||||
|
let arg2 = JsValue::from(tag);
|
||||||
|
self.on_message
|
||||||
|
.call2(&this, &arg1, &arg2)
|
||||||
|
.expect("on binary message failed!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
console_error!("we stopped receiving reconstructed messages!")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,15 +1,15 @@
|
|||||||
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
// Copyright 2022 - Nym Technologies SA <contact@nymtech.net>
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use client_core::config::GatewayEndpoint;
|
use client_core::config::GatewayEndpointConfig;
|
||||||
use wasm_bindgen::prelude::*;
|
use wasm_bindgen::prelude::*;
|
||||||
|
|
||||||
#[wasm_bindgen]
|
#[wasm_bindgen]
|
||||||
pub async fn get_gateway(api_server: String, preferred: Option<String>) -> GatewayEndpoint {
|
pub async fn get_gateway(api_server: String, preferred: Option<String>) -> GatewayEndpointConfig {
|
||||||
let validator_client = validator_client::client::ApiClient::new(api_server.parse().unwrap());
|
let validator_client = validator_client::client::ApiClient::new(api_server.parse().unwrap());
|
||||||
|
|
||||||
let gateways = match validator_client.get_cached_gateways().await {
|
let gateways = match validator_client.get_cached_gateways().await {
|
||||||
Err(err) => panic!("failed to obtain list of all gateways - {}", err),
|
Err(err) => panic!("failed to obtain list of all gateways - {err}"),
|
||||||
Ok(gateways) => gateways,
|
Ok(gateways) => gateways,
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -18,7 +18,7 @@ pub async fn get_gateway(api_server: String, preferred: Option<String>) -> Gatew
|
|||||||
.iter()
|
.iter()
|
||||||
.find(|g| g.gateway.identity_key == preferred)
|
.find(|g| g.gateway.identity_key == preferred)
|
||||||
{
|
{
|
||||||
return GatewayEndpoint {
|
return GatewayEndpointConfig {
|
||||||
gateway_id: details.gateway.identity_key.clone(),
|
gateway_id: details.gateway.identity_key.clone(),
|
||||||
gateway_owner: details.owner.to_string(),
|
gateway_owner: details.owner.to_string(),
|
||||||
gateway_listener: format!(
|
gateway_listener: format!(
|
||||||
@@ -33,7 +33,7 @@ pub async fn get_gateway(api_server: String, preferred: Option<String>) -> Gatew
|
|||||||
.first()
|
.first()
|
||||||
.expect("current topology holds no gateways");
|
.expect("current topology holds no gateways");
|
||||||
|
|
||||||
GatewayEndpoint {
|
GatewayEndpointConfig {
|
||||||
gateway_id: details.gateway.identity_key.clone(),
|
gateway_id: details.gateway.identity_key.clone(),
|
||||||
gateway_owner: details.owner.to_string(),
|
gateway_owner: details.owner.to_string(),
|
||||||
gateway_listener: format!(
|
gateway_listener: format!(
|
||||||
|
|||||||
@@ -10,9 +10,11 @@ pub type ConnectionId = u64;
|
|||||||
#[derive(Copy, Clone, Debug, Hash, PartialEq, Eq)]
|
#[derive(Copy, Clone, Debug, Hash, PartialEq, Eq)]
|
||||||
pub enum TransmissionLane {
|
pub enum TransmissionLane {
|
||||||
General,
|
General,
|
||||||
Reply,
|
// we need to treat surb-related requests and responses at higher priority
|
||||||
|
// so that the rest of underlying communication could actually continue
|
||||||
|
ReplySurbRequest,
|
||||||
|
AdditionalReplySurbs,
|
||||||
Retransmission,
|
Retransmission,
|
||||||
Control,
|
|
||||||
ConnectionId(ConnectionId),
|
ConnectionId(ConnectionId),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -26,6 +26,7 @@ network-defaults = { path = "../../network-defaults" }
|
|||||||
nymsphinx = { path = "../../nymsphinx" }
|
nymsphinx = { path = "../../nymsphinx" }
|
||||||
pemstore = { path = "../../pemstore" }
|
pemstore = { path = "../../pemstore" }
|
||||||
validator-client = { path = "../validator-client", optional = true }
|
validator-client = { path = "../validator-client", optional = true }
|
||||||
|
task = { path = "../../task" }
|
||||||
|
|
||||||
|
|
||||||
[dependencies.tungstenite]
|
[dependencies.tungstenite]
|
||||||
@@ -47,9 +48,6 @@ version = "0.14"
|
|||||||
[target."cfg(not(target_arch = \"wasm32\"))".dependencies.credential-storage]
|
[target."cfg(not(target_arch = \"wasm32\"))".dependencies.credential-storage]
|
||||||
path = "../../credential-storage"
|
path = "../../credential-storage"
|
||||||
|
|
||||||
[target."cfg(not(target_arch = \"wasm32\"))".dependencies.task]
|
|
||||||
path = "../../task"
|
|
||||||
|
|
||||||
# wasm-only dependencies
|
# wasm-only dependencies
|
||||||
[target."cfg(target_arch = \"wasm32\")".dependencies.wasm-bindgen]
|
[target."cfg(target_arch = \"wasm32\")".dependencies.wasm-bindgen]
|
||||||
version = "0.2"
|
version = "0.2"
|
||||||
@@ -59,6 +57,7 @@ version = "0.4"
|
|||||||
|
|
||||||
[target."cfg(target_arch = \"wasm32\")".dependencies.wasm-utils]
|
[target."cfg(target_arch = \"wasm32\")".dependencies.wasm-utils]
|
||||||
path = "../../wasm-utils"
|
path = "../../wasm-utils"
|
||||||
|
features = ["websocket"]
|
||||||
|
|
||||||
# only import it in wasm. Prefer proper tokio timer in non-wasm
|
# only import it in wasm. Prefer proper tokio timer in non-wasm
|
||||||
[target."cfg(target_arch = \"wasm32\")".dependencies.wasm-timer]
|
[target."cfg(target_arch = \"wasm32\")".dependencies.wasm-timer]
|
||||||
|
|||||||
@@ -26,8 +26,15 @@ use {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// TODO: make it nicer for wasm (I don't want to touch it for this experiment)
|
||||||
|
#[cfg(target_arch = "wasm32")]
|
||||||
|
use crate::wasm_storage::PersistentStorage;
|
||||||
|
|
||||||
|
#[cfg(not(target_arch = "wasm32"))]
|
||||||
|
use credential_storage::PersistentStorage;
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct BandwidthController<St: Storage> {
|
pub struct BandwidthController<St: Storage = PersistentStorage> {
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
storage: St,
|
storage: St,
|
||||||
#[cfg(feature = "coconut")]
|
#[cfg(feature = "coconut")]
|
||||||
|
|||||||
@@ -10,11 +10,11 @@ pub use crate::packet_router::{
|
|||||||
use crate::socket_state::{PartiallyDelegated, SocketState};
|
use crate::socket_state::{PartiallyDelegated, SocketState};
|
||||||
use crate::{cleanup_socket_message, try_decrypt_binary_message};
|
use crate::{cleanup_socket_message, try_decrypt_binary_message};
|
||||||
use crypto::asymmetric::identity;
|
use crypto::asymmetric::identity;
|
||||||
use futures::{FutureExt, SinkExt, StreamExt};
|
use futures::{SinkExt, StreamExt};
|
||||||
use gateway_requests::authentication::encrypted_address::EncryptedAddressBytes;
|
use gateway_requests::authentication::encrypted_address::EncryptedAddressBytes;
|
||||||
use gateway_requests::iv::IV;
|
use gateway_requests::iv::IV;
|
||||||
use gateway_requests::registration::handshake::{client_handshake, SharedKeys};
|
use gateway_requests::registration::handshake::{client_handshake, SharedKeys};
|
||||||
use gateway_requests::{BinaryRequest, ClientControlRequest, ServerResponse};
|
use gateway_requests::{BinaryRequest, ClientControlRequest, ServerResponse, PROTOCOL_VERSION};
|
||||||
use log::*;
|
use log::*;
|
||||||
use network_defaults::{REMAINING_BANDWIDTH_THRESHOLD, TOKENS_TO_BURN};
|
use network_defaults::{REMAINING_BANDWIDTH_THRESHOLD, TOKENS_TO_BURN};
|
||||||
use nymsphinx::forwarding::packet::MixPacket;
|
use nymsphinx::forwarding::packet::MixPacket;
|
||||||
@@ -22,6 +22,7 @@ use rand::rngs::OsRng;
|
|||||||
use std::convert::TryFrom;
|
use std::convert::TryFrom;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
use task::ShutdownListener;
|
||||||
use tungstenite::protocol::Message;
|
use tungstenite::protocol::Message;
|
||||||
|
|
||||||
#[cfg(feature = "coconut")]
|
#[cfg(feature = "coconut")]
|
||||||
@@ -30,8 +31,6 @@ use coconut_interface::Credential;
|
|||||||
#[cfg(not(target_arch = "wasm32"))]
|
#[cfg(not(target_arch = "wasm32"))]
|
||||||
use credential_storage::PersistentStorage;
|
use credential_storage::PersistentStorage;
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
#[cfg(not(target_arch = "wasm32"))]
|
||||||
use task::ShutdownListener;
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
use tokio_tungstenite::connect_async;
|
use tokio_tungstenite::connect_async;
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
#[cfg(target_arch = "wasm32")]
|
||||||
@@ -67,9 +66,8 @@ pub struct GatewayClient {
|
|||||||
/// Delay between each subsequent reconnection attempt.
|
/// Delay between each subsequent reconnection attempt.
|
||||||
reconnection_backoff: Duration,
|
reconnection_backoff: Duration,
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
/// Listen to shutdown messages.
|
/// Listen to shutdown messages.
|
||||||
shutdown: Option<ShutdownListener>,
|
shutdown: ShutdownListener,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl GatewayClient {
|
impl GatewayClient {
|
||||||
@@ -85,7 +83,7 @@ impl GatewayClient {
|
|||||||
ack_sender: AcknowledgementSender,
|
ack_sender: AcknowledgementSender,
|
||||||
response_timeout_duration: Duration,
|
response_timeout_duration: Duration,
|
||||||
bandwidth_controller: Option<BandwidthController<PersistentStorage>>,
|
bandwidth_controller: Option<BandwidthController<PersistentStorage>>,
|
||||||
#[cfg(not(target_arch = "wasm32"))] shutdown: Option<ShutdownListener>,
|
shutdown: ShutdownListener,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
GatewayClient {
|
GatewayClient {
|
||||||
authenticated: false,
|
authenticated: false,
|
||||||
@@ -97,18 +95,12 @@ impl GatewayClient {
|
|||||||
local_identity,
|
local_identity,
|
||||||
shared_key,
|
shared_key,
|
||||||
connection: SocketState::NotConnected,
|
connection: SocketState::NotConnected,
|
||||||
packet_router: PacketRouter::new(
|
packet_router: PacketRouter::new(ack_sender, mixnet_message_sender, shutdown.clone()),
|
||||||
ack_sender,
|
|
||||||
mixnet_message_sender,
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
shutdown.clone(),
|
|
||||||
),
|
|
||||||
response_timeout_duration,
|
response_timeout_duration,
|
||||||
bandwidth_controller,
|
bandwidth_controller,
|
||||||
should_reconnect_on_failure: true,
|
should_reconnect_on_failure: true,
|
||||||
reconnection_attempts: DEFAULT_RECONNECTION_ATTEMPTS,
|
reconnection_attempts: DEFAULT_RECONNECTION_ATTEMPTS,
|
||||||
reconnection_backoff: DEFAULT_RECONNECTION_BACKOFF,
|
reconnection_backoff: DEFAULT_RECONNECTION_BACKOFF,
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
shutdown,
|
shutdown,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -136,7 +128,6 @@ impl GatewayClient {
|
|||||||
gateway_owner: String,
|
gateway_owner: String,
|
||||||
local_identity: Arc<identity::KeyPair>,
|
local_identity: Arc<identity::KeyPair>,
|
||||||
response_timeout_duration: Duration,
|
response_timeout_duration: Duration,
|
||||||
#[cfg(not(target_arch = "wasm32"))] shutdown: Option<ShutdownListener>,
|
|
||||||
) -> Self {
|
) -> Self {
|
||||||
use futures::channel::mpsc;
|
use futures::channel::mpsc;
|
||||||
|
|
||||||
@@ -144,12 +135,8 @@ impl GatewayClient {
|
|||||||
// perfectly fine here, because it's not meant to be used
|
// perfectly fine here, because it's not meant to be used
|
||||||
let (ack_tx, _) = mpsc::unbounded();
|
let (ack_tx, _) = mpsc::unbounded();
|
||||||
let (mix_tx, _) = mpsc::unbounded();
|
let (mix_tx, _) = mpsc::unbounded();
|
||||||
let packet_router = PacketRouter::new(
|
let shutdown = ShutdownListener::dummy();
|
||||||
ack_tx,
|
let packet_router = PacketRouter::new(ack_tx, mix_tx, shutdown.clone());
|
||||||
mix_tx,
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
shutdown.clone(),
|
|
||||||
);
|
|
||||||
|
|
||||||
GatewayClient {
|
GatewayClient {
|
||||||
authenticated: false,
|
authenticated: false,
|
||||||
@@ -167,7 +154,6 @@ impl GatewayClient {
|
|||||||
should_reconnect_on_failure: false,
|
should_reconnect_on_failure: false,
|
||||||
reconnection_attempts: DEFAULT_RECONNECTION_ATTEMPTS,
|
reconnection_attempts: DEFAULT_RECONNECTION_ATTEMPTS,
|
||||||
reconnection_backoff: DEFAULT_RECONNECTION_BACKOFF,
|
reconnection_backoff: DEFAULT_RECONNECTION_BACKOFF,
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
shutdown,
|
shutdown,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -295,42 +281,19 @@ impl GatewayClient {
|
|||||||
// technically the `wasm_timer` also works outside wasm, but unless required,
|
// technically the `wasm_timer` also works outside wasm, but unless required,
|
||||||
// I really prefer to just stick to tokio
|
// I really prefer to just stick to tokio
|
||||||
#[cfg(target_arch = "wasm32")]
|
#[cfg(target_arch = "wasm32")]
|
||||||
let timeout = wasm_timer::Delay::new(self.response_timeout_duration);
|
let mut timeout = wasm_timer::Delay::new(self.response_timeout_duration);
|
||||||
|
|
||||||
let mut fused_timeout = timeout.fuse();
|
|
||||||
let mut fused_stream = conn.fuse();
|
|
||||||
|
|
||||||
// Bit of an ugly workaround for selecting on an `Option` without having access to
|
|
||||||
// `tokio::select`
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
let shutdown = {
|
|
||||||
let m_shutdown = self.shutdown.clone();
|
|
||||||
async {
|
|
||||||
if let Some(mut s) = m_shutdown {
|
|
||||||
s.recv().await
|
|
||||||
} else {
|
|
||||||
std::future::pending::<()>().await
|
|
||||||
}
|
|
||||||
}
|
|
||||||
.fuse()
|
|
||||||
};
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
tokio::pin!(shutdown);
|
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
let mut shutdown = std::future::pending::<()>().fuse();
|
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
futures::select! {
|
tokio::select! {
|
||||||
_ = shutdown => {
|
_ = self.shutdown.recv() => {
|
||||||
log::trace!("GatewayClient control response: Received shutdown");
|
log::trace!("GatewayClient control response: Received shutdown");
|
||||||
log::debug!("GatewayClient control response: Exiting");
|
log::debug!("GatewayClient control response: Exiting");
|
||||||
break Err(GatewayClientError::ConnectionClosedGatewayShutdown);
|
break Err(GatewayClientError::ConnectionClosedGatewayShutdown);
|
||||||
}
|
}
|
||||||
_ = &mut fused_timeout => {
|
_ = &mut timeout => {
|
||||||
break Err(GatewayClientError::Timeout);
|
break Err(GatewayClientError::Timeout);
|
||||||
}
|
}
|
||||||
msg = fused_stream.next() => {
|
msg = conn.next() => {
|
||||||
let ws_msg = match cleanup_socket_message(msg) {
|
let ws_msg = match cleanup_socket_message(msg) {
|
||||||
Err(err) => break Err(err),
|
Err(err) => break Err(err),
|
||||||
Ok(msg) => msg
|
Ok(msg) => msg
|
||||||
@@ -401,13 +364,10 @@ impl GatewayClient {
|
|||||||
.batch_send_without_response(messages)
|
.batch_send_without_response(messages)
|
||||||
.await
|
.await
|
||||||
{
|
{
|
||||||
error!("failed to batch send messages - {}...", err);
|
error!("failed to batch send messages - {err}...");
|
||||||
// we must ensure we do not leave the task still active
|
// we must ensure we do not leave the task still active
|
||||||
if let Err(err) = self.recover_socket_connection().await {
|
if let Err(err) = self.recover_socket_connection().await {
|
||||||
error!(
|
error!("... and the delegated stream has also errored out - {err}")
|
||||||
"... and the delegated stream has also errored out - {}",
|
|
||||||
err
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
Err(err)
|
Err(err)
|
||||||
} else {
|
} else {
|
||||||
@@ -427,13 +387,10 @@ impl GatewayClient {
|
|||||||
SocketState::Available(ref mut conn) => Ok(conn.send(msg).await?),
|
SocketState::Available(ref mut conn) => Ok(conn.send(msg).await?),
|
||||||
SocketState::PartiallyDelegated(ref mut partially_delegated) => {
|
SocketState::PartiallyDelegated(ref mut partially_delegated) => {
|
||||||
if let Err(err) = partially_delegated.send_without_response(msg).await {
|
if let Err(err) = partially_delegated.send_without_response(msg).await {
|
||||||
error!("failed to send message without response - {}...", err);
|
error!("failed to send message without response - {err}...");
|
||||||
// we must ensure we do not leave the task still active
|
// we must ensure we do not leave the task still active
|
||||||
if let Err(err) = self.recover_socket_connection().await {
|
if let Err(err) = self.recover_socket_connection().await {
|
||||||
error!(
|
error!("... and the delegated stream has also errored out - {err}")
|
||||||
"... and the delegated stream has also errored out - {}",
|
|
||||||
err
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
Err(err)
|
Err(err)
|
||||||
} else {
|
} else {
|
||||||
@@ -445,6 +402,33 @@ impl GatewayClient {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn check_gateway_protocol(
|
||||||
|
&self,
|
||||||
|
gateway_protocol: Option<u8>,
|
||||||
|
) -> Result<(), GatewayClientError> {
|
||||||
|
// right now there are no failure cases here, but this might change in the future
|
||||||
|
match gateway_protocol {
|
||||||
|
None => {
|
||||||
|
warn!("the gateway we're connected to has not specified its protocol version. It's probably running version < 1.1.X, but that's still fine for now. It will become a hard error in 1.2.0");
|
||||||
|
// note: in +1.2.0 we will have to return a hard error here
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
Some(v) if v != PROTOCOL_VERSION => {
|
||||||
|
let err = GatewayClientError::IncompatibleProtocol {
|
||||||
|
gateway: Some(v),
|
||||||
|
current: PROTOCOL_VERSION,
|
||||||
|
};
|
||||||
|
error!("{err}");
|
||||||
|
Err(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
Some(_) => {
|
||||||
|
info!("the gateway is using exactly the same protocol version as we are. We're good to continue!");
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
async fn register(&mut self) -> Result<(), GatewayClientError> {
|
async fn register(&mut self) -> Result<(), GatewayClientError> {
|
||||||
if !self.connection.is_established() {
|
if !self.connection.is_established() {
|
||||||
return Err(GatewayClientError::ConnectionNotEstablished);
|
return Err(GatewayClientError::ConnectionNotEstablished);
|
||||||
@@ -467,11 +451,20 @@ impl GatewayClient {
|
|||||||
.map_err(GatewayClientError::RegistrationFailure),
|
.map_err(GatewayClientError::RegistrationFailure),
|
||||||
_ => unreachable!(),
|
_ => unreachable!(),
|
||||||
}?;
|
}?;
|
||||||
self.authenticated = match self.read_control_response().await? {
|
let (authentication_status, gateway_protocol) = match self.read_control_response().await? {
|
||||||
ServerResponse::Register { status } => Ok(status),
|
ServerResponse::Register {
|
||||||
ServerResponse::Error { message } => Err(GatewayClientError::GatewayError(message)),
|
protocol_version,
|
||||||
_ => Err(GatewayClientError::UnexpectedResponse),
|
status,
|
||||||
}?;
|
} => (status, protocol_version),
|
||||||
|
ServerResponse::Error { message } => {
|
||||||
|
return Err(GatewayClientError::GatewayError(message))
|
||||||
|
}
|
||||||
|
_ => return Err(GatewayClientError::UnexpectedResponse),
|
||||||
|
};
|
||||||
|
|
||||||
|
self.check_gateway_protocol(gateway_protocol)?;
|
||||||
|
self.authenticated = authentication_status;
|
||||||
|
|
||||||
if self.authenticated {
|
if self.authenticated {
|
||||||
self.shared_key = Some(Arc::new(shared_key));
|
self.shared_key = Some(Arc::new(shared_key));
|
||||||
}
|
}
|
||||||
@@ -510,9 +503,11 @@ impl GatewayClient {
|
|||||||
|
|
||||||
match self.send_websocket_message(msg).await? {
|
match self.send_websocket_message(msg).await? {
|
||||||
ServerResponse::Authenticate {
|
ServerResponse::Authenticate {
|
||||||
|
protocol_version,
|
||||||
status,
|
status,
|
||||||
bandwidth_remaining,
|
bandwidth_remaining,
|
||||||
} => {
|
} => {
|
||||||
|
self.check_gateway_protocol(protocol_version)?;
|
||||||
self.authenticated = status;
|
self.authenticated = status;
|
||||||
self.bandwidth_remaining = bandwidth_remaining;
|
self.bandwidth_remaining = bandwidth_remaining;
|
||||||
Ok(())
|
Ok(())
|
||||||
@@ -748,7 +743,6 @@ impl GatewayClient {
|
|||||||
.as_ref()
|
.as_ref()
|
||||||
.expect("no shared key present even though we're authenticated!"),
|
.expect("no shared key present even though we're authenticated!"),
|
||||||
),
|
),
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
self.shutdown.clone(),
|
self.shutdown.clone(),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -85,6 +85,9 @@ pub enum GatewayClientError {
|
|||||||
|
|
||||||
#[error("Failed to send mixnet message")]
|
#[error("Failed to send mixnet message")]
|
||||||
MixnetMsgSenderFailedToSend,
|
MixnetMsgSenderFailedToSend,
|
||||||
|
|
||||||
|
#[error("Attempted to negotiate connection with gateway using incompatible protocol version. Ours is {current} and the gateway reports {gateway:?}")]
|
||||||
|
IncompatibleProtocol { gateway: Option<u8>, current: u8 },
|
||||||
}
|
}
|
||||||
|
|
||||||
impl GatewayClientError {
|
impl GatewayClientError {
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ pub mod client;
|
|||||||
pub mod error;
|
pub mod error;
|
||||||
pub mod packet_router;
|
pub mod packet_router;
|
||||||
pub mod socket_state;
|
pub mod socket_state;
|
||||||
#[cfg(feature = "wasm")]
|
#[cfg(target_arch = "wasm32")]
|
||||||
mod wasm_storage;
|
mod wasm_storage;
|
||||||
|
|
||||||
/// Helper method for reading from websocket stream. Helps to flatten the structure.
|
/// Helper method for reading from websocket stream. Helps to flatten the structure.
|
||||||
|
|||||||
@@ -4,15 +4,13 @@
|
|||||||
// JS: I personally don't like this name very much, but could not think of anything better.
|
// JS: I personally don't like this name very much, but could not think of anything better.
|
||||||
// I will gladly take any suggestions on how to rename this.
|
// I will gladly take any suggestions on how to rename this.
|
||||||
|
|
||||||
|
use crate::error::GatewayClientError;
|
||||||
use futures::channel::mpsc;
|
use futures::channel::mpsc;
|
||||||
use log::*;
|
use log::*;
|
||||||
use nymsphinx::addressing::nodes::MAX_NODE_ADDRESS_UNPADDED_LEN;
|
use nymsphinx::addressing::nodes::MAX_NODE_ADDRESS_UNPADDED_LEN;
|
||||||
use nymsphinx::params::packet_sizes::PacketSize;
|
use nymsphinx::params::packet_sizes::PacketSize;
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
use task::ShutdownListener;
|
use task::ShutdownListener;
|
||||||
|
|
||||||
use crate::error::GatewayClientError;
|
|
||||||
|
|
||||||
pub type MixnetMessageSender = mpsc::UnboundedSender<Vec<Vec<u8>>>;
|
pub type MixnetMessageSender = mpsc::UnboundedSender<Vec<Vec<u8>>>;
|
||||||
pub type MixnetMessageReceiver = mpsc::UnboundedReceiver<Vec<Vec<u8>>>;
|
pub type MixnetMessageReceiver = mpsc::UnboundedReceiver<Vec<Vec<u8>>>;
|
||||||
|
|
||||||
@@ -23,20 +21,18 @@ pub type AcknowledgementReceiver = mpsc::UnboundedReceiver<Vec<Vec<u8>>>;
|
|||||||
pub struct PacketRouter {
|
pub struct PacketRouter {
|
||||||
ack_sender: AcknowledgementSender,
|
ack_sender: AcknowledgementSender,
|
||||||
mixnet_message_sender: MixnetMessageSender,
|
mixnet_message_sender: MixnetMessageSender,
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
shutdown: ShutdownListener,
|
||||||
shutdown: Option<ShutdownListener>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl PacketRouter {
|
impl PacketRouter {
|
||||||
pub fn new(
|
pub fn new(
|
||||||
ack_sender: AcknowledgementSender,
|
ack_sender: AcknowledgementSender,
|
||||||
mixnet_message_sender: MixnetMessageSender,
|
mixnet_message_sender: MixnetMessageSender,
|
||||||
#[cfg(not(target_arch = "wasm32"))] shutdown: Option<ShutdownListener>,
|
shutdown: ShutdownListener,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
PacketRouter {
|
PacketRouter {
|
||||||
ack_sender,
|
ack_sender,
|
||||||
mixnet_message_sender,
|
mixnet_message_sender,
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
shutdown,
|
shutdown,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -86,13 +82,10 @@ impl PacketRouter {
|
|||||||
if !received_messages.is_empty() {
|
if !received_messages.is_empty() {
|
||||||
trace!("routing 'real'");
|
trace!("routing 'real'");
|
||||||
if let Err(err) = self.mixnet_message_sender.unbounded_send(received_messages) {
|
if let Err(err) = self.mixnet_message_sender.unbounded_send(received_messages) {
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
if self.shutdown.is_shutdown_poll() || self.shutdown.is_dummy() {
|
||||||
if let Some(shutdown) = &mut self.shutdown {
|
// This should ideally not happen, but it's ok
|
||||||
if shutdown.is_shutdown_poll() {
|
log::warn!("Failed to send mixnet message due to receiver task shutdown");
|
||||||
// This should ideally not happen, but it's ok
|
return Err(GatewayClientError::MixnetMsgSenderFailedToSend);
|
||||||
log::warn!("Failed to send mixnet message due to receiver task shutdown");
|
|
||||||
return Err(GatewayClientError::MixnetMsgSenderFailedToSend);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
// This should never happen during ordinary operation the way it's currently used.
|
// This should never happen during ordinary operation the way it's currently used.
|
||||||
// Abort to be on the safe side
|
// Abort to be on the safe side
|
||||||
|
|||||||
@@ -10,7 +10,6 @@ use futures::{SinkExt, StreamExt};
|
|||||||
use gateway_requests::registration::handshake::SharedKeys;
|
use gateway_requests::registration::handshake::SharedKeys;
|
||||||
use log::*;
|
use log::*;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
use task::ShutdownListener;
|
use task::ShutdownListener;
|
||||||
use tungstenite::Message;
|
use tungstenite::Message;
|
||||||
|
|
||||||
@@ -85,7 +84,7 @@ impl PartiallyDelegated {
|
|||||||
conn: WsConn,
|
conn: WsConn,
|
||||||
packet_router: PacketRouter,
|
packet_router: PacketRouter,
|
||||||
shared_key: Arc<SharedKeys>,
|
shared_key: Arc<SharedKeys>,
|
||||||
#[cfg(not(target_arch = "wasm32"))] shutdown: Option<ShutdownListener>,
|
mut shutdown: ShutdownListener,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
// when called for, it NEEDS TO yield back the stream so that we could merge it and
|
// when called for, it NEEDS TO yield back the stream so that we could merge it and
|
||||||
// read control request responses.
|
// read control request responses.
|
||||||
@@ -99,27 +98,9 @@ impl PartiallyDelegated {
|
|||||||
let mut chunk_stream = (&mut stream).ready_chunks(8);
|
let mut chunk_stream = (&mut stream).ready_chunks(8);
|
||||||
let mut packet_router = packet_router;
|
let mut packet_router = packet_router;
|
||||||
|
|
||||||
// Bit of an ugly workaround for selecting on an `Option` without having access to
|
|
||||||
// `tokio::select`
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
let shutdown = {
|
|
||||||
async {
|
|
||||||
if let Some(mut s) = shutdown {
|
|
||||||
s.recv().await
|
|
||||||
} else {
|
|
||||||
std::future::pending::<()>().await
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
|
||||||
tokio::pin!(shutdown);
|
|
||||||
|
|
||||||
#[cfg(target_arch = "wasm32")]
|
|
||||||
let mut shutdown = std::future::pending::<()>();
|
|
||||||
|
|
||||||
let ret_err = loop {
|
let ret_err = loop {
|
||||||
tokio::select! {
|
tokio::select! {
|
||||||
_ = &mut shutdown => {
|
_ = shutdown.recv() => {
|
||||||
log::trace!("GatewayClient listener: Received shutdown");
|
log::trace!("GatewayClient listener: Received shutdown");
|
||||||
log::debug!("GatewayClient listener: Exiting");
|
log::debug!("GatewayClient listener: Exiting");
|
||||||
return;
|
return;
|
||||||
@@ -142,7 +123,10 @@ impl PartiallyDelegated {
|
|||||||
|
|
||||||
if match ret_err {
|
if match ret_err {
|
||||||
Err(err) => stream_sender.send(Err(err)),
|
Err(err) => stream_sender.send(Err(err)),
|
||||||
Ok(_) => stream_sender.send(Ok(stream)),
|
Ok(_) => {
|
||||||
|
shutdown.mark_as_success();
|
||||||
|
stream_sender.send(Ok(stream))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
.is_err()
|
.is_err()
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -180,6 +180,35 @@ pub trait MixnetSigningClient {
|
|||||||
.await
|
.await
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async fn pledge_more(
|
||||||
|
&self,
|
||||||
|
additional_pledge: Coin,
|
||||||
|
fee: Option<Fee>,
|
||||||
|
) -> Result<ExecuteResult, NymdError> {
|
||||||
|
self.execute_mixnet_contract(
|
||||||
|
fee,
|
||||||
|
MixnetExecuteMsg::PledgeMore {},
|
||||||
|
vec![additional_pledge],
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn pledge_more_on_behalf(
|
||||||
|
&self,
|
||||||
|
owner: AccountId,
|
||||||
|
additional_pledge: Coin,
|
||||||
|
fee: Option<Fee>,
|
||||||
|
) -> Result<ExecuteResult, NymdError> {
|
||||||
|
self.execute_mixnet_contract(
|
||||||
|
fee,
|
||||||
|
MixnetExecuteMsg::PledgeMoreOnBehalf {
|
||||||
|
owner: owner.to_string(),
|
||||||
|
},
|
||||||
|
vec![additional_pledge],
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
async fn unbond_mixnode(&self, fee: Option<Fee>) -> Result<ExecuteResult, NymdError> {
|
async fn unbond_mixnode(&self, fee: Option<Fee>) -> Result<ExecuteResult, NymdError> {
|
||||||
self.execute_mixnet_contract(fee, MixnetExecuteMsg::UnbondMixnode {}, vec![])
|
self.execute_mixnet_contract(fee, MixnetExecuteMsg::UnbondMixnode {}, vec![])
|
||||||
.await
|
.await
|
||||||
|
|||||||
@@ -64,6 +64,21 @@ pub trait VestingSigningClient {
|
|||||||
fee: Option<Fee>,
|
fee: Option<Fee>,
|
||||||
) -> Result<ExecuteResult, NymdError>;
|
) -> Result<ExecuteResult, NymdError>;
|
||||||
|
|
||||||
|
async fn vesting_pledge_more(
|
||||||
|
&self,
|
||||||
|
additional_pledge: Coin,
|
||||||
|
fee: Option<Fee>,
|
||||||
|
) -> Result<ExecuteResult, NymdError> {
|
||||||
|
self.execute_vesting_contract(
|
||||||
|
fee,
|
||||||
|
VestingExecuteMsg::PledgeMore {
|
||||||
|
amount: additional_pledge.into(),
|
||||||
|
},
|
||||||
|
vec![],
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
async fn vesting_unbond_mixnode(&self, fee: Option<Fee>) -> Result<ExecuteResult, NymdError>;
|
async fn vesting_unbond_mixnode(&self, fee: Option<Fee>) -> Result<ExecuteResult, NymdError>;
|
||||||
|
|
||||||
async fn vesting_track_unbond_mixnode(
|
async fn vesting_track_unbond_mixnode(
|
||||||
|
|||||||
@@ -136,7 +136,12 @@ impl Client {
|
|||||||
&self,
|
&self,
|
||||||
) -> Result<Vec<MixNodeBondAnnotated>, ValidatorAPIError> {
|
) -> Result<Vec<MixNodeBondAnnotated>, ValidatorAPIError> {
|
||||||
self.query_validator_api(
|
self.query_validator_api(
|
||||||
&[routes::API_VERSION, routes::MIXNODES, routes::DETAILED],
|
&[
|
||||||
|
routes::API_VERSION,
|
||||||
|
routes::STATUS,
|
||||||
|
routes::MIXNODES,
|
||||||
|
routes::DETAILED,
|
||||||
|
],
|
||||||
NO_PARAMS,
|
NO_PARAMS,
|
||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
@@ -161,6 +166,7 @@ impl Client {
|
|||||||
self.query_validator_api(
|
self.query_validator_api(
|
||||||
&[
|
&[
|
||||||
routes::API_VERSION,
|
routes::API_VERSION,
|
||||||
|
routes::STATUS,
|
||||||
routes::MIXNODES,
|
routes::MIXNODES,
|
||||||
routes::ACTIVE,
|
routes::ACTIVE,
|
||||||
routes::DETAILED,
|
routes::DETAILED,
|
||||||
@@ -252,6 +258,7 @@ impl Client {
|
|||||||
self.query_validator_api(
|
self.query_validator_api(
|
||||||
&[
|
&[
|
||||||
routes::API_VERSION,
|
routes::API_VERSION,
|
||||||
|
routes::STATUS,
|
||||||
routes::MIXNODES,
|
routes::MIXNODES,
|
||||||
routes::REWARDED,
|
routes::REWARDED,
|
||||||
routes::DETAILED,
|
routes::DETAILED,
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ thiserror = "1"
|
|||||||
time = { version = "0.3.6", features = ["parsing", "formatting"] }
|
time = { version = "0.3.6", features = ["parsing", "formatting"] }
|
||||||
toml = "0.5.6"
|
toml = "0.5.6"
|
||||||
url = "2.2"
|
url = "2.2"
|
||||||
|
tap = "1"
|
||||||
|
|
||||||
cosmrs = { git = "https://github.com/neacsu/cosmos-rust", branch = "neacsu/feegrant_support" }
|
cosmrs = { git = "https://github.com/neacsu/cosmos-rust", branch = "neacsu/feegrant_support" }
|
||||||
cosmwasm-std = { version = "1.0.0" }
|
cosmwasm-std = { version = "1.0.0" }
|
||||||
|
|||||||
@@ -15,4 +15,10 @@ pub enum ContextError {
|
|||||||
// TODO: improve this to return known errors
|
// TODO: improve this to return known errors
|
||||||
#[error("failed to create client - {0}")]
|
#[error("failed to create client - {0}")]
|
||||||
NymdError(String),
|
NymdError(String),
|
||||||
|
|
||||||
|
#[error("{0}")]
|
||||||
|
NymdErrorPassthrough(#[from] validator_client::nymd::error::NymdError),
|
||||||
|
|
||||||
|
#[error("{0}")]
|
||||||
|
ValidatorClientError(#[from] validator_client::ValidatorClientError),
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ use network_defaults::{
|
|||||||
var_names::{API_VALIDATOR, MIXNET_CONTRACT_ADDRESS, NYMD_VALIDATOR, VESTING_CONTRACT_ADDRESS},
|
var_names::{API_VALIDATOR, MIXNET_CONTRACT_ADDRESS, NYMD_VALIDATOR, VESTING_CONTRACT_ADDRESS},
|
||||||
NymNetworkDetails,
|
NymNetworkDetails,
|
||||||
};
|
};
|
||||||
|
use tap::prelude::*;
|
||||||
use validator_client::nymd::{self, AccountId, NymdClient, QueryNymdClient, SigningNymdClient};
|
use validator_client::nymd::{self, AccountId, NymdClient, QueryNymdClient, SigningNymdClient};
|
||||||
pub use validator_client::validator_api::Client as ValidatorApiClient;
|
pub use validator_client::validator_api::Client as ValidatorApiClient;
|
||||||
|
|
||||||
@@ -58,7 +59,7 @@ pub fn create_signing_client(
|
|||||||
network_details: &NymNetworkDetails,
|
network_details: &NymNetworkDetails,
|
||||||
) -> Result<SigningClient, ContextError> {
|
) -> Result<SigningClient, ContextError> {
|
||||||
let client_config = nymd::Config::try_from_nym_network_details(network_details)
|
let client_config = nymd::Config::try_from_nym_network_details(network_details)
|
||||||
.expect("failed to construct valid validator client config with the provided network");
|
.tap_err(|e| log::error!("Failed to get client config - {:?}", e))?;
|
||||||
|
|
||||||
// get mnemonic
|
// get mnemonic
|
||||||
let mnemonic = match std::env::var("MNEMONIC") {
|
let mnemonic = match std::env::var("MNEMONIC") {
|
||||||
@@ -87,7 +88,7 @@ pub fn create_query_client(
|
|||||||
network_details: &NymNetworkDetails,
|
network_details: &NymNetworkDetails,
|
||||||
) -> Result<QueryClient, ContextError> {
|
) -> Result<QueryClient, ContextError> {
|
||||||
let client_config = nymd::Config::try_from_nym_network_details(network_details)
|
let client_config = nymd::Config::try_from_nym_network_details(network_details)
|
||||||
.expect("failed to construct valid validator client config with the provided network");
|
.tap_err(|e| log::error!("Failed to get client config - {:?}", e))?;
|
||||||
|
|
||||||
let nymd_url = network_details
|
let nymd_url = network_details
|
||||||
.endpoints
|
.endpoints
|
||||||
@@ -107,7 +108,7 @@ pub fn create_signing_client_with_validator_api(
|
|||||||
network_details: &NymNetworkDetails,
|
network_details: &NymNetworkDetails,
|
||||||
) -> Result<SigningClientWithValidatorAPI, ContextError> {
|
) -> Result<SigningClientWithValidatorAPI, ContextError> {
|
||||||
let client_config = validator_client::Config::try_from_nym_network_details(network_details)
|
let client_config = validator_client::Config::try_from_nym_network_details(network_details)
|
||||||
.expect("failed to construct valid validator client config with the provided network");
|
.tap_err(|e| log::error!("Failed to get client config - {:?}", e))?;
|
||||||
|
|
||||||
// get mnemonic
|
// get mnemonic
|
||||||
let mnemonic = match std::env::var("MNEMONIC") {
|
let mnemonic = match std::env::var("MNEMONIC") {
|
||||||
@@ -129,7 +130,7 @@ pub fn create_query_client_with_validator_api(
|
|||||||
network_details: &NymNetworkDetails,
|
network_details: &NymNetworkDetails,
|
||||||
) -> Result<QueryClientWithValidatorAPI, ContextError> {
|
) -> Result<QueryClientWithValidatorAPI, ContextError> {
|
||||||
let client_config = validator_client::Config::try_from_nym_network_details(network_details)
|
let client_config = validator_client::Config::try_from_nym_network_details(network_details)
|
||||||
.expect("failed to construct valid validator client config with the provided network");
|
.tap_err(|e| log::error!("Failed to get client config - {:?}", e))?;
|
||||||
|
|
||||||
match validator_client::client::Client::new_query(client_config) {
|
match validator_client::client::Client::new_query(client_config) {
|
||||||
Ok(client) => Ok(client),
|
Ok(client) => Ok(client),
|
||||||
|
|||||||
@@ -3,11 +3,11 @@
|
|||||||
|
|
||||||
use clap::Parser;
|
use clap::Parser;
|
||||||
use cosmrs::AccountId;
|
use cosmrs::AccountId;
|
||||||
use log::info;
|
use log::{error, info};
|
||||||
|
|
||||||
use validator_client::nymd::{Coin, VestingQueryClient};
|
use validator_client::nymd::{Coin, VestingQueryClient};
|
||||||
|
|
||||||
use crate::context::SigningClient;
|
use crate::context::QueryClient;
|
||||||
use crate::utils::show_error;
|
use crate::utils::show_error;
|
||||||
use crate::utils::{pretty_coin, pretty_cosmwasm_coin};
|
use crate::utils::{pretty_coin, pretty_cosmwasm_coin};
|
||||||
|
|
||||||
@@ -18,8 +18,16 @@ pub struct Args {
|
|||||||
pub address: Option<AccountId>,
|
pub address: Option<AccountId>,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn balance(args: Args, client: SigningClient) {
|
pub async fn balance(args: Args, client: QueryClient, address_from_mnemonic: Option<AccountId>) {
|
||||||
let account_id = args.address.unwrap_or_else(|| client.address().clone());
|
if args.address.is_none() && address_from_mnemonic.is_none() {
|
||||||
|
error!("Please specify an account address or a mnemonic to get the balance for");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let account_id = args
|
||||||
|
.address
|
||||||
|
.unwrap_or_else(|| address_from_mnemonic.expect("please provide a mnemonic"));
|
||||||
|
|
||||||
let vesting_address = account_id.to_string();
|
let vesting_address = account_id.to_string();
|
||||||
let denom = client.current_chain_details().mix_denom.base.as_str();
|
let denom = client.current_chain_details().mix_denom.base.as_str();
|
||||||
|
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user