forked from LBRYCommunity/lbry-sdk
Compare commits
14 commits
master
...
crazy_hub_
Author | SHA1 | Date | |
---|---|---|---|
|
1f64275da1 | ||
|
8c605864e7 | ||
|
6ec88e2101 | ||
|
17eeafd62c | ||
|
28e4b3eefd | ||
|
b2de89ca29 | ||
|
8bfff2d549 | ||
|
39d5078788 | ||
|
f8c46647d7 | ||
|
82de92c324 | ||
|
3dc3792478 | ||
|
ba1d0a12d1 | ||
|
e75047a0ab | ||
|
83a167bd37 |
168 changed files with 14526 additions and 5893 deletions
78
.github/workflows/main.yml
vendored
78
.github/workflows/main.yml
vendored
|
@ -1,24 +1,24 @@
|
||||||
name: ci
|
name: ci
|
||||||
on: ["push", "pull_request", "workflow_dispatch"]
|
on: ["push", "pull_request"]
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
|
||||||
lint:
|
lint:
|
||||||
name: lint
|
name: lint
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v2
|
||||||
- uses: actions/setup-python@v4
|
- uses: actions/setup-python@v1
|
||||||
with:
|
with:
|
||||||
python-version: '3.9'
|
python-version: '3.7'
|
||||||
- name: extract pip cache
|
- name: extract pip cache
|
||||||
uses: actions/cache@v3
|
uses: actions/cache@v2
|
||||||
with:
|
with:
|
||||||
path: ~/.cache/pip
|
path: ~/.cache/pip
|
||||||
key: ${{ runner.os }}-pip-${{ hashFiles('setup.py') }}
|
key: ${{ runner.os }}-pip-${{ hashFiles('setup.py') }}
|
||||||
restore-keys: ${{ runner.os }}-pip-
|
restore-keys: ${{ runner.os }}-pip-
|
||||||
- run: pip install --user --upgrade pip wheel
|
- run: pip install --user --upgrade pip wheel
|
||||||
- run: pip install -e .[lint]
|
- run: pip install -e .[torrent,lint]
|
||||||
- run: make lint
|
- run: make lint
|
||||||
|
|
||||||
tests-unit:
|
tests-unit:
|
||||||
|
@ -26,31 +26,31 @@ jobs:
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
os:
|
os:
|
||||||
- ubuntu-20.04
|
- ubuntu-latest
|
||||||
- macos-latest
|
- macos-latest
|
||||||
- windows-latest
|
- windows-latest
|
||||||
runs-on: ${{ matrix.os }}
|
runs-on: ${{ matrix.os }}
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v2
|
||||||
- uses: actions/setup-python@v4
|
- uses: actions/setup-python@v1
|
||||||
with:
|
with:
|
||||||
python-version: '3.9'
|
python-version: '3.7'
|
||||||
- name: set pip cache dir
|
- name: set pip cache dir
|
||||||
shell: bash
|
id: pip-cache
|
||||||
run: echo "PIP_CACHE_DIR=$(pip cache dir)" >> $GITHUB_ENV
|
run: echo "::set-output name=dir::$(pip cache dir)"
|
||||||
- name: extract pip cache
|
- name: extract pip cache
|
||||||
uses: actions/cache@v3
|
uses: actions/cache@v2
|
||||||
with:
|
with:
|
||||||
path: ${{ env.PIP_CACHE_DIR }}
|
path: ${{ steps.pip-cache.outputs.dir }}
|
||||||
key: ${{ runner.os }}-pip-${{ hashFiles('setup.py') }}
|
key: ${{ runner.os }}-pip-${{ hashFiles('setup.py') }}
|
||||||
restore-keys: ${{ runner.os }}-pip-
|
restore-keys: ${{ runner.os }}-pip-
|
||||||
- id: os-name
|
- id: os-name
|
||||||
uses: ASzc/change-string-case-action@v5
|
uses: ASzc/change-string-case-action@v1
|
||||||
with:
|
with:
|
||||||
string: ${{ runner.os }}
|
string: ${{ runner.os }}
|
||||||
- run: python -m pip install --user --upgrade pip wheel
|
- run: pip install --user --upgrade pip wheel
|
||||||
- if: startsWith(runner.os, 'linux')
|
- if: startsWith(runner.os, 'linux')
|
||||||
run: pip install -e .[test]
|
run: pip install -e .[torrent,test]
|
||||||
- if: startsWith(runner.os, 'linux')
|
- if: startsWith(runner.os, 'linux')
|
||||||
env:
|
env:
|
||||||
HOME: /tmp
|
HOME: /tmp
|
||||||
|
@ -72,7 +72,7 @@ jobs:
|
||||||
|
|
||||||
tests-integration:
|
tests-integration:
|
||||||
name: "tests / integration"
|
name: "tests / integration"
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-latest
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
test:
|
test:
|
||||||
|
@ -81,6 +81,8 @@ jobs:
|
||||||
- claims
|
- claims
|
||||||
- takeovers
|
- takeovers
|
||||||
- transactions
|
- transactions
|
||||||
|
- claims_legacy_search
|
||||||
|
- takeovers_legacy_search
|
||||||
- other
|
- other
|
||||||
steps:
|
steps:
|
||||||
- name: Configure sysctl limits
|
- name: Configure sysctl limits
|
||||||
|
@ -92,24 +94,22 @@ jobs:
|
||||||
- name: Runs Elasticsearch
|
- name: Runs Elasticsearch
|
||||||
uses: elastic/elastic-github-actions/elasticsearch@master
|
uses: elastic/elastic-github-actions/elasticsearch@master
|
||||||
with:
|
with:
|
||||||
stack-version: 7.12.1
|
stack-version: 7.6.0
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v2
|
||||||
- uses: actions/setup-python@v4
|
- uses: actions/setup-python@v1
|
||||||
with:
|
with:
|
||||||
python-version: '3.9'
|
python-version: '3.7'
|
||||||
- if: matrix.test == 'other'
|
- if: matrix.test == 'other'
|
||||||
run: |
|
run: |
|
||||||
sudo apt-get update
|
sudo apt-get update
|
||||||
sudo apt-get install -y --no-install-recommends ffmpeg
|
sudo apt-get install -y --no-install-recommends ffmpeg
|
||||||
- name: extract pip cache
|
- name: extract pip cache
|
||||||
uses: actions/cache@v3
|
uses: actions/cache@v2
|
||||||
with:
|
with:
|
||||||
path: ./.tox
|
path: ./.tox
|
||||||
key: tox-integration-${{ matrix.test }}-${{ hashFiles('setup.py') }}
|
key: tox-integration-${{ matrix.test }}-${{ hashFiles('setup.py') }}
|
||||||
restore-keys: txo-integration-${{ matrix.test }}-
|
restore-keys: txo-integration-${{ matrix.test }}-
|
||||||
- run: pip install tox coverage coveralls
|
- run: pip install tox coverage coveralls
|
||||||
- if: matrix.test == 'claims'
|
|
||||||
run: rm -rf .tox
|
|
||||||
- run: tox -e ${{ matrix.test }}
|
- run: tox -e ${{ matrix.test }}
|
||||||
- name: submit coverage report
|
- name: submit coverage report
|
||||||
env:
|
env:
|
||||||
|
@ -123,7 +123,7 @@ jobs:
|
||||||
|
|
||||||
coverage:
|
coverage:
|
||||||
needs: ["tests-unit", "tests-integration"]
|
needs: ["tests-unit", "tests-integration"]
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: finalize coverage report submission
|
- name: finalize coverage report submission
|
||||||
env:
|
env:
|
||||||
|
@ -138,29 +138,29 @@ jobs:
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
os:
|
os:
|
||||||
- ubuntu-20.04
|
- ubuntu-18.04
|
||||||
- macos-latest
|
- macos-latest
|
||||||
- windows-latest
|
- windows-latest
|
||||||
runs-on: ${{ matrix.os }}
|
runs-on: ${{ matrix.os }}
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v2
|
||||||
- uses: actions/setup-python@v4
|
- uses: actions/setup-python@v1
|
||||||
with:
|
with:
|
||||||
python-version: '3.9'
|
python-version: '3.7'
|
||||||
- id: os-name
|
- id: os-name
|
||||||
uses: ASzc/change-string-case-action@v5
|
uses: ASzc/change-string-case-action@v1
|
||||||
with:
|
with:
|
||||||
string: ${{ runner.os }}
|
string: ${{ runner.os }}
|
||||||
- name: set pip cache dir
|
- name: set pip cache dir
|
||||||
shell: bash
|
id: pip-cache
|
||||||
run: echo "PIP_CACHE_DIR=$(pip cache dir)" >> $GITHUB_ENV
|
run: echo "::set-output name=dir::$(pip cache dir)"
|
||||||
- name: extract pip cache
|
- name: extract pip cache
|
||||||
uses: actions/cache@v3
|
uses: actions/cache@v2
|
||||||
with:
|
with:
|
||||||
path: ${{ env.PIP_CACHE_DIR }}
|
path: ${{ steps.pip-cache.outputs.dir }}
|
||||||
key: ${{ runner.os }}-pip-${{ hashFiles('setup.py') }}
|
key: ${{ runner.os }}-pip-${{ hashFiles('setup.py') }}
|
||||||
restore-keys: ${{ runner.os }}-pip-
|
restore-keys: ${{ runner.os }}-pip-
|
||||||
- run: pip install pyinstaller==4.6
|
- run: pip install pyinstaller==4.4
|
||||||
- run: pip install -e .
|
- run: pip install -e .
|
||||||
- if: startsWith(github.ref, 'refs/tags/v')
|
- if: startsWith(github.ref, 'refs/tags/v')
|
||||||
run: python docker/set_build.py
|
run: python docker/set_build.py
|
||||||
|
@ -175,7 +175,7 @@ jobs:
|
||||||
pip install pywin32==301
|
pip install pywin32==301
|
||||||
pyinstaller --additional-hooks-dir=scripts/. --icon=icons/lbry256.ico --onefile --name lbrynet lbry/extras/cli.py
|
pyinstaller --additional-hooks-dir=scripts/. --icon=icons/lbry256.ico --onefile --name lbrynet lbry/extras/cli.py
|
||||||
dist/lbrynet.exe --version
|
dist/lbrynet.exe --version
|
||||||
- uses: actions/upload-artifact@v3
|
- uses: actions/upload-artifact@v2
|
||||||
with:
|
with:
|
||||||
name: lbrynet-${{ steps.os-name.outputs.lowercase }}
|
name: lbrynet-${{ steps.os-name.outputs.lowercase }}
|
||||||
path: dist/
|
path: dist/
|
||||||
|
@ -184,7 +184,7 @@ jobs:
|
||||||
name: "release"
|
name: "release"
|
||||||
if: startsWith(github.ref, 'refs/tags/v')
|
if: startsWith(github.ref, 'refs/tags/v')
|
||||||
needs: ["build"]
|
needs: ["build"]
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v1
|
- uses: actions/checkout@v1
|
||||||
- uses: actions/download-artifact@v2
|
- uses: actions/download-artifact@v2
|
||||||
|
|
2
.github/workflows/release.yml
vendored
2
.github/workflows/release.yml
vendored
|
@ -7,7 +7,7 @@ on:
|
||||||
jobs:
|
jobs:
|
||||||
release:
|
release:
|
||||||
name: "slack notification"
|
name: "slack notification"
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: LoveToKnow/slackify-markdown-action@v1.0.0
|
- uses: LoveToKnow/slackify-markdown-action@v1.0.0
|
||||||
id: markdown
|
id: markdown
|
||||||
|
|
2
LICENSE
2
LICENSE
|
@ -1,6 +1,6 @@
|
||||||
The MIT License (MIT)
|
The MIT License (MIT)
|
||||||
|
|
||||||
Copyright (c) 2015-2022 LBRY Inc
|
Copyright (c) 2015-2020 LBRY Inc
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the
|
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the
|
||||||
"Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish,
|
"Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
|
|
@ -2,7 +2,6 @@ FROM debian:10-slim
|
||||||
|
|
||||||
ARG user=lbry
|
ARG user=lbry
|
||||||
ARG projects_dir=/home/$user
|
ARG projects_dir=/home/$user
|
||||||
ARG db_dir=/database
|
|
||||||
|
|
||||||
ARG DOCKER_TAG
|
ARG DOCKER_TAG
|
||||||
ARG DOCKER_COMMIT=docker
|
ARG DOCKER_COMMIT=docker
|
||||||
|
@ -28,16 +27,12 @@ RUN groupadd -g 999 $user && useradd -m -u 999 -g $user $user
|
||||||
|
|
||||||
COPY . $projects_dir
|
COPY . $projects_dir
|
||||||
RUN chown -R $user:$user $projects_dir
|
RUN chown -R $user:$user $projects_dir
|
||||||
RUN mkdir -p $db_dir
|
|
||||||
RUN chown -R $user:$user $db_dir
|
|
||||||
|
|
||||||
USER $user
|
USER $user
|
||||||
WORKDIR $projects_dir
|
WORKDIR $projects_dir
|
||||||
|
|
||||||
RUN python3 -m pip install -U setuptools pip
|
|
||||||
RUN make install
|
RUN make install
|
||||||
RUN python3 docker/set_build.py
|
RUN python3 docker/set_build.py
|
||||||
RUN rm ~/.cache -rf
|
RUN rm ~/.cache -rf
|
||||||
VOLUME $db_dir
|
|
||||||
ENTRYPOINT ["python3", "scripts/dht_node.py"]
|
ENTRYPOINT ["python3", "scripts/dht_node.py"]
|
||||||
|
|
||||||
|
|
|
@ -20,7 +20,6 @@ RUN apt-get update && \
|
||||||
python3-dev \
|
python3-dev \
|
||||||
python3-pip \
|
python3-pip \
|
||||||
python3-wheel \
|
python3-wheel \
|
||||||
python3-cffi \
|
|
||||||
python3-setuptools && \
|
python3-setuptools && \
|
||||||
update-alternatives --install /usr/bin/pip pip /usr/bin/pip3 1 && \
|
update-alternatives --install /usr/bin/pip pip /usr/bin/pip3 1 && \
|
||||||
rm -rf /var/lib/apt/lists/*
|
rm -rf /var/lib/apt/lists/*
|
||||||
|
|
|
@ -18,27 +18,23 @@ services:
|
||||||
- "wallet_server:/database"
|
- "wallet_server:/database"
|
||||||
environment:
|
environment:
|
||||||
- DAEMON_URL=http://lbry:lbry@127.0.0.1:9245
|
- DAEMON_URL=http://lbry:lbry@127.0.0.1:9245
|
||||||
- MAX_QUERY_WORKERS=4
|
|
||||||
- CACHE_MB=1024
|
|
||||||
- CACHE_ALL_TX_HASHES=
|
|
||||||
- CACHE_ALL_CLAIM_TXOS=
|
|
||||||
- MAX_SEND=1000000000000000000
|
|
||||||
- MAX_RECEIVE=1000000000000000000
|
|
||||||
- MAX_SESSIONS=100000
|
|
||||||
- HOST=0.0.0.0
|
|
||||||
- TCP_PORT=50001
|
- TCP_PORT=50001
|
||||||
- PROMETHEUS_PORT=2112
|
- PROMETHEUS_PORT=2112
|
||||||
|
- QUERY_TIMEOUT_MS=3000 # how long search queries allowed to run before cancelling, in milliseconds
|
||||||
|
- TRENDING_ALGORITHMS=variable_decay
|
||||||
|
- MAX_SEND=10000000000000 # deprecated. leave it high until its removed
|
||||||
|
- MAX_SUBS=1000000000000 # deprecated. leave it high until its removed
|
||||||
- FILTERING_CHANNEL_IDS=770bd7ecba84fd2f7607fb15aedd2b172c2e153f 95e5db68a3101df19763f3a5182e4b12ba393ee8
|
- FILTERING_CHANNEL_IDS=770bd7ecba84fd2f7607fb15aedd2b172c2e153f 95e5db68a3101df19763f3a5182e4b12ba393ee8
|
||||||
- BLOCKING_CHANNEL_IDS=dd687b357950f6f271999971f43c785e8067c3a9 06871aa438032244202840ec59a469b303257cad b4a2528f436eca1bf3bf3e10ff3f98c57bd6c4c6
|
- BLOCKING_CHANNEL_IDS=dd687b357950f6f271999971f43c785e8067c3a9 06871aa438032244202840ec59a469b303257cad b4a2528f436eca1bf3bf3e10ff3f98c57bd6c4c6 e4e230b131082f6b10c8f7994bbb83f29e8e6fb9
|
||||||
es01:
|
es01:
|
||||||
image: docker.elastic.co/elasticsearch/elasticsearch:7.11.0
|
image: docker.elastic.co/elasticsearch/elasticsearch:7.11.0
|
||||||
container_name: es01
|
container_name: es01
|
||||||
environment:
|
environment:
|
||||||
- node.name=es01
|
- node.name=es01
|
||||||
- discovery.type=single-node
|
- discovery.type=single-node
|
||||||
- indices.query.bool.max_clause_count=8192
|
- indices.query.bool.max_clause_count=4096
|
||||||
- bootstrap.memory_lock=true
|
- bootstrap.memory_lock=true
|
||||||
- "ES_JAVA_OPTS=-Xms4g -Xmx4g" # no more than 32, remember to disable swap
|
- "ES_JAVA_OPTS=-Xms8g -Xmx8g" # no more than 32, remember to disable swap
|
||||||
ulimits:
|
ulimits:
|
||||||
memlock:
|
memlock:
|
||||||
soft: -1
|
soft: -1
|
||||||
|
|
|
@ -6,7 +6,7 @@ set -euo pipefail
|
||||||
|
|
||||||
SNAPSHOT_URL="${SNAPSHOT_URL:-}" #off by default. latest snapshot at https://lbry.com/snapshot/wallet
|
SNAPSHOT_URL="${SNAPSHOT_URL:-}" #off by default. latest snapshot at https://lbry.com/snapshot/wallet
|
||||||
|
|
||||||
if [[ -n "$SNAPSHOT_URL" ]] && [[ ! -f /database/lbry-leveldb ]]; then
|
if [[ -n "$SNAPSHOT_URL" ]] && [[ ! -f /database/claims.db ]]; then
|
||||||
files="$(ls)"
|
files="$(ls)"
|
||||||
echo "Downloading wallet snapshot from $SNAPSHOT_URL"
|
echo "Downloading wallet snapshot from $SNAPSHOT_URL"
|
||||||
wget --no-verbose --trust-server-names --content-disposition "$SNAPSHOT_URL"
|
wget --no-verbose --trust-server-names --content-disposition "$SNAPSHOT_URL"
|
||||||
|
@ -20,6 +20,6 @@ if [[ -n "$SNAPSHOT_URL" ]] && [[ ! -f /database/lbry-leveldb ]]; then
|
||||||
rm "$filename"
|
rm "$filename"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
/home/lbry/.local/bin/lbry-hub-elastic-sync
|
/home/lbry/.local/bin/lbry-hub-elastic-sync /database/claims.db
|
||||||
echo 'starting server'
|
echo 'starting server'
|
||||||
/home/lbry/.local/bin/lbry-hub "$@"
|
/home/lbry/.local/bin/lbry-hub "$@"
|
||||||
|
|
348
docs/api.json
348
docs/api.json
File diff suppressed because one or more lines are too long
|
@ -1,2 +1,2 @@
|
||||||
__version__ = "0.113.0"
|
__version__ = "0.105.0"
|
||||||
version = tuple(map(int, __version__.split('.'))) # pylint: disable=invalid-name
|
version = tuple(map(int, __version__.split('.'))) # pylint: disable=invalid-name
|
||||||
|
|
|
@ -87,8 +87,8 @@ class AbstractBlob:
|
||||||
self.blob_completed_callback = blob_completed_callback
|
self.blob_completed_callback = blob_completed_callback
|
||||||
self.blob_directory = blob_directory
|
self.blob_directory = blob_directory
|
||||||
self.writers: typing.Dict[typing.Tuple[typing.Optional[str], typing.Optional[int]], HashBlobWriter] = {}
|
self.writers: typing.Dict[typing.Tuple[typing.Optional[str], typing.Optional[int]], HashBlobWriter] = {}
|
||||||
self.verified: asyncio.Event = asyncio.Event()
|
self.verified: asyncio.Event = asyncio.Event(loop=self.loop)
|
||||||
self.writing: asyncio.Event = asyncio.Event()
|
self.writing: asyncio.Event = asyncio.Event(loop=self.loop)
|
||||||
self.readers: typing.List[typing.BinaryIO] = []
|
self.readers: typing.List[typing.BinaryIO] = []
|
||||||
self.added_on = added_on or time.time()
|
self.added_on = added_on or time.time()
|
||||||
self.is_mine = is_mine
|
self.is_mine = is_mine
|
||||||
|
@ -201,7 +201,7 @@ class AbstractBlob:
|
||||||
writer = blob.get_blob_writer()
|
writer = blob.get_blob_writer()
|
||||||
writer.write(blob_bytes)
|
writer.write(blob_bytes)
|
||||||
await blob.verified.wait()
|
await blob.verified.wait()
|
||||||
return BlobInfo(blob_num, length, binascii.hexlify(iv).decode(), added_on, blob_hash, is_mine)
|
return BlobInfo(blob_num, length, binascii.hexlify(iv).decode(), blob_hash, added_on, is_mine)
|
||||||
|
|
||||||
def save_verified_blob(self, verified_bytes: bytes):
|
def save_verified_blob(self, verified_bytes: bytes):
|
||||||
if self.verified.is_set():
|
if self.verified.is_set():
|
||||||
|
@ -222,7 +222,7 @@ class AbstractBlob:
|
||||||
peer_port: typing.Optional[int] = None) -> HashBlobWriter:
|
peer_port: typing.Optional[int] = None) -> HashBlobWriter:
|
||||||
if (peer_address, peer_port) in self.writers and not self.writers[(peer_address, peer_port)].closed():
|
if (peer_address, peer_port) in self.writers and not self.writers[(peer_address, peer_port)].closed():
|
||||||
raise OSError(f"attempted to download blob twice from {peer_address}:{peer_port}")
|
raise OSError(f"attempted to download blob twice from {peer_address}:{peer_port}")
|
||||||
fut = asyncio.Future()
|
fut = asyncio.Future(loop=self.loop)
|
||||||
writer = HashBlobWriter(self.blob_hash, self.get_length, fut)
|
writer = HashBlobWriter(self.blob_hash, self.get_length, fut)
|
||||||
self.writers[(peer_address, peer_port)] = writer
|
self.writers[(peer_address, peer_port)] = writer
|
||||||
|
|
||||||
|
|
|
@ -12,8 +12,8 @@ class BlobInfo:
|
||||||
]
|
]
|
||||||
|
|
||||||
def __init__(
|
def __init__(
|
||||||
self, blob_num: int, length: int, iv: str, added_on,
|
self, blob_num: int, length: int, iv: str,
|
||||||
blob_hash: typing.Optional[str] = None, is_mine=False):
|
blob_hash: typing.Optional[str] = None, added_on=0, is_mine=False):
|
||||||
self.blob_hash = blob_hash
|
self.blob_hash = blob_hash
|
||||||
self.blob_num = blob_num
|
self.blob_num = blob_num
|
||||||
self.length = length
|
self.length = length
|
||||||
|
|
|
@ -83,8 +83,6 @@ class BlobManager:
|
||||||
to_add = await self.storage.sync_missing_blobs(in_blobfiles_dir)
|
to_add = await self.storage.sync_missing_blobs(in_blobfiles_dir)
|
||||||
if to_add:
|
if to_add:
|
||||||
self.completed_blob_hashes.update(to_add)
|
self.completed_blob_hashes.update(to_add)
|
||||||
# check blobs that aren't set as finished but were seen on disk
|
|
||||||
await self.ensure_completed_blobs_status(in_blobfiles_dir - to_add)
|
|
||||||
if self.config.track_bandwidth:
|
if self.config.track_bandwidth:
|
||||||
self.connection_manager.start()
|
self.connection_manager.start()
|
||||||
return True
|
return True
|
||||||
|
@ -115,18 +113,9 @@ class BlobManager:
|
||||||
(blob.blob_hash, blob.length, blob.added_on, blob.is_mine), finished=False)
|
(blob.blob_hash, blob.length, blob.added_on, blob.is_mine), finished=False)
|
||||||
)
|
)
|
||||||
|
|
||||||
async def ensure_completed_blobs_status(self, blob_hashes: typing.Iterable[str]):
|
def check_completed_blobs(self, blob_hashes: typing.List[str]) -> typing.List[str]:
|
||||||
"""Ensures that completed blobs from a given list of blob hashes are set as 'finished' in the database."""
|
"""Returns of the blobhashes_to_check, which are valid"""
|
||||||
to_add = []
|
return [blob_hash for blob_hash in blob_hashes if self.is_blob_verified(blob_hash)]
|
||||||
for blob_hash in blob_hashes:
|
|
||||||
if not self.is_blob_verified(blob_hash):
|
|
||||||
continue
|
|
||||||
blob = self.get_blob(blob_hash)
|
|
||||||
to_add.append((blob.blob_hash, blob.length, blob.added_on, blob.is_mine))
|
|
||||||
if len(to_add) > 500:
|
|
||||||
await self.storage.add_blobs(*to_add, finished=True)
|
|
||||||
to_add.clear()
|
|
||||||
return await self.storage.add_blobs(*to_add, finished=True)
|
|
||||||
|
|
||||||
def delete_blob(self, blob_hash: str):
|
def delete_blob(self, blob_hash: str):
|
||||||
if not is_valid_blobhash(blob_hash):
|
if not is_valid_blobhash(blob_hash):
|
||||||
|
|
|
@ -14,51 +14,32 @@ class DiskSpaceManager:
|
||||||
self.running = False
|
self.running = False
|
||||||
self.task = None
|
self.task = None
|
||||||
self.analytics = analytics
|
self.analytics = analytics
|
||||||
self._used_space_bytes = None
|
|
||||||
|
|
||||||
async def get_free_space_mb(self, is_network_blob=False):
|
|
||||||
limit_mb = self.config.network_storage_limit if is_network_blob else self.config.blob_storage_limit
|
|
||||||
space_used_mb = await self.get_space_used_mb()
|
|
||||||
space_used_mb = space_used_mb['network_storage'] if is_network_blob else space_used_mb['content_storage']
|
|
||||||
return max(0, limit_mb - space_used_mb)
|
|
||||||
|
|
||||||
async def get_space_used_bytes(self):
|
async def get_space_used_bytes(self):
|
||||||
self._used_space_bytes = await self.db.get_stored_blob_disk_usage()
|
return await self.db.get_stored_blob_disk_usage()
|
||||||
return self._used_space_bytes
|
|
||||||
|
|
||||||
async def get_space_used_mb(self, cached=True):
|
async def get_space_used_mb(self):
|
||||||
cached = cached and self._used_space_bytes is not None
|
return int(await self.get_space_used_bytes()/1024.0/1024.0)
|
||||||
space_used_bytes = self._used_space_bytes if cached else await self.get_space_used_bytes()
|
|
||||||
return {key: int(value/1024.0/1024.0) for key, value in space_used_bytes.items()}
|
|
||||||
|
|
||||||
async def clean(self):
|
async def clean(self):
|
||||||
await self._clean(False)
|
space_used_bytes = await self.get_space_used_bytes()
|
||||||
await self._clean(True)
|
storage_limit = self.config.blob_storage_limit*1024*1024 if self.config.blob_storage_limit else None
|
||||||
|
|
||||||
async def _clean(self, is_network_blob=False):
|
|
||||||
space_used_mb = await self.get_space_used_mb(cached=False)
|
|
||||||
if is_network_blob:
|
|
||||||
space_used_mb = space_used_mb['network_storage']
|
|
||||||
else:
|
|
||||||
space_used_mb = space_used_mb['content_storage'] + space_used_mb['private_storage']
|
|
||||||
storage_limit_mb = self.config.network_storage_limit if is_network_blob else self.config.blob_storage_limit
|
|
||||||
if self.analytics:
|
if self.analytics:
|
||||||
asyncio.create_task(
|
asyncio.create_task(self.analytics.send_disk_space_used(space_used_bytes, storage_limit))
|
||||||
self.analytics.send_disk_space_used(space_used_mb, storage_limit_mb, is_network_blob)
|
if not storage_limit:
|
||||||
)
|
|
||||||
delete = []
|
|
||||||
available = storage_limit_mb - space_used_mb
|
|
||||||
if storage_limit_mb == 0 if not is_network_blob else available >= 0:
|
|
||||||
return 0
|
return 0
|
||||||
for blob_hash, file_size, _ in await self.db.get_stored_blobs(is_mine=False, is_network_blob=is_network_blob):
|
delete = []
|
||||||
|
available = storage_limit - space_used_bytes
|
||||||
|
if available > 0:
|
||||||
|
return 0
|
||||||
|
for blob_hash, file_size, _ in await self.db.get_stored_blobs(is_mine=False):
|
||||||
delete.append(blob_hash)
|
delete.append(blob_hash)
|
||||||
available += int(file_size/1024.0/1024.0)
|
available += file_size
|
||||||
if available >= 0:
|
if available > 0:
|
||||||
break
|
break
|
||||||
if delete:
|
if delete:
|
||||||
await self.db.stop_all_files()
|
await self.db.stop_all_files()
|
||||||
await self.blob_manager.delete_blobs(delete, delete_from_db=True)
|
await self.blob_manager.delete_blobs(delete, delete_from_db=True)
|
||||||
self._used_space_bytes = None
|
|
||||||
return len(delete)
|
return len(delete)
|
||||||
|
|
||||||
async def cleaning_loop(self):
|
async def cleaning_loop(self):
|
||||||
|
|
|
@ -32,7 +32,7 @@ class BlobExchangeClientProtocol(asyncio.Protocol):
|
||||||
self.buf = b''
|
self.buf = b''
|
||||||
|
|
||||||
# this is here to handle the race when the downloader is closed right as response_fut gets a result
|
# this is here to handle the race when the downloader is closed right as response_fut gets a result
|
||||||
self.closed = asyncio.Event()
|
self.closed = asyncio.Event(loop=self.loop)
|
||||||
|
|
||||||
def data_received(self, data: bytes):
|
def data_received(self, data: bytes):
|
||||||
if self.connection_manager:
|
if self.connection_manager:
|
||||||
|
@ -111,7 +111,7 @@ class BlobExchangeClientProtocol(asyncio.Protocol):
|
||||||
self.transport.write(msg)
|
self.transport.write(msg)
|
||||||
if self.connection_manager:
|
if self.connection_manager:
|
||||||
self.connection_manager.sent_data(f"{self.peer_address}:{self.peer_port}", len(msg))
|
self.connection_manager.sent_data(f"{self.peer_address}:{self.peer_port}", len(msg))
|
||||||
response: BlobResponse = await asyncio.wait_for(self._response_fut, self.peer_timeout)
|
response: BlobResponse = await asyncio.wait_for(self._response_fut, self.peer_timeout, loop=self.loop)
|
||||||
availability_response = response.get_availability_response()
|
availability_response = response.get_availability_response()
|
||||||
price_response = response.get_price_response()
|
price_response = response.get_price_response()
|
||||||
blob_response = response.get_blob_response()
|
blob_response = response.get_blob_response()
|
||||||
|
@ -151,7 +151,7 @@ class BlobExchangeClientProtocol(asyncio.Protocol):
|
||||||
f" timeout in {self.peer_timeout}"
|
f" timeout in {self.peer_timeout}"
|
||||||
log.debug(msg)
|
log.debug(msg)
|
||||||
msg = f"downloaded {self.blob.blob_hash[:8]} from {self.peer_address}:{self.peer_port}"
|
msg = f"downloaded {self.blob.blob_hash[:8]} from {self.peer_address}:{self.peer_port}"
|
||||||
await asyncio.wait_for(self.writer.finished, self.peer_timeout)
|
await asyncio.wait_for(self.writer.finished, self.peer_timeout, loop=self.loop)
|
||||||
# wait for the io to finish
|
# wait for the io to finish
|
||||||
await self.blob.verified.wait()
|
await self.blob.verified.wait()
|
||||||
log.info("%s at %fMB/s", msg,
|
log.info("%s at %fMB/s", msg,
|
||||||
|
@ -187,7 +187,7 @@ class BlobExchangeClientProtocol(asyncio.Protocol):
|
||||||
try:
|
try:
|
||||||
self._blob_bytes_received = 0
|
self._blob_bytes_received = 0
|
||||||
self.blob, self.writer = blob, blob.get_blob_writer(self.peer_address, self.peer_port)
|
self.blob, self.writer = blob, blob.get_blob_writer(self.peer_address, self.peer_port)
|
||||||
self._response_fut = asyncio.Future()
|
self._response_fut = asyncio.Future(loop=self.loop)
|
||||||
return await self._download_blob()
|
return await self._download_blob()
|
||||||
except OSError:
|
except OSError:
|
||||||
# i'm not sure how to fix this race condition - jack
|
# i'm not sure how to fix this race condition - jack
|
||||||
|
@ -244,7 +244,7 @@ async def request_blob(loop: asyncio.AbstractEventLoop, blob: Optional['Abstract
|
||||||
try:
|
try:
|
||||||
if not connected_protocol:
|
if not connected_protocol:
|
||||||
await asyncio.wait_for(loop.create_connection(lambda: protocol, address, tcp_port),
|
await asyncio.wait_for(loop.create_connection(lambda: protocol, address, tcp_port),
|
||||||
peer_connect_timeout)
|
peer_connect_timeout, loop=loop)
|
||||||
connected_protocol = protocol
|
connected_protocol = protocol
|
||||||
if blob is None or blob.get_is_verified() or not blob.is_writeable():
|
if blob is None or blob.get_is_verified() or not blob.is_writeable():
|
||||||
# blob is None happens when we are just opening a connection
|
# blob is None happens when we are just opening a connection
|
||||||
|
|
|
@ -3,7 +3,6 @@ import typing
|
||||||
import logging
|
import logging
|
||||||
from lbry.utils import cache_concurrent
|
from lbry.utils import cache_concurrent
|
||||||
from lbry.blob_exchange.client import request_blob
|
from lbry.blob_exchange.client import request_blob
|
||||||
from lbry.dht.node import get_kademlia_peers_from_hosts
|
|
||||||
if typing.TYPE_CHECKING:
|
if typing.TYPE_CHECKING:
|
||||||
from lbry.conf import Config
|
from lbry.conf import Config
|
||||||
from lbry.dht.node import Node
|
from lbry.dht.node import Node
|
||||||
|
@ -30,7 +29,7 @@ class BlobDownloader:
|
||||||
self.failures: typing.Dict['KademliaPeer', int] = {}
|
self.failures: typing.Dict['KademliaPeer', int] = {}
|
||||||
self.connection_failures: typing.Set['KademliaPeer'] = set()
|
self.connection_failures: typing.Set['KademliaPeer'] = set()
|
||||||
self.connections: typing.Dict['KademliaPeer', 'BlobExchangeClientProtocol'] = {}
|
self.connections: typing.Dict['KademliaPeer', 'BlobExchangeClientProtocol'] = {}
|
||||||
self.is_running = asyncio.Event()
|
self.is_running = asyncio.Event(loop=self.loop)
|
||||||
|
|
||||||
def should_race_continue(self, blob: 'AbstractBlob'):
|
def should_race_continue(self, blob: 'AbstractBlob'):
|
||||||
max_probes = self.config.max_connections_per_download * (1 if self.connections else 10)
|
max_probes = self.config.max_connections_per_download * (1 if self.connections else 10)
|
||||||
|
@ -64,8 +63,8 @@ class BlobDownloader:
|
||||||
self.scores[peer] = bytes_received / elapsed if bytes_received and elapsed else 1
|
self.scores[peer] = bytes_received / elapsed if bytes_received and elapsed else 1
|
||||||
|
|
||||||
async def new_peer_or_finished(self):
|
async def new_peer_or_finished(self):
|
||||||
active_tasks = list(self.active_connections.values()) + [asyncio.create_task(asyncio.sleep(1))]
|
active_tasks = list(self.active_connections.values()) + [asyncio.sleep(1)]
|
||||||
await asyncio.wait(active_tasks, return_when='FIRST_COMPLETED')
|
await asyncio.wait(active_tasks, loop=self.loop, return_when='FIRST_COMPLETED')
|
||||||
|
|
||||||
def cleanup_active(self):
|
def cleanup_active(self):
|
||||||
if not self.active_connections and not self.connections:
|
if not self.active_connections and not self.connections:
|
||||||
|
@ -88,6 +87,7 @@ class BlobDownloader:
|
||||||
if blob.get_is_verified():
|
if blob.get_is_verified():
|
||||||
return blob
|
return blob
|
||||||
self.is_running.set()
|
self.is_running.set()
|
||||||
|
tried_for_this_blob: typing.Set['KademliaPeer'] = set()
|
||||||
try:
|
try:
|
||||||
while not blob.get_is_verified() and self.is_running.is_set():
|
while not blob.get_is_verified() and self.is_running.is_set():
|
||||||
batch: typing.Set['KademliaPeer'] = set(self.connections.keys())
|
batch: typing.Set['KademliaPeer'] = set(self.connections.keys())
|
||||||
|
@ -97,14 +97,23 @@ class BlobDownloader:
|
||||||
"%s running, %d peers, %d ignored, %d active, %s connections", blob_hash[:6],
|
"%s running, %d peers, %d ignored, %d active, %s connections", blob_hash[:6],
|
||||||
len(batch), len(self.ignored), len(self.active_connections), len(self.connections)
|
len(batch), len(self.ignored), len(self.active_connections), len(self.connections)
|
||||||
)
|
)
|
||||||
|
re_add: typing.Set['KademliaPeer'] = set()
|
||||||
for peer in sorted(batch, key=lambda peer: self.scores.get(peer, 0), reverse=True):
|
for peer in sorted(batch, key=lambda peer: self.scores.get(peer, 0), reverse=True):
|
||||||
if peer in self.ignored:
|
if peer in self.ignored:
|
||||||
continue
|
continue
|
||||||
if peer in self.active_connections or not self.should_race_continue(blob):
|
if peer in tried_for_this_blob:
|
||||||
continue
|
continue
|
||||||
|
if peer in self.active_connections:
|
||||||
|
if peer not in re_add:
|
||||||
|
re_add.add(peer)
|
||||||
|
continue
|
||||||
|
if not self.should_race_continue(blob):
|
||||||
|
break
|
||||||
log.debug("request %s from %s:%i", blob_hash[:8], peer.address, peer.tcp_port)
|
log.debug("request %s from %s:%i", blob_hash[:8], peer.address, peer.tcp_port)
|
||||||
t = self.loop.create_task(self.request_blob_from_peer(blob, peer, connection_id))
|
t = self.loop.create_task(self.request_blob_from_peer(blob, peer, connection_id))
|
||||||
self.active_connections[peer] = t
|
self.active_connections[peer] = t
|
||||||
|
tried_for_this_blob.add(peer)
|
||||||
|
if not re_add:
|
||||||
self.peer_queue.put_nowait(list(batch))
|
self.peer_queue.put_nowait(list(batch))
|
||||||
await self.new_peer_or_finished()
|
await self.new_peer_or_finished()
|
||||||
self.cleanup_active()
|
self.cleanup_active()
|
||||||
|
@ -124,14 +133,11 @@ class BlobDownloader:
|
||||||
protocol.close()
|
protocol.close()
|
||||||
|
|
||||||
|
|
||||||
async def download_blob(loop, config: 'Config', blob_manager: 'BlobManager', dht_node: 'Node',
|
async def download_blob(loop, config: 'Config', blob_manager: 'BlobManager', node: 'Node',
|
||||||
blob_hash: str) -> 'AbstractBlob':
|
blob_hash: str) -> 'AbstractBlob':
|
||||||
search_queue = asyncio.Queue(maxsize=config.max_connections_per_download)
|
search_queue = asyncio.Queue(loop=loop, maxsize=config.max_connections_per_download)
|
||||||
search_queue.put_nowait(blob_hash)
|
search_queue.put_nowait(blob_hash)
|
||||||
peer_queue, accumulate_task = dht_node.accumulate_peers(search_queue)
|
peer_queue, accumulate_task = node.accumulate_peers(search_queue)
|
||||||
fixed_peers = None if not config.fixed_peers else await get_kademlia_peers_from_hosts(config.fixed_peers)
|
|
||||||
if fixed_peers:
|
|
||||||
loop.call_later(config.fixed_peer_delay, peer_queue.put_nowait, fixed_peers)
|
|
||||||
downloader = BlobDownloader(loop, config, blob_manager, peer_queue)
|
downloader = BlobDownloader(loop, config, blob_manager, peer_queue)
|
||||||
try:
|
try:
|
||||||
return await downloader.download_blob(blob_hash)
|
return await downloader.download_blob(blob_hash)
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
import asyncio
|
import asyncio
|
||||||
import binascii
|
import binascii
|
||||||
import logging
|
import logging
|
||||||
import socket
|
|
||||||
import typing
|
import typing
|
||||||
from json.decoder import JSONDecodeError
|
from json.decoder import JSONDecodeError
|
||||||
from lbry.blob_exchange.serialization import BlobResponse, BlobRequest, blob_response_types
|
from lbry.blob_exchange.serialization import BlobResponse, BlobRequest, blob_response_types
|
||||||
|
@ -25,19 +24,19 @@ class BlobServerProtocol(asyncio.Protocol):
|
||||||
self.idle_timeout = idle_timeout
|
self.idle_timeout = idle_timeout
|
||||||
self.transfer_timeout = transfer_timeout
|
self.transfer_timeout = transfer_timeout
|
||||||
self.server_task: typing.Optional[asyncio.Task] = None
|
self.server_task: typing.Optional[asyncio.Task] = None
|
||||||
self.started_listening = asyncio.Event()
|
self.started_listening = asyncio.Event(loop=self.loop)
|
||||||
self.buf = b''
|
self.buf = b''
|
||||||
self.transport: typing.Optional[asyncio.Transport] = None
|
self.transport: typing.Optional[asyncio.Transport] = None
|
||||||
self.lbrycrd_address = lbrycrd_address
|
self.lbrycrd_address = lbrycrd_address
|
||||||
self.peer_address_and_port: typing.Optional[str] = None
|
self.peer_address_and_port: typing.Optional[str] = None
|
||||||
self.started_transfer = asyncio.Event()
|
self.started_transfer = asyncio.Event(loop=self.loop)
|
||||||
self.transfer_finished = asyncio.Event()
|
self.transfer_finished = asyncio.Event(loop=self.loop)
|
||||||
self.close_on_idle_task: typing.Optional[asyncio.Task] = None
|
self.close_on_idle_task: typing.Optional[asyncio.Task] = None
|
||||||
|
|
||||||
async def close_on_idle(self):
|
async def close_on_idle(self):
|
||||||
while self.transport:
|
while self.transport:
|
||||||
try:
|
try:
|
||||||
await asyncio.wait_for(self.started_transfer.wait(), self.idle_timeout)
|
await asyncio.wait_for(self.started_transfer.wait(), self.idle_timeout, loop=self.loop)
|
||||||
except asyncio.TimeoutError:
|
except asyncio.TimeoutError:
|
||||||
log.debug("closing idle connection from %s", self.peer_address_and_port)
|
log.debug("closing idle connection from %s", self.peer_address_and_port)
|
||||||
return self.close()
|
return self.close()
|
||||||
|
@ -101,7 +100,7 @@ class BlobServerProtocol(asyncio.Protocol):
|
||||||
log.debug("send %s to %s:%i", blob_hash, peer_address, peer_port)
|
log.debug("send %s to %s:%i", blob_hash, peer_address, peer_port)
|
||||||
self.started_transfer.set()
|
self.started_transfer.set()
|
||||||
try:
|
try:
|
||||||
sent = await asyncio.wait_for(blob.sendfile(self), self.transfer_timeout)
|
sent = await asyncio.wait_for(blob.sendfile(self), self.transfer_timeout, loop=self.loop)
|
||||||
if sent and sent > 0:
|
if sent and sent > 0:
|
||||||
self.blob_manager.connection_manager.sent_data(self.peer_address_and_port, sent)
|
self.blob_manager.connection_manager.sent_data(self.peer_address_and_port, sent)
|
||||||
log.info("sent %s (%i bytes) to %s:%i", blob_hash, sent, peer_address, peer_port)
|
log.info("sent %s (%i bytes) to %s:%i", blob_hash, sent, peer_address, peer_port)
|
||||||
|
@ -138,7 +137,7 @@ class BlobServerProtocol(asyncio.Protocol):
|
||||||
try:
|
try:
|
||||||
request = BlobRequest.deserialize(self.buf + data)
|
request = BlobRequest.deserialize(self.buf + data)
|
||||||
self.buf = remainder
|
self.buf = remainder
|
||||||
except (UnicodeDecodeError, JSONDecodeError):
|
except JSONDecodeError:
|
||||||
log.error("request from %s is not valid json (%i bytes): %s", self.peer_address_and_port,
|
log.error("request from %s is not valid json (%i bytes): %s", self.peer_address_and_port,
|
||||||
len(self.buf + data), '' if not data else binascii.hexlify(self.buf + data).decode())
|
len(self.buf + data), '' if not data else binascii.hexlify(self.buf + data).decode())
|
||||||
self.close()
|
self.close()
|
||||||
|
@ -157,7 +156,7 @@ class BlobServer:
|
||||||
self.loop = loop
|
self.loop = loop
|
||||||
self.blob_manager = blob_manager
|
self.blob_manager = blob_manager
|
||||||
self.server_task: typing.Optional[asyncio.Task] = None
|
self.server_task: typing.Optional[asyncio.Task] = None
|
||||||
self.started_listening = asyncio.Event()
|
self.started_listening = asyncio.Event(loop=self.loop)
|
||||||
self.lbrycrd_address = lbrycrd_address
|
self.lbrycrd_address = lbrycrd_address
|
||||||
self.idle_timeout = idle_timeout
|
self.idle_timeout = idle_timeout
|
||||||
self.transfer_timeout = transfer_timeout
|
self.transfer_timeout = transfer_timeout
|
||||||
|
@ -168,13 +167,6 @@ class BlobServer:
|
||||||
raise Exception("already running")
|
raise Exception("already running")
|
||||||
|
|
||||||
async def _start_server():
|
async def _start_server():
|
||||||
# checking if the port is in use
|
|
||||||
# thx https://stackoverflow.com/a/52872579
|
|
||||||
with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
|
|
||||||
if s.connect_ex(('localhost', port)) == 0:
|
|
||||||
# the port is already in use!
|
|
||||||
log.error("Failed to bind TCP %s:%d", interface, port)
|
|
||||||
|
|
||||||
server = await self.loop.create_server(
|
server = await self.loop.create_server(
|
||||||
lambda: self.server_protocol_class(self.loop, self.blob_manager, self.lbrycrd_address,
|
lambda: self.server_protocol_class(self.loop, self.blob_manager, self.lbrycrd_address,
|
||||||
self.idle_timeout, self.transfer_timeout),
|
self.idle_timeout, self.transfer_timeout),
|
||||||
|
|
28
lbry/conf.py
28
lbry/conf.py
|
@ -613,7 +613,7 @@ class Config(CLIConfig):
|
||||||
"ports or have firewall rules you likely want to disable this.", True
|
"ports or have firewall rules you likely want to disable this.", True
|
||||||
)
|
)
|
||||||
udp_port = Integer("UDP port for communicating on the LBRY DHT", 4444, previous_names=['dht_node_port'])
|
udp_port = Integer("UDP port for communicating on the LBRY DHT", 4444, previous_names=['dht_node_port'])
|
||||||
tcp_port = Integer("TCP port to listen for incoming blob requests", 4444, previous_names=['peer_port'])
|
tcp_port = Integer("TCP port to listen for incoming blob requests", 3333, previous_names=['peer_port'])
|
||||||
prometheus_port = Integer("Port to expose prometheus metrics (off by default)", 0)
|
prometheus_port = Integer("Port to expose prometheus metrics (off by default)", 0)
|
||||||
network_interface = String("Interface to use for the DHT and blob exchange", '0.0.0.0')
|
network_interface = String("Interface to use for the DHT and blob exchange", '0.0.0.0')
|
||||||
|
|
||||||
|
@ -622,11 +622,7 @@ class Config(CLIConfig):
|
||||||
"Routing table bucket index below which we always split the bucket if given a new key to add to it and "
|
"Routing table bucket index below which we always split the bucket if given a new key to add to it and "
|
||||||
"the bucket is full. As this value is raised the depth of the routing table (and number of peers in it) "
|
"the bucket is full. As this value is raised the depth of the routing table (and number of peers in it) "
|
||||||
"will increase. This setting is used by seed nodes, you probably don't want to change it during normal "
|
"will increase. This setting is used by seed nodes, you probably don't want to change it during normal "
|
||||||
"use.", 2
|
"use.", 1
|
||||||
)
|
|
||||||
is_bootstrap_node = Toggle(
|
|
||||||
"When running as a bootstrap node, disable all logic related to balancing the routing table, so we can "
|
|
||||||
"add as many peers as possible and better help first-runs.", False
|
|
||||||
)
|
)
|
||||||
|
|
||||||
# protocol timeouts
|
# protocol timeouts
|
||||||
|
@ -638,7 +634,6 @@ class Config(CLIConfig):
|
||||||
|
|
||||||
# blob announcement and download
|
# blob announcement and download
|
||||||
save_blobs = Toggle("Save encrypted blob files for hosting, otherwise download blobs to memory only.", True)
|
save_blobs = Toggle("Save encrypted blob files for hosting, otherwise download blobs to memory only.", True)
|
||||||
network_storage_limit = Integer("Disk space in MB to be allocated for helping the P2P network. 0 = disable", 0)
|
|
||||||
blob_storage_limit = Integer("Disk space in MB to be allocated for blob storage. 0 = no limit", 0)
|
blob_storage_limit = Integer("Disk space in MB to be allocated for blob storage. 0 = no limit", 0)
|
||||||
blob_lru_cache_size = Integer(
|
blob_lru_cache_size = Integer(
|
||||||
"LRU cache size for decrypted downloaded blobs used to minimize re-downloading the same blobs when "
|
"LRU cache size for decrypted downloaded blobs used to minimize re-downloading the same blobs when "
|
||||||
|
@ -685,14 +680,6 @@ class Config(CLIConfig):
|
||||||
('cdn.reflector.lbry.com', 5567)
|
('cdn.reflector.lbry.com', 5567)
|
||||||
])
|
])
|
||||||
|
|
||||||
tracker_servers = Servers("BitTorrent-compatible (BEP15) UDP trackers for helping P2P discovery", [
|
|
||||||
('tracker.lbry.com', 9252),
|
|
||||||
('tracker.lbry.grin.io', 9252),
|
|
||||||
('tracker.lbry.pigg.es', 9252),
|
|
||||||
('tracker.lizard.technology', 9252),
|
|
||||||
('s1.lbry.network', 9252),
|
|
||||||
])
|
|
||||||
|
|
||||||
lbryum_servers = Servers("SPV wallet servers", [
|
lbryum_servers = Servers("SPV wallet servers", [
|
||||||
('spv11.lbry.com', 50001),
|
('spv11.lbry.com', 50001),
|
||||||
('spv12.lbry.com', 50001),
|
('spv12.lbry.com', 50001),
|
||||||
|
@ -703,20 +690,14 @@ class Config(CLIConfig):
|
||||||
('spv17.lbry.com', 50001),
|
('spv17.lbry.com', 50001),
|
||||||
('spv18.lbry.com', 50001),
|
('spv18.lbry.com', 50001),
|
||||||
('spv19.lbry.com', 50001),
|
('spv19.lbry.com', 50001),
|
||||||
('hub.lbry.grin.io', 50001),
|
|
||||||
('hub.lizard.technology', 50001),
|
|
||||||
('s1.lbry.network', 50001),
|
|
||||||
])
|
])
|
||||||
known_dht_nodes = Servers("Known nodes for bootstrapping connection to the DHT", [
|
known_dht_nodes = Servers("Known nodes for bootstrapping connection to the DHT", [
|
||||||
('dht.lbry.grin.io', 4444), # Grin
|
('dht.lbry.grin.io', 4444), # Grin
|
||||||
('dht.lbry.madiator.com', 4444), # Madiator
|
('dht.lbry.madiator.com', 4444), # Madiator
|
||||||
('dht.lbry.pigg.es', 4444), # Pigges
|
|
||||||
('lbrynet1.lbry.com', 4444), # US EAST
|
('lbrynet1.lbry.com', 4444), # US EAST
|
||||||
('lbrynet2.lbry.com', 4444), # US WEST
|
('lbrynet2.lbry.com', 4444), # US WEST
|
||||||
('lbrynet3.lbry.com', 4444), # EU
|
('lbrynet3.lbry.com', 4444), # EU
|
||||||
('lbrynet4.lbry.com', 4444), # ASIA
|
('lbrynet4.lbry.com', 4444) # ASIA
|
||||||
('dht.lizard.technology', 4444), # Jack
|
|
||||||
('s2.lbry.network', 4444),
|
|
||||||
])
|
])
|
||||||
|
|
||||||
# blockchain
|
# blockchain
|
||||||
|
@ -742,8 +723,7 @@ class Config(CLIConfig):
|
||||||
|
|
||||||
coin_selection_strategy = StringChoice(
|
coin_selection_strategy = StringChoice(
|
||||||
"Strategy to use when selecting UTXOs for a transaction",
|
"Strategy to use when selecting UTXOs for a transaction",
|
||||||
STRATEGIES, "prefer_confirmed"
|
STRATEGIES, "standard")
|
||||||
)
|
|
||||||
|
|
||||||
transaction_cache_size = Integer("Transaction cache size", 2 ** 17)
|
transaction_cache_size = Integer("Transaction cache size", 2 ** 17)
|
||||||
save_resolved_claims = Toggle(
|
save_resolved_claims = Toggle(
|
||||||
|
|
|
@ -67,7 +67,7 @@ class ConnectionManager:
|
||||||
|
|
||||||
while True:
|
while True:
|
||||||
last = time.perf_counter()
|
last = time.perf_counter()
|
||||||
await asyncio.sleep(0.1)
|
await asyncio.sleep(0.1, loop=self.loop)
|
||||||
self._status['incoming_bps'].clear()
|
self._status['incoming_bps'].clear()
|
||||||
self._status['outgoing_bps'].clear()
|
self._status['outgoing_bps'].clear()
|
||||||
now = time.perf_counter()
|
now = time.perf_counter()
|
||||||
|
|
|
@ -1,9 +1,6 @@
|
||||||
import asyncio
|
import asyncio
|
||||||
import typing
|
import typing
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
from prometheus_client import Counter, Gauge
|
|
||||||
|
|
||||||
if typing.TYPE_CHECKING:
|
if typing.TYPE_CHECKING:
|
||||||
from lbry.dht.node import Node
|
from lbry.dht.node import Node
|
||||||
from lbry.extras.daemon.storage import SQLiteStorage
|
from lbry.extras.daemon.storage import SQLiteStorage
|
||||||
|
@ -12,59 +9,45 @@ log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class BlobAnnouncer:
|
class BlobAnnouncer:
|
||||||
announcements_sent_metric = Counter(
|
|
||||||
"announcements_sent", "Number of announcements sent and their respective status.", namespace="dht_node",
|
|
||||||
labelnames=("peers", "error"),
|
|
||||||
)
|
|
||||||
announcement_queue_size_metric = Gauge(
|
|
||||||
"announcement_queue_size", "Number of hashes waiting to be announced.", namespace="dht_node",
|
|
||||||
labelnames=("scope",)
|
|
||||||
)
|
|
||||||
|
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop, node: 'Node', storage: 'SQLiteStorage'):
|
def __init__(self, loop: asyncio.AbstractEventLoop, node: 'Node', storage: 'SQLiteStorage'):
|
||||||
self.loop = loop
|
self.loop = loop
|
||||||
self.node = node
|
self.node = node
|
||||||
self.storage = storage
|
self.storage = storage
|
||||||
self.announce_task: asyncio.Task = None
|
self.announce_task: asyncio.Task = None
|
||||||
self.announce_queue: typing.List[str] = []
|
self.announce_queue: typing.List[str] = []
|
||||||
self._done = asyncio.Event()
|
|
||||||
self.announced = set()
|
|
||||||
|
|
||||||
async def _run_consumer(self):
|
async def _submit_announcement(self, blob_hash):
|
||||||
while self.announce_queue:
|
|
||||||
try:
|
try:
|
||||||
blob_hash = self.announce_queue.pop()
|
|
||||||
peers = len(await self.node.announce_blob(blob_hash))
|
peers = len(await self.node.announce_blob(blob_hash))
|
||||||
self.announcements_sent_metric.labels(peers=peers, error=False).inc()
|
|
||||||
if peers > 4:
|
if peers > 4:
|
||||||
self.announced.add(blob_hash)
|
return blob_hash
|
||||||
else:
|
else:
|
||||||
log.debug("failed to announce %s, could only find %d peers, retrying soon.", blob_hash[:8], peers)
|
log.debug("failed to announce %s, could only find %d peers, retrying soon.", blob_hash[:8], peers)
|
||||||
except Exception as err:
|
except Exception as err:
|
||||||
self.announcements_sent_metric.labels(peers=0, error=True).inc()
|
if isinstance(err, asyncio.CancelledError): # TODO: remove when updated to 3.8
|
||||||
|
raise err
|
||||||
log.warning("error announcing %s: %s", blob_hash[:8], str(err))
|
log.warning("error announcing %s: %s", blob_hash[:8], str(err))
|
||||||
|
|
||||||
async def _announce(self, batch_size: typing.Optional[int] = 10):
|
async def _announce(self, batch_size: typing.Optional[int] = 10):
|
||||||
while batch_size:
|
while batch_size:
|
||||||
if not self.node.joined.is_set():
|
if not self.node.joined.is_set():
|
||||||
await self.node.joined.wait()
|
await self.node.joined.wait()
|
||||||
await asyncio.sleep(60)
|
await asyncio.sleep(60, loop=self.loop)
|
||||||
if not self.node.protocol.routing_table.get_peers():
|
if not self.node.protocol.routing_table.get_peers():
|
||||||
log.warning("No peers in DHT, announce round skipped")
|
log.warning("No peers in DHT, announce round skipped")
|
||||||
continue
|
continue
|
||||||
self.announce_queue.extend(await self.storage.get_blobs_to_announce())
|
self.announce_queue.extend(await self.storage.get_blobs_to_announce())
|
||||||
self.announcement_queue_size_metric.labels(scope="global").set(len(self.announce_queue))
|
|
||||||
log.debug("announcer task wake up, %d blobs to announce", len(self.announce_queue))
|
log.debug("announcer task wake up, %d blobs to announce", len(self.announce_queue))
|
||||||
while len(self.announce_queue) > 0:
|
while len(self.announce_queue) > 0:
|
||||||
log.info("%i blobs to announce", len(self.announce_queue))
|
log.info("%i blobs to announce", len(self.announce_queue))
|
||||||
await asyncio.gather(*[self._run_consumer() for _ in range(batch_size)])
|
announced = await asyncio.gather(*[
|
||||||
announced = list(filter(None, self.announced))
|
self._submit_announcement(
|
||||||
|
self.announce_queue.pop()) for _ in range(batch_size) if self.announce_queue
|
||||||
|
], loop=self.loop)
|
||||||
|
announced = list(filter(None, announced))
|
||||||
if announced:
|
if announced:
|
||||||
await self.storage.update_last_announced_blobs(announced)
|
await self.storage.update_last_announced_blobs(announced)
|
||||||
log.info("announced %i blobs", len(announced))
|
log.info("announced %i blobs", len(announced))
|
||||||
self.announced.clear()
|
|
||||||
self._done.set()
|
|
||||||
self._done.clear()
|
|
||||||
|
|
||||||
def start(self, batch_size: typing.Optional[int] = 10):
|
def start(self, batch_size: typing.Optional[int] = 10):
|
||||||
assert not self.announce_task or self.announce_task.done(), "already running"
|
assert not self.announce_task or self.announce_task.done(), "already running"
|
||||||
|
@ -73,6 +56,3 @@ class BlobAnnouncer:
|
||||||
def stop(self):
|
def stop(self):
|
||||||
if self.announce_task and not self.announce_task.done():
|
if self.announce_task and not self.announce_task.done():
|
||||||
self.announce_task.cancel()
|
self.announce_task.cancel()
|
||||||
|
|
||||||
def wait(self):
|
|
||||||
return self._done.wait()
|
|
||||||
|
|
|
@ -20,6 +20,7 @@ MAYBE_PING_DELAY = 300 # 5 minutes
|
||||||
CHECK_REFRESH_INTERVAL = REFRESH_INTERVAL / 5
|
CHECK_REFRESH_INTERVAL = REFRESH_INTERVAL / 5
|
||||||
RPC_ID_LENGTH = 20
|
RPC_ID_LENGTH = 20
|
||||||
PROTOCOL_VERSION = 1
|
PROTOCOL_VERSION = 1
|
||||||
|
BOTTOM_OUT_LIMIT = 3
|
||||||
MSG_SIZE_LIMIT = 1400
|
MSG_SIZE_LIMIT = 1400
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -2,10 +2,7 @@ import logging
|
||||||
import asyncio
|
import asyncio
|
||||||
import typing
|
import typing
|
||||||
import socket
|
import socket
|
||||||
|
from lbry.utils import resolve_host
|
||||||
from prometheus_client import Gauge
|
|
||||||
|
|
||||||
from lbry.utils import aclosing, resolve_host
|
|
||||||
from lbry.dht import constants
|
from lbry.dht import constants
|
||||||
from lbry.dht.peer import make_kademlia_peer
|
from lbry.dht.peer import make_kademlia_peer
|
||||||
from lbry.dht.protocol.distance import Distance
|
from lbry.dht.protocol.distance import Distance
|
||||||
|
@ -20,32 +17,20 @@ log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class Node:
|
class Node:
|
||||||
storing_peers_metric = Gauge(
|
|
||||||
"storing_peers", "Number of peers storing blobs announced to this node", namespace="dht_node",
|
|
||||||
labelnames=("scope",),
|
|
||||||
)
|
|
||||||
stored_blob_with_x_bytes_colliding = Gauge(
|
|
||||||
"stored_blobs_x_bytes_colliding", "Number of blobs with at least X bytes colliding with this node id prefix",
|
|
||||||
namespace="dht_node", labelnames=("amount",)
|
|
||||||
)
|
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop, peer_manager: 'PeerManager', node_id: bytes, udp_port: int,
|
def __init__(self, loop: asyncio.AbstractEventLoop, peer_manager: 'PeerManager', node_id: bytes, udp_port: int,
|
||||||
internal_udp_port: int, peer_port: int, external_ip: str, rpc_timeout: float = constants.RPC_TIMEOUT,
|
internal_udp_port: int, peer_port: int, external_ip: str, rpc_timeout: float = constants.RPC_TIMEOUT,
|
||||||
split_buckets_under_index: int = constants.SPLIT_BUCKETS_UNDER_INDEX, is_bootstrap_node: bool = False,
|
split_buckets_under_index: int = constants.SPLIT_BUCKETS_UNDER_INDEX,
|
||||||
storage: typing.Optional['SQLiteStorage'] = None):
|
storage: typing.Optional['SQLiteStorage'] = None):
|
||||||
self.loop = loop
|
self.loop = loop
|
||||||
self.internal_udp_port = internal_udp_port
|
self.internal_udp_port = internal_udp_port
|
||||||
self.protocol = KademliaProtocol(loop, peer_manager, node_id, external_ip, udp_port, peer_port, rpc_timeout,
|
self.protocol = KademliaProtocol(loop, peer_manager, node_id, external_ip, udp_port, peer_port, rpc_timeout,
|
||||||
split_buckets_under_index, is_bootstrap_node)
|
split_buckets_under_index)
|
||||||
self.listening_port: asyncio.DatagramTransport = None
|
self.listening_port: asyncio.DatagramTransport = None
|
||||||
self.joined = asyncio.Event()
|
self.joined = asyncio.Event(loop=self.loop)
|
||||||
self._join_task: asyncio.Task = None
|
self._join_task: asyncio.Task = None
|
||||||
self._refresh_task: asyncio.Task = None
|
self._refresh_task: asyncio.Task = None
|
||||||
self._storage = storage
|
self._storage = storage
|
||||||
|
|
||||||
@property
|
|
||||||
def stored_blob_hashes(self):
|
|
||||||
return self.protocol.data_store.keys()
|
|
||||||
|
|
||||||
async def refresh_node(self, force_once=False):
|
async def refresh_node(self, force_once=False):
|
||||||
while True:
|
while True:
|
||||||
# remove peers with expired blob announcements from the datastore
|
# remove peers with expired blob announcements from the datastore
|
||||||
|
@ -55,21 +40,17 @@ class Node:
|
||||||
# add all peers in the routing table
|
# add all peers in the routing table
|
||||||
total_peers.extend(self.protocol.routing_table.get_peers())
|
total_peers.extend(self.protocol.routing_table.get_peers())
|
||||||
# add all the peers who have announced blobs to us
|
# add all the peers who have announced blobs to us
|
||||||
storing_peers = self.protocol.data_store.get_storing_contacts()
|
total_peers.extend(self.protocol.data_store.get_storing_contacts())
|
||||||
self.storing_peers_metric.labels("global").set(len(storing_peers))
|
|
||||||
total_peers.extend(storing_peers)
|
|
||||||
|
|
||||||
counts = {0: 0, 1: 0, 2: 0}
|
|
||||||
node_id = self.protocol.node_id
|
|
||||||
for blob_hash in self.protocol.data_store.keys():
|
|
||||||
bytes_colliding = 0 if blob_hash[0] != node_id[0] else 2 if blob_hash[1] == node_id[1] else 1
|
|
||||||
counts[bytes_colliding] += 1
|
|
||||||
self.stored_blob_with_x_bytes_colliding.labels(amount=0).set(counts[0])
|
|
||||||
self.stored_blob_with_x_bytes_colliding.labels(amount=1).set(counts[1])
|
|
||||||
self.stored_blob_with_x_bytes_colliding.labels(amount=2).set(counts[2])
|
|
||||||
|
|
||||||
# get ids falling in the midpoint of each bucket that hasn't been recently updated
|
# get ids falling in the midpoint of each bucket that hasn't been recently updated
|
||||||
node_ids = self.protocol.routing_table.get_refresh_list(0, True)
|
node_ids = self.protocol.routing_table.get_refresh_list(0, True)
|
||||||
|
# if we have 3 or fewer populated buckets get two random ids in the range of each to try and
|
||||||
|
# populate/split the buckets further
|
||||||
|
buckets_with_contacts = self.protocol.routing_table.buckets_with_contacts()
|
||||||
|
if buckets_with_contacts <= 3:
|
||||||
|
for i in range(buckets_with_contacts):
|
||||||
|
node_ids.append(self.protocol.routing_table.random_id_in_bucket_range(i))
|
||||||
|
node_ids.append(self.protocol.routing_table.random_id_in_bucket_range(i))
|
||||||
|
|
||||||
if self.protocol.routing_table.get_peers():
|
if self.protocol.routing_table.get_peers():
|
||||||
# if we have node ids to look up, perform the iterative search until we have k results
|
# if we have node ids to look up, perform the iterative search until we have k results
|
||||||
|
@ -79,7 +60,7 @@ class Node:
|
||||||
else:
|
else:
|
||||||
if force_once:
|
if force_once:
|
||||||
break
|
break
|
||||||
fut = asyncio.Future()
|
fut = asyncio.Future(loop=self.loop)
|
||||||
self.loop.call_later(constants.REFRESH_INTERVAL // 4, fut.set_result, None)
|
self.loop.call_later(constants.REFRESH_INTERVAL // 4, fut.set_result, None)
|
||||||
await fut
|
await fut
|
||||||
continue
|
continue
|
||||||
|
@ -93,7 +74,7 @@ class Node:
|
||||||
if force_once:
|
if force_once:
|
||||||
break
|
break
|
||||||
|
|
||||||
fut = asyncio.Future()
|
fut = asyncio.Future(loop=self.loop)
|
||||||
self.loop.call_later(constants.REFRESH_INTERVAL, fut.set_result, None)
|
self.loop.call_later(constants.REFRESH_INTERVAL, fut.set_result, None)
|
||||||
await fut
|
await fut
|
||||||
|
|
||||||
|
@ -108,7 +89,7 @@ class Node:
|
||||||
for peer in peers:
|
for peer in peers:
|
||||||
log.debug("store to %s %s %s", peer.address, peer.udp_port, peer.tcp_port)
|
log.debug("store to %s %s %s", peer.address, peer.udp_port, peer.tcp_port)
|
||||||
stored_to_tup = await asyncio.gather(
|
stored_to_tup = await asyncio.gather(
|
||||||
*(self.protocol.store_to_peer(hash_value, peer) for peer in peers)
|
*(self.protocol.store_to_peer(hash_value, peer) for peer in peers), loop=self.loop
|
||||||
)
|
)
|
||||||
stored_to = [node_id for node_id, contacted in stored_to_tup if contacted]
|
stored_to = [node_id for node_id, contacted in stored_to_tup if contacted]
|
||||||
if stored_to:
|
if stored_to:
|
||||||
|
@ -182,35 +163,38 @@ class Node:
|
||||||
for address, udp_port in known_node_urls or []
|
for address, udp_port in known_node_urls or []
|
||||||
]))
|
]))
|
||||||
except socket.gaierror:
|
except socket.gaierror:
|
||||||
await asyncio.sleep(30)
|
await asyncio.sleep(30, loop=self.loop)
|
||||||
continue
|
continue
|
||||||
|
|
||||||
self.protocol.peer_manager.reset()
|
self.protocol.peer_manager.reset()
|
||||||
self.protocol.ping_queue.enqueue_maybe_ping(*seed_peers, delay=0.0)
|
self.protocol.ping_queue.enqueue_maybe_ping(*seed_peers, delay=0.0)
|
||||||
await self.peer_search(self.protocol.node_id, shortlist=seed_peers, count=32)
|
await self.peer_search(self.protocol.node_id, shortlist=seed_peers, count=32)
|
||||||
|
|
||||||
await asyncio.sleep(1)
|
await asyncio.sleep(1, loop=self.loop)
|
||||||
|
|
||||||
def start(self, interface: str, known_node_urls: typing.Optional[typing.List[typing.Tuple[str, int]]] = None):
|
def start(self, interface: str, known_node_urls: typing.Optional[typing.List[typing.Tuple[str, int]]] = None):
|
||||||
self._join_task = self.loop.create_task(self.join_network(interface, known_node_urls))
|
self._join_task = self.loop.create_task(self.join_network(interface, known_node_urls))
|
||||||
|
|
||||||
def get_iterative_node_finder(self, key: bytes, shortlist: typing.Optional[typing.List['KademliaPeer']] = None,
|
def get_iterative_node_finder(self, key: bytes, shortlist: typing.Optional[typing.List['KademliaPeer']] = None,
|
||||||
|
bottom_out_limit: int = constants.BOTTOM_OUT_LIMIT,
|
||||||
max_results: int = constants.K) -> IterativeNodeFinder:
|
max_results: int = constants.K) -> IterativeNodeFinder:
|
||||||
shortlist = shortlist or self.protocol.routing_table.find_close_peers(key)
|
|
||||||
return IterativeNodeFinder(self.loop, self.protocol, key, max_results, shortlist)
|
return IterativeNodeFinder(self.loop, self.protocol.peer_manager, self.protocol.routing_table, self.protocol,
|
||||||
|
key, bottom_out_limit, max_results, None, shortlist)
|
||||||
|
|
||||||
def get_iterative_value_finder(self, key: bytes, shortlist: typing.Optional[typing.List['KademliaPeer']] = None,
|
def get_iterative_value_finder(self, key: bytes, shortlist: typing.Optional[typing.List['KademliaPeer']] = None,
|
||||||
|
bottom_out_limit: int = 40,
|
||||||
max_results: int = -1) -> IterativeValueFinder:
|
max_results: int = -1) -> IterativeValueFinder:
|
||||||
shortlist = shortlist or self.protocol.routing_table.find_close_peers(key)
|
|
||||||
return IterativeValueFinder(self.loop, self.protocol, key, max_results, shortlist)
|
return IterativeValueFinder(self.loop, self.protocol.peer_manager, self.protocol.routing_table, self.protocol,
|
||||||
|
key, bottom_out_limit, max_results, None, shortlist)
|
||||||
|
|
||||||
async def peer_search(self, node_id: bytes, count=constants.K, max_results=constants.K * 2,
|
async def peer_search(self, node_id: bytes, count=constants.K, max_results=constants.K * 2,
|
||||||
shortlist: typing.Optional[typing.List['KademliaPeer']] = None
|
bottom_out_limit=20, shortlist: typing.Optional[typing.List['KademliaPeer']] = None
|
||||||
) -> typing.List['KademliaPeer']:
|
) -> typing.List['KademliaPeer']:
|
||||||
peers = []
|
peers = []
|
||||||
async with aclosing(self.get_iterative_node_finder(
|
async for iteration_peers in self.get_iterative_node_finder(
|
||||||
node_id, shortlist=shortlist, max_results=max_results)) as node_finder:
|
node_id, shortlist=shortlist, bottom_out_limit=bottom_out_limit, max_results=max_results):
|
||||||
async for iteration_peers in node_finder:
|
|
||||||
peers.extend(iteration_peers)
|
peers.extend(iteration_peers)
|
||||||
distance = Distance(node_id)
|
distance = Distance(node_id)
|
||||||
peers.sort(key=lambda peer: distance(peer.node_id))
|
peers.sort(key=lambda peer: distance(peer.node_id))
|
||||||
|
@ -237,8 +221,8 @@ class Node:
|
||||||
|
|
||||||
# prioritize peers who reply to a dht ping first
|
# prioritize peers who reply to a dht ping first
|
||||||
# this minimizes attempting to make tcp connections that won't work later to dead or unreachable peers
|
# this minimizes attempting to make tcp connections that won't work later to dead or unreachable peers
|
||||||
async with aclosing(self.get_iterative_value_finder(bytes.fromhex(blob_hash))) as value_finder:
|
|
||||||
async for results in value_finder:
|
async for results in self.get_iterative_value_finder(bytes.fromhex(blob_hash)):
|
||||||
to_put = []
|
to_put = []
|
||||||
for peer in results:
|
for peer in results:
|
||||||
if peer.address == self.protocol.external_ip and self.protocol.peer_port == peer.tcp_port:
|
if peer.address == self.protocol.external_ip and self.protocol.peer_port == peer.tcp_port:
|
||||||
|
@ -271,12 +255,5 @@ class Node:
|
||||||
def accumulate_peers(self, search_queue: asyncio.Queue,
|
def accumulate_peers(self, search_queue: asyncio.Queue,
|
||||||
peer_queue: typing.Optional[asyncio.Queue] = None
|
peer_queue: typing.Optional[asyncio.Queue] = None
|
||||||
) -> typing.Tuple[asyncio.Queue, asyncio.Task]:
|
) -> typing.Tuple[asyncio.Queue, asyncio.Task]:
|
||||||
queue = peer_queue or asyncio.Queue()
|
queue = peer_queue or asyncio.Queue(loop=self.loop)
|
||||||
return queue, self.loop.create_task(self._accumulate_peers_for_value(search_queue, queue))
|
return queue, self.loop.create_task(self._accumulate_peers_for_value(search_queue, queue))
|
||||||
|
|
||||||
|
|
||||||
async def get_kademlia_peers_from_hosts(peer_list: typing.List[typing.Tuple[str, int]]) -> typing.List['KademliaPeer']:
|
|
||||||
peer_address_list = [(await resolve_host(url, port, proto='tcp'), port) for url, port in peer_list]
|
|
||||||
kademlia_peer_list = [make_kademlia_peer(None, address, None, tcp_port=port, allow_localhost=True)
|
|
||||||
for address, port in peer_address_list]
|
|
||||||
return kademlia_peer_list
|
|
||||||
|
|
|
@ -3,19 +3,15 @@ import asyncio
|
||||||
import logging
|
import logging
|
||||||
from dataclasses import dataclass, field
|
from dataclasses import dataclass, field
|
||||||
from functools import lru_cache
|
from functools import lru_cache
|
||||||
|
|
||||||
from prometheus_client import Gauge
|
|
||||||
|
|
||||||
from lbry.utils import is_valid_public_ipv4 as _is_valid_public_ipv4, LRUCache
|
from lbry.utils import is_valid_public_ipv4 as _is_valid_public_ipv4, LRUCache
|
||||||
from lbry.dht import constants
|
from lbry.dht import constants
|
||||||
from lbry.dht.serialization.datagram import make_compact_address, make_compact_ip, decode_compact_address
|
from lbry.dht.serialization.datagram import make_compact_address, make_compact_ip, decode_compact_address
|
||||||
|
|
||||||
ALLOW_LOCALHOST = False
|
ALLOW_LOCALHOST = False
|
||||||
CACHE_SIZE = 16384
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
@lru_cache(CACHE_SIZE)
|
@lru_cache(1024)
|
||||||
def make_kademlia_peer(node_id: typing.Optional[bytes], address: typing.Optional[str],
|
def make_kademlia_peer(node_id: typing.Optional[bytes], address: typing.Optional[str],
|
||||||
udp_port: typing.Optional[int] = None,
|
udp_port: typing.Optional[int] = None,
|
||||||
tcp_port: typing.Optional[int] = None,
|
tcp_port: typing.Optional[int] = None,
|
||||||
|
@ -29,26 +25,17 @@ def is_valid_public_ipv4(address, allow_localhost: bool = False):
|
||||||
|
|
||||||
|
|
||||||
class PeerManager:
|
class PeerManager:
|
||||||
peer_manager_keys_metric = Gauge(
|
|
||||||
"peer_manager_keys", "Number of keys tracked by PeerManager dicts (sum)", namespace="dht_node",
|
|
||||||
labelnames=("scope",)
|
|
||||||
)
|
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop):
|
def __init__(self, loop: asyncio.AbstractEventLoop):
|
||||||
self._loop = loop
|
self._loop = loop
|
||||||
self._rpc_failures: typing.Dict[
|
self._rpc_failures: typing.Dict[
|
||||||
typing.Tuple[str, int], typing.Tuple[typing.Optional[float], typing.Optional[float]]
|
typing.Tuple[str, int], typing.Tuple[typing.Optional[float], typing.Optional[float]]
|
||||||
] = LRUCache(CACHE_SIZE)
|
] = {}
|
||||||
self._last_replied: typing.Dict[typing.Tuple[str, int], float] = LRUCache(CACHE_SIZE)
|
self._last_replied: typing.Dict[typing.Tuple[str, int], float] = LRUCache(2048)
|
||||||
self._last_sent: typing.Dict[typing.Tuple[str, int], float] = LRUCache(CACHE_SIZE)
|
self._last_sent: typing.Dict[typing.Tuple[str, int], float] = LRUCache(2048)
|
||||||
self._last_requested: typing.Dict[typing.Tuple[str, int], float] = LRUCache(CACHE_SIZE)
|
self._last_requested: typing.Dict[typing.Tuple[str, int], float] = LRUCache(2048)
|
||||||
self._node_id_mapping: typing.Dict[typing.Tuple[str, int], bytes] = LRUCache(CACHE_SIZE)
|
self._node_id_mapping: typing.Dict[typing.Tuple[str, int], bytes] = LRUCache(2048)
|
||||||
self._node_id_reverse_mapping: typing.Dict[bytes, typing.Tuple[str, int]] = LRUCache(CACHE_SIZE)
|
self._node_id_reverse_mapping: typing.Dict[bytes, typing.Tuple[str, int]] = LRUCache(2048)
|
||||||
self._node_tokens: typing.Dict[bytes, (float, bytes)] = LRUCache(CACHE_SIZE)
|
self._node_tokens: typing.Dict[bytes, (float, bytes)] = LRUCache(2048)
|
||||||
|
|
||||||
def count_cache_keys(self):
|
|
||||||
return len(self._rpc_failures) + len(self._last_replied) + len(self._last_sent) + len(
|
|
||||||
self._last_requested) + len(self._node_id_mapping) + len(self._node_id_reverse_mapping) + len(
|
|
||||||
self._node_tokens)
|
|
||||||
|
|
||||||
def reset(self):
|
def reset(self):
|
||||||
for statistic in (self._rpc_failures, self._last_replied, self._last_sent, self._last_requested):
|
for statistic in (self._rpc_failures, self._last_replied, self._last_sent, self._last_requested):
|
||||||
|
@ -98,10 +85,6 @@ class PeerManager:
|
||||||
self._node_id_mapping.pop(self._node_id_reverse_mapping.pop(node_id))
|
self._node_id_mapping.pop(self._node_id_reverse_mapping.pop(node_id))
|
||||||
self._node_id_mapping[(address, udp_port)] = node_id
|
self._node_id_mapping[(address, udp_port)] = node_id
|
||||||
self._node_id_reverse_mapping[node_id] = (address, udp_port)
|
self._node_id_reverse_mapping[node_id] = (address, udp_port)
|
||||||
self.peer_manager_keys_metric.labels("global").set(self.count_cache_keys())
|
|
||||||
|
|
||||||
def get_node_id_for_endpoint(self, address, port):
|
|
||||||
return self._node_id_mapping.get((address, port))
|
|
||||||
|
|
||||||
def prune(self): # TODO: periodically call this
|
def prune(self): # TODO: periodically call this
|
||||||
now = self._loop.time()
|
now = self._loop.time()
|
||||||
|
@ -153,8 +136,7 @@ class PeerManager:
|
||||||
def peer_is_good(self, peer: 'KademliaPeer'):
|
def peer_is_good(self, peer: 'KademliaPeer'):
|
||||||
return self.contact_triple_is_good(peer.node_id, peer.address, peer.udp_port)
|
return self.contact_triple_is_good(peer.node_id, peer.address, peer.udp_port)
|
||||||
|
|
||||||
|
def decode_tcp_peer_from_compact_address(self, compact_address: bytes) -> 'KademliaPeer': # pylint: disable=no-self-use
|
||||||
def decode_tcp_peer_from_compact_address(compact_address: bytes) -> 'KademliaPeer': # pylint: disable=no-self-use
|
|
||||||
node_id, address, tcp_port = decode_compact_address(compact_address)
|
node_id, address, tcp_port = decode_compact_address(compact_address)
|
||||||
return make_kademlia_peer(node_id, address, udp_port=None, tcp_port=tcp_port)
|
return make_kademlia_peer(node_id, address, udp_port=None, tcp_port=tcp_port)
|
||||||
|
|
||||||
|
@ -194,6 +176,3 @@ class KademliaPeer:
|
||||||
|
|
||||||
def compact_ip(self):
|
def compact_ip(self):
|
||||||
return make_compact_ip(self.address)
|
return make_compact_ip(self.address)
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return f"{self.__class__.__name__}({self.node_id.hex()[:8]}@{self.address}:{self.udp_port}-{self.tcp_port})"
|
|
||||||
|
|
|
@ -16,9 +16,6 @@ class DictDataStore:
|
||||||
self._peer_manager = peer_manager
|
self._peer_manager = peer_manager
|
||||||
self.completed_blobs: typing.Set[str] = set()
|
self.completed_blobs: typing.Set[str] = set()
|
||||||
|
|
||||||
def keys(self):
|
|
||||||
return self._data_store.keys()
|
|
||||||
|
|
||||||
def __len__(self):
|
def __len__(self):
|
||||||
return self._data_store.__len__()
|
return self._data_store.__len__()
|
||||||
|
|
||||||
|
|
|
@ -1,17 +1,17 @@
|
||||||
import asyncio
|
import asyncio
|
||||||
from itertools import chain
|
from itertools import chain
|
||||||
from collections import defaultdict, OrderedDict
|
from collections import defaultdict
|
||||||
from collections.abc import AsyncIterator
|
|
||||||
import typing
|
import typing
|
||||||
import logging
|
import logging
|
||||||
from typing import TYPE_CHECKING
|
from typing import TYPE_CHECKING
|
||||||
from lbry.dht import constants
|
from lbry.dht import constants
|
||||||
from lbry.dht.error import RemoteException, TransportNotConnected
|
from lbry.dht.error import RemoteException, TransportNotConnected
|
||||||
from lbry.dht.protocol.distance import Distance
|
from lbry.dht.protocol.distance import Distance
|
||||||
from lbry.dht.peer import make_kademlia_peer, decode_tcp_peer_from_compact_address
|
from lbry.dht.peer import make_kademlia_peer
|
||||||
from lbry.dht.serialization.datagram import PAGE_KEY
|
from lbry.dht.serialization.datagram import PAGE_KEY
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
|
from lbry.dht.protocol.routing_table import TreeRoutingTable
|
||||||
from lbry.dht.protocol.protocol import KademliaProtocol
|
from lbry.dht.protocol.protocol import KademliaProtocol
|
||||||
from lbry.dht.peer import PeerManager, KademliaPeer
|
from lbry.dht.peer import PeerManager, KademliaPeer
|
||||||
|
|
||||||
|
@ -26,15 +26,6 @@ class FindResponse:
|
||||||
def get_close_triples(self) -> typing.List[typing.Tuple[bytes, str, int]]:
|
def get_close_triples(self) -> typing.List[typing.Tuple[bytes, str, int]]:
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
def get_close_kademlia_peers(self, peer_info) -> typing.Generator[typing.Iterator['KademliaPeer'], None, None]:
|
|
||||||
for contact_triple in self.get_close_triples():
|
|
||||||
node_id, address, udp_port = contact_triple
|
|
||||||
try:
|
|
||||||
yield make_kademlia_peer(node_id, address, udp_port)
|
|
||||||
except ValueError:
|
|
||||||
log.warning("misbehaving peer %s:%i returned peer with reserved ip %s:%i", peer_info.address,
|
|
||||||
peer_info.udp_port, address, udp_port)
|
|
||||||
|
|
||||||
|
|
||||||
class FindNodeResponse(FindResponse):
|
class FindNodeResponse(FindResponse):
|
||||||
def __init__(self, key: bytes, close_triples: typing.List[typing.Tuple[bytes, str, int]]):
|
def __init__(self, key: bytes, close_triples: typing.List[typing.Tuple[bytes, str, int]]):
|
||||||
|
@ -65,33 +56,57 @@ class FindValueResponse(FindResponse):
|
||||||
return [(node_id, address.decode(), port) for node_id, address, port in self.close_triples]
|
return [(node_id, address.decode(), port) for node_id, address, port in self.close_triples]
|
||||||
|
|
||||||
|
|
||||||
class IterativeFinder(AsyncIterator):
|
def get_shortlist(routing_table: 'TreeRoutingTable', key: bytes,
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop,
|
shortlist: typing.Optional[typing.List['KademliaPeer']]) -> typing.List['KademliaPeer']:
|
||||||
protocol: 'KademliaProtocol', key: bytes,
|
"""
|
||||||
max_results: typing.Optional[int] = constants.K,
|
If not provided, initialize the shortlist of peers to probe to the (up to) k closest peers in the routing table
|
||||||
|
|
||||||
|
:param routing_table: a TreeRoutingTable
|
||||||
|
:param key: a 48 byte hash
|
||||||
|
:param shortlist: optional manually provided shortlist, this is done during bootstrapping when there are no
|
||||||
|
peers in the routing table. During bootstrap the shortlist is set to be the seed nodes.
|
||||||
|
"""
|
||||||
|
if len(key) != constants.HASH_LENGTH:
|
||||||
|
raise ValueError("invalid key length: %i" % len(key))
|
||||||
|
return shortlist or routing_table.find_close_peers(key)
|
||||||
|
|
||||||
|
|
||||||
|
class IterativeFinder:
|
||||||
|
def __init__(self, loop: asyncio.AbstractEventLoop, peer_manager: 'PeerManager',
|
||||||
|
routing_table: 'TreeRoutingTable', protocol: 'KademliaProtocol', key: bytes,
|
||||||
|
bottom_out_limit: typing.Optional[int] = 2, max_results: typing.Optional[int] = constants.K,
|
||||||
|
exclude: typing.Optional[typing.List[typing.Tuple[str, int]]] = None,
|
||||||
shortlist: typing.Optional[typing.List['KademliaPeer']] = None):
|
shortlist: typing.Optional[typing.List['KademliaPeer']] = None):
|
||||||
if len(key) != constants.HASH_LENGTH:
|
if len(key) != constants.HASH_LENGTH:
|
||||||
raise ValueError("invalid key length: %i" % len(key))
|
raise ValueError("invalid key length: %i" % len(key))
|
||||||
self.loop = loop
|
self.loop = loop
|
||||||
self.peer_manager = protocol.peer_manager
|
self.peer_manager = peer_manager
|
||||||
|
self.routing_table = routing_table
|
||||||
self.protocol = protocol
|
self.protocol = protocol
|
||||||
|
|
||||||
self.key = key
|
self.key = key
|
||||||
self.max_results = max(constants.K, max_results)
|
self.bottom_out_limit = bottom_out_limit
|
||||||
|
self.max_results = max_results
|
||||||
|
self.exclude = exclude or []
|
||||||
|
|
||||||
self.active: typing.Dict['KademliaPeer', int] = OrderedDict() # peer: distance, sorted
|
self.active: typing.Set['KademliaPeer'] = set()
|
||||||
self.contacted: typing.Set['KademliaPeer'] = set()
|
self.contacted: typing.Set['KademliaPeer'] = set()
|
||||||
self.distance = Distance(key)
|
self.distance = Distance(key)
|
||||||
|
|
||||||
self.iteration_queue = asyncio.Queue()
|
self.closest_peer: typing.Optional['KademliaPeer'] = None
|
||||||
|
self.prev_closest_peer: typing.Optional['KademliaPeer'] = None
|
||||||
|
|
||||||
self.running_probes: typing.Dict['KademliaPeer', asyncio.Task] = {}
|
self.iteration_queue = asyncio.Queue(loop=self.loop)
|
||||||
|
|
||||||
|
self.running_probes: typing.Set[asyncio.Task] = set()
|
||||||
self.iteration_count = 0
|
self.iteration_count = 0
|
||||||
|
self.bottom_out_count = 0
|
||||||
self.running = False
|
self.running = False
|
||||||
self.tasks: typing.List[asyncio.Task] = []
|
self.tasks: typing.List[asyncio.Task] = []
|
||||||
for peer in shortlist:
|
self.delayed_calls: typing.List[asyncio.Handle] = []
|
||||||
|
for peer in get_shortlist(routing_table, key, shortlist):
|
||||||
if peer.node_id:
|
if peer.node_id:
|
||||||
self._add_active(peer, force=True)
|
self._add_active(peer)
|
||||||
else:
|
else:
|
||||||
# seed nodes
|
# seed nodes
|
||||||
self._schedule_probe(peer)
|
self._schedule_probe(peer)
|
||||||
|
@ -123,79 +138,66 @@ class IterativeFinder(AsyncIterator):
|
||||||
"""
|
"""
|
||||||
return []
|
return []
|
||||||
|
|
||||||
def _add_active(self, peer, force=False):
|
def _is_closer(self, peer: 'KademliaPeer') -> bool:
|
||||||
if not force and self.peer_manager.peer_is_good(peer) is False:
|
return not self.closest_peer or self.distance.is_closer(peer.node_id, self.closest_peer.node_id)
|
||||||
return
|
|
||||||
if peer in self.contacted:
|
def _add_active(self, peer):
|
||||||
return
|
|
||||||
if peer not in self.active and peer.node_id and peer.node_id != self.protocol.node_id:
|
if peer not in self.active and peer.node_id and peer.node_id != self.protocol.node_id:
|
||||||
self.active[peer] = self.distance(peer.node_id)
|
self.active.add(peer)
|
||||||
self.active = OrderedDict(sorted(self.active.items(), key=lambda item: item[1]))
|
if self._is_closer(peer):
|
||||||
|
self.prev_closest_peer = self.closest_peer
|
||||||
|
self.closest_peer = peer
|
||||||
|
|
||||||
async def _handle_probe_result(self, peer: 'KademliaPeer', response: FindResponse):
|
async def _handle_probe_result(self, peer: 'KademliaPeer', response: FindResponse):
|
||||||
self._add_active(peer)
|
self._add_active(peer)
|
||||||
for new_peer in response.get_close_kademlia_peers(peer):
|
for contact_triple in response.get_close_triples():
|
||||||
self._add_active(new_peer)
|
node_id, address, udp_port = contact_triple
|
||||||
|
try:
|
||||||
|
self._add_active(make_kademlia_peer(node_id, address, udp_port))
|
||||||
|
except ValueError:
|
||||||
|
log.warning("misbehaving peer %s:%i returned peer with reserved ip %s:%i", peer.address,
|
||||||
|
peer.udp_port, address, udp_port)
|
||||||
self.check_result_ready(response)
|
self.check_result_ready(response)
|
||||||
self._log_state(reason="check result")
|
|
||||||
|
|
||||||
def _reset_closest(self, peer):
|
|
||||||
if peer in self.active:
|
|
||||||
del self.active[peer]
|
|
||||||
|
|
||||||
async def _send_probe(self, peer: 'KademliaPeer'):
|
async def _send_probe(self, peer: 'KademliaPeer'):
|
||||||
try:
|
try:
|
||||||
response = await self.send_probe(peer)
|
response = await self.send_probe(peer)
|
||||||
except asyncio.TimeoutError:
|
except asyncio.TimeoutError:
|
||||||
self._reset_closest(peer)
|
self.active.discard(peer)
|
||||||
return
|
return
|
||||||
except asyncio.CancelledError:
|
|
||||||
log.debug("%s[%x] cancelled probe",
|
|
||||||
type(self).__name__, id(self))
|
|
||||||
raise
|
|
||||||
except ValueError as err:
|
except ValueError as err:
|
||||||
log.warning(str(err))
|
log.warning(str(err))
|
||||||
self._reset_closest(peer)
|
self.active.discard(peer)
|
||||||
return
|
return
|
||||||
except TransportNotConnected:
|
except TransportNotConnected:
|
||||||
await self._aclose(reason="not connected")
|
return self.aclose()
|
||||||
return
|
|
||||||
except RemoteException:
|
except RemoteException:
|
||||||
self._reset_closest(peer)
|
|
||||||
return
|
return
|
||||||
return await self._handle_probe_result(peer, response)
|
return await self._handle_probe_result(peer, response)
|
||||||
|
|
||||||
def _search_round(self):
|
async def _search_round(self):
|
||||||
"""
|
"""
|
||||||
Send up to constants.alpha (5) probes to closest active peers
|
Send up to constants.alpha (5) probes to closest active peers
|
||||||
"""
|
"""
|
||||||
|
|
||||||
added = 0
|
added = 0
|
||||||
for index, peer in enumerate(self.active.keys()):
|
to_probe = list(self.active - self.contacted)
|
||||||
if index == 0:
|
to_probe.sort(key=lambda peer: self.distance(self.key))
|
||||||
log.debug("%s[%x] closest to probe: %s",
|
for peer in to_probe:
|
||||||
type(self).__name__, id(self),
|
if added >= constants.ALPHA:
|
||||||
peer.node_id.hex()[:8])
|
|
||||||
if peer in self.contacted:
|
|
||||||
continue
|
|
||||||
if len(self.running_probes) >= constants.ALPHA:
|
|
||||||
break
|
|
||||||
if index > (constants.K + len(self.running_probes)):
|
|
||||||
break
|
break
|
||||||
origin_address = (peer.address, peer.udp_port)
|
origin_address = (peer.address, peer.udp_port)
|
||||||
|
if origin_address in self.exclude:
|
||||||
|
continue
|
||||||
if peer.node_id == self.protocol.node_id:
|
if peer.node_id == self.protocol.node_id:
|
||||||
continue
|
continue
|
||||||
if origin_address == (self.protocol.external_ip, self.protocol.udp_port):
|
if origin_address == (self.protocol.external_ip, self.protocol.udp_port):
|
||||||
continue
|
continue
|
||||||
self._schedule_probe(peer)
|
self._schedule_probe(peer)
|
||||||
added += 1
|
added += 1
|
||||||
log.debug("%s[%x] running %d probes for key %s",
|
log.debug("running %d probes", len(self.running_probes))
|
||||||
type(self).__name__, id(self),
|
|
||||||
len(self.running_probes), self.key.hex()[:8])
|
|
||||||
if not added and not self.running_probes:
|
if not added and not self.running_probes:
|
||||||
log.debug("%s[%x] search for %s exhausted",
|
log.debug("search for %s exhausted", self.key.hex()[:8])
|
||||||
type(self).__name__, id(self),
|
|
||||||
self.key.hex()[:8])
|
|
||||||
self.search_exhausted()
|
self.search_exhausted()
|
||||||
|
|
||||||
def _schedule_probe(self, peer: 'KademliaPeer'):
|
def _schedule_probe(self, peer: 'KademliaPeer'):
|
||||||
|
@ -204,24 +206,33 @@ class IterativeFinder(AsyncIterator):
|
||||||
t = self.loop.create_task(self._send_probe(peer))
|
t = self.loop.create_task(self._send_probe(peer))
|
||||||
|
|
||||||
def callback(_):
|
def callback(_):
|
||||||
self.running_probes.pop(peer, None)
|
self.running_probes.difference_update({
|
||||||
if self.running:
|
probe for probe in self.running_probes if probe.done() or probe == t
|
||||||
self._search_round()
|
})
|
||||||
|
if not self.running_probes:
|
||||||
|
self.tasks.append(self.loop.create_task(self._search_task(0.0)))
|
||||||
|
|
||||||
t.add_done_callback(callback)
|
t.add_done_callback(callback)
|
||||||
self.running_probes[peer] = t
|
self.running_probes.add(t)
|
||||||
|
|
||||||
def _log_state(self, reason="?"):
|
async def _search_task(self, delay: typing.Optional[float] = constants.ITERATIVE_LOOKUP_DELAY):
|
||||||
log.debug("%s[%x] [%s] %s: %i active nodes %i contacted %i produced %i queued",
|
try:
|
||||||
type(self).__name__, id(self), self.key.hex()[:8],
|
if self.running:
|
||||||
reason, len(self.active), len(self.contacted),
|
await self._search_round()
|
||||||
self.iteration_count, self.iteration_queue.qsize())
|
if self.running:
|
||||||
|
self.delayed_calls.append(self.loop.call_later(delay, self._search))
|
||||||
|
except (asyncio.CancelledError, StopAsyncIteration, TransportNotConnected):
|
||||||
|
if self.running:
|
||||||
|
self.loop.call_soon(self.aclose)
|
||||||
|
|
||||||
|
def _search(self):
|
||||||
|
self.tasks.append(self.loop.create_task(self._search_task()))
|
||||||
|
|
||||||
def __aiter__(self):
|
def __aiter__(self):
|
||||||
if self.running:
|
if self.running:
|
||||||
raise Exception("already running")
|
raise Exception("already running")
|
||||||
self.running = True
|
self.running = True
|
||||||
self.loop.call_soon(self._search_round)
|
self._search()
|
||||||
return self
|
return self
|
||||||
|
|
||||||
async def __anext__(self) -> typing.List['KademliaPeer']:
|
async def __anext__(self) -> typing.List['KademliaPeer']:
|
||||||
|
@ -234,57 +245,47 @@ class IterativeFinder(AsyncIterator):
|
||||||
raise StopAsyncIteration
|
raise StopAsyncIteration
|
||||||
self.iteration_count += 1
|
self.iteration_count += 1
|
||||||
return result
|
return result
|
||||||
except asyncio.CancelledError:
|
except (asyncio.CancelledError, StopAsyncIteration):
|
||||||
await self._aclose(reason="cancelled")
|
self.loop.call_soon(self.aclose)
|
||||||
raise
|
|
||||||
except StopAsyncIteration:
|
|
||||||
await self._aclose(reason="no more results")
|
|
||||||
raise
|
raise
|
||||||
|
|
||||||
async def _aclose(self, reason="?"):
|
def aclose(self):
|
||||||
log.debug("%s[%x] [%s] shutdown because %s: %i active nodes %i contacted %i produced %i queued",
|
|
||||||
type(self).__name__, id(self), self.key.hex()[:8],
|
|
||||||
reason, len(self.active), len(self.contacted),
|
|
||||||
self.iteration_count, self.iteration_queue.qsize())
|
|
||||||
self.running = False
|
self.running = False
|
||||||
self.iteration_queue.put_nowait(None)
|
self.iteration_queue.put_nowait(None)
|
||||||
for task in chain(self.tasks, self.running_probes.values()):
|
for task in chain(self.tasks, self.running_probes, self.delayed_calls):
|
||||||
task.cancel()
|
task.cancel()
|
||||||
self.tasks.clear()
|
self.tasks.clear()
|
||||||
self.running_probes.clear()
|
self.running_probes.clear()
|
||||||
|
self.delayed_calls.clear()
|
||||||
|
|
||||||
async def aclose(self):
|
|
||||||
if self.running:
|
|
||||||
await self._aclose(reason="aclose")
|
|
||||||
log.debug("%s[%x] [%s] async close completed",
|
|
||||||
type(self).__name__, id(self), self.key.hex()[:8])
|
|
||||||
|
|
||||||
class IterativeNodeFinder(IterativeFinder):
|
class IterativeNodeFinder(IterativeFinder):
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop,
|
def __init__(self, loop: asyncio.AbstractEventLoop, peer_manager: 'PeerManager',
|
||||||
protocol: 'KademliaProtocol', key: bytes,
|
routing_table: 'TreeRoutingTable', protocol: 'KademliaProtocol', key: bytes,
|
||||||
max_results: typing.Optional[int] = constants.K,
|
bottom_out_limit: typing.Optional[int] = 2, max_results: typing.Optional[int] = constants.K,
|
||||||
|
exclude: typing.Optional[typing.List[typing.Tuple[str, int]]] = None,
|
||||||
shortlist: typing.Optional[typing.List['KademliaPeer']] = None):
|
shortlist: typing.Optional[typing.List['KademliaPeer']] = None):
|
||||||
super().__init__(loop, protocol, key, max_results, shortlist)
|
super().__init__(loop, peer_manager, routing_table, protocol, key, bottom_out_limit, max_results, exclude,
|
||||||
|
shortlist)
|
||||||
self.yielded_peers: typing.Set['KademliaPeer'] = set()
|
self.yielded_peers: typing.Set['KademliaPeer'] = set()
|
||||||
|
|
||||||
async def send_probe(self, peer: 'KademliaPeer') -> FindNodeResponse:
|
async def send_probe(self, peer: 'KademliaPeer') -> FindNodeResponse:
|
||||||
log.debug("probe %s:%d (%s) for NODE %s",
|
log.debug("probing %s:%d %s", peer.address, peer.udp_port, peer.node_id.hex()[:8] if peer.node_id else '')
|
||||||
peer.address, peer.udp_port, peer.node_id.hex()[:8] if peer.node_id else '', self.key.hex()[:8])
|
|
||||||
response = await self.protocol.get_rpc_peer(peer).find_node(self.key)
|
response = await self.protocol.get_rpc_peer(peer).find_node(self.key)
|
||||||
return FindNodeResponse(self.key, response)
|
return FindNodeResponse(self.key, response)
|
||||||
|
|
||||||
def search_exhausted(self):
|
def search_exhausted(self):
|
||||||
self.put_result(self.active.keys(), finish=True)
|
self.put_result(self.active, finish=True)
|
||||||
|
|
||||||
def put_result(self, from_iter: typing.Iterable['KademliaPeer'], finish=False):
|
def put_result(self, from_iter: typing.Iterable['KademliaPeer'], finish=False):
|
||||||
not_yet_yielded = [
|
not_yet_yielded = [
|
||||||
peer for peer in from_iter
|
peer for peer in from_iter
|
||||||
if peer not in self.yielded_peers
|
if peer not in self.yielded_peers
|
||||||
and peer.node_id != self.protocol.node_id
|
and peer.node_id != self.protocol.node_id
|
||||||
and self.peer_manager.peer_is_good(peer) is True # return only peers who answered
|
and self.peer_manager.peer_is_good(peer) is not False
|
||||||
]
|
]
|
||||||
not_yet_yielded.sort(key=lambda peer: self.distance(peer.node_id))
|
not_yet_yielded.sort(key=lambda peer: self.distance(peer.node_id))
|
||||||
to_yield = not_yet_yielded[:max(constants.K, self.max_results)]
|
to_yield = not_yet_yielded[:min(constants.K, len(not_yet_yielded))]
|
||||||
if to_yield:
|
if to_yield:
|
||||||
self.yielded_peers.update(to_yield)
|
self.yielded_peers.update(to_yield)
|
||||||
self.iteration_queue.put_nowait(to_yield)
|
self.iteration_queue.put_nowait(to_yield)
|
||||||
|
@ -296,15 +297,27 @@ class IterativeNodeFinder(IterativeFinder):
|
||||||
|
|
||||||
if found:
|
if found:
|
||||||
log.debug("found")
|
log.debug("found")
|
||||||
return self.put_result(self.active.keys(), finish=True)
|
return self.put_result(self.active, finish=True)
|
||||||
|
if self.prev_closest_peer and self.closest_peer and not self._is_closer(self.prev_closest_peer):
|
||||||
|
# log.info("improving, %i %i %i %i %i", len(self.shortlist), len(self.active), len(self.contacted),
|
||||||
|
# self.bottom_out_count, self.iteration_count)
|
||||||
|
self.bottom_out_count = 0
|
||||||
|
elif self.prev_closest_peer and self.closest_peer:
|
||||||
|
self.bottom_out_count += 1
|
||||||
|
log.info("bottom out %i %i %i", len(self.active), len(self.contacted), self.bottom_out_count)
|
||||||
|
if self.bottom_out_count >= self.bottom_out_limit or self.iteration_count >= self.bottom_out_limit:
|
||||||
|
log.info("limit hit")
|
||||||
|
self.put_result(self.active, True)
|
||||||
|
|
||||||
|
|
||||||
class IterativeValueFinder(IterativeFinder):
|
class IterativeValueFinder(IterativeFinder):
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop,
|
def __init__(self, loop: asyncio.AbstractEventLoop, peer_manager: 'PeerManager',
|
||||||
protocol: 'KademliaProtocol', key: bytes,
|
routing_table: 'TreeRoutingTable', protocol: 'KademliaProtocol', key: bytes,
|
||||||
max_results: typing.Optional[int] = constants.K,
|
bottom_out_limit: typing.Optional[int] = 2, max_results: typing.Optional[int] = constants.K,
|
||||||
|
exclude: typing.Optional[typing.List[typing.Tuple[str, int]]] = None,
|
||||||
shortlist: typing.Optional[typing.List['KademliaPeer']] = None):
|
shortlist: typing.Optional[typing.List['KademliaPeer']] = None):
|
||||||
super().__init__(loop, protocol, key, max_results, shortlist)
|
super().__init__(loop, peer_manager, routing_table, protocol, key, bottom_out_limit, max_results, exclude,
|
||||||
|
shortlist)
|
||||||
self.blob_peers: typing.Set['KademliaPeer'] = set()
|
self.blob_peers: typing.Set['KademliaPeer'] = set()
|
||||||
# this tracks the index of the most recent page we requested from each peer
|
# this tracks the index of the most recent page we requested from each peer
|
||||||
self.peer_pages: typing.DefaultDict['KademliaPeer', int] = defaultdict(int)
|
self.peer_pages: typing.DefaultDict['KademliaPeer', int] = defaultdict(int)
|
||||||
|
@ -312,8 +325,6 @@ class IterativeValueFinder(IterativeFinder):
|
||||||
self.discovered_peers: typing.Dict['KademliaPeer', typing.Set['KademliaPeer']] = defaultdict(set)
|
self.discovered_peers: typing.Dict['KademliaPeer', typing.Set['KademliaPeer']] = defaultdict(set)
|
||||||
|
|
||||||
async def send_probe(self, peer: 'KademliaPeer') -> FindValueResponse:
|
async def send_probe(self, peer: 'KademliaPeer') -> FindValueResponse:
|
||||||
log.debug("probe %s:%d (%s) for VALUE %s",
|
|
||||||
peer.address, peer.udp_port, peer.node_id.hex()[:8], self.key.hex()[:8])
|
|
||||||
page = self.peer_pages[peer]
|
page = self.peer_pages[peer]
|
||||||
response = await self.protocol.get_rpc_peer(peer).find_value(self.key, page=page)
|
response = await self.protocol.get_rpc_peer(peer).find_value(self.key, page=page)
|
||||||
parsed = FindValueResponse(self.key, response)
|
parsed = FindValueResponse(self.key, response)
|
||||||
|
@ -323,7 +334,7 @@ class IterativeValueFinder(IterativeFinder):
|
||||||
decoded_peers = set()
|
decoded_peers = set()
|
||||||
for compact_addr in parsed.found_compact_addresses:
|
for compact_addr in parsed.found_compact_addresses:
|
||||||
try:
|
try:
|
||||||
decoded_peers.add(decode_tcp_peer_from_compact_address(compact_addr))
|
decoded_peers.add(self.peer_manager.decode_tcp_peer_from_compact_address(compact_addr))
|
||||||
except ValueError:
|
except ValueError:
|
||||||
log.warning("misbehaving peer %s:%i returned invalid peer for blob",
|
log.warning("misbehaving peer %s:%i returned invalid peer for blob",
|
||||||
peer.address, peer.udp_port)
|
peer.address, peer.udp_port)
|
||||||
|
@ -335,6 +346,7 @@ class IterativeValueFinder(IterativeFinder):
|
||||||
already_known + len(parsed.found_compact_addresses))
|
already_known + len(parsed.found_compact_addresses))
|
||||||
if len(self.discovered_peers[peer]) != already_known + len(parsed.found_compact_addresses):
|
if len(self.discovered_peers[peer]) != already_known + len(parsed.found_compact_addresses):
|
||||||
log.warning("misbehaving peer %s:%i returned duplicate peers for blob", peer.address, peer.udp_port)
|
log.warning("misbehaving peer %s:%i returned duplicate peers for blob", peer.address, peer.udp_port)
|
||||||
|
parsed.found_compact_addresses.clear()
|
||||||
elif len(parsed.found_compact_addresses) >= constants.K and self.peer_pages[peer] < parsed.pages:
|
elif len(parsed.found_compact_addresses) >= constants.K and self.peer_pages[peer] < parsed.pages:
|
||||||
# the peer returned a full page and indicates it has more
|
# the peer returned a full page and indicates it has more
|
||||||
self.peer_pages[peer] += 1
|
self.peer_pages[peer] += 1
|
||||||
|
@ -345,15 +357,26 @@ class IterativeValueFinder(IterativeFinder):
|
||||||
|
|
||||||
def check_result_ready(self, response: FindValueResponse):
|
def check_result_ready(self, response: FindValueResponse):
|
||||||
if response.found:
|
if response.found:
|
||||||
blob_peers = [decode_tcp_peer_from_compact_address(compact_addr)
|
blob_peers = [self.peer_manager.decode_tcp_peer_from_compact_address(compact_addr)
|
||||||
for compact_addr in response.found_compact_addresses]
|
for compact_addr in response.found_compact_addresses]
|
||||||
to_yield = []
|
to_yield = []
|
||||||
|
self.bottom_out_count = 0
|
||||||
for blob_peer in blob_peers:
|
for blob_peer in blob_peers:
|
||||||
if blob_peer not in self.blob_peers:
|
if blob_peer not in self.blob_peers:
|
||||||
self.blob_peers.add(blob_peer)
|
self.blob_peers.add(blob_peer)
|
||||||
to_yield.append(blob_peer)
|
to_yield.append(blob_peer)
|
||||||
if to_yield:
|
if to_yield:
|
||||||
|
# log.info("found %i new peers for blob", len(to_yield))
|
||||||
self.iteration_queue.put_nowait(to_yield)
|
self.iteration_queue.put_nowait(to_yield)
|
||||||
|
# if self.max_results and len(self.blob_peers) >= self.max_results:
|
||||||
|
# log.info("enough blob peers found")
|
||||||
|
# if not self.finished.is_set():
|
||||||
|
# self.finished.set()
|
||||||
|
elif self.prev_closest_peer and self.closest_peer:
|
||||||
|
self.bottom_out_count += 1
|
||||||
|
if self.bottom_out_count >= self.bottom_out_limit:
|
||||||
|
log.info("blob peer search bottomed out")
|
||||||
|
self.iteration_queue.put_nowait(None)
|
||||||
|
|
||||||
def get_initial_result(self) -> typing.List['KademliaPeer']:
|
def get_initial_result(self) -> typing.List['KademliaPeer']:
|
||||||
if self.protocol.data_store.has_peers_for_blob(self.key):
|
if self.protocol.data_store.has_peers_for_blob(self.key):
|
||||||
|
|
|
@ -3,14 +3,11 @@ import socket
|
||||||
import functools
|
import functools
|
||||||
import hashlib
|
import hashlib
|
||||||
import asyncio
|
import asyncio
|
||||||
import time
|
|
||||||
import typing
|
import typing
|
||||||
import random
|
import random
|
||||||
from asyncio.protocols import DatagramProtocol
|
from asyncio.protocols import DatagramProtocol
|
||||||
from asyncio.transports import DatagramTransport
|
from asyncio.transports import DatagramTransport
|
||||||
|
|
||||||
from prometheus_client import Gauge, Counter, Histogram
|
|
||||||
|
|
||||||
from lbry.dht import constants
|
from lbry.dht import constants
|
||||||
from lbry.dht.serialization.bencoding import DecodeError
|
from lbry.dht.serialization.bencoding import DecodeError
|
||||||
from lbry.dht.serialization.datagram import decode_datagram, ErrorDatagram, ResponseDatagram, RequestDatagram
|
from lbry.dht.serialization.datagram import decode_datagram, ErrorDatagram, ResponseDatagram, RequestDatagram
|
||||||
|
@ -33,11 +30,6 @@ OLD_PROTOCOL_ERRORS = {
|
||||||
|
|
||||||
|
|
||||||
class KademliaRPC:
|
class KademliaRPC:
|
||||||
stored_blob_metric = Gauge(
|
|
||||||
"stored_blobs", "Number of blobs announced by other peers", namespace="dht_node",
|
|
||||||
labelnames=("scope",),
|
|
||||||
)
|
|
||||||
|
|
||||||
def __init__(self, protocol: 'KademliaProtocol', loop: asyncio.AbstractEventLoop, peer_port: int = 3333):
|
def __init__(self, protocol: 'KademliaProtocol', loop: asyncio.AbstractEventLoop, peer_port: int = 3333):
|
||||||
self.protocol = protocol
|
self.protocol = protocol
|
||||||
self.loop = loop
|
self.loop = loop
|
||||||
|
@ -69,7 +61,6 @@ class KademliaRPC:
|
||||||
self.protocol.data_store.add_peer_to_blob(
|
self.protocol.data_store.add_peer_to_blob(
|
||||||
rpc_contact, blob_hash
|
rpc_contact, blob_hash
|
||||||
)
|
)
|
||||||
self.stored_blob_metric.labels("global").set(len(self.protocol.data_store))
|
|
||||||
return b'OK'
|
return b'OK'
|
||||||
|
|
||||||
def find_node(self, rpc_contact: 'KademliaPeer', key: bytes) -> typing.List[typing.Tuple[bytes, str, int]]:
|
def find_node(self, rpc_contact: 'KademliaPeer', key: bytes) -> typing.List[typing.Tuple[bytes, str, int]]:
|
||||||
|
@ -218,10 +209,6 @@ class PingQueue:
|
||||||
def running(self):
|
def running(self):
|
||||||
return self._running
|
return self._running
|
||||||
|
|
||||||
@property
|
|
||||||
def busy(self):
|
|
||||||
return self._running and (any(self._running_pings) or any(self._pending_contacts))
|
|
||||||
|
|
||||||
def enqueue_maybe_ping(self, *peers: 'KademliaPeer', delay: typing.Optional[float] = None):
|
def enqueue_maybe_ping(self, *peers: 'KademliaPeer', delay: typing.Optional[float] = None):
|
||||||
delay = delay if delay is not None else self._default_delay
|
delay = delay if delay is not None else self._default_delay
|
||||||
now = self._loop.time()
|
now = self._loop.time()
|
||||||
|
@ -233,7 +220,7 @@ class PingQueue:
|
||||||
async def ping_task():
|
async def ping_task():
|
||||||
try:
|
try:
|
||||||
if self._protocol.peer_manager.peer_is_good(peer):
|
if self._protocol.peer_manager.peer_is_good(peer):
|
||||||
if not self._protocol.routing_table.get_peer(peer.node_id):
|
if peer not in self._protocol.routing_table.get_peers():
|
||||||
self._protocol.add_peer(peer)
|
self._protocol.add_peer(peer)
|
||||||
return
|
return
|
||||||
await self._protocol.get_rpc_peer(peer).ping()
|
await self._protocol.get_rpc_peer(peer).ping()
|
||||||
|
@ -253,7 +240,7 @@ class PingQueue:
|
||||||
del self._pending_contacts[peer]
|
del self._pending_contacts[peer]
|
||||||
self.maybe_ping(peer)
|
self.maybe_ping(peer)
|
||||||
break
|
break
|
||||||
await asyncio.sleep(1)
|
await asyncio.sleep(1, loop=self._loop)
|
||||||
|
|
||||||
def start(self):
|
def start(self):
|
||||||
assert not self._running
|
assert not self._running
|
||||||
|
@ -272,33 +259,9 @@ class PingQueue:
|
||||||
|
|
||||||
|
|
||||||
class KademliaProtocol(DatagramProtocol):
|
class KademliaProtocol(DatagramProtocol):
|
||||||
request_sent_metric = Counter(
|
|
||||||
"request_sent", "Number of requests send from DHT RPC protocol", namespace="dht_node",
|
|
||||||
labelnames=("method",),
|
|
||||||
)
|
|
||||||
request_success_metric = Counter(
|
|
||||||
"request_success", "Number of successful requests", namespace="dht_node",
|
|
||||||
labelnames=("method",),
|
|
||||||
)
|
|
||||||
request_error_metric = Counter(
|
|
||||||
"request_error", "Number of errors returned from request to other peers", namespace="dht_node",
|
|
||||||
labelnames=("method",),
|
|
||||||
)
|
|
||||||
HISTOGRAM_BUCKETS = (
|
|
||||||
.005, .01, .025, .05, .075, .1, .25, .5, .75, 1.0, 2.5, 3.0, 3.5, 4.0, 4.50, 5.0, 5.50, 6.0, float('inf')
|
|
||||||
)
|
|
||||||
response_time_metric = Histogram(
|
|
||||||
"response_time", "Response times of DHT RPC requests", namespace="dht_node", buckets=HISTOGRAM_BUCKETS,
|
|
||||||
labelnames=("method",)
|
|
||||||
)
|
|
||||||
received_request_metric = Counter(
|
|
||||||
"received_request", "Number of received DHT RPC requests", namespace="dht_node",
|
|
||||||
labelnames=("method",),
|
|
||||||
)
|
|
||||||
|
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop, peer_manager: 'PeerManager', node_id: bytes, external_ip: str,
|
def __init__(self, loop: asyncio.AbstractEventLoop, peer_manager: 'PeerManager', node_id: bytes, external_ip: str,
|
||||||
udp_port: int, peer_port: int, rpc_timeout: float = constants.RPC_TIMEOUT,
|
udp_port: int, peer_port: int, rpc_timeout: float = constants.RPC_TIMEOUT,
|
||||||
split_buckets_under_index: int = constants.SPLIT_BUCKETS_UNDER_INDEX, is_boostrap_node: bool = False):
|
split_buckets_under_index: int = constants.SPLIT_BUCKETS_UNDER_INDEX):
|
||||||
self.peer_manager = peer_manager
|
self.peer_manager = peer_manager
|
||||||
self.loop = loop
|
self.loop = loop
|
||||||
self.node_id = node_id
|
self.node_id = node_id
|
||||||
|
@ -313,16 +276,15 @@ class KademliaProtocol(DatagramProtocol):
|
||||||
self.transport: DatagramTransport = None
|
self.transport: DatagramTransport = None
|
||||||
self.old_token_secret = constants.generate_id()
|
self.old_token_secret = constants.generate_id()
|
||||||
self.token_secret = constants.generate_id()
|
self.token_secret = constants.generate_id()
|
||||||
self.routing_table = TreeRoutingTable(
|
self.routing_table = TreeRoutingTable(self.loop, self.peer_manager, self.node_id, split_buckets_under_index)
|
||||||
self.loop, self.peer_manager, self.node_id, split_buckets_under_index, is_bootstrap_node=is_boostrap_node)
|
|
||||||
self.data_store = DictDataStore(self.loop, self.peer_manager)
|
self.data_store = DictDataStore(self.loop, self.peer_manager)
|
||||||
self.ping_queue = PingQueue(self.loop, self)
|
self.ping_queue = PingQueue(self.loop, self)
|
||||||
self.node_rpc = KademliaRPC(self, self.loop, self.peer_port)
|
self.node_rpc = KademliaRPC(self, self.loop, self.peer_port)
|
||||||
self.rpc_timeout = rpc_timeout
|
self.rpc_timeout = rpc_timeout
|
||||||
self._split_lock = asyncio.Lock()
|
self._split_lock = asyncio.Lock(loop=self.loop)
|
||||||
self._to_remove: typing.Set['KademliaPeer'] = set()
|
self._to_remove: typing.Set['KademliaPeer'] = set()
|
||||||
self._to_add: typing.Set['KademliaPeer'] = set()
|
self._to_add: typing.Set['KademliaPeer'] = set()
|
||||||
self._wakeup_routing_task = asyncio.Event()
|
self._wakeup_routing_task = asyncio.Event(loop=self.loop)
|
||||||
self.maintaing_routing_task: typing.Optional[asyncio.Task] = None
|
self.maintaing_routing_task: typing.Optional[asyncio.Task] = None
|
||||||
|
|
||||||
@functools.lru_cache(128)
|
@functools.lru_cache(128)
|
||||||
|
@ -361,10 +323,72 @@ class KademliaProtocol(DatagramProtocol):
|
||||||
return args, {}
|
return args, {}
|
||||||
|
|
||||||
async def _add_peer(self, peer: 'KademliaPeer'):
|
async def _add_peer(self, peer: 'KademliaPeer'):
|
||||||
async def probe(some_peer: 'KademliaPeer'):
|
if not peer.node_id:
|
||||||
rpc_peer = self.get_rpc_peer(some_peer)
|
log.warning("Tried adding a peer with no node id!")
|
||||||
await rpc_peer.ping()
|
return False
|
||||||
return await self.routing_table.add_peer(peer, probe)
|
for my_peer in self.routing_table.get_peers():
|
||||||
|
if (my_peer.address, my_peer.udp_port) == (peer.address, peer.udp_port) and my_peer.node_id != peer.node_id:
|
||||||
|
self.routing_table.remove_peer(my_peer)
|
||||||
|
self.routing_table.join_buckets()
|
||||||
|
bucket_index = self.routing_table.kbucket_index(peer.node_id)
|
||||||
|
if self.routing_table.buckets[bucket_index].add_peer(peer):
|
||||||
|
return True
|
||||||
|
|
||||||
|
# The bucket is full; see if it can be split (by checking if its range includes the host node's node_id)
|
||||||
|
if self.routing_table.should_split(bucket_index, peer.node_id):
|
||||||
|
self.routing_table.split_bucket(bucket_index)
|
||||||
|
# Retry the insertion attempt
|
||||||
|
result = await self._add_peer(peer)
|
||||||
|
self.routing_table.join_buckets()
|
||||||
|
return result
|
||||||
|
else:
|
||||||
|
# We can't split the k-bucket
|
||||||
|
#
|
||||||
|
# The 13 page kademlia paper specifies that the least recently contacted node in the bucket
|
||||||
|
# shall be pinged. If it fails to reply it is replaced with the new contact. If the ping is successful
|
||||||
|
# the new contact is ignored and not added to the bucket (sections 2.2 and 2.4).
|
||||||
|
#
|
||||||
|
# A reasonable extension to this is BEP 0005, which extends the above:
|
||||||
|
#
|
||||||
|
# Not all nodes that we learn about are equal. Some are "good" and some are not.
|
||||||
|
# Many nodes using the DHT are able to send queries and receive responses,
|
||||||
|
# but are not able to respond to queries from other nodes. It is important that
|
||||||
|
# each node's routing table must contain only known good nodes. A good node is
|
||||||
|
# a node has responded to one of our queries within the last 15 minutes. A node
|
||||||
|
# is also good if it has ever responded to one of our queries and has sent us a
|
||||||
|
# query within the last 15 minutes. After 15 minutes of inactivity, a node becomes
|
||||||
|
# questionable. Nodes become bad when they fail to respond to multiple queries
|
||||||
|
# in a row. Nodes that we know are good are given priority over nodes with unknown status.
|
||||||
|
#
|
||||||
|
# When there are bad or questionable nodes in the bucket, the least recent is selected for
|
||||||
|
# potential replacement (BEP 0005). When all nodes in the bucket are fresh, the head (least recent)
|
||||||
|
# contact is selected as described in section 2.2 of the kademlia paper. In both cases the new contact
|
||||||
|
# is ignored if the pinged node replies.
|
||||||
|
|
||||||
|
not_good_contacts = self.routing_table.buckets[bucket_index].get_bad_or_unknown_peers()
|
||||||
|
not_recently_replied = []
|
||||||
|
for my_peer in not_good_contacts:
|
||||||
|
last_replied = self.peer_manager.get_last_replied(my_peer.address, my_peer.udp_port)
|
||||||
|
if not last_replied or last_replied + 60 < self.loop.time():
|
||||||
|
not_recently_replied.append(my_peer)
|
||||||
|
if not_recently_replied:
|
||||||
|
to_replace = not_recently_replied[0]
|
||||||
|
else:
|
||||||
|
to_replace = self.routing_table.buckets[bucket_index].peers[0]
|
||||||
|
last_replied = self.peer_manager.get_last_replied(to_replace.address, to_replace.udp_port)
|
||||||
|
if last_replied and last_replied + 60 > self.loop.time():
|
||||||
|
return False
|
||||||
|
log.debug("pinging %s:%s", to_replace.address, to_replace.udp_port)
|
||||||
|
try:
|
||||||
|
to_replace_rpc = self.get_rpc_peer(to_replace)
|
||||||
|
await to_replace_rpc.ping()
|
||||||
|
return False
|
||||||
|
except asyncio.TimeoutError:
|
||||||
|
log.debug("Replacing dead contact in bucket %i: %s:%i with %s:%i ", bucket_index,
|
||||||
|
to_replace.address, to_replace.udp_port, peer.address, peer.udp_port)
|
||||||
|
if to_replace in self.routing_table.buckets[bucket_index]:
|
||||||
|
self.routing_table.buckets[bucket_index].remove_peer(to_replace)
|
||||||
|
return await self._add_peer(peer)
|
||||||
|
|
||||||
def add_peer(self, peer: 'KademliaPeer'):
|
def add_peer(self, peer: 'KademliaPeer'):
|
||||||
if peer.node_id == self.node_id:
|
if peer.node_id == self.node_id:
|
||||||
|
@ -382,10 +406,11 @@ class KademliaProtocol(DatagramProtocol):
|
||||||
async with self._split_lock:
|
async with self._split_lock:
|
||||||
peer = self._to_remove.pop()
|
peer = self._to_remove.pop()
|
||||||
self.routing_table.remove_peer(peer)
|
self.routing_table.remove_peer(peer)
|
||||||
|
self.routing_table.join_buckets()
|
||||||
while self._to_add:
|
while self._to_add:
|
||||||
async with self._split_lock:
|
async with self._split_lock:
|
||||||
await self._add_peer(self._to_add.pop())
|
await self._add_peer(self._to_add.pop())
|
||||||
await asyncio.gather(self._wakeup_routing_task.wait(), asyncio.sleep(.1))
|
await asyncio.gather(self._wakeup_routing_task.wait(), asyncio.sleep(.1, loop=self.loop), loop=self.loop)
|
||||||
self._wakeup_routing_task.clear()
|
self._wakeup_routing_task.clear()
|
||||||
|
|
||||||
def _handle_rpc(self, sender_contact: 'KademliaPeer', message: RequestDatagram):
|
def _handle_rpc(self, sender_contact: 'KademliaPeer', message: RequestDatagram):
|
||||||
|
@ -422,10 +447,10 @@ class KademliaProtocol(DatagramProtocol):
|
||||||
|
|
||||||
def handle_request_datagram(self, address: typing.Tuple[str, int], request_datagram: RequestDatagram):
|
def handle_request_datagram(self, address: typing.Tuple[str, int], request_datagram: RequestDatagram):
|
||||||
# This is an RPC method request
|
# This is an RPC method request
|
||||||
self.received_request_metric.labels(method=request_datagram.method).inc()
|
|
||||||
self.peer_manager.report_last_requested(address[0], address[1])
|
self.peer_manager.report_last_requested(address[0], address[1])
|
||||||
|
try:
|
||||||
peer = self.routing_table.get_peer(request_datagram.node_id)
|
peer = self.routing_table.get_peer(request_datagram.node_id)
|
||||||
if not peer:
|
except IndexError:
|
||||||
try:
|
try:
|
||||||
peer = make_kademlia_peer(request_datagram.node_id, address[0], address[1])
|
peer = make_kademlia_peer(request_datagram.node_id, address[0], address[1])
|
||||||
except ValueError as err:
|
except ValueError as err:
|
||||||
|
@ -550,19 +575,14 @@ class KademliaProtocol(DatagramProtocol):
|
||||||
self._send(peer, request)
|
self._send(peer, request)
|
||||||
response_fut = self.sent_messages[request.rpc_id][1]
|
response_fut = self.sent_messages[request.rpc_id][1]
|
||||||
try:
|
try:
|
||||||
self.request_sent_metric.labels(method=request.method).inc()
|
|
||||||
start = time.perf_counter()
|
|
||||||
response = await asyncio.wait_for(response_fut, self.rpc_timeout)
|
response = await asyncio.wait_for(response_fut, self.rpc_timeout)
|
||||||
self.response_time_metric.labels(method=request.method).observe(time.perf_counter() - start)
|
|
||||||
self.peer_manager.report_last_replied(peer.address, peer.udp_port)
|
self.peer_manager.report_last_replied(peer.address, peer.udp_port)
|
||||||
self.request_success_metric.labels(method=request.method).inc()
|
|
||||||
return response
|
return response
|
||||||
except asyncio.CancelledError:
|
except asyncio.CancelledError:
|
||||||
if not response_fut.done():
|
if not response_fut.done():
|
||||||
response_fut.cancel()
|
response_fut.cancel()
|
||||||
raise
|
raise
|
||||||
except (asyncio.TimeoutError, RemoteException):
|
except (asyncio.TimeoutError, RemoteException):
|
||||||
self.request_error_metric.labels(method=request.method).inc()
|
|
||||||
self.peer_manager.report_failure(peer.address, peer.udp_port)
|
self.peer_manager.report_failure(peer.address, peer.udp_port)
|
||||||
if self.peer_manager.peer_is_good(peer) is False:
|
if self.peer_manager.peer_is_good(peer) is False:
|
||||||
self.remove_peer(peer)
|
self.remove_peer(peer)
|
||||||
|
|
|
@ -4,11 +4,7 @@ import logging
|
||||||
import typing
|
import typing
|
||||||
import itertools
|
import itertools
|
||||||
|
|
||||||
from prometheus_client import Gauge
|
|
||||||
|
|
||||||
from lbry import utils
|
|
||||||
from lbry.dht import constants
|
from lbry.dht import constants
|
||||||
from lbry.dht.error import RemoteException
|
|
||||||
from lbry.dht.protocol.distance import Distance
|
from lbry.dht.protocol.distance import Distance
|
||||||
if typing.TYPE_CHECKING:
|
if typing.TYPE_CHECKING:
|
||||||
from lbry.dht.peer import KademliaPeer, PeerManager
|
from lbry.dht.peer import KademliaPeer, PeerManager
|
||||||
|
@ -17,20 +13,10 @@ log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class KBucket:
|
class KBucket:
|
||||||
|
""" Description - later
|
||||||
"""
|
"""
|
||||||
Kademlia K-bucket implementation.
|
|
||||||
"""
|
|
||||||
peer_in_routing_table_metric = Gauge(
|
|
||||||
"peers_in_routing_table", "Number of peers on routing table", namespace="dht_node",
|
|
||||||
labelnames=("scope",)
|
|
||||||
)
|
|
||||||
peer_with_x_bit_colliding_metric = Gauge(
|
|
||||||
"peer_x_bit_colliding", "Number of peers with at least X bits colliding with this node id",
|
|
||||||
namespace="dht_node", labelnames=("amount",)
|
|
||||||
)
|
|
||||||
|
|
||||||
def __init__(self, peer_manager: 'PeerManager', range_min: int, range_max: int,
|
def __init__(self, peer_manager: 'PeerManager', range_min: int, range_max: int, node_id: bytes):
|
||||||
node_id: bytes, capacity: int = constants.K):
|
|
||||||
"""
|
"""
|
||||||
@param range_min: The lower boundary for the range in the n-bit ID
|
@param range_min: The lower boundary for the range in the n-bit ID
|
||||||
space covered by this k-bucket
|
space covered by this k-bucket
|
||||||
|
@ -38,12 +24,12 @@ class KBucket:
|
||||||
covered by this k-bucket
|
covered by this k-bucket
|
||||||
"""
|
"""
|
||||||
self._peer_manager = peer_manager
|
self._peer_manager = peer_manager
|
||||||
|
self.last_accessed = 0
|
||||||
self.range_min = range_min
|
self.range_min = range_min
|
||||||
self.range_max = range_max
|
self.range_max = range_max
|
||||||
self.peers: typing.List['KademliaPeer'] = []
|
self.peers: typing.List['KademliaPeer'] = []
|
||||||
self._node_id = node_id
|
self._node_id = node_id
|
||||||
self._distance_to_self = Distance(node_id)
|
self._distance_to_self = Distance(node_id)
|
||||||
self.capacity = capacity
|
|
||||||
|
|
||||||
def add_peer(self, peer: 'KademliaPeer') -> bool:
|
def add_peer(self, peer: 'KademliaPeer') -> bool:
|
||||||
""" Add contact to _contact list in the right order. This will move the
|
""" Add contact to _contact list in the right order. This will move the
|
||||||
|
@ -70,19 +56,18 @@ class KBucket:
|
||||||
self.peers.remove(local_peer)
|
self.peers.remove(local_peer)
|
||||||
self.peers.append(peer)
|
self.peers.append(peer)
|
||||||
return True
|
return True
|
||||||
if len(self.peers) < self.capacity:
|
if len(self.peers) < constants.K:
|
||||||
self.peers.append(peer)
|
self.peers.append(peer)
|
||||||
self.peer_in_routing_table_metric.labels("global").inc()
|
|
||||||
bits_colliding = utils.get_colliding_prefix_bits(peer.node_id, self._node_id)
|
|
||||||
self.peer_with_x_bit_colliding_metric.labels(amount=bits_colliding).inc()
|
|
||||||
return True
|
return True
|
||||||
else:
|
else:
|
||||||
return False
|
return False
|
||||||
|
# raise BucketFull("No space in bucket to insert contact")
|
||||||
|
|
||||||
def get_peer(self, node_id: bytes) -> 'KademliaPeer':
|
def get_peer(self, node_id: bytes) -> 'KademliaPeer':
|
||||||
for peer in self.peers:
|
for peer in self.peers:
|
||||||
if peer.node_id == node_id:
|
if peer.node_id == node_id:
|
||||||
return peer
|
return peer
|
||||||
|
raise IndexError(node_id)
|
||||||
|
|
||||||
def get_peers(self, count=-1, exclude_contact=None, sort_distance_to=None) -> typing.List['KademliaPeer']:
|
def get_peers(self, count=-1, exclude_contact=None, sort_distance_to=None) -> typing.List['KademliaPeer']:
|
||||||
""" Returns a list containing up to the first count number of contacts
|
""" Returns a list containing up to the first count number of contacts
|
||||||
|
@ -139,9 +124,6 @@ class KBucket:
|
||||||
|
|
||||||
def remove_peer(self, peer: 'KademliaPeer') -> None:
|
def remove_peer(self, peer: 'KademliaPeer') -> None:
|
||||||
self.peers.remove(peer)
|
self.peers.remove(peer)
|
||||||
self.peer_in_routing_table_metric.labels("global").dec()
|
|
||||||
bits_colliding = utils.get_colliding_prefix_bits(peer.node_id, self._node_id)
|
|
||||||
self.peer_with_x_bit_colliding_metric.labels(amount=bits_colliding).dec()
|
|
||||||
|
|
||||||
def key_in_range(self, key: bytes) -> bool:
|
def key_in_range(self, key: bytes) -> bool:
|
||||||
""" Tests whether the specified key (i.e. node ID) is in the range
|
""" Tests whether the specified key (i.e. node ID) is in the range
|
||||||
|
@ -179,36 +161,24 @@ class TreeRoutingTable:
|
||||||
version of the Kademlia paper, in section 2.4. It does, however, use the
|
version of the Kademlia paper, in section 2.4. It does, however, use the
|
||||||
ping RPC-based k-bucket eviction algorithm described in section 2.2 of
|
ping RPC-based k-bucket eviction algorithm described in section 2.2 of
|
||||||
that paper.
|
that paper.
|
||||||
|
|
||||||
BOOTSTRAP MODE: if set to True, we always add all peers. This is so a
|
|
||||||
bootstrap node does not get a bias towards its own node id and replies are
|
|
||||||
the best it can provide (joining peer knows its neighbors immediately).
|
|
||||||
Over time, this will need to be optimized so we use the disk as holding
|
|
||||||
everything in memory won't be feasible anymore.
|
|
||||||
See: https://github.com/bittorrent/bootstrap-dht
|
|
||||||
"""
|
"""
|
||||||
bucket_in_routing_table_metric = Gauge(
|
|
||||||
"buckets_in_routing_table", "Number of buckets on routing table", namespace="dht_node",
|
|
||||||
labelnames=("scope",)
|
|
||||||
)
|
|
||||||
|
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop, peer_manager: 'PeerManager', parent_node_id: bytes,
|
def __init__(self, loop: asyncio.AbstractEventLoop, peer_manager: 'PeerManager', parent_node_id: bytes,
|
||||||
split_buckets_under_index: int = constants.SPLIT_BUCKETS_UNDER_INDEX, is_bootstrap_node: bool = False):
|
split_buckets_under_index: int = constants.SPLIT_BUCKETS_UNDER_INDEX):
|
||||||
self._loop = loop
|
self._loop = loop
|
||||||
self._peer_manager = peer_manager
|
self._peer_manager = peer_manager
|
||||||
self._parent_node_id = parent_node_id
|
self._parent_node_id = parent_node_id
|
||||||
self._split_buckets_under_index = split_buckets_under_index
|
self._split_buckets_under_index = split_buckets_under_index
|
||||||
self.buckets: typing.List[KBucket] = [
|
self.buckets: typing.List[KBucket] = [
|
||||||
KBucket(
|
KBucket(
|
||||||
self._peer_manager, range_min=0, range_max=2 ** constants.HASH_BITS, node_id=self._parent_node_id,
|
self._peer_manager, range_min=0, range_max=2 ** constants.HASH_BITS, node_id=self._parent_node_id
|
||||||
capacity=1 << 32 if is_bootstrap_node else constants.K
|
|
||||||
)
|
)
|
||||||
]
|
]
|
||||||
|
|
||||||
def get_peers(self) -> typing.List['KademliaPeer']:
|
def get_peers(self) -> typing.List['KademliaPeer']:
|
||||||
return list(itertools.chain.from_iterable(map(lambda bucket: bucket.peers, self.buckets)))
|
return list(itertools.chain.from_iterable(map(lambda bucket: bucket.peers, self.buckets)))
|
||||||
|
|
||||||
def _should_split(self, bucket_index: int, to_add: bytes) -> bool:
|
def should_split(self, bucket_index: int, to_add: bytes) -> bool:
|
||||||
# https://stackoverflow.com/questions/32129978/highly-unbalanced-kademlia-routing-table/32187456#32187456
|
# https://stackoverflow.com/questions/32129978/highly-unbalanced-kademlia-routing-table/32187456#32187456
|
||||||
if bucket_index < self._split_buckets_under_index:
|
if bucket_index < self._split_buckets_under_index:
|
||||||
return True
|
return True
|
||||||
|
@ -233,32 +203,39 @@ class TreeRoutingTable:
|
||||||
return []
|
return []
|
||||||
|
|
||||||
def get_peer(self, contact_id: bytes) -> 'KademliaPeer':
|
def get_peer(self, contact_id: bytes) -> 'KademliaPeer':
|
||||||
return self.buckets[self._kbucket_index(contact_id)].get_peer(contact_id)
|
"""
|
||||||
|
@raise IndexError: No contact with the specified contact ID is known
|
||||||
|
by this node
|
||||||
|
"""
|
||||||
|
return self.buckets[self.kbucket_index(contact_id)].get_peer(contact_id)
|
||||||
|
|
||||||
def get_refresh_list(self, start_index: int = 0, force: bool = False) -> typing.List[bytes]:
|
def get_refresh_list(self, start_index: int = 0, force: bool = False) -> typing.List[bytes]:
|
||||||
|
bucket_index = start_index
|
||||||
refresh_ids = []
|
refresh_ids = []
|
||||||
for offset, _ in enumerate(self.buckets[start_index:]):
|
now = int(self._loop.time())
|
||||||
refresh_ids.append(self._midpoint_id_in_bucket_range(start_index + offset))
|
for bucket in self.buckets[start_index:]:
|
||||||
# if we have 3 or fewer populated buckets get two random ids in the range of each to try and
|
if force or now - bucket.last_accessed >= constants.REFRESH_INTERVAL:
|
||||||
# populate/split the buckets further
|
to_search = self.midpoint_id_in_bucket_range(bucket_index)
|
||||||
buckets_with_contacts = self.buckets_with_contacts()
|
refresh_ids.append(to_search)
|
||||||
if buckets_with_contacts <= 3:
|
bucket_index += 1
|
||||||
for i in range(buckets_with_contacts):
|
|
||||||
refresh_ids.append(self._random_id_in_bucket_range(i))
|
|
||||||
refresh_ids.append(self._random_id_in_bucket_range(i))
|
|
||||||
return refresh_ids
|
return refresh_ids
|
||||||
|
|
||||||
def remove_peer(self, peer: 'KademliaPeer') -> None:
|
def remove_peer(self, peer: 'KademliaPeer') -> None:
|
||||||
if not peer.node_id:
|
if not peer.node_id:
|
||||||
return
|
return
|
||||||
bucket_index = self._kbucket_index(peer.node_id)
|
bucket_index = self.kbucket_index(peer.node_id)
|
||||||
try:
|
try:
|
||||||
self.buckets[bucket_index].remove_peer(peer)
|
self.buckets[bucket_index].remove_peer(peer)
|
||||||
self._join_buckets()
|
|
||||||
except ValueError:
|
except ValueError:
|
||||||
return
|
return
|
||||||
|
|
||||||
def _kbucket_index(self, key: bytes) -> int:
|
def touch_kbucket(self, key: bytes) -> None:
|
||||||
|
self.touch_kbucket_by_index(self.kbucket_index(key))
|
||||||
|
|
||||||
|
def touch_kbucket_by_index(self, bucket_index: int):
|
||||||
|
self.buckets[bucket_index].last_accessed = int(self._loop.time())
|
||||||
|
|
||||||
|
def kbucket_index(self, key: bytes) -> int:
|
||||||
i = 0
|
i = 0
|
||||||
for bucket in self.buckets:
|
for bucket in self.buckets:
|
||||||
if bucket.key_in_range(key):
|
if bucket.key_in_range(key):
|
||||||
|
@ -267,19 +244,19 @@ class TreeRoutingTable:
|
||||||
i += 1
|
i += 1
|
||||||
return i
|
return i
|
||||||
|
|
||||||
def _random_id_in_bucket_range(self, bucket_index: int) -> bytes:
|
def random_id_in_bucket_range(self, bucket_index: int) -> bytes:
|
||||||
random_id = int(random.randrange(self.buckets[bucket_index].range_min, self.buckets[bucket_index].range_max))
|
random_id = int(random.randrange(self.buckets[bucket_index].range_min, self.buckets[bucket_index].range_max))
|
||||||
return Distance(
|
return Distance(
|
||||||
self._parent_node_id
|
self._parent_node_id
|
||||||
)(random_id.to_bytes(constants.HASH_LENGTH, 'big')).to_bytes(constants.HASH_LENGTH, 'big')
|
)(random_id.to_bytes(constants.HASH_LENGTH, 'big')).to_bytes(constants.HASH_LENGTH, 'big')
|
||||||
|
|
||||||
def _midpoint_id_in_bucket_range(self, bucket_index: int) -> bytes:
|
def midpoint_id_in_bucket_range(self, bucket_index: int) -> bytes:
|
||||||
half = int((self.buckets[bucket_index].range_max - self.buckets[bucket_index].range_min) // 2)
|
half = int((self.buckets[bucket_index].range_max - self.buckets[bucket_index].range_min) // 2)
|
||||||
return Distance(self._parent_node_id)(
|
return Distance(self._parent_node_id)(
|
||||||
int(self.buckets[bucket_index].range_min + half).to_bytes(constants.HASH_LENGTH, 'big')
|
int(self.buckets[bucket_index].range_min + half).to_bytes(constants.HASH_LENGTH, 'big')
|
||||||
).to_bytes(constants.HASH_LENGTH, 'big')
|
).to_bytes(constants.HASH_LENGTH, 'big')
|
||||||
|
|
||||||
def _split_bucket(self, old_bucket_index: int) -> None:
|
def split_bucket(self, old_bucket_index: int) -> None:
|
||||||
""" Splits the specified k-bucket into two new buckets which together
|
""" Splits the specified k-bucket into two new buckets which together
|
||||||
cover the same range in the key/ID space
|
cover the same range in the key/ID space
|
||||||
|
|
||||||
|
@ -302,9 +279,8 @@ class TreeRoutingTable:
|
||||||
# ...and remove them from the old bucket
|
# ...and remove them from the old bucket
|
||||||
for contact in new_bucket.peers:
|
for contact in new_bucket.peers:
|
||||||
old_bucket.remove_peer(contact)
|
old_bucket.remove_peer(contact)
|
||||||
self.bucket_in_routing_table_metric.labels("global").set(len(self.buckets))
|
|
||||||
|
|
||||||
def _join_buckets(self):
|
def join_buckets(self):
|
||||||
if len(self.buckets) == 1:
|
if len(self.buckets) == 1:
|
||||||
return
|
return
|
||||||
to_pop = [i for i, bucket in enumerate(self.buckets) if len(bucket) == 0]
|
to_pop = [i for i, bucket in enumerate(self.buckets) if len(bucket) == 0]
|
||||||
|
@ -326,8 +302,14 @@ class TreeRoutingTable:
|
||||||
elif can_go_higher:
|
elif can_go_higher:
|
||||||
self.buckets[bucket_index_to_pop + 1].range_min = bucket.range_min
|
self.buckets[bucket_index_to_pop + 1].range_min = bucket.range_min
|
||||||
self.buckets.remove(bucket)
|
self.buckets.remove(bucket)
|
||||||
self.bucket_in_routing_table_metric.labels("global").set(len(self.buckets))
|
return self.join_buckets()
|
||||||
return self._join_buckets()
|
|
||||||
|
def contact_in_routing_table(self, address_tuple: typing.Tuple[str, int]) -> bool:
|
||||||
|
for bucket in self.buckets:
|
||||||
|
for contact in bucket.get_peers(sort_distance_to=False):
|
||||||
|
if address_tuple[0] == contact.address and address_tuple[1] == contact.udp_port:
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
def buckets_with_contacts(self) -> int:
|
def buckets_with_contacts(self) -> int:
|
||||||
count = 0
|
count = 0
|
||||||
|
@ -335,70 +317,3 @@ class TreeRoutingTable:
|
||||||
if len(bucket) > 0:
|
if len(bucket) > 0:
|
||||||
count += 1
|
count += 1
|
||||||
return count
|
return count
|
||||||
|
|
||||||
async def add_peer(self, peer: 'KademliaPeer', probe: typing.Callable[['KademliaPeer'], typing.Awaitable]):
|
|
||||||
if not peer.node_id:
|
|
||||||
log.warning("Tried adding a peer with no node id!")
|
|
||||||
return False
|
|
||||||
for my_peer in self.get_peers():
|
|
||||||
if (my_peer.address, my_peer.udp_port) == (peer.address, peer.udp_port) and my_peer.node_id != peer.node_id:
|
|
||||||
self.remove_peer(my_peer)
|
|
||||||
self._join_buckets()
|
|
||||||
bucket_index = self._kbucket_index(peer.node_id)
|
|
||||||
if self.buckets[bucket_index].add_peer(peer):
|
|
||||||
return True
|
|
||||||
|
|
||||||
# The bucket is full; see if it can be split (by checking if its range includes the host node's node_id)
|
|
||||||
if self._should_split(bucket_index, peer.node_id):
|
|
||||||
self._split_bucket(bucket_index)
|
|
||||||
# Retry the insertion attempt
|
|
||||||
result = await self.add_peer(peer, probe)
|
|
||||||
self._join_buckets()
|
|
||||||
return result
|
|
||||||
else:
|
|
||||||
# We can't split the k-bucket
|
|
||||||
#
|
|
||||||
# The 13 page kademlia paper specifies that the least recently contacted node in the bucket
|
|
||||||
# shall be pinged. If it fails to reply it is replaced with the new contact. If the ping is successful
|
|
||||||
# the new contact is ignored and not added to the bucket (sections 2.2 and 2.4).
|
|
||||||
#
|
|
||||||
# A reasonable extension to this is BEP 0005, which extends the above:
|
|
||||||
#
|
|
||||||
# Not all nodes that we learn about are equal. Some are "good" and some are not.
|
|
||||||
# Many nodes using the DHT are able to send queries and receive responses,
|
|
||||||
# but are not able to respond to queries from other nodes. It is important that
|
|
||||||
# each node's routing table must contain only known good nodes. A good node is
|
|
||||||
# a node has responded to one of our queries within the last 15 minutes. A node
|
|
||||||
# is also good if it has ever responded to one of our queries and has sent us a
|
|
||||||
# query within the last 15 minutes. After 15 minutes of inactivity, a node becomes
|
|
||||||
# questionable. Nodes become bad when they fail to respond to multiple queries
|
|
||||||
# in a row. Nodes that we know are good are given priority over nodes with unknown status.
|
|
||||||
#
|
|
||||||
# When there are bad or questionable nodes in the bucket, the least recent is selected for
|
|
||||||
# potential replacement (BEP 0005). When all nodes in the bucket are fresh, the head (least recent)
|
|
||||||
# contact is selected as described in section 2.2 of the kademlia paper. In both cases the new contact
|
|
||||||
# is ignored if the pinged node replies.
|
|
||||||
|
|
||||||
not_good_contacts = self.buckets[bucket_index].get_bad_or_unknown_peers()
|
|
||||||
not_recently_replied = []
|
|
||||||
for my_peer in not_good_contacts:
|
|
||||||
last_replied = self._peer_manager.get_last_replied(my_peer.address, my_peer.udp_port)
|
|
||||||
if not last_replied or last_replied + 60 < self._loop.time():
|
|
||||||
not_recently_replied.append(my_peer)
|
|
||||||
if not_recently_replied:
|
|
||||||
to_replace = not_recently_replied[0]
|
|
||||||
else:
|
|
||||||
to_replace = self.buckets[bucket_index].peers[0]
|
|
||||||
last_replied = self._peer_manager.get_last_replied(to_replace.address, to_replace.udp_port)
|
|
||||||
if last_replied and last_replied + 60 > self._loop.time():
|
|
||||||
return False
|
|
||||||
log.debug("pinging %s:%s", to_replace.address, to_replace.udp_port)
|
|
||||||
try:
|
|
||||||
await probe(to_replace)
|
|
||||||
return False
|
|
||||||
except (asyncio.TimeoutError, RemoteException):
|
|
||||||
log.debug("Replacing dead contact in bucket %i: %s:%i with %s:%i ", bucket_index,
|
|
||||||
to_replace.address, to_replace.udp_port, peer.address, peer.udp_port)
|
|
||||||
if to_replace in self.buckets[bucket_index]:
|
|
||||||
self.buckets[bucket_index].remove_peer(to_replace)
|
|
||||||
return await self.add_peer(peer, probe)
|
|
||||||
|
|
|
@ -38,7 +38,6 @@ Code | Name | Message
|
||||||
114 | InputStringIsBlank | {argument} cannot be blank.
|
114 | InputStringIsBlank | {argument} cannot be blank.
|
||||||
115 | EmptyPublishedFile | Cannot publish empty file: {file_path}
|
115 | EmptyPublishedFile | Cannot publish empty file: {file_path}
|
||||||
116 | MissingPublishedFile | File does not exist: {file_path}
|
116 | MissingPublishedFile | File does not exist: {file_path}
|
||||||
117 | InvalidStreamURL | Invalid LBRY stream URL: '{url}' -- When an URL cannot be downloaded, such as '@Channel/' or a collection
|
|
||||||
**2xx** | Configuration | Configuration errors.
|
**2xx** | Configuration | Configuration errors.
|
||||||
201 | ConfigWrite | Cannot write configuration file '{path}'. -- When writing the default config fails on startup, such as due to permission issues.
|
201 | ConfigWrite | Cannot write configuration file '{path}'. -- When writing the default config fails on startup, such as due to permission issues.
|
||||||
202 | ConfigRead | Cannot find provided configuration file '{path}'. -- Can't open the config file user provided via command line args.
|
202 | ConfigRead | Cannot find provided configuration file '{path}'. -- Can't open the config file user provided via command line args.
|
||||||
|
|
|
@ -105,16 +105,6 @@ class MissingPublishedFileError(InputValueError):
|
||||||
super().__init__(f"File does not exist: {file_path}")
|
super().__init__(f"File does not exist: {file_path}")
|
||||||
|
|
||||||
|
|
||||||
class InvalidStreamURLError(InputValueError):
|
|
||||||
"""
|
|
||||||
When an URL cannot be downloaded, such as '@Channel/' or a collection
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, url):
|
|
||||||
self.url = url
|
|
||||||
super().__init__(f"Invalid LBRY stream URL: '{url}'")
|
|
||||||
|
|
||||||
|
|
||||||
class ConfigurationError(BaseError):
|
class ConfigurationError(BaseError):
|
||||||
"""
|
"""
|
||||||
Configuration errors.
|
Configuration errors.
|
||||||
|
|
|
@ -226,9 +226,6 @@ def get_argument_parser():
|
||||||
def ensure_directory_exists(path: str):
|
def ensure_directory_exists(path: str):
|
||||||
if not os.path.isdir(path):
|
if not os.path.isdir(path):
|
||||||
pathlib.Path(path).mkdir(parents=True, exist_ok=True)
|
pathlib.Path(path).mkdir(parents=True, exist_ok=True)
|
||||||
use_effective_ids = os.access in os.supports_effective_ids
|
|
||||||
if not os.access(path, os.W_OK, effective_ids=use_effective_ids):
|
|
||||||
raise PermissionError(f"The following directory is not writable: {path}")
|
|
||||||
|
|
||||||
|
|
||||||
LOG_MODULES = 'lbry', 'aioupnp'
|
LOG_MODULES = 'lbry', 'aioupnp'
|
||||||
|
|
|
@ -170,12 +170,11 @@ class AnalyticsManager:
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
|
|
||||||
async def send_disk_space_used(self, storage_used, storage_limit, is_from_network_quota):
|
async def send_disk_space_used(self, storage_used, storage_limit):
|
||||||
await self.track(
|
await self.track(
|
||||||
self._event(DISK_SPACE, {
|
self._event(DISK_SPACE, {
|
||||||
'used': storage_used,
|
'used': storage_used,
|
||||||
'limit': storage_limit,
|
'limit': storage_limit,
|
||||||
'from_network_quota': is_from_network_quota
|
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
from lbry.extras.cli import execute_command
|
|
||||||
from lbry.conf import Config
|
from lbry.conf import Config
|
||||||
|
from lbry.extras.cli import execute_command
|
||||||
|
|
||||||
|
|
||||||
def daemon_rpc(conf: Config, method: str, **kwargs):
|
def daemon_rpc(conf: Config, method: str, **kwargs):
|
||||||
|
|
|
@ -37,7 +37,7 @@ class Component(metaclass=ComponentType):
|
||||||
def running(self):
|
def running(self):
|
||||||
return self._running
|
return self._running
|
||||||
|
|
||||||
async def get_status(self): # pylint: disable=no-self-use
|
async def get_status(self):
|
||||||
return
|
return
|
||||||
|
|
||||||
async def start(self):
|
async def start(self):
|
||||||
|
|
|
@ -42,7 +42,7 @@ class ComponentManager:
|
||||||
self.analytics_manager = analytics_manager
|
self.analytics_manager = analytics_manager
|
||||||
self.component_classes = {}
|
self.component_classes = {}
|
||||||
self.components = set()
|
self.components = set()
|
||||||
self.started = asyncio.Event()
|
self.started = asyncio.Event(loop=self.loop)
|
||||||
self.peer_manager = peer_manager or PeerManager(asyncio.get_event_loop_policy().get_event_loop())
|
self.peer_manager = peer_manager or PeerManager(asyncio.get_event_loop_policy().get_event_loop())
|
||||||
|
|
||||||
for component_name, component_class in self.default_component_classes.items():
|
for component_name, component_class in self.default_component_classes.items():
|
||||||
|
@ -118,7 +118,7 @@ class ComponentManager:
|
||||||
component._setup() for component in stage if not component.running
|
component._setup() for component in stage if not component.running
|
||||||
]
|
]
|
||||||
if needing_start:
|
if needing_start:
|
||||||
await asyncio.wait(map(asyncio.create_task, needing_start))
|
await asyncio.wait(needing_start)
|
||||||
self.started.set()
|
self.started.set()
|
||||||
|
|
||||||
async def stop(self):
|
async def stop(self):
|
||||||
|
@ -131,7 +131,7 @@ class ComponentManager:
|
||||||
component._stop() for component in stage if component.running
|
component._stop() for component in stage if component.running
|
||||||
]
|
]
|
||||||
if needing_stop:
|
if needing_stop:
|
||||||
await asyncio.wait(map(asyncio.create_task, needing_stop))
|
await asyncio.wait(needing_stop)
|
||||||
|
|
||||||
def all_components_running(self, *component_names):
|
def all_components_running(self, *component_names):
|
||||||
"""
|
"""
|
||||||
|
|
|
@ -4,7 +4,6 @@ import asyncio
|
||||||
import logging
|
import logging
|
||||||
import binascii
|
import binascii
|
||||||
import typing
|
import typing
|
||||||
|
|
||||||
import base58
|
import base58
|
||||||
|
|
||||||
from aioupnp import __version__ as aioupnp_version
|
from aioupnp import __version__ as aioupnp_version
|
||||||
|
@ -18,7 +17,6 @@ from lbry.dht.blob_announcer import BlobAnnouncer
|
||||||
from lbry.blob.blob_manager import BlobManager
|
from lbry.blob.blob_manager import BlobManager
|
||||||
from lbry.blob.disk_space_manager import DiskSpaceManager
|
from lbry.blob.disk_space_manager import DiskSpaceManager
|
||||||
from lbry.blob_exchange.server import BlobServer
|
from lbry.blob_exchange.server import BlobServer
|
||||||
from lbry.stream.background_downloader import BackgroundDownloader
|
|
||||||
from lbry.stream.stream_manager import StreamManager
|
from lbry.stream.stream_manager import StreamManager
|
||||||
from lbry.file.file_manager import FileManager
|
from lbry.file.file_manager import FileManager
|
||||||
from lbry.extras.daemon.component import Component
|
from lbry.extras.daemon.component import Component
|
||||||
|
@ -27,8 +25,10 @@ from lbry.extras.daemon.storage import SQLiteStorage
|
||||||
from lbry.torrent.torrent_manager import TorrentManager
|
from lbry.torrent.torrent_manager import TorrentManager
|
||||||
from lbry.wallet import WalletManager
|
from lbry.wallet import WalletManager
|
||||||
from lbry.wallet.usage_payment import WalletServerPayer
|
from lbry.wallet.usage_payment import WalletServerPayer
|
||||||
from lbry.torrent.tracker import TrackerClient
|
try:
|
||||||
from lbry.torrent.session import TorrentSession
|
from lbry.torrent.session import TorrentSession
|
||||||
|
except ImportError:
|
||||||
|
TorrentSession = None
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
@ -42,11 +42,9 @@ DHT_COMPONENT = "dht"
|
||||||
HASH_ANNOUNCER_COMPONENT = "hash_announcer"
|
HASH_ANNOUNCER_COMPONENT = "hash_announcer"
|
||||||
FILE_MANAGER_COMPONENT = "file_manager"
|
FILE_MANAGER_COMPONENT = "file_manager"
|
||||||
DISK_SPACE_COMPONENT = "disk_space"
|
DISK_SPACE_COMPONENT = "disk_space"
|
||||||
BACKGROUND_DOWNLOADER_COMPONENT = "background_downloader"
|
|
||||||
PEER_PROTOCOL_SERVER_COMPONENT = "peer_protocol_server"
|
PEER_PROTOCOL_SERVER_COMPONENT = "peer_protocol_server"
|
||||||
UPNP_COMPONENT = "upnp"
|
UPNP_COMPONENT = "upnp"
|
||||||
EXCHANGE_RATE_MANAGER_COMPONENT = "exchange_rate_manager"
|
EXCHANGE_RATE_MANAGER_COMPONENT = "exchange_rate_manager"
|
||||||
TRACKER_ANNOUNCER_COMPONENT = "tracker_announcer_component"
|
|
||||||
LIBTORRENT_COMPONENT = "libtorrent_component"
|
LIBTORRENT_COMPONENT = "libtorrent_component"
|
||||||
|
|
||||||
|
|
||||||
|
@ -293,7 +291,6 @@ class DHTComponent(Component):
|
||||||
peer_port=self.external_peer_port,
|
peer_port=self.external_peer_port,
|
||||||
rpc_timeout=self.conf.node_rpc_timeout,
|
rpc_timeout=self.conf.node_rpc_timeout,
|
||||||
split_buckets_under_index=self.conf.split_buckets_under_index,
|
split_buckets_under_index=self.conf.split_buckets_under_index,
|
||||||
is_bootstrap_node=self.conf.is_bootstrap_node,
|
|
||||||
storage=storage
|
storage=storage
|
||||||
)
|
)
|
||||||
self.dht_node.start(self.conf.network_interface, self.conf.known_dht_nodes)
|
self.dht_node.start(self.conf.network_interface, self.conf.known_dht_nodes)
|
||||||
|
@ -357,6 +354,10 @@ class FileManagerComponent(Component):
|
||||||
wallet = self.component_manager.get_component(WALLET_COMPONENT)
|
wallet = self.component_manager.get_component(WALLET_COMPONENT)
|
||||||
node = self.component_manager.get_component(DHT_COMPONENT) \
|
node = self.component_manager.get_component(DHT_COMPONENT) \
|
||||||
if self.component_manager.has_component(DHT_COMPONENT) else None
|
if self.component_manager.has_component(DHT_COMPONENT) else None
|
||||||
|
try:
|
||||||
|
torrent = self.component_manager.get_component(LIBTORRENT_COMPONENT) if TorrentSession else None
|
||||||
|
except NameError:
|
||||||
|
torrent = None
|
||||||
log.info('Starting the file manager')
|
log.info('Starting the file manager')
|
||||||
loop = asyncio.get_event_loop()
|
loop = asyncio.get_event_loop()
|
||||||
self.file_manager = FileManager(
|
self.file_manager = FileManager(
|
||||||
|
@ -365,8 +366,7 @@ class FileManagerComponent(Component):
|
||||||
self.file_manager.source_managers['stream'] = StreamManager(
|
self.file_manager.source_managers['stream'] = StreamManager(
|
||||||
loop, self.conf, blob_manager, wallet, storage, node,
|
loop, self.conf, blob_manager, wallet, storage, node,
|
||||||
)
|
)
|
||||||
if self.component_manager.has_component(LIBTORRENT_COMPONENT):
|
if TorrentSession and LIBTORRENT_COMPONENT not in self.conf.components_to_skip:
|
||||||
torrent = self.component_manager.get_component(LIBTORRENT_COMPONENT)
|
|
||||||
self.file_manager.source_managers['torrent'] = TorrentManager(
|
self.file_manager.source_managers['torrent'] = TorrentManager(
|
||||||
loop, self.conf, torrent, storage, self.component_manager.analytics_manager
|
loop, self.conf, torrent, storage, self.component_manager.analytics_manager
|
||||||
)
|
)
|
||||||
|
@ -374,69 +374,7 @@ class FileManagerComponent(Component):
|
||||||
log.info('Done setting up file manager')
|
log.info('Done setting up file manager')
|
||||||
|
|
||||||
async def stop(self):
|
async def stop(self):
|
||||||
await self.file_manager.stop()
|
self.file_manager.stop()
|
||||||
|
|
||||||
|
|
||||||
class BackgroundDownloaderComponent(Component):
|
|
||||||
MIN_PREFIX_COLLIDING_BITS = 8
|
|
||||||
component_name = BACKGROUND_DOWNLOADER_COMPONENT
|
|
||||||
depends_on = [DATABASE_COMPONENT, BLOB_COMPONENT, DISK_SPACE_COMPONENT]
|
|
||||||
|
|
||||||
def __init__(self, component_manager):
|
|
||||||
super().__init__(component_manager)
|
|
||||||
self.background_task: typing.Optional[asyncio.Task] = None
|
|
||||||
self.download_loop_delay_seconds = 60
|
|
||||||
self.ongoing_download: typing.Optional[asyncio.Task] = None
|
|
||||||
self.space_manager: typing.Optional[DiskSpaceManager] = None
|
|
||||||
self.blob_manager: typing.Optional[BlobManager] = None
|
|
||||||
self.background_downloader: typing.Optional[BackgroundDownloader] = None
|
|
||||||
self.dht_node: typing.Optional[Node] = None
|
|
||||||
self.space_available: typing.Optional[int] = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def is_busy(self):
|
|
||||||
return bool(self.ongoing_download and not self.ongoing_download.done())
|
|
||||||
|
|
||||||
@property
|
|
||||||
def component(self) -> 'BackgroundDownloaderComponent':
|
|
||||||
return self
|
|
||||||
|
|
||||||
async def get_status(self):
|
|
||||||
return {'running': self.background_task is not None and not self.background_task.done(),
|
|
||||||
'available_free_space_mb': self.space_available,
|
|
||||||
'ongoing_download': self.is_busy}
|
|
||||||
|
|
||||||
async def download_blobs_in_background(self):
|
|
||||||
while True:
|
|
||||||
self.space_available = await self.space_manager.get_free_space_mb(True)
|
|
||||||
if not self.is_busy and self.space_available > 10:
|
|
||||||
self._download_next_close_blob_hash()
|
|
||||||
await asyncio.sleep(self.download_loop_delay_seconds)
|
|
||||||
|
|
||||||
def _download_next_close_blob_hash(self):
|
|
||||||
node_id = self.dht_node.protocol.node_id
|
|
||||||
for blob_hash in self.dht_node.stored_blob_hashes:
|
|
||||||
if blob_hash.hex() in self.blob_manager.completed_blob_hashes:
|
|
||||||
continue
|
|
||||||
if utils.get_colliding_prefix_bits(node_id, blob_hash) >= self.MIN_PREFIX_COLLIDING_BITS:
|
|
||||||
self.ongoing_download = asyncio.create_task(self.background_downloader.download_blobs(blob_hash.hex()))
|
|
||||||
return
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
self.space_manager: DiskSpaceManager = self.component_manager.get_component(DISK_SPACE_COMPONENT)
|
|
||||||
if not self.component_manager.has_component(DHT_COMPONENT):
|
|
||||||
return
|
|
||||||
self.dht_node = self.component_manager.get_component(DHT_COMPONENT)
|
|
||||||
self.blob_manager = self.component_manager.get_component(BLOB_COMPONENT)
|
|
||||||
storage = self.component_manager.get_component(DATABASE_COMPONENT)
|
|
||||||
self.background_downloader = BackgroundDownloader(self.conf, storage, self.blob_manager, self.dht_node)
|
|
||||||
self.background_task = asyncio.create_task(self.download_blobs_in_background())
|
|
||||||
|
|
||||||
async def stop(self):
|
|
||||||
if self.ongoing_download and not self.ongoing_download.done():
|
|
||||||
self.ongoing_download.cancel()
|
|
||||||
if self.background_task:
|
|
||||||
self.background_task.cancel()
|
|
||||||
|
|
||||||
|
|
||||||
class DiskSpaceComponent(Component):
|
class DiskSpaceComponent(Component):
|
||||||
|
@ -453,15 +391,11 @@ class DiskSpaceComponent(Component):
|
||||||
|
|
||||||
async def get_status(self):
|
async def get_status(self):
|
||||||
if self.disk_space_manager:
|
if self.disk_space_manager:
|
||||||
space_used = await self.disk_space_manager.get_space_used_mb(cached=True)
|
|
||||||
return {
|
return {
|
||||||
'total_used_mb': space_used['total'],
|
'space_used': str(await self.disk_space_manager.get_space_used_mb()),
|
||||||
'published_blobs_storage_used_mb': space_used['private_storage'],
|
|
||||||
'content_blobs_storage_used_mb': space_used['content_storage'],
|
|
||||||
'seed_blobs_storage_used_mb': space_used['network_storage'],
|
|
||||||
'running': self.disk_space_manager.running,
|
'running': self.disk_space_manager.running,
|
||||||
}
|
}
|
||||||
return {'space_used': '0', 'network_seeding_space_used': '0', 'running': False}
|
return {'space_used': '0', 'running': False}
|
||||||
|
|
||||||
async def start(self):
|
async def start(self):
|
||||||
db = self.component_manager.get_component(DATABASE_COMPONENT)
|
db = self.component_manager.get_component(DATABASE_COMPONENT)
|
||||||
|
@ -495,6 +429,7 @@ class TorrentComponent(Component):
|
||||||
}
|
}
|
||||||
|
|
||||||
async def start(self):
|
async def start(self):
|
||||||
|
if TorrentSession:
|
||||||
self.torrent_session = TorrentSession(asyncio.get_event_loop(), None)
|
self.torrent_session = TorrentSession(asyncio.get_event_loop(), None)
|
||||||
await self.torrent_session.bind() # TODO: specify host/port
|
await self.torrent_session.bind() # TODO: specify host/port
|
||||||
|
|
||||||
|
@ -551,7 +486,7 @@ class UPnPComponent(Component):
|
||||||
while True:
|
while True:
|
||||||
if now:
|
if now:
|
||||||
await self._maintain_redirects()
|
await self._maintain_redirects()
|
||||||
await asyncio.sleep(360)
|
await asyncio.sleep(360, loop=self.component_manager.loop)
|
||||||
|
|
||||||
async def _maintain_redirects(self):
|
async def _maintain_redirects(self):
|
||||||
# setup the gateway if necessary
|
# setup the gateway if necessary
|
||||||
|
@ -560,6 +495,8 @@ class UPnPComponent(Component):
|
||||||
self.upnp = await UPnP.discover(loop=self.component_manager.loop)
|
self.upnp = await UPnP.discover(loop=self.component_manager.loop)
|
||||||
log.info("found upnp gateway: %s", self.upnp.gateway.manufacturer_string)
|
log.info("found upnp gateway: %s", self.upnp.gateway.manufacturer_string)
|
||||||
except Exception as err:
|
except Exception as err:
|
||||||
|
if isinstance(err, asyncio.CancelledError): # TODO: remove when updated to 3.8
|
||||||
|
raise
|
||||||
log.warning("upnp discovery failed: %s", err)
|
log.warning("upnp discovery failed: %s", err)
|
||||||
self.upnp = None
|
self.upnp = None
|
||||||
|
|
||||||
|
@ -671,7 +608,7 @@ class UPnPComponent(Component):
|
||||||
log.info("Removing upnp redirects: %s", self.upnp_redirects)
|
log.info("Removing upnp redirects: %s", self.upnp_redirects)
|
||||||
await asyncio.wait([
|
await asyncio.wait([
|
||||||
self.upnp.delete_port_mapping(port, protocol) for protocol, port in self.upnp_redirects.items()
|
self.upnp.delete_port_mapping(port, protocol) for protocol, port in self.upnp_redirects.items()
|
||||||
])
|
], loop=self.component_manager.loop)
|
||||||
if self._maintain_redirects_task and not self._maintain_redirects_task.done():
|
if self._maintain_redirects_task and not self._maintain_redirects_task.done():
|
||||||
self._maintain_redirects_task.cancel()
|
self._maintain_redirects_task.cancel()
|
||||||
|
|
||||||
|
@ -702,49 +639,3 @@ class ExchangeRateManagerComponent(Component):
|
||||||
|
|
||||||
async def stop(self):
|
async def stop(self):
|
||||||
self.exchange_rate_manager.stop()
|
self.exchange_rate_manager.stop()
|
||||||
|
|
||||||
|
|
||||||
class TrackerAnnouncerComponent(Component):
|
|
||||||
component_name = TRACKER_ANNOUNCER_COMPONENT
|
|
||||||
depends_on = [FILE_MANAGER_COMPONENT]
|
|
||||||
|
|
||||||
def __init__(self, component_manager):
|
|
||||||
super().__init__(component_manager)
|
|
||||||
self.file_manager = None
|
|
||||||
self.announce_task = None
|
|
||||||
self.tracker_client: typing.Optional[TrackerClient] = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def component(self):
|
|
||||||
return self.tracker_client
|
|
||||||
|
|
||||||
@property
|
|
||||||
def running(self):
|
|
||||||
return self._running and self.announce_task and not self.announce_task.done()
|
|
||||||
|
|
||||||
async def announce_forever(self):
|
|
||||||
while True:
|
|
||||||
sleep_seconds = 60.0
|
|
||||||
announce_sd_hashes = []
|
|
||||||
for file in self.file_manager.get_filtered():
|
|
||||||
if not file.downloader:
|
|
||||||
continue
|
|
||||||
announce_sd_hashes.append(bytes.fromhex(file.sd_hash))
|
|
||||||
await self.tracker_client.announce_many(*announce_sd_hashes)
|
|
||||||
await asyncio.sleep(sleep_seconds)
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
node = self.component_manager.get_component(DHT_COMPONENT) \
|
|
||||||
if self.component_manager.has_component(DHT_COMPONENT) else None
|
|
||||||
node_id = node.protocol.node_id if node else None
|
|
||||||
self.tracker_client = TrackerClient(node_id, self.conf.tcp_port, lambda: self.conf.tracker_servers)
|
|
||||||
await self.tracker_client.start()
|
|
||||||
self.file_manager = self.component_manager.get_component(FILE_MANAGER_COMPONENT)
|
|
||||||
self.announce_task = asyncio.create_task(self.announce_forever())
|
|
||||||
|
|
||||||
async def stop(self):
|
|
||||||
self.file_manager = None
|
|
||||||
if self.announce_task and not self.announce_task.done():
|
|
||||||
self.announce_task.cancel()
|
|
||||||
self.announce_task = None
|
|
||||||
self.tracker_client.stop()
|
|
||||||
|
|
|
@ -8,14 +8,16 @@ import time
|
||||||
import inspect
|
import inspect
|
||||||
import typing
|
import typing
|
||||||
import random
|
import random
|
||||||
|
import hashlib
|
||||||
import tracemalloc
|
import tracemalloc
|
||||||
import itertools
|
from decimal import Decimal
|
||||||
from urllib.parse import urlencode, quote
|
from urllib.parse import urlencode, quote
|
||||||
from typing import Callable, Optional, List
|
from typing import Callable, Optional, List
|
||||||
from binascii import hexlify, unhexlify
|
from binascii import hexlify, unhexlify
|
||||||
from traceback import format_exc
|
from traceback import format_exc
|
||||||
from functools import wraps, partial
|
from functools import wraps, partial
|
||||||
|
|
||||||
|
import ecdsa
|
||||||
import base58
|
import base58
|
||||||
from aiohttp import web
|
from aiohttp import web
|
||||||
from prometheus_client import generate_latest as prom_generate_latest, Gauge, Histogram, Counter
|
from prometheus_client import generate_latest as prom_generate_latest, Gauge, Histogram, Counter
|
||||||
|
@ -27,8 +29,6 @@ from lbry.wallet import (
|
||||||
)
|
)
|
||||||
from lbry.wallet.dewies import dewies_to_lbc, lbc_to_dewies, dict_values_to_lbc
|
from lbry.wallet.dewies import dewies_to_lbc, lbc_to_dewies, dict_values_to_lbc
|
||||||
from lbry.wallet.constants import TXO_TYPES, CLAIM_TYPE_NAMES
|
from lbry.wallet.constants import TXO_TYPES, CLAIM_TYPE_NAMES
|
||||||
from lbry.wallet.bip32 import PrivateKey
|
|
||||||
from lbry.crypto.base58 import Base58
|
|
||||||
|
|
||||||
from lbry import utils
|
from lbry import utils
|
||||||
from lbry.conf import Config, Setting, NOT_SET
|
from lbry.conf import Config, Setting, NOT_SET
|
||||||
|
@ -38,13 +38,12 @@ from lbry.dht.peer import make_kademlia_peer
|
||||||
from lbry.error import (
|
from lbry.error import (
|
||||||
DownloadSDTimeoutError, ComponentsNotStartedError, ComponentStartConditionNotMetError,
|
DownloadSDTimeoutError, ComponentsNotStartedError, ComponentStartConditionNotMetError,
|
||||||
CommandDoesNotExistError, BaseError, WalletNotFoundError, WalletAlreadyLoadedError, WalletAlreadyExistsError,
|
CommandDoesNotExistError, BaseError, WalletNotFoundError, WalletAlreadyLoadedError, WalletAlreadyExistsError,
|
||||||
ConflictingInputValueError, AlreadyPurchasedError, PrivateKeyNotFoundError, InputStringIsBlankError,
|
ConflictingInputValueError, AlreadyPurchasedError, PrivateKeyNotFoundError, InputStringIsBlankError
|
||||||
InputValueError
|
|
||||||
)
|
)
|
||||||
from lbry.extras import system_info
|
from lbry.extras import system_info
|
||||||
from lbry.extras.daemon import analytics
|
from lbry.extras.daemon import analytics
|
||||||
from lbry.extras.daemon.components import WALLET_COMPONENT, DATABASE_COMPONENT, DHT_COMPONENT, BLOB_COMPONENT
|
from lbry.extras.daemon.components import WALLET_COMPONENT, DATABASE_COMPONENT, DHT_COMPONENT, BLOB_COMPONENT
|
||||||
from lbry.extras.daemon.components import FILE_MANAGER_COMPONENT, DISK_SPACE_COMPONENT, TRACKER_ANNOUNCER_COMPONENT
|
from lbry.extras.daemon.components import FILE_MANAGER_COMPONENT, DISK_SPACE_COMPONENT
|
||||||
from lbry.extras.daemon.components import EXCHANGE_RATE_MANAGER_COMPONENT, UPNP_COMPONENT
|
from lbry.extras.daemon.components import EXCHANGE_RATE_MANAGER_COMPONENT, UPNP_COMPONENT
|
||||||
from lbry.extras.daemon.componentmanager import RequiredCondition
|
from lbry.extras.daemon.componentmanager import RequiredCondition
|
||||||
from lbry.extras.daemon.componentmanager import ComponentManager
|
from lbry.extras.daemon.componentmanager import ComponentManager
|
||||||
|
@ -53,8 +52,9 @@ from lbry.extras.daemon.undecorated import undecorated
|
||||||
from lbry.extras.daemon.security import ensure_request_allowed
|
from lbry.extras.daemon.security import ensure_request_allowed
|
||||||
from lbry.file_analysis import VideoFileAnalyzer
|
from lbry.file_analysis import VideoFileAnalyzer
|
||||||
from lbry.schema.claim import Claim
|
from lbry.schema.claim import Claim
|
||||||
from lbry.schema.url import URL
|
from lbry.schema.url import URL, normalize_name
|
||||||
|
from lbry.wallet.server.db.elasticsearch.constants import RANGE_FIELDS, REPLACEMENTS
|
||||||
|
MY_RANGE_FIELDS = RANGE_FIELDS - {"limit_claims_per_channel"}
|
||||||
|
|
||||||
if typing.TYPE_CHECKING:
|
if typing.TYPE_CHECKING:
|
||||||
from lbry.blob.blob_manager import BlobManager
|
from lbry.blob.blob_manager import BlobManager
|
||||||
|
@ -67,29 +67,6 @@ if typing.TYPE_CHECKING:
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
RANGE_FIELDS = {
|
|
||||||
'height', 'creation_height', 'activation_height', 'expiration_height',
|
|
||||||
'timestamp', 'creation_timestamp', 'duration', 'release_time', 'fee_amount',
|
|
||||||
'tx_position', 'repost_count', 'limit_claims_per_channel',
|
|
||||||
'amount', 'effective_amount', 'support_amount',
|
|
||||||
'trending_score', 'censor_type', 'tx_num'
|
|
||||||
}
|
|
||||||
MY_RANGE_FIELDS = RANGE_FIELDS - {"limit_claims_per_channel"}
|
|
||||||
REPLACEMENTS = {
|
|
||||||
'claim_name': 'normalized_name',
|
|
||||||
'name': 'normalized_name',
|
|
||||||
'txid': 'tx_id',
|
|
||||||
'nout': 'tx_nout',
|
|
||||||
'trending_group': 'trending_score',
|
|
||||||
'trending_mixed': 'trending_score',
|
|
||||||
'trending_global': 'trending_score',
|
|
||||||
'trending_local': 'trending_score',
|
|
||||||
'reposted': 'repost_count',
|
|
||||||
'stream_types': 'stream_type',
|
|
||||||
'media_types': 'media_type',
|
|
||||||
'valid_channel_signature': 'is_signature_valid'
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def is_transactional_function(name):
|
def is_transactional_function(name):
|
||||||
for action in ('create', 'update', 'abandon', 'send', 'fund'):
|
for action in ('create', 'update', 'abandon', 'send', 'fund'):
|
||||||
|
@ -195,6 +172,48 @@ def paginate_list(items: List, page: Optional[int], page_size: Optional[int]):
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def fix_kwargs_for_hub(**kwargs):
|
||||||
|
repeated_fields = {"media_type", "stream_type", "claim_type"}
|
||||||
|
value_fields = {"tx_nout", "has_source", "is_signature_valid"}
|
||||||
|
opcodes = {'=': 0, '<=': 1, '>=': 2, '<': 3, '>': 4}
|
||||||
|
for key, value in list(kwargs.items()):
|
||||||
|
if value in (None, [], False):
|
||||||
|
kwargs.pop(key)
|
||||||
|
continue
|
||||||
|
if key in REPLACEMENTS:
|
||||||
|
kwargs[REPLACEMENTS[key]] = kwargs.pop(key)
|
||||||
|
key = REPLACEMENTS[key]
|
||||||
|
|
||||||
|
if key == "normalized_name":
|
||||||
|
kwargs[key] = normalize_name(value)
|
||||||
|
if key == "limit_claims_per_channel":
|
||||||
|
value = kwargs.pop("limit_claims_per_channel") or 0
|
||||||
|
if value > 0:
|
||||||
|
kwargs["limit_claims_per_channel"] = value
|
||||||
|
elif key == "invalid_channel_signature":
|
||||||
|
kwargs["is_signature_valid"] = {"value": not kwargs.pop("invalid_channel_signature")}
|
||||||
|
elif key == "has_no_source":
|
||||||
|
kwargs["has_source"] = {"value": not kwargs.pop("has_no_source")}
|
||||||
|
elif key in value_fields:
|
||||||
|
kwargs[key] = {"value": value} if not isinstance(value, dict) else value
|
||||||
|
elif key in repeated_fields and isinstance(value, str):
|
||||||
|
kwargs[key] = [value]
|
||||||
|
elif key in ("claim_id", "channel_id"):
|
||||||
|
kwargs[key] = {"invert": False, "value": [kwargs[key]]}
|
||||||
|
elif key in ("claim_ids", "channel_ids"):
|
||||||
|
kwargs[key[:-1]] = {"invert": False, "value": kwargs.pop(key)}
|
||||||
|
elif key == "not_channel_ids":
|
||||||
|
kwargs["channel_id"] = {"invert": True, "value": kwargs.pop("not_channel_ids")}
|
||||||
|
elif key in MY_RANGE_FIELDS:
|
||||||
|
operator = '='
|
||||||
|
if isinstance(value, str) and value[0] in opcodes:
|
||||||
|
operator_length = 2 if value[:2] in opcodes else 1
|
||||||
|
operator, value = value[:operator_length], value[operator_length:]
|
||||||
|
value = [str(value if key != 'fee_amount' else Decimal(value)*1000)]
|
||||||
|
kwargs[key] = {"op": opcodes[operator], "value": value}
|
||||||
|
return kwargs
|
||||||
|
|
||||||
|
|
||||||
DHT_HAS_CONTACTS = "dht_has_contacts"
|
DHT_HAS_CONTACTS = "dht_has_contacts"
|
||||||
|
|
||||||
|
|
||||||
|
@ -614,8 +633,7 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
content_type='application/json'
|
content_type='application/json'
|
||||||
)
|
)
|
||||||
|
|
||||||
@staticmethod
|
async def handle_metrics_get_request(self, request: web.Request):
|
||||||
async def handle_metrics_get_request(request: web.Request):
|
|
||||||
try:
|
try:
|
||||||
return web.Response(
|
return web.Response(
|
||||||
text=prom_generate_latest().decode(),
|
text=prom_generate_latest().decode(),
|
||||||
|
@ -933,12 +951,7 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
},
|
},
|
||||||
'total_outgoing_mps': (float) megabytes per second sent,
|
'total_outgoing_mps': (float) megabytes per second sent,
|
||||||
'total_incoming_mps': (float) megabytes per second received,
|
'total_incoming_mps': (float) megabytes per second received,
|
||||||
'max_outgoing_mbs': (float) maximum bandwidth (megabytes per second) sent, since the
|
'time': (float) timestamp
|
||||||
daemon was started
|
|
||||||
'max_incoming_mbs': (float) maximum bandwidth (megabytes per second) received, since the
|
|
||||||
daemon was started
|
|
||||||
'total_sent' : (int) total number of bytes sent since the daemon was started
|
|
||||||
'total_received' : (int) total number of bytes received since the daemon was started
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
'hash_announcer': {
|
'hash_announcer': {
|
||||||
|
@ -1328,65 +1341,6 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
c.wallets += [wallet_id]
|
c.wallets += [wallet_id]
|
||||||
return wallet
|
return wallet
|
||||||
|
|
||||||
@requires("wallet")
|
|
||||||
async def jsonrpc_wallet_export(self, password=None, wallet_id=None):
|
|
||||||
"""
|
|
||||||
Exports encrypted wallet data if password is supplied; otherwise plain JSON.
|
|
||||||
|
|
||||||
Wallet must be unlocked to perform this operation.
|
|
||||||
|
|
||||||
Usage:
|
|
||||||
wallet_export [--password=<password>] [--wallet_id=<wallet_id>]
|
|
||||||
|
|
||||||
Options:
|
|
||||||
--password=<password> : (str) password to encrypt outgoing data
|
|
||||||
--wallet_id=<wallet_id> : (str) wallet being exported
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
(str) data: base64-encoded encrypted wallet, or cleartext JSON
|
|
||||||
|
|
||||||
"""
|
|
||||||
wallet = self.wallet_manager.get_wallet_or_default(wallet_id)
|
|
||||||
if password is None:
|
|
||||||
return wallet.to_json()
|
|
||||||
return wallet.pack(password).decode()
|
|
||||||
|
|
||||||
@requires("wallet")
|
|
||||||
async def jsonrpc_wallet_import(self, data, password=None, wallet_id=None, blocking=False):
|
|
||||||
"""
|
|
||||||
Import wallet data and merge accounts and preferences. Data is expected to be JSON if
|
|
||||||
password is not supplied.
|
|
||||||
|
|
||||||
Wallet must be unlocked to perform this operation.
|
|
||||||
|
|
||||||
Usage:
|
|
||||||
wallet_import (<data> | --data=<data>) [<password> | --password=<password>]
|
|
||||||
[--wallet_id=<wallet_id>] [--blocking]
|
|
||||||
|
|
||||||
Options:
|
|
||||||
--data=<data> : (str) incoming wallet data
|
|
||||||
--password=<password> : (str) password to decrypt incoming data
|
|
||||||
--wallet_id=<wallet_id> : (str) wallet being merged into
|
|
||||||
--blocking : (bool) wait until any new accounts have merged
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
(str) base64-encoded encrypted wallet, or cleartext JSON
|
|
||||||
"""
|
|
||||||
wallet = self.wallet_manager.get_wallet_or_default(wallet_id)
|
|
||||||
added_accounts, merged_accounts = wallet.merge(self.wallet_manager, password, data)
|
|
||||||
for new_account in itertools.chain(added_accounts, merged_accounts):
|
|
||||||
await new_account.maybe_migrate_certificates()
|
|
||||||
if added_accounts and self.ledger.network.is_connected:
|
|
||||||
if blocking:
|
|
||||||
await asyncio.wait([
|
|
||||||
a.ledger.subscribe_account(a) for a in added_accounts
|
|
||||||
])
|
|
||||||
else:
|
|
||||||
for new_account in added_accounts:
|
|
||||||
asyncio.create_task(self.ledger.subscribe_account(new_account))
|
|
||||||
wallet.save()
|
|
||||||
return await self.jsonrpc_wallet_export(password=password, wallet_id=wallet_id)
|
|
||||||
|
|
||||||
@requires("wallet")
|
@requires("wallet")
|
||||||
async def jsonrpc_wallet_add(self, wallet_id):
|
async def jsonrpc_wallet_add(self, wallet_id):
|
||||||
"""
|
"""
|
||||||
|
@ -1855,7 +1809,7 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
Options:
|
Options:
|
||||||
--to_account=<to_account> : (str) send to this account
|
--to_account=<to_account> : (str) send to this account
|
||||||
--from_account=<from_account> : (str) spend from this account
|
--from_account=<from_account> : (str) spend from this account
|
||||||
--amount=<amount> : (decimal) the amount to transfer lbc
|
--amount=<amount> : (str) the amount to transfer lbc
|
||||||
--everything : (bool) transfer everything (excluding claims), default: false.
|
--everything : (bool) transfer everything (excluding claims), default: false.
|
||||||
--outputs=<outputs> : (int) split payment across many outputs, default: 1.
|
--outputs=<outputs> : (int) split payment across many outputs, default: 1.
|
||||||
--wallet_id=<wallet_id> : (str) limit operation to specific wallet.
|
--wallet_id=<wallet_id> : (str) limit operation to specific wallet.
|
||||||
|
@ -1878,48 +1832,6 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
outputs=outputs, broadcast=broadcast
|
outputs=outputs, broadcast=broadcast
|
||||||
)
|
)
|
||||||
|
|
||||||
@requires("wallet")
|
|
||||||
async def jsonrpc_account_deposit(
|
|
||||||
self, txid, nout, redeem_script, private_key,
|
|
||||||
to_account=None, wallet_id=None, preview=False, blocking=False
|
|
||||||
):
|
|
||||||
"""
|
|
||||||
Spend a time locked transaction into your account.
|
|
||||||
|
|
||||||
Usage:
|
|
||||||
account_deposit <txid> <nout> <redeem_script> <private_key>
|
|
||||||
[<to_account> | --to_account=<to_account>]
|
|
||||||
[--wallet_id=<wallet_id>] [--preview] [--blocking]
|
|
||||||
|
|
||||||
Options:
|
|
||||||
--txid=<txid> : (str) id of the transaction
|
|
||||||
--nout=<nout> : (int) output number in the transaction
|
|
||||||
--redeem_script=<redeem_script> : (str) redeem script for output
|
|
||||||
--private_key=<private_key> : (str) private key to sign transaction
|
|
||||||
--to_account=<to_account> : (str) deposit to this account
|
|
||||||
--wallet_id=<wallet_id> : (str) limit operation to specific wallet.
|
|
||||||
--preview : (bool) do not broadcast the transaction
|
|
||||||
--blocking : (bool) wait until tx has synced
|
|
||||||
|
|
||||||
Returns: {Transaction}
|
|
||||||
"""
|
|
||||||
wallet = self.wallet_manager.get_wallet_or_default(wallet_id)
|
|
||||||
account = wallet.get_account_or_default(to_account)
|
|
||||||
other_tx = await self.wallet_manager.get_transaction(txid)
|
|
||||||
tx = await Transaction.spend_time_lock(
|
|
||||||
other_tx.outputs[nout], unhexlify(redeem_script), account
|
|
||||||
)
|
|
||||||
pk = PrivateKey.from_bytes(
|
|
||||||
account.ledger, Base58.decode_check(private_key)[1:-1]
|
|
||||||
)
|
|
||||||
await tx.sign([account], {pk.address: pk})
|
|
||||||
if not preview:
|
|
||||||
await self.broadcast_or_release(tx, blocking)
|
|
||||||
self.component_manager.loop.create_task(self.analytics_manager.send_credits_sent())
|
|
||||||
else:
|
|
||||||
await self.ledger.release_tx(tx)
|
|
||||||
return tx
|
|
||||||
|
|
||||||
@requires(WALLET_COMPONENT)
|
@requires(WALLET_COMPONENT)
|
||||||
def jsonrpc_account_send(self, amount, addresses, account_id=None, wallet_id=None, preview=False, blocking=False):
|
def jsonrpc_account_send(self, amount, addresses, account_id=None, wallet_id=None, preview=False, blocking=False):
|
||||||
"""
|
"""
|
||||||
|
@ -1991,9 +1903,7 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
wallet = self.wallet_manager.get_wallet_or_default(wallet_id)
|
wallet = self.wallet_manager.get_wallet_or_default(wallet_id)
|
||||||
wallet_changed = False
|
wallet_changed = False
|
||||||
if data is not None:
|
if data is not None:
|
||||||
added_accounts, merged_accounts = wallet.merge(self.wallet_manager, password, data)
|
added_accounts = wallet.merge(self.wallet_manager, password, data)
|
||||||
for new_account in itertools.chain(added_accounts, merged_accounts):
|
|
||||||
await new_account.maybe_migrate_certificates()
|
|
||||||
if added_accounts and self.ledger.network.is_connected:
|
if added_accounts and self.ledger.network.is_connected:
|
||||||
if blocking:
|
if blocking:
|
||||||
await asyncio.wait([
|
await asyncio.wait([
|
||||||
|
@ -2410,7 +2320,6 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
|
|
||||||
Usage:
|
Usage:
|
||||||
claim_list [--claim_type=<claim_type>...] [--claim_id=<claim_id>...] [--name=<name>...] [--is_spent]
|
claim_list [--claim_type=<claim_type>...] [--claim_id=<claim_id>...] [--name=<name>...] [--is_spent]
|
||||||
[--reposted_claim_id=<reposted_claim_id>...]
|
|
||||||
[--channel_id=<channel_id>...] [--account_id=<account_id>] [--wallet_id=<wallet_id>]
|
[--channel_id=<channel_id>...] [--account_id=<account_id>] [--wallet_id=<wallet_id>]
|
||||||
[--has_source | --has_no_source] [--page=<page>] [--page_size=<page_size>]
|
[--has_source | --has_no_source] [--page=<page>] [--page_size=<page_size>]
|
||||||
[--resolve] [--order_by=<order_by>] [--no_totals] [--include_received_tips]
|
[--resolve] [--order_by=<order_by>] [--no_totals] [--include_received_tips]
|
||||||
|
@ -2421,7 +2330,6 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
--channel_id=<channel_id> : (str or list) streams in this channel
|
--channel_id=<channel_id> : (str or list) streams in this channel
|
||||||
--name=<name> : (str or list) claim name
|
--name=<name> : (str or list) claim name
|
||||||
--is_spent : (bool) shows previous claim updates and abandons
|
--is_spent : (bool) shows previous claim updates and abandons
|
||||||
--reposted_claim_id=<reposted_claim_id> : (str or list) reposted claim id
|
|
||||||
--account_id=<account_id> : (str) id of the account to query
|
--account_id=<account_id> : (str) id of the account to query
|
||||||
--wallet_id=<wallet_id> : (str) restrict results to specific wallet
|
--wallet_id=<wallet_id> : (str) restrict results to specific wallet
|
||||||
--has_source : (bool) list claims containing a source field
|
--has_source : (bool) list claims containing a source field
|
||||||
|
@ -2485,9 +2393,6 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
value with an equality constraint such as '>', '>=', '<' and '<='
|
value with an equality constraint such as '>', '>=', '<' and '<='
|
||||||
eg. --height=">400000" would limit results to only claims above 400k block height.
|
eg. --height=">400000" would limit results to only claims above 400k block height.
|
||||||
|
|
||||||
They also support multiple constraints passed as a list of the args described above.
|
|
||||||
eg. --release_time=[">1000000", "<2000000"]
|
|
||||||
|
|
||||||
Usage:
|
Usage:
|
||||||
claim_search [<name> | --name=<name>] [--text=<text>] [--txid=<txid>] [--nout=<nout>]
|
claim_search [<name> | --name=<name>] [--text=<text>] [--txid=<txid>] [--nout=<nout>]
|
||||||
[--claim_id=<claim_id> | --claim_ids=<claim_ids>...]
|
[--claim_id=<claim_id> | --claim_ids=<claim_ids>...]
|
||||||
|
@ -2502,7 +2407,7 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
[--amount=<amount>] [--effective_amount=<effective_amount>]
|
[--amount=<amount>] [--effective_amount=<effective_amount>]
|
||||||
[--support_amount=<support_amount>] [--trending_group=<trending_group>]
|
[--support_amount=<support_amount>] [--trending_group=<trending_group>]
|
||||||
[--trending_mixed=<trending_mixed>] [--trending_local=<trending_local>]
|
[--trending_mixed=<trending_mixed>] [--trending_local=<trending_local>]
|
||||||
[--trending_global=<trending_global] [--trending_score=<trending_score]
|
[--trending_global=<trending_global]
|
||||||
[--reposted_claim_id=<reposted_claim_id>] [--reposted=<reposted>]
|
[--reposted_claim_id=<reposted_claim_id>] [--reposted=<reposted>]
|
||||||
[--claim_type=<claim_type>] [--stream_types=<stream_types>...] [--media_types=<media_types>...]
|
[--claim_type=<claim_type>] [--stream_types=<stream_types>...] [--media_types=<media_types>...]
|
||||||
[--fee_currency=<fee_currency>] [--fee_amount=<fee_amount>]
|
[--fee_currency=<fee_currency>] [--fee_amount=<fee_amount>]
|
||||||
|
@ -2514,7 +2419,7 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
[--not_locations=<not_locations>...]
|
[--not_locations=<not_locations>...]
|
||||||
[--order_by=<order_by>...] [--no_totals] [--page=<page>] [--page_size=<page_size>]
|
[--order_by=<order_by>...] [--no_totals] [--page=<page>] [--page_size=<page_size>]
|
||||||
[--wallet_id=<wallet_id>] [--include_purchase_receipt] [--include_is_my_output]
|
[--wallet_id=<wallet_id>] [--include_purchase_receipt] [--include_is_my_output]
|
||||||
[--remove_duplicates] [--has_source | --has_no_source] [--sd_hash=<sd_hash>]
|
[--remove_duplicates] [--has_source | --has_no_source]
|
||||||
[--new_sdk_server=<new_sdk_server>]
|
[--new_sdk_server=<new_sdk_server>]
|
||||||
|
|
||||||
Options:
|
Options:
|
||||||
|
@ -2570,11 +2475,25 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
all tips and supports received), this amount is
|
all tips and supports received), this amount is
|
||||||
blank until claim has reached activation height
|
blank until claim has reached activation height
|
||||||
(supports equality constraints)
|
(supports equality constraints)
|
||||||
--trending_score=<trending_score>: (int) limit by trending score (supports equality constraints)
|
--trending_group=<trending_group>: (int) group numbers 1 through 4 representing the
|
||||||
--trending_group=<trending_group>: (int) DEPRECATED - instead please use trending_score
|
trending groups of the content: 4 means
|
||||||
--trending_mixed=<trending_mixed>: (int) DEPRECATED - instead please use trending_score
|
content is trending globally and independently,
|
||||||
--trending_local=<trending_local>: (int) DEPRECATED - instead please use trending_score
|
3 means content is not trending globally but is
|
||||||
--trending_global=<trending_global>: (int) DEPRECATED - instead please use trending_score
|
trending independently (locally), 2 means it is
|
||||||
|
trending globally but not independently and 1
|
||||||
|
means it's not trending globally or locally
|
||||||
|
(supports equality constraints)
|
||||||
|
--trending_mixed=<trending_mixed>: (int) trending amount taken from the global or local
|
||||||
|
value depending on the trending group:
|
||||||
|
4 - global value, 3 - local value, 2 - global
|
||||||
|
value, 1 - local value (supports equality
|
||||||
|
constraints)
|
||||||
|
--trending_local=<trending_local>: (int) trending value calculated relative only to
|
||||||
|
the individual contents past history (supports
|
||||||
|
equality constraints)
|
||||||
|
--trending_global=<trending_global>: (int) trending value calculated relative to all
|
||||||
|
trending content globally (supports
|
||||||
|
equality constraints)
|
||||||
--reposted_claim_id=<reposted_claim_id>: (str) all reposts of the specified original claim id
|
--reposted_claim_id=<reposted_claim_id>: (str) all reposts of the specified original claim id
|
||||||
--reposted=<reposted> : (int) claims reposted this many times (supports
|
--reposted=<reposted> : (int) claims reposted this many times (supports
|
||||||
equality constraints)
|
equality constraints)
|
||||||
|
@ -2612,13 +2531,26 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
--remove_duplicates : (bool) removes duplicated content from search by picking either the
|
--remove_duplicates : (bool) removes duplicated content from search by picking either the
|
||||||
original claim or the oldest matching repost
|
original claim or the oldest matching repost
|
||||||
--has_source : (bool) find claims containing a source field
|
--has_source : (bool) find claims containing a source field
|
||||||
--sd_hash=<sd_hash> : (str) find claims where the source stream descriptor hash matches
|
|
||||||
(partially or completely) the given hexadecimal string
|
|
||||||
--has_no_source : (bool) find claims not containing a source field
|
--has_no_source : (bool) find claims not containing a source field
|
||||||
--new_sdk_server=<new_sdk_server> : (str) URL of the new SDK server (EXPERIMENTAL)
|
--new_sdk_server=<new_sdk_server> : (str) URL of the new SDK server (EXPERIMENTAL)
|
||||||
|
|
||||||
Returns: {Paginated[Output]}
|
Returns: {Paginated[Output]}
|
||||||
"""
|
"""
|
||||||
|
if self.ledger.config.get('use_go_hub'):
|
||||||
|
host = self.ledger.network.client.server[0]
|
||||||
|
port = "50051"
|
||||||
|
kwargs['new_sdk_server'] = f"{host}:{port}"
|
||||||
|
if kwargs.get("channel"):
|
||||||
|
channel = kwargs.pop("channel")
|
||||||
|
channel_obj = (await self.jsonrpc_resolve(channel))[channel]
|
||||||
|
if isinstance(channel_obj, dict):
|
||||||
|
# This happens when the channel doesn't exist
|
||||||
|
kwargs["channel_id"] = ""
|
||||||
|
else:
|
||||||
|
kwargs["channel_id"] = channel_obj.claim_id
|
||||||
|
kwargs = fix_kwargs_for_hub(**kwargs)
|
||||||
|
else:
|
||||||
|
# Don't do this if using the hub server, it screws everything up
|
||||||
if "claim_ids" in kwargs and not kwargs["claim_ids"]:
|
if "claim_ids" in kwargs and not kwargs["claim_ids"]:
|
||||||
kwargs.pop("claim_ids")
|
kwargs.pop("claim_ids")
|
||||||
if {'claim_id', 'claim_ids'}.issubset(kwargs):
|
if {'claim_id', 'claim_ids'}.issubset(kwargs):
|
||||||
|
@ -2629,17 +2561,6 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
kwargs['signature_valid'] = 0
|
kwargs['signature_valid'] = 0
|
||||||
if 'has_no_source' in kwargs:
|
if 'has_no_source' in kwargs:
|
||||||
kwargs['has_source'] = not kwargs.pop('has_no_source')
|
kwargs['has_source'] = not kwargs.pop('has_no_source')
|
||||||
if 'order_by' in kwargs: # TODO: remove this after removing support for old trending args from the api
|
|
||||||
value = kwargs.pop('order_by')
|
|
||||||
value = value if isinstance(value, list) else [value]
|
|
||||||
new_value = []
|
|
||||||
for new_v in value:
|
|
||||||
migrated = new_v if new_v not in (
|
|
||||||
'trending_mixed', 'trending_local', 'trending_global', 'trending_group'
|
|
||||||
) else 'trending_score'
|
|
||||||
if migrated not in new_value:
|
|
||||||
new_value.append(migrated)
|
|
||||||
kwargs['order_by'] = new_value
|
|
||||||
page_num, page_size = abs(kwargs.pop('page', 1)), min(abs(kwargs.pop('page_size', DEFAULT_PAGE_SIZE)), 50)
|
page_num, page_size = abs(kwargs.pop('page', 1)), min(abs(kwargs.pop('page_size', DEFAULT_PAGE_SIZE)), 50)
|
||||||
wallet = self.wallet_manager.get_wallet_or_default(kwargs.pop('wallet_id', None))
|
wallet = self.wallet_manager.get_wallet_or_default(kwargs.pop('wallet_id', None))
|
||||||
kwargs.update({'offset': page_size * (page_num - 1), 'limit': page_size})
|
kwargs.update({'offset': page_size * (page_num - 1), 'limit': page_size})
|
||||||
|
@ -2765,17 +2686,16 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
name, claim, amount, claim_address, funding_accounts, funding_accounts[0]
|
name, claim, amount, claim_address, funding_accounts, funding_accounts[0]
|
||||||
)
|
)
|
||||||
txo = tx.outputs[0]
|
txo = tx.outputs[0]
|
||||||
txo.set_channel_private_key(
|
await txo.generate_channel_private_key()
|
||||||
await funding_accounts[0].generate_channel_private_key()
|
|
||||||
)
|
|
||||||
|
|
||||||
await tx.sign(funding_accounts)
|
await tx.sign(funding_accounts)
|
||||||
|
|
||||||
if not preview:
|
if not preview:
|
||||||
|
account.add_channel_private_key(txo.private_key)
|
||||||
wallet.save()
|
wallet.save()
|
||||||
await self.broadcast_or_release(tx, blocking)
|
await self.broadcast_or_release(tx, blocking)
|
||||||
self.component_manager.loop.create_task(self.storage.save_claims([self._old_get_temp_claim_info(
|
self.component_manager.loop.create_task(self.storage.save_claims([self._old_get_temp_claim_info(
|
||||||
tx, txo, claim_address, claim, name
|
tx, txo, claim_address, claim, name, dewies_to_lbc(amount)
|
||||||
)]))
|
)]))
|
||||||
self.component_manager.loop.create_task(self.analytics_manager.send_new_channel())
|
self.component_manager.loop.create_task(self.analytics_manager.send_new_channel())
|
||||||
else:
|
else:
|
||||||
|
@ -2920,9 +2840,7 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
new_txo = tx.outputs[0]
|
new_txo = tx.outputs[0]
|
||||||
|
|
||||||
if new_signing_key:
|
if new_signing_key:
|
||||||
new_txo.set_channel_private_key(
|
await new_txo.generate_channel_private_key()
|
||||||
await funding_accounts[0].generate_channel_private_key()
|
|
||||||
)
|
|
||||||
else:
|
else:
|
||||||
new_txo.private_key = old_txo.private_key
|
new_txo.private_key = old_txo.private_key
|
||||||
|
|
||||||
|
@ -2931,10 +2849,11 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
await tx.sign(funding_accounts)
|
await tx.sign(funding_accounts)
|
||||||
|
|
||||||
if not preview:
|
if not preview:
|
||||||
|
account.add_channel_private_key(new_txo.private_key)
|
||||||
wallet.save()
|
wallet.save()
|
||||||
await self.broadcast_or_release(tx, blocking)
|
await self.broadcast_or_release(tx, blocking)
|
||||||
self.component_manager.loop.create_task(self.storage.save_claims([self._old_get_temp_claim_info(
|
self.component_manager.loop.create_task(self.storage.save_claims([self._old_get_temp_claim_info(
|
||||||
tx, new_txo, claim_address, new_txo.claim, new_txo.claim_name
|
tx, new_txo, claim_address, new_txo.claim, new_txo.claim_name, dewies_to_lbc(amount)
|
||||||
)]))
|
)]))
|
||||||
self.component_manager.loop.create_task(self.analytics_manager.send_new_channel())
|
self.component_manager.loop.create_task(self.analytics_manager.send_new_channel())
|
||||||
else:
|
else:
|
||||||
|
@ -2944,21 +2863,19 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
|
|
||||||
@requires(WALLET_COMPONENT)
|
@requires(WALLET_COMPONENT)
|
||||||
async def jsonrpc_channel_sign(
|
async def jsonrpc_channel_sign(
|
||||||
self, channel_name=None, channel_id=None, hexdata=None, salt=None,
|
self, channel_name=None, channel_id=None, hexdata=None, channel_account_id=None, wallet_id=None):
|
||||||
channel_account_id=None, wallet_id=None):
|
|
||||||
"""
|
"""
|
||||||
Signs data using the specified channel signing key.
|
Signs data using the specified channel signing key.
|
||||||
|
|
||||||
Usage:
|
Usage:
|
||||||
channel_sign [<channel_name> | --channel_name=<channel_name>] [<channel_id> | --channel_id=<channel_id>]
|
channel_sign [<channel_name> | --channel_name=<channel_name>]
|
||||||
[<hexdata> | --hexdata=<hexdata>] [<salt> | --salt=<salt>]
|
[<channel_id> | --channel_id=<channel_id>] [<hexdata> | --hexdata=<hexdata>]
|
||||||
[--channel_account_id=<channel_account_id>...] [--wallet_id=<wallet_id>]
|
[--channel_account_id=<channel_account_id>...] [--wallet_id=<wallet_id>]
|
||||||
|
|
||||||
Options:
|
Options:
|
||||||
--channel_name=<channel_name> : (str) name of channel used to sign (or use channel id)
|
--channel_name=<channel_name> : (str) name of channel used to sign (or use channel id)
|
||||||
--channel_id=<channel_id> : (str) claim id of channel used to sign (or use channel name)
|
--channel_id=<channel_id> : (str) claim id of channel used to sign (or use channel name)
|
||||||
--hexdata=<hexdata> : (str) data to sign, encoded as hexadecimal
|
--hexdata=<hexdata> : (str) data to sign, encoded as hexadecimal
|
||||||
--salt=<salt> : (str) salt to use for signing, default is to use timestamp
|
|
||||||
--channel_account_id=<channel_account_id>: (str) one or more account ids for accounts to look in
|
--channel_account_id=<channel_account_id>: (str) one or more account ids for accounts to look in
|
||||||
for channel certificates, defaults to all accounts.
|
for channel certificates, defaults to all accounts.
|
||||||
--wallet_id=<wallet_id> : (str) restrict operation to specific wallet
|
--wallet_id=<wallet_id> : (str) restrict operation to specific wallet
|
||||||
|
@ -2975,13 +2892,11 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
signing_channel = await self.get_channel_or_error(
|
signing_channel = await self.get_channel_or_error(
|
||||||
wallet, channel_account_id, channel_id, channel_name, for_signing=True
|
wallet, channel_account_id, channel_id, channel_name, for_signing=True
|
||||||
)
|
)
|
||||||
if salt is None:
|
timestamp = str(int(time.time()))
|
||||||
salt = str(int(time.time()))
|
signature = signing_channel.sign_data(unhexlify(hexdata), timestamp)
|
||||||
signature = signing_channel.sign_data(unhexlify(str(hexdata)), salt)
|
|
||||||
return {
|
return {
|
||||||
'signature': signature,
|
'signature': signature,
|
||||||
'signing_ts': salt, # DEPRECATED
|
'signing_ts': timestamp
|
||||||
'salt': salt,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@requires(WALLET_COMPONENT)
|
@requires(WALLET_COMPONENT)
|
||||||
|
@ -3106,7 +3021,7 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
'channel_id': channel.claim_id,
|
'channel_id': channel.claim_id,
|
||||||
'holding_address': address,
|
'holding_address': address,
|
||||||
'holding_public_key': public_key.extended_key_string(),
|
'holding_public_key': public_key.extended_key_string(),
|
||||||
'signing_private_key': channel.private_key.signing_key.to_pem().decode()
|
'signing_private_key': channel.private_key.to_pem().decode()
|
||||||
}
|
}
|
||||||
return base58.b58encode(json.dumps(export, separators=(',', ':')))
|
return base58.b58encode(json.dumps(export, separators=(',', ':')))
|
||||||
|
|
||||||
|
@ -3129,14 +3044,15 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
|
|
||||||
decoded = base58.b58decode(channel_data)
|
decoded = base58.b58decode(channel_data)
|
||||||
data = json.loads(decoded)
|
data = json.loads(decoded)
|
||||||
channel_private_key = PrivateKey.from_pem(
|
channel_private_key = ecdsa.SigningKey.from_pem(
|
||||||
self.ledger, data['signing_private_key']
|
data['signing_private_key'], hashfunc=hashlib.sha256
|
||||||
)
|
)
|
||||||
|
public_key_der = channel_private_key.get_verifying_key().to_der()
|
||||||
|
|
||||||
# check that the holding_address hasn't changed since the export was made
|
# check that the holding_address hasn't changed since the export was made
|
||||||
holding_address = data['holding_address']
|
holding_address = data['holding_address']
|
||||||
channels, _, _, _ = await self.ledger.claim_search(
|
channels, _, _, _ = await self.ledger.claim_search(
|
||||||
wallet.accounts, public_key_id=channel_private_key.address
|
wallet.accounts, public_key_id=self.ledger.public_key_to_address(public_key_der)
|
||||||
)
|
)
|
||||||
if channels and channels[0].get_address(self.ledger) != holding_address:
|
if channels and channels[0].get_address(self.ledger) != holding_address:
|
||||||
holding_address = channels[0].get_address(self.ledger)
|
holding_address = channels[0].get_address(self.ledger)
|
||||||
|
@ -3299,17 +3215,15 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
)
|
)
|
||||||
|
|
||||||
@requires(WALLET_COMPONENT, FILE_MANAGER_COMPONENT, BLOB_COMPONENT, DATABASE_COMPONENT)
|
@requires(WALLET_COMPONENT, FILE_MANAGER_COMPONENT, BLOB_COMPONENT, DATABASE_COMPONENT)
|
||||||
async def jsonrpc_stream_repost(
|
async def jsonrpc_stream_repost(self, name, bid, claim_id, allow_duplicate_name=False, channel_id=None,
|
||||||
self, name, bid, claim_id, allow_duplicate_name=False, channel_id=None,
|
|
||||||
channel_name=None, channel_account_id=None, account_id=None, wallet_id=None,
|
channel_name=None, channel_account_id=None, account_id=None, wallet_id=None,
|
||||||
claim_address=None, funding_account_ids=None, preview=False, blocking=False, **kwargs):
|
claim_address=None, funding_account_ids=None, preview=False, blocking=False):
|
||||||
"""
|
"""
|
||||||
Creates a claim that references an existing stream by its claim id.
|
Creates a claim that references an existing stream by its claim id.
|
||||||
|
|
||||||
Usage:
|
Usage:
|
||||||
stream_repost (<name> | --name=<name>) (<bid> | --bid=<bid>) (<claim_id> | --claim_id=<claim_id>)
|
stream_repost (<name> | --name=<name>) (<bid> | --bid=<bid>) (<claim_id> | --claim_id=<claim_id>)
|
||||||
[--allow_duplicate_name=<allow_duplicate_name>]
|
[--allow_duplicate_name=<allow_duplicate_name>]
|
||||||
[--title=<title>] [--description=<description>] [--tags=<tags>...]
|
|
||||||
[--channel_id=<channel_id> | --channel_name=<channel_name>]
|
[--channel_id=<channel_id> | --channel_name=<channel_name>]
|
||||||
[--channel_account_id=<channel_account_id>...]
|
[--channel_account_id=<channel_account_id>...]
|
||||||
[--account_id=<account_id>] [--wallet_id=<wallet_id>]
|
[--account_id=<account_id>] [--wallet_id=<wallet_id>]
|
||||||
|
@ -3322,9 +3236,6 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
--claim_id=<claim_id> : (str) id of the claim being reposted
|
--claim_id=<claim_id> : (str) id of the claim being reposted
|
||||||
--allow_duplicate_name=<allow_duplicate_name> : (bool) create new claim even if one already exists with
|
--allow_duplicate_name=<allow_duplicate_name> : (bool) create new claim even if one already exists with
|
||||||
given name. default: false.
|
given name. default: false.
|
||||||
--title=<title> : (str) title of the repost
|
|
||||||
--description=<description> : (str) description of the repost
|
|
||||||
--tags=<tags> : (list) add repost tags
|
|
||||||
--channel_id=<channel_id> : (str) claim id of the publisher channel
|
--channel_id=<channel_id> : (str) claim id of the publisher channel
|
||||||
--channel_name=<channel_name> : (str) name of the publisher channel
|
--channel_name=<channel_name> : (str) name of the publisher channel
|
||||||
--channel_account_id=<channel_account_id>: (str) one or more account ids for accounts to look in
|
--channel_account_id=<channel_account_id>: (str) one or more account ids for accounts to look in
|
||||||
|
@ -3359,7 +3270,6 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
raise Exception('Invalid claim id. It is expected to be a 40 characters long hexadecimal string.')
|
raise Exception('Invalid claim id. It is expected to be a 40 characters long hexadecimal string.')
|
||||||
|
|
||||||
claim = Claim()
|
claim = Claim()
|
||||||
claim.repost.update(**kwargs)
|
|
||||||
claim.repost.reference.claim_id = claim_id
|
claim.repost.reference.claim_id = claim_id
|
||||||
tx = await Transaction.claim_create(
|
tx = await Transaction.claim_create(
|
||||||
name, claim, amount, claim_address, funding_accounts, funding_accounts[0], channel
|
name, claim, amount, claim_address, funding_accounts, funding_accounts[0], channel
|
||||||
|
@ -3537,7 +3447,7 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
|
|
||||||
async def save_claims():
|
async def save_claims():
|
||||||
await self.storage.save_claims([self._old_get_temp_claim_info(
|
await self.storage.save_claims([self._old_get_temp_claim_info(
|
||||||
tx, new_txo, claim_address, claim, name
|
tx, new_txo, claim_address, claim, name, dewies_to_lbc(amount)
|
||||||
)])
|
)])
|
||||||
if file_path is not None:
|
if file_path is not None:
|
||||||
await self.storage.save_content_claim(file_stream.stream_hash, new_txo.id)
|
await self.storage.save_content_claim(file_stream.stream_hash, new_txo.id)
|
||||||
|
@ -3684,17 +3594,15 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
)
|
)
|
||||||
if len(existing_claims) != 1:
|
if len(existing_claims) != 1:
|
||||||
account_ids = ', '.join(f"'{account.id}'" for account in accounts)
|
account_ids = ', '.join(f"'{account.id}'" for account in accounts)
|
||||||
raise InputValueError(
|
# TODO: use error from lbry.error
|
||||||
|
raise Exception(
|
||||||
f"Can't find the stream '{claim_id}' in account(s) {account_ids}."
|
f"Can't find the stream '{claim_id}' in account(s) {account_ids}."
|
||||||
)
|
)
|
||||||
|
|
||||||
old_txo = existing_claims[0]
|
old_txo = existing_claims[0]
|
||||||
if not old_txo.claim.is_stream and not old_txo.claim.is_repost:
|
if not old_txo.claim.is_stream:
|
||||||
# in principle it should work with any type of claim, but its safer to
|
# TODO: use error from lbry.error
|
||||||
# limit it to ones we know won't be broken. in the future we can expand
|
raise Exception(
|
||||||
# this if we have a test case for e.g. channel or support claims
|
f"A claim with id '{claim_id}' was found but it is not a stream claim."
|
||||||
raise InputValueError(
|
|
||||||
f"A claim with id '{claim_id}' was found but it is not a stream or repost claim."
|
|
||||||
)
|
)
|
||||||
|
|
||||||
if bid is not None:
|
if bid is not None:
|
||||||
|
@ -3725,7 +3633,6 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
|
|
||||||
if replace:
|
if replace:
|
||||||
claim = Claim()
|
claim = Claim()
|
||||||
if old_txo.claim.is_stream:
|
|
||||||
if old_txo.claim.stream.has_source:
|
if old_txo.claim.stream.has_source:
|
||||||
claim.stream.message.source.CopyFrom(
|
claim.stream.message.source.CopyFrom(
|
||||||
old_txo.claim.stream.message.source
|
old_txo.claim.stream.message.source
|
||||||
|
@ -3735,24 +3642,19 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
old_stream_type = getattr(old_txo.claim.stream.message, stream_type)
|
old_stream_type = getattr(old_txo.claim.stream.message, stream_type)
|
||||||
new_stream_type = getattr(claim.stream.message, stream_type)
|
new_stream_type = getattr(claim.stream.message, stream_type)
|
||||||
new_stream_type.CopyFrom(old_stream_type)
|
new_stream_type.CopyFrom(old_stream_type)
|
||||||
|
claim.stream.update(file_path=file_path, **kwargs)
|
||||||
else:
|
else:
|
||||||
claim = Claim.from_bytes(old_txo.claim.to_bytes())
|
claim = Claim.from_bytes(old_txo.claim.to_bytes())
|
||||||
|
|
||||||
if old_txo.claim.is_stream:
|
|
||||||
claim.stream.update(file_path=file_path, **kwargs)
|
claim.stream.update(file_path=file_path, **kwargs)
|
||||||
elif old_txo.claim.is_repost:
|
|
||||||
claim.repost.update(**kwargs)
|
|
||||||
|
|
||||||
if clear_channel:
|
if clear_channel:
|
||||||
claim.clear_signature()
|
claim.clear_signature()
|
||||||
tx = await Transaction.claim_update(
|
tx = await Transaction.claim_update(
|
||||||
old_txo, claim, amount, claim_address, funding_accounts, funding_accounts[0],
|
old_txo, claim, amount, claim_address, funding_accounts, funding_accounts[0],
|
||||||
channel if not clear_channel else None
|
channel if not clear_channel else None
|
||||||
)
|
)
|
||||||
|
|
||||||
new_txo = tx.outputs[0]
|
new_txo = tx.outputs[0]
|
||||||
stream_hash = None
|
stream_hash = None
|
||||||
if not preview and old_txo.claim.is_stream:
|
if not preview:
|
||||||
old_stream = self.file_manager.get_filtered(sd_hash=old_txo.claim.stream.source.sd_hash)
|
old_stream = self.file_manager.get_filtered(sd_hash=old_txo.claim.stream.source.sd_hash)
|
||||||
old_stream = old_stream[0] if old_stream else None
|
old_stream = old_stream[0] if old_stream else None
|
||||||
if file_path is not None:
|
if file_path is not None:
|
||||||
|
@ -3774,7 +3676,7 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
|
|
||||||
async def save_claims():
|
async def save_claims():
|
||||||
await self.storage.save_claims([self._old_get_temp_claim_info(
|
await self.storage.save_claims([self._old_get_temp_claim_info(
|
||||||
tx, new_txo, claim_address, new_txo.claim, new_txo.claim_name
|
tx, new_txo, claim_address, new_txo.claim, new_txo.claim_name, dewies_to_lbc(amount)
|
||||||
)])
|
)])
|
||||||
if stream_hash:
|
if stream_hash:
|
||||||
await self.storage.save_content_claim(stream_hash, new_txo.id)
|
await self.storage.save_content_claim(stream_hash, new_txo.id)
|
||||||
|
@ -4036,8 +3938,6 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
[--languages=<languages>...] [--clear_languages]
|
[--languages=<languages>...] [--clear_languages]
|
||||||
[--locations=<locations>...] [--clear_locations]
|
[--locations=<locations>...] [--clear_locations]
|
||||||
[--thumbnail_url=<thumbnail_url>] [--cover_url=<cover_url>]
|
[--thumbnail_url=<thumbnail_url>] [--cover_url=<cover_url>]
|
||||||
[--channel_id=<channel_id> | --channel_name=<channel_name>]
|
|
||||||
[--channel_account_id=<channel_account_id>...]
|
|
||||||
[--account_id=<account_id>] [--wallet_id=<wallet_id>]
|
[--account_id=<account_id>] [--wallet_id=<wallet_id>]
|
||||||
[--claim_address=<claim_address>]
|
[--claim_address=<claim_address>]
|
||||||
[--funding_account_ids=<funding_account_ids>...]
|
[--funding_account_ids=<funding_account_ids>...]
|
||||||
|
@ -4093,10 +3993,6 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
|
|
||||||
--clear_locations : (bool) clear existing locations (prior to adding new ones)
|
--clear_locations : (bool) clear existing locations (prior to adding new ones)
|
||||||
--thumbnail_url=<thumbnail_url>: (str) thumbnail url
|
--thumbnail_url=<thumbnail_url>: (str) thumbnail url
|
||||||
--channel_id=<channel_id> : (str) claim id of the publisher channel
|
|
||||||
--channel_name=<channel_name> : (str) name of the publisher channel
|
|
||||||
--channel_account_id=<channel_account_id>: (str) one or more account ids for accounts to look in
|
|
||||||
for channel certificates, defaults to all accounts.
|
|
||||||
--account_id=<account_id> : (str) account in which to look for collection (default: all)
|
--account_id=<account_id> : (str) account in which to look for collection (default: all)
|
||||||
--wallet_id=<wallet_id> : (str) restrict operation to specific wallet
|
--wallet_id=<wallet_id> : (str) restrict operation to specific wallet
|
||||||
--funding_account_ids=<funding_account_ids>: (list) ids of accounts to fund this transaction
|
--funding_account_ids=<funding_account_ids>: (list) ids of accounts to fund this transaction
|
||||||
|
@ -4344,7 +4240,7 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
'nout': tx.position,
|
'nout': tx.position,
|
||||||
'address': claim_address,
|
'address': claim_address,
|
||||||
'claim_id': claim_id,
|
'claim_id': claim_id,
|
||||||
'amount': dewies_to_lbc(new_txo.amount)
|
'amount': dewies_to_lbc(amount)
|
||||||
}]})
|
}]})
|
||||||
self.component_manager.loop.create_task(self.analytics_manager.send_claim_action('new_support'))
|
self.component_manager.loop.create_task(self.analytics_manager.send_claim_action('new_support'))
|
||||||
else:
|
else:
|
||||||
|
@ -4961,16 +4857,21 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
DHT / Blob Exchange peer commands.
|
DHT / Blob Exchange peer commands.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
async def jsonrpc_peer_list(self, blob_hash, page=None, page_size=None):
|
@requires(DHT_COMPONENT)
|
||||||
|
async def jsonrpc_peer_list(self, blob_hash, search_bottom_out_limit=None, page=None, page_size=None):
|
||||||
"""
|
"""
|
||||||
Get peers for blob hash
|
Get peers for blob hash
|
||||||
|
|
||||||
Usage:
|
Usage:
|
||||||
peer_list (<blob_hash> | --blob_hash=<blob_hash>)
|
peer_list (<blob_hash> | --blob_hash=<blob_hash>)
|
||||||
|
[<search_bottom_out_limit> | --search_bottom_out_limit=<search_bottom_out_limit>]
|
||||||
[--page=<page>] [--page_size=<page_size>]
|
[--page=<page>] [--page_size=<page_size>]
|
||||||
|
|
||||||
Options:
|
Options:
|
||||||
--blob_hash=<blob_hash> : (str) find available peers for this blob hash
|
--blob_hash=<blob_hash> : (str) find available peers for this blob hash
|
||||||
|
--search_bottom_out_limit=<search_bottom_out_limit> : (int) the number of search probes in a row
|
||||||
|
that don't find any new peers
|
||||||
|
before giving up and returning
|
||||||
--page=<page> : (int) page to return during paginating
|
--page=<page> : (int) page to return during paginating
|
||||||
--page_size=<page_size> : (int) number of items on page during pagination
|
--page_size=<page_size> : (int) number of items on page during pagination
|
||||||
|
|
||||||
|
@ -4982,29 +4883,28 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
if not is_valid_blobhash(blob_hash):
|
if not is_valid_blobhash(blob_hash):
|
||||||
# TODO: use error from lbry.error
|
# TODO: use error from lbry.error
|
||||||
raise Exception("invalid blob hash")
|
raise Exception("invalid blob hash")
|
||||||
peer_q = asyncio.Queue(loop=self.component_manager.loop)
|
if search_bottom_out_limit is not None:
|
||||||
if self.component_manager.has_component(TRACKER_ANNOUNCER_COMPONENT):
|
search_bottom_out_limit = int(search_bottom_out_limit)
|
||||||
tracker = self.component_manager.get_component(TRACKER_ANNOUNCER_COMPONENT)
|
if search_bottom_out_limit <= 0:
|
||||||
tracker_peers = await tracker.get_kademlia_peer_list(bytes.fromhex(blob_hash))
|
# TODO: use error from lbry.error
|
||||||
log.info("Found %d peers for %s from trackers.", len(tracker_peers), blob_hash[:8])
|
raise Exception("invalid bottom out limit")
|
||||||
peer_q.put_nowait(tracker_peers)
|
else:
|
||||||
elif not self.component_manager.has_component(DHT_COMPONENT):
|
search_bottom_out_limit = 4
|
||||||
raise Exception("Peer list needs, at least, either a DHT component or a Tracker component for discovery.")
|
|
||||||
peers = []
|
peers = []
|
||||||
if self.component_manager.has_component(DHT_COMPONENT):
|
peer_q = asyncio.Queue(loop=self.component_manager.loop)
|
||||||
await self.dht_node._peers_for_value_producer(blob_hash, peer_q)
|
await self.dht_node._peers_for_value_producer(blob_hash, peer_q)
|
||||||
while not peer_q.empty():
|
while not peer_q.empty():
|
||||||
peers.extend(peer_q.get_nowait())
|
peers.extend(peer_q.get_nowait())
|
||||||
results = {
|
results = [
|
||||||
(peer.address, peer.tcp_port): {
|
{
|
||||||
"node_id": hexlify(peer.node_id).decode() if peer.node_id else None,
|
"node_id": hexlify(peer.node_id).decode(),
|
||||||
"address": peer.address,
|
"address": peer.address,
|
||||||
"udp_port": peer.udp_port,
|
"udp_port": peer.udp_port,
|
||||||
"tcp_port": peer.tcp_port,
|
"tcp_port": peer.tcp_port,
|
||||||
}
|
}
|
||||||
for peer in peers
|
for peer in peers
|
||||||
}
|
]
|
||||||
return paginate_list(list(results.values()), page, page_size)
|
return paginate_list(results, page, page_size)
|
||||||
|
|
||||||
@requires(DATABASE_COMPONENT)
|
@requires(DATABASE_COMPONENT)
|
||||||
async def jsonrpc_blob_announce(self, blob_hash=None, stream_hash=None, sd_hash=None):
|
async def jsonrpc_blob_announce(self, blob_hash=None, stream_hash=None, sd_hash=None):
|
||||||
|
@ -5059,8 +4959,7 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
--finished : (bool) only return finished blobs
|
--finished : (bool) only return finished blobs
|
||||||
--uri=<uri> : (str) filter blobs by stream in a uri
|
--uri=<uri> : (str) filter blobs by stream in a uri
|
||||||
--stream_hash=<stream_hash> : (str) filter blobs by stream hash
|
--stream_hash=<stream_hash> : (str) filter blobs by stream hash
|
||||||
--sd_hash=<sd_hash> : (str) filter blobs in a stream by sd hash, ie the hash of the stream
|
--sd_hash=<sd_hash> : (str) filter blobs by sd hash
|
||||||
descriptor blob for a stream that has been downloaded
|
|
||||||
--page=<page> : (int) page to return during paginating
|
--page=<page> : (int) page to return during paginating
|
||||||
--page_size=<page_size> : (int) number of items on page during pagination
|
--page_size=<page_size> : (int) number of items on page during pagination
|
||||||
|
|
||||||
|
@ -5229,12 +5128,10 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_id": (str) the local dht node id
|
"node_id": (str) the local dht node id
|
||||||
"prefix_neighbors_count": (int) the amount of peers sharing the same byte prefix of the local node id
|
|
||||||
}
|
}
|
||||||
"""
|
"""
|
||||||
result = {
|
result = {
|
||||||
'buckets': {},
|
'buckets': {}
|
||||||
'prefix_neighbors_count': 0
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for i, _ in enumerate(self.dht_node.protocol.routing_table.buckets):
|
for i, _ in enumerate(self.dht_node.protocol.routing_table.buckets):
|
||||||
|
@ -5247,7 +5144,6 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
"node_id": hexlify(peer.node_id).decode(),
|
"node_id": hexlify(peer.node_id).decode(),
|
||||||
}
|
}
|
||||||
result['buckets'][i].append(host)
|
result['buckets'][i].append(host)
|
||||||
result['prefix_neighbors_count'] += 1 if peer.node_id[0] == self.dht_node.protocol.node_id[0] else 0
|
|
||||||
|
|
||||||
result['node_id'] = hexlify(self.dht_node.protocol.node_id).decode()
|
result['node_id'] = hexlify(self.dht_node.protocol.node_id).decode()
|
||||||
return result
|
return result
|
||||||
|
@ -5474,11 +5370,11 @@ class Daemon(metaclass=JSONRPCServerType):
|
||||||
return results
|
return results
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def _old_get_temp_claim_info(tx, txo, address, claim_dict, name):
|
def _old_get_temp_claim_info(tx, txo, address, claim_dict, name, bid):
|
||||||
return {
|
return {
|
||||||
"claim_id": txo.claim_id,
|
"claim_id": txo.claim_id,
|
||||||
"name": name,
|
"name": name,
|
||||||
"amount": dewies_to_lbc(txo.amount),
|
"amount": bid,
|
||||||
"address": address,
|
"address": address,
|
||||||
"txid": tx.id,
|
"txid": tx.id,
|
||||||
"nout": txo.position,
|
"nout": txo.position,
|
||||||
|
|
|
@ -80,6 +80,8 @@ class MarketFeed:
|
||||||
self.rate = ExchangeRate(self.market, rate, int(time.time()))
|
self.rate = ExchangeRate(self.market, rate, int(time.time()))
|
||||||
self.last_check = time.time()
|
self.last_check = time.time()
|
||||||
return self.rate
|
return self.rate
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
raise
|
||||||
except asyncio.TimeoutError:
|
except asyncio.TimeoutError:
|
||||||
log.warning("Timed out fetching exchange rate from %s.", self.name)
|
log.warning("Timed out fetching exchange rate from %s.", self.name)
|
||||||
except json.JSONDecodeError as e:
|
except json.JSONDecodeError as e:
|
||||||
|
@ -194,9 +196,9 @@ FEEDS: Iterable[Type[MarketFeed]] = (
|
||||||
BittrexUSDFeed,
|
BittrexUSDFeed,
|
||||||
CoinExBTCFeed,
|
CoinExBTCFeed,
|
||||||
CoinExUSDFeed,
|
CoinExUSDFeed,
|
||||||
# HotbitBTCFeed,
|
HotbitBTCFeed,
|
||||||
# HotbitUSDFeed,
|
HotbitUSDFeed,
|
||||||
# UPbitBTCFeed,
|
UPbitBTCFeed,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -10,7 +10,7 @@ from lbry.schema.claim import Claim
|
||||||
from lbry.schema.support import Support
|
from lbry.schema.support import Support
|
||||||
from lbry.torrent.torrent_manager import TorrentSource
|
from lbry.torrent.torrent_manager import TorrentSource
|
||||||
from lbry.wallet import Wallet, Ledger, Account, Transaction, Output
|
from lbry.wallet import Wallet, Ledger, Account, Transaction, Output
|
||||||
from lbry.wallet.bip32 import PublicKey
|
from lbry.wallet.bip32 import PubKey
|
||||||
from lbry.wallet.dewies import dewies_to_lbc
|
from lbry.wallet.dewies import dewies_to_lbc
|
||||||
from lbry.stream.managed_stream import ManagedStream
|
from lbry.stream.managed_stream import ManagedStream
|
||||||
|
|
||||||
|
@ -138,7 +138,7 @@ class JSONResponseEncoder(JSONEncoder):
|
||||||
return self.encode_claim(obj)
|
return self.encode_claim(obj)
|
||||||
if isinstance(obj, Support):
|
if isinstance(obj, Support):
|
||||||
return obj.to_dict()
|
return obj.to_dict()
|
||||||
if isinstance(obj, PublicKey):
|
if isinstance(obj, PubKey):
|
||||||
return obj.extended_key_string()
|
return obj.extended_key_string()
|
||||||
if isinstance(obj, datetime):
|
if isinstance(obj, datetime):
|
||||||
return obj.strftime("%Y%m%dT%H:%M:%S")
|
return obj.strftime("%Y%m%dT%H:%M:%S")
|
||||||
|
@ -328,8 +328,8 @@ class JSONResponseEncoder(JSONEncoder):
|
||||||
result.update({
|
result.update({
|
||||||
'streaming_url': managed_stream.stream_url,
|
'streaming_url': managed_stream.stream_url,
|
||||||
'stream_hash': managed_stream.stream_hash,
|
'stream_hash': managed_stream.stream_hash,
|
||||||
'stream_name': managed_stream.stream_name,
|
'stream_name': managed_stream.descriptor.stream_name,
|
||||||
'suggested_file_name': managed_stream.suggested_file_name,
|
'suggested_file_name': managed_stream.descriptor.suggested_file_name,
|
||||||
'sd_hash': managed_stream.descriptor.sd_hash,
|
'sd_hash': managed_stream.descriptor.sd_hash,
|
||||||
'mime_type': managed_stream.mime_type,
|
'mime_type': managed_stream.mime_type,
|
||||||
'key': managed_stream.descriptor.key,
|
'key': managed_stream.descriptor.key,
|
||||||
|
|
|
@ -37,8 +37,6 @@ def migrate_db(conf, start, end):
|
||||||
from .migrate13to14 import do_migration
|
from .migrate13to14 import do_migration
|
||||||
elif current == 14:
|
elif current == 14:
|
||||||
from .migrate14to15 import do_migration
|
from .migrate14to15 import do_migration
|
||||||
elif current == 15:
|
|
||||||
from .migrate15to16 import do_migration
|
|
||||||
else:
|
else:
|
||||||
raise Exception(f"DB migration of version {current} to {current+1} is not available")
|
raise Exception(f"DB migration of version {current} to {current+1} is not available")
|
||||||
try:
|
try:
|
||||||
|
|
|
@ -1,17 +0,0 @@
|
||||||
import os
|
|
||||||
import sqlite3
|
|
||||||
|
|
||||||
|
|
||||||
def do_migration(conf):
|
|
||||||
db_path = os.path.join(conf.data_dir, "lbrynet.sqlite")
|
|
||||||
connection = sqlite3.connect(db_path)
|
|
||||||
cursor = connection.cursor()
|
|
||||||
|
|
||||||
cursor.executescript("""
|
|
||||||
update blob set should_announce=0
|
|
||||||
where should_announce=1 and
|
|
||||||
blob.blob_hash in (select stream_blob.blob_hash from stream_blob where position=0);
|
|
||||||
""")
|
|
||||||
|
|
||||||
connection.commit()
|
|
||||||
connection.close()
|
|
|
@ -20,7 +20,7 @@ def do_migration(conf):
|
||||||
"left outer join blob b ON b.blob_hash=s.blob_hash order by s.position").fetchall()
|
"left outer join blob b ON b.blob_hash=s.blob_hash order by s.position").fetchall()
|
||||||
blobs_by_stream = {}
|
blobs_by_stream = {}
|
||||||
for stream_hash, position, iv, blob_hash, blob_length in blobs:
|
for stream_hash, position, iv, blob_hash, blob_length in blobs:
|
||||||
blobs_by_stream.setdefault(stream_hash, []).append(BlobInfo(position, blob_length or 0, iv, 0, blob_hash))
|
blobs_by_stream.setdefault(stream_hash, []).append(BlobInfo(position, blob_length or 0, iv, blob_hash))
|
||||||
|
|
||||||
for stream_name, stream_key, suggested_filename, sd_hash, stream_hash in streams:
|
for stream_name, stream_key, suggested_filename, sd_hash, stream_hash in streams:
|
||||||
sd = StreamDescriptor(None, blob_dir, stream_name, stream_key, suggested_filename,
|
sd = StreamDescriptor(None, blob_dir, stream_name, stream_key, suggested_filename,
|
||||||
|
|
|
@ -187,8 +187,8 @@ def store_stream(transaction: sqlite3.Connection, sd_blob: 'BlobFile', descripto
|
||||||
).fetchall()
|
).fetchall()
|
||||||
# ensure should_announce is set regardless if insert was ignored
|
# ensure should_announce is set regardless if insert was ignored
|
||||||
transaction.execute(
|
transaction.execute(
|
||||||
"update blob set should_announce=1 where blob_hash in (?)",
|
"update blob set should_announce=1 where blob_hash in (?, ?)",
|
||||||
(sd_blob.blob_hash,)
|
(sd_blob.blob_hash, descriptor.blobs[0].blob_hash,)
|
||||||
).fetchall()
|
).fetchall()
|
||||||
|
|
||||||
|
|
||||||
|
@ -337,7 +337,6 @@ class SQLiteStorage(SQLiteMixin):
|
||||||
tcp_port integer,
|
tcp_port integer,
|
||||||
unique (address, udp_port)
|
unique (address, udp_port)
|
||||||
);
|
);
|
||||||
create index if not exists blob_data on blob(blob_hash, blob_length, is_mine);
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, conf: Config, path, loop=None, time_getter: typing.Optional[typing.Callable[[], float]] = None):
|
def __init__(self, conf: Config, path, loop=None, time_getter: typing.Optional[typing.Callable[[], float]] = None):
|
||||||
|
@ -381,11 +380,6 @@ class SQLiteStorage(SQLiteMixin):
|
||||||
"select status from blob where blob_hash=?", blob_hash
|
"select status from blob where blob_hash=?", blob_hash
|
||||||
)
|
)
|
||||||
|
|
||||||
def set_announce(self, *blob_hashes):
|
|
||||||
return self.db.execute_fetchall(
|
|
||||||
"update blob set should_announce=1 where blob_hash in (?, ?)", blob_hashes
|
|
||||||
)
|
|
||||||
|
|
||||||
def update_last_announced_blobs(self, blob_hashes: typing.List[str]):
|
def update_last_announced_blobs(self, blob_hashes: typing.List[str]):
|
||||||
def _update_last_announced_blobs(transaction: sqlite3.Connection):
|
def _update_last_announced_blobs(transaction: sqlite3.Connection):
|
||||||
last_announced = self.time_getter()
|
last_announced = self.time_getter()
|
||||||
|
@ -443,53 +437,20 @@ class SQLiteStorage(SQLiteMixin):
|
||||||
def get_all_blob_hashes(self):
|
def get_all_blob_hashes(self):
|
||||||
return self.run_and_return_list("select blob_hash from blob")
|
return self.run_and_return_list("select blob_hash from blob")
|
||||||
|
|
||||||
async def get_stored_blobs(self, is_mine: bool, is_network_blob=False):
|
async def get_stored_blobs(self, is_mine: bool):
|
||||||
is_mine = 1 if is_mine else 0
|
is_mine = 1 if is_mine else 0
|
||||||
if is_network_blob:
|
|
||||||
return await self.db.execute_fetchall(
|
return await self.db.execute_fetchall(
|
||||||
"select blob.blob_hash, blob.blob_length, blob.added_on "
|
"select blob_hash, blob_length, added_on from blob where is_mine=? order by added_on asc",
|
||||||
"from blob left join stream_blob using (blob_hash) "
|
|
||||||
"where stream_blob.stream_hash is null and blob.is_mine=? and blob.status='finished'"
|
|
||||||
"order by blob.blob_length desc, blob.added_on asc",
|
|
||||||
(is_mine,)
|
(is_mine,)
|
||||||
)
|
)
|
||||||
|
|
||||||
sd_blobs = await self.db.execute_fetchall(
|
async def get_stored_blob_disk_usage(self, is_mine: Optional[bool] = None):
|
||||||
"select blob.blob_hash, blob.blob_length, blob.added_on "
|
if is_mine is None:
|
||||||
"from blob join stream on blob.blob_hash=stream.sd_hash join file using (stream_hash) "
|
sql, args = "select coalesce(sum(blob_length), 0) from blob", ()
|
||||||
"where blob.is_mine=? order by blob.added_on asc",
|
else:
|
||||||
(is_mine,)
|
is_mine = 1 if is_mine else 0
|
||||||
)
|
sql, args = "select coalesce(sum(blob_length), 0) from blob where is_mine=?", (is_mine,)
|
||||||
content_blobs = await self.db.execute_fetchall(
|
return (await self.db.execute_fetchone(sql, args))[0]
|
||||||
"select blob.blob_hash, blob.blob_length, blob.added_on "
|
|
||||||
"from blob join stream_blob using (blob_hash) cross join stream using (stream_hash)"
|
|
||||||
"cross join file using (stream_hash)"
|
|
||||||
"where blob.is_mine=? and blob.status='finished' order by blob.added_on asc, blob.blob_length asc",
|
|
||||||
(is_mine,)
|
|
||||||
)
|
|
||||||
return content_blobs + sd_blobs
|
|
||||||
|
|
||||||
async def get_stored_blob_disk_usage(self):
|
|
||||||
total, network_size, content_size, private_size = await self.db.execute_fetchone("""
|
|
||||||
select coalesce(sum(blob_length), 0) as total,
|
|
||||||
coalesce(sum(case when
|
|
||||||
stream_blob.stream_hash is null
|
|
||||||
then blob_length else 0 end), 0) as network_storage,
|
|
||||||
coalesce(sum(case when
|
|
||||||
stream_blob.blob_hash is not null and is_mine=0
|
|
||||||
then blob_length else 0 end), 0) as content_storage,
|
|
||||||
coalesce(sum(case when
|
|
||||||
is_mine=1
|
|
||||||
then blob_length else 0 end), 0) as private_storage
|
|
||||||
from blob left join stream_blob using (blob_hash)
|
|
||||||
where blob_hash not in (select sd_hash from stream) and blob.status="finished"
|
|
||||||
""")
|
|
||||||
return {
|
|
||||||
'network_storage': network_size,
|
|
||||||
'content_storage': content_size,
|
|
||||||
'private_storage': private_size,
|
|
||||||
'total': total
|
|
||||||
}
|
|
||||||
|
|
||||||
async def update_blob_ownership(self, sd_hash, is_mine: bool):
|
async def update_blob_ownership(self, sd_hash, is_mine: bool):
|
||||||
is_mine = 1 if is_mine else 0
|
is_mine = 1 if is_mine else 0
|
||||||
|
@ -534,8 +495,7 @@ class SQLiteStorage(SQLiteMixin):
|
||||||
def _get_blobs_for_stream(transaction):
|
def _get_blobs_for_stream(transaction):
|
||||||
crypt_blob_infos = []
|
crypt_blob_infos = []
|
||||||
stream_blobs = transaction.execute(
|
stream_blobs = transaction.execute(
|
||||||
"select s.blob_hash, s.position, s.iv, b.added_on "
|
"select blob_hash, position, iv from stream_blob where stream_hash=? "
|
||||||
"from stream_blob s left outer join blob b on b.blob_hash=s.blob_hash where stream_hash=? "
|
|
||||||
"order by position asc", (stream_hash, )
|
"order by position asc", (stream_hash, )
|
||||||
).fetchall()
|
).fetchall()
|
||||||
if only_completed:
|
if only_completed:
|
||||||
|
@ -555,10 +515,9 @@ class SQLiteStorage(SQLiteMixin):
|
||||||
for blob_hash, length in lengths:
|
for blob_hash, length in lengths:
|
||||||
blob_length_dict[blob_hash] = length
|
blob_length_dict[blob_hash] = length
|
||||||
|
|
||||||
current_time = time.time()
|
for blob_hash, position, iv in stream_blobs:
|
||||||
for blob_hash, position, iv, added_on in stream_blobs:
|
|
||||||
blob_length = blob_length_dict.get(blob_hash, 0)
|
blob_length = blob_length_dict.get(blob_hash, 0)
|
||||||
crypt_blob_infos.append(BlobInfo(position, blob_length, iv, added_on or current_time, blob_hash))
|
crypt_blob_infos.append(BlobInfo(position, blob_length, iv, blob_hash))
|
||||||
if not blob_hash:
|
if not blob_hash:
|
||||||
break
|
break
|
||||||
return crypt_blob_infos
|
return crypt_blob_infos
|
||||||
|
@ -793,7 +752,7 @@ class SQLiteStorage(SQLiteMixin):
|
||||||
|
|
||||||
await self.db.run(_save_claims)
|
await self.db.run(_save_claims)
|
||||||
if update_file_callbacks:
|
if update_file_callbacks:
|
||||||
await asyncio.wait(map(asyncio.create_task, update_file_callbacks))
|
await asyncio.wait(update_file_callbacks)
|
||||||
if claim_id_to_supports:
|
if claim_id_to_supports:
|
||||||
await self.save_supports(claim_id_to_supports)
|
await self.save_supports(claim_id_to_supports)
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,6 @@ from typing import Optional
|
||||||
from aiohttp.web import Request
|
from aiohttp.web import Request
|
||||||
from lbry.error import ResolveError, DownloadSDTimeoutError, InsufficientFundsError
|
from lbry.error import ResolveError, DownloadSDTimeoutError, InsufficientFundsError
|
||||||
from lbry.error import ResolveTimeoutError, DownloadDataTimeoutError, KeyFeeAboveMaxAllowedError
|
from lbry.error import ResolveTimeoutError, DownloadDataTimeoutError, KeyFeeAboveMaxAllowedError
|
||||||
from lbry.error import InvalidStreamURLError
|
|
||||||
from lbry.stream.managed_stream import ManagedStream
|
from lbry.stream.managed_stream import ManagedStream
|
||||||
from lbry.torrent.torrent_manager import TorrentSource
|
from lbry.torrent.torrent_manager import TorrentSource
|
||||||
from lbry.utils import cache_concurrent
|
from lbry.utils import cache_concurrent
|
||||||
|
@ -13,12 +12,11 @@ from lbry.schema.url import URL
|
||||||
from lbry.wallet.dewies import dewies_to_lbc
|
from lbry.wallet.dewies import dewies_to_lbc
|
||||||
from lbry.file.source_manager import SourceManager
|
from lbry.file.source_manager import SourceManager
|
||||||
from lbry.file.source import ManagedDownloadSource
|
from lbry.file.source import ManagedDownloadSource
|
||||||
from lbry.extras.daemon.storage import StoredContentClaim
|
|
||||||
if typing.TYPE_CHECKING:
|
if typing.TYPE_CHECKING:
|
||||||
from lbry.conf import Config
|
from lbry.conf import Config
|
||||||
from lbry.extras.daemon.analytics import AnalyticsManager
|
from lbry.extras.daemon.analytics import AnalyticsManager
|
||||||
from lbry.extras.daemon.storage import SQLiteStorage
|
from lbry.extras.daemon.storage import SQLiteStorage
|
||||||
from lbry.wallet import WalletManager
|
from lbry.wallet import WalletManager, Output
|
||||||
from lbry.extras.daemon.exchange_rate_manager import ExchangeRateManager
|
from lbry.extras.daemon.exchange_rate_manager import ExchangeRateManager
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
@ -50,10 +48,10 @@ class FileManager:
|
||||||
await manager.started.wait()
|
await manager.started.wait()
|
||||||
self.started.set()
|
self.started.set()
|
||||||
|
|
||||||
async def stop(self):
|
def stop(self):
|
||||||
for manager in self.source_managers.values():
|
for manager in self.source_managers.values():
|
||||||
# fixme: pop or not?
|
# fixme: pop or not?
|
||||||
await manager.stop()
|
manager.stop()
|
||||||
self.started.clear()
|
self.started.clear()
|
||||||
|
|
||||||
@cache_concurrent
|
@cache_concurrent
|
||||||
|
@ -83,11 +81,8 @@ class FileManager:
|
||||||
payment = None
|
payment = None
|
||||||
try:
|
try:
|
||||||
# resolve the claim
|
# resolve the claim
|
||||||
try:
|
|
||||||
if not URL.parse(uri).has_stream:
|
if not URL.parse(uri).has_stream:
|
||||||
raise InvalidStreamURLError(uri)
|
raise ResolveError("cannot download a channel claim, specify a /path")
|
||||||
except ValueError:
|
|
||||||
raise InvalidStreamURLError(uri)
|
|
||||||
try:
|
try:
|
||||||
resolved_result = await asyncio.wait_for(
|
resolved_result = await asyncio.wait_for(
|
||||||
self.wallet_manager.ledger.resolve(
|
self.wallet_manager.ledger.resolve(
|
||||||
|
@ -99,6 +94,8 @@ class FileManager:
|
||||||
except asyncio.TimeoutError:
|
except asyncio.TimeoutError:
|
||||||
raise ResolveTimeoutError(uri)
|
raise ResolveTimeoutError(uri)
|
||||||
except Exception as err:
|
except Exception as err:
|
||||||
|
if isinstance(err, asyncio.CancelledError):
|
||||||
|
raise
|
||||||
log.exception("Unexpected error resolving stream:")
|
log.exception("Unexpected error resolving stream:")
|
||||||
raise ResolveError(f"Unexpected error resolving stream: {str(err)}")
|
raise ResolveError(f"Unexpected error resolving stream: {str(err)}")
|
||||||
if 'error' in resolved_result:
|
if 'error' in resolved_result:
|
||||||
|
@ -193,24 +190,21 @@ class FileManager:
|
||||||
####################
|
####################
|
||||||
# make downloader and wait for start
|
# make downloader and wait for start
|
||||||
####################
|
####################
|
||||||
# temporary with fields we know so downloader can start. Missing fields are populated later.
|
|
||||||
stored_claim = StoredContentClaim(outpoint=outpoint, claim_id=txo.claim_id, name=txo.claim_name,
|
|
||||||
amount=txo.amount, height=txo.tx_ref.height,
|
|
||||||
serialized=claim.to_bytes().hex())
|
|
||||||
|
|
||||||
if not claim.stream.source.bt_infohash:
|
if not claim.stream.source.bt_infohash:
|
||||||
# fixme: this shouldnt be here
|
# fixme: this shouldnt be here
|
||||||
stream = ManagedStream(
|
stream = ManagedStream(
|
||||||
self.loop, self.config, source_manager.blob_manager, claim.stream.source.sd_hash,
|
self.loop, self.config, source_manager.blob_manager, claim.stream.source.sd_hash,
|
||||||
download_directory, file_name, ManagedStream.STATUS_RUNNING, content_fee=payment,
|
download_directory, file_name, ManagedStream.STATUS_RUNNING, content_fee=payment,
|
||||||
analytics_manager=self.analytics_manager, claim=stored_claim
|
analytics_manager=self.analytics_manager
|
||||||
)
|
)
|
||||||
stream.downloader.node = source_manager.node
|
stream.downloader.node = source_manager.node
|
||||||
else:
|
else:
|
||||||
stream = TorrentSource(
|
stream = TorrentSource(
|
||||||
self.loop, self.config, self.storage, identifier=claim.stream.source.bt_infohash,
|
self.loop, self.config, self.storage, identifier=claim.stream.source.bt_infohash,
|
||||||
file_name=file_name, download_directory=download_directory or self.config.download_dir,
|
file_name=file_name, download_directory=download_directory or self.config.download_dir,
|
||||||
status=ManagedStream.STATUS_RUNNING, claim=stored_claim, analytics_manager=self.analytics_manager,
|
status=ManagedStream.STATUS_RUNNING,
|
||||||
|
analytics_manager=self.analytics_manager,
|
||||||
torrent_session=source_manager.torrent_session
|
torrent_session=source_manager.torrent_session
|
||||||
)
|
)
|
||||||
log.info("starting download for %s", uri)
|
log.info("starting download for %s", uri)
|
||||||
|
@ -242,14 +236,15 @@ class FileManager:
|
||||||
claim_info = await self.storage.get_content_claim_for_torrent(stream.identifier)
|
claim_info = await self.storage.get_content_claim_for_torrent(stream.identifier)
|
||||||
stream.set_claim(claim_info, claim)
|
stream.set_claim(claim_info, claim)
|
||||||
if save_file:
|
if save_file:
|
||||||
await asyncio.wait_for(stream.save_file(), timeout - (self.loop.time() - before_download))
|
await asyncio.wait_for(stream.save_file(), timeout - (self.loop.time() - before_download),
|
||||||
|
loop=self.loop)
|
||||||
return stream
|
return stream
|
||||||
except asyncio.TimeoutError:
|
except asyncio.TimeoutError:
|
||||||
error = DownloadDataTimeoutError(stream.sd_hash)
|
error = DownloadDataTimeoutError(stream.sd_hash)
|
||||||
raise error
|
raise error
|
||||||
except (Exception, asyncio.CancelledError) as err: # forgive data timeout, don't delete stream
|
except Exception as err: # forgive data timeout, don't delete stream
|
||||||
expected = (DownloadSDTimeoutError, DownloadDataTimeoutError, InsufficientFundsError,
|
expected = (DownloadSDTimeoutError, DownloadDataTimeoutError, InsufficientFundsError,
|
||||||
KeyFeeAboveMaxAllowedError, ResolveError, InvalidStreamURLError)
|
KeyFeeAboveMaxAllowedError)
|
||||||
if isinstance(err, expected):
|
if isinstance(err, expected):
|
||||||
log.warning("Failed to download %s: %s", uri, str(err))
|
log.warning("Failed to download %s: %s", uri, str(err))
|
||||||
elif isinstance(err, asyncio.CancelledError):
|
elif isinstance(err, asyncio.CancelledError):
|
||||||
|
|
|
@ -45,12 +45,11 @@ class ManagedDownloadSource:
|
||||||
self.purchase_receipt = None
|
self.purchase_receipt = None
|
||||||
self._added_on = added_on
|
self._added_on = added_on
|
||||||
self.analytics_manager = analytics_manager
|
self.analytics_manager = analytics_manager
|
||||||
self.downloader = None
|
|
||||||
|
|
||||||
self.saving = asyncio.Event()
|
self.saving = asyncio.Event(loop=self.loop)
|
||||||
self.finished_writing = asyncio.Event()
|
self.finished_writing = asyncio.Event(loop=self.loop)
|
||||||
self.started_writing = asyncio.Event()
|
self.started_writing = asyncio.Event(loop=self.loop)
|
||||||
self.finished_write_attempt = asyncio.Event()
|
self.finished_write_attempt = asyncio.Event(loop=self.loop)
|
||||||
|
|
||||||
# @classmethod
|
# @classmethod
|
||||||
# async def create(cls, loop: asyncio.AbstractEventLoop, config: 'Config', file_path: str,
|
# async def create(cls, loop: asyncio.AbstractEventLoop, config: 'Config', file_path: str,
|
||||||
|
@ -67,7 +66,7 @@ class ManagedDownloadSource:
|
||||||
async def save_file(self, file_name: Optional[str] = None, download_directory: Optional[str] = None):
|
async def save_file(self, file_name: Optional[str] = None, download_directory: Optional[str] = None):
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
async def stop_tasks(self):
|
def stop_tasks(self):
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
def set_claim(self, claim_info: typing.Dict, claim: 'Claim'):
|
def set_claim(self, claim_info: typing.Dict, claim: 'Claim'):
|
||||||
|
|
|
@ -54,16 +54,16 @@ class SourceManager:
|
||||||
self.storage = storage
|
self.storage = storage
|
||||||
self.analytics_manager = analytics_manager
|
self.analytics_manager = analytics_manager
|
||||||
self._sources: typing.Dict[str, ManagedDownloadSource] = {}
|
self._sources: typing.Dict[str, ManagedDownloadSource] = {}
|
||||||
self.started = asyncio.Event()
|
self.started = asyncio.Event(loop=self.loop)
|
||||||
|
|
||||||
def add(self, source: ManagedDownloadSource):
|
def add(self, source: ManagedDownloadSource):
|
||||||
self._sources[source.identifier] = source
|
self._sources[source.identifier] = source
|
||||||
|
|
||||||
async def remove(self, source: ManagedDownloadSource):
|
def remove(self, source: ManagedDownloadSource):
|
||||||
if source.identifier not in self._sources:
|
if source.identifier not in self._sources:
|
||||||
return
|
return
|
||||||
self._sources.pop(source.identifier)
|
self._sources.pop(source.identifier)
|
||||||
await source.stop_tasks()
|
source.stop_tasks()
|
||||||
|
|
||||||
async def initialize_from_database(self):
|
async def initialize_from_database(self):
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
@ -72,10 +72,10 @@ class SourceManager:
|
||||||
await self.initialize_from_database()
|
await self.initialize_from_database()
|
||||||
self.started.set()
|
self.started.set()
|
||||||
|
|
||||||
async def stop(self):
|
def stop(self):
|
||||||
while self._sources:
|
while self._sources:
|
||||||
_, source = self._sources.popitem()
|
_, source = self._sources.popitem()
|
||||||
await source.stop_tasks()
|
source.stop_tasks()
|
||||||
self.started.clear()
|
self.started.clear()
|
||||||
|
|
||||||
async def create(self, file_path: str, key: Optional[bytes] = None,
|
async def create(self, file_path: str, key: Optional[bytes] = None,
|
||||||
|
@ -83,7 +83,7 @@ class SourceManager:
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
async def delete(self, source: ManagedDownloadSource, delete_file: Optional[bool] = False):
|
async def delete(self, source: ManagedDownloadSource, delete_file: Optional[bool] = False):
|
||||||
await self.remove(source)
|
self.remove(source)
|
||||||
if delete_file and source.output_file_exists:
|
if delete_file and source.output_file_exists:
|
||||||
os.remove(source.full_path)
|
os.remove(source.full_path)
|
||||||
|
|
||||||
|
|
|
@ -2,5 +2,4 @@ build:
|
||||||
rm types/v2/* -rf
|
rm types/v2/* -rf
|
||||||
touch types/v2/__init__.py
|
touch types/v2/__init__.py
|
||||||
cd types/v2/ && protoc --python_out=. -I ../../../../../types/v2/proto/ ../../../../../types/v2/proto/*.proto
|
cd types/v2/ && protoc --python_out=. -I ../../../../../types/v2/proto/ ../../../../../types/v2/proto/*.proto
|
||||||
cd types/v2/ && cp ../../../../../types/jsonschema/* ./
|
|
||||||
sed -e 's/^import\ \(.*\)_pb2\ /from . import\ \1_pb2\ /g' -i types/v2/*.py
|
sed -e 's/^import\ \(.*\)_pb2\ /from . import\ \1_pb2\ /g' -i types/v2/*.py
|
||||||
|
|
|
@ -2,9 +2,6 @@ import logging
|
||||||
from typing import List
|
from typing import List
|
||||||
from binascii import hexlify, unhexlify
|
from binascii import hexlify, unhexlify
|
||||||
|
|
||||||
from asn1crypto.keys import PublicKeyInfo
|
|
||||||
from coincurve import PublicKey as cPublicKey
|
|
||||||
|
|
||||||
from google.protobuf.json_format import MessageToDict
|
from google.protobuf.json_format import MessageToDict
|
||||||
from google.protobuf.message import DecodeError
|
from google.protobuf.message import DecodeError
|
||||||
from hachoir.core.log import log as hachoir_log
|
from hachoir.core.log import log as hachoir_log
|
||||||
|
@ -349,7 +346,7 @@ class Channel(BaseClaim):
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def public_key(self) -> str:
|
def public_key(self) -> str:
|
||||||
return hexlify(self.public_key_bytes).decode()
|
return hexlify(self.message.public_key).decode()
|
||||||
|
|
||||||
@public_key.setter
|
@public_key.setter
|
||||||
def public_key(self, sd_public_key: str):
|
def public_key(self, sd_public_key: str):
|
||||||
|
@ -357,11 +354,7 @@ class Channel(BaseClaim):
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def public_key_bytes(self) -> bytes:
|
def public_key_bytes(self) -> bytes:
|
||||||
if len(self.message.public_key) == 33:
|
|
||||||
return self.message.public_key
|
return self.message.public_key
|
||||||
public_key_info = PublicKeyInfo.load(self.message.public_key)
|
|
||||||
public_key = cPublicKey(public_key_info.native['public_key'])
|
|
||||||
return public_key.format(compressed=True)
|
|
||||||
|
|
||||||
@public_key_bytes.setter
|
@public_key_bytes.setter
|
||||||
def public_key_bytes(self, public_key: bytes):
|
def public_key_bytes(self, public_key: bytes):
|
||||||
|
@ -398,12 +391,6 @@ class Repost(BaseClaim):
|
||||||
|
|
||||||
claim_type = Claim.REPOST
|
claim_type = Claim.REPOST
|
||||||
|
|
||||||
def to_dict(self):
|
|
||||||
claim = super().to_dict()
|
|
||||||
if claim.pop('claim_hash', None):
|
|
||||||
claim['claim_id'] = self.reference.claim_id
|
|
||||||
return claim
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def reference(self) -> ClaimReference:
|
def reference(self) -> ClaimReference:
|
||||||
return ClaimReference(self.message)
|
return ClaimReference(self.message)
|
||||||
|
|
|
@ -1,6 +1,4 @@
|
||||||
import os
|
import os
|
||||||
import filetype
|
|
||||||
import logging
|
|
||||||
|
|
||||||
types_map = {
|
types_map = {
|
||||||
# http://www.iana.org/assignments/media-types
|
# http://www.iana.org/assignments/media-types
|
||||||
|
@ -168,38 +166,10 @@ types_map = {
|
||||||
'.wmv': ('video/x-ms-wmv', 'video')
|
'.wmv': ('video/x-ms-wmv', 'video')
|
||||||
}
|
}
|
||||||
|
|
||||||
# maps detected extensions to the possible analogs
|
|
||||||
# i.e. .cbz file is actually a .zip
|
|
||||||
synonyms_map = {
|
|
||||||
'.zip': ['.cbz'],
|
|
||||||
'.rar': ['.cbr'],
|
|
||||||
'.ar': ['.a']
|
|
||||||
}
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
def guess_media_type(path):
|
def guess_media_type(path):
|
||||||
_, ext = os.path.splitext(path)
|
_, ext = os.path.splitext(path)
|
||||||
extension = ext.strip().lower()
|
extension = ext.strip().lower()
|
||||||
|
|
||||||
try:
|
|
||||||
kind = filetype.guess(path)
|
|
||||||
if kind:
|
|
||||||
real_extension = f".{kind.extension}"
|
|
||||||
|
|
||||||
if extension != real_extension:
|
|
||||||
if extension:
|
|
||||||
log.warning(f"file extension does not match it's contents: {path}, identified as {real_extension}")
|
|
||||||
else:
|
|
||||||
log.debug(f"file {path} does not have extension, identified by it's contents as {real_extension}")
|
|
||||||
|
|
||||||
if extension not in synonyms_map.get(real_extension, []):
|
|
||||||
extension = real_extension
|
|
||||||
|
|
||||||
except OSError as error:
|
|
||||||
pass
|
|
||||||
|
|
||||||
if extension[1:]:
|
if extension[1:]:
|
||||||
if extension in types_map:
|
if extension in types_map:
|
||||||
return types_map[extension]
|
return types_map[extension]
|
||||||
|
|
|
@ -1,11 +1,13 @@
|
||||||
import base64
|
import base64
|
||||||
from typing import List, Union, Optional, NamedTuple
|
from typing import List, TYPE_CHECKING, Union, Optional
|
||||||
from binascii import hexlify
|
from binascii import hexlify
|
||||||
from itertools import chain
|
from itertools import chain
|
||||||
|
|
||||||
from lbry.error import ResolveCensoredError
|
from lbry.error import ResolveCensoredError
|
||||||
from lbry.schema.types.v2.result_pb2 import Outputs as OutputsMessage
|
from lbry.schema.types.v2.result_pb2 import Outputs as OutputsMessage
|
||||||
from lbry.schema.types.v2.result_pb2 import Error as ErrorMessage
|
from lbry.schema.types.v2.result_pb2 import Error as ErrorMessage
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from lbry.wallet.server.leveldb import ResolveResult
|
||||||
|
|
||||||
INVALID = ErrorMessage.Code.Name(ErrorMessage.INVALID)
|
INVALID = ErrorMessage.Code.Name(ErrorMessage.INVALID)
|
||||||
NOT_FOUND = ErrorMessage.Code.Name(ErrorMessage.NOT_FOUND)
|
NOT_FOUND = ErrorMessage.Code.Name(ErrorMessage.NOT_FOUND)
|
||||||
|
@ -22,31 +24,6 @@ def set_reference(reference, claim_hash, rows):
|
||||||
return
|
return
|
||||||
|
|
||||||
|
|
||||||
class ResolveResult(NamedTuple):
|
|
||||||
name: str
|
|
||||||
normalized_name: str
|
|
||||||
claim_hash: bytes
|
|
||||||
tx_num: int
|
|
||||||
position: int
|
|
||||||
tx_hash: bytes
|
|
||||||
height: int
|
|
||||||
amount: int
|
|
||||||
short_url: str
|
|
||||||
is_controlling: bool
|
|
||||||
canonical_url: str
|
|
||||||
creation_height: int
|
|
||||||
activation_height: int
|
|
||||||
expiration_height: int
|
|
||||||
effective_amount: int
|
|
||||||
support_amount: int
|
|
||||||
reposted: int
|
|
||||||
last_takeover_height: Optional[int]
|
|
||||||
claims_in_channel: Optional[int]
|
|
||||||
channel_hash: Optional[bytes]
|
|
||||||
reposted_claim_hash: Optional[bytes]
|
|
||||||
signature_valid: Optional[bool]
|
|
||||||
|
|
||||||
|
|
||||||
class Censor:
|
class Censor:
|
||||||
|
|
||||||
NOT_CENSORED = 0
|
NOT_CENSORED = 0
|
||||||
|
@ -177,6 +154,19 @@ class Outputs:
|
||||||
outputs.blocked, outputs.blocked_total
|
outputs.blocked, outputs.blocked_total
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_grpc(cls, outputs: OutputsMessage) -> 'Outputs':
|
||||||
|
txs = set()
|
||||||
|
for txo_message in chain(outputs.txos, outputs.extra_txos):
|
||||||
|
if txo_message.WhichOneof('meta') == 'error':
|
||||||
|
continue
|
||||||
|
txs.add((hexlify(txo_message.tx_hash[::-1]).decode(), txo_message.height))
|
||||||
|
return cls(
|
||||||
|
outputs.txos, outputs.extra_txos, txs,
|
||||||
|
outputs.offset, outputs.total,
|
||||||
|
outputs.blocked, outputs.blocked_total
|
||||||
|
)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def to_base64(cls, txo_rows, extra_txo_rows, offset=0, total=None, blocked=None) -> str:
|
def to_base64(cls, txo_rows, extra_txo_rows, offset=0, total=None, blocked=None) -> str:
|
||||||
return base64.b64encode(cls.to_bytes(txo_rows, extra_txo_rows, offset, total, blocked)).decode()
|
return base64.b64encode(cls.to_bytes(txo_rows, extra_txo_rows, offset, total, blocked)).decode()
|
||||||
|
|
716
lbry/schema/types/v2/hub_pb2.py
Normal file
716
lbry/schema/types/v2/hub_pb2.py
Normal file
|
@ -0,0 +1,716 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Generated by the protocol buffer compiler. DO NOT EDIT!
|
||||||
|
# source: hub.proto
|
||||||
|
"""Generated protocol buffer code."""
|
||||||
|
from google.protobuf import descriptor as _descriptor
|
||||||
|
from google.protobuf import message as _message
|
||||||
|
from google.protobuf import reflection as _reflection
|
||||||
|
from google.protobuf import symbol_database as _symbol_database
|
||||||
|
# @@protoc_insertion_point(imports)
|
||||||
|
|
||||||
|
_sym_db = _symbol_database.Default()
|
||||||
|
|
||||||
|
|
||||||
|
from . import result_pb2 as result__pb2
|
||||||
|
|
||||||
|
|
||||||
|
DESCRIPTOR = _descriptor.FileDescriptor(
|
||||||
|
name='hub.proto',
|
||||||
|
package='pb',
|
||||||
|
syntax='proto3',
|
||||||
|
serialized_options=b'Z$github.com/lbryio/hub/protobuf/go/pb',
|
||||||
|
create_key=_descriptor._internal_create_key,
|
||||||
|
serialized_pb=b'\n\thub.proto\x12\x02pb\x1a\x0cresult.proto\"0\n\x0fInvertibleField\x12\x0e\n\x06invert\x18\x01 \x01(\x08\x12\r\n\x05value\x18\x02 \x03(\t\"\x1a\n\tBoolValue\x12\r\n\x05value\x18\x01 \x01(\x08\"\x1c\n\x0bUInt32Value\x12\r\n\x05value\x18\x01 \x01(\r\"j\n\nRangeField\x12\x1d\n\x02op\x18\x01 \x01(\x0e\x32\x11.pb.RangeField.Op\x12\r\n\x05value\x18\x02 \x03(\t\".\n\x02Op\x12\x06\n\x02\x45Q\x10\x00\x12\x07\n\x03LTE\x10\x01\x12\x07\n\x03GTE\x10\x02\x12\x06\n\x02LT\x10\x03\x12\x06\n\x02GT\x10\x04\"\xa3\x0c\n\rSearchRequest\x12%\n\x08\x63laim_id\x18\x01 \x01(\x0b\x32\x13.pb.InvertibleField\x12\'\n\nchannel_id\x18\x02 \x01(\x0b\x32\x13.pb.InvertibleField\x12\x0c\n\x04text\x18\x03 \x01(\t\x12\r\n\x05limit\x18\x04 \x01(\r\x12\x10\n\x08order_by\x18\x05 \x03(\t\x12\x0e\n\x06offset\x18\x06 \x01(\r\x12\x16\n\x0eis_controlling\x18\x07 \x01(\x08\x12\x1d\n\x15last_take_over_height\x18\x08 \x01(\t\x12\x12\n\nclaim_name\x18\t \x01(\t\x12\x17\n\x0fnormalized_name\x18\n \x01(\t\x12#\n\x0btx_position\x18\x0b \x01(\x0b\x32\x0e.pb.RangeField\x12\x1e\n\x06\x61mount\x18\x0c \x01(\x0b\x32\x0e.pb.RangeField\x12!\n\ttimestamp\x18\r \x01(\x0b\x32\x0e.pb.RangeField\x12*\n\x12\x63reation_timestamp\x18\x0e \x01(\x0b\x32\x0e.pb.RangeField\x12\x1e\n\x06height\x18\x0f \x01(\x0b\x32\x0e.pb.RangeField\x12\'\n\x0f\x63reation_height\x18\x10 \x01(\x0b\x32\x0e.pb.RangeField\x12)\n\x11\x61\x63tivation_height\x18\x11 \x01(\x0b\x32\x0e.pb.RangeField\x12)\n\x11\x65xpiration_height\x18\x12 \x01(\x0b\x32\x0e.pb.RangeField\x12$\n\x0crelease_time\x18\x13 \x01(\x0b\x32\x0e.pb.RangeField\x12\x11\n\tshort_url\x18\x14 \x01(\t\x12\x15\n\rcanonical_url\x18\x15 \x01(\t\x12\r\n\x05title\x18\x16 \x01(\t\x12\x0e\n\x06\x61uthor\x18\x17 \x01(\t\x12\x13\n\x0b\x64\x65scription\x18\x18 \x01(\t\x12\x12\n\nclaim_type\x18\x19 \x03(\t\x12$\n\x0crepost_count\x18\x1a \x01(\x0b\x32\x0e.pb.RangeField\x12\x13\n\x0bstream_type\x18\x1b \x03(\t\x12\x12\n\nmedia_type\x18\x1c \x03(\t\x12\"\n\nfee_amount\x18\x1d \x01(\x0b\x32\x0e.pb.RangeField\x12\x14\n\x0c\x66\x65\x65_currency\x18\x1e \x01(\t\x12 \n\x08\x64uration\x18\x1f \x01(\x0b\x32\x0e.pb.RangeField\x12\x19\n\x11reposted_claim_id\x18 \x01(\t\x12#\n\x0b\x63\x65nsor_type\x18! \x01(\x0b\x32\x0e.pb.RangeField\x12\x19\n\x11\x63laims_in_channel\x18\" \x01(\t\x12$\n\x0c\x63hannel_join\x18# \x01(\x0b\x32\x0e.pb.RangeField\x12)\n\x12is_signature_valid\x18$ \x01(\x0b\x32\r.pb.BoolValue\x12(\n\x10\x65\x66\x66\x65\x63tive_amount\x18% \x01(\x0b\x32\x0e.pb.RangeField\x12&\n\x0esupport_amount\x18& \x01(\x0b\x32\x0e.pb.RangeField\x12&\n\x0etrending_score\x18\' \x01(\x0b\x32\x0e.pb.RangeField\x12\r\n\x05tx_id\x18+ \x01(\t\x12 \n\x07tx_nout\x18, \x01(\x0b\x32\x0f.pb.UInt32Value\x12\x11\n\tsignature\x18- \x01(\t\x12\x18\n\x10signature_digest\x18. \x01(\t\x12\x18\n\x10public_key_bytes\x18/ \x01(\t\x12\x15\n\rpublic_key_id\x18\x30 \x01(\t\x12\x10\n\x08\x61ny_tags\x18\x31 \x03(\t\x12\x10\n\x08\x61ll_tags\x18\x32 \x03(\t\x12\x10\n\x08not_tags\x18\x33 \x03(\t\x12\x1d\n\x15has_channel_signature\x18\x34 \x01(\x08\x12!\n\nhas_source\x18\x35 \x01(\x0b\x32\r.pb.BoolValue\x12 \n\x18limit_claims_per_channel\x18\x36 \x01(\r\x12\x15\n\rany_languages\x18\x37 \x03(\t\x12\x15\n\rall_languages\x18\x38 \x03(\t\x12\x19\n\x11remove_duplicates\x18\x39 \x01(\x08\x12\x11\n\tno_totals\x18: \x01(\x08\x32\x31\n\x03Hub\x12*\n\x06Search\x12\x11.pb.SearchRequest\x1a\x0b.pb.Outputs\"\x00\x42&Z$github.com/lbryio/hub/protobuf/go/pbb\x06proto3'
|
||||||
|
,
|
||||||
|
dependencies=[result__pb2.DESCRIPTOR,])
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
_RANGEFIELD_OP = _descriptor.EnumDescriptor(
|
||||||
|
name='Op',
|
||||||
|
full_name='pb.RangeField.Op',
|
||||||
|
filename=None,
|
||||||
|
file=DESCRIPTOR,
|
||||||
|
create_key=_descriptor._internal_create_key,
|
||||||
|
values=[
|
||||||
|
_descriptor.EnumValueDescriptor(
|
||||||
|
name='EQ', index=0, number=0,
|
||||||
|
serialized_options=None,
|
||||||
|
type=None,
|
||||||
|
create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.EnumValueDescriptor(
|
||||||
|
name='LTE', index=1, number=1,
|
||||||
|
serialized_options=None,
|
||||||
|
type=None,
|
||||||
|
create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.EnumValueDescriptor(
|
||||||
|
name='GTE', index=2, number=2,
|
||||||
|
serialized_options=None,
|
||||||
|
type=None,
|
||||||
|
create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.EnumValueDescriptor(
|
||||||
|
name='LT', index=3, number=3,
|
||||||
|
serialized_options=None,
|
||||||
|
type=None,
|
||||||
|
create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.EnumValueDescriptor(
|
||||||
|
name='GT', index=4, number=4,
|
||||||
|
serialized_options=None,
|
||||||
|
type=None,
|
||||||
|
create_key=_descriptor._internal_create_key),
|
||||||
|
],
|
||||||
|
containing_type=None,
|
||||||
|
serialized_options=None,
|
||||||
|
serialized_start=199,
|
||||||
|
serialized_end=245,
|
||||||
|
)
|
||||||
|
_sym_db.RegisterEnumDescriptor(_RANGEFIELD_OP)
|
||||||
|
|
||||||
|
|
||||||
|
_INVERTIBLEFIELD = _descriptor.Descriptor(
|
||||||
|
name='InvertibleField',
|
||||||
|
full_name='pb.InvertibleField',
|
||||||
|
filename=None,
|
||||||
|
file=DESCRIPTOR,
|
||||||
|
containing_type=None,
|
||||||
|
create_key=_descriptor._internal_create_key,
|
||||||
|
fields=[
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='invert', full_name='pb.InvertibleField.invert', index=0,
|
||||||
|
number=1, type=8, cpp_type=7, label=1,
|
||||||
|
has_default_value=False, default_value=False,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='value', full_name='pb.InvertibleField.value', index=1,
|
||||||
|
number=2, type=9, cpp_type=9, label=3,
|
||||||
|
has_default_value=False, default_value=[],
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
],
|
||||||
|
extensions=[
|
||||||
|
],
|
||||||
|
nested_types=[],
|
||||||
|
enum_types=[
|
||||||
|
],
|
||||||
|
serialized_options=None,
|
||||||
|
is_extendable=False,
|
||||||
|
syntax='proto3',
|
||||||
|
extension_ranges=[],
|
||||||
|
oneofs=[
|
||||||
|
],
|
||||||
|
serialized_start=31,
|
||||||
|
serialized_end=79,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
_BOOLVALUE = _descriptor.Descriptor(
|
||||||
|
name='BoolValue',
|
||||||
|
full_name='pb.BoolValue',
|
||||||
|
filename=None,
|
||||||
|
file=DESCRIPTOR,
|
||||||
|
containing_type=None,
|
||||||
|
create_key=_descriptor._internal_create_key,
|
||||||
|
fields=[
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='value', full_name='pb.BoolValue.value', index=0,
|
||||||
|
number=1, type=8, cpp_type=7, label=1,
|
||||||
|
has_default_value=False, default_value=False,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
],
|
||||||
|
extensions=[
|
||||||
|
],
|
||||||
|
nested_types=[],
|
||||||
|
enum_types=[
|
||||||
|
],
|
||||||
|
serialized_options=None,
|
||||||
|
is_extendable=False,
|
||||||
|
syntax='proto3',
|
||||||
|
extension_ranges=[],
|
||||||
|
oneofs=[
|
||||||
|
],
|
||||||
|
serialized_start=81,
|
||||||
|
serialized_end=107,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
_UINT32VALUE = _descriptor.Descriptor(
|
||||||
|
name='UInt32Value',
|
||||||
|
full_name='pb.UInt32Value',
|
||||||
|
filename=None,
|
||||||
|
file=DESCRIPTOR,
|
||||||
|
containing_type=None,
|
||||||
|
create_key=_descriptor._internal_create_key,
|
||||||
|
fields=[
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='value', full_name='pb.UInt32Value.value', index=0,
|
||||||
|
number=1, type=13, cpp_type=3, label=1,
|
||||||
|
has_default_value=False, default_value=0,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
],
|
||||||
|
extensions=[
|
||||||
|
],
|
||||||
|
nested_types=[],
|
||||||
|
enum_types=[
|
||||||
|
],
|
||||||
|
serialized_options=None,
|
||||||
|
is_extendable=False,
|
||||||
|
syntax='proto3',
|
||||||
|
extension_ranges=[],
|
||||||
|
oneofs=[
|
||||||
|
],
|
||||||
|
serialized_start=109,
|
||||||
|
serialized_end=137,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
_RANGEFIELD = _descriptor.Descriptor(
|
||||||
|
name='RangeField',
|
||||||
|
full_name='pb.RangeField',
|
||||||
|
filename=None,
|
||||||
|
file=DESCRIPTOR,
|
||||||
|
containing_type=None,
|
||||||
|
create_key=_descriptor._internal_create_key,
|
||||||
|
fields=[
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='op', full_name='pb.RangeField.op', index=0,
|
||||||
|
number=1, type=14, cpp_type=8, label=1,
|
||||||
|
has_default_value=False, default_value=0,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='value', full_name='pb.RangeField.value', index=1,
|
||||||
|
number=2, type=9, cpp_type=9, label=3,
|
||||||
|
has_default_value=False, default_value=[],
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
],
|
||||||
|
extensions=[
|
||||||
|
],
|
||||||
|
nested_types=[],
|
||||||
|
enum_types=[
|
||||||
|
_RANGEFIELD_OP,
|
||||||
|
],
|
||||||
|
serialized_options=None,
|
||||||
|
is_extendable=False,
|
||||||
|
syntax='proto3',
|
||||||
|
extension_ranges=[],
|
||||||
|
oneofs=[
|
||||||
|
],
|
||||||
|
serialized_start=139,
|
||||||
|
serialized_end=245,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
_SEARCHREQUEST = _descriptor.Descriptor(
|
||||||
|
name='SearchRequest',
|
||||||
|
full_name='pb.SearchRequest',
|
||||||
|
filename=None,
|
||||||
|
file=DESCRIPTOR,
|
||||||
|
containing_type=None,
|
||||||
|
create_key=_descriptor._internal_create_key,
|
||||||
|
fields=[
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='claim_id', full_name='pb.SearchRequest.claim_id', index=0,
|
||||||
|
number=1, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='channel_id', full_name='pb.SearchRequest.channel_id', index=1,
|
||||||
|
number=2, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='text', full_name='pb.SearchRequest.text', index=2,
|
||||||
|
number=3, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='limit', full_name='pb.SearchRequest.limit', index=3,
|
||||||
|
number=4, type=13, cpp_type=3, label=1,
|
||||||
|
has_default_value=False, default_value=0,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='order_by', full_name='pb.SearchRequest.order_by', index=4,
|
||||||
|
number=5, type=9, cpp_type=9, label=3,
|
||||||
|
has_default_value=False, default_value=[],
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='offset', full_name='pb.SearchRequest.offset', index=5,
|
||||||
|
number=6, type=13, cpp_type=3, label=1,
|
||||||
|
has_default_value=False, default_value=0,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='is_controlling', full_name='pb.SearchRequest.is_controlling', index=6,
|
||||||
|
number=7, type=8, cpp_type=7, label=1,
|
||||||
|
has_default_value=False, default_value=False,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='last_take_over_height', full_name='pb.SearchRequest.last_take_over_height', index=7,
|
||||||
|
number=8, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='claim_name', full_name='pb.SearchRequest.claim_name', index=8,
|
||||||
|
number=9, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='normalized_name', full_name='pb.SearchRequest.normalized_name', index=9,
|
||||||
|
number=10, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='tx_position', full_name='pb.SearchRequest.tx_position', index=10,
|
||||||
|
number=11, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='amount', full_name='pb.SearchRequest.amount', index=11,
|
||||||
|
number=12, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='timestamp', full_name='pb.SearchRequest.timestamp', index=12,
|
||||||
|
number=13, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='creation_timestamp', full_name='pb.SearchRequest.creation_timestamp', index=13,
|
||||||
|
number=14, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='height', full_name='pb.SearchRequest.height', index=14,
|
||||||
|
number=15, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='creation_height', full_name='pb.SearchRequest.creation_height', index=15,
|
||||||
|
number=16, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='activation_height', full_name='pb.SearchRequest.activation_height', index=16,
|
||||||
|
number=17, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='expiration_height', full_name='pb.SearchRequest.expiration_height', index=17,
|
||||||
|
number=18, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='release_time', full_name='pb.SearchRequest.release_time', index=18,
|
||||||
|
number=19, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='short_url', full_name='pb.SearchRequest.short_url', index=19,
|
||||||
|
number=20, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='canonical_url', full_name='pb.SearchRequest.canonical_url', index=20,
|
||||||
|
number=21, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='title', full_name='pb.SearchRequest.title', index=21,
|
||||||
|
number=22, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='author', full_name='pb.SearchRequest.author', index=22,
|
||||||
|
number=23, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='description', full_name='pb.SearchRequest.description', index=23,
|
||||||
|
number=24, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='claim_type', full_name='pb.SearchRequest.claim_type', index=24,
|
||||||
|
number=25, type=9, cpp_type=9, label=3,
|
||||||
|
has_default_value=False, default_value=[],
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='repost_count', full_name='pb.SearchRequest.repost_count', index=25,
|
||||||
|
number=26, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='stream_type', full_name='pb.SearchRequest.stream_type', index=26,
|
||||||
|
number=27, type=9, cpp_type=9, label=3,
|
||||||
|
has_default_value=False, default_value=[],
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='media_type', full_name='pb.SearchRequest.media_type', index=27,
|
||||||
|
number=28, type=9, cpp_type=9, label=3,
|
||||||
|
has_default_value=False, default_value=[],
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='fee_amount', full_name='pb.SearchRequest.fee_amount', index=28,
|
||||||
|
number=29, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='fee_currency', full_name='pb.SearchRequest.fee_currency', index=29,
|
||||||
|
number=30, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='duration', full_name='pb.SearchRequest.duration', index=30,
|
||||||
|
number=31, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='reposted_claim_id', full_name='pb.SearchRequest.reposted_claim_id', index=31,
|
||||||
|
number=32, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='censor_type', full_name='pb.SearchRequest.censor_type', index=32,
|
||||||
|
number=33, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='claims_in_channel', full_name='pb.SearchRequest.claims_in_channel', index=33,
|
||||||
|
number=34, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='channel_join', full_name='pb.SearchRequest.channel_join', index=34,
|
||||||
|
number=35, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='is_signature_valid', full_name='pb.SearchRequest.is_signature_valid', index=35,
|
||||||
|
number=36, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='effective_amount', full_name='pb.SearchRequest.effective_amount', index=36,
|
||||||
|
number=37, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='support_amount', full_name='pb.SearchRequest.support_amount', index=37,
|
||||||
|
number=38, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='trending_score', full_name='pb.SearchRequest.trending_score', index=38,
|
||||||
|
number=39, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='tx_id', full_name='pb.SearchRequest.tx_id', index=39,
|
||||||
|
number=43, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='tx_nout', full_name='pb.SearchRequest.tx_nout', index=40,
|
||||||
|
number=44, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='signature', full_name='pb.SearchRequest.signature', index=41,
|
||||||
|
number=45, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='signature_digest', full_name='pb.SearchRequest.signature_digest', index=42,
|
||||||
|
number=46, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='public_key_bytes', full_name='pb.SearchRequest.public_key_bytes', index=43,
|
||||||
|
number=47, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='public_key_id', full_name='pb.SearchRequest.public_key_id', index=44,
|
||||||
|
number=48, type=9, cpp_type=9, label=1,
|
||||||
|
has_default_value=False, default_value=b"".decode('utf-8'),
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='any_tags', full_name='pb.SearchRequest.any_tags', index=45,
|
||||||
|
number=49, type=9, cpp_type=9, label=3,
|
||||||
|
has_default_value=False, default_value=[],
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='all_tags', full_name='pb.SearchRequest.all_tags', index=46,
|
||||||
|
number=50, type=9, cpp_type=9, label=3,
|
||||||
|
has_default_value=False, default_value=[],
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='not_tags', full_name='pb.SearchRequest.not_tags', index=47,
|
||||||
|
number=51, type=9, cpp_type=9, label=3,
|
||||||
|
has_default_value=False, default_value=[],
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='has_channel_signature', full_name='pb.SearchRequest.has_channel_signature', index=48,
|
||||||
|
number=52, type=8, cpp_type=7, label=1,
|
||||||
|
has_default_value=False, default_value=False,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='has_source', full_name='pb.SearchRequest.has_source', index=49,
|
||||||
|
number=53, type=11, cpp_type=10, label=1,
|
||||||
|
has_default_value=False, default_value=None,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='limit_claims_per_channel', full_name='pb.SearchRequest.limit_claims_per_channel', index=50,
|
||||||
|
number=54, type=13, cpp_type=3, label=1,
|
||||||
|
has_default_value=False, default_value=0,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='any_languages', full_name='pb.SearchRequest.any_languages', index=51,
|
||||||
|
number=55, type=9, cpp_type=9, label=3,
|
||||||
|
has_default_value=False, default_value=[],
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='all_languages', full_name='pb.SearchRequest.all_languages', index=52,
|
||||||
|
number=56, type=9, cpp_type=9, label=3,
|
||||||
|
has_default_value=False, default_value=[],
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='remove_duplicates', full_name='pb.SearchRequest.remove_duplicates', index=53,
|
||||||
|
number=57, type=8, cpp_type=7, label=1,
|
||||||
|
has_default_value=False, default_value=False,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
_descriptor.FieldDescriptor(
|
||||||
|
name='no_totals', full_name='pb.SearchRequest.no_totals', index=54,
|
||||||
|
number=58, type=8, cpp_type=7, label=1,
|
||||||
|
has_default_value=False, default_value=False,
|
||||||
|
message_type=None, enum_type=None, containing_type=None,
|
||||||
|
is_extension=False, extension_scope=None,
|
||||||
|
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
|
||||||
|
],
|
||||||
|
extensions=[
|
||||||
|
],
|
||||||
|
nested_types=[],
|
||||||
|
enum_types=[
|
||||||
|
],
|
||||||
|
serialized_options=None,
|
||||||
|
is_extendable=False,
|
||||||
|
syntax='proto3',
|
||||||
|
extension_ranges=[],
|
||||||
|
oneofs=[
|
||||||
|
],
|
||||||
|
serialized_start=248,
|
||||||
|
serialized_end=1819,
|
||||||
|
)
|
||||||
|
|
||||||
|
_RANGEFIELD.fields_by_name['op'].enum_type = _RANGEFIELD_OP
|
||||||
|
_RANGEFIELD_OP.containing_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['claim_id'].message_type = _INVERTIBLEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['channel_id'].message_type = _INVERTIBLEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['tx_position'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['amount'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['timestamp'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['creation_timestamp'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['height'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['creation_height'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['activation_height'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['expiration_height'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['release_time'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['repost_count'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['fee_amount'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['duration'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['censor_type'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['channel_join'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['is_signature_valid'].message_type = _BOOLVALUE
|
||||||
|
_SEARCHREQUEST.fields_by_name['effective_amount'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['support_amount'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['trending_score'].message_type = _RANGEFIELD
|
||||||
|
_SEARCHREQUEST.fields_by_name['tx_nout'].message_type = _UINT32VALUE
|
||||||
|
_SEARCHREQUEST.fields_by_name['has_source'].message_type = _BOOLVALUE
|
||||||
|
DESCRIPTOR.message_types_by_name['InvertibleField'] = _INVERTIBLEFIELD
|
||||||
|
DESCRIPTOR.message_types_by_name['BoolValue'] = _BOOLVALUE
|
||||||
|
DESCRIPTOR.message_types_by_name['UInt32Value'] = _UINT32VALUE
|
||||||
|
DESCRIPTOR.message_types_by_name['RangeField'] = _RANGEFIELD
|
||||||
|
DESCRIPTOR.message_types_by_name['SearchRequest'] = _SEARCHREQUEST
|
||||||
|
_sym_db.RegisterFileDescriptor(DESCRIPTOR)
|
||||||
|
|
||||||
|
InvertibleField = _reflection.GeneratedProtocolMessageType('InvertibleField', (_message.Message,), {
|
||||||
|
'DESCRIPTOR' : _INVERTIBLEFIELD,
|
||||||
|
'__module__' : 'hub_pb2'
|
||||||
|
# @@protoc_insertion_point(class_scope:pb.InvertibleField)
|
||||||
|
})
|
||||||
|
_sym_db.RegisterMessage(InvertibleField)
|
||||||
|
|
||||||
|
BoolValue = _reflection.GeneratedProtocolMessageType('BoolValue', (_message.Message,), {
|
||||||
|
'DESCRIPTOR' : _BOOLVALUE,
|
||||||
|
'__module__' : 'hub_pb2'
|
||||||
|
# @@protoc_insertion_point(class_scope:pb.BoolValue)
|
||||||
|
})
|
||||||
|
_sym_db.RegisterMessage(BoolValue)
|
||||||
|
|
||||||
|
UInt32Value = _reflection.GeneratedProtocolMessageType('UInt32Value', (_message.Message,), {
|
||||||
|
'DESCRIPTOR' : _UINT32VALUE,
|
||||||
|
'__module__' : 'hub_pb2'
|
||||||
|
# @@protoc_insertion_point(class_scope:pb.UInt32Value)
|
||||||
|
})
|
||||||
|
_sym_db.RegisterMessage(UInt32Value)
|
||||||
|
|
||||||
|
RangeField = _reflection.GeneratedProtocolMessageType('RangeField', (_message.Message,), {
|
||||||
|
'DESCRIPTOR' : _RANGEFIELD,
|
||||||
|
'__module__' : 'hub_pb2'
|
||||||
|
# @@protoc_insertion_point(class_scope:pb.RangeField)
|
||||||
|
})
|
||||||
|
_sym_db.RegisterMessage(RangeField)
|
||||||
|
|
||||||
|
SearchRequest = _reflection.GeneratedProtocolMessageType('SearchRequest', (_message.Message,), {
|
||||||
|
'DESCRIPTOR' : _SEARCHREQUEST,
|
||||||
|
'__module__' : 'hub_pb2'
|
||||||
|
# @@protoc_insertion_point(class_scope:pb.SearchRequest)
|
||||||
|
})
|
||||||
|
_sym_db.RegisterMessage(SearchRequest)
|
||||||
|
|
||||||
|
|
||||||
|
DESCRIPTOR._options = None
|
||||||
|
|
||||||
|
_HUB = _descriptor.ServiceDescriptor(
|
||||||
|
name='Hub',
|
||||||
|
full_name='pb.Hub',
|
||||||
|
file=DESCRIPTOR,
|
||||||
|
index=0,
|
||||||
|
serialized_options=None,
|
||||||
|
create_key=_descriptor._internal_create_key,
|
||||||
|
serialized_start=1821,
|
||||||
|
serialized_end=1870,
|
||||||
|
methods=[
|
||||||
|
_descriptor.MethodDescriptor(
|
||||||
|
name='Search',
|
||||||
|
full_name='pb.Hub.Search',
|
||||||
|
index=0,
|
||||||
|
containing_service=None,
|
||||||
|
input_type=_SEARCHREQUEST,
|
||||||
|
output_type=result__pb2._OUTPUTS,
|
||||||
|
serialized_options=None,
|
||||||
|
create_key=_descriptor._internal_create_key,
|
||||||
|
),
|
||||||
|
])
|
||||||
|
_sym_db.RegisterServiceDescriptor(_HUB)
|
||||||
|
|
||||||
|
DESCRIPTOR.services_by_name['Hub'] = _HUB
|
||||||
|
|
||||||
|
# @@protoc_insertion_point(module_scope)
|
67
lbry/schema/types/v2/hub_pb2_grpc.py
Normal file
67
lbry/schema/types/v2/hub_pb2_grpc.py
Normal file
|
@ -0,0 +1,67 @@
|
||||||
|
# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
|
||||||
|
"""Client and server classes corresponding to protobuf-defined services."""
|
||||||
|
import grpc
|
||||||
|
|
||||||
|
from . import hub_pb2 as hub__pb2
|
||||||
|
from . import result_pb2 as result__pb2
|
||||||
|
|
||||||
|
|
||||||
|
class HubStub(object):
|
||||||
|
"""Missing associated documentation comment in .proto file."""
|
||||||
|
|
||||||
|
def __init__(self, channel):
|
||||||
|
"""Constructor.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
channel: A grpc.Channel.
|
||||||
|
"""
|
||||||
|
self.Search = channel.unary_unary(
|
||||||
|
'/pb.Hub/Search',
|
||||||
|
request_serializer=hub__pb2.SearchRequest.SerializeToString,
|
||||||
|
response_deserializer=result__pb2.Outputs.FromString,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class HubServicer(object):
|
||||||
|
"""Missing associated documentation comment in .proto file."""
|
||||||
|
|
||||||
|
def Search(self, request, context):
|
||||||
|
"""Missing associated documentation comment in .proto file."""
|
||||||
|
context.set_code(grpc.StatusCode.UNIMPLEMENTED)
|
||||||
|
context.set_details('Method not implemented!')
|
||||||
|
raise NotImplementedError('Method not implemented!')
|
||||||
|
|
||||||
|
|
||||||
|
def add_HubServicer_to_server(servicer, server):
|
||||||
|
rpc_method_handlers = {
|
||||||
|
'Search': grpc.unary_unary_rpc_method_handler(
|
||||||
|
servicer.Search,
|
||||||
|
request_deserializer=hub__pb2.SearchRequest.FromString,
|
||||||
|
response_serializer=result__pb2.Outputs.SerializeToString,
|
||||||
|
),
|
||||||
|
}
|
||||||
|
generic_handler = grpc.method_handlers_generic_handler(
|
||||||
|
'pb.Hub', rpc_method_handlers)
|
||||||
|
server.add_generic_rpc_handlers((generic_handler,))
|
||||||
|
|
||||||
|
|
||||||
|
# This class is part of an EXPERIMENTAL API.
|
||||||
|
class Hub(object):
|
||||||
|
"""Missing associated documentation comment in .proto file."""
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def Search(request,
|
||||||
|
target,
|
||||||
|
options=(),
|
||||||
|
channel_credentials=None,
|
||||||
|
call_credentials=None,
|
||||||
|
insecure=False,
|
||||||
|
compression=None,
|
||||||
|
wait_for_ready=None,
|
||||||
|
timeout=None,
|
||||||
|
metadata=None):
|
||||||
|
return grpc.experimental.unary_unary(request, target, '/pb.Hub/Search',
|
||||||
|
hub__pb2.SearchRequest.SerializeToString,
|
||||||
|
result__pb2.Outputs.FromString,
|
||||||
|
options, channel_credentials,
|
||||||
|
insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
|
4
lbry/schema/types/v2/result_pb2_grpc.py
Normal file
4
lbry/schema/types/v2/result_pb2_grpc.py
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
|
||||||
|
"""Client and server classes corresponding to protobuf-defined services."""
|
||||||
|
import grpc
|
||||||
|
|
|
@ -1,139 +0,0 @@
|
||||||
{
|
|
||||||
"title": "Wallet",
|
|
||||||
"description": "An LBC wallet",
|
|
||||||
"type": "object",
|
|
||||||
"required": ["name", "version", "accounts", "preferences"],
|
|
||||||
"additionalProperties": false,
|
|
||||||
"properties": {
|
|
||||||
"name": {
|
|
||||||
"description": "Human readable name for this wallet",
|
|
||||||
"type": "string"
|
|
||||||
},
|
|
||||||
"version": {
|
|
||||||
"description": "Wallet spec version",
|
|
||||||
"type": "integer",
|
|
||||||
"$comment": "Should this be a string? We may need some sort of decimal type if we want exact decimal versions."
|
|
||||||
},
|
|
||||||
"accounts": {
|
|
||||||
"description": "Accounts associated with this wallet",
|
|
||||||
"type": "array",
|
|
||||||
"items": {
|
|
||||||
"type": "object",
|
|
||||||
"required": ["address_generator", "certificates", "encrypted", "ledger", "modified_on", "name", "private_key", "public_key", "seed"],
|
|
||||||
"additionalProperties": false,
|
|
||||||
"properties": {
|
|
||||||
"address_generator": {
|
|
||||||
"description": "Higher level manager of either singular or deterministically generated addresses",
|
|
||||||
"type": "object",
|
|
||||||
"oneOf": [
|
|
||||||
{
|
|
||||||
"required": ["name", "change", "receiving"],
|
|
||||||
"additionalProperties": false,
|
|
||||||
"properties": {
|
|
||||||
"name": {
|
|
||||||
"description": "type of address generator: a deterministic chain of addresses",
|
|
||||||
"enum": ["deterministic-chain"],
|
|
||||||
"type": "string"
|
|
||||||
},
|
|
||||||
"change": {
|
|
||||||
"$ref": "#/$defs/address_manager",
|
|
||||||
"description": "Manager for deterministically generated change address (not used for single address)"
|
|
||||||
},
|
|
||||||
"receiving": {
|
|
||||||
"$ref": "#/$defs/address_manager",
|
|
||||||
"description": "Manager for deterministically generated receiving address (not used for single address)"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}, {
|
|
||||||
"required": ["name"],
|
|
||||||
"additionalProperties": false,
|
|
||||||
"properties": {
|
|
||||||
"name": {
|
|
||||||
"description": "type of address generator: a single address",
|
|
||||||
"enum": ["single-address"],
|
|
||||||
"type": "string"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"certificates": {
|
|
||||||
"type": "object",
|
|
||||||
"description": "Channel keys. Mapping from public key address to pem-formatted private key.",
|
|
||||||
"additionalProperties": {"type": "string"}
|
|
||||||
},
|
|
||||||
"encrypted": {
|
|
||||||
"type": "boolean",
|
|
||||||
"description": "Whether private key and seed are encrypted with a password"
|
|
||||||
},
|
|
||||||
"ledger": {
|
|
||||||
"description": "Which network to use",
|
|
||||||
"type": "string",
|
|
||||||
"examples": [
|
|
||||||
"lbc_mainnet",
|
|
||||||
"lbc_testnet"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"modified_on": {
|
|
||||||
"description": "last modified time in Unix Time",
|
|
||||||
"type": "integer"
|
|
||||||
},
|
|
||||||
"name": {
|
|
||||||
"description": "Name for account, possibly human readable",
|
|
||||||
"type": "string"
|
|
||||||
},
|
|
||||||
"private_key": {
|
|
||||||
"description": "Private key for address if `address_generator` is a single address. Root of chain of private keys for addresses if `address_generator` is a deterministic chain of addresses. Encrypted if `encrypted` is true.",
|
|
||||||
"type": "string"
|
|
||||||
},
|
|
||||||
"public_key": {
|
|
||||||
"description": "Public key for address if `address_generator` is a single address. Root of chain of public keys for addresses if `address_generator` is a deterministic chain of addresses.",
|
|
||||||
"type": "string"
|
|
||||||
},
|
|
||||||
"seed": {
|
|
||||||
"description": "Human readable representation of `private_key`. encrypted if `encrypted` is set to `true`",
|
|
||||||
"type": "string"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"preferences": {
|
|
||||||
"description": "Timestamped application-level preferences. Values can be objects or of a primitive type.",
|
|
||||||
"$comment": "enable-sync is seen in example wallet. encrypt-on-disk is seen in example wallet. they both have a boolean `value` field. Do we want them explicitly defined here? local and shared seem to have at least a similar structure (type, value [yes, again], version), value being the free-form part. Should we define those here? Or can there be any key under preferences, and `value` be literally be anything in any form?",
|
|
||||||
"type": "object",
|
|
||||||
"additionalProperties": {
|
|
||||||
"type": "object",
|
|
||||||
"required": ["ts", "value"],
|
|
||||||
"additionalProperties": false,
|
|
||||||
"properties": {
|
|
||||||
"ts": {
|
|
||||||
"type": "number",
|
|
||||||
"description": "When the item was set, in Unix time format.",
|
|
||||||
"$comment": "Do we want a string (decimal)?"
|
|
||||||
},
|
|
||||||
"value": {
|
|
||||||
"$comment": "Sometimes this has been an object, sometimes just a boolean. I don't want to prescribe anything."
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"$defs": {
|
|
||||||
"address_manager": {
|
|
||||||
"description": "Manager for deterministically generated addresses",
|
|
||||||
"type": "object",
|
|
||||||
"required": ["gap", "maximum_uses_per_address"],
|
|
||||||
"additionalProperties": false,
|
|
||||||
"properties": {
|
|
||||||
"gap": {
|
|
||||||
"description": "Maximum allowed consecutive generated addresses with no transactions",
|
|
||||||
"type": "integer"
|
|
||||||
},
|
|
||||||
"maximum_uses_per_address": {
|
|
||||||
"description": "Maximum number of uses for each generated address",
|
|
||||||
"type": "integer"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,31 +0,0 @@
|
||||||
import asyncio
|
|
||||||
import logging
|
|
||||||
|
|
||||||
from lbry.stream.downloader import StreamDownloader
|
|
||||||
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class BackgroundDownloader:
|
|
||||||
def __init__(self, conf, storage, blob_manager, dht_node=None):
|
|
||||||
self.storage = storage
|
|
||||||
self.blob_manager = blob_manager
|
|
||||||
self.node = dht_node
|
|
||||||
self.conf = conf
|
|
||||||
|
|
||||||
async def download_blobs(self, sd_hash):
|
|
||||||
downloader = StreamDownloader(asyncio.get_running_loop(), self.conf, self.blob_manager, sd_hash)
|
|
||||||
try:
|
|
||||||
await downloader.start(self.node, save_stream=False)
|
|
||||||
for blob_info in downloader.descriptor.blobs[:-1]:
|
|
||||||
await downloader.download_stream_blob(blob_info)
|
|
||||||
except ValueError:
|
|
||||||
return
|
|
||||||
except asyncio.CancelledError:
|
|
||||||
log.debug("Cancelled background downloader")
|
|
||||||
raise
|
|
||||||
except Exception:
|
|
||||||
log.error("Unexpected download error on background downloader")
|
|
||||||
finally:
|
|
||||||
downloader.stop()
|
|
|
@ -194,13 +194,12 @@ class StreamDescriptor:
|
||||||
raise InvalidStreamDescriptorError("Stream terminator blob should not have a hash")
|
raise InvalidStreamDescriptorError("Stream terminator blob should not have a hash")
|
||||||
if any(i != blob_info['blob_num'] for i, blob_info in enumerate(decoded['blobs'])):
|
if any(i != blob_info['blob_num'] for i, blob_info in enumerate(decoded['blobs'])):
|
||||||
raise InvalidStreamDescriptorError("Stream contains out of order or skipped blobs")
|
raise InvalidStreamDescriptorError("Stream contains out of order or skipped blobs")
|
||||||
added_on = time.time()
|
|
||||||
descriptor = cls(
|
descriptor = cls(
|
||||||
loop, blob_dir,
|
loop, blob_dir,
|
||||||
binascii.unhexlify(decoded['stream_name']).decode(),
|
binascii.unhexlify(decoded['stream_name']).decode(),
|
||||||
decoded['key'],
|
decoded['key'],
|
||||||
binascii.unhexlify(decoded['suggested_file_name']).decode(),
|
binascii.unhexlify(decoded['suggested_file_name']).decode(),
|
||||||
[BlobInfo(info['blob_num'], info['length'], info['iv'], added_on, info.get('blob_hash'))
|
[BlobInfo(info['blob_num'], info['length'], info['iv'], info.get('blob_hash'))
|
||||||
for info in decoded['blobs']],
|
for info in decoded['blobs']],
|
||||||
decoded['stream_hash'],
|
decoded['stream_hash'],
|
||||||
blob.blob_hash
|
blob.blob_hash
|
||||||
|
@ -267,7 +266,7 @@ class StreamDescriptor:
|
||||||
blobs.append(blob_info)
|
blobs.append(blob_info)
|
||||||
blobs.append(
|
blobs.append(
|
||||||
# add the stream terminator
|
# add the stream terminator
|
||||||
BlobInfo(len(blobs), 0, binascii.hexlify(next(iv_generator)).decode(), added_on, None, True)
|
BlobInfo(len(blobs), 0, binascii.hexlify(next(iv_generator)).decode(), None, added_on, True)
|
||||||
)
|
)
|
||||||
file_name = os.path.basename(file_path)
|
file_name = os.path.basename(file_path)
|
||||||
suggested_file_name = sanitize_file_name(file_name)
|
suggested_file_name = sanitize_file_name(file_name)
|
||||||
|
|
|
@ -3,13 +3,11 @@ import typing
|
||||||
import logging
|
import logging
|
||||||
import binascii
|
import binascii
|
||||||
|
|
||||||
from lbry.dht.node import get_kademlia_peers_from_hosts
|
from lbry.dht.peer import make_kademlia_peer
|
||||||
from lbry.error import DownloadSDTimeoutError
|
from lbry.error import DownloadSDTimeoutError
|
||||||
from lbry.utils import lru_cache_concurrent
|
from lbry.utils import resolve_host, lru_cache_concurrent
|
||||||
from lbry.stream.descriptor import StreamDescriptor
|
from lbry.stream.descriptor import StreamDescriptor
|
||||||
from lbry.blob_exchange.downloader import BlobDownloader
|
from lbry.blob_exchange.downloader import BlobDownloader
|
||||||
from lbry.torrent.tracker import enqueue_tracker_search
|
|
||||||
|
|
||||||
if typing.TYPE_CHECKING:
|
if typing.TYPE_CHECKING:
|
||||||
from lbry.conf import Config
|
from lbry.conf import Config
|
||||||
from lbry.dht.node import Node
|
from lbry.dht.node import Node
|
||||||
|
@ -27,8 +25,8 @@ class StreamDownloader:
|
||||||
self.config = config
|
self.config = config
|
||||||
self.blob_manager = blob_manager
|
self.blob_manager = blob_manager
|
||||||
self.sd_hash = sd_hash
|
self.sd_hash = sd_hash
|
||||||
self.search_queue = asyncio.Queue() # blob hashes to feed into the iterative finder
|
self.search_queue = asyncio.Queue(loop=loop) # blob hashes to feed into the iterative finder
|
||||||
self.peer_queue = asyncio.Queue() # new peers to try
|
self.peer_queue = asyncio.Queue(loop=loop) # new peers to try
|
||||||
self.blob_downloader = BlobDownloader(self.loop, self.config, self.blob_manager, self.peer_queue)
|
self.blob_downloader = BlobDownloader(self.loop, self.config, self.blob_manager, self.peer_queue)
|
||||||
self.descriptor: typing.Optional[StreamDescriptor] = descriptor
|
self.descriptor: typing.Optional[StreamDescriptor] = descriptor
|
||||||
self.node: typing.Optional['Node'] = None
|
self.node: typing.Optional['Node'] = None
|
||||||
|
@ -50,19 +48,26 @@ class StreamDownloader:
|
||||||
self.cached_read_blob = cached_read_blob
|
self.cached_read_blob = cached_read_blob
|
||||||
|
|
||||||
async def add_fixed_peers(self):
|
async def add_fixed_peers(self):
|
||||||
def _add_fixed_peers(fixed_peers):
|
def _delayed_add_fixed_peers():
|
||||||
self.peer_queue.put_nowait(fixed_peers)
|
|
||||||
self.added_fixed_peers = True
|
self.added_fixed_peers = True
|
||||||
|
self.peer_queue.put_nowait([
|
||||||
|
make_kademlia_peer(None, address, None, tcp_port=port, allow_localhost=True)
|
||||||
|
for address, port in addresses
|
||||||
|
])
|
||||||
|
|
||||||
if not self.config.fixed_peers:
|
if not self.config.fixed_peers:
|
||||||
return
|
return
|
||||||
|
addresses = [
|
||||||
|
(await resolve_host(url, port, proto='tcp'), port)
|
||||||
|
for url, port in self.config.fixed_peers
|
||||||
|
]
|
||||||
if 'dht' in self.config.components_to_skip or not self.node or not \
|
if 'dht' in self.config.components_to_skip or not self.node or not \
|
||||||
len(self.node.protocol.routing_table.get_peers()) > 0:
|
len(self.node.protocol.routing_table.get_peers()) > 0:
|
||||||
self.fixed_peers_delay = 0.0
|
self.fixed_peers_delay = 0.0
|
||||||
else:
|
else:
|
||||||
self.fixed_peers_delay = self.config.fixed_peer_delay
|
self.fixed_peers_delay = self.config.fixed_peer_delay
|
||||||
fixed_peers = await get_kademlia_peers_from_hosts(self.config.fixed_peers)
|
|
||||||
self.fixed_peers_handle = self.loop.call_later(self.fixed_peers_delay, _add_fixed_peers, fixed_peers)
|
self.fixed_peers_handle = self.loop.call_later(self.fixed_peers_delay, _delayed_add_fixed_peers)
|
||||||
|
|
||||||
async def load_descriptor(self, connection_id: int = 0):
|
async def load_descriptor(self, connection_id: int = 0):
|
||||||
# download or get the sd blob
|
# download or get the sd blob
|
||||||
|
@ -72,7 +77,7 @@ class StreamDownloader:
|
||||||
now = self.loop.time()
|
now = self.loop.time()
|
||||||
sd_blob = await asyncio.wait_for(
|
sd_blob = await asyncio.wait_for(
|
||||||
self.blob_downloader.download_blob(self.sd_hash, connection_id),
|
self.blob_downloader.download_blob(self.sd_hash, connection_id),
|
||||||
self.config.blob_download_timeout
|
self.config.blob_download_timeout, loop=self.loop
|
||||||
)
|
)
|
||||||
log.info("downloaded sd blob %s", self.sd_hash)
|
log.info("downloaded sd blob %s", self.sd_hash)
|
||||||
self.time_to_descriptor = self.loop.time() - now
|
self.time_to_descriptor = self.loop.time() - now
|
||||||
|
@ -85,7 +90,7 @@ class StreamDownloader:
|
||||||
)
|
)
|
||||||
log.info("loaded stream manifest %s", self.sd_hash)
|
log.info("loaded stream manifest %s", self.sd_hash)
|
||||||
|
|
||||||
async def start(self, node: typing.Optional['Node'] = None, connection_id: int = 0, save_stream=True):
|
async def start(self, node: typing.Optional['Node'] = None, connection_id: int = 0):
|
||||||
# set up peer accumulation
|
# set up peer accumulation
|
||||||
self.node = node or self.node # fixme: this shouldnt be set here!
|
self.node = node or self.node # fixme: this shouldnt be set here!
|
||||||
if self.node:
|
if self.node:
|
||||||
|
@ -93,7 +98,6 @@ class StreamDownloader:
|
||||||
self.accumulate_task.cancel()
|
self.accumulate_task.cancel()
|
||||||
_, self.accumulate_task = self.node.accumulate_peers(self.search_queue, self.peer_queue)
|
_, self.accumulate_task = self.node.accumulate_peers(self.search_queue, self.peer_queue)
|
||||||
await self.add_fixed_peers()
|
await self.add_fixed_peers()
|
||||||
enqueue_tracker_search(bytes.fromhex(self.sd_hash), self.peer_queue)
|
|
||||||
# start searching for peers for the sd hash
|
# start searching for peers for the sd hash
|
||||||
self.search_queue.put_nowait(self.sd_hash)
|
self.search_queue.put_nowait(self.sd_hash)
|
||||||
log.info("searching for peers for stream %s", self.sd_hash)
|
log.info("searching for peers for stream %s", self.sd_hash)
|
||||||
|
@ -101,7 +105,11 @@ class StreamDownloader:
|
||||||
if not self.descriptor:
|
if not self.descriptor:
|
||||||
await self.load_descriptor(connection_id)
|
await self.load_descriptor(connection_id)
|
||||||
|
|
||||||
if not await self.blob_manager.storage.stream_exists(self.sd_hash) and save_stream:
|
# add the head blob to the peer search
|
||||||
|
self.search_queue.put_nowait(self.descriptor.blobs[0].blob_hash)
|
||||||
|
log.info("added head blob to peer search for stream %s", self.sd_hash)
|
||||||
|
|
||||||
|
if not await self.blob_manager.storage.stream_exists(self.sd_hash):
|
||||||
await self.blob_manager.storage.store_stream(
|
await self.blob_manager.storage.store_stream(
|
||||||
self.blob_manager.get_blob(self.sd_hash, length=self.descriptor.length), self.descriptor
|
self.blob_manager.get_blob(self.sd_hash, length=self.descriptor.length), self.descriptor
|
||||||
)
|
)
|
||||||
|
@ -111,7 +119,7 @@ class StreamDownloader:
|
||||||
raise ValueError(f"blob {blob_info.blob_hash} is not part of stream with sd hash {self.sd_hash}")
|
raise ValueError(f"blob {blob_info.blob_hash} is not part of stream with sd hash {self.sd_hash}")
|
||||||
blob = await asyncio.wait_for(
|
blob = await asyncio.wait_for(
|
||||||
self.blob_downloader.download_blob(blob_info.blob_hash, blob_info.length, connection_id),
|
self.blob_downloader.download_blob(blob_info.blob_hash, blob_info.length, connection_id),
|
||||||
self.config.blob_download_timeout * 10
|
self.config.blob_download_timeout * 10, loop=self.loop
|
||||||
)
|
)
|
||||||
return blob
|
return blob
|
||||||
|
|
||||||
|
|
|
@ -16,8 +16,10 @@ from lbry.file.source import ManagedDownloadSource
|
||||||
|
|
||||||
if typing.TYPE_CHECKING:
|
if typing.TYPE_CHECKING:
|
||||||
from lbry.conf import Config
|
from lbry.conf import Config
|
||||||
|
from lbry.schema.claim import Claim
|
||||||
from lbry.blob.blob_manager import BlobManager
|
from lbry.blob.blob_manager import BlobManager
|
||||||
from lbry.blob.blob_info import BlobInfo
|
from lbry.blob.blob_info import BlobInfo
|
||||||
|
from lbry.dht.node import Node
|
||||||
from lbry.extras.daemon.analytics import AnalyticsManager
|
from lbry.extras.daemon.analytics import AnalyticsManager
|
||||||
from lbry.wallet.transaction import Transaction
|
from lbry.wallet.transaction import Transaction
|
||||||
|
|
||||||
|
@ -60,9 +62,9 @@ class ManagedStream(ManagedDownloadSource):
|
||||||
self.file_output_task: typing.Optional[asyncio.Task] = None
|
self.file_output_task: typing.Optional[asyncio.Task] = None
|
||||||
self.delayed_stop_task: typing.Optional[asyncio.Task] = None
|
self.delayed_stop_task: typing.Optional[asyncio.Task] = None
|
||||||
self.streaming_responses: typing.List[typing.Tuple[Request, StreamResponse]] = []
|
self.streaming_responses: typing.List[typing.Tuple[Request, StreamResponse]] = []
|
||||||
self.fully_reflected = asyncio.Event()
|
self.fully_reflected = asyncio.Event(loop=self.loop)
|
||||||
self.streaming = asyncio.Event()
|
self.streaming = asyncio.Event(loop=self.loop)
|
||||||
self._running = asyncio.Event()
|
self._running = asyncio.Event(loop=self.loop)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def sd_hash(self) -> str:
|
def sd_hash(self) -> str:
|
||||||
|
@ -82,19 +84,7 @@ class ManagedStream(ManagedDownloadSource):
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def file_name(self) -> Optional[str]:
|
def file_name(self) -> Optional[str]:
|
||||||
return self._file_name or self.suggested_file_name
|
return self._file_name or (self.descriptor.suggested_file_name if self.descriptor else None)
|
||||||
|
|
||||||
@property
|
|
||||||
def suggested_file_name(self) -> Optional[str]:
|
|
||||||
first_option = ((self.descriptor and self.descriptor.suggested_file_name) or '').strip()
|
|
||||||
return sanitize_file_name(first_option or (self.stream_claim_info and self.stream_claim_info.claim and
|
|
||||||
self.stream_claim_info.claim.stream.source.name))
|
|
||||||
|
|
||||||
@property
|
|
||||||
def stream_name(self) -> Optional[str]:
|
|
||||||
first_option = ((self.descriptor and self.descriptor.stream_name) or '').strip()
|
|
||||||
return first_option or (self.stream_claim_info and self.stream_claim_info.claim and
|
|
||||||
self.stream_claim_info.claim.stream.source.name)
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def written_bytes(self) -> int:
|
def written_bytes(self) -> int:
|
||||||
|
@ -128,7 +118,7 @@ class ManagedStream(ManagedDownloadSource):
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def mime_type(self):
|
def mime_type(self):
|
||||||
return guess_media_type(os.path.basename(self.suggested_file_name))[0]
|
return guess_media_type(os.path.basename(self.descriptor.suggested_file_name))[0]
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def download_path(self):
|
def download_path(self):
|
||||||
|
@ -161,7 +151,7 @@ class ManagedStream(ManagedDownloadSource):
|
||||||
log.info("start downloader for stream (sd hash: %s)", self.sd_hash)
|
log.info("start downloader for stream (sd hash: %s)", self.sd_hash)
|
||||||
self._running.set()
|
self._running.set()
|
||||||
try:
|
try:
|
||||||
await asyncio.wait_for(self.downloader.start(), timeout)
|
await asyncio.wait_for(self.downloader.start(), timeout, loop=self.loop)
|
||||||
except asyncio.TimeoutError:
|
except asyncio.TimeoutError:
|
||||||
self._running.clear()
|
self._running.clear()
|
||||||
raise DownloadSDTimeoutError(self.sd_hash)
|
raise DownloadSDTimeoutError(self.sd_hash)
|
||||||
|
@ -174,7 +164,7 @@ class ManagedStream(ManagedDownloadSource):
|
||||||
if not self._file_name:
|
if not self._file_name:
|
||||||
self._file_name = await get_next_available_file_name(
|
self._file_name = await get_next_available_file_name(
|
||||||
self.loop, self.download_directory,
|
self.loop, self.download_directory,
|
||||||
self._file_name or sanitize_file_name(self.suggested_file_name)
|
self._file_name or sanitize_file_name(self.descriptor.suggested_file_name)
|
||||||
)
|
)
|
||||||
file_name, download_dir = self._file_name, self.download_directory
|
file_name, download_dir = self._file_name, self.download_directory
|
||||||
else:
|
else:
|
||||||
|
@ -191,7 +181,7 @@ class ManagedStream(ManagedDownloadSource):
|
||||||
Stop any running save/stream tasks as well as the downloader and update the status in the database
|
Stop any running save/stream tasks as well as the downloader and update the status in the database
|
||||||
"""
|
"""
|
||||||
|
|
||||||
await self.stop_tasks()
|
self.stop_tasks()
|
||||||
if (finished and self.status != self.STATUS_FINISHED) or self.status == self.STATUS_RUNNING:
|
if (finished and self.status != self.STATUS_FINISHED) or self.status == self.STATUS_RUNNING:
|
||||||
await self.update_status(self.STATUS_FINISHED if finished else self.STATUS_STOPPED)
|
await self.update_status(self.STATUS_FINISHED if finished else self.STATUS_STOPPED)
|
||||||
|
|
||||||
|
@ -279,7 +269,7 @@ class ManagedStream(ManagedDownloadSource):
|
||||||
log.info("finished saving file for lbry://%s#%s (sd hash %s...) -> %s", self.claim_name, self.claim_id,
|
log.info("finished saving file for lbry://%s#%s (sd hash %s...) -> %s", self.claim_name, self.claim_id,
|
||||||
self.sd_hash[:6], self.full_path)
|
self.sd_hash[:6], self.full_path)
|
||||||
await self.blob_manager.storage.set_saved_file(self.stream_hash)
|
await self.blob_manager.storage.set_saved_file(self.stream_hash)
|
||||||
except (Exception, asyncio.CancelledError) as err:
|
except Exception as err:
|
||||||
if os.path.isfile(output_path):
|
if os.path.isfile(output_path):
|
||||||
log.warning("removing incomplete download %s for %s", output_path, self.sd_hash)
|
log.warning("removing incomplete download %s for %s", output_path, self.sd_hash)
|
||||||
os.remove(output_path)
|
os.remove(output_path)
|
||||||
|
@ -306,14 +296,14 @@ class ManagedStream(ManagedDownloadSource):
|
||||||
self.download_directory = download_directory or self.download_directory or self.config.download_dir
|
self.download_directory = download_directory or self.download_directory or self.config.download_dir
|
||||||
if not self.download_directory:
|
if not self.download_directory:
|
||||||
raise ValueError("no directory to download to")
|
raise ValueError("no directory to download to")
|
||||||
if not (file_name or self._file_name or self.suggested_file_name):
|
if not (file_name or self._file_name or self.descriptor.suggested_file_name):
|
||||||
raise ValueError("no file name to download to")
|
raise ValueError("no file name to download to")
|
||||||
if not os.path.isdir(self.download_directory):
|
if not os.path.isdir(self.download_directory):
|
||||||
log.warning("download directory '%s' does not exist, attempting to make it", self.download_directory)
|
log.warning("download directory '%s' does not exist, attempting to make it", self.download_directory)
|
||||||
os.mkdir(self.download_directory)
|
os.mkdir(self.download_directory)
|
||||||
self._file_name = await get_next_available_file_name(
|
self._file_name = await get_next_available_file_name(
|
||||||
self.loop, self.download_directory,
|
self.loop, self.download_directory,
|
||||||
file_name or self._file_name or sanitize_file_name(self.suggested_file_name)
|
file_name or self._file_name or sanitize_file_name(self.descriptor.suggested_file_name)
|
||||||
)
|
)
|
||||||
await self.blob_manager.storage.change_file_download_dir_and_file_name(
|
await self.blob_manager.storage.change_file_download_dir_and_file_name(
|
||||||
self.stream_hash, self.download_directory, self.file_name
|
self.stream_hash, self.download_directory, self.file_name
|
||||||
|
@ -321,16 +311,15 @@ class ManagedStream(ManagedDownloadSource):
|
||||||
await self.update_status(ManagedStream.STATUS_RUNNING)
|
await self.update_status(ManagedStream.STATUS_RUNNING)
|
||||||
self.file_output_task = self.loop.create_task(self._save_file(self.full_path))
|
self.file_output_task = self.loop.create_task(self._save_file(self.full_path))
|
||||||
try:
|
try:
|
||||||
await asyncio.wait_for(self.started_writing.wait(), self.config.download_timeout)
|
await asyncio.wait_for(self.started_writing.wait(), self.config.download_timeout, loop=self.loop)
|
||||||
except asyncio.TimeoutError:
|
except asyncio.TimeoutError:
|
||||||
log.warning("timeout starting to write data for lbry://%s#%s", self.claim_name, self.claim_id)
|
log.warning("timeout starting to write data for lbry://%s#%s", self.claim_name, self.claim_id)
|
||||||
await self.stop_tasks()
|
self.stop_tasks()
|
||||||
await self.update_status(ManagedStream.STATUS_STOPPED)
|
await self.update_status(ManagedStream.STATUS_STOPPED)
|
||||||
|
|
||||||
async def stop_tasks(self):
|
def stop_tasks(self):
|
||||||
if self.file_output_task and not self.file_output_task.done():
|
if self.file_output_task and not self.file_output_task.done():
|
||||||
self.file_output_task.cancel()
|
self.file_output_task.cancel()
|
||||||
await asyncio.gather(self.file_output_task, return_exceptions=True)
|
|
||||||
self.file_output_task = None
|
self.file_output_task = None
|
||||||
while self.streaming_responses:
|
while self.streaming_responses:
|
||||||
req, response = self.streaming_responses.pop()
|
req, response = self.streaming_responses.pop()
|
||||||
|
@ -367,7 +356,7 @@ class ManagedStream(ManagedDownloadSource):
|
||||||
return sent
|
return sent
|
||||||
except ConnectionError:
|
except ConnectionError:
|
||||||
return sent
|
return sent
|
||||||
except (OSError, Exception, asyncio.CancelledError) as err:
|
except (OSError, Exception) as err:
|
||||||
if isinstance(err, asyncio.CancelledError):
|
if isinstance(err, asyncio.CancelledError):
|
||||||
log.warning("stopped uploading %s#%s to reflector", self.claim_name, self.claim_id)
|
log.warning("stopped uploading %s#%s to reflector", self.claim_name, self.claim_id)
|
||||||
elif isinstance(err, OSError):
|
elif isinstance(err, OSError):
|
||||||
|
@ -402,7 +391,7 @@ class ManagedStream(ManagedDownloadSource):
|
||||||
self.sd_hash[:6])
|
self.sd_hash[:6])
|
||||||
await self.stop()
|
await self.stop()
|
||||||
return
|
return
|
||||||
await asyncio.sleep(1)
|
await asyncio.sleep(1, loop=self.loop)
|
||||||
|
|
||||||
def _prepare_range_response_headers(self, get_range: str) -> typing.Tuple[typing.Dict[str, str], int, int, int]:
|
def _prepare_range_response_headers(self, get_range: str) -> typing.Tuple[typing.Dict[str, str], int, int, int]:
|
||||||
if '=' in get_range:
|
if '=' in get_range:
|
||||||
|
|
|
@ -21,7 +21,7 @@ class ReflectorServerProtocol(asyncio.Protocol):
|
||||||
self.loop = asyncio.get_event_loop()
|
self.loop = asyncio.get_event_loop()
|
||||||
self.blob_manager = blob_manager
|
self.blob_manager = blob_manager
|
||||||
self.server_task: asyncio.Task = None
|
self.server_task: asyncio.Task = None
|
||||||
self.started_listening = asyncio.Event()
|
self.started_listening = asyncio.Event(loop=self.loop)
|
||||||
self.buf = b''
|
self.buf = b''
|
||||||
self.transport: asyncio.StreamWriter = None
|
self.transport: asyncio.StreamWriter = None
|
||||||
self.writer: typing.Optional['HashBlobWriter'] = None
|
self.writer: typing.Optional['HashBlobWriter'] = None
|
||||||
|
@ -29,9 +29,9 @@ class ReflectorServerProtocol(asyncio.Protocol):
|
||||||
self.descriptor: typing.Optional['StreamDescriptor'] = None
|
self.descriptor: typing.Optional['StreamDescriptor'] = None
|
||||||
self.sd_blob: typing.Optional['BlobFile'] = None
|
self.sd_blob: typing.Optional['BlobFile'] = None
|
||||||
self.received = []
|
self.received = []
|
||||||
self.incoming = incoming_event or asyncio.Event()
|
self.incoming = incoming_event or asyncio.Event(loop=self.loop)
|
||||||
self.not_incoming = not_incoming_event or asyncio.Event()
|
self.not_incoming = not_incoming_event or asyncio.Event(loop=self.loop)
|
||||||
self.stop_event = stop_event or asyncio.Event()
|
self.stop_event = stop_event or asyncio.Event(loop=self.loop)
|
||||||
self.chunk_size = response_chunk_size
|
self.chunk_size = response_chunk_size
|
||||||
self.wait_for_stop_task: typing.Optional[asyncio.Task] = None
|
self.wait_for_stop_task: typing.Optional[asyncio.Task] = None
|
||||||
self.partial_event = partial_event
|
self.partial_event = partial_event
|
||||||
|
@ -94,7 +94,7 @@ class ReflectorServerProtocol(asyncio.Protocol):
|
||||||
self.incoming.set()
|
self.incoming.set()
|
||||||
self.send_response({"send_sd_blob": True})
|
self.send_response({"send_sd_blob": True})
|
||||||
try:
|
try:
|
||||||
await asyncio.wait_for(self.sd_blob.verified.wait(), 30)
|
await asyncio.wait_for(self.sd_blob.verified.wait(), 30, loop=self.loop)
|
||||||
self.descriptor = await StreamDescriptor.from_stream_descriptor_blob(
|
self.descriptor = await StreamDescriptor.from_stream_descriptor_blob(
|
||||||
self.loop, self.blob_manager.blob_dir, self.sd_blob
|
self.loop, self.blob_manager.blob_dir, self.sd_blob
|
||||||
)
|
)
|
||||||
|
@ -140,7 +140,7 @@ class ReflectorServerProtocol(asyncio.Protocol):
|
||||||
self.incoming.set()
|
self.incoming.set()
|
||||||
self.send_response({"send_blob": True})
|
self.send_response({"send_blob": True})
|
||||||
try:
|
try:
|
||||||
await asyncio.wait_for(blob.verified.wait(), 30)
|
await asyncio.wait_for(blob.verified.wait(), 30, loop=self.loop)
|
||||||
self.send_response({"received_blob": True})
|
self.send_response({"received_blob": True})
|
||||||
except asyncio.TimeoutError:
|
except asyncio.TimeoutError:
|
||||||
self.send_response({"received_blob": False})
|
self.send_response({"received_blob": False})
|
||||||
|
@ -162,10 +162,10 @@ class ReflectorServer:
|
||||||
self.loop = asyncio.get_event_loop()
|
self.loop = asyncio.get_event_loop()
|
||||||
self.blob_manager = blob_manager
|
self.blob_manager = blob_manager
|
||||||
self.server_task: typing.Optional[asyncio.Task] = None
|
self.server_task: typing.Optional[asyncio.Task] = None
|
||||||
self.started_listening = asyncio.Event()
|
self.started_listening = asyncio.Event(loop=self.loop)
|
||||||
self.stopped_listening = asyncio.Event()
|
self.stopped_listening = asyncio.Event(loop=self.loop)
|
||||||
self.incoming_event = incoming_event or asyncio.Event()
|
self.incoming_event = incoming_event or asyncio.Event(loop=self.loop)
|
||||||
self.not_incoming_event = not_incoming_event or asyncio.Event()
|
self.not_incoming_event = not_incoming_event or asyncio.Event(loop=self.loop)
|
||||||
self.response_chunk_size = response_chunk_size
|
self.response_chunk_size = response_chunk_size
|
||||||
self.stop_event = stop_event
|
self.stop_event = stop_event
|
||||||
self.partial_needs = partial_needs # for testing cases where it doesn't know what it wants
|
self.partial_needs = partial_needs # for testing cases where it doesn't know what it wants
|
||||||
|
|
|
@ -54,7 +54,7 @@ class StreamManager(SourceManager):
|
||||||
self.re_reflect_task: Optional[asyncio.Task] = None
|
self.re_reflect_task: Optional[asyncio.Task] = None
|
||||||
self.update_stream_finished_futs: typing.List[asyncio.Future] = []
|
self.update_stream_finished_futs: typing.List[asyncio.Future] = []
|
||||||
self.running_reflector_uploads: typing.Dict[str, asyncio.Task] = {}
|
self.running_reflector_uploads: typing.Dict[str, asyncio.Task] = {}
|
||||||
self.started = asyncio.Event()
|
self.started = asyncio.Event(loop=self.loop)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def streams(self):
|
def streams(self):
|
||||||
|
@ -70,7 +70,6 @@ class StreamManager(SourceManager):
|
||||||
|
|
||||||
async def recover_streams(self, file_infos: typing.List[typing.Dict]):
|
async def recover_streams(self, file_infos: typing.List[typing.Dict]):
|
||||||
to_restore = []
|
to_restore = []
|
||||||
to_check = []
|
|
||||||
|
|
||||||
async def recover_stream(sd_hash: str, stream_hash: str, stream_name: str,
|
async def recover_stream(sd_hash: str, stream_hash: str, stream_name: str,
|
||||||
suggested_file_name: str, key: str,
|
suggested_file_name: str, key: str,
|
||||||
|
@ -83,7 +82,6 @@ class StreamManager(SourceManager):
|
||||||
if not descriptor:
|
if not descriptor:
|
||||||
return
|
return
|
||||||
to_restore.append((descriptor, sd_blob, content_fee))
|
to_restore.append((descriptor, sd_blob, content_fee))
|
||||||
to_check.extend([sd_blob.blob_hash] + [blob.blob_hash for blob in descriptor.blobs[:-1]])
|
|
||||||
|
|
||||||
await asyncio.gather(*[
|
await asyncio.gather(*[
|
||||||
recover_stream(
|
recover_stream(
|
||||||
|
@ -95,8 +93,6 @@ class StreamManager(SourceManager):
|
||||||
|
|
||||||
if to_restore:
|
if to_restore:
|
||||||
await self.storage.recover_streams(to_restore, self.config.download_dir)
|
await self.storage.recover_streams(to_restore, self.config.download_dir)
|
||||||
if to_check:
|
|
||||||
await self.blob_manager.ensure_completed_blobs_status(to_check)
|
|
||||||
|
|
||||||
# if self.blob_manager._save_blobs:
|
# if self.blob_manager._save_blobs:
|
||||||
# log.info("Recovered %i/%i attempted streams", len(to_restore), len(file_infos))
|
# log.info("Recovered %i/%i attempted streams", len(to_restore), len(file_infos))
|
||||||
|
@ -150,7 +146,7 @@ class StreamManager(SourceManager):
|
||||||
file_info['added_on'], file_info['fully_reflected']
|
file_info['added_on'], file_info['fully_reflected']
|
||||||
)))
|
)))
|
||||||
if add_stream_tasks:
|
if add_stream_tasks:
|
||||||
await asyncio.gather(*add_stream_tasks)
|
await asyncio.gather(*add_stream_tasks, loop=self.loop)
|
||||||
log.info("Started stream manager with %i files", len(self._sources))
|
log.info("Started stream manager with %i files", len(self._sources))
|
||||||
if not self.node:
|
if not self.node:
|
||||||
log.info("no DHT node given, resuming downloads trusting that we can contact reflector")
|
log.info("no DHT node given, resuming downloads trusting that we can contact reflector")
|
||||||
|
@ -159,11 +155,14 @@ class StreamManager(SourceManager):
|
||||||
self.resume_saving_task = asyncio.ensure_future(asyncio.gather(
|
self.resume_saving_task = asyncio.ensure_future(asyncio.gather(
|
||||||
*(self._sources[sd_hash].save_file(file_name, download_directory)
|
*(self._sources[sd_hash].save_file(file_name, download_directory)
|
||||||
for (file_name, download_directory, sd_hash) in to_resume_saving),
|
for (file_name, download_directory, sd_hash) in to_resume_saving),
|
||||||
|
loop=self.loop
|
||||||
))
|
))
|
||||||
|
|
||||||
async def reflect_streams(self):
|
async def reflect_streams(self):
|
||||||
try:
|
try:
|
||||||
return await self._reflect_streams()
|
return await self._reflect_streams()
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
raise
|
||||||
except Exception:
|
except Exception:
|
||||||
log.exception("reflector task encountered an unexpected error!")
|
log.exception("reflector task encountered an unexpected error!")
|
||||||
|
|
||||||
|
@ -183,21 +182,21 @@ class StreamManager(SourceManager):
|
||||||
batch.append(self.reflect_stream(stream))
|
batch.append(self.reflect_stream(stream))
|
||||||
if len(batch) >= self.config.concurrent_reflector_uploads:
|
if len(batch) >= self.config.concurrent_reflector_uploads:
|
||||||
log.debug("waiting for batch of %s reflecting streams", len(batch))
|
log.debug("waiting for batch of %s reflecting streams", len(batch))
|
||||||
await asyncio.gather(*batch)
|
await asyncio.gather(*batch, loop=self.loop)
|
||||||
log.debug("done processing %s streams", len(batch))
|
log.debug("done processing %s streams", len(batch))
|
||||||
batch = []
|
batch = []
|
||||||
if batch:
|
if batch:
|
||||||
log.debug("waiting for batch of %s reflecting streams", len(batch))
|
log.debug("waiting for batch of %s reflecting streams", len(batch))
|
||||||
await asyncio.gather(*batch)
|
await asyncio.gather(*batch, loop=self.loop)
|
||||||
log.debug("done processing %s streams", len(batch))
|
log.debug("done processing %s streams", len(batch))
|
||||||
await asyncio.sleep(300)
|
await asyncio.sleep(300, loop=self.loop)
|
||||||
|
|
||||||
async def start(self):
|
async def start(self):
|
||||||
await super().start()
|
await super().start()
|
||||||
self.re_reflect_task = self.loop.create_task(self.reflect_streams())
|
self.re_reflect_task = self.loop.create_task(self.reflect_streams())
|
||||||
|
|
||||||
async def stop(self):
|
def stop(self):
|
||||||
await super().stop()
|
super().stop()
|
||||||
if self.resume_saving_task and not self.resume_saving_task.done():
|
if self.resume_saving_task and not self.resume_saving_task.done():
|
||||||
self.resume_saving_task.cancel()
|
self.resume_saving_task.cancel()
|
||||||
if self.re_reflect_task and not self.re_reflect_task.done():
|
if self.re_reflect_task and not self.re_reflect_task.done():
|
||||||
|
@ -224,8 +223,7 @@ class StreamManager(SourceManager):
|
||||||
)
|
)
|
||||||
return task
|
return task
|
||||||
|
|
||||||
@staticmethod
|
async def _retriable_reflect_stream(self, stream, host, port):
|
||||||
async def _retriable_reflect_stream(stream, host, port):
|
|
||||||
sent = await stream.upload_to_reflector(host, port)
|
sent = await stream.upload_to_reflector(host, port)
|
||||||
while not stream.is_fully_reflected and stream.reflector_progress > 0 and len(sent) > 0:
|
while not stream.is_fully_reflected and stream.reflector_progress > 0 and len(sent) > 0:
|
||||||
stream.reflector_progress = 0
|
stream.reflector_progress = 0
|
||||||
|
@ -260,7 +258,7 @@ class StreamManager(SourceManager):
|
||||||
return
|
return
|
||||||
if source.identifier in self.running_reflector_uploads:
|
if source.identifier in self.running_reflector_uploads:
|
||||||
self.running_reflector_uploads[source.identifier].cancel()
|
self.running_reflector_uploads[source.identifier].cancel()
|
||||||
await source.stop_tasks()
|
source.stop_tasks()
|
||||||
if source.identifier in self.streams:
|
if source.identifier in self.streams:
|
||||||
del self.streams[source.identifier]
|
del self.streams[source.identifier]
|
||||||
blob_hashes = [source.identifier] + [b.blob_hash for b in source.descriptor.blobs[:-1]]
|
blob_hashes = [source.identifier] + [b.blob_hash for b in source.descriptor.blobs[:-1]]
|
||||||
|
|
118
lbry/testcase.py
118
lbry/testcase.py
|
@ -17,10 +17,8 @@ from functools import partial
|
||||||
from lbry.wallet import WalletManager, Wallet, Ledger, Account, Transaction
|
from lbry.wallet import WalletManager, Wallet, Ledger, Account, Transaction
|
||||||
from lbry.conf import Config
|
from lbry.conf import Config
|
||||||
from lbry.wallet.util import satoshis_to_coins
|
from lbry.wallet.util import satoshis_to_coins
|
||||||
from lbry.wallet.dewies import lbc_to_dewies
|
|
||||||
from lbry.wallet.orchstr8 import Conductor
|
from lbry.wallet.orchstr8 import Conductor
|
||||||
from lbry.wallet.orchstr8.node import LBCWalletNode, WalletNode
|
from lbry.wallet.orchstr8.node import BlockchainNode, WalletNode, HubNode
|
||||||
from lbry.schema.claim import Claim
|
|
||||||
|
|
||||||
from lbry.extras.daemon.daemon import Daemon, jsonrpc_dumps_pretty
|
from lbry.extras.daemon.daemon import Daemon, jsonrpc_dumps_pretty
|
||||||
from lbry.extras.daemon.components import Component, WalletComponent
|
from lbry.extras.daemon.components import Component, WalletComponent
|
||||||
|
@ -134,18 +132,17 @@ class AsyncioTestCase(unittest.TestCase):
|
||||||
|
|
||||||
with outcome.testPartExecutor(self):
|
with outcome.testPartExecutor(self):
|
||||||
self.setUp()
|
self.setUp()
|
||||||
self.add_timeout()
|
|
||||||
self.loop.run_until_complete(self.asyncSetUp())
|
self.loop.run_until_complete(self.asyncSetUp())
|
||||||
if outcome.success:
|
if outcome.success:
|
||||||
outcome.expecting_failure = expecting_failure
|
outcome.expecting_failure = expecting_failure
|
||||||
with outcome.testPartExecutor(self, isTest=True):
|
with outcome.testPartExecutor(self, isTest=True):
|
||||||
maybe_coroutine = testMethod()
|
maybe_coroutine = testMethod()
|
||||||
if asyncio.iscoroutine(maybe_coroutine):
|
if asyncio.iscoroutine(maybe_coroutine):
|
||||||
self.add_timeout()
|
if self.TIMEOUT:
|
||||||
|
self.loop.call_later(self.TIMEOUT, self.cancel)
|
||||||
self.loop.run_until_complete(maybe_coroutine)
|
self.loop.run_until_complete(maybe_coroutine)
|
||||||
outcome.expecting_failure = False
|
outcome.expecting_failure = False
|
||||||
with outcome.testPartExecutor(self):
|
with outcome.testPartExecutor(self):
|
||||||
self.add_timeout()
|
|
||||||
self.loop.run_until_complete(self.asyncTearDown())
|
self.loop.run_until_complete(self.asyncTearDown())
|
||||||
self.tearDown()
|
self.tearDown()
|
||||||
|
|
||||||
|
@ -193,7 +190,6 @@ class AsyncioTestCase(unittest.TestCase):
|
||||||
with outcome.testPartExecutor(self):
|
with outcome.testPartExecutor(self):
|
||||||
maybe_coroutine = function(*args, **kwargs)
|
maybe_coroutine = function(*args, **kwargs)
|
||||||
if asyncio.iscoroutine(maybe_coroutine):
|
if asyncio.iscoroutine(maybe_coroutine):
|
||||||
self.add_timeout()
|
|
||||||
self.loop.run_until_complete(maybe_coroutine)
|
self.loop.run_until_complete(maybe_coroutine)
|
||||||
|
|
||||||
def cancel(self):
|
def cancel(self):
|
||||||
|
@ -202,16 +198,6 @@ class AsyncioTestCase(unittest.TestCase):
|
||||||
task.print_stack()
|
task.print_stack()
|
||||||
task.cancel()
|
task.cancel()
|
||||||
|
|
||||||
def add_timeout(self):
|
|
||||||
if self.TIMEOUT:
|
|
||||||
self.loop.call_later(self.TIMEOUT, self.check_timeout, time())
|
|
||||||
|
|
||||||
def check_timeout(self, started):
|
|
||||||
if time() - started >= self.TIMEOUT:
|
|
||||||
self.cancel()
|
|
||||||
else:
|
|
||||||
self.loop.call_later(self.TIMEOUT, self.check_timeout, started)
|
|
||||||
|
|
||||||
|
|
||||||
class AdvanceTimeTestCase(AsyncioTestCase):
|
class AdvanceTimeTestCase(AsyncioTestCase):
|
||||||
|
|
||||||
|
@ -236,7 +222,8 @@ class IntegrationTestCase(AsyncioTestCase):
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
super().__init__(*args, **kwargs)
|
super().__init__(*args, **kwargs)
|
||||||
self.conductor: Optional[Conductor] = None
|
self.conductor: Optional[Conductor] = None
|
||||||
self.blockchain: Optional[LBCWalletNode] = None
|
self.blockchain: Optional[BlockchainNode] = None
|
||||||
|
self.hub: Optional[HubNode] = None
|
||||||
self.wallet_node: Optional[WalletNode] = None
|
self.wallet_node: Optional[WalletNode] = None
|
||||||
self.manager: Optional[WalletManager] = None
|
self.manager: Optional[WalletManager] = None
|
||||||
self.ledger: Optional[Ledger] = None
|
self.ledger: Optional[Ledger] = None
|
||||||
|
@ -245,15 +232,16 @@ class IntegrationTestCase(AsyncioTestCase):
|
||||||
|
|
||||||
async def asyncSetUp(self):
|
async def asyncSetUp(self):
|
||||||
self.conductor = Conductor(seed=self.SEED)
|
self.conductor = Conductor(seed=self.SEED)
|
||||||
await self.conductor.start_lbcd()
|
await self.conductor.start_blockchain()
|
||||||
self.addCleanup(self.conductor.stop_lbcd)
|
self.addCleanup(self.conductor.stop_blockchain)
|
||||||
await self.conductor.start_lbcwallet()
|
|
||||||
self.addCleanup(self.conductor.stop_lbcwallet)
|
|
||||||
await self.conductor.start_spv()
|
await self.conductor.start_spv()
|
||||||
self.addCleanup(self.conductor.stop_spv)
|
self.addCleanup(self.conductor.stop_spv)
|
||||||
await self.conductor.start_wallet()
|
await self.conductor.start_wallet()
|
||||||
self.addCleanup(self.conductor.stop_wallet)
|
self.addCleanup(self.conductor.stop_wallet)
|
||||||
self.blockchain = self.conductor.lbcwallet_node
|
await self.conductor.start_hub()
|
||||||
|
self.addCleanup(self.conductor.stop_hub)
|
||||||
|
self.blockchain = self.conductor.blockchain_node
|
||||||
|
self.hub = self.conductor.hub_node
|
||||||
self.wallet_node = self.conductor.wallet_node
|
self.wallet_node = self.conductor.wallet_node
|
||||||
self.manager = self.wallet_node.manager
|
self.manager = self.wallet_node.manager
|
||||||
self.ledger = self.wallet_node.ledger
|
self.ledger = self.wallet_node.ledger
|
||||||
|
@ -267,13 +255,6 @@ class IntegrationTestCase(AsyncioTestCase):
|
||||||
def broadcast(self, tx):
|
def broadcast(self, tx):
|
||||||
return self.ledger.broadcast(tx)
|
return self.ledger.broadcast(tx)
|
||||||
|
|
||||||
async def broadcast_and_confirm(self, tx, ledger=None):
|
|
||||||
ledger = ledger or self.ledger
|
|
||||||
notifications = asyncio.create_task(ledger.wait(tx))
|
|
||||||
await ledger.broadcast(tx)
|
|
||||||
await notifications
|
|
||||||
await self.generate_and_wait(1, [tx.id], ledger)
|
|
||||||
|
|
||||||
async def on_header(self, height):
|
async def on_header(self, height):
|
||||||
if self.ledger.headers.height < height:
|
if self.ledger.headers.height < height:
|
||||||
await self.ledger.on_header.where(
|
await self.ledger.on_header.where(
|
||||||
|
@ -281,29 +262,11 @@ class IntegrationTestCase(AsyncioTestCase):
|
||||||
)
|
)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
async def send_to_address_and_wait(self, address, amount, blocks_to_generate=0, ledger=None):
|
def on_transaction_id(self, txid, ledger=None):
|
||||||
tx_watch = []
|
return (ledger or self.ledger).on_transaction.where(
|
||||||
txid = None
|
lambda e: e.tx.id == txid
|
||||||
done = False
|
|
||||||
watcher = (ledger or self.ledger).on_transaction.where(
|
|
||||||
lambda e: e.tx.id == txid or done or tx_watch.append(e.tx.id)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
txid = await self.blockchain.send_to_address(address, amount)
|
|
||||||
done = txid in tx_watch
|
|
||||||
await watcher
|
|
||||||
|
|
||||||
await self.generate_and_wait(blocks_to_generate, [txid], ledger)
|
|
||||||
return txid
|
|
||||||
|
|
||||||
async def generate_and_wait(self, blocks_to_generate, txids, ledger=None):
|
|
||||||
if blocks_to_generate > 0:
|
|
||||||
watcher = (ledger or self.ledger).on_transaction.where(
|
|
||||||
lambda e: ((e.tx.id in txids and txids.remove(e.tx.id)), len(txids) <= 0)[-1] # multi-statement lambda
|
|
||||||
)
|
|
||||||
await self.generate(blocks_to_generate)
|
|
||||||
await watcher
|
|
||||||
|
|
||||||
def on_address_update(self, address):
|
def on_address_update(self, address):
|
||||||
return self.ledger.on_transaction.where(
|
return self.ledger.on_transaction.where(
|
||||||
lambda e: e.address == address
|
lambda e: e.address == address
|
||||||
|
@ -314,22 +277,6 @@ class IntegrationTestCase(AsyncioTestCase):
|
||||||
lambda e: e.tx.id == tx.id and e.address == address
|
lambda e: e.tx.id == tx.id and e.address == address
|
||||||
)
|
)
|
||||||
|
|
||||||
async def generate(self, blocks):
|
|
||||||
""" Ask lbrycrd to generate some blocks and wait until ledger has them. """
|
|
||||||
prepare = self.ledger.on_header.where(self.blockchain.is_expected_block)
|
|
||||||
self.conductor.spv_node.server.synchronized.clear()
|
|
||||||
await self.blockchain.generate(blocks)
|
|
||||||
height = self.blockchain.block_expected
|
|
||||||
await prepare # no guarantee that it didn't happen already, so start waiting from before calling generate
|
|
||||||
while True:
|
|
||||||
await self.conductor.spv_node.server.synchronized.wait()
|
|
||||||
self.conductor.spv_node.server.synchronized.clear()
|
|
||||||
if self.conductor.spv_node.server.db.db_height < height:
|
|
||||||
continue
|
|
||||||
if self.conductor.spv_node.server._es_height < height:
|
|
||||||
continue
|
|
||||||
break
|
|
||||||
|
|
||||||
|
|
||||||
class FakeExchangeRateManager(ExchangeRateManager):
|
class FakeExchangeRateManager(ExchangeRateManager):
|
||||||
|
|
||||||
|
@ -390,19 +337,20 @@ class CommandTestCase(IntegrationTestCase):
|
||||||
self.skip_libtorrent = True
|
self.skip_libtorrent = True
|
||||||
|
|
||||||
async def asyncSetUp(self):
|
async def asyncSetUp(self):
|
||||||
|
await super().asyncSetUp()
|
||||||
|
|
||||||
logging.getLogger('lbry.blob_exchange').setLevel(self.VERBOSITY)
|
logging.getLogger('lbry.blob_exchange').setLevel(self.VERBOSITY)
|
||||||
logging.getLogger('lbry.daemon').setLevel(self.VERBOSITY)
|
logging.getLogger('lbry.daemon').setLevel(self.VERBOSITY)
|
||||||
logging.getLogger('lbry.stream').setLevel(self.VERBOSITY)
|
logging.getLogger('lbry.stream').setLevel(self.VERBOSITY)
|
||||||
logging.getLogger('lbry.wallet').setLevel(self.VERBOSITY)
|
logging.getLogger('lbry.wallet').setLevel(self.VERBOSITY)
|
||||||
|
|
||||||
await super().asyncSetUp()
|
|
||||||
|
|
||||||
self.daemon = await self.add_daemon(self.wallet_node)
|
self.daemon = await self.add_daemon(self.wallet_node)
|
||||||
|
|
||||||
await self.account.ensure_address_gap()
|
await self.account.ensure_address_gap()
|
||||||
address = (await self.account.receiving.get_addresses(limit=1, only_usable=True))[0]
|
address = (await self.account.receiving.get_addresses(limit=1, only_usable=True))[0]
|
||||||
await self.send_to_address_and_wait(address, 10, 6)
|
sendtxid = await self.blockchain.send_to_address(address, 10)
|
||||||
|
await self.confirm_tx(sendtxid)
|
||||||
|
await self.generate(5)
|
||||||
|
|
||||||
server_tmp_dir = tempfile.mkdtemp()
|
server_tmp_dir = tempfile.mkdtemp()
|
||||||
self.addCleanup(shutil.rmtree, server_tmp_dir)
|
self.addCleanup(shutil.rmtree, server_tmp_dir)
|
||||||
|
@ -499,14 +447,9 @@ class CommandTestCase(IntegrationTestCase):
|
||||||
|
|
||||||
async def confirm_tx(self, txid, ledger=None):
|
async def confirm_tx(self, txid, ledger=None):
|
||||||
""" Wait for tx to be in mempool, then generate a block, wait for tx to be in a block. """
|
""" Wait for tx to be in mempool, then generate a block, wait for tx to be in a block. """
|
||||||
# await (ledger or self.ledger).on_transaction.where(lambda e: e.tx.id == txid)
|
await self.on_transaction_id(txid, ledger)
|
||||||
on_tx = (ledger or self.ledger).on_transaction.where(lambda e: e.tx.id == txid)
|
await self.generate(1)
|
||||||
await asyncio.wait([self.generate(1), on_tx], timeout=5)
|
await self.on_transaction_id(txid, ledger)
|
||||||
|
|
||||||
# # actually, if it's in the mempool or in the block we're fine
|
|
||||||
# await self.generate_and_wait(1, [txid], ledger=ledger)
|
|
||||||
# return txid
|
|
||||||
|
|
||||||
return txid
|
return txid
|
||||||
|
|
||||||
async def on_transaction_dict(self, tx):
|
async def on_transaction_dict(self, tx):
|
||||||
|
@ -521,6 +464,12 @@ class CommandTestCase(IntegrationTestCase):
|
||||||
addresses.add(txo['address'])
|
addresses.add(txo['address'])
|
||||||
return list(addresses)
|
return list(addresses)
|
||||||
|
|
||||||
|
async def generate(self, blocks):
|
||||||
|
""" Ask lbrycrd to generate some blocks and wait until ledger has them. """
|
||||||
|
prepare = self.ledger.on_header.where(self.blockchain.is_expected_block)
|
||||||
|
await self.blockchain.generate(blocks)
|
||||||
|
await prepare # no guarantee that it didn't happen already, so start waiting from before calling generate
|
||||||
|
|
||||||
async def blockchain_claim_name(self, name: str, value: str, amount: str, confirm=True):
|
async def blockchain_claim_name(self, name: str, value: str, amount: str, confirm=True):
|
||||||
txid = await self.blockchain._cli_cmnd('claimname', name, value, amount)
|
txid = await self.blockchain._cli_cmnd('claimname', name, value, amount)
|
||||||
if confirm:
|
if confirm:
|
||||||
|
@ -551,19 +500,6 @@ class CommandTestCase(IntegrationTestCase):
|
||||||
return self.sout(tx)
|
return self.sout(tx)
|
||||||
return tx
|
return tx
|
||||||
|
|
||||||
async def create_nondeterministic_channel(self, name, price, pubkey_bytes, daemon=None, blocking=False):
|
|
||||||
account = (daemon or self.daemon).wallet_manager.default_account
|
|
||||||
claim_address = await account.receiving.get_or_create_usable_address()
|
|
||||||
claim = Claim()
|
|
||||||
claim.channel.public_key_bytes = pubkey_bytes
|
|
||||||
tx = await Transaction.claim_create(
|
|
||||||
name, claim, lbc_to_dewies(price),
|
|
||||||
claim_address, [self.account], self.account
|
|
||||||
)
|
|
||||||
await tx.sign([self.account])
|
|
||||||
await (daemon or self.daemon).broadcast_or_release(tx, blocking)
|
|
||||||
return self.sout(tx)
|
|
||||||
|
|
||||||
def create_upload_file(self, data, prefix=None, suffix=None):
|
def create_upload_file(self, data, prefix=None, suffix=None):
|
||||||
file_path = tempfile.mktemp(prefix=prefix or "tmp", suffix=suffix or "", dir=self.daemon.conf.upload_dir)
|
file_path = tempfile.mktemp(prefix=prefix or "tmp", suffix=suffix or "", dir=self.daemon.conf.upload_dir)
|
||||||
with open(file_path, 'w+b') as file:
|
with open(file_path, 'w+b') as file:
|
||||||
|
|
|
@ -10,13 +10,47 @@ from typing import Optional
|
||||||
import libtorrent
|
import libtorrent
|
||||||
|
|
||||||
|
|
||||||
|
NOTIFICATION_MASKS = [
|
||||||
|
"error",
|
||||||
|
"peer",
|
||||||
|
"port_mapping",
|
||||||
|
"storage",
|
||||||
|
"tracker",
|
||||||
|
"debug",
|
||||||
|
"status",
|
||||||
|
"progress",
|
||||||
|
"ip_block",
|
||||||
|
"dht",
|
||||||
|
"stats",
|
||||||
|
"session_log",
|
||||||
|
"torrent_log",
|
||||||
|
"peer_log",
|
||||||
|
"incoming_request",
|
||||||
|
"dht_log",
|
||||||
|
"dht_operation",
|
||||||
|
"port_mapping_log",
|
||||||
|
"picker_log",
|
||||||
|
"file_progress",
|
||||||
|
"piece_progress",
|
||||||
|
"upload",
|
||||||
|
"block_progress"
|
||||||
|
]
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
DEFAULT_FLAGS = ( # fixme: somehow the logic here is inverted?
|
DEFAULT_FLAGS = ( # fixme: somehow the logic here is inverted?
|
||||||
libtorrent.add_torrent_params_flags_t.flag_auto_managed
|
libtorrent.add_torrent_params_flags_t.flag_auto_managed
|
||||||
| libtorrent.add_torrent_params_flags_t.flag_update_subscribe
|
| libtorrent.add_torrent_params_flags_t.flag_update_subscribe
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def get_notification_type(notification) -> str:
|
||||||
|
for i, notification_type in enumerate(NOTIFICATION_MASKS):
|
||||||
|
if (1 << i) & notification:
|
||||||
|
return notification_type
|
||||||
|
raise ValueError("unrecognized notification type")
|
||||||
|
|
||||||
|
|
||||||
class TorrentHandle:
|
class TorrentHandle:
|
||||||
def __init__(self, loop, executor, handle):
|
def __init__(self, loop, executor, handle):
|
||||||
self._loop = loop
|
self._loop = loop
|
||||||
|
@ -87,7 +121,7 @@ class TorrentHandle:
|
||||||
self._show_status()
|
self._show_status()
|
||||||
if self.finished.is_set():
|
if self.finished.is_set():
|
||||||
break
|
break
|
||||||
await asyncio.sleep(0.1)
|
await asyncio.sleep(0.1, loop=self._loop)
|
||||||
|
|
||||||
async def pause(self):
|
async def pause(self):
|
||||||
await self._loop.run_in_executor(
|
await self._loop.run_in_executor(
|
||||||
|
@ -122,8 +156,10 @@ class TorrentSession:
|
||||||
async def bind(self, interface: str = '0.0.0.0', port: int = 10889):
|
async def bind(self, interface: str = '0.0.0.0', port: int = 10889):
|
||||||
settings = {
|
settings = {
|
||||||
'listen_interfaces': f"{interface}:{port}",
|
'listen_interfaces': f"{interface}:{port}",
|
||||||
'enable_natpmp': False,
|
'enable_outgoing_utp': True,
|
||||||
'enable_upnp': False
|
'enable_incoming_utp': True,
|
||||||
|
'enable_outgoing_tcp': False,
|
||||||
|
'enable_incoming_tcp': False
|
||||||
}
|
}
|
||||||
self._session = await self._loop.run_in_executor(
|
self._session = await self._loop.run_in_executor(
|
||||||
self._executor, libtorrent.session, settings # pylint: disable=c-extension-no-member
|
self._executor, libtorrent.session, settings # pylint: disable=c-extension-no-member
|
||||||
|
@ -150,7 +186,7 @@ class TorrentSession:
|
||||||
await self._loop.run_in_executor(
|
await self._loop.run_in_executor(
|
||||||
self._executor, self._pop_alerts
|
self._executor, self._pop_alerts
|
||||||
)
|
)
|
||||||
await asyncio.sleep(1)
|
await asyncio.sleep(1, loop=self._loop)
|
||||||
|
|
||||||
async def pause(self):
|
async def pause(self):
|
||||||
await self._loop.run_in_executor(
|
await self._loop.run_in_executor(
|
||||||
|
|
|
@ -36,7 +36,7 @@ class Torrent:
|
||||||
def __init__(self, loop, handle):
|
def __init__(self, loop, handle):
|
||||||
self._loop = loop
|
self._loop = loop
|
||||||
self._handle = handle
|
self._handle = handle
|
||||||
self.finished = asyncio.Event()
|
self.finished = asyncio.Event(loop=loop)
|
||||||
|
|
||||||
def _threaded_update_status(self):
|
def _threaded_update_status(self):
|
||||||
status = self._handle.status()
|
status = self._handle.status()
|
||||||
|
@ -58,7 +58,7 @@ class Torrent:
|
||||||
log.info("finished downloading torrent!")
|
log.info("finished downloading torrent!")
|
||||||
await self.pause()
|
await self.pause()
|
||||||
break
|
break
|
||||||
await asyncio.sleep(1)
|
await asyncio.sleep(1, loop=self._loop)
|
||||||
|
|
||||||
async def pause(self):
|
async def pause(self):
|
||||||
log.info("pause torrent")
|
log.info("pause torrent")
|
||||||
|
|
|
@ -74,7 +74,7 @@ class TorrentSource(ManagedDownloadSource):
|
||||||
def bt_infohash(self):
|
def bt_infohash(self):
|
||||||
return self.identifier
|
return self.identifier
|
||||||
|
|
||||||
async def stop_tasks(self):
|
def stop_tasks(self):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
@ -118,8 +118,8 @@ class TorrentManager(SourceManager):
|
||||||
async def start(self):
|
async def start(self):
|
||||||
await super().start()
|
await super().start()
|
||||||
|
|
||||||
async def stop(self):
|
def stop(self):
|
||||||
await super().stop()
|
super().stop()
|
||||||
log.info("finished stopping the torrent manager")
|
log.info("finished stopping the torrent manager")
|
||||||
|
|
||||||
async def delete(self, source: ManagedDownloadSource, delete_file: Optional[bool] = False):
|
async def delete(self, source: ManagedDownloadSource, delete_file: Optional[bool] = False):
|
||||||
|
|
|
@ -1,285 +0,0 @@
|
||||||
import random
|
|
||||||
import socket
|
|
||||||
import string
|
|
||||||
import struct
|
|
||||||
import asyncio
|
|
||||||
import logging
|
|
||||||
import time
|
|
||||||
import ipaddress
|
|
||||||
from collections import namedtuple
|
|
||||||
from functools import reduce
|
|
||||||
from typing import Optional
|
|
||||||
|
|
||||||
from lbry.dht.node import get_kademlia_peers_from_hosts
|
|
||||||
from lbry.utils import resolve_host, async_timed_cache, cache_concurrent
|
|
||||||
from lbry.wallet.stream import StreamController
|
|
||||||
from lbry import version
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
CONNECTION_EXPIRES_AFTER_SECONDS = 50
|
|
||||||
PREFIX = 'LB' # todo: PR BEP20 to add ourselves
|
|
||||||
DEFAULT_TIMEOUT_SECONDS = 10.0
|
|
||||||
DEFAULT_CONCURRENCY_LIMIT = 100
|
|
||||||
# see: http://bittorrent.org/beps/bep_0015.html and http://xbtt.sourceforge.net/udp_tracker_protocol.html
|
|
||||||
ConnectRequest = namedtuple("ConnectRequest", ["connection_id", "action", "transaction_id"])
|
|
||||||
ConnectResponse = namedtuple("ConnectResponse", ["action", "transaction_id", "connection_id"])
|
|
||||||
AnnounceRequest = namedtuple("AnnounceRequest",
|
|
||||||
["connection_id", "action", "transaction_id", "info_hash", "peer_id", "downloaded", "left",
|
|
||||||
"uploaded", "event", "ip_addr", "key", "num_want", "port"])
|
|
||||||
AnnounceResponse = namedtuple("AnnounceResponse",
|
|
||||||
["action", "transaction_id", "interval", "leechers", "seeders", "peers"])
|
|
||||||
CompactIPv4Peer = namedtuple("CompactPeer", ["address", "port"])
|
|
||||||
ScrapeRequest = namedtuple("ScrapeRequest", ["connection_id", "action", "transaction_id", "infohashes"])
|
|
||||||
ScrapeResponse = namedtuple("ScrapeResponse", ["action", "transaction_id", "items"])
|
|
||||||
ScrapeResponseItem = namedtuple("ScrapeResponseItem", ["seeders", "completed", "leechers"])
|
|
||||||
ErrorResponse = namedtuple("ErrorResponse", ["action", "transaction_id", "message"])
|
|
||||||
structs = {
|
|
||||||
ConnectRequest: struct.Struct(">QII"),
|
|
||||||
ConnectResponse: struct.Struct(">IIQ"),
|
|
||||||
AnnounceRequest: struct.Struct(">QII20s20sQQQIIIiH"),
|
|
||||||
AnnounceResponse: struct.Struct(">IIIII"),
|
|
||||||
CompactIPv4Peer: struct.Struct(">IH"),
|
|
||||||
ScrapeRequest: struct.Struct(">QII"),
|
|
||||||
ScrapeResponse: struct.Struct(">II"),
|
|
||||||
ScrapeResponseItem: struct.Struct(">III"),
|
|
||||||
ErrorResponse: struct.Struct(">II")
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def decode(cls, data, offset=0):
|
|
||||||
decoder = structs[cls]
|
|
||||||
if cls is AnnounceResponse:
|
|
||||||
return AnnounceResponse(*decoder.unpack_from(data, offset),
|
|
||||||
peers=[decode(CompactIPv4Peer, data, index) for index in range(20, len(data), 6)])
|
|
||||||
elif cls is ScrapeResponse:
|
|
||||||
return ScrapeResponse(*decoder.unpack_from(data, offset),
|
|
||||||
items=[decode(ScrapeResponseItem, data, index) for index in range(8, len(data), 12)])
|
|
||||||
elif cls is ErrorResponse:
|
|
||||||
return ErrorResponse(*decoder.unpack_from(data, offset), data[decoder.size:])
|
|
||||||
return cls(*decoder.unpack_from(data, offset))
|
|
||||||
|
|
||||||
|
|
||||||
def encode(obj):
|
|
||||||
if isinstance(obj, ScrapeRequest):
|
|
||||||
return structs[ScrapeRequest].pack(*obj[:-1]) + b''.join(obj.infohashes)
|
|
||||||
elif isinstance(obj, ErrorResponse):
|
|
||||||
return structs[ErrorResponse].pack(*obj[:-1]) + obj.message
|
|
||||||
elif isinstance(obj, AnnounceResponse):
|
|
||||||
return structs[AnnounceResponse].pack(*obj[:-1]) + b''.join([encode(peer) for peer in obj.peers])
|
|
||||||
return structs[type(obj)].pack(*obj)
|
|
||||||
|
|
||||||
|
|
||||||
def make_peer_id(random_part: Optional[str] = None) -> bytes:
|
|
||||||
# see https://wiki.theory.org/BitTorrentSpecification#peer_id and https://www.bittorrent.org/beps/bep_0020.html
|
|
||||||
# not to confuse with node id; peer id identifies uniquely the software, version and instance
|
|
||||||
random_part = random_part or ''.join(random.choice(string.ascii_letters) for _ in range(20))
|
|
||||||
return f"{PREFIX}-{'-'.join(map(str, version))}-{random_part}"[:20].encode()
|
|
||||||
|
|
||||||
|
|
||||||
class UDPTrackerClientProtocol(asyncio.DatagramProtocol):
|
|
||||||
def __init__(self, timeout: float = DEFAULT_TIMEOUT_SECONDS):
|
|
||||||
self.transport = None
|
|
||||||
self.data_queue = {}
|
|
||||||
self.timeout = timeout
|
|
||||||
self.semaphore = asyncio.Semaphore(DEFAULT_CONCURRENCY_LIMIT)
|
|
||||||
|
|
||||||
def connection_made(self, transport: asyncio.DatagramTransport) -> None:
|
|
||||||
self.transport = transport
|
|
||||||
|
|
||||||
async def request(self, obj, tracker_ip, tracker_port):
|
|
||||||
self.data_queue[obj.transaction_id] = asyncio.get_running_loop().create_future()
|
|
||||||
try:
|
|
||||||
async with self.semaphore:
|
|
||||||
self.transport.sendto(encode(obj), (tracker_ip, tracker_port))
|
|
||||||
return await asyncio.wait_for(self.data_queue[obj.transaction_id], self.timeout)
|
|
||||||
finally:
|
|
||||||
self.data_queue.pop(obj.transaction_id, None)
|
|
||||||
|
|
||||||
async def connect(self, tracker_ip, tracker_port):
|
|
||||||
transaction_id = random.getrandbits(32)
|
|
||||||
return decode(ConnectResponse,
|
|
||||||
await self.request(ConnectRequest(0x41727101980, 0, transaction_id), tracker_ip, tracker_port))
|
|
||||||
|
|
||||||
@cache_concurrent
|
|
||||||
@async_timed_cache(CONNECTION_EXPIRES_AFTER_SECONDS)
|
|
||||||
async def ensure_connection_id(self, peer_id, tracker_ip, tracker_port):
|
|
||||||
# peer_id is just to ensure cache coherency
|
|
||||||
return (await self.connect(tracker_ip, tracker_port)).connection_id
|
|
||||||
|
|
||||||
async def announce(self, info_hash, peer_id, port, tracker_ip, tracker_port, stopped=False):
|
|
||||||
connection_id = await self.ensure_connection_id(peer_id, tracker_ip, tracker_port)
|
|
||||||
# this should make the key deterministic but unique per info hash + peer id
|
|
||||||
key = int.from_bytes(info_hash[:4], "big") ^ int.from_bytes(peer_id[:4], "big") ^ port
|
|
||||||
transaction_id = random.getrandbits(32)
|
|
||||||
req = AnnounceRequest(
|
|
||||||
connection_id, 1, transaction_id, info_hash, peer_id, 0, 0, 0, 3 if stopped else 1, 0, key, -1, port)
|
|
||||||
return decode(AnnounceResponse, await self.request(req, tracker_ip, tracker_port))
|
|
||||||
|
|
||||||
async def scrape(self, infohashes, tracker_ip, tracker_port, connection_id=None):
|
|
||||||
connection_id = await self.ensure_connection_id(None, tracker_ip, tracker_port)
|
|
||||||
transaction_id = random.getrandbits(32)
|
|
||||||
reply = await self.request(
|
|
||||||
ScrapeRequest(connection_id, 2, transaction_id, infohashes), tracker_ip, tracker_port)
|
|
||||||
return decode(ScrapeResponse, reply), connection_id
|
|
||||||
|
|
||||||
def datagram_received(self, data: bytes, addr: (str, int)) -> None:
|
|
||||||
if len(data) < 8:
|
|
||||||
return
|
|
||||||
transaction_id = int.from_bytes(data[4:8], byteorder="big", signed=False)
|
|
||||||
if transaction_id in self.data_queue:
|
|
||||||
if not self.data_queue[transaction_id].done():
|
|
||||||
if data[3] == 3:
|
|
||||||
return self.data_queue[transaction_id].set_exception(Exception(decode(ErrorResponse, data).message))
|
|
||||||
return self.data_queue[transaction_id].set_result(data)
|
|
||||||
log.debug("unexpected packet (can be a response for a previously timed out request): %s", data.hex())
|
|
||||||
|
|
||||||
def connection_lost(self, exc: Exception = None) -> None:
|
|
||||||
self.transport = None
|
|
||||||
|
|
||||||
|
|
||||||
class TrackerClient:
|
|
||||||
event_controller = StreamController()
|
|
||||||
|
|
||||||
def __init__(self, node_id, announce_port, get_servers, timeout=10.0):
|
|
||||||
self.client = UDPTrackerClientProtocol(timeout=timeout)
|
|
||||||
self.transport = None
|
|
||||||
self.peer_id = make_peer_id(node_id.hex() if node_id else None)
|
|
||||||
self.announce_port = announce_port
|
|
||||||
self._get_servers = get_servers
|
|
||||||
self.results = {} # we can't probe the server before the interval, so we keep the result here until it expires
|
|
||||||
self.tasks = {}
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
self.transport, _ = await asyncio.get_running_loop().create_datagram_endpoint(
|
|
||||||
lambda: self.client, local_addr=("0.0.0.0", 0))
|
|
||||||
self.event_controller.stream.listen(
|
|
||||||
lambda request: self.on_hash(request[1], request[2]) if request[0] == 'search' else None)
|
|
||||||
|
|
||||||
def stop(self):
|
|
||||||
while self.tasks:
|
|
||||||
self.tasks.popitem()[1].cancel()
|
|
||||||
if self.transport is not None:
|
|
||||||
self.transport.close()
|
|
||||||
self.client = None
|
|
||||||
self.transport = None
|
|
||||||
self.event_controller.close()
|
|
||||||
|
|
||||||
def on_hash(self, info_hash, on_announcement=None):
|
|
||||||
if info_hash not in self.tasks:
|
|
||||||
task = asyncio.create_task(self.get_peer_list(info_hash, on_announcement=on_announcement))
|
|
||||||
task.add_done_callback(lambda *_: self.tasks.pop(info_hash, None))
|
|
||||||
self.tasks[info_hash] = task
|
|
||||||
|
|
||||||
async def announce_many(self, *info_hashes, stopped=False):
|
|
||||||
await asyncio.gather(
|
|
||||||
*[self._announce_many(server, info_hashes, stopped=stopped) for server in self._get_servers()],
|
|
||||||
return_exceptions=True)
|
|
||||||
|
|
||||||
async def _announce_many(self, server, info_hashes, stopped=False):
|
|
||||||
tracker_ip = await resolve_host(*server, 'udp')
|
|
||||||
still_good_info_hashes = {
|
|
||||||
info_hash for (info_hash, (next_announcement, _)) in self.results.get(tracker_ip, {}).items()
|
|
||||||
if time.time() < next_announcement
|
|
||||||
}
|
|
||||||
results = await asyncio.gather(
|
|
||||||
*[self._probe_server(info_hash, tracker_ip, server[1], stopped=stopped)
|
|
||||||
for info_hash in info_hashes if info_hash not in still_good_info_hashes],
|
|
||||||
return_exceptions=True)
|
|
||||||
if results:
|
|
||||||
errors = sum([1 for result in results if result is None or isinstance(result, Exception)])
|
|
||||||
log.info("Tracker: finished announcing %d files to %s:%d, %d errors", len(results), *server, errors)
|
|
||||||
|
|
||||||
async def get_peer_list(self, info_hash, stopped=False, on_announcement=None, no_port=False):
|
|
||||||
found = []
|
|
||||||
probes = [self._probe_server(info_hash, *server, stopped, no_port) for server in self._get_servers()]
|
|
||||||
for done in asyncio.as_completed(probes):
|
|
||||||
result = await done
|
|
||||||
if result is not None:
|
|
||||||
await asyncio.gather(*filter(asyncio.iscoroutine, [on_announcement(result)] if on_announcement else []))
|
|
||||||
found.append(result)
|
|
||||||
return found
|
|
||||||
|
|
||||||
async def get_kademlia_peer_list(self, info_hash):
|
|
||||||
responses = await self.get_peer_list(info_hash, no_port=True)
|
|
||||||
return await announcement_to_kademlia_peers(*responses)
|
|
||||||
|
|
||||||
async def _probe_server(self, info_hash, tracker_host, tracker_port, stopped=False, no_port=False):
|
|
||||||
result = None
|
|
||||||
try:
|
|
||||||
tracker_host = await resolve_host(tracker_host, tracker_port, 'udp')
|
|
||||||
except socket.error:
|
|
||||||
log.warning("DNS failure while resolving tracker host: %s, skipping.", tracker_host)
|
|
||||||
return
|
|
||||||
self.results.setdefault(tracker_host, {})
|
|
||||||
if info_hash in self.results[tracker_host]:
|
|
||||||
next_announcement, result = self.results[tracker_host][info_hash]
|
|
||||||
if time.time() < next_announcement:
|
|
||||||
return result
|
|
||||||
try:
|
|
||||||
result = await self.client.announce(
|
|
||||||
info_hash, self.peer_id, 0 if no_port else self.announce_port, tracker_host, tracker_port, stopped)
|
|
||||||
self.results[tracker_host][info_hash] = (time.time() + result.interval, result)
|
|
||||||
except asyncio.TimeoutError: # todo: this is UDP, timeout is common, we need a better metric for failures
|
|
||||||
self.results[tracker_host][info_hash] = (time.time() + 60.0, result)
|
|
||||||
log.debug("Tracker timed out: %s:%d", tracker_host, tracker_port)
|
|
||||||
return None
|
|
||||||
log.debug("Announced: %s found %d peers for %s", tracker_host, len(result.peers), info_hash.hex()[:8])
|
|
||||||
return result
|
|
||||||
|
|
||||||
|
|
||||||
def enqueue_tracker_search(info_hash: bytes, peer_q: asyncio.Queue):
|
|
||||||
async def on_announcement(announcement: AnnounceResponse):
|
|
||||||
peers = await announcement_to_kademlia_peers(announcement)
|
|
||||||
log.info("Found %d peers from tracker for %s", len(peers), info_hash.hex()[:8])
|
|
||||||
peer_q.put_nowait(peers)
|
|
||||||
TrackerClient.event_controller.add(('search', info_hash, on_announcement))
|
|
||||||
|
|
||||||
|
|
||||||
def announcement_to_kademlia_peers(*announcements: AnnounceResponse):
|
|
||||||
peers = [
|
|
||||||
(str(ipaddress.ip_address(peer.address)), peer.port)
|
|
||||||
for announcement in announcements for peer in announcement.peers if peer.port > 1024 # no privileged or 0
|
|
||||||
]
|
|
||||||
return get_kademlia_peers_from_hosts(peers)
|
|
||||||
|
|
||||||
|
|
||||||
class UDPTrackerServerProtocol(asyncio.DatagramProtocol): # for testing. Not suitable for production
|
|
||||||
def __init__(self):
|
|
||||||
self.transport = None
|
|
||||||
self.known_conns = set()
|
|
||||||
self.peers = {}
|
|
||||||
|
|
||||||
def connection_made(self, transport: asyncio.DatagramTransport) -> None:
|
|
||||||
self.transport = transport
|
|
||||||
|
|
||||||
def add_peer(self, info_hash, ip_address: str, port: int):
|
|
||||||
self.peers.setdefault(info_hash, [])
|
|
||||||
self.peers[info_hash].append(encode_peer(ip_address, port))
|
|
||||||
|
|
||||||
def datagram_received(self, data: bytes, addr: (str, int)) -> None:
|
|
||||||
if len(data) < 16:
|
|
||||||
return
|
|
||||||
action = int.from_bytes(data[8:12], "big", signed=False)
|
|
||||||
if action == 0:
|
|
||||||
req = decode(ConnectRequest, data)
|
|
||||||
connection_id = random.getrandbits(32)
|
|
||||||
self.known_conns.add(connection_id)
|
|
||||||
return self.transport.sendto(encode(ConnectResponse(0, req.transaction_id, connection_id)), addr)
|
|
||||||
elif action == 1:
|
|
||||||
req = decode(AnnounceRequest, data)
|
|
||||||
if req.connection_id not in self.known_conns:
|
|
||||||
resp = encode(ErrorResponse(3, req.transaction_id, b'Connection ID missmatch.\x00'))
|
|
||||||
else:
|
|
||||||
compact_address = encode_peer(addr[0], req.port)
|
|
||||||
if req.event != 3:
|
|
||||||
self.add_peer(req.info_hash, addr[0], req.port)
|
|
||||||
elif compact_address in self.peers.get(req.info_hash, []):
|
|
||||||
self.peers[req.info_hash].remove(compact_address)
|
|
||||||
peers = [decode(CompactIPv4Peer, peer) for peer in self.peers[req.info_hash]]
|
|
||||||
resp = encode(AnnounceResponse(1, req.transaction_id, 1700, 0, len(peers), peers))
|
|
||||||
return self.transport.sendto(resp, addr)
|
|
||||||
|
|
||||||
|
|
||||||
def encode_peer(ip_address: str, port: int):
|
|
||||||
compact_ip = reduce(lambda buff, x: buff + bytearray([int(x)]), ip_address.split('.'), bytearray())
|
|
||||||
return compact_ip + port.to_bytes(2, "big", signed=False)
|
|
|
@ -130,16 +130,21 @@ def get_sd_hash(stream_info):
|
||||||
def json_dumps_pretty(obj, **kwargs):
|
def json_dumps_pretty(obj, **kwargs):
|
||||||
return json.dumps(obj, sort_keys=True, indent=2, separators=(',', ': '), **kwargs)
|
return json.dumps(obj, sort_keys=True, indent=2, separators=(',', ': '), **kwargs)
|
||||||
|
|
||||||
try:
|
|
||||||
# the standard contextlib.aclosing() is available in 3.10+
|
def cancel_task(task: typing.Optional[asyncio.Task]):
|
||||||
from contextlib import aclosing # pylint: disable=unused-import
|
if task and not task.done():
|
||||||
except ImportError:
|
task.cancel()
|
||||||
@contextlib.asynccontextmanager
|
|
||||||
async def aclosing(thing):
|
|
||||||
try:
|
def cancel_tasks(tasks: typing.List[typing.Optional[asyncio.Task]]):
|
||||||
yield thing
|
for task in tasks:
|
||||||
finally:
|
cancel_task(task)
|
||||||
await thing.aclose()
|
|
||||||
|
|
||||||
|
def drain_tasks(tasks: typing.List[typing.Optional[asyncio.Task]]):
|
||||||
|
while tasks:
|
||||||
|
cancel_task(tasks.pop())
|
||||||
|
|
||||||
|
|
||||||
def async_timed_cache(duration: int):
|
def async_timed_cache(duration: int):
|
||||||
def wrapper(func):
|
def wrapper(func):
|
||||||
|
@ -400,7 +405,7 @@ async def fallback_get_external_ip(): # used if spv servers can't be used for i
|
||||||
|
|
||||||
async def _get_external_ip(default_servers) -> typing.Tuple[typing.Optional[str], typing.Optional[str]]:
|
async def _get_external_ip(default_servers) -> typing.Tuple[typing.Optional[str], typing.Optional[str]]:
|
||||||
# used if upnp is disabled or non-functioning
|
# used if upnp is disabled or non-functioning
|
||||||
from lbry.wallet.udp import SPVStatusClientProtocol # pylint: disable=C0415
|
from lbry.wallet.server.udp import SPVStatusClientProtocol # pylint: disable=C0415
|
||||||
|
|
||||||
hostname_to_ip = {}
|
hostname_to_ip = {}
|
||||||
ip_to_hostnames = collections.defaultdict(list)
|
ip_to_hostnames = collections.defaultdict(list)
|
||||||
|
@ -450,8 +455,8 @@ def is_running_from_bundle():
|
||||||
|
|
||||||
|
|
||||||
class LockWithMetrics(asyncio.Lock):
|
class LockWithMetrics(asyncio.Lock):
|
||||||
def __init__(self, acquire_metric, held_time_metric):
|
def __init__(self, acquire_metric, held_time_metric, loop=None):
|
||||||
super().__init__()
|
super().__init__(loop=loop)
|
||||||
self._acquire_metric = acquire_metric
|
self._acquire_metric = acquire_metric
|
||||||
self._lock_held_time_metric = held_time_metric
|
self._lock_held_time_metric = held_time_metric
|
||||||
self._lock_acquired_time = None
|
self._lock_acquired_time = None
|
||||||
|
@ -469,18 +474,3 @@ class LockWithMetrics(asyncio.Lock):
|
||||||
return super().release()
|
return super().release()
|
||||||
finally:
|
finally:
|
||||||
self._lock_held_time_metric.observe(time.perf_counter() - self._lock_acquired_time)
|
self._lock_held_time_metric.observe(time.perf_counter() - self._lock_acquired_time)
|
||||||
|
|
||||||
|
|
||||||
def get_colliding_prefix_bits(first_value: bytes, second_value: bytes):
|
|
||||||
"""
|
|
||||||
Calculates the amount of colliding prefix bits between <first_value> and <second_value>.
|
|
||||||
This is given by the amount of bits that are the same until the first different one (via XOR),
|
|
||||||
starting from the most significant bit to the least significant bit.
|
|
||||||
:param first_value: first value to compare, bigger than size.
|
|
||||||
:param second_value: second value to compare, bigger than size.
|
|
||||||
:return: amount of prefix colliding bits.
|
|
||||||
"""
|
|
||||||
assert len(first_value) == len(second_value), "length should be the same"
|
|
||||||
size = len(first_value) * 8
|
|
||||||
first_value, second_value = int.from_bytes(first_value, "big"), int.from_bytes(second_value, "big")
|
|
||||||
return size - (first_value ^ second_value).bit_length()
|
|
||||||
|
|
|
@ -1,23 +1,17 @@
|
||||||
__lbcd__ = 'lbcd'
|
__node_daemon__ = 'lbrycrdd'
|
||||||
__lbcctl__ = 'lbcctl'
|
__node_cli__ = 'lbrycrd-cli'
|
||||||
__lbcwallet__ = 'lbcwallet'
|
__node_bin__ = ''
|
||||||
__lbcd_url__ = (
|
__node_url__ = (
|
||||||
'https://github.com/lbryio/lbcd/releases/download/' +
|
'https://github.com/lbryio/lbrycrd/releases/download/v0.17.4.6/lbrycrd-linux-1746.zip'
|
||||||
'v0.22.100-rc.0/lbcd_0.22.100-rc.0_TARGET_PLATFORM.tar.gz'
|
|
||||||
)
|
|
||||||
__lbcwallet_url__ = (
|
|
||||||
'https://github.com/lbryio/lbcwallet/releases/download/' +
|
|
||||||
'v0.13.100-alpha.0/lbcwallet_0.13.100-alpha.0_TARGET_PLATFORM.tar.gz'
|
|
||||||
)
|
)
|
||||||
__spvserver__ = 'lbry.wallet.server.coin.LBCRegTest'
|
__spvserver__ = 'lbry.wallet.server.coin.LBCRegTest'
|
||||||
|
|
||||||
from lbry.wallet.wallet import Wallet, WalletStorage, TimestampedPreferences, ENCRYPT_ON_DISK
|
from .wallet import Wallet, WalletStorage, TimestampedPreferences, ENCRYPT_ON_DISK
|
||||||
from lbry.wallet.manager import WalletManager
|
from .manager import WalletManager
|
||||||
from lbry.wallet.network import Network
|
from .network import Network
|
||||||
from lbry.wallet.ledger import Ledger, RegTestLedger, TestNetLedger, BlockHeightEvent
|
from .ledger import Ledger, RegTestLedger, TestNetLedger, BlockHeightEvent
|
||||||
from lbry.wallet.account import Account, AddressManager, SingleKey, HierarchicalDeterministic, \
|
from .account import Account, AddressManager, SingleKey, HierarchicalDeterministic
|
||||||
DeterministicChannelKeyManager
|
from .transaction import Transaction, Output, Input
|
||||||
from lbry.wallet.transaction import Transaction, Output, Input
|
from .script import OutputScript, InputScript
|
||||||
from lbry.wallet.script import OutputScript, InputScript
|
from .database import SQLiteMixin, Database
|
||||||
from lbry.wallet.database import SQLiteMixin, Database
|
from .header import Headers
|
||||||
from lbry.wallet.header import Headers
|
|
||||||
|
|
BIN
lbry/wallet/__init__.pyc
Normal file
BIN
lbry/wallet/__init__.pyc
Normal file
Binary file not shown.
|
@ -9,10 +9,11 @@ from hashlib import sha256
|
||||||
from string import hexdigits
|
from string import hexdigits
|
||||||
from typing import Type, Dict, Tuple, Optional, Any, List
|
from typing import Type, Dict, Tuple, Optional, Any, List
|
||||||
|
|
||||||
|
import ecdsa
|
||||||
from lbry.error import InvalidPasswordError
|
from lbry.error import InvalidPasswordError
|
||||||
from lbry.crypto.crypt import aes_encrypt, aes_decrypt
|
from lbry.crypto.crypt import aes_encrypt, aes_decrypt
|
||||||
|
|
||||||
from .bip32 import PrivateKey, PublicKey, KeyPath, from_extended_key_string
|
from .bip32 import PrivateKey, PubKey, from_extended_key_string
|
||||||
from .mnemonic import Mnemonic
|
from .mnemonic import Mnemonic
|
||||||
from .constants import COIN, TXO_TYPES
|
from .constants import COIN, TXO_TYPES
|
||||||
from .transaction import Transaction, Input, Output
|
from .transaction import Transaction, Input, Output
|
||||||
|
@ -33,49 +34,6 @@ def validate_claim_id(claim_id):
|
||||||
raise Exception("Claim id is not hex encoded")
|
raise Exception("Claim id is not hex encoded")
|
||||||
|
|
||||||
|
|
||||||
class DeterministicChannelKeyManager:
|
|
||||||
|
|
||||||
def __init__(self, account: 'Account'):
|
|
||||||
self.account = account
|
|
||||||
self.last_known = 0
|
|
||||||
self.cache = {}
|
|
||||||
self._private_key: Optional[PrivateKey] = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def private_key(self):
|
|
||||||
if self._private_key is None:
|
|
||||||
if self.account.private_key is not None:
|
|
||||||
self._private_key = self.account.private_key.child(KeyPath.CHANNEL)
|
|
||||||
return self._private_key
|
|
||||||
|
|
||||||
def maybe_generate_deterministic_key_for_channel(self, txo):
|
|
||||||
if self.private_key is None:
|
|
||||||
return
|
|
||||||
next_private_key = self.private_key.child(self.last_known)
|
|
||||||
public_key = next_private_key.public_key
|
|
||||||
public_key_bytes = public_key.pubkey_bytes
|
|
||||||
if txo.claim.channel.public_key_bytes == public_key_bytes:
|
|
||||||
self.cache[public_key.address] = next_private_key
|
|
||||||
self.last_known += 1
|
|
||||||
|
|
||||||
async def ensure_cache_primed(self):
|
|
||||||
if self.private_key is not None:
|
|
||||||
await self.generate_next_key()
|
|
||||||
|
|
||||||
async def generate_next_key(self) -> PrivateKey:
|
|
||||||
db = self.account.ledger.db
|
|
||||||
while True:
|
|
||||||
next_private_key = self.private_key.child(self.last_known)
|
|
||||||
public_key = next_private_key.public_key
|
|
||||||
self.cache[public_key.address] = next_private_key
|
|
||||||
if not await db.is_channel_key_used(self.account, public_key):
|
|
||||||
return next_private_key
|
|
||||||
self.last_known += 1
|
|
||||||
|
|
||||||
def get_private_key_from_pubkey_hash(self, pubkey_hash) -> PrivateKey:
|
|
||||||
return self.cache.get(pubkey_hash)
|
|
||||||
|
|
||||||
|
|
||||||
class AddressManager:
|
class AddressManager:
|
||||||
|
|
||||||
name: str
|
name: str
|
||||||
|
@ -121,7 +79,7 @@ class AddressManager:
|
||||||
def get_private_key(self, index: int) -> PrivateKey:
|
def get_private_key(self, index: int) -> PrivateKey:
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
def get_public_key(self, index: int) -> PublicKey:
|
def get_public_key(self, index: int) -> PubKey:
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
async def get_max_gap(self):
|
async def get_max_gap(self):
|
||||||
|
@ -161,8 +119,8 @@ class HierarchicalDeterministic(AddressManager):
|
||||||
@classmethod
|
@classmethod
|
||||||
def from_dict(cls, account: 'Account', d: dict) -> Tuple[AddressManager, AddressManager]:
|
def from_dict(cls, account: 'Account', d: dict) -> Tuple[AddressManager, AddressManager]:
|
||||||
return (
|
return (
|
||||||
cls(account, KeyPath.RECEIVE, **d.get('receiving', {'gap': 20, 'maximum_uses_per_address': 1})),
|
cls(account, 0, **d.get('receiving', {'gap': 20, 'maximum_uses_per_address': 1})),
|
||||||
cls(account, KeyPath.CHANGE, **d.get('change', {'gap': 6, 'maximum_uses_per_address': 1}))
|
cls(account, 1, **d.get('change', {'gap': 6, 'maximum_uses_per_address': 1}))
|
||||||
)
|
)
|
||||||
|
|
||||||
def merge(self, d: dict):
|
def merge(self, d: dict):
|
||||||
|
@ -175,7 +133,7 @@ class HierarchicalDeterministic(AddressManager):
|
||||||
def get_private_key(self, index: int) -> PrivateKey:
|
def get_private_key(self, index: int) -> PrivateKey:
|
||||||
return self.account.private_key.child(self.chain_number).child(index)
|
return self.account.private_key.child(self.chain_number).child(index)
|
||||||
|
|
||||||
def get_public_key(self, index: int) -> PublicKey:
|
def get_public_key(self, index: int) -> PubKey:
|
||||||
return self.account.public_key.child(self.chain_number).child(index)
|
return self.account.public_key.child(self.chain_number).child(index)
|
||||||
|
|
||||||
async def get_max_gap(self) -> int:
|
async def get_max_gap(self) -> int:
|
||||||
|
@ -235,7 +193,7 @@ class SingleKey(AddressManager):
|
||||||
@classmethod
|
@classmethod
|
||||||
def from_dict(cls, account: 'Account', d: dict) \
|
def from_dict(cls, account: 'Account', d: dict) \
|
||||||
-> Tuple[AddressManager, AddressManager]:
|
-> Tuple[AddressManager, AddressManager]:
|
||||||
same_address_manager = cls(account, account.public_key, KeyPath.RECEIVE)
|
same_address_manager = cls(account, account.public_key, 0)
|
||||||
return same_address_manager, same_address_manager
|
return same_address_manager, same_address_manager
|
||||||
|
|
||||||
def to_dict_instance(self):
|
def to_dict_instance(self):
|
||||||
|
@ -244,7 +202,7 @@ class SingleKey(AddressManager):
|
||||||
def get_private_key(self, index: int) -> PrivateKey:
|
def get_private_key(self, index: int) -> PrivateKey:
|
||||||
return self.account.private_key
|
return self.account.private_key
|
||||||
|
|
||||||
def get_public_key(self, index: int) -> PublicKey:
|
def get_public_key(self, index: int) -> PubKey:
|
||||||
return self.account.public_key
|
return self.account.public_key
|
||||||
|
|
||||||
async def get_max_gap(self) -> int:
|
async def get_max_gap(self) -> int:
|
||||||
|
@ -266,6 +224,9 @@ class SingleKey(AddressManager):
|
||||||
|
|
||||||
class Account:
|
class Account:
|
||||||
|
|
||||||
|
mnemonic_class = Mnemonic
|
||||||
|
private_key_class = PrivateKey
|
||||||
|
public_key_class = PubKey
|
||||||
address_generators: Dict[str, Type[AddressManager]] = {
|
address_generators: Dict[str, Type[AddressManager]] = {
|
||||||
SingleKey.name: SingleKey,
|
SingleKey.name: SingleKey,
|
||||||
HierarchicalDeterministic.name: HierarchicalDeterministic,
|
HierarchicalDeterministic.name: HierarchicalDeterministic,
|
||||||
|
@ -273,7 +234,7 @@ class Account:
|
||||||
|
|
||||||
def __init__(self, ledger: 'Ledger', wallet: 'Wallet', name: str,
|
def __init__(self, ledger: 'Ledger', wallet: 'Wallet', name: str,
|
||||||
seed: str, private_key_string: str, encrypted: bool,
|
seed: str, private_key_string: str, encrypted: bool,
|
||||||
private_key: Optional[PrivateKey], public_key: PublicKey,
|
private_key: Optional[PrivateKey], public_key: PubKey,
|
||||||
address_generator: dict, modified_on: float, channel_keys: dict) -> None:
|
address_generator: dict, modified_on: float, channel_keys: dict) -> None:
|
||||||
self.ledger = ledger
|
self.ledger = ledger
|
||||||
self.wallet = wallet
|
self.wallet = wallet
|
||||||
|
@ -284,14 +245,13 @@ class Account:
|
||||||
self.private_key_string = private_key_string
|
self.private_key_string = private_key_string
|
||||||
self.init_vectors: Dict[str, bytes] = {}
|
self.init_vectors: Dict[str, bytes] = {}
|
||||||
self.encrypted = encrypted
|
self.encrypted = encrypted
|
||||||
self.private_key: Optional[PrivateKey] = private_key
|
self.private_key = private_key
|
||||||
self.public_key: PublicKey = public_key
|
self.public_key = public_key
|
||||||
generator_name = address_generator.get('name', HierarchicalDeterministic.name)
|
generator_name = address_generator.get('name', HierarchicalDeterministic.name)
|
||||||
self.address_generator = self.address_generators[generator_name]
|
self.address_generator = self.address_generators[generator_name]
|
||||||
self.receiving, self.change = self.address_generator.from_dict(self, address_generator)
|
self.receiving, self.change = self.address_generator.from_dict(self, address_generator)
|
||||||
self.address_managers = {am.chain_number: am for am in (self.receiving, self.change)}
|
self.address_managers = {am.chain_number: am for am in (self.receiving, self.change)}
|
||||||
self.channel_keys = channel_keys
|
self.channel_keys = channel_keys
|
||||||
self.deterministic_channel_keys = DeterministicChannelKeyManager(self)
|
|
||||||
ledger.add_account(self)
|
ledger.add_account(self)
|
||||||
wallet.add_account(self)
|
wallet.add_account(self)
|
||||||
|
|
||||||
|
@ -306,19 +266,19 @@ class Account:
|
||||||
name: str = None, address_generator: dict = None):
|
name: str = None, address_generator: dict = None):
|
||||||
return cls.from_dict(ledger, wallet, {
|
return cls.from_dict(ledger, wallet, {
|
||||||
'name': name,
|
'name': name,
|
||||||
'seed': Mnemonic().make_seed(),
|
'seed': cls.mnemonic_class().make_seed(),
|
||||||
'address_generator': address_generator or {}
|
'address_generator': address_generator or {}
|
||||||
})
|
})
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def get_private_key_from_seed(cls, ledger: 'Ledger', seed: str, password: str):
|
def get_private_key_from_seed(cls, ledger: 'Ledger', seed: str, password: str):
|
||||||
return PrivateKey.from_seed(
|
return cls.private_key_class.from_seed(
|
||||||
ledger, Mnemonic.mnemonic_to_seed(seed, password or 'lbryum')
|
ledger, cls.mnemonic_class.mnemonic_to_seed(seed, password or 'lbryum')
|
||||||
)
|
)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def keys_from_dict(cls, ledger: 'Ledger', d: dict) \
|
def keys_from_dict(cls, ledger: 'Ledger', d: dict) \
|
||||||
-> Tuple[str, Optional[PrivateKey], PublicKey]:
|
-> Tuple[str, Optional[PrivateKey], PubKey]:
|
||||||
seed = d.get('seed', '')
|
seed = d.get('seed', '')
|
||||||
private_key_string = d.get('private_key', '')
|
private_key_string = d.get('private_key', '')
|
||||||
private_key = None
|
private_key = None
|
||||||
|
@ -489,7 +449,7 @@ class Account:
|
||||||
assert not self.encrypted, "Cannot get private key on encrypted wallet account."
|
assert not self.encrypted, "Cannot get private key on encrypted wallet account."
|
||||||
return self.address_managers[chain].get_private_key(index)
|
return self.address_managers[chain].get_private_key(index)
|
||||||
|
|
||||||
def get_public_key(self, chain: int, index: int) -> PublicKey:
|
def get_public_key(self, chain: int, index: int) -> PubKey:
|
||||||
return self.address_managers[chain].get_public_key(index)
|
return self.address_managers[chain].get_public_key(index)
|
||||||
|
|
||||||
def get_balance(self, confirmations=0, include_claims=False, read_only=False, **constraints):
|
def get_balance(self, confirmations=0, include_claims=False, read_only=False, **constraints):
|
||||||
|
@ -560,30 +520,33 @@ class Account:
|
||||||
|
|
||||||
return tx
|
return tx
|
||||||
|
|
||||||
async def generate_channel_private_key(self):
|
def add_channel_private_key(self, private_key):
|
||||||
return await self.deterministic_channel_keys.generate_next_key()
|
public_key_bytes = private_key.get_verifying_key().to_der()
|
||||||
|
channel_pubkey_hash = self.ledger.public_key_to_address(public_key_bytes)
|
||||||
|
self.channel_keys[channel_pubkey_hash] = private_key.to_pem().decode()
|
||||||
|
|
||||||
def add_channel_private_key(self, private_key: PrivateKey):
|
async def get_channel_private_key(self, public_key_bytes):
|
||||||
self.channel_keys[private_key.address] = private_key.to_pem().decode()
|
|
||||||
|
|
||||||
async def get_channel_private_key(self, public_key_bytes) -> PrivateKey:
|
|
||||||
channel_pubkey_hash = self.ledger.public_key_to_address(public_key_bytes)
|
channel_pubkey_hash = self.ledger.public_key_to_address(public_key_bytes)
|
||||||
private_key_pem = self.channel_keys.get(channel_pubkey_hash)
|
private_key_pem = self.channel_keys.get(channel_pubkey_hash)
|
||||||
if private_key_pem:
|
if private_key_pem:
|
||||||
return PrivateKey.from_pem(self.ledger, private_key_pem)
|
return await asyncio.get_event_loop().run_in_executor(
|
||||||
return self.deterministic_channel_keys.get_private_key_from_pubkey_hash(channel_pubkey_hash)
|
None, ecdsa.SigningKey.from_pem, private_key_pem, sha256
|
||||||
|
)
|
||||||
|
|
||||||
async def maybe_migrate_certificates(self):
|
async def maybe_migrate_certificates(self):
|
||||||
|
def to_der(private_key_pem):
|
||||||
|
return ecdsa.SigningKey.from_pem(private_key_pem, hashfunc=sha256).get_verifying_key().to_der()
|
||||||
|
|
||||||
if not self.channel_keys:
|
if not self.channel_keys:
|
||||||
return
|
return
|
||||||
channel_keys = {}
|
channel_keys = {}
|
||||||
for private_key_pem in self.channel_keys.values():
|
for private_key_pem in self.channel_keys.values():
|
||||||
if not isinstance(private_key_pem, str):
|
if not isinstance(private_key_pem, str):
|
||||||
continue
|
continue
|
||||||
if not private_key_pem.startswith("-----BEGIN"):
|
if "-----BEGIN EC PRIVATE KEY-----" not in private_key_pem:
|
||||||
continue
|
continue
|
||||||
private_key = PrivateKey.from_pem(self.ledger, private_key_pem)
|
public_key_der = await asyncio.get_event_loop().run_in_executor(None, to_der, private_key_pem)
|
||||||
channel_keys[private_key.address] = private_key_pem
|
channel_keys[self.ledger.public_key_to_address(public_key_der)] = private_key_pem
|
||||||
if self.channel_keys != channel_keys:
|
if self.channel_keys != channel_keys:
|
||||||
self.channel_keys = channel_keys
|
self.channel_keys = channel_keys
|
||||||
self.wallet.save()
|
self.wallet.save()
|
||||||
|
|
|
@ -1,21 +1,10 @@
|
||||||
from asn1crypto.keys import PrivateKeyInfo, ECPrivateKey
|
from coincurve import PublicKey, PrivateKey as _PrivateKey
|
||||||
from coincurve import PublicKey as cPublicKey, PrivateKey as cPrivateKey
|
|
||||||
from coincurve.utils import (
|
|
||||||
pem_to_der, lib as libsecp256k1, ffi as libsecp256k1_ffi
|
|
||||||
)
|
|
||||||
from coincurve.ecdsa import CDATA_SIG_LENGTH
|
|
||||||
|
|
||||||
from lbry.crypto.hash import hmac_sha512, hash160, double_sha256
|
from lbry.crypto.hash import hmac_sha512, hash160, double_sha256
|
||||||
from lbry.crypto.base58 import Base58
|
from lbry.crypto.base58 import Base58
|
||||||
from .util import cachedproperty
|
from .util import cachedproperty
|
||||||
|
|
||||||
|
|
||||||
class KeyPath:
|
|
||||||
RECEIVE = 0
|
|
||||||
CHANGE = 1
|
|
||||||
CHANNEL = 2
|
|
||||||
|
|
||||||
|
|
||||||
class DerivationError(Exception):
|
class DerivationError(Exception):
|
||||||
""" Raised when an invalid derivation occurs. """
|
""" Raised when an invalid derivation occurs. """
|
||||||
|
|
||||||
|
@ -82,30 +71,26 @@ class _KeyBase:
|
||||||
return Base58.encode_check(self.extended_key())
|
return Base58.encode_check(self.extended_key())
|
||||||
|
|
||||||
|
|
||||||
class PublicKey(_KeyBase):
|
class PubKey(_KeyBase):
|
||||||
""" A BIP32 public key. """
|
""" A BIP32 public key. """
|
||||||
|
|
||||||
def __init__(self, ledger, pubkey, chain_code, n, depth, parent=None):
|
def __init__(self, ledger, pubkey, chain_code, n, depth, parent=None):
|
||||||
super().__init__(ledger, chain_code, n, depth, parent)
|
super().__init__(ledger, chain_code, n, depth, parent)
|
||||||
if isinstance(pubkey, cPublicKey):
|
if isinstance(pubkey, PublicKey):
|
||||||
self.verifying_key = pubkey
|
self.verifying_key = pubkey
|
||||||
else:
|
else:
|
||||||
self.verifying_key = self._verifying_key_from_pubkey(pubkey)
|
self.verifying_key = self._verifying_key_from_pubkey(pubkey)
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def from_compressed(cls, public_key_bytes, ledger=None) -> 'PublicKey':
|
|
||||||
return cls(ledger, public_key_bytes, bytes((0,)*32), 0, 0)
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def _verifying_key_from_pubkey(cls, pubkey):
|
def _verifying_key_from_pubkey(cls, pubkey):
|
||||||
""" Converts a 33-byte compressed pubkey into an coincurve.PublicKey object. """
|
""" Converts a 33-byte compressed pubkey into an PublicKey object. """
|
||||||
if not isinstance(pubkey, (bytes, bytearray)):
|
if not isinstance(pubkey, (bytes, bytearray)):
|
||||||
raise TypeError('pubkey must be raw bytes')
|
raise TypeError('pubkey must be raw bytes')
|
||||||
if len(pubkey) != 33:
|
if len(pubkey) != 33:
|
||||||
raise ValueError('pubkey must be 33 bytes')
|
raise ValueError('pubkey must be 33 bytes')
|
||||||
if pubkey[0] not in (2, 3):
|
if pubkey[0] not in (2, 3):
|
||||||
raise ValueError('invalid pubkey prefix byte')
|
raise ValueError('invalid pubkey prefix byte')
|
||||||
return cPublicKey(pubkey)
|
return PublicKey(pubkey)
|
||||||
|
|
||||||
@cachedproperty
|
@cachedproperty
|
||||||
def pubkey_bytes(self):
|
def pubkey_bytes(self):
|
||||||
|
@ -120,7 +105,7 @@ class PublicKey(_KeyBase):
|
||||||
def ec_point(self):
|
def ec_point(self):
|
||||||
return self.verifying_key.point()
|
return self.verifying_key.point()
|
||||||
|
|
||||||
def child(self, n: int) -> 'PublicKey':
|
def child(self, n: int):
|
||||||
""" Return the derived child extended pubkey at index N. """
|
""" Return the derived child extended pubkey at index N. """
|
||||||
if not 0 <= n < (1 << 31):
|
if not 0 <= n < (1 << 31):
|
||||||
raise ValueError('invalid BIP32 public key child number')
|
raise ValueError('invalid BIP32 public key child number')
|
||||||
|
@ -128,7 +113,7 @@ class PublicKey(_KeyBase):
|
||||||
msg = self.pubkey_bytes + n.to_bytes(4, 'big')
|
msg = self.pubkey_bytes + n.to_bytes(4, 'big')
|
||||||
L_b, R_b = self._hmac_sha512(msg) # pylint: disable=invalid-name
|
L_b, R_b = self._hmac_sha512(msg) # pylint: disable=invalid-name
|
||||||
derived_key = self.verifying_key.add(L_b)
|
derived_key = self.verifying_key.add(L_b)
|
||||||
return PublicKey(self.ledger, derived_key, R_b, n, self.depth + 1, self)
|
return PubKey(self.ledger, derived_key, R_b, n, self.depth + 1, self)
|
||||||
|
|
||||||
def identifier(self):
|
def identifier(self):
|
||||||
""" Return the key's identifier as 20 bytes. """
|
""" Return the key's identifier as 20 bytes. """
|
||||||
|
@ -141,36 +126,6 @@ class PublicKey(_KeyBase):
|
||||||
self.pubkey_bytes
|
self.pubkey_bytes
|
||||||
)
|
)
|
||||||
|
|
||||||
def verify(self, signature, digest) -> bool:
|
|
||||||
""" Verify that a signature is valid for a 32 byte digest. """
|
|
||||||
|
|
||||||
if len(signature) != 64:
|
|
||||||
raise ValueError('Signature must be 64 bytes long.')
|
|
||||||
|
|
||||||
if len(digest) != 32:
|
|
||||||
raise ValueError('Digest must be 32 bytes long.')
|
|
||||||
|
|
||||||
key = self.verifying_key
|
|
||||||
|
|
||||||
raw_signature = libsecp256k1_ffi.new('secp256k1_ecdsa_signature *')
|
|
||||||
|
|
||||||
parsed = libsecp256k1.secp256k1_ecdsa_signature_parse_compact(
|
|
||||||
key.context.ctx, raw_signature, signature
|
|
||||||
)
|
|
||||||
assert parsed == 1
|
|
||||||
|
|
||||||
normalized_signature = libsecp256k1_ffi.new('secp256k1_ecdsa_signature *')
|
|
||||||
|
|
||||||
libsecp256k1.secp256k1_ecdsa_signature_normalize(
|
|
||||||
key.context.ctx, normalized_signature, raw_signature
|
|
||||||
)
|
|
||||||
|
|
||||||
verified = libsecp256k1.secp256k1_ecdsa_verify(
|
|
||||||
key.context.ctx, normalized_signature, digest, key.public_key
|
|
||||||
)
|
|
||||||
|
|
||||||
return bool(verified)
|
|
||||||
|
|
||||||
|
|
||||||
class PrivateKey(_KeyBase):
|
class PrivateKey(_KeyBase):
|
||||||
"""A BIP32 private key."""
|
"""A BIP32 private key."""
|
||||||
|
@ -179,7 +134,7 @@ class PrivateKey(_KeyBase):
|
||||||
|
|
||||||
def __init__(self, ledger, privkey, chain_code, n, depth, parent=None):
|
def __init__(self, ledger, privkey, chain_code, n, depth, parent=None):
|
||||||
super().__init__(ledger, chain_code, n, depth, parent)
|
super().__init__(ledger, chain_code, n, depth, parent)
|
||||||
if isinstance(privkey, cPrivateKey):
|
if isinstance(privkey, _PrivateKey):
|
||||||
self.signing_key = privkey
|
self.signing_key = privkey
|
||||||
else:
|
else:
|
||||||
self.signing_key = self._signing_key_from_privkey(privkey)
|
self.signing_key = self._signing_key_from_privkey(privkey)
|
||||||
|
@ -187,7 +142,7 @@ class PrivateKey(_KeyBase):
|
||||||
@classmethod
|
@classmethod
|
||||||
def _signing_key_from_privkey(cls, private_key):
|
def _signing_key_from_privkey(cls, private_key):
|
||||||
""" Converts a 32-byte private key into an coincurve.PrivateKey object. """
|
""" Converts a 32-byte private key into an coincurve.PrivateKey object. """
|
||||||
return cPrivateKey.from_int(PrivateKey._private_key_secret_exponent(private_key))
|
return _PrivateKey.from_int(PrivateKey._private_key_secret_exponent(private_key))
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def _private_key_secret_exponent(cls, private_key):
|
def _private_key_secret_exponent(cls, private_key):
|
||||||
|
@ -199,40 +154,24 @@ class PrivateKey(_KeyBase):
|
||||||
return int.from_bytes(private_key, 'big')
|
return int.from_bytes(private_key, 'big')
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def from_seed(cls, ledger, seed) -> 'PrivateKey':
|
def from_seed(cls, ledger, seed):
|
||||||
# This hard-coded message string seems to be coin-independent...
|
# This hard-coded message string seems to be coin-independent...
|
||||||
hmac = hmac_sha512(b'Bitcoin seed', seed)
|
hmac = hmac_sha512(b'Bitcoin seed', seed)
|
||||||
privkey, chain_code = hmac[:32], hmac[32:]
|
privkey, chain_code = hmac[:32], hmac[32:]
|
||||||
return cls(ledger, privkey, chain_code, 0, 0)
|
return cls(ledger, privkey, chain_code, 0, 0)
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def from_pem(cls, ledger, pem) -> 'PrivateKey':
|
|
||||||
der = pem_to_der(pem.encode())
|
|
||||||
try:
|
|
||||||
key_int = ECPrivateKey.load(der).native['private_key']
|
|
||||||
except ValueError:
|
|
||||||
key_int = PrivateKeyInfo.load(der).native['private_key']['private_key']
|
|
||||||
private_key = cPrivateKey.from_int(key_int)
|
|
||||||
return cls(ledger, private_key, bytes((0,)*32), 0, 0)
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def from_bytes(cls, ledger, key_bytes) -> 'PrivateKey':
|
|
||||||
return cls(ledger, cPrivateKey(key_bytes), bytes((0,)*32), 0, 0)
|
|
||||||
|
|
||||||
@cachedproperty
|
@cachedproperty
|
||||||
def private_key_bytes(self):
|
def private_key_bytes(self):
|
||||||
""" Return the serialized private key (no leading zero byte). """
|
""" Return the serialized private key (no leading zero byte). """
|
||||||
return self.signing_key.secret
|
return self.signing_key.secret
|
||||||
|
|
||||||
@cachedproperty
|
@cachedproperty
|
||||||
def public_key(self) -> PublicKey:
|
def public_key(self):
|
||||||
""" Return the corresponding extended public key. """
|
""" Return the corresponding extended public key. """
|
||||||
verifying_key = self.signing_key.public_key
|
verifying_key = self.signing_key.public_key
|
||||||
parent_pubkey = self.parent.public_key if self.parent else None
|
parent_pubkey = self.parent.public_key if self.parent else None
|
||||||
return PublicKey(
|
return PubKey(self.ledger, verifying_key, self.chain_code, self.n, self.depth,
|
||||||
self.ledger, verifying_key, self.chain_code,
|
parent_pubkey)
|
||||||
self.n, self.depth, parent_pubkey
|
|
||||||
)
|
|
||||||
|
|
||||||
def ec_point(self):
|
def ec_point(self):
|
||||||
return self.public_key.ec_point()
|
return self.public_key.ec_point()
|
||||||
|
@ -245,12 +184,11 @@ class PrivateKey(_KeyBase):
|
||||||
""" Return the private key encoded in Wallet Import Format. """
|
""" Return the private key encoded in Wallet Import Format. """
|
||||||
return self.ledger.private_key_to_wif(self.private_key_bytes)
|
return self.ledger.private_key_to_wif(self.private_key_bytes)
|
||||||
|
|
||||||
@property
|
|
||||||
def address(self):
|
def address(self):
|
||||||
""" The public key as a P2PKH address. """
|
""" The public key as a P2PKH address. """
|
||||||
return self.public_key.address
|
return self.public_key.address
|
||||||
|
|
||||||
def child(self, n) -> 'PrivateKey':
|
def child(self, n):
|
||||||
""" Return the derived child extended private key at index N."""
|
""" Return the derived child extended private key at index N."""
|
||||||
if not 0 <= n < (1 << 32):
|
if not 0 <= n < (1 << 32):
|
||||||
raise ValueError('invalid BIP32 private key child number')
|
raise ValueError('invalid BIP32 private key child number')
|
||||||
|
@ -269,28 +207,6 @@ class PrivateKey(_KeyBase):
|
||||||
""" Produce a signature for piece of data by double hashing it and signing the hash. """
|
""" Produce a signature for piece of data by double hashing it and signing the hash. """
|
||||||
return self.signing_key.sign(data, hasher=double_sha256)
|
return self.signing_key.sign(data, hasher=double_sha256)
|
||||||
|
|
||||||
def sign_compact(self, digest):
|
|
||||||
""" Produce a compact signature. """
|
|
||||||
key = self.signing_key
|
|
||||||
|
|
||||||
signature = libsecp256k1_ffi.new('secp256k1_ecdsa_signature *')
|
|
||||||
signed = libsecp256k1.secp256k1_ecdsa_sign(
|
|
||||||
key.context.ctx, signature, digest, key.secret,
|
|
||||||
libsecp256k1_ffi.NULL, libsecp256k1_ffi.NULL
|
|
||||||
)
|
|
||||||
|
|
||||||
if not signed:
|
|
||||||
raise ValueError('The private key was invalid.')
|
|
||||||
|
|
||||||
serialized = libsecp256k1_ffi.new('unsigned char[%d]' % CDATA_SIG_LENGTH)
|
|
||||||
compacted = libsecp256k1.secp256k1_ecdsa_signature_serialize_compact(
|
|
||||||
key.context.ctx, serialized, signature
|
|
||||||
)
|
|
||||||
if compacted != 1:
|
|
||||||
raise ValueError('The signature could not be compacted.')
|
|
||||||
|
|
||||||
return bytes(libsecp256k1_ffi.buffer(serialized, CDATA_SIG_LENGTH))
|
|
||||||
|
|
||||||
def identifier(self):
|
def identifier(self):
|
||||||
"""Return the key's identifier as 20 bytes."""
|
"""Return the key's identifier as 20 bytes."""
|
||||||
return self.public_key.identifier()
|
return self.public_key.identifier()
|
||||||
|
@ -302,12 +218,9 @@ class PrivateKey(_KeyBase):
|
||||||
b'\0' + self.private_key_bytes
|
b'\0' + self.private_key_bytes
|
||||||
)
|
)
|
||||||
|
|
||||||
def to_pem(self):
|
|
||||||
return self.signing_key.to_pem()
|
|
||||||
|
|
||||||
|
|
||||||
def _from_extended_key(ledger, ekey):
|
def _from_extended_key(ledger, ekey):
|
||||||
"""Return a PublicKey or PrivateKey from an extended key raw bytes."""
|
"""Return a PubKey or PrivateKey from an extended key raw bytes."""
|
||||||
if not isinstance(ekey, (bytes, bytearray)):
|
if not isinstance(ekey, (bytes, bytearray)):
|
||||||
raise TypeError('extended key must be raw bytes')
|
raise TypeError('extended key must be raw bytes')
|
||||||
if len(ekey) != 78:
|
if len(ekey) != 78:
|
||||||
|
@ -319,7 +232,7 @@ def _from_extended_key(ledger, ekey):
|
||||||
|
|
||||||
if ekey[:4] == ledger.extended_public_key_prefix:
|
if ekey[:4] == ledger.extended_public_key_prefix:
|
||||||
pubkey = ekey[45:]
|
pubkey = ekey[45:]
|
||||||
key = PublicKey(ledger, pubkey, chain_code, n, depth)
|
key = PubKey(ledger, pubkey, chain_code, n, depth)
|
||||||
elif ekey[:4] == ledger.extended_private_key_prefix:
|
elif ekey[:4] == ledger.extended_private_key_prefix:
|
||||||
if ekey[45] != 0:
|
if ekey[45] != 0:
|
||||||
raise ValueError('invalid extended private key prefix byte')
|
raise ValueError('invalid extended private key prefix byte')
|
||||||
|
@ -337,6 +250,6 @@ def from_extended_key_string(ledger, ekey_str):
|
||||||
xpub6BsnM1W2Y7qLMiuhi7f7dbAwQZ5Cz5gYJCRzTNainXzQXYjFwtuQXHd
|
xpub6BsnM1W2Y7qLMiuhi7f7dbAwQZ5Cz5gYJCRzTNainXzQXYjFwtuQXHd
|
||||||
3qfi3t3KJtHxshXezfjft93w4UE7BGMtKwhqEHae3ZA7d823DVrL
|
3qfi3t3KJtHxshXezfjft93w4UE7BGMtKwhqEHae3ZA7d823DVrL
|
||||||
|
|
||||||
return a PublicKey or PrivateKey.
|
return a PubKey or PrivateKey.
|
||||||
"""
|
"""
|
||||||
return _from_extended_key(ledger, Base58.decode_check(ekey_str))
|
return _from_extended_key(ledger, Base58.decode_check(ekey_str))
|
||||||
|
|
|
@ -881,365 +881,4 @@ HASHES = {
|
||||||
879000: '0eb0810f4b81d1845b0a88f05449408df2e45715c9210a656f45278c5fdf7956',
|
879000: '0eb0810f4b81d1845b0a88f05449408df2e45715c9210a656f45278c5fdf7956',
|
||||||
880000: 'e7d613027e3b4ca38d09bbef07998b57db237c6d67f1e8ea50024d2e0d9a1a72',
|
880000: 'e7d613027e3b4ca38d09bbef07998b57db237c6d67f1e8ea50024d2e0d9a1a72',
|
||||||
881000: '21af4d355d8756b8bf0369b2d79b5c824148ae069026ba5c14f9dd6b7555e1db',
|
881000: '21af4d355d8756b8bf0369b2d79b5c824148ae069026ba5c14f9dd6b7555e1db',
|
||||||
882000: 'bc26f028e547ec44fc3864925bd1493211773b5cb9a9583ba4c1909b89fe0d33',
|
|
||||||
883000: '170a624f4be04cd2fd435cfb6ba1f31b9ef5d7b084a25dfa23cd118c2752029e',
|
|
||||||
884000: '46cccb7a12b4d01d07c211b7b8db41321cd73f30069df27bcdb3bb600c0272b0',
|
|
||||||
885000: '7c27f79d5a99baf0f81f2b09eb5c1bf905976a0f872e02bd4ca9e82f0ed50cb0',
|
|
||||||
886000: '256e3e00cecc72dbbfef5cea627ecf1d43b56edd5fd1642a2bc4e97c17056f34',
|
|
||||||
887000: '658ebac7dfa62bc7a22b1a9ba4e5b425a866f7550a6b40fd07de47119fd1f7e8',
|
|
||||||
888000: '497a9d02868605b9ff6e7f15948a83a7e07606829107e63c2e091c90c7a7b4d4',
|
|
||||||
889000: '561daaa7ebc87e586d37a96ecfbc72484d7eb602824f38f484ed333e78208e9e',
|
|
||||||
890000: 'ab5a8cb625b28343f8fac858eab6576c856dab88bde8cda02b80b3edfd307d71',
|
|
||||||
891000: '2e81d9fc885ddc09222b298ac9efbb73638a5721802b9256de6505ecf122dbaa',
|
|
||||||
892000: '73be08881b8832e986c0bb9a06c70fff346edb2afaf69630e47e4a4a90c5fece',
|
|
||||||
893000: 'd39079dcaa4d8af1c26f0edf7e16df43cd857a31e0aa4c4123226793f1ab497f',
|
|
||||||
894000: '0a3b677d72c590d4b1ff7a9b4098d6b52d0dc10d64c30c2766d18e6eb02872cd',
|
|
||||||
895000: 'a3bbba831f48c5b68e494ee63015b487782c64c5c24bb29436283360c28fd1e0',
|
|
||||||
896000: '20af178a192ca43975ab6c838fe97ca42ba6c682682eddbc6481efd153ecb0a2',
|
|
||||||
897000: '8d0ee14b9fdb853a09ab2951d26b8f7cb8bc8038b09513bd330ee4b0bdcc4780',
|
|
||||||
898000: 'c97fbb70f804408b131a98f9fb4c04cdf2df1655d3e8ff2e0d58ed8537349f4e',
|
|
||||||
899000: 'eba2be80478e8dec2d66ca40b853580c5dad040351c64c177e3d8c25aff6c1b6',
|
|
||||||
900000: 'c4dc344a993558418b93b3f60aaef0030e2a4116086577fbf1e2f544bdbddae1',
|
|
||||||
901000: '36d84229afa63045875fc8fea0c55de8eb90694b3a37cceb825c87abf1fea998',
|
|
||||||
902000: '8ca4890ecfc5e3f9d767e4fcdf318a1e3e3597675bbcfe534d64e76bc4e8fbf4',
|
|
||||||
903000: '8b9f6a7514033c57668ca94fb3758cc6d1ef37ac982c2ff5a9f0f206fcd8d0a8',
|
|
||||||
904000: 'e9ae813991f35ca89af2fe1f1b6adf9e93c6b1dd6a74f003ebbe699a30b252ea',
|
|
||||||
905000: 'd426489d01d4f4c829f2eb68a67721d2c0e1c71e8c33ef9253593447e8603462',
|
|
||||||
906000: '63000bbed97451e68d64485c02c1c3d90b4156237dac315f4e012ffb538e375b',
|
|
||||||
907000: '96759653a4e514541effa7ef86d9f22a272ddde7b069149d17e9d9203a1edafb',
|
|
||||||
908000: 'eec6477d2f3b71bde76dc2380d6e06aa8aa306ca56ba1dd15a31c22ae0db501b',
|
|
||||||
909000: 'd5c2984cf130335aa29296ba5b17672d00360fe0ec73977326180014908c0b55',
|
|
||||||
910000: '7b99cb1c94144f606937903e173bd9ef63bfffd3db8110693fa4c2caa0abc21f',
|
|
||||||
911000: '95eed0d9dd9869ac6f83fa67863e77f24df69bcb90fef70918f30b2400e24ea8',
|
|
||||||
912000: '34c3c8780c54ecced50f0a6b394309d09ee6ce37cd98794699c63771d1d91144',
|
|
||||||
913000: '536052ddcd445702160288ef3f669ce56868c085315556c9f5ca081ef0c0b9e1',
|
|
||||||
914000: '1bcd1fe9632f93a0a1fe7d8a1891a4fc6ef1be40ccf887524a9095ed7aa9fa44',
|
|
||||||
915000: '139bad9fa12ec72a37b62ad8511300ebfda89330fa5d5a83861f864b6adeae67',
|
|
||||||
916000: '81d15282214ff83e2a034212eb58abeafcb5664d3734bff13b22b4c093b20fea',
|
|
||||||
917000: 'f31081031cebe450e4450ef397d91790fc0068e98e6746cd0aab86d17e4448f5',
|
|
||||||
918000: '4af8eb28616ef0e859b5471650c7f8e910cd692a6b4ff3a7171a709db2f18e4e',
|
|
||||||
919000: '78a197b5f9733e9e4dc9820e1c79bd335beb19f6b87056e48e8e21fbe27d83d6',
|
|
||||||
920000: '33d20f86d1367f07d6731e1e2cc9305252b281b1b092403133924cc1052f501d',
|
|
||||||
921000: '6926f1e31e7fe9b8f7a81efa73d5635f8f28c1db1708e4d57f6e7ead951a4beb',
|
|
||||||
922000: '811e2335798eb54696a4b11ca3a44b9d79486262119383d542491afa9ae80204',
|
|
||||||
923000: '8f47ac365bc380885db809f2818ffc7dd2076aaa0f9bf6c180df1b4358dc842e',
|
|
||||||
924000: '535e79802c10630c17fb8fddec3ba2bf85eedbc0c076f3575f8189fe887ba993',
|
|
||||||
925000: 'ca43bd24d17d75d55e72e45549384b395c62e1daf0d3f58f296e18168b918fbf',
|
|
||||||
926000: '9a03be89e0725877d42296e6c995d9c48bb5f4bbd971f5a9add191af2d1c144b',
|
|
||||||
927000: 'a14e0ef6bd1bc221dbba99031c16ddbbd76394186677c29bdf07b89fa2a6efac',
|
|
||||||
928000: 'b16931bd7392e9db26be975b072024210fb5fe6ee22fc0809d51980aa8068a98',
|
|
||||||
929000: '4da56a2e66fcd98a70039d9061ea5eb0fb6d9460b437d2191e47441182419a04',
|
|
||||||
930000: '87e820e2237a54c4ea100bdd0145598f05add92185cd3d0929aa2d5099f4d5e0',
|
|
||||||
931000: '515b22c91172157c443a47cf213014aff144181a77e276e291535ab3762bb1ae',
|
|
||||||
932000: 'e130c6a9eb416f96256d1f90256a148957daa32f56af228d2d9ce6ff27ce2011',
|
|
||||||
933000: '30c992ec7a9a320fb4db260373121efc7b5e7fc744f4b31defbe6a7608e0749e',
|
|
||||||
934000: 'ec490fa0de6b1d78a4121a5044f501bbb3bd9e448c18121cea87eb8e3cadba41',
|
|
||||||
935000: '603e4ae6a6d936c79b3f1c9f9e88305930953b9b390dac442976a6e8395fc520',
|
|
||||||
936000: '2b756fe2de4328e598ed511b8828e5c2c6b5cdda1b5e7c1c26f8e0424c81afa9',
|
|
||||||
937000: '1ae0f15f14a0d4819e34a6c18de9428a9e43e17d75383bffa9ffb18358e93b63',
|
|
||||||
938000: 'cbd7001825ec87b8c6917d6e9e7dc5c8d7767788b6ffd61a61d0c612dbe5de66',
|
|
||||||
939000: 'd770d0395aa79076044783fb37a1bb173cb95c93ff1ba82c34a72c4d8e425a03',
|
|
||||||
940000: '3341d0a0349d091d88d233cd6ea6e0ad553d52039b4d47af51b8a8e7573a7916',
|
|
||||||
941000: '16123b8758e99344ebe6670cd95826881b274c31d4da2a051052955a32bade3a',
|
|
||||||
942000: 'ac7430961e77f902918fe79a52cbf6b523e3f2804ec83d0b17908e131ea9ea68',
|
|
||||||
943000: '2ad08a6877e4687dcb7a623adeddc88403e8082efd6de28328b351282dc141e2',
|
|
||||||
944000: '81382e8c1f47fa7c03fa1726f9b09ed1cd38140fe50683896eaa1b403d7e5fe3',
|
|
||||||
945000: '152bfbb166da04dab16030af28ae65b3275819eed1d0bbfc11eba65616ebefd6',
|
|
||||||
946000: '25b3da0962f87a0d3e4aec8b16483efbcab9514893a42fd31f4cb544ddc45a1f',
|
|
||||||
947000: '2cb738ba342436628ff292797e3d36c4752d71bdc1af87fe758d469d06e36e0e',
|
|
||||||
948000: 'b3683e18570fcc8b986720514539181ec43fb5dbc20fe314c56ab6bd31ab766a',
|
|
||||||
949000: '94ced5bfba55ccffc909bf098d537e047d8d4cbb79f5e2a74146073f39804865',
|
|
||||||
950000: 'b11543cd2aedae27f6ddc3d2b431c897fdcfe59ed3c926b0777bc1e99de4d12a',
|
|
||||||
951000: '21508881a7f80fcd0b9b27bbcfba634b39c6525f5313968c4605cd55b4fec446',
|
|
||||||
952000: 'f9b3ed919c9ca20cd2927d899ee7a86c93c2dd919dafb6fdb792f2d9f1895cb0',
|
|
||||||
953000: 'cf578d8e80eec4102dc1b5321f10b36020b3b32f4b5d4664c90c412ca2ef6b42',
|
|
||||||
954000: 'ed17c919ae5c4be835966b47f667d6082c75917b95584b2d2aff0e32f5c8aa98',
|
|
||||||
955000: '948ea467fa01a20122e2146669214fdd3bb025038554609f7299ece5bca63e39',
|
|
||||||
956000: 'b50ff4c02957ed8764215d25f206f6f1fe6d0eb712a378b937ff952dd479afd2',
|
|
||||||
957000: '169922a3e51517ba6104a883d29aac03a9d20b4d448bd2773137b0d790e3db6b',
|
|
||||||
958000: '92258ac2e8b53167dc30436d93f385d432bd549711ab9790ba4e8263c5c54382',
|
|
||||||
959000: '7ca824697459eb302bcd7fba9d255fb269555abe7cf9d2dd5e54e196d751e682',
|
|
||||||
960000: '89f9ec925d23698076d84f9e852ab04fc956ac4465827303de0c3bb0b685eb32',
|
|
||||||
961000: '41cf75cd71bc12b93674c416e8b01b7410eb9e09eb8727ad93ff0b833c9966c9',
|
|
||||||
962000: '7db1f1dbff3e389713067879bfedf9513ec74bb1e128b13fc2fe23ad55fd0306',
|
|
||||||
963000: 'a35e71c611b2227adeac824d151d2f09bdbecd5765a4e62c6e74a3e4290abc66',
|
|
||||||
964000: 'dc1811130e249d2208d6f85838512b4e5482efb0bd2f619164a68a0c60d7f248',
|
|
||||||
965000: '92f5e25dd1c03102720dd0c3136b1a0769901bf89fcc0262a5e24405f349ca07',
|
|
||||||
966000: '08243d780d8ba96a940f409b87d9c6b8a95c92804173b9156ada0dad35b628dc',
|
|
||||||
967000: 'cb769a8935bb6faeb981da74f4079babbbb89476f825cc897f43e79790295260',
|
|
||||||
968000: 'ff3fc27d2998f4dc4ac1ff378afe14c7d0f43cc328deb9c978ec0e067d1dfaf9',
|
|
||||||
969000: 'e41a3452f45d5f025627d08c9c41017679e9c4804371dd1cc02f3ed49f85dbb2',
|
|
||||||
970000: 'f5eaaf7ba6b47245a4a8096a7785c7b25dc6db342ac2ccbba0c321e97ab58284',
|
|
||||||
971000: '75414062f1d4ed675dadc8f04ba10147a484aaca1ae316dc0b896a92809b3db6',
|
|
||||||
972000: '5bcf2ee00133774c7d060a1a1863dfccc20d5127ecb542470f607dec2504fe6f',
|
|
||||||
973000: '07d15b9656ecde2cd86a9d22c3de8b6505d6bab2aa5a94560b0db9119f1f6f6c',
|
|
||||||
974000: '2059e7924d7a210a88f5a65abc61152506a82edccd27416e796c81b9b8003f13',
|
|
||||||
975000: '7fcf5d8b2c0e51cfbdaa2502a9da0bdb323646899dad37dacc39af9f9e16fc5c',
|
|
||||||
976000: '02acb8cf87a0900436eccfca50371948531041d7b8b410a902205f84dd7fb88e',
|
|
||||||
977000: '2636dfd5a47016c893265473e78ecbf2000769d886f0d01ee7a91e9397210d15',
|
|
||||||
978000: 'ce92f52a35096b94bea73a7d4e113bc4564a4a589b66f1ab86f61c822cf9ee76',
|
|
||||||
979000: '21b8102f5b76be0c8e20d537ebc78ebe46bfcea6b6d2dda950ce5b48e85f72d7',
|
|
||||||
980000: 'f4df0bd63b36105705de62266d654612d9804bad7069d41344de269657e6f084',
|
|
||||||
981000: 'f006cd2718d98d774a5cd18394db7744c812fa149c8a63e76bab934aee89f571',
|
|
||||||
982000: 'da5d6609265d9153022d823b0260aa07e7511ceff7a3fd2ca7ce83cb3900a661',
|
|
||||||
983000: '3a26f3f02aa145fa8c5268fbe10dd9c3546d7dda57489ca5d4b161beb0d5a6e2',
|
|
||||||
984000: '968e8cd37a1137797d40f39f106cae62d1e252b46c7473b9434ad5f870ee88fb',
|
|
||||||
985000: '3129c3bf20deace1a9c92646a9d769da7a07f18dcd5b7a7b1e8cf5fd5390f8e1',
|
|
||||||
986000: '6ce830ca5da322ddbb97fc572ea03218913d070e5910516b33c6113b02b23c21',
|
|
||||||
987000: '7fb1a8635623847132ab766a99b792953379f782d1115b9649f5f9c5a742ca04',
|
|
||||||
988000: '5e8e6c6da7f271129c20c4dd891dcb1df4f9d690ee7cf391c6b7fbd028a0da4c',
|
|
||||||
989000: '12919e34bb9a9ac1d2a01e221eb8c511117fc4e1b3ae15355d95caf4673bdb08',
|
|
||||||
990000: '016f8b18227a0c09da55594a98638ad5b0fbb4896e2ab6163ac40b6015b2811e',
|
|
||||||
991000: 'ddf8cd6e2f4ee07530ae7567cef4fa2c2fd4a655cb20e20422e66fd49bde6489',
|
|
||||||
992000: 'dca77707c0caa3a9605f3dadf593402339c29448869907fb31f6c624e942dcbd',
|
|
||||||
993000: 'de9acc4c7c482ecac741fd6acbbc3a333afab52f3fe5eea4130c0770299a56dd',
|
|
||||||
994000: '54420631f8a801a1b8f391088f599ee22cedc06f24bf67f18272feb8fe70c682',
|
|
||||||
995000: '4b44b26e3e2495716dfd86fc42594cd4b1e4b70bdab4f0905cce4cb9556e008a',
|
|
||||||
996000: 'd6e41fd301fc5f519c343ceb39c9ff845656a4482e4e182abdcd3963fd5fde1c',
|
|
||||||
997000: 'd68b6a509d742b182ffb5a98b0e585a2320a5d3fe6977ad3e6cd06835ef2ea55',
|
|
||||||
998000: '1efcdcbadbec54ce3a93a1857253614536c34f05a0b1924f24bff194dc3392e1',
|
|
||||||
999000: '10a7713e46f47527f3819b4a9257a03f3e207d18e4917d6bcb43fdea3ba82b9a',
|
|
||||||
1000000: '1b4ddb1436df05f07807d6337b93ee1aa8b600fd6a910a8fd5313a39e0440eec',
|
|
||||||
1001000: 'cde0df1abdae26d2c2bdc111be15fb33231c5e167bb8b8f8eec667d71379fee4',
|
|
||||||
1002000: 'd7ce7a96a3ca73a4dfd6a1780e23f834f339142519ea7f45d256c113e27e4857',
|
|
||||||
1003000: 'b1a9b1c562ec62b9dd746d336b4211afc37482d0274ff692a44fa17ac9fe9a28',
|
|
||||||
1004000: '7afd6d0fb0014fbe16a31c84d3f1731736eaeef35e40bb1a1f232fb00345deae',
|
|
||||||
1005000: '4af61ce4cda5de58277f7a67cadea5d3f6ce56e54785b188e32306e00b0414df',
|
|
||||||
1006000: '08e1fb7295efd4a48cb999d899a3d481b682ddbce738fecd88a6d32cbe8234f0',
|
|
||||||
1007000: '14a367a41603dd690541daee8aa4a2882260059e3f85bd8978b7431e8f7db844',
|
|
||||||
1008000: 'e673230e62aaefad0678611f94ff35ee8a6e18eb96438bdfb4b614f54f54dba7',
|
|
||||||
1009000: 'e191af8fb71d0d91419abd19443af3d3f23ee4fe359bb8c390429cc838132bde',
|
|
||||||
1010000: 'ffdba58f184cf60838b75b7899b6633e7cfd34cf36eded572c0133d07387bc49',
|
|
||||||
1011000: '40801af3a5546cb9d53e05e21b74be09de9a421b762ca1d52d2266f5c2055ce8',
|
|
||||||
1012000: '552519acebed0e38102f5270dc60b1da7a123600b6b94169ae74462ae454693f',
|
|
||||||
1013000: '1eee96f48418929927eaa9642777bc806d326cfffaf077bc8695a7ecd438d631',
|
|
||||||
1014000: 'a471093e1de2a8db586412d7351c8d88e44ea890f46e9b43251af427a0a4a879',
|
|
||||||
1015000: '57532f5a522295cc139f008bdcb7a1e6d02e6035d5221b2687c7c216f06297a2',
|
|
||||||
1016000: 'ec46dba07addcb6e62f58456a53c513d876f1c49ae7d76d230adb8debd26027d',
|
|
||||||
1017000: '33ea8d25f342a7465ed71e4bab2b91007991e0994c61d321e3625301a1390322',
|
|
||||||
1018000: '4871c03cc95d4ce0a39bd2cebbb001b2ea1cce1b3561bb841d88f43bb9d12ffd',
|
|
||||||
1019000: 'f5248257576eb2ff4139d6374cc7ce34121cc942598cf9e04d2bd572e09189bb',
|
|
||||||
1020000: 'e7785286897c85cfb0276957bff216039eeb11bc1ebca89d0bb586022caa5750',
|
|
||||||
1021000: 'a30220f17d060634c5f6a1ddc5ea34b01c18fb5eb7e0e8267b66bf5a49525627',
|
|
||||||
1022000: '6083ea49e64ac0d4507c674237cf87d30b90b285ec63d082e626df0223eb7c9c',
|
|
||||||
1023000: '1dc5596d716bc33ee0f56fc40c1f073155a58a7692935c9e5854ef3b65b76828',
|
|
||||||
1024000: '065adfee40dc33abff07fb55339571712b959bc1830dc60b6691e36eab1508ae',
|
|
||||||
1025000: 'bb6903752d31278570e774b80a80782179c78f099e58c3dc4cba7afea7a471c4',
|
|
||||||
1026000: 'f3050f3c2f3a76f5084856b0f089383517caa3f51530fbc29335308f5f170625',
|
|
||||||
1027000: '746ed3701510d07958d11a06f22dbb839d9858373dc5a33249dd69e91bab01fd',
|
|
||||||
1028000: '43f7a96ea6a45b78c29ad4a2f8680ef184438c2bd3686172b0564e0ae6dd7ba1',
|
|
||||||
1029000: 'cbb9916099c59e14fe61d284374f4feaa3d43afec59e4698ed92143576f24b34',
|
|
||||||
1030000: '2e805fc2331e32e586ea692bc3d4e6b11e1ec3f1cab6e331b459f9f1ac9a1f1e',
|
|
||||||
1031000: '04f324f8f6d4f9901cf65f78dc91d6010ea6cf125f5ac0253b57b5f1f79e81e0',
|
|
||||||
1032000: '60ca62f52fdfd858b0ee0fdb380648bde85ca14e2a73565205ed4ee0bc861c77',
|
|
||||||
1033000: 'eb60aac23d599d3099cf98ed8fc3213f1bc06bc1c677429b303e9c81f79f1340',
|
|
||||||
1034000: 'f0328df2daf119ce673ddfa7a39a84576985f701f7a7dec3f56f58c2019ebd4d',
|
|
||||||
1035000: 'f9d3cbce3854de168d8835c96917c01be6244c8f82641e8d9398dfffec4e7107',
|
|
||||||
1036000: '7dca97e6e1d6ed70aa7805f74b768009a270e7ebe1dd951e8727d1d2f2d271f2',
|
|
||||||
1037000: '5329504126b2845b3044f423b521e77ff58d7d242f24bf87c87f4d8d4e03a947',
|
|
||||||
1038000: '5bad3ad55e3daa415f3182a1f2a099fe1767e8fae34e9bb95d47e242b8971434',
|
|
||||||
1039000: 'c29729b8ba49ac0043fe4aa6fc971f8ac3eda68ff92970957ada39a2989b2491',
|
|
||||||
1040000: 'f303aebfc9267600c081d0c021065743f93790df6f5c924a86b773788e0c45be',
|
|
||||||
1041000: 'a1cbe5059fa2275707785b77970c36d79b12c1ba93121bc9064ab9b64abacf7b',
|
|
||||||
1042000: '004b0dd4e438abc54ae832d733df32a6ba35b75e6d3e0c9c1dee5a7950507295',
|
|
||||||
1043000: '31893a3fe7bb4f6dd546c7a8de4a65990e94046aab442d18c68b6bf6acd54518',
|
|
||||||
1044000: '2c4dd479948acc42946f94050810000b0539864ad24a67a7251bff1c4971b035',
|
|
||||||
1045000: '1cea782d60df35a88b30ae205ce37e30abc7cad2b22181722be150bd92c53814',
|
|
||||||
1046000: 'ee808f0efb0f2ef93e8599d8b7f0e2e7c3cdc42353e4ea5165028b961f43d548',
|
|
||||||
1047000: '75f057e2a8cb1d46e5c943d63cc56936a6bac8b1cb89300593845a20baf39765',
|
|
||||||
1048000: '2abcd227f5314baed85e3c5b49d3888a60085c1845c955a8bf96aa3dd6394798',
|
|
||||||
1049000: '5d0ec24b9acd5ab21b42f68e1f3142b7bf83433b98f2fa9794586c8eff45893e',
|
|
||||||
1050000: '1d364b13a4c17bd67a6d1e5f77c26d02faa014d7cd152b4da70380f168b8e0ff',
|
|
||||||
1051000: 'b9a20cec21de84433be9b85817dd4803e875d9275dbc02907b29888431859bae',
|
|
||||||
1052000: '424cb56b00407d73b309b2081dd0bf89213cf024e3aafb3090506aa0ba10f835',
|
|
||||||
1053000: '6df3041a32fafd6a4e08778546d077cf591e1a2a16e77fe7a610efc2b542a9ff',
|
|
||||||
1054000: '78f8dee794f3d4366019339d7ba74ad2b543ecd25dc575620f66e1d535411971',
|
|
||||||
1055000: '43b8e9dae5addd58a7cccf62ba57ab46ffdaa2dcd113cc8ca537e9101b54c096',
|
|
||||||
1056000: '86b7f3741343f85d93410b78cc3fbf03d49b60a664e908703016aa56a206ae7e',
|
|
||||||
1057000: 'b033cf6ec622be6a99dff536a2cf73b36d3c3f8c3835ee17e0dd357403e85c41',
|
|
||||||
1058000: 'a65a6db692a8358e399a5ac3c818902fdb60595262ae05531084848febead249',
|
|
||||||
1059000: 'f6d781d2e2fdb4b7b074d1d8123875d899cdbd6be375cb4288e86f1d14a929f6',
|
|
||||||
1060000: 'cd9019bb1de4926cca16a7bef1a46786f10a3260d467cda0775f73361795abc9',
|
|
||||||
1061000: 'ed4f5dc6f475f95b40595632fafd9e7e5eef388b6cc15772204c0b0e9ee4e542',
|
|
||||||
1062000: 'c44d02a890aa66979b10d1cfa597c877f498841b4e12dd9a7bdf8d4a5fccab80',
|
|
||||||
1063000: '1c093734f5f241b36c1b9971e2759983f88f4033405a2588b4ebfd6998ac7465',
|
|
||||||
1064000: '9e354a83b71bbb9704053bfeea038a9c3d5daad080c6406c698b047c634706a6',
|
|
||||||
1065000: '563188accc4a6e311bd5046516a92a233f11f891b2304d37f151c5a6002b6958',
|
|
||||||
1066000: '333f1b4e996fac87e32dec667533715b31f1736b4342806a81d568b5c5238456',
|
|
||||||
1067000: 'df59a0b7319d5269bdf55043d91ec62bbb30829bb7054da623717a394b6ed678',
|
|
||||||
1068000: '06d8b674a205393edaf20c1d837baadc9caf0b0a675645246263cc163302241d',
|
|
||||||
1069000: 'ac065c48fad1383039d39e23c8367bad7cf9a37e07a5294cd7b04af5827b9961',
|
|
||||||
1070000: '90cd8b50f94208bc459081356474a961f6b764a1217f8fd291f5e4828081b730',
|
|
||||||
1071000: '3c0aa207ba9eea45458ab4fa26d6a027862592adb9bcce30915816e777dc6cfc',
|
|
||||||
1072000: '3d556c08f2300b67b704d3cbf46e22866e3ac164472b5930e2ada23b08475a0f',
|
|
||||||
1073000: 'a39b5c54c24efe3066aa203358b96baea405cd59aac6b0b48930e77799b4dd7d',
|
|
||||||
1074000: 'e8c8273d5a50a60e8744716c9f31496fb29eca87b4d68643f4ecd7ec4e400e23',
|
|
||||||
1075000: 'b8043ae41a1d0d7d4310c85764fcba1424733df347ffc2e8cbda1fe6ccbb5153',
|
|
||||||
1076000: '58468db1f91805e767d334824d6bffe54e0f900d1fb2a89b105086a493053b3d',
|
|
||||||
1077000: '04a78749b58465efa3a56d1735cd082c1f0f796e26486c7136950dbaf6effaa4',
|
|
||||||
1078000: 'e1dd6b58c75b01a67d4a4594dc7b4b2ee9e7d7fa7b25fd6246ce0e86eff33c75',
|
|
||||||
1079000: 'd239af017a6bb664485b14ad15e0eb703775e43018a045a8612b3697794460da',
|
|
||||||
1080000: '29ae5503f8c1249fefeb63fd967a71a70588ee0db1c97497e16366163a684341',
|
|
||||||
1081000: '05103ab27469e0859cbcd3daf42faa2bae798f522534697c7f2b34f7a050ee0f',
|
|
||||||
1082000: '4553d2cb7e90b6db11d242e287fe96822e6cd60e6388b94bf9006411f202ba03',
|
|
||||||
1083000: '97995acd178b2a142d571d5ae1c2a3deaf93a909fd91fb9c541d57f73e32dc99',
|
|
||||||
1084000: '9e3f23376af14d76ab24cd54e321dec019af73ad61067d959ff90043acc5ffcc',
|
|
||||||
1085000: '81c056b14f13cee0d6d6c8079fdd5a1a84c3a5c76cc9448612e8ef6d3531300e',
|
|
||||||
1086000: '8a0004f6809bdd075915a804e43991dfe8f22e05679d2fdaf8e373f101bac5c2',
|
|
||||||
1087000: '27c45a4c9ad24e038f2ebe40835a1c49ac7221d7185082866ee354351ba87c7a',
|
|
||||||
1088000: 'fd27e21747117b00b4ada1cba161ac49edb57cca540f86ac5ba885050f08f824',
|
|
||||||
1089000: 'bff867335767103bc3ed15ede5b9fde88016f8ede15dc5bf3e81ea40dcfc61ae',
|
|
||||||
1090000: '608f75016d1db08888dd59640f63e838c19bdfa833c0cc177ad3d2b818b0db5b',
|
|
||||||
1091000: '90750b452bd4dedaab6b57fecbfe88f71ce3d5437fad7f9ec0fdd270445c7526',
|
|
||||||
1092000: '98287b39f9f1233017dc5d932e5c77f0521ca84587eb3f39f0e7b6c297c749af',
|
|
||||||
1093000: '68a5846ed05c9bb142197849106838765f90f15c10b2cc938eef49b95eaa9d33',
|
|
||||||
1094000: '5660a1aac2fc763a417fc656c8887fc8186bf613ae1ccbb1a664fb43ce1fa1d6',
|
|
||||||
1095000: '62bad3db418b3f4cad3596881b645b72479c71deb0d39c7a4c8bd1577dc225fd',
|
|
||||||
1096000: 'e0e4b2b183591f10dd5614c289412f2fb5e320b7d3278f7c028f42f591872666',
|
|
||||||
1097000: 'a233a233fc2aa5dab9e75106d91388343ef969458ea974f1409a2ab5fc441911',
|
|
||||||
1098000: '16dfa5fa6cbd1188e562697b5f00ac206960d0851ed84adf37ae975fd5ffdd6a',
|
|
||||||
1099000: 'b8a870b7dc6d3263730c00f59d52aa6cce35dc59aa8fba715034cc2d14927260',
|
|
||||||
1100000: 'a3cd7749743da22a3846dcc2edbf1df21b938e829419389e3bc09284797c5b43',
|
|
||||||
1101000: '75b14c2a95e2a095949729b7c0b624bd725a2de98404a8e3247b60c977d0198e',
|
|
||||||
1102000: '4d3af64d37064dd5f57e25d61f248a1e21c1b1cadd7bb1404e35c9fbe06f1fd4',
|
|
||||||
1103000: 'd73c92bfed358dfcd7659228974ab75ea2fc86f2301ee47133adad8075203872',
|
|
||||||
1104000: '30cd82354f37bc0b412123867c7e1835206022a7501853bf8c0d3df02f291645',
|
|
||||||
1105000: '1d2ef984f26693dce77460cd2694e5da46e675077e91a1cea26051733b01a7ef',
|
|
||||||
1106000: '51c076c304222fe3ca308ba6968c46fef448f85be13a095cecb75b90e7954698',
|
|
||||||
1107000: '99e2221339e16acc34c9816f2ef7b866c2dd753aa3cbe484ae831959a23ece68',
|
|
||||||
1108000: '0f1227c250296bfe88eb7eb41703f99f633cfe02870816111e0cadfe778ddb19',
|
|
||||||
1109000: 'b35447f1ad76f95bc4f5886e4028d33acb3ad7b5000dd15516d3f11ce4baa990',
|
|
||||||
1110000: 'ac7baff996062bfaaaddd7d496b17e3ec1c8d34b2143095645ff22fb3888ae00',
|
|
||||||
1111000: '430bbbdcca36b2d69b6a2dd8b07c583a060a467e5f9acbc6de62462e1f7c7036',
|
|
||||||
1112000: 'e5274dea029dc44baff55c05b0555f91b74d29ffd40e3a8c4e2c5b57f9d40bef',
|
|
||||||
1113000: 'cf43863249fa42cfe108220dd40169dac702b0dd9cf5cb699cf2fc96feda8371',
|
|
||||||
1114000: 'fa1c0e551784d21c451564124d2d730e616724f3e535de3c186bcdeb47e80a8f',
|
|
||||||
1115000: '49fe6ecee35a397b83b5a704e950ad028cfb4b7e7a524021e789f4acc0fd6ffe',
|
|
||||||
1116000: '74ecded36751aa8b7901b31f0d16d75d111fc3c40b567f649c04f74ed028aa5c',
|
|
||||||
1117000: 'd9ca760a22190bdf545766b47d963c738a4edcc27f4d15ca801b35751577cfa7',
|
|
||||||
1118000: 'c28d42f871682800ac4e867608227cfb6bc4c00b618e83a8556f201a1c28813c',
|
|
||||||
1119000: 'c5fafc4e1785b0b9e84bb052e392154a5ba1aefe612998017e90772bcd554e08',
|
|
||||||
1120000: 'aa054d428bc9ccee0761da92163817163413065fe1e67ef79a056c5233ea3476',
|
|
||||||
1121000: '0df295bb944218503bd1bf66d2ece0c50fd22dae3391b80673a7ad1e4e5c3934',
|
|
||||||
1122000: 'a13abb350a26673b3933b1de307a60a6845ca594d502599548c6253e21a6d8e8',
|
|
||||||
1123000: 'a4bc6a3abf9ed1f4b14338ff0f03f83456312bc91a93fa89ae6db493050115e1',
|
|
||||||
1124000: '65869938df99adf0dda76200291ce09a54c9bcc787e4bb62cd72c367db58f4f0',
|
|
||||||
1125000: 'ea5e918233b14c3c73d488a906e3741c61bdcafe0393bd0404168fe80c950a46',
|
|
||||||
1126000: 'ce88cd35104fcec51bcee77302e03162dc694802536f5b668786b2245e61bca5',
|
|
||||||
1127000: 'ea19c0c8d205be4be87d02c5301c9ed331e7d75e25b93d1c2137c248882af515',
|
|
||||||
1128000: '006f32d63c2a3adcf4fbad0b0629c97f1beab6446a9c27fbde9472f2d066219e',
|
|
||||||
1129000: '218e5392e1ecf471c3bbc3d79c24dee30ac8db315dbeb61317318efb3f221163',
|
|
||||||
1130000: '30b9da0bd8364e9cd5551b2529341a01a3b7257a238d15b2560e2c99fdb324e8',
|
|
||||||
1131000: '8a7f382cfa023d2eba6639443e67206f8883b57d23ce7e1339234b8bb3098a82',
|
|
||||||
1132000: 'bf9af68a6fe2112d8fe311dfd52334ae2e7b0bac6675c9ebfddb1f386c212668',
|
|
||||||
1133000: '1a30951e2be633502a47c255a93ddbb9ed231d6bb4c55a807c0e910b437766b3',
|
|
||||||
1134000: 'a9bcaf3300b7915e701a8e396eb13f0c7287576323420be7aab3c3ba48020f76',
|
|
||||||
1135000: '337eed9ed072b5ad862af2d3d651f1b49fa852abc590b7e1c2dc381b496f438a',
|
|
||||||
1136000: '208761dbc29ec58302d722a05e937a3cf9e78bfb6495be395dd7b54f02e169dc',
|
|
||||||
1137000: '4e5b67ff3324b64e268049fdc3d82982b847ee359d409ade6368864c38a111e5',
|
|
||||||
1138000: '55d1d0833021a664e85eec8cc90a0985e67cc80d28841aaa8c2231ec28087ebb',
|
|
||||||
1139000: 'e750ada1ec9fa0f2f2461ed68958c7d116a699a82ec12911da5563139f8df19e',
|
|
||||||
1140000: '9cf81407b6ccc8046f0233f97484166945758f7392bb54841c912fcb34cf205c',
|
|
||||||
1141000: 'fccf32b2fae03e3b6b562483776625f9843cd68734c55659e2069cde7e383170',
|
|
||||||
1142000: 'c3608c215dd6569da6c1871c4d72a09ab1caa9663647f2a9454b5693d5d72a65',
|
|
||||||
1143000: 'bd39cb8c4e529d15bbea6baeec66afe52ca18afe32bd812f28fbb0676647cdff',
|
|
||||||
1144000: '6e42d02538565ce7e2d9bf31a304f1fd0ac122d35d17a030160575815901b0b1',
|
|
||||||
1145000: 'b9722e1de2904ce1219140fffb1f4f9f5a041f885faa634404238d103c738b4c',
|
|
||||||
1146000: 'd4de4271459966cee774f538a243d7db0689b213b296463d42e45c93194d7861',
|
|
||||||
1147000: '51fadf109f22bb85574d0fbcbd0b20992983e89aee3d415a7b1c37c44775d9a9',
|
|
||||||
1148000: '137e1fe8da31680d21a42e7421eb608a883a497314e4404625ce44b0edadde6a',
|
|
||||||
1149000: 'cb87867eb04203ce15e0763a2f4389376cea75e0a2877f55e2911c575bef07a8',
|
|
||||||
1150000: '977528ca7953a2c9c19fefaa3aab7ebdec3ac324d74a07d83764ba25d9be0689',
|
|
||||||
1151000: 'a09c51c832600ded63a19201df008075273ea248fd406886e93a2cbaa3bba46b',
|
|
||||||
1152000: '0e5367cfa0f00dd932a5bcc00dcc807fa6825161806bed588e16a57947b4b32d',
|
|
||||||
1153000: '55a9de3dcde2efb56a3c5fea7d22b98c1e180db9a4d4f4f6be7aae1f1cbd7608',
|
|
||||||
1154000: 'abc58cf71c4691ebfaef920252730cf69abbe9de88b424c03051b9b03e85d45a',
|
|
||||||
1155000: '4f074ce73c8a096620b8a32498362eb66a072eae95d561f2d53557cd513ae785',
|
|
||||||
1156000: '540a838a0f0a8834466b17dd456d35b8acae2ec8419f8bd9a704d9ea439062ac',
|
|
||||||
1157000: 'd5310ac671abdb658ea028db86c23fc729af965f91d67a37218c1412cf32a1f5',
|
|
||||||
1158000: '162d906a07e6c35e7c3ebf7069a200521605a97920f5b589d31b19bfd7766ee2',
|
|
||||||
1159000: '600bd8f5e1e62219e220f4dcb650db5812e79956f95ae8a50e83126932685ee0',
|
|
||||||
1160000: '91319398d1a805fac8582c8485e6d84e7490d6cfa6e44e2c630665b6bce0e6b8',
|
|
||||||
1161000: 'f7ad3cff6ee76e1e3df4abe70c600e4af66e1df55bf7b03aee12251d4455a1d4',
|
|
||||||
1162000: '85b9fbba669c2a4d3f85cdb5123f9538c05bd66172b7236d756703f99258454d',
|
|
||||||
1163000: '966085d767d1e5e2e8baf8eda8c11472ec5351181c418b503585284009aaea79',
|
|
||||||
1164000: '1c94e1b531215c019b12caf407296d8868481f49524b7180c7161b0363c1f789',
|
|
||||||
1165000: '803b6bf93735aeae2cf607824e2adf0d754b58da2516c2da1e485c697e472143',
|
|
||||||
1166000: '872561a82f7991633d0927d25cb659d096bbe556fe6dac7a0b6a679820733069',
|
|
||||||
1167000: '6bd7cdd605a3179b54c8af88d1638bf8133fab12cbf0a78d37cf21eddf4395a1',
|
|
||||||
1168000: '79946f5758c1817239cc642d27298bd710983551a8236e49832c6d818b097337',
|
|
||||||
1169000: 'b0994c60728e74de4aa361f37fa85e5296ce3188ae4e0b66d7b34fe86a239c9c',
|
|
||||||
1170000: 'a54188a5a64e0cf8da2406d16a0ac3983b087fc7d6231b6f8abf92cf11dc78cd',
|
|
||||||
1171000: 'ec2924d98e470cc6359821e6468df2c15d60301861d443188730342581230ef2',
|
|
||||||
1172000: 'b4ac11116aa73ce19428009a80e583e19dc9bcd380f7f7ce272a92921d5868d2',
|
|
||||||
1173000: '501d3551f762999dd5a799f3c5658fff2a7f3aff0511488272cd7693fefb8f9d',
|
|
||||||
1174000: '4660074ea48a78ae453cb14b694b2844cc0fb63ed9352ed20d11158bbb5c1f28',
|
|
||||||
1175000: '0727f6b1d9f8fe5677a9ffa0d475f53f5a419ef90b80896c22c2c95de22175de',
|
|
||||||
1176000: '150633d6a35496c24a93c9e19817e90f649c56b7e2558f99e97325bfd5df8b17',
|
|
||||||
1177000: '0849e19f22571b62dba8ff02f6b5a064a7ac36e7ed491321b3663567e8e17294',
|
|
||||||
1178000: '770dd463e7bad80f689f12934e4ae06e24378d1545dcf211fd143beaef49464e',
|
|
||||||
1179000: '059d383dcc60a49b658b674d92fc35cab07b06329c58d73818b6387cb0c06534',
|
|
||||||
1180000: 'e547cb3c636243ca9ae4cfb92c30a0f583eda84e329a5c1e5f64a26fc6fc791e',
|
|
||||||
1181000: '4521a4396ab02f73d45d7a3393ea1c602d255778d52c12079c88bfbad32aab43',
|
|
||||||
1182000: '051cfe993e4b0b34233403a9e8c397dd50e8b78a30fb07e9c260604ee9e624a9',
|
|
||||||
1183000: '44a69c99bb8b85e84ae279f2d8e5400d51cb3d5f0bcd178db49d55548cd66191',
|
|
||||||
1184000: '2a1d23c9bb3c71a533e0c9d25b03bfa7e9db8e014645f3e7fbede6d99fff0191',
|
|
||||||
1185000: 'bb90d6c6d77819163a9e909ee621d874707cdb21c91b1d9e861b204cf37d0ffa',
|
|
||||||
1186000: '4a92051b738ea0e28c64c64f1eb6f0405bc7c3427bef91ff20f4c43cf084d750',
|
|
||||||
1187000: 'f782ac330ca20fb5d8a094ee0f0f8c086a76e3f03ecc6a2c42f8fd07e52e0f41',
|
|
||||||
1188000: '94cb7b653dd3d838c186420158cf0e73db73ec28deaf67d9a2ca902caba4141a',
|
|
||||||
1189000: 'c8128e59b9ec948de890184578a113478ea63f7d57cb75c2c8d5c001a5a724c0',
|
|
||||||
1190000: '4da643bd35e5b98932ae21515a6bffb9c72f2cd8d514cd2d7eac1922af785c3f',
|
|
||||||
1191000: '0f922d86658ac3f53c5f9db360c68ab3f3253a925f23e1323820e3384214719a',
|
|
||||||
1192000: '4c3ab631cf5ba0c236f7c64af6f790fc24448319de6f75dbd28df4e2648d0b7d',
|
|
||||||
1193000: 'eda118d1fac3470a1f8f01f5c78108c8ecdcd6420be30f6d20f1d1831e7b6975',
|
|
||||||
1194000: '5723fff88abd9bb5088476fa5f4221a61c6f8a718703a92f13248ad350abeea2',
|
|
||||||
1195000: '1715846f82d011919e3446c6ce675a65fb80338bd791d4e735702c4767d9adc4',
|
|
||||||
1196000: 'b497667996aee2db61e88f442e728be15ab0b2b64cfd43198691fcf6cdafacc8',
|
|
||||||
1197000: '309a6170d837b8cb334fb888a64ed4e47e6592747e93c8e9d1bf7d608cfef87d',
|
|
||||||
1198000: '3ea918ef64a67dec20051519e6aefaeb7aca2d8583baca9ad5c5bd07073e513a',
|
|
||||||
1199000: '4ec7b7361b0243e5b2996a16e3b27acd662126b95fe542a487c7030e47ea3667',
|
|
||||||
1200000: 'b829c742686fcd642d0f9443336d7e2c4eab81667c90ce553df1350ed10b4233',
|
|
||||||
1201000: '44c022887f1e126fd281b1cae26b2017fa6415a64b105762c87643204ce165a5',
|
|
||||||
1202000: 'b11cc739eb28a14f4e47be125aa7e62d6d6f90c8f8014ee70044ed506d53d938',
|
|
||||||
1203000: '997a7c5fd7a98b39c9ca0790519924d73c3567656b605c97a6fdb7b406c3c64d',
|
|
||||||
1204000: '7d25d872e17195ee277243f7a5a39aa64d8750cec62e4777146acf61a8e76b04',
|
|
||||||
1205000: 'ce8486ae745a4645bee081ef3291d9505174bed05b0668d963b2998b7643dbb0',
|
|
||||||
1206000: '46a0bcea3c411c600dffe3e06e3d1dfbf5879a7ec4dcf3848e794cefcbf2bc0b',
|
|
||||||
1207000: '37e6297bf6e4e2bdd40401d4d7f95e3e3bdafd4a7f76b9c52865cefc6b82b20b',
|
|
||||||
1208000: 'd09e3982a9827b8cf56a5a2f4031dc6b082926c1fd57b63beaaa6cfd534eb902',
|
|
||||||
1209000: '54ae9010a9f146c83464e7ee60b30d9dbee36418561abc4e8d61bce9baa2d21d',
|
|
||||||
1210000: '5dcfd33f8e5ac21c9ba8553758b8cd8afae7961cad428530b5109c2db2ebf39f',
|
|
||||||
1211000: '91c952348bb2c3dfac0d6531a3dac770ea6dab571af257530e9c55493c96bdd9',
|
|
||||||
1212000: 'e62cc3fe044a7f5de4c04a8aed5619548f9d5c6fad9f989d3382cb96de1d780d',
|
|
||||||
1213000: '66b46ffdca8acf1dd04528dadb28b6ac4ce38807c1b84abd685d4ddb3dc59a34',
|
|
||||||
1214000: '2ce4091756ad23746bab4906f46545953cadaf61deae0d78e8a10d4eb51866b1',
|
|
||||||
1215000: '83ce3ca087799cdc4b4c5e7cfeb4a127708724a7ca76aa5f7f4ec1ed48b5fca6',
|
|
||||||
1216000: '7d07b739b7991fbd74926281bf51bba9d5721afab39598720f9ff5f7410a6721',
|
|
||||||
1217000: '76adf49491670d0e8379058eacf0228f330f3c18955dfea1ebe43bc11ee065f3',
|
|
||||||
1218000: '77f422e7301a81692dec69e5c6d35fa988a00a4d820ad0ebb1d595add36558cc',
|
|
||||||
1219000: '8ba9d944f8c468c81799294aeea8dc05ed1bb90bb26552fcd190bd88fedcddf2',
|
|
||||||
1220000: '00330367c255e0fe51b374597995c53353bc5700ad7d603cbd4197141933fe9c',
|
|
||||||
1221000: '3ba8b316b7964f31fdf628ed869a6fd023680cca6611257a31efe22e4d17e578',
|
|
||||||
1222000: '016e58d3fb6a29a3f9281789359460e776e9feb2f0db500482b6e231e1272aef',
|
|
||||||
1223000: 'fdfe767c29a3de7acd913b627d1e5fa887a1af9974f6a8a6474db822468c785c',
|
|
||||||
1224000: '92239f6207bff3689c554e92b24fe2e7be4a2203104ad8ef08b2c6bedd9aeccf',
|
|
||||||
1225000: '9a2f2dd9527b533d3d743efc55236e73e15192171bc8d0cd910918d1ab00aef7',
|
|
||||||
1226000: 'eb8269c75b8c5f66e6ea88ad70883dddcf8a75a45198ca7a46eb0ec606a791bb',
|
|
||||||
1227000: '5c82e624390cd57942dc9d64344eaa3d8991e0437e01802473053245b706290c',
|
|
||||||
1228000: '51e9a7d727f07fc01be7c03e3dd854eb666697f05bf89259baac628520d4402c',
|
|
||||||
1229000: 'c4bfdb651c9abdeda717fb9c8a4c8a6c9c0f78c13d3e6cae3f24f504d734c643',
|
|
||||||
1230000: '9f1ce781d16f2334567cbfb22fff42c14d2b9290cc2883746f435a1fb127021d',
|
|
||||||
1231000: '5c996634b377412ae0a3d8f541f3cc4a354aab72c198aa23a5cfc2678cbabf09',
|
|
||||||
1232000: '86702316a2d1730fbae01a08f36fffe5bf6d3ebb7d76b35a1617713766698b46',
|
|
||||||
1233000: 'fb16b63916c0287cb9b01d0c5aad626ced1b73c49a374c9009703aa90fd27a82',
|
|
||||||
1234000: '7c6f7904602ccd86bfb05cb8d6b5547c989c57cb2e214e93f1220fa4fe29bcb0',
|
|
||||||
1235000: '898b0f20811f52aa5a6bd0c35eff86fca3fbe3b066e423644fa77b2e269d9513',
|
|
||||||
1236000: '39128910ef624b6a8bbd390a311b5587c0991cda834eed996d814fe410cac352',
|
|
||||||
1237000: 'a0709afeedb64af4168ce8cf3dbda667a248df8e91da96acb2333686a2b89325',
|
|
||||||
1238000: 'e00075e7ba8c18cc277bfc5115ae6ff6b9678e6e99efd6e45f549ef8a3981a3d',
|
|
||||||
1239000: '3fba891600738f2d37e279209d52bbe6dc7ce005eeed62048247c96f370e7cd5',
|
|
||||||
1240000: 'def9bf1bec9325db90bb070f532972cfdd74e814c2b5e74a4d5a7c09a963a5f1',
|
|
||||||
1241000: '6a5d187e32bc189ac786959e1fe846031b97ae1ce202c22e1bdb1d2a963005fd',
|
|
||||||
1242000: 'a74d7c0b104eaf76c53a3a31ce51b75bbd8e05b5e84c31f593f505a13d83634c',
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -141,7 +141,7 @@ class CoinSelector:
|
||||||
_) -> List[OutputEffectiveAmountEstimator]:
|
_) -> List[OutputEffectiveAmountEstimator]:
|
||||||
""" Accumulate UTXOs at random until there is enough to cover the target. """
|
""" Accumulate UTXOs at random until there is enough to cover the target. """
|
||||||
target = self.target + self.cost_of_change
|
target = self.target + self.cost_of_change
|
||||||
self.random.shuffle(txos, random=self.random.random) # pylint: disable=deprecated-argument
|
self.random.shuffle(txos, self.random.random)
|
||||||
selection = []
|
selection = []
|
||||||
amount = 0
|
amount = 0
|
||||||
for coin in txos:
|
for coin in txos:
|
||||||
|
|
|
@ -2,7 +2,6 @@ NULL_HASH32 = b'\x00'*32
|
||||||
|
|
||||||
CENT = 1000000
|
CENT = 1000000
|
||||||
COIN = 100*CENT
|
COIN = 100*CENT
|
||||||
DUST = 1000
|
|
||||||
|
|
||||||
TIMEOUT = 30.0
|
TIMEOUT = 30.0
|
||||||
|
|
||||||
|
|
|
@ -9,11 +9,10 @@ from dataclasses import dataclass
|
||||||
from contextvars import ContextVar
|
from contextvars import ContextVar
|
||||||
from typing import Tuple, List, Union, Callable, Any, Awaitable, Iterable, Dict, Optional
|
from typing import Tuple, List, Union, Callable, Any, Awaitable, Iterable, Dict, Optional
|
||||||
from datetime import date
|
from datetime import date
|
||||||
|
|
||||||
from prometheus_client import Gauge, Counter, Histogram
|
from prometheus_client import Gauge, Counter, Histogram
|
||||||
from lbry.utils import LockWithMetrics
|
from lbry.utils import LockWithMetrics
|
||||||
|
|
||||||
from .bip32 import PublicKey
|
from .bip32 import PubKey
|
||||||
from .transaction import Transaction, Output, OutputScript, TXRefImmutable, Input
|
from .transaction import Transaction, Output, OutputScript, TXRefImmutable, Input
|
||||||
from .constants import TXO_TYPES, CLAIM_TYPES
|
from .constants import TXO_TYPES, CLAIM_TYPES
|
||||||
from .util import date_to_julian_day
|
from .util import date_to_julian_day
|
||||||
|
@ -976,9 +975,7 @@ class Database(SQLiteMixin):
|
||||||
sql.append("LEFT JOIN txi ON (txi.position=0 AND txi.txid=txo.txid)")
|
sql.append("LEFT JOIN txi ON (txi.position=0 AND txi.txid=txo.txid)")
|
||||||
return await self.db.execute_fetchall(*query(' '.join(sql), **constraints), read_only=read_only)
|
return await self.db.execute_fetchall(*query(' '.join(sql), **constraints), read_only=read_only)
|
||||||
|
|
||||||
async def get_txos(
|
async def get_txos(self, wallet=None, no_tx=False, no_channel_info=False, read_only=False, **constraints):
|
||||||
self, wallet=None, no_tx=False, no_channel_info=False, read_only=False, **constraints
|
|
||||||
) -> List[Output]:
|
|
||||||
include_is_spent = constraints.get('include_is_spent', False)
|
include_is_spent = constraints.get('include_is_spent', False)
|
||||||
include_is_my_input = constraints.get('include_is_my_input', False)
|
include_is_my_input = constraints.get('include_is_my_input', False)
|
||||||
include_is_my_output = constraints.pop('include_is_my_output', False)
|
include_is_my_output = constraints.pop('include_is_my_output', False)
|
||||||
|
@ -1204,14 +1201,13 @@ class Database(SQLiteMixin):
|
||||||
addresses = await self.select_addresses(', '.join(cols), read_only=read_only, **constraints)
|
addresses = await self.select_addresses(', '.join(cols), read_only=read_only, **constraints)
|
||||||
if 'pubkey' in cols:
|
if 'pubkey' in cols:
|
||||||
for address in addresses:
|
for address in addresses:
|
||||||
address['pubkey'] = PublicKey(
|
address['pubkey'] = PubKey(
|
||||||
self.ledger, address.pop('pubkey'), address.pop('chain_code'),
|
self.ledger, address.pop('pubkey'), address.pop('chain_code'),
|
||||||
address.pop('n'), address.pop('depth')
|
address.pop('n'), address.pop('depth')
|
||||||
)
|
)
|
||||||
return addresses
|
return addresses
|
||||||
|
|
||||||
async def get_address_count(self, cols=None, read_only=False, **constraints):
|
async def get_address_count(self, cols=None, read_only=False, **constraints):
|
||||||
self._clean_txo_constraints_for_aggregation(constraints)
|
|
||||||
count = await self.select_addresses('COUNT(*) as total', read_only=read_only, **constraints)
|
count = await self.select_addresses('COUNT(*) as total', read_only=read_only, **constraints)
|
||||||
return count[0]['total'] or 0
|
return count[0]['total'] or 0
|
||||||
|
|
||||||
|
@ -1245,18 +1241,6 @@ class Database(SQLiteMixin):
|
||||||
async def set_address_history(self, address, history):
|
async def set_address_history(self, address, history):
|
||||||
await self._set_address_history(address, history)
|
await self._set_address_history(address, history)
|
||||||
|
|
||||||
async def is_channel_key_used(self, account, key: PublicKey):
|
|
||||||
channels = await self.get_txos(
|
|
||||||
accounts=[account], txo_type=TXO_TYPES['channel'],
|
|
||||||
no_tx=True, no_channel_info=True
|
|
||||||
)
|
|
||||||
other_key_bytes = key.pubkey_bytes
|
|
||||||
for channel in channels:
|
|
||||||
claim = channel.can_decode_claim
|
|
||||||
if claim and claim.channel.public_key_bytes == other_key_bytes:
|
|
||||||
return True
|
|
||||||
return False
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def constrain_purchases(constraints):
|
def constrain_purchases(constraints):
|
||||||
accounts = constraints.pop('accounts', None)
|
accounts = constraints.pop('accounts', None)
|
||||||
|
|
|
@ -16,18 +16,18 @@ from lbry.crypto.hash import hash160, double_sha256, sha256
|
||||||
from lbry.crypto.base58 import Base58
|
from lbry.crypto.base58 import Base58
|
||||||
from lbry.utils import LRUCacheWithMetrics
|
from lbry.utils import LRUCacheWithMetrics
|
||||||
|
|
||||||
from lbry.wallet.tasks import TaskGroup
|
from .tasks import TaskGroup
|
||||||
from lbry.wallet.database import Database
|
from .database import Database
|
||||||
from lbry.wallet.stream import StreamController
|
from .stream import StreamController
|
||||||
from lbry.wallet.dewies import dewies_to_lbc
|
from .dewies import dewies_to_lbc
|
||||||
from lbry.wallet.account import Account, AddressManager, SingleKey
|
from .account import Account, AddressManager, SingleKey
|
||||||
from lbry.wallet.network import Network
|
from .network import Network
|
||||||
from lbry.wallet.transaction import Transaction, Output
|
from .transaction import Transaction, Output
|
||||||
from lbry.wallet.header import Headers, UnvalidatedHeaders
|
from .header import Headers, UnvalidatedHeaders
|
||||||
from lbry.wallet.checkpoints import HASHES
|
from .checkpoints import HASHES
|
||||||
from lbry.wallet.constants import TXO_TYPES, CLAIM_TYPES, COIN, NULL_HASH32
|
from .constants import TXO_TYPES, CLAIM_TYPES, COIN, NULL_HASH32
|
||||||
from lbry.wallet.bip32 import PublicKey, PrivateKey
|
from .bip32 import PubKey, PrivateKey
|
||||||
from lbry.wallet.coinselection import CoinSelector
|
from .coinselection import CoinSelector
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
@ -106,7 +106,7 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
target_timespan = 150
|
target_timespan = 150
|
||||||
|
|
||||||
default_fee_per_byte = 50
|
default_fee_per_byte = 50
|
||||||
default_fee_per_name_char = 0
|
default_fee_per_name_char = 200000
|
||||||
|
|
||||||
checkpoints = HASHES
|
checkpoints = HASHES
|
||||||
|
|
||||||
|
@ -226,7 +226,7 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
return account.get_private_key(address_info['chain'], address_info['pubkey'].n)
|
return account.get_private_key(address_info['chain'], address_info['pubkey'].n)
|
||||||
return None
|
return None
|
||||||
|
|
||||||
async def get_public_key_for_address(self, wallet, address) -> Optional[PublicKey]:
|
async def get_public_key_for_address(self, wallet, address) -> Optional[PubKey]:
|
||||||
match = await self._get_account_and_address_info_for_address(wallet, address)
|
match = await self._get_account_and_address_info_for_address(wallet, address)
|
||||||
if match:
|
if match:
|
||||||
_, address_info = match
|
_, address_info = match
|
||||||
|
@ -329,10 +329,10 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
async def start(self):
|
async def start(self):
|
||||||
if not os.path.exists(self.path):
|
if not os.path.exists(self.path):
|
||||||
os.mkdir(self.path)
|
os.mkdir(self.path)
|
||||||
await asyncio.wait(map(asyncio.create_task, [
|
await asyncio.wait([
|
||||||
self.db.open(),
|
self.db.open(),
|
||||||
self.headers.open()
|
self.headers.open()
|
||||||
]))
|
])
|
||||||
fully_synced = self.on_ready.first
|
fully_synced = self.on_ready.first
|
||||||
asyncio.create_task(self.network.start())
|
asyncio.create_task(self.network.start())
|
||||||
await self.network.on_connected.first
|
await self.network.on_connected.first
|
||||||
|
@ -365,10 +365,6 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
await self.db.close()
|
await self.db.close()
|
||||||
await self.headers.close()
|
await self.headers.close()
|
||||||
|
|
||||||
async def tasks_are_done(self):
|
|
||||||
await self._update_tasks.done.wait()
|
|
||||||
await self._other_tasks.done.wait()
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def local_height_including_downloaded_height(self):
|
def local_height_including_downloaded_height(self):
|
||||||
return max(self.headers.height, self._download_height)
|
return max(self.headers.height, self._download_height)
|
||||||
|
@ -466,15 +462,14 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
async def subscribe_accounts(self):
|
async def subscribe_accounts(self):
|
||||||
if self.network.is_connected and self.accounts:
|
if self.network.is_connected and self.accounts:
|
||||||
log.info("Subscribe to %i accounts", len(self.accounts))
|
log.info("Subscribe to %i accounts", len(self.accounts))
|
||||||
await asyncio.wait(map(asyncio.create_task, [
|
await asyncio.wait([
|
||||||
self.subscribe_account(a) for a in self.accounts
|
self.subscribe_account(a) for a in self.accounts
|
||||||
]))
|
])
|
||||||
|
|
||||||
async def subscribe_account(self, account: Account):
|
async def subscribe_account(self, account: Account):
|
||||||
for address_manager in account.address_managers.values():
|
for address_manager in account.address_managers.values():
|
||||||
await self.subscribe_addresses(address_manager, await address_manager.get_addresses())
|
await self.subscribe_addresses(address_manager, await address_manager.get_addresses())
|
||||||
await account.ensure_address_gap()
|
await account.ensure_address_gap()
|
||||||
await account.deterministic_channel_keys.ensure_cache_primed()
|
|
||||||
|
|
||||||
async def unsubscribe_account(self, account: Account):
|
async def unsubscribe_account(self, account: Account):
|
||||||
for address in await account.get_addresses():
|
for address in await account.get_addresses():
|
||||||
|
@ -555,7 +550,6 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
)
|
)
|
||||||
remote_history_txids = {txid for txid, _ in remote_history}
|
remote_history_txids = {txid for txid, _ in remote_history}
|
||||||
async for tx in self.request_synced_transactions(to_request, remote_history_txids, address):
|
async for tx in self.request_synced_transactions(to_request, remote_history_txids, address):
|
||||||
self.maybe_has_channel_key(tx)
|
|
||||||
pending_synced_history[tx_indexes[tx.id]] = f"{tx.id}:{tx.height}:"
|
pending_synced_history[tx_indexes[tx.id]] = f"{tx.id}:{tx.height}:"
|
||||||
if len(pending_synced_history) % 100 == 0:
|
if len(pending_synced_history) % 100 == 0:
|
||||||
log.info("Syncing address %s: %d/%d", address, len(pending_synced_history), len(to_request))
|
log.info("Syncing address %s: %d/%d", address, len(pending_synced_history), len(to_request))
|
||||||
|
@ -623,12 +617,6 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
tx.is_verified = merkle_root == header['merkle_root']
|
tx.is_verified = merkle_root == header['merkle_root']
|
||||||
return tx
|
return tx
|
||||||
|
|
||||||
def maybe_has_channel_key(self, tx):
|
|
||||||
for txo in tx._outputs:
|
|
||||||
if txo.can_decode_claim and txo.claim.is_channel:
|
|
||||||
for account in self.accounts:
|
|
||||||
account.deterministic_channel_keys.maybe_generate_deterministic_key_for_channel(txo)
|
|
||||||
|
|
||||||
async def request_transactions(self, to_request: Tuple[Tuple[str, int], ...], cached=False):
|
async def request_transactions(self, to_request: Tuple[Tuple[str, int], ...], cached=False):
|
||||||
batches = [[]]
|
batches = [[]]
|
||||||
remote_heights = {}
|
remote_heights = {}
|
||||||
|
@ -722,15 +710,6 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
return account.address_managers[details['chain']]
|
return account.address_managers[details['chain']]
|
||||||
return None
|
return None
|
||||||
|
|
||||||
async def broadcast_or_release(self, tx, blocking=False):
|
|
||||||
try:
|
|
||||||
await self.broadcast(tx)
|
|
||||||
except:
|
|
||||||
await self.release_tx(tx)
|
|
||||||
raise
|
|
||||||
if blocking:
|
|
||||||
await self.wait(tx, timeout=None)
|
|
||||||
|
|
||||||
def broadcast(self, tx):
|
def broadcast(self, tx):
|
||||||
# broadcast can't be a retriable call yet
|
# broadcast can't be a retriable call yet
|
||||||
return self.network.broadcast(hexlify(tx.raw).decode())
|
return self.network.broadcast(hexlify(tx.raw).decode())
|
||||||
|
@ -752,7 +731,7 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
while timeout and (int(time.perf_counter()) - start) <= timeout:
|
while timeout and (int(time.perf_counter()) - start) <= timeout:
|
||||||
if await self._wait_round(tx, height, addresses):
|
if await self._wait_round(tx, height, addresses):
|
||||||
return
|
return
|
||||||
raise asyncio.TimeoutError(f'Timed out waiting for transaction. {tx.id}')
|
raise asyncio.TimeoutError('Timed out waiting for transaction.')
|
||||||
|
|
||||||
async def _wait_round(self, tx: Transaction, height: int, addresses: Iterable[str]):
|
async def _wait_round(self, tx: Transaction, height: int, addresses: Iterable[str]):
|
||||||
records = await self.db.get_addresses(address__in=addresses)
|
records = await self.db.get_addresses(address__in=addresses)
|
||||||
|
@ -789,9 +768,13 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
include_is_my_output=False,
|
include_is_my_output=False,
|
||||||
include_sent_supports=False,
|
include_sent_supports=False,
|
||||||
include_sent_tips=False,
|
include_sent_tips=False,
|
||||||
include_received_tips=False) -> Tuple[List[Output], dict, int, int]:
|
include_received_tips=False,
|
||||||
|
hub_server=False) -> Tuple[List[Output], dict, int, int]:
|
||||||
encoded_outputs = await query
|
encoded_outputs = await query
|
||||||
outputs = Outputs.from_base64(encoded_outputs or '') # TODO: why is the server returning None?
|
if hub_server:
|
||||||
|
outputs = Outputs.from_grpc(encoded_outputs)
|
||||||
|
else:
|
||||||
|
outputs = Outputs.from_base64(encoded_outputs or b'') # TODO: why is the server returning None?
|
||||||
txs: List[Transaction] = []
|
txs: List[Transaction] = []
|
||||||
if len(outputs.txs) > 0:
|
if len(outputs.txs) > 0:
|
||||||
async for tx in self.request_transactions(tuple(outputs.txs), cached=True):
|
async for tx in self.request_transactions(tuple(outputs.txs), cached=True):
|
||||||
|
@ -867,9 +850,12 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
txo.received_tips = tips
|
txo.received_tips = tips
|
||||||
return txos, blocked, outputs.offset, outputs.total
|
return txos, blocked, outputs.offset, outputs.total
|
||||||
|
|
||||||
async def resolve(self, accounts, urls, **kwargs):
|
async def resolve(self, accounts, urls, new_sdk_server=None, **kwargs):
|
||||||
txos = []
|
txos = []
|
||||||
urls_copy = list(urls)
|
urls_copy = list(urls)
|
||||||
|
if new_sdk_server:
|
||||||
|
resolve = partial(self.network.new_resolve, new_sdk_server)
|
||||||
|
else:
|
||||||
resolve = partial(self.network.retriable_call, self.network.resolve)
|
resolve = partial(self.network.retriable_call, self.network.resolve)
|
||||||
while urls_copy:
|
while urls_copy:
|
||||||
batch, urls_copy = urls_copy[:100], urls_copy[100:]
|
batch, urls_copy = urls_copy[:100], urls_copy[100:]
|
||||||
|
@ -895,14 +881,17 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
return await self.network.sum_supports(new_sdk_server, **kwargs)
|
return await self.network.sum_supports(new_sdk_server, **kwargs)
|
||||||
|
|
||||||
async def claim_search(
|
async def claim_search(
|
||||||
self, accounts,
|
self, accounts, include_purchase_receipt=False, include_is_my_output=False,
|
||||||
include_purchase_receipt=False,
|
new_sdk_server=None, **kwargs) -> Tuple[List[Output], dict, int, int]:
|
||||||
include_is_my_output=False,
|
if new_sdk_server:
|
||||||
**kwargs) -> Tuple[List[Output], dict, int, int]:
|
claim_search = partial(self.network.new_claim_search, new_sdk_server)
|
||||||
|
else:
|
||||||
|
claim_search = self.network.claim_search
|
||||||
return await self._inflate_outputs(
|
return await self._inflate_outputs(
|
||||||
self.network.claim_search(**kwargs), accounts,
|
claim_search(**kwargs), accounts,
|
||||||
include_purchase_receipt=include_purchase_receipt,
|
include_purchase_receipt=include_purchase_receipt,
|
||||||
include_is_my_output=include_is_my_output
|
include_is_my_output=include_is_my_output,
|
||||||
|
hub_server=new_sdk_server is not None
|
||||||
)
|
)
|
||||||
|
|
||||||
# async def get_claim_by_claim_id(self, accounts, claim_id, **kwargs) -> Output:
|
# async def get_claim_by_claim_id(self, accounts, claim_id, **kwargs) -> Output:
|
||||||
|
@ -938,7 +927,9 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
"%d change addresses (gap: %d), %d channels, %d certificates and %d claims. ",
|
"%d change addresses (gap: %d), %d channels, %d certificates and %d claims. ",
|
||||||
account.id, balance, total_receiving, account.receiving.gap, total_change,
|
account.id, balance, total_receiving, account.receiving.gap, total_change,
|
||||||
account.change.gap, channel_count, len(account.channel_keys), claim_count)
|
account.change.gap, channel_count, len(account.channel_keys), claim_count)
|
||||||
except Exception:
|
except Exception as err:
|
||||||
|
if isinstance(err, asyncio.CancelledError): # TODO: remove when updated to 3.8
|
||||||
|
raise
|
||||||
log.exception(
|
log.exception(
|
||||||
'Failed to display wallet state, please file issue '
|
'Failed to display wallet state, please file issue '
|
||||||
'for this bug along with the traceback you see below:')
|
'for this bug along with the traceback you see below:')
|
||||||
|
@ -961,7 +952,9 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
claim_ids = [p.purchased_claim_id for p in purchases]
|
claim_ids = [p.purchased_claim_id for p in purchases]
|
||||||
try:
|
try:
|
||||||
resolved, _, _, _ = await self.claim_search([], claim_ids=claim_ids)
|
resolved, _, _, _ = await self.claim_search([], claim_ids=claim_ids)
|
||||||
except Exception:
|
except Exception as err:
|
||||||
|
if isinstance(err, asyncio.CancelledError): # TODO: remove when updated to 3.8
|
||||||
|
raise
|
||||||
log.exception("Resolve failed while looking up purchased claim ids:")
|
log.exception("Resolve failed while looking up purchased claim ids:")
|
||||||
resolved = []
|
resolved = []
|
||||||
lookup = {claim.claim_id: claim for claim in resolved}
|
lookup = {claim.claim_id: claim for claim in resolved}
|
||||||
|
@ -1041,7 +1034,9 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
claim_ids = collection.claim.collection.claims.ids[offset:page_size + offset]
|
claim_ids = collection.claim.collection.claims.ids[offset:page_size + offset]
|
||||||
try:
|
try:
|
||||||
resolve_results, _, _, _ = await self.claim_search([], claim_ids=claim_ids)
|
resolve_results, _, _, _ = await self.claim_search([], claim_ids=claim_ids)
|
||||||
except Exception:
|
except Exception as err:
|
||||||
|
if isinstance(err, asyncio.CancelledError): # TODO: remove when updated to 3.8
|
||||||
|
raise
|
||||||
log.exception("Resolve failed while looking up collection claim ids:")
|
log.exception("Resolve failed while looking up collection claim ids:")
|
||||||
return []
|
return []
|
||||||
claims = []
|
claims = []
|
||||||
|
|
|
@ -3,6 +3,7 @@ import json
|
||||||
import typing
|
import typing
|
||||||
import logging
|
import logging
|
||||||
import asyncio
|
import asyncio
|
||||||
|
from distutils.util import strtobool
|
||||||
|
|
||||||
from binascii import unhexlify
|
from binascii import unhexlify
|
||||||
from decimal import Decimal
|
from decimal import Decimal
|
||||||
|
@ -11,13 +12,13 @@ from typing import List, Type, MutableSequence, MutableMapping, Optional
|
||||||
from lbry.error import KeyFeeAboveMaxAllowedError, WalletNotLoadedError
|
from lbry.error import KeyFeeAboveMaxAllowedError, WalletNotLoadedError
|
||||||
from lbry.conf import Config, NOT_SET
|
from lbry.conf import Config, NOT_SET
|
||||||
|
|
||||||
from lbry.wallet.dewies import dewies_to_lbc
|
from .dewies import dewies_to_lbc
|
||||||
from lbry.wallet.account import Account
|
from .account import Account
|
||||||
from lbry.wallet.ledger import Ledger, LedgerRegistry
|
from .ledger import Ledger, LedgerRegistry
|
||||||
from lbry.wallet.transaction import Transaction, Output
|
from .transaction import Transaction, Output
|
||||||
from lbry.wallet.database import Database
|
from .database import Database
|
||||||
from lbry.wallet.wallet import Wallet, WalletStorage, ENCRYPT_ON_DISK
|
from .wallet import Wallet, WalletStorage, ENCRYPT_ON_DISK
|
||||||
from lbry.wallet.rpc.jsonrpc import CodeMessageError
|
from .rpc.jsonrpc import CodeMessageError
|
||||||
|
|
||||||
if typing.TYPE_CHECKING:
|
if typing.TYPE_CHECKING:
|
||||||
from lbry.extras.daemon.exchange_rate_manager import ExchangeRateManager
|
from lbry.extras.daemon.exchange_rate_manager import ExchangeRateManager
|
||||||
|
@ -182,6 +183,7 @@ class WalletManager:
|
||||||
}[config.blockchain_name]
|
}[config.blockchain_name]
|
||||||
|
|
||||||
ledger_config = {
|
ledger_config = {
|
||||||
|
'use_go_hub': not strtobool(os.environ.get('ENABLE_LEGACY_SEARCH') or 'yes'),
|
||||||
'auto_connect': True,
|
'auto_connect': True,
|
||||||
'explicit_servers': [],
|
'explicit_servers': [],
|
||||||
'hub_timeout': config.hub_timeout,
|
'hub_timeout': config.hub_timeout,
|
||||||
|
@ -192,8 +194,6 @@ class WalletManager:
|
||||||
'data_path': config.wallet_dir,
|
'data_path': config.wallet_dir,
|
||||||
'tx_cache_size': config.transaction_cache_size
|
'tx_cache_size': config.transaction_cache_size
|
||||||
}
|
}
|
||||||
if 'LBRY_FEE_PER_NAME_CHAR' in os.environ:
|
|
||||||
ledger_config['fee_per_name_char'] = int(os.environ.get('LBRY_FEE_PER_NAME_CHAR'))
|
|
||||||
|
|
||||||
wallets_directory = os.path.join(config.wallet_dir, 'wallets')
|
wallets_directory = os.path.join(config.wallet_dir, 'wallets')
|
||||||
if not os.path.exists(wallets_directory):
|
if not os.path.exists(wallets_directory):
|
||||||
|
@ -236,6 +236,7 @@ class WalletManager:
|
||||||
|
|
||||||
async def reset(self):
|
async def reset(self):
|
||||||
self.ledger.config = {
|
self.ledger.config = {
|
||||||
|
'use_go_hub': not strtobool(os.environ.get('ENABLE_LEGACY_SEARCH') or 'yes'),
|
||||||
'auto_connect': True,
|
'auto_connect': True,
|
||||||
'explicit_servers': [],
|
'explicit_servers': [],
|
||||||
'default_servers': Config.lbryum_servers.default,
|
'default_servers': Config.lbryum_servers.default,
|
||||||
|
@ -317,4 +318,10 @@ class WalletManager:
|
||||||
)
|
)
|
||||||
|
|
||||||
async def broadcast_or_release(self, tx, blocking=False):
|
async def broadcast_or_release(self, tx, blocking=False):
|
||||||
await self.ledger.broadcast_or_release(tx, blocking=blocking)
|
try:
|
||||||
|
await self.ledger.broadcast(tx)
|
||||||
|
except:
|
||||||
|
await self.ledger.release_tx(tx)
|
||||||
|
raise
|
||||||
|
if blocking:
|
||||||
|
await self.ledger.wait(tx, timeout=None)
|
||||||
|
|
|
@ -7,13 +7,16 @@ from time import perf_counter
|
||||||
from collections import defaultdict
|
from collections import defaultdict
|
||||||
from typing import Dict, Optional, Tuple
|
from typing import Dict, Optional, Tuple
|
||||||
import aiohttp
|
import aiohttp
|
||||||
|
import grpc
|
||||||
|
from lbry.schema.types.v2 import hub_pb2_grpc
|
||||||
|
from lbry.schema.types.v2.hub_pb2 import SearchRequest
|
||||||
|
|
||||||
from lbry import __version__
|
from lbry import __version__
|
||||||
from lbry.utils import resolve_host
|
from lbry.utils import resolve_host
|
||||||
from lbry.error import IncompatibleWalletServerError
|
from lbry.error import IncompatibleWalletServerError
|
||||||
from lbry.wallet.rpc import RPCSession as BaseClientSession, Connector, RPCError, ProtocolError
|
from lbry.wallet.rpc import RPCSession as BaseClientSession, Connector, RPCError, ProtocolError
|
||||||
from lbry.wallet.stream import StreamController
|
from lbry.wallet.stream import StreamController
|
||||||
from lbry.wallet.udp import SPVStatusClientProtocol, SPVPong
|
from lbry.wallet.server.udp import SPVStatusClientProtocol, SPVPong
|
||||||
from lbry.conf import KnownHubsList
|
from lbry.conf import KnownHubsList
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
@ -117,9 +120,9 @@ class ClientSession(BaseClientSession):
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
await asyncio.sleep(max(0, max_idle - (now - self.last_send)))
|
await asyncio.sleep(max(0, max_idle - (now - self.last_send)))
|
||||||
except (Exception, asyncio.CancelledError) as err:
|
except Exception as err:
|
||||||
if isinstance(err, asyncio.CancelledError):
|
if isinstance(err, asyncio.CancelledError):
|
||||||
log.info("closing connection to %s:%i", *self.server)
|
log.warning("closing connection to %s:%i", *self.server)
|
||||||
else:
|
else:
|
||||||
log.exception("lost connection to spv")
|
log.exception("lost connection to spv")
|
||||||
finally:
|
finally:
|
||||||
|
@ -137,7 +140,7 @@ class ClientSession(BaseClientSession):
|
||||||
controller.add(request.args)
|
controller.add(request.args)
|
||||||
|
|
||||||
def connection_lost(self, exc):
|
def connection_lost(self, exc):
|
||||||
log.debug("Connection lost: %s:%d", *self.server)
|
log.warning("Connection lost: %s:%d", *self.server)
|
||||||
super().connection_lost(exc)
|
super().connection_lost(exc)
|
||||||
self.response_time = None
|
self.response_time = None
|
||||||
self.connection_latency = None
|
self.connection_latency = None
|
||||||
|
@ -214,7 +217,7 @@ class Network:
|
||||||
def loop_task_done_callback(f):
|
def loop_task_done_callback(f):
|
||||||
try:
|
try:
|
||||||
f.result()
|
f.result()
|
||||||
except (Exception, asyncio.CancelledError):
|
except Exception:
|
||||||
if self.running:
|
if self.running:
|
||||||
log.exception("wallet server connection loop crashed")
|
log.exception("wallet server connection loop crashed")
|
||||||
|
|
||||||
|
@ -300,7 +303,7 @@ class Network:
|
||||||
concurrency=self.config.get('concurrent_hub_requests', 30))
|
concurrency=self.config.get('concurrent_hub_requests', 30))
|
||||||
try:
|
try:
|
||||||
await client.create_connection()
|
await client.create_connection()
|
||||||
log.info("Connected to spv server %s:%i", host, port)
|
log.warning("Connected to spv server %s:%i", host, port)
|
||||||
await client.ensure_server_version()
|
await client.ensure_server_version()
|
||||||
return client
|
return client
|
||||||
except (asyncio.TimeoutError, ConnectionError, OSError, IncompatibleWalletServerError, RPCError):
|
except (asyncio.TimeoutError, ConnectionError, OSError, IncompatibleWalletServerError, RPCError):
|
||||||
|
@ -312,8 +315,7 @@ class Network:
|
||||||
sleep_delay = 30
|
sleep_delay = 30
|
||||||
while self.running:
|
while self.running:
|
||||||
await asyncio.wait(
|
await asyncio.wait(
|
||||||
map(asyncio.create_task, [asyncio.sleep(30), self._urgent_need_reconnect.wait()]),
|
[asyncio.sleep(30), self._urgent_need_reconnect.wait()], return_when=asyncio.FIRST_COMPLETED
|
||||||
return_when=asyncio.FIRST_COMPLETED
|
|
||||||
)
|
)
|
||||||
if self._urgent_need_reconnect.is_set():
|
if self._urgent_need_reconnect.is_set():
|
||||||
sleep_delay = 30
|
sleep_delay = 30
|
||||||
|
@ -339,13 +341,14 @@ class Network:
|
||||||
try:
|
try:
|
||||||
if not self._urgent_need_reconnect.is_set():
|
if not self._urgent_need_reconnect.is_set():
|
||||||
await asyncio.wait(
|
await asyncio.wait(
|
||||||
[self._keepalive_task, asyncio.create_task(self._urgent_need_reconnect.wait())],
|
[self._keepalive_task, self._urgent_need_reconnect.wait()],
|
||||||
return_when=asyncio.FIRST_COMPLETED
|
return_when=asyncio.FIRST_COMPLETED
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
await self._keepalive_task
|
await self._keepalive_task
|
||||||
if self._urgent_need_reconnect.is_set():
|
if self._urgent_need_reconnect.is_set():
|
||||||
log.warning("urgent reconnect needed")
|
log.warning("urgent reconnect needed")
|
||||||
|
self._urgent_need_reconnect.clear()
|
||||||
if self._keepalive_task and not self._keepalive_task.done():
|
if self._keepalive_task and not self._keepalive_task.done():
|
||||||
self._keepalive_task.cancel()
|
self._keepalive_task.cancel()
|
||||||
except asyncio.CancelledError:
|
except asyncio.CancelledError:
|
||||||
|
@ -354,7 +357,7 @@ class Network:
|
||||||
self._keepalive_task = None
|
self._keepalive_task = None
|
||||||
self.client = None
|
self.client = None
|
||||||
self.server_features = None
|
self.server_features = None
|
||||||
log.info("connection lost to %s", server_str)
|
log.warning("connection lost to %s", server_str)
|
||||||
log.info("network loop finished")
|
log.info("network loop finished")
|
||||||
|
|
||||||
async def stop(self):
|
async def stop(self):
|
||||||
|
@ -391,6 +394,7 @@ class Network:
|
||||||
log.warning("Wallet server call timed out, retrying.")
|
log.warning("Wallet server call timed out, retrying.")
|
||||||
except ConnectionError:
|
except ConnectionError:
|
||||||
log.warning("connection error")
|
log.warning("connection error")
|
||||||
|
|
||||||
raise asyncio.CancelledError() # if we got here, we are shutting down
|
raise asyncio.CancelledError() # if we got here, we are shutting down
|
||||||
|
|
||||||
def _update_remote_height(self, header_args):
|
def _update_remote_height(self, header_args):
|
||||||
|
@ -473,6 +477,21 @@ class Network:
|
||||||
def claim_search(self, session_override=None, **kwargs):
|
def claim_search(self, session_override=None, **kwargs):
|
||||||
return self.rpc('blockchain.claimtrie.search', kwargs, False, session_override)
|
return self.rpc('blockchain.claimtrie.search', kwargs, False, session_override)
|
||||||
|
|
||||||
|
async def new_resolve(self, server, urls):
|
||||||
|
message = {"method": "resolve", "params": {"urls": urls, "protobuf": True}}
|
||||||
|
async with self.aiohttp_session.post(server, json=message) as r:
|
||||||
|
result = await r.json()
|
||||||
|
return result['result']
|
||||||
|
|
||||||
|
async def new_claim_search(self, server, **kwargs):
|
||||||
|
async with grpc.aio.insecure_channel(server) as channel:
|
||||||
|
stub = hub_pb2_grpc.HubStub(channel)
|
||||||
|
try:
|
||||||
|
response = await stub.Search(SearchRequest(**kwargs))
|
||||||
|
except grpc.aio.AioRpcError as error:
|
||||||
|
raise RPCError(error.code(), error.details())
|
||||||
|
return response
|
||||||
|
|
||||||
async def sum_supports(self, server, **kwargs):
|
async def sum_supports(self, server, **kwargs):
|
||||||
message = {"method": "support_sum", "params": kwargs}
|
message = {"method": "support_sum", "params": kwargs}
|
||||||
async with self.aiohttp_session.post(server, json=message) as r:
|
async with self.aiohttp_session.post(server, json=message) as r:
|
||||||
|
|
|
@ -1,2 +1,5 @@
|
||||||
from lbry.wallet.orchstr8.node import Conductor
|
__hub_url__ = (
|
||||||
from lbry.wallet.orchstr8.service import ConductorService
|
"https://github.com/lbryio/hub/releases/download/leveldb-hub/hub"
|
||||||
|
)
|
||||||
|
from .node import Conductor
|
||||||
|
from .service import ConductorService
|
||||||
|
|
|
@ -5,9 +5,7 @@ import aiohttp
|
||||||
|
|
||||||
from lbry import wallet
|
from lbry import wallet
|
||||||
from lbry.wallet.orchstr8.node import (
|
from lbry.wallet.orchstr8.node import (
|
||||||
Conductor,
|
Conductor, get_blockchain_node_from_ledger
|
||||||
get_lbcd_node_from_ledger,
|
|
||||||
get_lbcwallet_node_from_ledger
|
|
||||||
)
|
)
|
||||||
from lbry.wallet.orchstr8.service import ConductorService
|
from lbry.wallet.orchstr8.service import ConductorService
|
||||||
|
|
||||||
|
@ -18,11 +16,10 @@ def get_argument_parser():
|
||||||
)
|
)
|
||||||
subparsers = parser.add_subparsers(dest='command', help='sub-command help')
|
subparsers = parser.add_subparsers(dest='command', help='sub-command help')
|
||||||
|
|
||||||
subparsers.add_parser("download", help="Download lbcd and lbcwallet node binaries.")
|
subparsers.add_parser("download", help="Download blockchain node binary.")
|
||||||
|
|
||||||
start = subparsers.add_parser("start", help="Start orchstr8 service.")
|
start = subparsers.add_parser("start", help="Start orchstr8 service.")
|
||||||
start.add_argument("--lbcd", help="Hostname to start lbcd node.")
|
start.add_argument("--blockchain", help="Hostname to start blockchain node.")
|
||||||
start.add_argument("--lbcwallet", help="Hostname to start lbcwallet node.")
|
|
||||||
start.add_argument("--spv", help="Hostname to start SPV server.")
|
start.add_argument("--spv", help="Hostname to start SPV server.")
|
||||||
start.add_argument("--wallet", help="Hostname to start wallet daemon.")
|
start.add_argument("--wallet", help="Hostname to start wallet daemon.")
|
||||||
|
|
||||||
|
@ -50,8 +47,7 @@ def main():
|
||||||
|
|
||||||
if command == 'download':
|
if command == 'download':
|
||||||
logging.getLogger('blockchain').setLevel(logging.INFO)
|
logging.getLogger('blockchain').setLevel(logging.INFO)
|
||||||
get_lbcd_node_from_ledger(wallet).ensure()
|
get_blockchain_node_from_ledger(wallet).ensure()
|
||||||
get_lbcwallet_node_from_ledger(wallet).ensure()
|
|
||||||
|
|
||||||
elif command == 'generate':
|
elif command == 'generate':
|
||||||
loop.run_until_complete(run_remote_command(
|
loop.run_until_complete(run_remote_command(
|
||||||
|
@ -61,12 +57,9 @@ def main():
|
||||||
elif command == 'start':
|
elif command == 'start':
|
||||||
|
|
||||||
conductor = Conductor()
|
conductor = Conductor()
|
||||||
if getattr(args, 'lbcd', False):
|
if getattr(args, 'blockchain', False):
|
||||||
conductor.lbcd_node.hostname = args.lbcd
|
conductor.blockchain_node.hostname = args.blockchain
|
||||||
loop.run_until_complete(conductor.start_lbcd())
|
loop.run_until_complete(conductor.start_blockchain())
|
||||||
if getattr(args, 'lbcwallet', False):
|
|
||||||
conductor.lbcwallet_node.hostname = args.lbcwallet
|
|
||||||
loop.run_until_complete(conductor.start_lbcwallet())
|
|
||||||
if getattr(args, 'spv', False):
|
if getattr(args, 'spv', False):
|
||||||
conductor.spv_node.hostname = args.spv
|
conductor.spv_node.hostname = args.spv
|
||||||
loop.run_until_complete(conductor.start_spv())
|
loop.run_until_complete(conductor.start_spv())
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
# pylint: disable=import-error
|
|
||||||
import os
|
import os
|
||||||
import json
|
import json
|
||||||
import shutil
|
import shutil
|
||||||
|
@ -8,44 +7,35 @@ import tarfile
|
||||||
import logging
|
import logging
|
||||||
import tempfile
|
import tempfile
|
||||||
import subprocess
|
import subprocess
|
||||||
import platform
|
import importlib
|
||||||
|
from distutils.util import strtobool
|
||||||
|
|
||||||
from binascii import hexlify
|
from binascii import hexlify
|
||||||
from typing import Type, Optional
|
from typing import Type, Optional
|
||||||
import urllib.request
|
import urllib.request
|
||||||
from uuid import uuid4
|
from uuid import uuid4
|
||||||
|
|
||||||
|
|
||||||
import lbry
|
import lbry
|
||||||
|
from lbry.wallet.server.server import Server
|
||||||
|
from lbry.wallet.server.env import Env
|
||||||
from lbry.wallet import Wallet, Ledger, RegTestLedger, WalletManager, Account, BlockHeightEvent
|
from lbry.wallet import Wallet, Ledger, RegTestLedger, WalletManager, Account, BlockHeightEvent
|
||||||
from lbry.conf import KnownHubsList, Config
|
from lbry.conf import KnownHubsList, Config
|
||||||
|
from lbry.wallet.orchstr8 import __hub_url__
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
try:
|
|
||||||
from hub.herald.env import ServerEnv
|
def get_spvserver_from_ledger(ledger_module):
|
||||||
from hub.scribe.env import BlockchainEnv
|
spvserver_path, regtest_class_name = ledger_module.__spvserver__.rsplit('.', 1)
|
||||||
from hub.elastic_sync.env import ElasticEnv
|
spvserver_module = importlib.import_module(spvserver_path)
|
||||||
from hub.herald.service import HubServerService
|
return getattr(spvserver_module, regtest_class_name)
|
||||||
from hub.elastic_sync.service import ElasticSyncService
|
|
||||||
from hub.scribe.service import BlockchainProcessorService
|
|
||||||
except ImportError:
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
def get_lbcd_node_from_ledger(ledger_module):
|
def get_blockchain_node_from_ledger(ledger_module):
|
||||||
return LBCDNode(
|
return BlockchainNode(
|
||||||
ledger_module.__lbcd_url__,
|
ledger_module.__node_url__,
|
||||||
ledger_module.__lbcd__,
|
os.path.join(ledger_module.__node_bin__, ledger_module.__node_daemon__),
|
||||||
ledger_module.__lbcctl__
|
os.path.join(ledger_module.__node_bin__, ledger_module.__node_cli__)
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def get_lbcwallet_node_from_ledger(ledger_module):
|
|
||||||
return LBCWalletNode(
|
|
||||||
ledger_module.__lbcwallet_url__,
|
|
||||||
ledger_module.__lbcwallet__,
|
|
||||||
ledger_module.__lbcctl__
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@ -53,37 +43,53 @@ class Conductor:
|
||||||
|
|
||||||
def __init__(self, seed=None):
|
def __init__(self, seed=None):
|
||||||
self.manager_module = WalletManager
|
self.manager_module = WalletManager
|
||||||
self.lbcd_node = get_lbcd_node_from_ledger(lbry.wallet)
|
self.spv_module = get_spvserver_from_ledger(lbry.wallet)
|
||||||
self.lbcwallet_node = get_lbcwallet_node_from_ledger(lbry.wallet)
|
|
||||||
self.spv_node = SPVNode()
|
self.blockchain_node = get_blockchain_node_from_ledger(lbry.wallet)
|
||||||
|
self.spv_node = SPVNode(self.spv_module)
|
||||||
self.wallet_node = WalletNode(
|
self.wallet_node = WalletNode(
|
||||||
self.manager_module, RegTestLedger, default_seed=seed
|
self.manager_module, RegTestLedger, default_seed=seed
|
||||||
)
|
)
|
||||||
self.lbcd_started = False
|
self.hub_node = HubNode(__hub_url__, "hub", self.spv_node)
|
||||||
self.lbcwallet_started = False
|
|
||||||
|
self.blockchain_started = False
|
||||||
self.spv_started = False
|
self.spv_started = False
|
||||||
self.wallet_started = False
|
self.wallet_started = False
|
||||||
|
self.hub_started = False
|
||||||
|
|
||||||
self.log = log.getChild('conductor')
|
self.log = log.getChild('conductor')
|
||||||
|
|
||||||
async def start_lbcd(self):
|
async def start_blockchain(self):
|
||||||
if not self.lbcd_started:
|
if not self.blockchain_started:
|
||||||
await self.lbcd_node.start()
|
asyncio.create_task(self.blockchain_node.start())
|
||||||
self.lbcd_started = True
|
await self.blockchain_node.running.wait()
|
||||||
|
await self.blockchain_node.generate(200)
|
||||||
|
self.blockchain_started = True
|
||||||
|
|
||||||
async def stop_lbcd(self, cleanup=True):
|
async def stop_blockchain(self):
|
||||||
if self.lbcd_started:
|
if self.blockchain_started:
|
||||||
await self.lbcd_node.stop(cleanup)
|
await self.blockchain_node.stop(cleanup=True)
|
||||||
self.lbcd_started = False
|
self.blockchain_started = False
|
||||||
|
|
||||||
|
async def start_hub(self):
|
||||||
|
if not self.hub_started:
|
||||||
|
asyncio.create_task(self.hub_node.start())
|
||||||
|
await self.blockchain_node.running.wait()
|
||||||
|
self.hub_started = True
|
||||||
|
|
||||||
|
async def stop_hub(self):
|
||||||
|
if self.hub_started:
|
||||||
|
await self.hub_node.stop(cleanup=True)
|
||||||
|
self.hub_started = False
|
||||||
|
|
||||||
async def start_spv(self):
|
async def start_spv(self):
|
||||||
if not self.spv_started:
|
if not self.spv_started:
|
||||||
await self.spv_node.start(self.lbcwallet_node)
|
await self.spv_node.start(self.blockchain_node)
|
||||||
self.spv_started = True
|
self.spv_started = True
|
||||||
|
|
||||||
async def stop_spv(self, cleanup=True):
|
async def stop_spv(self):
|
||||||
if self.spv_started:
|
if self.spv_started:
|
||||||
await self.spv_node.stop(cleanup)
|
await self.spv_node.stop(cleanup=True)
|
||||||
self.spv_started = False
|
self.spv_started = False
|
||||||
|
|
||||||
async def start_wallet(self):
|
async def start_wallet(self):
|
||||||
|
@ -91,30 +97,13 @@ class Conductor:
|
||||||
await self.wallet_node.start(self.spv_node)
|
await self.wallet_node.start(self.spv_node)
|
||||||
self.wallet_started = True
|
self.wallet_started = True
|
||||||
|
|
||||||
async def stop_wallet(self, cleanup=True):
|
async def stop_wallet(self):
|
||||||
if self.wallet_started:
|
if self.wallet_started:
|
||||||
await self.wallet_node.stop(cleanup)
|
await self.wallet_node.stop(cleanup=True)
|
||||||
self.wallet_started = False
|
self.wallet_started = False
|
||||||
|
|
||||||
async def start_lbcwallet(self, clean=True):
|
|
||||||
if not self.lbcwallet_started:
|
|
||||||
await self.lbcwallet_node.start()
|
|
||||||
if clean:
|
|
||||||
mining_addr = await self.lbcwallet_node.get_new_address()
|
|
||||||
self.lbcwallet_node.mining_addr = mining_addr
|
|
||||||
await self.lbcwallet_node.generate(200)
|
|
||||||
# unlock the wallet for the next 1 hour
|
|
||||||
await self.lbcwallet_node.wallet_passphrase("password", 3600)
|
|
||||||
self.lbcwallet_started = True
|
|
||||||
|
|
||||||
async def stop_lbcwallet(self, cleanup=True):
|
|
||||||
if self.lbcwallet_started:
|
|
||||||
await self.lbcwallet_node.stop(cleanup)
|
|
||||||
self.lbcwallet_started = False
|
|
||||||
|
|
||||||
async def start(self):
|
async def start(self):
|
||||||
await self.start_lbcd()
|
await self.start_blockchain()
|
||||||
await self.start_lbcwallet()
|
|
||||||
await self.start_spv()
|
await self.start_spv()
|
||||||
await self.start_wallet()
|
await self.start_wallet()
|
||||||
|
|
||||||
|
@ -122,8 +111,7 @@ class Conductor:
|
||||||
all_the_stops = [
|
all_the_stops = [
|
||||||
self.stop_wallet,
|
self.stop_wallet,
|
||||||
self.stop_spv,
|
self.stop_spv,
|
||||||
self.stop_lbcwallet,
|
self.stop_blockchain
|
||||||
self.stop_lbcd
|
|
||||||
]
|
]
|
||||||
for stop in all_the_stops:
|
for stop in all_the_stops:
|
||||||
try:
|
try:
|
||||||
|
@ -131,12 +119,6 @@ class Conductor:
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
log.exception('Exception raised while stopping services:', exc_info=e)
|
log.exception('Exception raised while stopping services:', exc_info=e)
|
||||||
|
|
||||||
async def clear_mempool(self):
|
|
||||||
await self.stop_lbcwallet(cleanup=False)
|
|
||||||
await self.stop_lbcd(cleanup=False)
|
|
||||||
await self.start_lbcd()
|
|
||||||
await self.start_lbcwallet(clean=False)
|
|
||||||
|
|
||||||
|
|
||||||
class WalletNode:
|
class WalletNode:
|
||||||
|
|
||||||
|
@ -157,14 +139,14 @@ class WalletNode:
|
||||||
|
|
||||||
async def start(self, spv_node: 'SPVNode', seed=None, connect=True, config=None):
|
async def start(self, spv_node: 'SPVNode', seed=None, connect=True, config=None):
|
||||||
wallets_dir = os.path.join(self.data_path, 'wallets')
|
wallets_dir = os.path.join(self.data_path, 'wallets')
|
||||||
wallet_file_name = os.path.join(wallets_dir, 'my_wallet.json')
|
|
||||||
if not os.path.isdir(wallets_dir):
|
|
||||||
os.mkdir(wallets_dir)
|
os.mkdir(wallets_dir)
|
||||||
|
wallet_file_name = os.path.join(wallets_dir, 'my_wallet.json')
|
||||||
with open(wallet_file_name, 'w') as wallet_file:
|
with open(wallet_file_name, 'w') as wallet_file:
|
||||||
wallet_file.write('{"version": 1, "accounts": []}\n')
|
wallet_file.write('{"version": 1, "accounts": []}\n')
|
||||||
self.manager = self.manager_class.from_config({
|
self.manager = self.manager_class.from_config({
|
||||||
'ledgers': {
|
'ledgers': {
|
||||||
self.ledger_class.get_id(): {
|
self.ledger_class.get_id(): {
|
||||||
|
'use_go_hub': not strtobool(os.environ.get('ENABLE_LEGACY_SEARCH') or 'yes'),
|
||||||
'api_port': self.port,
|
'api_port': self.port,
|
||||||
'explicit_servers': [(spv_node.hostname, spv_node.port)],
|
'explicit_servers': [(spv_node.hostname, spv_node.port)],
|
||||||
'default_servers': Config.lbryum_servers.default,
|
'default_servers': Config.lbryum_servers.default,
|
||||||
|
@ -172,7 +154,6 @@ class WalletNode:
|
||||||
'known_hubs': config.known_hubs if config else KnownHubsList(),
|
'known_hubs': config.known_hubs if config else KnownHubsList(),
|
||||||
'hub_timeout': 30,
|
'hub_timeout': 30,
|
||||||
'concurrent_hub_requests': 32,
|
'concurrent_hub_requests': 32,
|
||||||
'fee_per_name_char': 200000
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
'wallets': [wallet_file_name]
|
'wallets': [wallet_file_name]
|
||||||
|
@ -203,83 +184,55 @@ class WalletNode:
|
||||||
|
|
||||||
|
|
||||||
class SPVNode:
|
class SPVNode:
|
||||||
def __init__(self, node_number=1):
|
|
||||||
self.node_number = node_number
|
def __init__(self, coin_class, node_number=1):
|
||||||
|
self.coin_class = coin_class
|
||||||
self.controller = None
|
self.controller = None
|
||||||
self.data_path = None
|
self.data_path = None
|
||||||
self.server: Optional[HubServerService] = None
|
self.server = None
|
||||||
self.writer: Optional[BlockchainProcessorService] = None
|
|
||||||
self.es_writer: Optional[ElasticSyncService] = None
|
|
||||||
self.hostname = 'localhost'
|
self.hostname = 'localhost'
|
||||||
self.port = 50001 + node_number # avoid conflict with default daemon
|
self.port = 50001 + node_number # avoid conflict with default daemon
|
||||||
self.udp_port = self.port
|
self.udp_port = self.port
|
||||||
self.elastic_notifier_port = 19080 + node_number
|
|
||||||
self.elastic_services = f'localhost:9200/localhost:{self.elastic_notifier_port}'
|
|
||||||
self.session_timeout = 600
|
self.session_timeout = 600
|
||||||
self.stopped = True
|
self.rpc_port = '0' # disabled by default
|
||||||
|
self.stopped = False
|
||||||
self.index_name = uuid4().hex
|
self.index_name = uuid4().hex
|
||||||
|
|
||||||
async def start(self, lbcwallet_node: 'LBCWalletNode', extraconf=None):
|
async def start(self, blockchain_node: 'BlockchainNode', extraconf=None):
|
||||||
if not self.stopped:
|
|
||||||
log.warning("spv node is already running")
|
|
||||||
return
|
|
||||||
self.stopped = False
|
|
||||||
try:
|
|
||||||
self.data_path = tempfile.mkdtemp()
|
self.data_path = tempfile.mkdtemp()
|
||||||
conf = {
|
conf = {
|
||||||
'description': '',
|
'DESCRIPTION': '',
|
||||||
'payment_address': '',
|
'PAYMENT_ADDRESS': '',
|
||||||
'daily_fee': '0',
|
'DAILY_FEE': '0',
|
||||||
'db_dir': self.data_path,
|
'DB_DIRECTORY': self.data_path,
|
||||||
'daemon_url': lbcwallet_node.rpc_url,
|
'DAEMON_URL': blockchain_node.rpc_url,
|
||||||
'reorg_limit': 100,
|
'REORG_LIMIT': '100',
|
||||||
'host': self.hostname,
|
'HOST': self.hostname,
|
||||||
'tcp_port': self.port,
|
'TCP_PORT': str(self.port),
|
||||||
'udp_port': self.udp_port,
|
'UDP_PORT': str(self.udp_port),
|
||||||
'elastic_services': self.elastic_services,
|
'SESSION_TIMEOUT': str(self.session_timeout),
|
||||||
'session_timeout': self.session_timeout,
|
'MAX_QUERY_WORKERS': '0',
|
||||||
'max_query_workers': 0,
|
'INDIVIDUAL_TAG_INDEXES': '',
|
||||||
'es_index_prefix': self.index_name,
|
'RPC_PORT': self.rpc_port,
|
||||||
'chain': 'regtest',
|
'ES_INDEX_PREFIX': self.index_name,
|
||||||
'index_address_status': False
|
'ES_MODE': 'writer',
|
||||||
}
|
}
|
||||||
if extraconf:
|
if extraconf:
|
||||||
conf.update(extraconf)
|
conf.update(extraconf)
|
||||||
self.writer = BlockchainProcessorService(
|
# TODO: don't use os.environ
|
||||||
BlockchainEnv(db_dir=self.data_path, daemon_url=lbcwallet_node.rpc_url,
|
os.environ.update(conf)
|
||||||
reorg_limit=100, max_query_workers=0, chain='regtest', index_address_status=False)
|
self.server = Server(Env(self.coin_class))
|
||||||
)
|
self.server.bp.mempool.refresh_secs = self.server.bp.prefetcher.polling_delay = 0.5
|
||||||
self.server = HubServerService(ServerEnv(**conf))
|
|
||||||
self.es_writer = ElasticSyncService(
|
|
||||||
ElasticEnv(
|
|
||||||
db_dir=self.data_path, reorg_limit=100, max_query_workers=0, chain='regtest',
|
|
||||||
elastic_notifier_port=self.elastic_notifier_port,
|
|
||||||
es_index_prefix=self.index_name,
|
|
||||||
filtering_channel_ids=(extraconf or {}).get('filtering_channel_ids'),
|
|
||||||
blocking_channel_ids=(extraconf or {}).get('blocking_channel_ids')
|
|
||||||
)
|
|
||||||
)
|
|
||||||
await self.writer.start()
|
|
||||||
await self.es_writer.start()
|
|
||||||
await self.server.start()
|
await self.server.start()
|
||||||
except Exception as e:
|
|
||||||
self.stopped = True
|
|
||||||
log.exception("failed to start spv node")
|
|
||||||
raise e
|
|
||||||
|
|
||||||
async def stop(self, cleanup=True):
|
async def stop(self, cleanup=True):
|
||||||
if self.stopped:
|
if self.stopped:
|
||||||
log.warning("spv node is already stopped")
|
|
||||||
return
|
return
|
||||||
try:
|
try:
|
||||||
|
await self.server.db.search_index.delete_index()
|
||||||
|
await self.server.db.search_index.stop()
|
||||||
await self.server.stop()
|
await self.server.stop()
|
||||||
await self.es_writer.delete_index()
|
|
||||||
await self.es_writer.stop()
|
|
||||||
await self.writer.stop()
|
|
||||||
self.stopped = True
|
self.stopped = True
|
||||||
except Exception as e:
|
|
||||||
log.exception("failed to stop spv node")
|
|
||||||
raise e
|
|
||||||
finally:
|
finally:
|
||||||
cleanup and self.cleanup()
|
cleanup and self.cleanup()
|
||||||
|
|
||||||
|
@ -287,19 +240,18 @@ class SPVNode:
|
||||||
shutil.rmtree(self.data_path, ignore_errors=True)
|
shutil.rmtree(self.data_path, ignore_errors=True)
|
||||||
|
|
||||||
|
|
||||||
class LBCDProcess(asyncio.SubprocessProtocol):
|
class BlockchainProcess(asyncio.SubprocessProtocol):
|
||||||
|
|
||||||
IGNORE_OUTPUT = [
|
IGNORE_OUTPUT = [
|
||||||
b'keypool keep',
|
b'keypool keep',
|
||||||
b'keypool reserve',
|
b'keypool reserve',
|
||||||
b'keypool return',
|
b'keypool return',
|
||||||
b'Block submitted',
|
|
||||||
]
|
]
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.ready = asyncio.Event()
|
self.ready = asyncio.Event()
|
||||||
self.stopped = asyncio.Event()
|
self.stopped = asyncio.Event()
|
||||||
self.log = log.getChild('lbcd')
|
self.log = log.getChild('blockchain')
|
||||||
|
|
||||||
def pipe_data_received(self, fd, data):
|
def pipe_data_received(self, fd, data):
|
||||||
if self.log and not any(ignore in data for ignore in self.IGNORE_OUTPUT):
|
if self.log and not any(ignore in data for ignore in self.IGNORE_OUTPUT):
|
||||||
|
@ -310,7 +262,7 @@ class LBCDProcess(asyncio.SubprocessProtocol):
|
||||||
if b'Error:' in data:
|
if b'Error:' in data:
|
||||||
self.ready.set()
|
self.ready.set()
|
||||||
raise SystemError(data.decode())
|
raise SystemError(data.decode())
|
||||||
if b'RPCS: RPC server listening on' in data:
|
if b'Done loading' in data:
|
||||||
self.ready.set()
|
self.ready.set()
|
||||||
|
|
||||||
def process_exited(self):
|
def process_exited(self):
|
||||||
|
@ -318,57 +270,39 @@ class LBCDProcess(asyncio.SubprocessProtocol):
|
||||||
self.ready.set()
|
self.ready.set()
|
||||||
|
|
||||||
|
|
||||||
class WalletProcess(asyncio.SubprocessProtocol):
|
class BlockchainNode:
|
||||||
|
|
||||||
IGNORE_OUTPUT = [
|
P2SH_SEGWIT_ADDRESS = "p2sh-segwit"
|
||||||
]
|
BECH32_ADDRESS = "bech32"
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
self.ready = asyncio.Event()
|
|
||||||
self.stopped = asyncio.Event()
|
|
||||||
self.log = log.getChild('lbcwallet')
|
|
||||||
self.transport: Optional[asyncio.transports.SubprocessTransport] = None
|
|
||||||
|
|
||||||
def pipe_data_received(self, fd, data):
|
|
||||||
if self.log and not any(ignore in data for ignore in self.IGNORE_OUTPUT):
|
|
||||||
if b'Error:' in data:
|
|
||||||
self.log.error(data.decode())
|
|
||||||
else:
|
|
||||||
self.log.info(data.decode())
|
|
||||||
if b'Error:' in data:
|
|
||||||
self.ready.set()
|
|
||||||
raise SystemError(data.decode())
|
|
||||||
if b'WLLT: Finished rescan' in data:
|
|
||||||
self.ready.set()
|
|
||||||
|
|
||||||
def process_exited(self):
|
|
||||||
self.stopped.set()
|
|
||||||
self.ready.set()
|
|
||||||
|
|
||||||
|
|
||||||
class LBCDNode:
|
|
||||||
def __init__(self, url, daemon, cli):
|
def __init__(self, url, daemon, cli):
|
||||||
self.latest_release_url = url
|
self.latest_release_url = url
|
||||||
self.project_dir = os.path.dirname(os.path.dirname(__file__))
|
self.project_dir = os.path.dirname(os.path.dirname(__file__))
|
||||||
self.bin_dir = os.path.join(self.project_dir, 'bin')
|
self.bin_dir = os.path.join(self.project_dir, 'bin')
|
||||||
self.daemon_bin = os.path.join(self.bin_dir, daemon)
|
self.daemon_bin = os.path.join(self.bin_dir, daemon)
|
||||||
self.cli_bin = os.path.join(self.bin_dir, cli)
|
self.cli_bin = os.path.join(self.bin_dir, cli)
|
||||||
self.log = log.getChild('lbcd')
|
self.log = log.getChild('blockchain')
|
||||||
self.data_path = tempfile.mkdtemp()
|
self.data_path = None
|
||||||
self.protocol = None
|
self.protocol = None
|
||||||
self.transport = None
|
self.transport = None
|
||||||
|
self.block_expected = 0
|
||||||
self.hostname = 'localhost'
|
self.hostname = 'localhost'
|
||||||
self.peerport = 29246
|
self.peerport = 9246 + 2 # avoid conflict with default peer port
|
||||||
self.rpcport = 29245
|
self.rpcport = 9245 + 2 # avoid conflict with default rpc port
|
||||||
self.rpcuser = 'rpcuser'
|
self.rpcuser = 'rpcuser'
|
||||||
self.rpcpassword = 'rpcpassword'
|
self.rpcpassword = 'rpcpassword'
|
||||||
self.stopped = True
|
self.stopped = False
|
||||||
|
self.restart_ready = asyncio.Event()
|
||||||
|
self.restart_ready.set()
|
||||||
self.running = asyncio.Event()
|
self.running = asyncio.Event()
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def rpc_url(self):
|
def rpc_url(self):
|
||||||
return f'http://{self.rpcuser}:{self.rpcpassword}@{self.hostname}:{self.rpcport}/'
|
return f'http://{self.rpcuser}:{self.rpcpassword}@{self.hostname}:{self.rpcport}/'
|
||||||
|
|
||||||
|
def is_expected_block(self, e: BlockHeightEvent):
|
||||||
|
return self.block_expected == e.height
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def exists(self):
|
def exists(self):
|
||||||
return (
|
return (
|
||||||
|
@ -377,12 +311,6 @@ class LBCDNode:
|
||||||
)
|
)
|
||||||
|
|
||||||
def download(self):
|
def download(self):
|
||||||
uname = platform.uname()
|
|
||||||
target_os = str.lower(uname.system)
|
|
||||||
target_arch = str.replace(uname.machine, 'x86_64', 'amd64')
|
|
||||||
target_platform = target_os + '_' + target_arch
|
|
||||||
self.latest_release_url = str.replace(self.latest_release_url, 'TARGET_PLATFORM', target_platform)
|
|
||||||
|
|
||||||
downloaded_file = os.path.join(
|
downloaded_file = os.path.join(
|
||||||
self.bin_dir,
|
self.bin_dir,
|
||||||
self.latest_release_url[self.latest_release_url.rfind('/')+1:]
|
self.latest_release_url[self.latest_release_url.rfind('/')+1:]
|
||||||
|
@ -416,106 +344,170 @@ class LBCDNode:
|
||||||
return self.exists or self.download()
|
return self.exists or self.download()
|
||||||
|
|
||||||
async def start(self):
|
async def start(self):
|
||||||
if not self.stopped:
|
|
||||||
return
|
|
||||||
self.stopped = False
|
|
||||||
try:
|
|
||||||
assert self.ensure()
|
assert self.ensure()
|
||||||
|
self.data_path = tempfile.mkdtemp()
|
||||||
loop = asyncio.get_event_loop()
|
loop = asyncio.get_event_loop()
|
||||||
asyncio.get_child_watcher().attach_loop(loop)
|
asyncio.get_child_watcher().attach_loop(loop)
|
||||||
command = [
|
command = [
|
||||||
self.daemon_bin,
|
self.daemon_bin,
|
||||||
'--notls',
|
f'-datadir={self.data_path}', '-printtoconsole', '-regtest', '-server', '-txindex',
|
||||||
f'--datadir={self.data_path}',
|
f'-rpcuser={self.rpcuser}', f'-rpcpassword={self.rpcpassword}', f'-rpcport={self.rpcport}',
|
||||||
'--regtest', f'--listen=127.0.0.1:{self.peerport}', f'--rpclisten=127.0.0.1:{self.rpcport}',
|
f'-port={self.peerport}'
|
||||||
'--txindex', f'--rpcuser={self.rpcuser}', f'--rpcpass={self.rpcpassword}'
|
|
||||||
]
|
]
|
||||||
self.log.info(' '.join(command))
|
self.log.info(' '.join(command))
|
||||||
|
while not self.stopped:
|
||||||
|
if self.running.is_set():
|
||||||
|
await asyncio.sleep(1)
|
||||||
|
continue
|
||||||
|
await self.restart_ready.wait()
|
||||||
|
try:
|
||||||
self.transport, self.protocol = await loop.subprocess_exec(
|
self.transport, self.protocol = await loop.subprocess_exec(
|
||||||
LBCDProcess, *command
|
BlockchainProcess, *command
|
||||||
)
|
)
|
||||||
await self.protocol.ready.wait()
|
await self.protocol.ready.wait()
|
||||||
assert not self.protocol.stopped.is_set()
|
assert not self.protocol.stopped.is_set()
|
||||||
self.running.set()
|
self.running.set()
|
||||||
except asyncio.CancelledError:
|
except asyncio.CancelledError:
|
||||||
self.running.clear()
|
self.running.clear()
|
||||||
self.stopped = True
|
|
||||||
raise
|
raise
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.running.clear()
|
self.running.clear()
|
||||||
self.stopped = True
|
log.exception('failed to start lbrycrdd', exc_info=e)
|
||||||
log.exception('failed to start lbcd', exc_info=e)
|
|
||||||
raise
|
|
||||||
|
|
||||||
async def stop(self, cleanup=True):
|
async def stop(self, cleanup=True):
|
||||||
if self.stopped:
|
self.stopped = True
|
||||||
return
|
|
||||||
try:
|
try:
|
||||||
if self.transport:
|
|
||||||
self.transport.terminate()
|
self.transport.terminate()
|
||||||
await self.protocol.stopped.wait()
|
await self.protocol.stopped.wait()
|
||||||
self.transport.close()
|
self.transport.close()
|
||||||
except Exception as e:
|
|
||||||
log.exception('failed to stop lbcd', exc_info=e)
|
|
||||||
raise
|
|
||||||
finally:
|
finally:
|
||||||
self.log.info("Done shutting down " + self.daemon_bin)
|
|
||||||
self.stopped = True
|
|
||||||
if cleanup:
|
if cleanup:
|
||||||
self.cleanup()
|
self.cleanup()
|
||||||
|
|
||||||
|
async def clear_mempool(self):
|
||||||
|
self.restart_ready.clear()
|
||||||
|
self.transport.terminate()
|
||||||
|
await self.protocol.stopped.wait()
|
||||||
|
self.transport.close()
|
||||||
self.running.clear()
|
self.running.clear()
|
||||||
|
os.remove(os.path.join(self.data_path, 'regtest', 'mempool.dat'))
|
||||||
|
self.restart_ready.set()
|
||||||
|
await self.running.wait()
|
||||||
|
|
||||||
def cleanup(self):
|
def cleanup(self):
|
||||||
assert self.stopped
|
|
||||||
shutil.rmtree(self.data_path, ignore_errors=True)
|
shutil.rmtree(self.data_path, ignore_errors=True)
|
||||||
|
|
||||||
|
async def _cli_cmnd(self, *args):
|
||||||
|
cmnd_args = [
|
||||||
|
self.cli_bin, f'-datadir={self.data_path}', '-regtest',
|
||||||
|
f'-rpcuser={self.rpcuser}', f'-rpcpassword={self.rpcpassword}', f'-rpcport={self.rpcport}'
|
||||||
|
] + list(args)
|
||||||
|
self.log.info(' '.join(cmnd_args))
|
||||||
|
loop = asyncio.get_event_loop()
|
||||||
|
asyncio.get_child_watcher().attach_loop(loop)
|
||||||
|
process = await asyncio.create_subprocess_exec(
|
||||||
|
*cmnd_args, stdout=subprocess.PIPE, stderr=subprocess.STDOUT
|
||||||
|
)
|
||||||
|
out, _ = await process.communicate()
|
||||||
|
result = out.decode().strip()
|
||||||
|
self.log.info(result)
|
||||||
|
if result.startswith('error code'):
|
||||||
|
raise Exception(result)
|
||||||
|
return result
|
||||||
|
|
||||||
class LBCWalletNode:
|
def generate(self, blocks):
|
||||||
P2SH_SEGWIT_ADDRESS = "p2sh-segwit"
|
self.block_expected += blocks
|
||||||
BECH32_ADDRESS = "bech32"
|
return self._cli_cmnd('generate', str(blocks))
|
||||||
|
|
||||||
|
def invalidate_block(self, blockhash):
|
||||||
|
return self._cli_cmnd('invalidateblock', blockhash)
|
||||||
|
|
||||||
|
def get_block_hash(self, block):
|
||||||
|
return self._cli_cmnd('getblockhash', str(block))
|
||||||
|
|
||||||
|
def sendrawtransaction(self, tx):
|
||||||
|
return self._cli_cmnd('sendrawtransaction', tx)
|
||||||
|
|
||||||
|
async def get_block(self, block_hash):
|
||||||
|
return json.loads(await self._cli_cmnd('getblock', block_hash, '1'))
|
||||||
|
|
||||||
|
def get_raw_change_address(self):
|
||||||
|
return self._cli_cmnd('getrawchangeaddress')
|
||||||
|
|
||||||
|
def get_new_address(self, address_type):
|
||||||
|
return self._cli_cmnd('getnewaddress', "", address_type)
|
||||||
|
|
||||||
|
async def get_balance(self):
|
||||||
|
return await self._cli_cmnd('getbalance')
|
||||||
|
|
||||||
|
def send_to_address(self, address, amount):
|
||||||
|
return self._cli_cmnd('sendtoaddress', address, str(amount))
|
||||||
|
|
||||||
|
def send_raw_transaction(self, tx):
|
||||||
|
return self._cli_cmnd('sendrawtransaction', tx.decode())
|
||||||
|
|
||||||
|
def create_raw_transaction(self, inputs, outputs):
|
||||||
|
return self._cli_cmnd('createrawtransaction', json.dumps(inputs), json.dumps(outputs))
|
||||||
|
|
||||||
|
async def sign_raw_transaction_with_wallet(self, tx):
|
||||||
|
return json.loads(await self._cli_cmnd('signrawtransactionwithwallet', tx))['hex'].encode()
|
||||||
|
|
||||||
|
def decode_raw_transaction(self, tx):
|
||||||
|
return self._cli_cmnd('decoderawtransaction', hexlify(tx.raw).decode())
|
||||||
|
|
||||||
|
def get_raw_transaction(self, txid):
|
||||||
|
return self._cli_cmnd('getrawtransaction', txid, '1')
|
||||||
|
|
||||||
|
|
||||||
|
class HubProcess(asyncio.SubprocessProtocol):
|
||||||
|
def __init__(self):
|
||||||
|
self.ready = asyncio.Event()
|
||||||
|
self.stopped = asyncio.Event()
|
||||||
|
self.log = log.getChild('hub')
|
||||||
|
|
||||||
|
def pipe_data_received(self, fd, data):
|
||||||
|
if self.log:
|
||||||
|
self.log.info(data.decode())
|
||||||
|
if b'error' in data.lower():
|
||||||
|
self.ready.set()
|
||||||
|
raise SystemError(data.decode())
|
||||||
|
if b'listening on' in data:
|
||||||
|
self.ready.set()
|
||||||
|
|
||||||
|
def process_exited(self):
|
||||||
|
self.stopped.set()
|
||||||
|
self.ready.set()
|
||||||
|
|
||||||
|
|
||||||
|
class HubNode:
|
||||||
|
|
||||||
|
def __init__(self, url, daemon, spv_node):
|
||||||
|
self.spv_node = spv_node
|
||||||
|
self.debug = False
|
||||||
|
|
||||||
def __init__(self, url, lbcwallet, cli):
|
|
||||||
self.latest_release_url = url
|
self.latest_release_url = url
|
||||||
self.project_dir = os.path.dirname(os.path.dirname(__file__))
|
self.project_dir = os.path.dirname(os.path.dirname(__file__))
|
||||||
self.bin_dir = os.path.join(self.project_dir, 'bin')
|
self.bin_dir = os.path.join(self.project_dir, 'bin')
|
||||||
self.lbcwallet_bin = os.path.join(self.bin_dir, lbcwallet)
|
self.daemon_bin = os.path.join(self.bin_dir, daemon)
|
||||||
self.cli_bin = os.path.join(self.bin_dir, cli)
|
self.cli_bin = os.path.join(self.bin_dir, daemon)
|
||||||
self.log = log.getChild('lbcwallet')
|
self.log = log.getChild('hub')
|
||||||
self.protocol = None
|
|
||||||
self.transport = None
|
self.transport = None
|
||||||
|
self.protocol = None
|
||||||
self.hostname = 'localhost'
|
self.hostname = 'localhost'
|
||||||
self.lbcd_rpcport = 29245
|
self.rpcport = 50051 # avoid conflict with default rpc port
|
||||||
self.lbcwallet_rpcport = 29244
|
self.stopped = False
|
||||||
self.rpcuser = 'rpcuser'
|
self.restart_ready = asyncio.Event()
|
||||||
self.rpcpassword = 'rpcpassword'
|
self.restart_ready.set()
|
||||||
self.data_path = tempfile.mkdtemp()
|
|
||||||
self.stopped = True
|
|
||||||
self.running = asyncio.Event()
|
self.running = asyncio.Event()
|
||||||
self.block_expected = 0
|
|
||||||
self.mining_addr = ''
|
|
||||||
|
|
||||||
@property
|
|
||||||
def rpc_url(self):
|
|
||||||
# FIXME: somehow the hub/sdk doesn't learn the blocks through the Walet RPC port, why?
|
|
||||||
# return f'http://{self.rpcuser}:{self.rpcpassword}@{self.hostname}:{self.lbcwallet_rpcport}/'
|
|
||||||
return f'http://{self.rpcuser}:{self.rpcpassword}@{self.hostname}:{self.lbcd_rpcport}/'
|
|
||||||
|
|
||||||
def is_expected_block(self, e: BlockHeightEvent):
|
|
||||||
return self.block_expected == e.height
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def exists(self):
|
def exists(self):
|
||||||
return (
|
return (
|
||||||
os.path.exists(self.lbcwallet_bin)
|
os.path.exists(self.cli_bin) and
|
||||||
|
os.path.exists(self.daemon_bin)
|
||||||
)
|
)
|
||||||
|
|
||||||
def download(self):
|
def download(self):
|
||||||
uname = platform.uname()
|
|
||||||
target_os = str.lower(uname.system)
|
|
||||||
target_arch = str.replace(uname.machine, 'x86_64', 'amd64')
|
|
||||||
target_platform = target_os + '_' + target_arch
|
|
||||||
self.latest_release_url = str.replace(self.latest_release_url, 'TARGET_PLATFORM', target_platform)
|
|
||||||
|
|
||||||
downloaded_file = os.path.join(
|
downloaded_file = os.path.join(
|
||||||
self.bin_dir,
|
self.bin_dir,
|
||||||
self.latest_release_url[self.latest_release_url.rfind('/')+1:]
|
self.latest_release_url[self.latest_release_url.rfind('/')+1:]
|
||||||
|
@ -536,12 +528,15 @@ class LBCWalletNode:
|
||||||
with zipfile.ZipFile(downloaded_file) as dotzip:
|
with zipfile.ZipFile(downloaded_file) as dotzip:
|
||||||
dotzip.extractall(self.bin_dir)
|
dotzip.extractall(self.bin_dir)
|
||||||
# zipfile bug https://bugs.python.org/issue15795
|
# zipfile bug https://bugs.python.org/issue15795
|
||||||
os.chmod(self.lbcwallet_bin, 0o755)
|
os.chmod(self.cli_bin, 0o755)
|
||||||
|
os.chmod(self.daemon_bin, 0o755)
|
||||||
|
|
||||||
elif downloaded_file.endswith('.tar.gz'):
|
elif downloaded_file.endswith('.tar.gz'):
|
||||||
with tarfile.open(downloaded_file) as tar:
|
with tarfile.open(downloaded_file) as tar:
|
||||||
tar.extractall(self.bin_dir)
|
tar.extractall(self.bin_dir)
|
||||||
|
|
||||||
|
os.chmod(self.daemon_bin, 0o755)
|
||||||
|
|
||||||
return self.exists
|
return self.exists
|
||||||
|
|
||||||
def ensure(self):
|
def ensure(self):
|
||||||
|
@ -551,125 +546,40 @@ class LBCWalletNode:
|
||||||
assert self.ensure()
|
assert self.ensure()
|
||||||
loop = asyncio.get_event_loop()
|
loop = asyncio.get_event_loop()
|
||||||
asyncio.get_child_watcher().attach_loop(loop)
|
asyncio.get_child_watcher().attach_loop(loop)
|
||||||
|
|
||||||
command = [
|
command = [
|
||||||
self.lbcwallet_bin,
|
self.daemon_bin, 'serve', '--esindex', self.spv_node.index_name + 'claims', '--debug'
|
||||||
'--noservertls', '--noclienttls',
|
|
||||||
'--regtest',
|
|
||||||
f'--rpcconnect=127.0.0.1:{self.lbcd_rpcport}', f'--rpclisten=127.0.0.1:{self.lbcwallet_rpcport}',
|
|
||||||
'--createtemp', f'--appdata={self.data_path}',
|
|
||||||
f'--username={self.rpcuser}', f'--password={self.rpcpassword}'
|
|
||||||
]
|
]
|
||||||
self.log.info(' '.join(command))
|
self.log.info(' '.join(command))
|
||||||
|
while not self.stopped:
|
||||||
|
if self.running.is_set():
|
||||||
|
await asyncio.sleep(1)
|
||||||
|
continue
|
||||||
|
await self.restart_ready.wait()
|
||||||
try:
|
try:
|
||||||
|
if not self.debug:
|
||||||
self.transport, self.protocol = await loop.subprocess_exec(
|
self.transport, self.protocol = await loop.subprocess_exec(
|
||||||
WalletProcess, *command
|
HubProcess, *command
|
||||||
)
|
)
|
||||||
self.protocol.transport = self.transport
|
|
||||||
await self.protocol.ready.wait()
|
await self.protocol.ready.wait()
|
||||||
assert not self.protocol.stopped.is_set()
|
assert not self.protocol.stopped.is_set()
|
||||||
self.running.set()
|
self.running.set()
|
||||||
self.stopped = False
|
|
||||||
except asyncio.CancelledError:
|
except asyncio.CancelledError:
|
||||||
self.running.clear()
|
self.running.clear()
|
||||||
raise
|
raise
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.running.clear()
|
self.running.clear()
|
||||||
log.exception('failed to start lbcwallet', exc_info=e)
|
log.exception('failed to start hub', exc_info=e)
|
||||||
|
|
||||||
def cleanup(self):
|
|
||||||
assert self.stopped
|
|
||||||
shutil.rmtree(self.data_path, ignore_errors=True)
|
|
||||||
|
|
||||||
async def stop(self, cleanup=True):
|
async def stop(self, cleanup=True):
|
||||||
if self.stopped:
|
self.stopped = True
|
||||||
return
|
|
||||||
try:
|
try:
|
||||||
|
if not self.debug:
|
||||||
self.transport.terminate()
|
self.transport.terminate()
|
||||||
await self.protocol.stopped.wait()
|
await self.protocol.stopped.wait()
|
||||||
self.transport.close()
|
self.transport.close()
|
||||||
except Exception as e:
|
|
||||||
log.exception('failed to stop lbcwallet', exc_info=e)
|
|
||||||
raise
|
|
||||||
finally:
|
finally:
|
||||||
self.log.info("Done shutting down " + self.lbcwallet_bin)
|
|
||||||
self.stopped = True
|
|
||||||
if cleanup:
|
if cleanup:
|
||||||
self.cleanup()
|
self.cleanup()
|
||||||
self.running.clear()
|
|
||||||
|
|
||||||
async def _cli_cmnd(self, *args):
|
def cleanup(self):
|
||||||
cmnd_args = [
|
pass
|
||||||
self.cli_bin,
|
|
||||||
f'--rpcuser={self.rpcuser}', f'--rpcpass={self.rpcpassword}', '--notls', '--regtest', '--wallet'
|
|
||||||
] + list(args)
|
|
||||||
self.log.info(' '.join(cmnd_args))
|
|
||||||
loop = asyncio.get_event_loop()
|
|
||||||
asyncio.get_child_watcher().attach_loop(loop)
|
|
||||||
process = await asyncio.create_subprocess_exec(
|
|
||||||
*cmnd_args, stdout=subprocess.PIPE, stderr=subprocess.PIPE
|
|
||||||
)
|
|
||||||
out, err = await process.communicate()
|
|
||||||
result = out.decode().strip()
|
|
||||||
err = err.decode().strip()
|
|
||||||
if len(result) <= 0 and err.startswith('-'):
|
|
||||||
raise Exception(err)
|
|
||||||
if err and 'creating a default config file' not in err:
|
|
||||||
log.warning(err)
|
|
||||||
self.log.info(result)
|
|
||||||
if result.startswith('error code'):
|
|
||||||
raise Exception(result)
|
|
||||||
return result
|
|
||||||
|
|
||||||
def generate(self, blocks):
|
|
||||||
self.block_expected += blocks
|
|
||||||
return self._cli_cmnd('generatetoaddress', str(blocks), self.mining_addr)
|
|
||||||
|
|
||||||
def generate_to_address(self, blocks, addr):
|
|
||||||
self.block_expected += blocks
|
|
||||||
return self._cli_cmnd('generatetoaddress', str(blocks), addr)
|
|
||||||
|
|
||||||
def wallet_passphrase(self, passphrase, timeout):
|
|
||||||
return self._cli_cmnd('walletpassphrase', passphrase, str(timeout))
|
|
||||||
|
|
||||||
def invalidate_block(self, blockhash):
|
|
||||||
return self._cli_cmnd('invalidateblock', blockhash)
|
|
||||||
|
|
||||||
def get_block_hash(self, block):
|
|
||||||
return self._cli_cmnd('getblockhash', str(block))
|
|
||||||
|
|
||||||
def sendrawtransaction(self, tx):
|
|
||||||
return self._cli_cmnd('sendrawtransaction', tx)
|
|
||||||
|
|
||||||
async def get_block(self, block_hash):
|
|
||||||
return json.loads(await self._cli_cmnd('getblock', block_hash, '1'))
|
|
||||||
|
|
||||||
def get_raw_change_address(self):
|
|
||||||
return self._cli_cmnd('getrawchangeaddress')
|
|
||||||
|
|
||||||
def get_new_address(self, address_type='legacy'):
|
|
||||||
return self._cli_cmnd('getnewaddress', "", address_type)
|
|
||||||
|
|
||||||
async def get_balance(self):
|
|
||||||
return await self._cli_cmnd('getbalance')
|
|
||||||
|
|
||||||
def send_to_address(self, address, amount):
|
|
||||||
return self._cli_cmnd('sendtoaddress', address, str(amount))
|
|
||||||
|
|
||||||
def send_raw_transaction(self, tx):
|
|
||||||
return self._cli_cmnd('sendrawtransaction', tx.decode())
|
|
||||||
|
|
||||||
def create_raw_transaction(self, inputs, outputs):
|
|
||||||
return self._cli_cmnd('createrawtransaction', json.dumps(inputs), json.dumps(outputs))
|
|
||||||
|
|
||||||
async def sign_raw_transaction_with_wallet(self, tx):
|
|
||||||
# the "withwallet" portion should only come into play if we are doing segwit.
|
|
||||||
# and "withwallet" doesn't exist on lbcd yet.
|
|
||||||
result = await self._cli_cmnd('signrawtransaction', tx)
|
|
||||||
return json.loads(result)['hex'].encode()
|
|
||||||
|
|
||||||
def decode_raw_transaction(self, tx):
|
|
||||||
return self._cli_cmnd('decoderawtransaction', hexlify(tx.raw).decode())
|
|
||||||
|
|
||||||
def get_raw_transaction(self, txid):
|
|
||||||
return self._cli_cmnd('getrawtransaction', txid, '1')
|
|
||||||
|
|
|
@ -61,10 +61,8 @@ class ConductorService:
|
||||||
#set_logging(
|
#set_logging(
|
||||||
# self.stack.ledger_module, logging.DEBUG, WebSocketLogHandler(self.send_message)
|
# self.stack.ledger_module, logging.DEBUG, WebSocketLogHandler(self.send_message)
|
||||||
#)
|
#)
|
||||||
self.stack.lbcd_started or await self.stack.start_lbcd()
|
self.stack.blockchain_started or await self.stack.start_blockchain()
|
||||||
self.send_message({'type': 'service', 'name': 'lbcd', 'port': self.stack.lbcd_node.port})
|
self.send_message({'type': 'service', 'name': 'blockchain', 'port': self.stack.blockchain_node.port})
|
||||||
self.stack.lbcwallet_started or await self.stack.start_lbcwallet()
|
|
||||||
self.send_message({'type': 'service', 'name': 'lbcwallet', 'port': self.stack.lbcwallet_node.port})
|
|
||||||
self.stack.spv_started or await self.stack.start_spv()
|
self.stack.spv_started or await self.stack.start_spv()
|
||||||
self.send_message({'type': 'service', 'name': 'spv', 'port': self.stack.spv_node.port})
|
self.send_message({'type': 'service', 'name': 'spv', 'port': self.stack.spv_node.port})
|
||||||
self.stack.wallet_started or await self.stack.start_wallet()
|
self.stack.wallet_started or await self.stack.start_wallet()
|
||||||
|
@ -76,7 +74,7 @@ class ConductorService:
|
||||||
async def generate(self, request):
|
async def generate(self, request):
|
||||||
data = await request.post()
|
data = await request.post()
|
||||||
blocks = data.get('blocks', 1)
|
blocks = data.get('blocks', 1)
|
||||||
await self.stack.lbcwallet_node.generate(int(blocks))
|
await self.stack.blockchain_node.generate(int(blocks))
|
||||||
return json_response({'blocks': blocks})
|
return json_response({'blocks': blocks})
|
||||||
|
|
||||||
async def transfer(self, request):
|
async def transfer(self, request):
|
||||||
|
@ -87,14 +85,11 @@ class ConductorService:
|
||||||
if not address:
|
if not address:
|
||||||
raise ValueError("No address was provided.")
|
raise ValueError("No address was provided.")
|
||||||
amount = data.get('amount', 1)
|
amount = data.get('amount', 1)
|
||||||
|
txid = await self.stack.blockchain_node.send_to_address(address, amount)
|
||||||
if self.stack.wallet_started:
|
if self.stack.wallet_started:
|
||||||
watcher = self.stack.wallet_node.ledger.on_transaction.where(
|
await self.stack.wallet_node.ledger.on_transaction.where(
|
||||||
lambda e: e.address == address # and e.tx.id == txid -- might stall; see send_to_address_and_wait
|
lambda e: e.tx.id == txid and e.address == address
|
||||||
)
|
)
|
||||||
txid = await self.stack.lbcwallet_node.send_to_address(address, amount)
|
|
||||||
await watcher
|
|
||||||
else:
|
|
||||||
txid = await self.stack.lbcwallet_node.send_to_address(address, amount)
|
|
||||||
return json_response({
|
return json_response({
|
||||||
'address': address,
|
'address': address,
|
||||||
'amount': amount,
|
'amount': amount,
|
||||||
|
@ -103,7 +98,7 @@ class ConductorService:
|
||||||
|
|
||||||
async def balance(self, _):
|
async def balance(self, _):
|
||||||
return json_response({
|
return json_response({
|
||||||
'balance': await self.stack.lbcwallet_node.get_balance()
|
'balance': await self.stack.blockchain_node.get_balance()
|
||||||
})
|
})
|
||||||
|
|
||||||
async def log(self, request):
|
async def log(self, request):
|
||||||
|
@ -134,7 +129,7 @@ class ConductorService:
|
||||||
'type': 'status',
|
'type': 'status',
|
||||||
'height': self.stack.wallet_node.ledger.headers.height,
|
'height': self.stack.wallet_node.ledger.headers.height,
|
||||||
'balance': satoshis_to_coins(await self.stack.wallet_node.account.get_balance()),
|
'balance': satoshis_to_coins(await self.stack.wallet_node.account.get_balance()),
|
||||||
'miner': await self.stack.lbcwallet_node.get_balance()
|
'miner': await self.stack.blockchain_node.get_balance()
|
||||||
})
|
})
|
||||||
|
|
||||||
def send_message(self, msg):
|
def send_message(self, msg):
|
||||||
|
|
|
@ -395,8 +395,8 @@ class RPCSession(SessionBase):
|
||||||
namespace=NAMESPACE, labelnames=("version",)
|
namespace=NAMESPACE, labelnames=("version",)
|
||||||
)
|
)
|
||||||
|
|
||||||
def __init__(self, *, framer=None, connection=None):
|
def __init__(self, *, framer=None, loop=None, connection=None):
|
||||||
super().__init__(framer=framer)
|
super().__init__(framer=framer, loop=loop)
|
||||||
self.connection = connection or self.default_connection()
|
self.connection = connection or self.default_connection()
|
||||||
self.client_version = 'unknown'
|
self.client_version = 'unknown'
|
||||||
|
|
||||||
|
@ -424,10 +424,11 @@ class RPCSession(SessionBase):
|
||||||
self.max_errors = 0
|
self.max_errors = 0
|
||||||
self._bump_errors()
|
self._bump_errors()
|
||||||
else:
|
else:
|
||||||
for request in requests:
|
self.schedule_requests(requests)
|
||||||
await self._task_group.add(self._handle_request(request))
|
|
||||||
|
|
||||||
async def _handle_request(self, request):
|
async def _handle_request(self, request):
|
||||||
|
if self.is_closing():
|
||||||
|
return
|
||||||
start = time.perf_counter()
|
start = time.perf_counter()
|
||||||
try:
|
try:
|
||||||
result = await self.handle_request(request)
|
result = await self.handle_request(request)
|
||||||
|
@ -472,6 +473,10 @@ class RPCSession(SessionBase):
|
||||||
async def handle_request(self, request):
|
async def handle_request(self, request):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
def schedule_requests(self, requests):
|
||||||
|
for request in requests:
|
||||||
|
self._task_group.add(self._handle_request(request))
|
||||||
|
|
||||||
async def send_request(self, method, args=()):
|
async def send_request(self, method, args=()):
|
||||||
"""Send an RPC request over the network."""
|
"""Send an RPC request over the network."""
|
||||||
if self.is_closing():
|
if self.is_closing():
|
||||||
|
|
|
@ -17,7 +17,6 @@ OP_HASH160 = 0xa9
|
||||||
OP_EQUALVERIFY = 0x88
|
OP_EQUALVERIFY = 0x88
|
||||||
OP_CHECKSIG = 0xac
|
OP_CHECKSIG = 0xac
|
||||||
OP_CHECKMULTISIG = 0xae
|
OP_CHECKMULTISIG = 0xae
|
||||||
OP_CHECKLOCKTIMEVERIFY = 0xb1
|
|
||||||
OP_EQUAL = 0x87
|
OP_EQUAL = 0x87
|
||||||
OP_PUSHDATA1 = 0x4c
|
OP_PUSHDATA1 = 0x4c
|
||||||
OP_PUSHDATA2 = 0x4d
|
OP_PUSHDATA2 = 0x4d
|
||||||
|
@ -277,7 +276,7 @@ class Template:
|
||||||
elif isinstance(opcode, PUSH_INTEGER):
|
elif isinstance(opcode, PUSH_INTEGER):
|
||||||
data = values[opcode.name]
|
data = values[opcode.name]
|
||||||
source.write_many(push_data(
|
source.write_many(push_data(
|
||||||
data.to_bytes((data.bit_length() + 8) // 8, byteorder='little', signed=True)
|
data.to_bytes((data.bit_length() + 7) // 8, byteorder='little')
|
||||||
))
|
))
|
||||||
elif isinstance(opcode, PUSH_SUBSCRIPT):
|
elif isinstance(opcode, PUSH_SUBSCRIPT):
|
||||||
data = values[opcode.name]
|
data = values[opcode.name]
|
||||||
|
@ -358,27 +357,19 @@ class InputScript(Script):
|
||||||
REDEEM_PUBKEY_HASH = Template('pubkey_hash', (
|
REDEEM_PUBKEY_HASH = Template('pubkey_hash', (
|
||||||
PUSH_SINGLE('signature'), PUSH_SINGLE('pubkey')
|
PUSH_SINGLE('signature'), PUSH_SINGLE('pubkey')
|
||||||
))
|
))
|
||||||
MULTI_SIG_SCRIPT = Template('multi_sig', (
|
REDEEM_SCRIPT = Template('script', (
|
||||||
SMALL_INTEGER('signatures_count'), PUSH_MANY('pubkeys'), SMALL_INTEGER('pubkeys_count'),
|
SMALL_INTEGER('signatures_count'), PUSH_MANY('pubkeys'), SMALL_INTEGER('pubkeys_count'),
|
||||||
OP_CHECKMULTISIG
|
OP_CHECKMULTISIG
|
||||||
))
|
))
|
||||||
REDEEM_SCRIPT_HASH_MULTI_SIG = Template('script_hash+multi_sig', (
|
REDEEM_SCRIPT_HASH = Template('script_hash', (
|
||||||
OP_0, PUSH_MANY('signatures'), PUSH_SUBSCRIPT('script', MULTI_SIG_SCRIPT)
|
OP_0, PUSH_MANY('signatures'), PUSH_SUBSCRIPT('script', REDEEM_SCRIPT)
|
||||||
))
|
|
||||||
TIME_LOCK_SCRIPT = Template('timelock', (
|
|
||||||
PUSH_INTEGER('height'), OP_CHECKLOCKTIMEVERIFY, OP_DROP,
|
|
||||||
# rest is identical to OutputScript.PAY_PUBKEY_HASH:
|
|
||||||
OP_DUP, OP_HASH160, PUSH_SINGLE('pubkey_hash'), OP_EQUALVERIFY, OP_CHECKSIG
|
|
||||||
))
|
|
||||||
REDEEM_SCRIPT_HASH_TIME_LOCK = Template('script_hash+timelock', (
|
|
||||||
PUSH_SINGLE('signature'), PUSH_SINGLE('pubkey'), PUSH_SUBSCRIPT('script', TIME_LOCK_SCRIPT)
|
|
||||||
))
|
))
|
||||||
|
|
||||||
templates = [
|
templates = [
|
||||||
REDEEM_PUBKEY,
|
REDEEM_PUBKEY,
|
||||||
REDEEM_PUBKEY_HASH,
|
REDEEM_PUBKEY_HASH,
|
||||||
REDEEM_SCRIPT_HASH_TIME_LOCK,
|
REDEEM_SCRIPT_HASH,
|
||||||
REDEEM_SCRIPT_HASH_MULTI_SIG,
|
REDEEM_SCRIPT
|
||||||
]
|
]
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
|
@ -389,37 +380,19 @@ class InputScript(Script):
|
||||||
})
|
})
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def redeem_multi_sig_script_hash(cls, signatures, pubkeys):
|
def redeem_script_hash(cls, signatures, pubkeys):
|
||||||
return cls(template=cls.REDEEM_SCRIPT_HASH_MULTI_SIG, values={
|
return cls(template=cls.REDEEM_SCRIPT_HASH, values={
|
||||||
'signatures': signatures,
|
'signatures': signatures,
|
||||||
'script': cls(template=cls.MULTI_SIG_SCRIPT, values={
|
'script': cls.redeem_script(signatures, pubkeys)
|
||||||
|
})
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def redeem_script(cls, signatures, pubkeys):
|
||||||
|
return cls(template=cls.REDEEM_SCRIPT, values={
|
||||||
'signatures_count': len(signatures),
|
'signatures_count': len(signatures),
|
||||||
'pubkeys': pubkeys,
|
'pubkeys': pubkeys,
|
||||||
'pubkeys_count': len(pubkeys)
|
'pubkeys_count': len(pubkeys)
|
||||||
})
|
})
|
||||||
})
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def redeem_time_lock_script_hash(cls, signature, pubkey, height=None, pubkey_hash=None, script_source=None):
|
|
||||||
if height and pubkey_hash:
|
|
||||||
script = cls(template=cls.TIME_LOCK_SCRIPT, values={
|
|
||||||
'height': height,
|
|
||||||
'pubkey_hash': pubkey_hash
|
|
||||||
})
|
|
||||||
elif script_source:
|
|
||||||
script = cls(source=script_source, template=cls.TIME_LOCK_SCRIPT)
|
|
||||||
script.parse(script.template)
|
|
||||||
else:
|
|
||||||
raise ValueError("script_source or both height and pubkey_hash are required.")
|
|
||||||
return cls(template=cls.REDEEM_SCRIPT_HASH_TIME_LOCK, values={
|
|
||||||
'signature': signature,
|
|
||||||
'pubkey': pubkey,
|
|
||||||
'script': script
|
|
||||||
})
|
|
||||||
|
|
||||||
@property
|
|
||||||
def is_script_hash(self):
|
|
||||||
return self.template.name.startswith('script_hash+')
|
|
||||||
|
|
||||||
|
|
||||||
class OutputScript(Script):
|
class OutputScript(Script):
|
||||||
|
@ -487,6 +460,21 @@ class OutputScript(Script):
|
||||||
UPDATE_CLAIM_OPCODES + PAY_SCRIPT_HASH.opcodes
|
UPDATE_CLAIM_OPCODES + PAY_SCRIPT_HASH.opcodes
|
||||||
))
|
))
|
||||||
|
|
||||||
|
SELL_SCRIPT = Template('sell_script', (
|
||||||
|
OP_VERIFY, OP_DROP, OP_DROP, OP_DROP, PUSH_INTEGER('price'), OP_PRICECHECK
|
||||||
|
))
|
||||||
|
SELL_CLAIM = Template('sell_claim+pay_script_hash', (
|
||||||
|
OP_SELL_CLAIM, PUSH_SINGLE('claim_id'), PUSH_SUBSCRIPT('sell_script', SELL_SCRIPT),
|
||||||
|
PUSH_SUBSCRIPT('receive_script', InputScript.REDEEM_SCRIPT), OP_2DROP, OP_2DROP
|
||||||
|
) + PAY_SCRIPT_HASH.opcodes)
|
||||||
|
|
||||||
|
BUY_CLAIM = Template('buy_claim+pay_script_hash', (
|
||||||
|
OP_BUY_CLAIM, PUSH_SINGLE('sell_id'),
|
||||||
|
PUSH_SINGLE('claim_id'), PUSH_SINGLE('claim_version'),
|
||||||
|
PUSH_SINGLE('owner_pubkey_hash'), PUSH_SINGLE('negotiation_signature'),
|
||||||
|
OP_2DROP, OP_2DROP, OP_2DROP,
|
||||||
|
) + PAY_SCRIPT_HASH.opcodes)
|
||||||
|
|
||||||
templates = [
|
templates = [
|
||||||
PAY_PUBKEY_FULL,
|
PAY_PUBKEY_FULL,
|
||||||
PAY_PUBKEY_HASH,
|
PAY_PUBKEY_HASH,
|
||||||
|
@ -501,6 +489,8 @@ class OutputScript(Script):
|
||||||
SUPPORT_CLAIM_DATA_SCRIPT,
|
SUPPORT_CLAIM_DATA_SCRIPT,
|
||||||
UPDATE_CLAIM_PUBKEY,
|
UPDATE_CLAIM_PUBKEY,
|
||||||
UPDATE_CLAIM_SCRIPT,
|
UPDATE_CLAIM_SCRIPT,
|
||||||
|
SELL_CLAIM, SELL_SCRIPT,
|
||||||
|
BUY_CLAIM,
|
||||||
]
|
]
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
|
@ -560,6 +550,30 @@ class OutputScript(Script):
|
||||||
'pubkey_hash': pubkey_hash
|
'pubkey_hash': pubkey_hash
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def sell_script(cls, price):
|
||||||
|
return cls(template=cls.SELL_SCRIPT, values={
|
||||||
|
'price': price,
|
||||||
|
})
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def sell_claim(cls, claim_id, price, signatures, pubkeys):
|
||||||
|
return cls(template=cls.SELL_CLAIM, values={
|
||||||
|
'claim_id': claim_id,
|
||||||
|
'sell_script': OutputScript.sell_script(price),
|
||||||
|
'receive_script': InputScript.redeem_script(signatures, pubkeys)
|
||||||
|
})
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def buy_claim(cls, sell_id, claim_id, claim_version, owner_pubkey_hash, negotiation_signature):
|
||||||
|
return cls(template=cls.BUY_CLAIM, values={
|
||||||
|
'sell_id': sell_id,
|
||||||
|
'claim_id': claim_id,
|
||||||
|
'claim_version': claim_version,
|
||||||
|
'owner_pubkey_hash': owner_pubkey_hash,
|
||||||
|
'negotiation_signature': negotiation_signature,
|
||||||
|
})
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def is_pay_pubkey_hash(self):
|
def is_pay_pubkey_hash(self):
|
||||||
return self.template.name.endswith('pay_pubkey_hash')
|
return self.template.name.endswith('pay_pubkey_hash')
|
||||||
|
@ -588,6 +602,17 @@ class OutputScript(Script):
|
||||||
def is_support_claim_data(self):
|
def is_support_claim_data(self):
|
||||||
return self.template.name.startswith('support_claim+data+')
|
return self.template.name.startswith('support_claim+data+')
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_sell_claim(self):
|
||||||
|
return self.template.name.startswith('sell_claim+')
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_buy_claim(self):
|
||||||
|
return self.template.name.startswith('buy_claim+')
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def is_claim_involved(self):
|
def is_claim_involved(self):
|
||||||
return any((self.is_claim_name, self.is_support_claim, self.is_update_claim))
|
return any((
|
||||||
|
self.is_claim_name, self.is_support_claim, self.is_update_claim,
|
||||||
|
self.is_sell_claim, self.is_buy_claim
|
||||||
|
))
|
||||||
|
|
1760
lbry/wallet/server/block_processor.py
Normal file
1760
lbry/wallet/server/block_processor.py
Normal file
File diff suppressed because it is too large
Load diff
43
lbry/wallet/server/cli.py
Normal file
43
lbry/wallet/server/cli.py
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
import logging
|
||||||
|
import traceback
|
||||||
|
import argparse
|
||||||
|
import importlib
|
||||||
|
from lbry.wallet.server.env import Env
|
||||||
|
from lbry.wallet.server.server import Server
|
||||||
|
|
||||||
|
|
||||||
|
def get_argument_parser():
|
||||||
|
parser = argparse.ArgumentParser(
|
||||||
|
prog="lbry-hub"
|
||||||
|
)
|
||||||
|
parser.add_argument("spvserver", type=str, help="Python class path to SPV server implementation.",
|
||||||
|
nargs="?", default="lbry.wallet.server.coin.LBC")
|
||||||
|
return parser
|
||||||
|
|
||||||
|
|
||||||
|
def get_coin_class(spvserver):
|
||||||
|
spvserver_path, coin_class_name = spvserver.rsplit('.', 1)
|
||||||
|
spvserver_module = importlib.import_module(spvserver_path)
|
||||||
|
return getattr(spvserver_module, coin_class_name)
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
parser = get_argument_parser()
|
||||||
|
args = parser.parse_args()
|
||||||
|
coin_class = get_coin_class(args.spvserver)
|
||||||
|
logging.basicConfig(level=logging.INFO, format="%(asctime)s %(levelname)-4s %(name)s:%(lineno)d: %(message)s")
|
||||||
|
logging.info('lbry.server starting')
|
||||||
|
logging.getLogger('aiohttp').setLevel(logging.WARNING)
|
||||||
|
logging.getLogger('elasticsearch').setLevel(logging.WARNING)
|
||||||
|
try:
|
||||||
|
server = Server(Env(coin_class))
|
||||||
|
server.run()
|
||||||
|
except Exception:
|
||||||
|
traceback.print_exc()
|
||||||
|
logging.critical('lbry.server terminated abnormally')
|
||||||
|
else:
|
||||||
|
logging.info('lbry.server terminated normally')
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
386
lbry/wallet/server/coin.py
Normal file
386
lbry/wallet/server/coin.py
Normal file
|
@ -0,0 +1,386 @@
|
||||||
|
import re
|
||||||
|
import struct
|
||||||
|
from typing import List
|
||||||
|
from hashlib import sha256
|
||||||
|
from decimal import Decimal
|
||||||
|
from collections import namedtuple
|
||||||
|
|
||||||
|
import lbry.wallet.server.tx as lib_tx
|
||||||
|
from lbry.wallet.script import OutputScript, OP_CLAIM_NAME, OP_UPDATE_CLAIM, OP_SUPPORT_CLAIM
|
||||||
|
from lbry.wallet.server.tx import DeserializerSegWit
|
||||||
|
from lbry.wallet.server.util import cachedproperty, subclasses
|
||||||
|
from lbry.wallet.server.hash import Base58, hash160, double_sha256, hash_to_hex_str, HASHX_LEN
|
||||||
|
from lbry.wallet.server.daemon import Daemon, LBCDaemon
|
||||||
|
from lbry.wallet.server.script import ScriptPubKey, OpCodes
|
||||||
|
from lbry.wallet.server.leveldb import LevelDB
|
||||||
|
from lbry.wallet.server.session import LBRYElectrumX, LBRYSessionManager
|
||||||
|
from lbry.wallet.server.block_processor import BlockProcessor
|
||||||
|
|
||||||
|
|
||||||
|
Block = namedtuple("Block", "raw header transactions")
|
||||||
|
OP_RETURN = OpCodes.OP_RETURN
|
||||||
|
|
||||||
|
|
||||||
|
class CoinError(Exception):
|
||||||
|
"""Exception raised for coin-related errors."""
|
||||||
|
|
||||||
|
|
||||||
|
class Coin:
|
||||||
|
"""Base class of coin hierarchy."""
|
||||||
|
|
||||||
|
REORG_LIMIT = 200
|
||||||
|
# Not sure if these are coin-specific
|
||||||
|
RPC_URL_REGEX = re.compile('.+@(\\[[0-9a-fA-F:]+\\]|[^:]+)(:[0-9]+)?')
|
||||||
|
VALUE_PER_COIN = 100000000
|
||||||
|
CHUNK_SIZE = 2016
|
||||||
|
BASIC_HEADER_SIZE = 80
|
||||||
|
STATIC_BLOCK_HEADERS = True
|
||||||
|
SESSIONCLS = LBRYElectrumX
|
||||||
|
DESERIALIZER = lib_tx.Deserializer
|
||||||
|
DAEMON = Daemon
|
||||||
|
BLOCK_PROCESSOR = BlockProcessor
|
||||||
|
SESSION_MANAGER = LBRYSessionManager
|
||||||
|
DB = LevelDB
|
||||||
|
HEADER_VALUES = [
|
||||||
|
'version', 'prev_block_hash', 'merkle_root', 'timestamp', 'bits', 'nonce'
|
||||||
|
]
|
||||||
|
HEADER_UNPACK = struct.Struct('< I 32s 32s I I I').unpack_from
|
||||||
|
MEMPOOL_HISTOGRAM_REFRESH_SECS = 500
|
||||||
|
XPUB_VERBYTES = bytes('????', 'utf-8')
|
||||||
|
XPRV_VERBYTES = bytes('????', 'utf-8')
|
||||||
|
ENCODE_CHECK = Base58.encode_check
|
||||||
|
DECODE_CHECK = Base58.decode_check
|
||||||
|
# Peer discovery
|
||||||
|
PEER_DEFAULT_PORTS = {'t': '50001', 's': '50002'}
|
||||||
|
PEERS: List[str] = []
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def lookup_coin_class(cls, name, net):
|
||||||
|
"""Return a coin class given name and network.
|
||||||
|
|
||||||
|
Raise an exception if unrecognised."""
|
||||||
|
req_attrs = ['TX_COUNT', 'TX_COUNT_HEIGHT', 'TX_PER_BLOCK']
|
||||||
|
for coin in subclasses(Coin):
|
||||||
|
if (coin.NAME.lower() == name.lower() and
|
||||||
|
coin.NET.lower() == net.lower()):
|
||||||
|
coin_req_attrs = req_attrs.copy()
|
||||||
|
missing = [attr for attr in coin_req_attrs
|
||||||
|
if not hasattr(coin, attr)]
|
||||||
|
if missing:
|
||||||
|
raise CoinError(f'coin {name} missing {missing} attributes')
|
||||||
|
return coin
|
||||||
|
raise CoinError(f'unknown coin {name} and network {net} combination')
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def sanitize_url(cls, url):
|
||||||
|
# Remove surrounding ws and trailing /s
|
||||||
|
url = url.strip().rstrip('/')
|
||||||
|
match = cls.RPC_URL_REGEX.match(url)
|
||||||
|
if not match:
|
||||||
|
raise CoinError(f'invalid daemon URL: "{url}"')
|
||||||
|
if match.groups()[1] is None:
|
||||||
|
url += f':{cls.RPC_PORT:d}'
|
||||||
|
if not url.startswith('http://') and not url.startswith('https://'):
|
||||||
|
url = 'http://' + url
|
||||||
|
return url + '/'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def genesis_block(cls, block):
|
||||||
|
"""Check the Genesis block is the right one for this coin.
|
||||||
|
|
||||||
|
Return the block less its unspendable coinbase.
|
||||||
|
"""
|
||||||
|
header = cls.block_header(block, 0)
|
||||||
|
header_hex_hash = hash_to_hex_str(cls.header_hash(header))
|
||||||
|
if header_hex_hash != cls.GENESIS_HASH:
|
||||||
|
raise CoinError(f'genesis block has hash {header_hex_hash} expected {cls.GENESIS_HASH}')
|
||||||
|
|
||||||
|
return header + bytes(1)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def hashX_from_script(cls, script):
|
||||||
|
"""Returns a hashX from a script, or None if the script is provably
|
||||||
|
unspendable so the output can be dropped.
|
||||||
|
"""
|
||||||
|
if script and script[0] == OP_RETURN:
|
||||||
|
return None
|
||||||
|
return sha256(script).digest()[:HASHX_LEN]
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def lookup_xverbytes(verbytes):
|
||||||
|
"""Return a (is_xpub, coin_class) pair given xpub/xprv verbytes."""
|
||||||
|
# Order means BTC testnet will override NMC testnet
|
||||||
|
for coin in subclasses(Coin):
|
||||||
|
if verbytes == coin.XPUB_VERBYTES:
|
||||||
|
return True, coin
|
||||||
|
if verbytes == coin.XPRV_VERBYTES:
|
||||||
|
return False, coin
|
||||||
|
raise CoinError('version bytes unrecognised')
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def address_to_hashX(cls, address):
|
||||||
|
"""Return a hashX given a coin address."""
|
||||||
|
return cls.hashX_from_script(cls.pay_to_address_script(address))
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def P2PKH_address_from_hash160(cls, hash160):
|
||||||
|
"""Return a P2PKH address given a public key."""
|
||||||
|
assert len(hash160) == 20
|
||||||
|
return cls.ENCODE_CHECK(cls.P2PKH_VERBYTE + hash160)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def P2PKH_address_from_pubkey(cls, pubkey):
|
||||||
|
"""Return a coin address given a public key."""
|
||||||
|
return cls.P2PKH_address_from_hash160(hash160(pubkey))
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def P2SH_address_from_hash160(cls, hash160):
|
||||||
|
"""Return a coin address given a hash160."""
|
||||||
|
assert len(hash160) == 20
|
||||||
|
return cls.ENCODE_CHECK(cls.P2SH_VERBYTES[0] + hash160)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def hash160_to_P2PKH_script(cls, hash160):
|
||||||
|
return ScriptPubKey.P2PKH_script(hash160)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def hash160_to_P2PKH_hashX(cls, hash160):
|
||||||
|
return cls.hashX_from_script(cls.hash160_to_P2PKH_script(hash160))
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def pay_to_address_script(cls, address):
|
||||||
|
"""Return a pubkey script that pays to a pubkey hash.
|
||||||
|
|
||||||
|
Pass the address (either P2PKH or P2SH) in base58 form.
|
||||||
|
"""
|
||||||
|
raw = cls.DECODE_CHECK(address)
|
||||||
|
|
||||||
|
# Require version byte(s) plus hash160.
|
||||||
|
verbyte = -1
|
||||||
|
verlen = len(raw) - 20
|
||||||
|
if verlen > 0:
|
||||||
|
verbyte, hash160 = raw[:verlen], raw[verlen:]
|
||||||
|
|
||||||
|
if verbyte == cls.P2PKH_VERBYTE:
|
||||||
|
return cls.hash160_to_P2PKH_script(hash160)
|
||||||
|
if verbyte in cls.P2SH_VERBYTES:
|
||||||
|
return ScriptPubKey.P2SH_script(hash160)
|
||||||
|
|
||||||
|
raise CoinError(f'invalid address: {address}')
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def privkey_WIF(cls, privkey_bytes, compressed):
|
||||||
|
"""Return the private key encoded in Wallet Import Format."""
|
||||||
|
payload = bytearray(cls.WIF_BYTE) + privkey_bytes
|
||||||
|
if compressed:
|
||||||
|
payload.append(0x01)
|
||||||
|
return cls.ENCODE_CHECK(payload)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def header_hash(cls, header):
|
||||||
|
"""Given a header return hash"""
|
||||||
|
return double_sha256(header)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def header_prevhash(cls, header):
|
||||||
|
"""Given a header return previous hash"""
|
||||||
|
return header[4:36]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def static_header_offset(cls, height):
|
||||||
|
"""Given a header height return its offset in the headers file.
|
||||||
|
|
||||||
|
If header sizes change at some point, this is the only code
|
||||||
|
that needs updating."""
|
||||||
|
assert cls.STATIC_BLOCK_HEADERS
|
||||||
|
return height * cls.BASIC_HEADER_SIZE
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def static_header_len(cls, height):
|
||||||
|
"""Given a header height return its length."""
|
||||||
|
return (cls.static_header_offset(height + 1)
|
||||||
|
- cls.static_header_offset(height))
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def block_header(cls, block, height):
|
||||||
|
"""Returns the block header given a block and its height."""
|
||||||
|
return block[:cls.static_header_len(height)]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def block(cls, raw_block, height):
|
||||||
|
"""Return a Block namedtuple given a raw block and its height."""
|
||||||
|
header = cls.block_header(raw_block, height)
|
||||||
|
txs = cls.DESERIALIZER(raw_block, start=len(header)).read_tx_block()
|
||||||
|
return Block(raw_block, header, txs)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def transaction(cls, raw_tx: bytes):
|
||||||
|
"""Return a Block namedtuple given a raw block and its height."""
|
||||||
|
return cls.DESERIALIZER(raw_tx).read_tx()
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def decimal_value(cls, value):
|
||||||
|
"""Return the number of standard coin units as a Decimal given a
|
||||||
|
quantity of smallest units.
|
||||||
|
|
||||||
|
For example 1 BTC is returned for 100 million satoshis.
|
||||||
|
"""
|
||||||
|
return Decimal(value) / cls.VALUE_PER_COIN
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def electrum_header(cls, header, height):
|
||||||
|
h = dict(zip(cls.HEADER_VALUES, cls.HEADER_UNPACK(header)))
|
||||||
|
# Add the height that is not present in the header itself
|
||||||
|
h['block_height'] = height
|
||||||
|
# Convert bytes to str
|
||||||
|
h['prev_block_hash'] = hash_to_hex_str(h['prev_block_hash'])
|
||||||
|
h['merkle_root'] = hash_to_hex_str(h['merkle_root'])
|
||||||
|
return h
|
||||||
|
|
||||||
|
|
||||||
|
class LBC(Coin):
|
||||||
|
DAEMON = LBCDaemon
|
||||||
|
SESSIONCLS = LBRYElectrumX
|
||||||
|
SESSION_MANAGER = LBRYSessionManager
|
||||||
|
DESERIALIZER = DeserializerSegWit
|
||||||
|
DB = LevelDB
|
||||||
|
NAME = "LBRY"
|
||||||
|
SHORTNAME = "LBC"
|
||||||
|
NET = "mainnet"
|
||||||
|
BASIC_HEADER_SIZE = 112
|
||||||
|
CHUNK_SIZE = 96
|
||||||
|
XPUB_VERBYTES = bytes.fromhex("0488b21e")
|
||||||
|
XPRV_VERBYTES = bytes.fromhex("0488ade4")
|
||||||
|
P2PKH_VERBYTE = bytes.fromhex("55")
|
||||||
|
P2SH_VERBYTES = bytes.fromhex("7A")
|
||||||
|
WIF_BYTE = bytes.fromhex("1C")
|
||||||
|
GENESIS_HASH = ('9c89283ba0f3227f6c03b70216b9f665'
|
||||||
|
'f0118d5e0fa729cedf4fb34d6a34f463')
|
||||||
|
TX_COUNT = 2716936
|
||||||
|
TX_COUNT_HEIGHT = 329554
|
||||||
|
TX_PER_BLOCK = 1
|
||||||
|
RPC_PORT = 9245
|
||||||
|
REORG_LIMIT = 200
|
||||||
|
|
||||||
|
nOriginalClaimExpirationTime = 262974
|
||||||
|
nExtendedClaimExpirationTime = 2102400
|
||||||
|
nExtendedClaimExpirationForkHeight = 400155
|
||||||
|
nNormalizedNameForkHeight = 539940 # targeting 21 March 2019
|
||||||
|
nMinTakeoverWorkaroundHeight = 496850
|
||||||
|
nMaxTakeoverWorkaroundHeight = 658300 # targeting 30 Oct 2019
|
||||||
|
nWitnessForkHeight = 680770 # targeting 11 Dec 2019
|
||||||
|
nAllClaimsInMerkleForkHeight = 658310 # targeting 30 Oct 2019
|
||||||
|
proportionalDelayFactor = 32
|
||||||
|
maxTakeoverDelay = 4032
|
||||||
|
|
||||||
|
PEERS = [
|
||||||
|
]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def genesis_block(cls, block):
|
||||||
|
'''Check the Genesis block is the right one for this coin.
|
||||||
|
|
||||||
|
Return the block less its unspendable coinbase.
|
||||||
|
'''
|
||||||
|
header = cls.block_header(block, 0)
|
||||||
|
header_hex_hash = hash_to_hex_str(cls.header_hash(header))
|
||||||
|
if header_hex_hash != cls.GENESIS_HASH:
|
||||||
|
raise CoinError(f'genesis block has hash {header_hex_hash} expected {cls.GENESIS_HASH}')
|
||||||
|
|
||||||
|
return block
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def electrum_header(cls, header, height):
|
||||||
|
version, = struct.unpack('<I', header[:4])
|
||||||
|
timestamp, bits, nonce = struct.unpack('<III', header[100:112])
|
||||||
|
return {
|
||||||
|
'version': version,
|
||||||
|
'prev_block_hash': hash_to_hex_str(header[4:36]),
|
||||||
|
'merkle_root': hash_to_hex_str(header[36:68]),
|
||||||
|
'claim_trie_root': hash_to_hex_str(header[68:100]),
|
||||||
|
'timestamp': timestamp,
|
||||||
|
'bits': bits,
|
||||||
|
'nonce': nonce,
|
||||||
|
'block_height': height,
|
||||||
|
}
|
||||||
|
|
||||||
|
@cachedproperty
|
||||||
|
def address_handlers(self):
|
||||||
|
return ScriptPubKey.PayToHandlers(
|
||||||
|
address=self.P2PKH_address_from_hash160,
|
||||||
|
script_hash=self.P2SH_address_from_hash160,
|
||||||
|
pubkey=self.P2PKH_address_from_pubkey,
|
||||||
|
unspendable=lambda: None,
|
||||||
|
strange=self.claim_address_handler,
|
||||||
|
)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def address_from_script(cls, script):
|
||||||
|
'''Given a pk_script, return the address it pays to, or None.'''
|
||||||
|
return ScriptPubKey.pay_to(cls.address_handlers, script)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def claim_address_handler(cls, script):
|
||||||
|
'''Parse a claim script, returns the address
|
||||||
|
'''
|
||||||
|
output = OutputScript(script)
|
||||||
|
if output.is_pay_pubkey_hash:
|
||||||
|
return cls.P2PKH_address_from_hash160(output.values['pubkey_hash'])
|
||||||
|
if output.is_pay_script_hash:
|
||||||
|
return cls.P2SH_address_from_hash160(output.values['script_hash'])
|
||||||
|
if output.is_pay_pubkey:
|
||||||
|
return cls.P2PKH_address_from_pubkey(output.values['pubkey'])
|
||||||
|
if output.is_return_data:
|
||||||
|
return None
|
||||||
|
return None
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def hashX_from_script(cls, script):
|
||||||
|
'''
|
||||||
|
Overrides electrumx hashX from script by extracting addresses from claim scripts.
|
||||||
|
'''
|
||||||
|
if script and script[0] == OpCodes.OP_RETURN or not script:
|
||||||
|
return None
|
||||||
|
if script[0] in [
|
||||||
|
OP_CLAIM_NAME,
|
||||||
|
OP_UPDATE_CLAIM,
|
||||||
|
OP_SUPPORT_CLAIM,
|
||||||
|
]:
|
||||||
|
return cls.address_to_hashX(cls.claim_address_handler(script))
|
||||||
|
else:
|
||||||
|
return sha256(script).digest()[:HASHX_LEN]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def get_expiration_height(cls, last_updated_height: int, extended: bool = False) -> int:
|
||||||
|
if extended:
|
||||||
|
return last_updated_height + cls.nExtendedClaimExpirationTime
|
||||||
|
if last_updated_height < cls.nExtendedClaimExpirationForkHeight:
|
||||||
|
return last_updated_height + cls.nOriginalClaimExpirationTime
|
||||||
|
return last_updated_height + cls.nExtendedClaimExpirationTime
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def get_delay_for_name(cls, blocks_of_continuous_ownership: int) -> int:
|
||||||
|
return min(blocks_of_continuous_ownership // cls.proportionalDelayFactor, cls.maxTakeoverDelay)
|
||||||
|
|
||||||
|
|
||||||
|
class LBCRegTest(LBC):
|
||||||
|
NET = "regtest"
|
||||||
|
GENESIS_HASH = '6e3fcf1299d4ec5d79c3a4c91d624a4acf9e2e173d95a1a0504f677669687556'
|
||||||
|
XPUB_VERBYTES = bytes.fromhex('043587cf')
|
||||||
|
XPRV_VERBYTES = bytes.fromhex('04358394')
|
||||||
|
P2PKH_VERBYTE = bytes.fromhex("6f")
|
||||||
|
P2SH_VERBYTES = bytes.fromhex("c4")
|
||||||
|
|
||||||
|
nOriginalClaimExpirationTime = 500
|
||||||
|
nExtendedClaimExpirationTime = 600
|
||||||
|
nExtendedClaimExpirationForkHeight = 800
|
||||||
|
nNormalizedNameForkHeight = 250
|
||||||
|
nMinTakeoverWorkaroundHeight = -1
|
||||||
|
nMaxTakeoverWorkaroundHeight = -1
|
||||||
|
nWitnessForkHeight = 150
|
||||||
|
nAllClaimsInMerkleForkHeight = 350
|
||||||
|
|
||||||
|
|
||||||
|
class LBCTestNet(LBCRegTest):
|
||||||
|
NET = "testnet"
|
||||||
|
GENESIS_HASH = '9c89283ba0f3227f6c03b70216b9f665f0118d5e0fa729cedf4fb34d6a34f463'
|
375
lbry/wallet/server/daemon.py
Normal file
375
lbry/wallet/server/daemon.py
Normal file
|
@ -0,0 +1,375 @@
|
||||||
|
import asyncio
|
||||||
|
import itertools
|
||||||
|
import json
|
||||||
|
import time
|
||||||
|
from functools import wraps
|
||||||
|
|
||||||
|
import aiohttp
|
||||||
|
from prometheus_client import Gauge, Histogram
|
||||||
|
from lbry.utils import LRUCacheWithMetrics
|
||||||
|
from lbry.wallet.rpc.jsonrpc import RPCError
|
||||||
|
from lbry.wallet.server.util import hex_to_bytes, class_logger
|
||||||
|
from lbry.wallet.rpc import JSONRPC
|
||||||
|
|
||||||
|
|
||||||
|
class DaemonError(Exception):
|
||||||
|
"""Raised when the daemon returns an error in its results."""
|
||||||
|
|
||||||
|
|
||||||
|
class WarmingUpError(Exception):
|
||||||
|
"""Internal - when the daemon is warming up."""
|
||||||
|
|
||||||
|
|
||||||
|
class WorkQueueFullError(Exception):
|
||||||
|
"""Internal - when the daemon's work queue is full."""
|
||||||
|
|
||||||
|
|
||||||
|
NAMESPACE = "wallet_server"
|
||||||
|
|
||||||
|
|
||||||
|
class Daemon:
|
||||||
|
"""Handles connections to a daemon at the given URL."""
|
||||||
|
|
||||||
|
WARMING_UP = -28
|
||||||
|
id_counter = itertools.count()
|
||||||
|
|
||||||
|
lbrycrd_request_time_metric = Histogram(
|
||||||
|
"lbrycrd_request", "lbrycrd requests count", namespace=NAMESPACE, labelnames=("method",)
|
||||||
|
)
|
||||||
|
lbrycrd_pending_count_metric = Gauge(
|
||||||
|
"lbrycrd_pending_count", "Number of lbrycrd rpcs that are in flight", namespace=NAMESPACE,
|
||||||
|
labelnames=("method",)
|
||||||
|
)
|
||||||
|
|
||||||
|
def __init__(self, coin, url, max_workqueue=10, init_retry=0.25,
|
||||||
|
max_retry=4.0):
|
||||||
|
self.coin = coin
|
||||||
|
self.logger = class_logger(__name__, self.__class__.__name__)
|
||||||
|
self.set_url(url)
|
||||||
|
# Limit concurrent RPC calls to this number.
|
||||||
|
# See DEFAULT_HTTP_WORKQUEUE in bitcoind, which is typically 16
|
||||||
|
self.workqueue_semaphore = asyncio.Semaphore(value=max_workqueue)
|
||||||
|
self.init_retry = init_retry
|
||||||
|
self.max_retry = max_retry
|
||||||
|
self._height = None
|
||||||
|
self.available_rpcs = {}
|
||||||
|
self.connector = aiohttp.TCPConnector()
|
||||||
|
self._block_hash_cache = LRUCacheWithMetrics(100000)
|
||||||
|
self._block_cache = LRUCacheWithMetrics(2 ** 13, metric_name='block', namespace=NAMESPACE)
|
||||||
|
|
||||||
|
async def close(self):
|
||||||
|
if self.connector:
|
||||||
|
await self.connector.close()
|
||||||
|
self.connector = None
|
||||||
|
|
||||||
|
def set_url(self, url):
|
||||||
|
"""Set the URLS to the given list, and switch to the first one."""
|
||||||
|
urls = url.split(',')
|
||||||
|
urls = [self.coin.sanitize_url(url) for url in urls]
|
||||||
|
for n, url in enumerate(urls):
|
||||||
|
status = '' if n else ' (current)'
|
||||||
|
logged_url = self.logged_url(url)
|
||||||
|
self.logger.info(f'daemon #{n + 1} at {logged_url}{status}')
|
||||||
|
self.url_index = 0
|
||||||
|
self.urls = urls
|
||||||
|
|
||||||
|
def current_url(self):
|
||||||
|
"""Returns the current daemon URL."""
|
||||||
|
return self.urls[self.url_index]
|
||||||
|
|
||||||
|
def logged_url(self, url=None):
|
||||||
|
"""The host and port part, for logging."""
|
||||||
|
url = url or self.current_url()
|
||||||
|
return url[url.rindex('@') + 1:]
|
||||||
|
|
||||||
|
def failover(self):
|
||||||
|
"""Call to fail-over to the next daemon URL.
|
||||||
|
|
||||||
|
Returns False if there is only one, otherwise True.
|
||||||
|
"""
|
||||||
|
if len(self.urls) > 1:
|
||||||
|
self.url_index = (self.url_index + 1) % len(self.urls)
|
||||||
|
self.logger.info(f'failing over to {self.logged_url()}')
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
|
def client_session(self):
|
||||||
|
"""An aiohttp client session."""
|
||||||
|
return aiohttp.ClientSession(connector=self.connector, connector_owner=False)
|
||||||
|
|
||||||
|
async def _send_data(self, data):
|
||||||
|
if not self.connector:
|
||||||
|
raise asyncio.CancelledError('Tried to send request during shutdown.')
|
||||||
|
async with self.workqueue_semaphore:
|
||||||
|
async with self.client_session() as session:
|
||||||
|
async with session.post(self.current_url(), data=data) as resp:
|
||||||
|
kind = resp.headers.get('Content-Type', None)
|
||||||
|
if kind == 'application/json':
|
||||||
|
return await resp.json()
|
||||||
|
# bitcoind's HTTP protocol "handling" is a bad joke
|
||||||
|
text = await resp.text()
|
||||||
|
if 'Work queue depth exceeded' in text:
|
||||||
|
raise WorkQueueFullError
|
||||||
|
text = text.strip() or resp.reason
|
||||||
|
self.logger.error(text)
|
||||||
|
raise DaemonError(text)
|
||||||
|
|
||||||
|
async def _send(self, payload, processor):
|
||||||
|
"""Send a payload to be converted to JSON.
|
||||||
|
|
||||||
|
Handles temporary connection issues. Daemon response errors
|
||||||
|
are raise through DaemonError.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def log_error(error):
|
||||||
|
nonlocal last_error_log, retry
|
||||||
|
now = time.time()
|
||||||
|
if now - last_error_log > 60:
|
||||||
|
last_error_log = now
|
||||||
|
self.logger.error(f'{error} Retrying occasionally...')
|
||||||
|
if retry == self.max_retry and self.failover():
|
||||||
|
retry = 0
|
||||||
|
|
||||||
|
on_good_message = None
|
||||||
|
last_error_log = 0
|
||||||
|
data = json.dumps(payload)
|
||||||
|
retry = self.init_retry
|
||||||
|
methods = tuple(
|
||||||
|
[payload['method']] if isinstance(payload, dict) else [request['method'] for request in payload]
|
||||||
|
)
|
||||||
|
while True:
|
||||||
|
try:
|
||||||
|
for method in methods:
|
||||||
|
self.lbrycrd_pending_count_metric.labels(method=method).inc()
|
||||||
|
result = await self._send_data(data)
|
||||||
|
result = processor(result)
|
||||||
|
if on_good_message:
|
||||||
|
self.logger.info(on_good_message)
|
||||||
|
return result
|
||||||
|
except asyncio.TimeoutError:
|
||||||
|
log_error('timeout error.')
|
||||||
|
except aiohttp.ServerDisconnectedError:
|
||||||
|
log_error('disconnected.')
|
||||||
|
on_good_message = 'connection restored'
|
||||||
|
except aiohttp.ClientConnectionError:
|
||||||
|
log_error('connection problem - is your daemon running?')
|
||||||
|
on_good_message = 'connection restored'
|
||||||
|
except aiohttp.ClientError as e:
|
||||||
|
log_error(f'daemon error: {e}')
|
||||||
|
on_good_message = 'running normally'
|
||||||
|
except WarmingUpError:
|
||||||
|
log_error('starting up checking blocks.')
|
||||||
|
on_good_message = 'running normally'
|
||||||
|
except WorkQueueFullError:
|
||||||
|
log_error('work queue full.')
|
||||||
|
on_good_message = 'running normally'
|
||||||
|
finally:
|
||||||
|
for method in methods:
|
||||||
|
self.lbrycrd_pending_count_metric.labels(method=method).dec()
|
||||||
|
await asyncio.sleep(retry)
|
||||||
|
retry = max(min(self.max_retry, retry * 2), self.init_retry)
|
||||||
|
|
||||||
|
async def _send_single(self, method, params=None):
|
||||||
|
"""Send a single request to the daemon."""
|
||||||
|
|
||||||
|
start = time.perf_counter()
|
||||||
|
|
||||||
|
def processor(result):
|
||||||
|
err = result['error']
|
||||||
|
if not err:
|
||||||
|
return result['result']
|
||||||
|
if err.get('code') == self.WARMING_UP:
|
||||||
|
raise WarmingUpError
|
||||||
|
raise DaemonError(err)
|
||||||
|
|
||||||
|
payload = {'method': method, 'id': next(self.id_counter)}
|
||||||
|
if params:
|
||||||
|
payload['params'] = params
|
||||||
|
result = await self._send(payload, processor)
|
||||||
|
self.lbrycrd_request_time_metric.labels(method=method).observe(time.perf_counter() - start)
|
||||||
|
return result
|
||||||
|
|
||||||
|
async def _send_vector(self, method, params_iterable, replace_errs=False):
|
||||||
|
"""Send several requests of the same method.
|
||||||
|
|
||||||
|
The result will be an array of the same length as params_iterable.
|
||||||
|
If replace_errs is true, any item with an error is returned as None,
|
||||||
|
otherwise an exception is raised."""
|
||||||
|
|
||||||
|
start = time.perf_counter()
|
||||||
|
|
||||||
|
def processor(result):
|
||||||
|
errs = [item['error'] for item in result if item['error']]
|
||||||
|
if any(err.get('code') == self.WARMING_UP for err in errs):
|
||||||
|
raise WarmingUpError
|
||||||
|
if not errs or replace_errs:
|
||||||
|
return [item['result'] for item in result]
|
||||||
|
raise DaemonError(errs)
|
||||||
|
|
||||||
|
payload = [{'method': method, 'params': p, 'id': next(self.id_counter)}
|
||||||
|
for p in params_iterable]
|
||||||
|
result = []
|
||||||
|
if payload:
|
||||||
|
result = await self._send(payload, processor)
|
||||||
|
self.lbrycrd_request_time_metric.labels(method=method).observe(time.perf_counter() - start)
|
||||||
|
return result
|
||||||
|
|
||||||
|
async def _is_rpc_available(self, method):
|
||||||
|
"""Return whether given RPC method is available in the daemon.
|
||||||
|
|
||||||
|
Results are cached and the daemon will generally not be queried with
|
||||||
|
the same method more than once."""
|
||||||
|
available = self.available_rpcs.get(method)
|
||||||
|
if available is None:
|
||||||
|
available = True
|
||||||
|
try:
|
||||||
|
await self._send_single(method)
|
||||||
|
except DaemonError as e:
|
||||||
|
err = e.args[0]
|
||||||
|
error_code = err.get("code")
|
||||||
|
available = error_code != JSONRPC.METHOD_NOT_FOUND
|
||||||
|
self.available_rpcs[method] = available
|
||||||
|
return available
|
||||||
|
|
||||||
|
async def block_hex_hashes(self, first, count):
|
||||||
|
"""Return the hex hashes of count block starting at height first."""
|
||||||
|
if first + count < (self.cached_height() or 0) - 200:
|
||||||
|
return await self._cached_block_hex_hashes(first, count)
|
||||||
|
params_iterable = ((h, ) for h in range(first, first + count))
|
||||||
|
return await self._send_vector('getblockhash', params_iterable)
|
||||||
|
|
||||||
|
async def _cached_block_hex_hashes(self, first, count):
|
||||||
|
"""Return the hex hashes of count block starting at height first."""
|
||||||
|
cached = self._block_hash_cache.get((first, count))
|
||||||
|
if cached:
|
||||||
|
return cached
|
||||||
|
params_iterable = ((h, ) for h in range(first, first + count))
|
||||||
|
self._block_hash_cache[(first, count)] = await self._send_vector('getblockhash', params_iterable)
|
||||||
|
return self._block_hash_cache[(first, count)]
|
||||||
|
|
||||||
|
async def deserialised_block(self, hex_hash):
|
||||||
|
"""Return the deserialised block with the given hex hash."""
|
||||||
|
if hex_hash not in self._block_cache:
|
||||||
|
block = await self._send_single('getblock', (hex_hash, True))
|
||||||
|
self._block_cache[hex_hash] = block
|
||||||
|
return block
|
||||||
|
return self._block_cache[hex_hash]
|
||||||
|
|
||||||
|
async def raw_blocks(self, hex_hashes):
|
||||||
|
"""Return the raw binary blocks with the given hex hashes."""
|
||||||
|
params_iterable = ((h, False) for h in hex_hashes)
|
||||||
|
blocks = await self._send_vector('getblock', params_iterable)
|
||||||
|
# Convert hex string to bytes
|
||||||
|
return [hex_to_bytes(block) for block in blocks]
|
||||||
|
|
||||||
|
async def mempool_hashes(self):
|
||||||
|
"""Update our record of the daemon's mempool hashes."""
|
||||||
|
return await self._send_single('getrawmempool')
|
||||||
|
|
||||||
|
async def estimatefee(self, block_count):
|
||||||
|
"""Return the fee estimate for the block count. Units are whole
|
||||||
|
currency units per KB, e.g. 0.00000995, or -1 if no estimate
|
||||||
|
is available.
|
||||||
|
"""
|
||||||
|
args = (block_count, )
|
||||||
|
if await self._is_rpc_available('estimatesmartfee'):
|
||||||
|
estimate = await self._send_single('estimatesmartfee', args)
|
||||||
|
return estimate.get('feerate', -1)
|
||||||
|
return await self._send_single('estimatefee', args)
|
||||||
|
|
||||||
|
async def getnetworkinfo(self):
|
||||||
|
"""Return the result of the 'getnetworkinfo' RPC call."""
|
||||||
|
return await self._send_single('getnetworkinfo')
|
||||||
|
|
||||||
|
async def relayfee(self):
|
||||||
|
"""The minimum fee a low-priority tx must pay in order to be accepted
|
||||||
|
to the daemon's memory pool."""
|
||||||
|
network_info = await self.getnetworkinfo()
|
||||||
|
return network_info['relayfee']
|
||||||
|
|
||||||
|
async def getrawtransaction(self, hex_hash, verbose=False):
|
||||||
|
"""Return the serialized raw transaction with the given hash."""
|
||||||
|
# Cast to int because some coin daemons are old and require it
|
||||||
|
return await self._send_single('getrawtransaction',
|
||||||
|
(hex_hash, int(verbose)))
|
||||||
|
|
||||||
|
async def getrawtransactions(self, hex_hashes, replace_errs=True):
|
||||||
|
"""Return the serialized raw transactions with the given hashes.
|
||||||
|
|
||||||
|
Replaces errors with None by default."""
|
||||||
|
params_iterable = ((hex_hash, 0) for hex_hash in hex_hashes)
|
||||||
|
txs = await self._send_vector('getrawtransaction', params_iterable,
|
||||||
|
replace_errs=replace_errs)
|
||||||
|
# Convert hex strings to bytes
|
||||||
|
return [hex_to_bytes(tx) if tx else None for tx in txs]
|
||||||
|
|
||||||
|
async def broadcast_transaction(self, raw_tx):
|
||||||
|
"""Broadcast a transaction to the network."""
|
||||||
|
return await self._send_single('sendrawtransaction', (raw_tx, ))
|
||||||
|
|
||||||
|
async def height(self):
|
||||||
|
"""Query the daemon for its current height."""
|
||||||
|
self._height = await self._send_single('getblockcount')
|
||||||
|
return self._height
|
||||||
|
|
||||||
|
def cached_height(self):
|
||||||
|
"""Return the cached daemon height.
|
||||||
|
|
||||||
|
If the daemon has not been queried yet this returns None."""
|
||||||
|
return self._height
|
||||||
|
|
||||||
|
|
||||||
|
def handles_errors(decorated_function):
|
||||||
|
@wraps(decorated_function)
|
||||||
|
async def wrapper(*args, **kwargs):
|
||||||
|
try:
|
||||||
|
return await decorated_function(*args, **kwargs)
|
||||||
|
except DaemonError as daemon_error:
|
||||||
|
raise RPCError(1, daemon_error.args[0])
|
||||||
|
return wrapper
|
||||||
|
|
||||||
|
|
||||||
|
class LBCDaemon(Daemon):
|
||||||
|
@handles_errors
|
||||||
|
async def getrawtransaction(self, hex_hash, verbose=False):
|
||||||
|
return await super().getrawtransaction(hex_hash=hex_hash, verbose=verbose)
|
||||||
|
|
||||||
|
@handles_errors
|
||||||
|
async def getclaimbyid(self, claim_id):
|
||||||
|
'''Given a claim id, retrieves claim information.'''
|
||||||
|
return await self._send_single('getclaimbyid', (claim_id,))
|
||||||
|
|
||||||
|
@handles_errors
|
||||||
|
async def getclaimsbyids(self, claim_ids):
|
||||||
|
'''Given a list of claim ids, batches calls to retrieve claim information.'''
|
||||||
|
return await self._send_vector('getclaimbyid', ((claim_id,) for claim_id in claim_ids))
|
||||||
|
|
||||||
|
@handles_errors
|
||||||
|
async def getclaimsforname(self, name):
|
||||||
|
'''Given a name, retrieves all claims matching that name.'''
|
||||||
|
return await self._send_single('getclaimsforname', (name,))
|
||||||
|
|
||||||
|
@handles_errors
|
||||||
|
async def getclaimsfortx(self, txid):
|
||||||
|
'''Given a txid, returns the claims it make.'''
|
||||||
|
return await self._send_single('getclaimsfortx', (txid,)) or []
|
||||||
|
|
||||||
|
@handles_errors
|
||||||
|
async def getnameproof(self, name, block_hash=None):
|
||||||
|
'''Given a name and optional block_hash, returns a name proof and winner, if any.'''
|
||||||
|
return await self._send_single('getnameproof', (name, block_hash,) if block_hash else (name,))
|
||||||
|
|
||||||
|
@handles_errors
|
||||||
|
async def getvalueforname(self, name):
|
||||||
|
'''Given a name, returns the winning claim value.'''
|
||||||
|
return await self._send_single('getvalueforname', (name,))
|
||||||
|
|
||||||
|
@handles_errors
|
||||||
|
async def getnamesintrie(self):
|
||||||
|
'''Given a name, returns the winning claim value.'''
|
||||||
|
return await self._send_single('getnamesintrie')
|
||||||
|
|
||||||
|
@handles_errors
|
||||||
|
async def claimname(self, name, hexvalue, amount):
|
||||||
|
'''Claim a name, used for functional tests only.'''
|
||||||
|
return await self._send_single('claimname', (name, hexvalue, float(amount)))
|
42
lbry/wallet/server/db/__init__.py
Normal file
42
lbry/wallet/server/db/__init__.py
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
import enum
|
||||||
|
|
||||||
|
|
||||||
|
@enum.unique
|
||||||
|
class DB_PREFIXES(enum.Enum):
|
||||||
|
claim_to_support = b'K'
|
||||||
|
support_to_claim = b'L'
|
||||||
|
|
||||||
|
claim_to_txo = b'E'
|
||||||
|
txo_to_claim = b'G'
|
||||||
|
|
||||||
|
claim_to_channel = b'I'
|
||||||
|
channel_to_claim = b'J'
|
||||||
|
|
||||||
|
claim_short_id_prefix = b'F'
|
||||||
|
effective_amount = b'D'
|
||||||
|
claim_expiration = b'O'
|
||||||
|
|
||||||
|
claim_takeover = b'P'
|
||||||
|
pending_activation = b'Q'
|
||||||
|
activated_claim_and_support = b'R'
|
||||||
|
active_amount = b'S'
|
||||||
|
|
||||||
|
repost = b'V'
|
||||||
|
reposted_claim = b'W'
|
||||||
|
|
||||||
|
undo = b'M'
|
||||||
|
claim_diff = b'Y'
|
||||||
|
|
||||||
|
tx = b'B'
|
||||||
|
block_hash = b'C'
|
||||||
|
header = b'H'
|
||||||
|
tx_num = b'N'
|
||||||
|
tx_count = b'T'
|
||||||
|
tx_hash = b'X'
|
||||||
|
utxo = b'u'
|
||||||
|
hashx_utxo = b'h'
|
||||||
|
hashx_history = b'x'
|
||||||
|
db_state = b's'
|
||||||
|
channel_count = b'Z'
|
||||||
|
support_amount = b'a'
|
||||||
|
block_txs = b'b'
|
447
lbry/wallet/server/db/common.py
Normal file
447
lbry/wallet/server/db/common.py
Normal file
|
@ -0,0 +1,447 @@
|
||||||
|
import typing
|
||||||
|
|
||||||
|
CLAIM_TYPES = {
|
||||||
|
'stream': 1,
|
||||||
|
'channel': 2,
|
||||||
|
'repost': 3,
|
||||||
|
'collection': 4,
|
||||||
|
}
|
||||||
|
|
||||||
|
STREAM_TYPES = {
|
||||||
|
'video': 1,
|
||||||
|
'audio': 2,
|
||||||
|
'image': 3,
|
||||||
|
'document': 4,
|
||||||
|
'binary': 5,
|
||||||
|
'model': 6,
|
||||||
|
}
|
||||||
|
|
||||||
|
# 9/21/2020
|
||||||
|
MOST_USED_TAGS = {
|
||||||
|
"gaming",
|
||||||
|
"people & blogs",
|
||||||
|
"entertainment",
|
||||||
|
"music",
|
||||||
|
"pop culture",
|
||||||
|
"education",
|
||||||
|
"technology",
|
||||||
|
"blockchain",
|
||||||
|
"news",
|
||||||
|
"funny",
|
||||||
|
"science & technology",
|
||||||
|
"learning",
|
||||||
|
"gameplay",
|
||||||
|
"news & politics",
|
||||||
|
"comedy",
|
||||||
|
"bitcoin",
|
||||||
|
"beliefs",
|
||||||
|
"nature",
|
||||||
|
"art",
|
||||||
|
"economics",
|
||||||
|
"film & animation",
|
||||||
|
"lets play",
|
||||||
|
"games",
|
||||||
|
"sports",
|
||||||
|
"howto & style",
|
||||||
|
"game",
|
||||||
|
"cryptocurrency",
|
||||||
|
"playstation 4",
|
||||||
|
"automotive",
|
||||||
|
"crypto",
|
||||||
|
"mature",
|
||||||
|
"sony interactive entertainment",
|
||||||
|
"walkthrough",
|
||||||
|
"tutorial",
|
||||||
|
"video game",
|
||||||
|
"weapons",
|
||||||
|
"playthrough",
|
||||||
|
"pc",
|
||||||
|
"anime",
|
||||||
|
"how to",
|
||||||
|
"btc",
|
||||||
|
"fun",
|
||||||
|
"ethereum",
|
||||||
|
"food",
|
||||||
|
"travel & events",
|
||||||
|
"minecraft",
|
||||||
|
"science",
|
||||||
|
"autos & vehicles",
|
||||||
|
"play",
|
||||||
|
"politics",
|
||||||
|
"commentary",
|
||||||
|
"twitch",
|
||||||
|
"ps4live",
|
||||||
|
"love",
|
||||||
|
"ps4",
|
||||||
|
"nonprofits & activism",
|
||||||
|
"ps4share",
|
||||||
|
"fortnite",
|
||||||
|
"xbox",
|
||||||
|
"porn",
|
||||||
|
"video games",
|
||||||
|
"trump",
|
||||||
|
"español",
|
||||||
|
"money",
|
||||||
|
"music video",
|
||||||
|
"nintendo",
|
||||||
|
"movie",
|
||||||
|
"coronavirus",
|
||||||
|
"donald trump",
|
||||||
|
"steam",
|
||||||
|
"trailer",
|
||||||
|
"android",
|
||||||
|
"podcast",
|
||||||
|
"xbox one",
|
||||||
|
"survival",
|
||||||
|
"audio",
|
||||||
|
"linux",
|
||||||
|
"travel",
|
||||||
|
"funny moments",
|
||||||
|
"litecoin",
|
||||||
|
"animation",
|
||||||
|
"gamer",
|
||||||
|
"lets",
|
||||||
|
"playstation",
|
||||||
|
"bitcoin news",
|
||||||
|
"history",
|
||||||
|
"xxx",
|
||||||
|
"fox news",
|
||||||
|
"dance",
|
||||||
|
"god",
|
||||||
|
"adventure",
|
||||||
|
"liberal",
|
||||||
|
"2020",
|
||||||
|
"horror",
|
||||||
|
"government",
|
||||||
|
"freedom",
|
||||||
|
"reaction",
|
||||||
|
"meme",
|
||||||
|
"photography",
|
||||||
|
"truth",
|
||||||
|
"health",
|
||||||
|
"lbry",
|
||||||
|
"family",
|
||||||
|
"online",
|
||||||
|
"eth",
|
||||||
|
"crypto news",
|
||||||
|
"diy",
|
||||||
|
"trading",
|
||||||
|
"gold",
|
||||||
|
"memes",
|
||||||
|
"world",
|
||||||
|
"space",
|
||||||
|
"lol",
|
||||||
|
"covid-19",
|
||||||
|
"rpg",
|
||||||
|
"humor",
|
||||||
|
"democrat",
|
||||||
|
"film",
|
||||||
|
"call of duty",
|
||||||
|
"tech",
|
||||||
|
"religion",
|
||||||
|
"conspiracy",
|
||||||
|
"rap",
|
||||||
|
"cnn",
|
||||||
|
"hangoutsonair",
|
||||||
|
"unboxing",
|
||||||
|
"fiction",
|
||||||
|
"conservative",
|
||||||
|
"cars",
|
||||||
|
"hoa",
|
||||||
|
"epic",
|
||||||
|
"programming",
|
||||||
|
"progressive",
|
||||||
|
"cryptocurrency news",
|
||||||
|
"classical",
|
||||||
|
"jesus",
|
||||||
|
"movies",
|
||||||
|
"book",
|
||||||
|
"ps3",
|
||||||
|
"republican",
|
||||||
|
"fitness",
|
||||||
|
"books",
|
||||||
|
"multiplayer",
|
||||||
|
"animals",
|
||||||
|
"pokemon",
|
||||||
|
"bitcoin price",
|
||||||
|
"facebook",
|
||||||
|
"sharefactory",
|
||||||
|
"criptomonedas",
|
||||||
|
"cod",
|
||||||
|
"bible",
|
||||||
|
"business",
|
||||||
|
"stream",
|
||||||
|
"comics",
|
||||||
|
"how",
|
||||||
|
"fail",
|
||||||
|
"nsfw",
|
||||||
|
"new music",
|
||||||
|
"satire",
|
||||||
|
"pets & animals",
|
||||||
|
"computer",
|
||||||
|
"classical music",
|
||||||
|
"indie",
|
||||||
|
"musica",
|
||||||
|
"msnbc",
|
||||||
|
"fps",
|
||||||
|
"mod",
|
||||||
|
"sport",
|
||||||
|
"sony",
|
||||||
|
"ripple",
|
||||||
|
"auto",
|
||||||
|
"rock",
|
||||||
|
"marvel",
|
||||||
|
"complete",
|
||||||
|
"mining",
|
||||||
|
"political",
|
||||||
|
"mobile",
|
||||||
|
"pubg",
|
||||||
|
"hip hop",
|
||||||
|
"flat earth",
|
||||||
|
"xbox 360",
|
||||||
|
"reviews",
|
||||||
|
"vlogging",
|
||||||
|
"latest news",
|
||||||
|
"hack",
|
||||||
|
"tarot",
|
||||||
|
"iphone",
|
||||||
|
"media",
|
||||||
|
"cute",
|
||||||
|
"christian",
|
||||||
|
"free speech",
|
||||||
|
"trap",
|
||||||
|
"war",
|
||||||
|
"remix",
|
||||||
|
"ios",
|
||||||
|
"xrp",
|
||||||
|
"spirituality",
|
||||||
|
"song",
|
||||||
|
"league of legends",
|
||||||
|
"cat"
|
||||||
|
}
|
||||||
|
|
||||||
|
MATURE_TAGS = [
|
||||||
|
'nsfw', 'porn', 'xxx', 'mature', 'adult', 'sex'
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
def normalize_tag(tag):
|
||||||
|
return tag.replace(" ", "_").replace("&", "and").replace("-", "_")
|
||||||
|
|
||||||
|
|
||||||
|
COMMON_TAGS = {
|
||||||
|
tag: normalize_tag(tag) for tag in list(MOST_USED_TAGS)
|
||||||
|
}
|
||||||
|
|
||||||
|
INDEXED_LANGUAGES = [
|
||||||
|
'none',
|
||||||
|
'en',
|
||||||
|
'aa',
|
||||||
|
'ab',
|
||||||
|
'ae',
|
||||||
|
'af',
|
||||||
|
'ak',
|
||||||
|
'am',
|
||||||
|
'an',
|
||||||
|
'ar',
|
||||||
|
'as',
|
||||||
|
'av',
|
||||||
|
'ay',
|
||||||
|
'az',
|
||||||
|
'ba',
|
||||||
|
'be',
|
||||||
|
'bg',
|
||||||
|
'bh',
|
||||||
|
'bi',
|
||||||
|
'bm',
|
||||||
|
'bn',
|
||||||
|
'bo',
|
||||||
|
'br',
|
||||||
|
'bs',
|
||||||
|
'ca',
|
||||||
|
'ce',
|
||||||
|
'ch',
|
||||||
|
'co',
|
||||||
|
'cr',
|
||||||
|
'cs',
|
||||||
|
'cu',
|
||||||
|
'cv',
|
||||||
|
'cy',
|
||||||
|
'da',
|
||||||
|
'de',
|
||||||
|
'dv',
|
||||||
|
'dz',
|
||||||
|
'ee',
|
||||||
|
'el',
|
||||||
|
'eo',
|
||||||
|
'es',
|
||||||
|
'et',
|
||||||
|
'eu',
|
||||||
|
'fa',
|
||||||
|
'ff',
|
||||||
|
'fi',
|
||||||
|
'fj',
|
||||||
|
'fo',
|
||||||
|
'fr',
|
||||||
|
'fy',
|
||||||
|
'ga',
|
||||||
|
'gd',
|
||||||
|
'gl',
|
||||||
|
'gn',
|
||||||
|
'gu',
|
||||||
|
'gv',
|
||||||
|
'ha',
|
||||||
|
'he',
|
||||||
|
'hi',
|
||||||
|
'ho',
|
||||||
|
'hr',
|
||||||
|
'ht',
|
||||||
|
'hu',
|
||||||
|
'hy',
|
||||||
|
'hz',
|
||||||
|
'ia',
|
||||||
|
'id',
|
||||||
|
'ie',
|
||||||
|
'ig',
|
||||||
|
'ii',
|
||||||
|
'ik',
|
||||||
|
'io',
|
||||||
|
'is',
|
||||||
|
'it',
|
||||||
|
'iu',
|
||||||
|
'ja',
|
||||||
|
'jv',
|
||||||
|
'ka',
|
||||||
|
'kg',
|
||||||
|
'ki',
|
||||||
|
'kj',
|
||||||
|
'kk',
|
||||||
|
'kl',
|
||||||
|
'km',
|
||||||
|
'kn',
|
||||||
|
'ko',
|
||||||
|
'kr',
|
||||||
|
'ks',
|
||||||
|
'ku',
|
||||||
|
'kv',
|
||||||
|
'kw',
|
||||||
|
'ky',
|
||||||
|
'la',
|
||||||
|
'lb',
|
||||||
|
'lg',
|
||||||
|
'li',
|
||||||
|
'ln',
|
||||||
|
'lo',
|
||||||
|
'lt',
|
||||||
|
'lu',
|
||||||
|
'lv',
|
||||||
|
'mg',
|
||||||
|
'mh',
|
||||||
|
'mi',
|
||||||
|
'mk',
|
||||||
|
'ml',
|
||||||
|
'mn',
|
||||||
|
'mr',
|
||||||
|
'ms',
|
||||||
|
'mt',
|
||||||
|
'my',
|
||||||
|
'na',
|
||||||
|
'nb',
|
||||||
|
'nd',
|
||||||
|
'ne',
|
||||||
|
'ng',
|
||||||
|
'nl',
|
||||||
|
'nn',
|
||||||
|
'no',
|
||||||
|
'nr',
|
||||||
|
'nv',
|
||||||
|
'ny',
|
||||||
|
'oc',
|
||||||
|
'oj',
|
||||||
|
'om',
|
||||||
|
'or',
|
||||||
|
'os',
|
||||||
|
'pa',
|
||||||
|
'pi',
|
||||||
|
'pl',
|
||||||
|
'ps',
|
||||||
|
'pt',
|
||||||
|
'qu',
|
||||||
|
'rm',
|
||||||
|
'rn',
|
||||||
|
'ro',
|
||||||
|
'ru',
|
||||||
|
'rw',
|
||||||
|
'sa',
|
||||||
|
'sc',
|
||||||
|
'sd',
|
||||||
|
'se',
|
||||||
|
'sg',
|
||||||
|
'si',
|
||||||
|
'sk',
|
||||||
|
'sl',
|
||||||
|
'sm',
|
||||||
|
'sn',
|
||||||
|
'so',
|
||||||
|
'sq',
|
||||||
|
'sr',
|
||||||
|
'ss',
|
||||||
|
'st',
|
||||||
|
'su',
|
||||||
|
'sv',
|
||||||
|
'sw',
|
||||||
|
'ta',
|
||||||
|
'te',
|
||||||
|
'tg',
|
||||||
|
'th',
|
||||||
|
'ti',
|
||||||
|
'tk',
|
||||||
|
'tl',
|
||||||
|
'tn',
|
||||||
|
'to',
|
||||||
|
'tr',
|
||||||
|
'ts',
|
||||||
|
'tt',
|
||||||
|
'tw',
|
||||||
|
'ty',
|
||||||
|
'ug',
|
||||||
|
'uk',
|
||||||
|
'ur',
|
||||||
|
'uz',
|
||||||
|
've',
|
||||||
|
'vi',
|
||||||
|
'vo',
|
||||||
|
'wa',
|
||||||
|
'wo',
|
||||||
|
'xh',
|
||||||
|
'yi',
|
||||||
|
'yo',
|
||||||
|
'za',
|
||||||
|
'zh',
|
||||||
|
'zu'
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class ResolveResult(typing.NamedTuple):
|
||||||
|
name: str
|
||||||
|
normalized_name: str
|
||||||
|
claim_hash: bytes
|
||||||
|
tx_num: int
|
||||||
|
position: int
|
||||||
|
tx_hash: bytes
|
||||||
|
height: int
|
||||||
|
amount: int
|
||||||
|
short_url: str
|
||||||
|
is_controlling: bool
|
||||||
|
canonical_url: str
|
||||||
|
creation_height: int
|
||||||
|
activation_height: int
|
||||||
|
expiration_height: int
|
||||||
|
effective_amount: int
|
||||||
|
support_amount: int
|
||||||
|
reposted: int
|
||||||
|
last_takeover_height: typing.Optional[int]
|
||||||
|
claims_in_channel: typing.Optional[int]
|
||||||
|
channel_hash: typing.Optional[bytes]
|
||||||
|
reposted_claim_hash: typing.Optional[bytes]
|
||||||
|
signature_valid: typing.Optional[bool]
|
119
lbry/wallet/server/db/db.py
Normal file
119
lbry/wallet/server/db/db.py
Normal file
|
@ -0,0 +1,119 @@
|
||||||
|
import struct
|
||||||
|
from typing import Optional
|
||||||
|
from lbry.wallet.server.db import DB_PREFIXES
|
||||||
|
from lbry.wallet.server.db.revertable import RevertableOpStack, RevertablePut, RevertableDelete
|
||||||
|
|
||||||
|
|
||||||
|
class KeyValueStorage:
|
||||||
|
def get(self, key: bytes, fill_cache: bool = True) -> Optional[bytes]:
|
||||||
|
raise NotImplemented()
|
||||||
|
|
||||||
|
def iterator(self, reverse=False, start=None, stop=None, include_start=True, include_stop=False, prefix=None,
|
||||||
|
include_key=True, include_value=True, fill_cache=True):
|
||||||
|
raise NotImplemented()
|
||||||
|
|
||||||
|
def write_batch(self, transaction: bool = False):
|
||||||
|
raise NotImplemented()
|
||||||
|
|
||||||
|
def close(self):
|
||||||
|
raise NotImplemented()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def closed(self) -> bool:
|
||||||
|
raise NotImplemented()
|
||||||
|
|
||||||
|
|
||||||
|
class PrefixDB:
|
||||||
|
UNDO_KEY_STRUCT = struct.Struct(b'>Q')
|
||||||
|
|
||||||
|
def __init__(self, db: KeyValueStorage, max_undo_depth: int = 200, unsafe_prefixes=None):
|
||||||
|
self._db = db
|
||||||
|
self._op_stack = RevertableOpStack(db.get, unsafe_prefixes=unsafe_prefixes)
|
||||||
|
self._max_undo_depth = max_undo_depth
|
||||||
|
|
||||||
|
def unsafe_commit(self):
|
||||||
|
"""
|
||||||
|
Write staged changes to the database without keeping undo information
|
||||||
|
Changes written cannot be undone
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
with self._db.write_batch(transaction=True) as batch:
|
||||||
|
batch_put = batch.put
|
||||||
|
batch_delete = batch.delete
|
||||||
|
for staged_change in self._op_stack:
|
||||||
|
if staged_change.is_put:
|
||||||
|
batch_put(staged_change.key, staged_change.value)
|
||||||
|
else:
|
||||||
|
batch_delete(staged_change.key)
|
||||||
|
finally:
|
||||||
|
self._op_stack.clear()
|
||||||
|
|
||||||
|
def commit(self, height: int):
|
||||||
|
"""
|
||||||
|
Write changes for a block height to the database and keep undo information so that the changes can be reverted
|
||||||
|
"""
|
||||||
|
undo_ops = self._op_stack.get_undo_ops()
|
||||||
|
delete_undos = []
|
||||||
|
if height > self._max_undo_depth:
|
||||||
|
delete_undos.extend(self._db.iterator(
|
||||||
|
start=DB_PREFIXES.undo.value + self.UNDO_KEY_STRUCT.pack(0),
|
||||||
|
stop=DB_PREFIXES.undo.value + self.UNDO_KEY_STRUCT.pack(height - self._max_undo_depth),
|
||||||
|
include_value=False
|
||||||
|
))
|
||||||
|
try:
|
||||||
|
with self._db.write_batch(transaction=True) as batch:
|
||||||
|
batch_put = batch.put
|
||||||
|
batch_delete = batch.delete
|
||||||
|
for staged_change in self._op_stack:
|
||||||
|
if staged_change.is_put:
|
||||||
|
batch_put(staged_change.key, staged_change.value)
|
||||||
|
else:
|
||||||
|
batch_delete(staged_change.key)
|
||||||
|
for undo_to_delete in delete_undos:
|
||||||
|
batch_delete(undo_to_delete)
|
||||||
|
batch_put(DB_PREFIXES.undo.value + self.UNDO_KEY_STRUCT.pack(height), undo_ops)
|
||||||
|
finally:
|
||||||
|
self._op_stack.clear()
|
||||||
|
|
||||||
|
def rollback(self, height: int):
|
||||||
|
"""
|
||||||
|
Revert changes for a block height
|
||||||
|
"""
|
||||||
|
undo_key = DB_PREFIXES.undo.value + self.UNDO_KEY_STRUCT.pack(height)
|
||||||
|
self._op_stack.apply_packed_undo_ops(self._db.get(undo_key))
|
||||||
|
try:
|
||||||
|
with self._db.write_batch(transaction=True) as batch:
|
||||||
|
batch_put = batch.put
|
||||||
|
batch_delete = batch.delete
|
||||||
|
for staged_change in self._op_stack:
|
||||||
|
if staged_change.is_put:
|
||||||
|
batch_put(staged_change.key, staged_change.value)
|
||||||
|
else:
|
||||||
|
batch_delete(staged_change.key)
|
||||||
|
batch_delete(undo_key)
|
||||||
|
finally:
|
||||||
|
self._op_stack.clear()
|
||||||
|
|
||||||
|
def get(self, key: bytes, fill_cache: bool = True) -> Optional[bytes]:
|
||||||
|
return self._db.get(key, fill_cache=fill_cache)
|
||||||
|
|
||||||
|
def iterator(self, reverse=False, start=None, stop=None, include_start=True, include_stop=False, prefix=None,
|
||||||
|
include_key=True, include_value=True, fill_cache=True):
|
||||||
|
return self._db.iterator(
|
||||||
|
reverse=reverse, start=start, stop=stop, include_start=include_start, include_stop=include_stop,
|
||||||
|
prefix=prefix, include_key=include_key, include_value=include_value, fill_cache=fill_cache
|
||||||
|
)
|
||||||
|
|
||||||
|
def close(self):
|
||||||
|
if not self._db.closed:
|
||||||
|
self._db.close()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def closed(self):
|
||||||
|
return self._db.closed
|
||||||
|
|
||||||
|
def stage_raw_put(self, key: bytes, value: bytes):
|
||||||
|
self._op_stack.append_op(RevertablePut(key, value))
|
||||||
|
|
||||||
|
def stage_raw_delete(self, key: bytes, value: bytes):
|
||||||
|
self._op_stack.append_op(RevertableDelete(key, value))
|
1
lbry/wallet/server/db/elasticsearch/__init__.py
Normal file
1
lbry/wallet/server/db/elasticsearch/__init__.py
Normal file
|
@ -0,0 +1 @@
|
||||||
|
from .search import SearchIndex
|
84
lbry/wallet/server/db/elasticsearch/constants.py
Normal file
84
lbry/wallet/server/db/elasticsearch/constants.py
Normal file
|
@ -0,0 +1,84 @@
|
||||||
|
INDEX_DEFAULT_SETTINGS = {
|
||||||
|
"settings":
|
||||||
|
{"analysis":
|
||||||
|
{"analyzer": {
|
||||||
|
"default": {"tokenizer": "whitespace", "filter": ["lowercase", "porter_stem"]}}},
|
||||||
|
"index":
|
||||||
|
{"refresh_interval": -1,
|
||||||
|
"number_of_shards": 1,
|
||||||
|
"number_of_replicas": 0,
|
||||||
|
"sort": {
|
||||||
|
"field": ["trending_score", "release_time"],
|
||||||
|
"order": ["desc", "desc"]
|
||||||
|
}}
|
||||||
|
},
|
||||||
|
"mappings": {
|
||||||
|
"properties": {
|
||||||
|
"claim_id": {
|
||||||
|
"fields": {
|
||||||
|
"keyword": {
|
||||||
|
"ignore_above": 256,
|
||||||
|
"type": "keyword"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"type": "text",
|
||||||
|
"index_prefixes": {
|
||||||
|
"min_chars": 1,
|
||||||
|
"max_chars": 10
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"height": {"type": "integer"},
|
||||||
|
"claim_type": {"type": "byte"},
|
||||||
|
"censor_type": {"type": "byte"},
|
||||||
|
"trending_score": {"type": "double"},
|
||||||
|
"release_time": {"type": "long"}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
FIELDS = {
|
||||||
|
'_id',
|
||||||
|
'claim_id', 'claim_type', 'claim_name', 'normalized_name',
|
||||||
|
'tx_id', 'tx_nout', 'tx_position',
|
||||||
|
'short_url', 'canonical_url',
|
||||||
|
'is_controlling', 'last_take_over_height',
|
||||||
|
'public_key_bytes', 'public_key_id', 'claims_in_channel', 'channel_join_height',
|
||||||
|
'channel_id', 'signature', 'signature_digest', 'is_signature_valid',
|
||||||
|
'amount', 'effective_amount', 'support_amount',
|
||||||
|
'fee_amount', 'fee_currency',
|
||||||
|
'height', 'creation_height', 'activation_height', 'expiration_height',
|
||||||
|
'stream_type', 'media_type', 'censor_type',
|
||||||
|
'title', 'author', 'description',
|
||||||
|
'timestamp', 'creation_timestamp',
|
||||||
|
'duration', 'release_time',
|
||||||
|
'tags', 'languages', 'has_source', 'reposted_claim_type',
|
||||||
|
'reposted_claim_id', 'repost_count',
|
||||||
|
'trending_score', 'tx_num'
|
||||||
|
}
|
||||||
|
|
||||||
|
TEXT_FIELDS = {'author', 'canonical_url', 'channel_id', 'description', 'claim_id', 'censoring_channel_id',
|
||||||
|
'media_type', 'normalized_name', 'public_key_bytes', 'public_key_id', 'short_url', 'signature',
|
||||||
|
'claim_name', 'signature_digest', 'title', 'tx_id', 'fee_currency', 'reposted_claim_id',
|
||||||
|
'tags'}
|
||||||
|
|
||||||
|
RANGE_FIELDS = {
|
||||||
|
'height', 'creation_height', 'activation_height', 'expiration_height',
|
||||||
|
'timestamp', 'creation_timestamp', 'duration', 'release_time', 'fee_amount',
|
||||||
|
'tx_position', 'channel_join', 'repost_count', 'limit_claims_per_channel',
|
||||||
|
'amount', 'effective_amount', 'support_amount',
|
||||||
|
'trending_score', 'censor_type', 'tx_num'
|
||||||
|
}
|
||||||
|
|
||||||
|
ALL_FIELDS = RANGE_FIELDS | TEXT_FIELDS | FIELDS
|
||||||
|
|
||||||
|
REPLACEMENTS = {
|
||||||
|
'claim_name': 'normalized_name',
|
||||||
|
'name': 'normalized_name',
|
||||||
|
'txid': 'tx_id',
|
||||||
|
'nout': 'tx_nout',
|
||||||
|
'trending_mixed': 'trending_score',
|
||||||
|
'reposted': 'repost_count',
|
||||||
|
'stream_types': 'stream_type',
|
||||||
|
'media_types': 'media_type',
|
||||||
|
'valid_channel_signature': 'is_signature_valid'
|
||||||
|
}
|
649
lbry/wallet/server/db/elasticsearch/search.py
Normal file
649
lbry/wallet/server/db/elasticsearch/search.py
Normal file
|
@ -0,0 +1,649 @@
|
||||||
|
import time
|
||||||
|
import asyncio
|
||||||
|
import struct
|
||||||
|
from binascii import unhexlify
|
||||||
|
from collections import Counter, deque
|
||||||
|
from decimal import Decimal
|
||||||
|
from operator import itemgetter
|
||||||
|
from typing import Optional, List, Iterable, Union
|
||||||
|
|
||||||
|
from elasticsearch import AsyncElasticsearch, NotFoundError, ConnectionError
|
||||||
|
from elasticsearch.helpers import async_streaming_bulk
|
||||||
|
from lbry.error import ResolveCensoredError, TooManyClaimSearchParametersError
|
||||||
|
from lbry.schema.result import Outputs, Censor
|
||||||
|
from lbry.schema.tags import clean_tags
|
||||||
|
from lbry.schema.url import URL, normalize_name
|
||||||
|
from lbry.utils import LRUCache
|
||||||
|
from lbry.wallet.server.db.common import CLAIM_TYPES, STREAM_TYPES
|
||||||
|
from lbry.wallet.server.db.elasticsearch.constants import INDEX_DEFAULT_SETTINGS, REPLACEMENTS, FIELDS, TEXT_FIELDS, \
|
||||||
|
RANGE_FIELDS, ALL_FIELDS
|
||||||
|
from lbry.wallet.server.util import class_logger
|
||||||
|
from lbry.wallet.server.db.common import ResolveResult
|
||||||
|
|
||||||
|
|
||||||
|
class ChannelResolution(str):
|
||||||
|
@classmethod
|
||||||
|
def lookup_error(cls, url):
|
||||||
|
return LookupError(f'Could not find channel in "{url}".')
|
||||||
|
|
||||||
|
|
||||||
|
class StreamResolution(str):
|
||||||
|
@classmethod
|
||||||
|
def lookup_error(cls, url):
|
||||||
|
return LookupError(f'Could not find claim at "{url}".')
|
||||||
|
|
||||||
|
|
||||||
|
class IndexVersionMismatch(Exception):
|
||||||
|
def __init__(self, got_version, expected_version):
|
||||||
|
self.got_version = got_version
|
||||||
|
self.expected_version = expected_version
|
||||||
|
|
||||||
|
|
||||||
|
class SearchIndex:
|
||||||
|
VERSION = 1
|
||||||
|
|
||||||
|
def __init__(self, index_prefix: str, search_timeout=3.0, elastic_host='localhost', elastic_port=9200,
|
||||||
|
half_life=0.4, whale_threshold=10000, whale_half_life=0.99):
|
||||||
|
self.search_timeout = search_timeout
|
||||||
|
self.sync_timeout = 600 # wont hit that 99% of the time, but can hit on a fresh import
|
||||||
|
self.search_client: Optional[AsyncElasticsearch] = None
|
||||||
|
self.sync_client: Optional[AsyncElasticsearch] = None
|
||||||
|
self.index = index_prefix + 'claims'
|
||||||
|
self.logger = class_logger(__name__, self.__class__.__name__)
|
||||||
|
self.claim_cache = LRUCache(2 ** 15)
|
||||||
|
self.search_cache = LRUCache(2 ** 17)
|
||||||
|
self._elastic_host = elastic_host
|
||||||
|
self._elastic_port = elastic_port
|
||||||
|
self._trending_half_life = half_life
|
||||||
|
self._trending_whale_threshold = whale_threshold
|
||||||
|
self._trending_whale_half_life = whale_half_life
|
||||||
|
|
||||||
|
async def get_index_version(self) -> int:
|
||||||
|
try:
|
||||||
|
template = await self.sync_client.indices.get_template(self.index)
|
||||||
|
return template[self.index]['version']
|
||||||
|
except NotFoundError:
|
||||||
|
return 0
|
||||||
|
|
||||||
|
async def set_index_version(self, version):
|
||||||
|
await self.sync_client.indices.put_template(
|
||||||
|
self.index, body={'version': version, 'index_patterns': ['ignored']}, ignore=400
|
||||||
|
)
|
||||||
|
|
||||||
|
async def start(self) -> bool:
|
||||||
|
if self.sync_client:
|
||||||
|
return False
|
||||||
|
hosts = [{'host': self._elastic_host, 'port': self._elastic_port}]
|
||||||
|
self.sync_client = AsyncElasticsearch(hosts, timeout=self.sync_timeout)
|
||||||
|
self.search_client = AsyncElasticsearch(hosts, timeout=self.search_timeout)
|
||||||
|
while True:
|
||||||
|
try:
|
||||||
|
await self.sync_client.cluster.health(wait_for_status='yellow')
|
||||||
|
break
|
||||||
|
except ConnectionError:
|
||||||
|
self.logger.warning("Failed to connect to Elasticsearch. Waiting for it!")
|
||||||
|
await asyncio.sleep(1)
|
||||||
|
|
||||||
|
res = await self.sync_client.indices.create(self.index, INDEX_DEFAULT_SETTINGS, ignore=400)
|
||||||
|
acked = res.get('acknowledged', False)
|
||||||
|
if acked:
|
||||||
|
await self.set_index_version(self.VERSION)
|
||||||
|
return acked
|
||||||
|
index_version = await self.get_index_version()
|
||||||
|
if index_version != self.VERSION:
|
||||||
|
self.logger.error("es search index has an incompatible version: %s vs %s", index_version, self.VERSION)
|
||||||
|
raise IndexVersionMismatch(index_version, self.VERSION)
|
||||||
|
await self.sync_client.indices.refresh(self.index)
|
||||||
|
return acked
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
clients = [self.sync_client, self.search_client]
|
||||||
|
self.sync_client, self.search_client = None, None
|
||||||
|
return asyncio.ensure_future(asyncio.gather(*(client.close() for client in clients)))
|
||||||
|
|
||||||
|
def delete_index(self):
|
||||||
|
return self.sync_client.indices.delete(self.index, ignore_unavailable=True)
|
||||||
|
|
||||||
|
async def _consume_claim_producer(self, claim_producer):
|
||||||
|
count = 0
|
||||||
|
async for op, doc in claim_producer:
|
||||||
|
if op == 'delete':
|
||||||
|
yield {
|
||||||
|
'_index': self.index,
|
||||||
|
'_op_type': 'delete',
|
||||||
|
'_id': doc
|
||||||
|
}
|
||||||
|
else:
|
||||||
|
yield {
|
||||||
|
'doc': {key: value for key, value in doc.items() if key in ALL_FIELDS},
|
||||||
|
'_id': doc['claim_id'],
|
||||||
|
'_index': self.index,
|
||||||
|
'_op_type': 'update',
|
||||||
|
'doc_as_upsert': True
|
||||||
|
}
|
||||||
|
count += 1
|
||||||
|
if count % 100 == 0:
|
||||||
|
self.logger.info("Indexing in progress, %d claims.", count)
|
||||||
|
if count:
|
||||||
|
self.logger.info("Indexing done for %d claims.", count)
|
||||||
|
else:
|
||||||
|
self.logger.debug("Indexing done for %d claims.", count)
|
||||||
|
|
||||||
|
async def claim_consumer(self, claim_producer):
|
||||||
|
touched = set()
|
||||||
|
async for ok, item in async_streaming_bulk(self.sync_client, self._consume_claim_producer(claim_producer),
|
||||||
|
raise_on_error=False):
|
||||||
|
if not ok:
|
||||||
|
self.logger.warning("indexing failed for an item: %s", item)
|
||||||
|
else:
|
||||||
|
item = item.popitem()[1]
|
||||||
|
touched.add(item['_id'])
|
||||||
|
await self.sync_client.indices.refresh(self.index)
|
||||||
|
self.logger.debug("Indexing done.")
|
||||||
|
|
||||||
|
def update_filter_query(self, censor_type, blockdict, channels=False):
|
||||||
|
blockdict = {blocked.hex(): blocker.hex() for blocked, blocker in blockdict.items()}
|
||||||
|
if channels:
|
||||||
|
update = expand_query(channel_id__in=list(blockdict.keys()), censor_type=f"<{censor_type}")
|
||||||
|
else:
|
||||||
|
update = expand_query(claim_id__in=list(blockdict.keys()), censor_type=f"<{censor_type}")
|
||||||
|
key = 'channel_id' if channels else 'claim_id'
|
||||||
|
update['script'] = {
|
||||||
|
"source": f"ctx._source.censor_type={censor_type}; "
|
||||||
|
f"ctx._source.censoring_channel_id=params[ctx._source.{key}];",
|
||||||
|
"lang": "painless",
|
||||||
|
"params": blockdict
|
||||||
|
}
|
||||||
|
return update
|
||||||
|
|
||||||
|
async def update_trending_score(self, params):
|
||||||
|
update_trending_score_script = """
|
||||||
|
double softenLBC(double lbc) { Math.pow(lbc, 1.0f / 3.0f) }
|
||||||
|
double inflateUnits(int height) {
|
||||||
|
int renormalizationPeriod = 100000;
|
||||||
|
double doublingRate = 400.0f;
|
||||||
|
Math.pow(2.0, (height % renormalizationPeriod) / doublingRate)
|
||||||
|
}
|
||||||
|
double spikePower(double newAmount) {
|
||||||
|
if (newAmount < 50.0) {
|
||||||
|
0.5
|
||||||
|
} else if (newAmount < 85.0) {
|
||||||
|
newAmount / 100.0
|
||||||
|
} else {
|
||||||
|
0.85
|
||||||
|
}
|
||||||
|
}
|
||||||
|
double spikeMass(double oldAmount, double newAmount) {
|
||||||
|
double softenedChange = softenLBC(Math.abs(newAmount - oldAmount));
|
||||||
|
double changeInSoftened = Math.abs(softenLBC(newAmount) - softenLBC(oldAmount));
|
||||||
|
double power = spikePower(newAmount);
|
||||||
|
if (oldAmount > newAmount) {
|
||||||
|
-1.0 * Math.pow(changeInSoftened, power) * Math.pow(softenedChange, 1.0 - power)
|
||||||
|
} else {
|
||||||
|
Math.pow(changeInSoftened, power) * Math.pow(softenedChange, 1.0 - power)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (i in params.src.changes) {
|
||||||
|
double units = inflateUnits(i.height);
|
||||||
|
if (i.added) {
|
||||||
|
if (ctx._source.trending_score == null) {
|
||||||
|
ctx._source.trending_score = (units * spikeMass(i.prev_amount, i.prev_amount + i.new_amount));
|
||||||
|
} else {
|
||||||
|
ctx._source.trending_score += (units * spikeMass(i.prev_amount, i.prev_amount + i.new_amount));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (ctx._source.trending_score == null) {
|
||||||
|
ctx._source.trending_score = (units * spikeMass(i.prev_amount, i.prev_amount - i.new_amount));
|
||||||
|
} else {
|
||||||
|
ctx._source.trending_score += (units * spikeMass(i.prev_amount, i.prev_amount - i.new_amount));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
start = time.perf_counter()
|
||||||
|
|
||||||
|
def producer():
|
||||||
|
for claim_id, claim_updates in params.items():
|
||||||
|
yield {
|
||||||
|
'_id': claim_id,
|
||||||
|
'_index': self.index,
|
||||||
|
'_op_type': 'update',
|
||||||
|
'script': {
|
||||||
|
'lang': 'painless',
|
||||||
|
'source': update_trending_score_script,
|
||||||
|
'params': {'src': {
|
||||||
|
'changes': [
|
||||||
|
{
|
||||||
|
'height': p.height,
|
||||||
|
'added': p.added,
|
||||||
|
'prev_amount': p.prev_amount * 1E-9,
|
||||||
|
'new_amount': p.new_amount * 1E-9,
|
||||||
|
} for p in claim_updates
|
||||||
|
]
|
||||||
|
}}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
if not params:
|
||||||
|
return
|
||||||
|
async for ok, item in async_streaming_bulk(self.sync_client, producer(), raise_on_error=False):
|
||||||
|
if not ok:
|
||||||
|
self.logger.warning("updating trending failed for an item: %s", item)
|
||||||
|
await self.sync_client.indices.refresh(self.index)
|
||||||
|
self.logger.info("updated trending scores in %ims", int((time.perf_counter() - start) * 1000))
|
||||||
|
|
||||||
|
async def apply_filters(self, blocked_streams, blocked_channels, filtered_streams, filtered_channels):
|
||||||
|
if filtered_streams:
|
||||||
|
await self.sync_client.update_by_query(
|
||||||
|
self.index, body=self.update_filter_query(Censor.SEARCH, filtered_streams), slices=4)
|
||||||
|
await self.sync_client.indices.refresh(self.index)
|
||||||
|
if filtered_channels:
|
||||||
|
await self.sync_client.update_by_query(
|
||||||
|
self.index, body=self.update_filter_query(Censor.SEARCH, filtered_channels), slices=4)
|
||||||
|
await self.sync_client.indices.refresh(self.index)
|
||||||
|
await self.sync_client.update_by_query(
|
||||||
|
self.index, body=self.update_filter_query(Censor.SEARCH, filtered_channels, True), slices=4)
|
||||||
|
await self.sync_client.indices.refresh(self.index)
|
||||||
|
if blocked_streams:
|
||||||
|
await self.sync_client.update_by_query(
|
||||||
|
self.index, body=self.update_filter_query(Censor.RESOLVE, blocked_streams), slices=4)
|
||||||
|
await self.sync_client.indices.refresh(self.index)
|
||||||
|
if blocked_channels:
|
||||||
|
await self.sync_client.update_by_query(
|
||||||
|
self.index, body=self.update_filter_query(Censor.RESOLVE, blocked_channels), slices=4)
|
||||||
|
await self.sync_client.indices.refresh(self.index)
|
||||||
|
await self.sync_client.update_by_query(
|
||||||
|
self.index, body=self.update_filter_query(Censor.RESOLVE, blocked_channels, True), slices=4)
|
||||||
|
await self.sync_client.indices.refresh(self.index)
|
||||||
|
self.clear_caches()
|
||||||
|
|
||||||
|
def clear_caches(self):
|
||||||
|
self.search_cache.clear()
|
||||||
|
self.claim_cache.clear()
|
||||||
|
|
||||||
|
async def cached_search(self, kwargs):
|
||||||
|
total_referenced = []
|
||||||
|
cache_item = ResultCacheItem.from_cache(str(kwargs), self.search_cache)
|
||||||
|
if cache_item.result is not None:
|
||||||
|
return cache_item.result
|
||||||
|
async with cache_item.lock:
|
||||||
|
if cache_item.result:
|
||||||
|
return cache_item.result
|
||||||
|
censor = Censor(Censor.SEARCH)
|
||||||
|
if kwargs.get('no_totals'):
|
||||||
|
response, offset, total = await self.search(**kwargs, censor_type=Censor.NOT_CENSORED)
|
||||||
|
else:
|
||||||
|
response, offset, total = await self.search(**kwargs)
|
||||||
|
censor.apply(response)
|
||||||
|
total_referenced.extend(response)
|
||||||
|
|
||||||
|
if censor.censored:
|
||||||
|
response, _, _ = await self.search(**kwargs, censor_type=Censor.NOT_CENSORED)
|
||||||
|
total_referenced.extend(response)
|
||||||
|
response = [
|
||||||
|
ResolveResult(
|
||||||
|
name=r['claim_name'],
|
||||||
|
normalized_name=r['normalized_name'],
|
||||||
|
claim_hash=r['claim_hash'],
|
||||||
|
tx_num=r['tx_num'],
|
||||||
|
position=r['tx_nout'],
|
||||||
|
tx_hash=r['tx_hash'],
|
||||||
|
height=r['height'],
|
||||||
|
amount=r['amount'],
|
||||||
|
short_url=r['short_url'],
|
||||||
|
is_controlling=r['is_controlling'],
|
||||||
|
canonical_url=r['canonical_url'],
|
||||||
|
creation_height=r['creation_height'],
|
||||||
|
activation_height=r['activation_height'],
|
||||||
|
expiration_height=r['expiration_height'],
|
||||||
|
effective_amount=r['effective_amount'],
|
||||||
|
support_amount=r['support_amount'],
|
||||||
|
last_takeover_height=r['last_take_over_height'],
|
||||||
|
claims_in_channel=r['claims_in_channel'],
|
||||||
|
channel_hash=r['channel_hash'],
|
||||||
|
reposted_claim_hash=r['reposted_claim_hash'],
|
||||||
|
reposted=r['reposted'],
|
||||||
|
signature_valid=r['signature_valid']
|
||||||
|
) for r in response
|
||||||
|
]
|
||||||
|
extra = [
|
||||||
|
ResolveResult(
|
||||||
|
name=r['claim_name'],
|
||||||
|
normalized_name=r['normalized_name'],
|
||||||
|
claim_hash=r['claim_hash'],
|
||||||
|
tx_num=r['tx_num'],
|
||||||
|
position=r['tx_nout'],
|
||||||
|
tx_hash=r['tx_hash'],
|
||||||
|
height=r['height'],
|
||||||
|
amount=r['amount'],
|
||||||
|
short_url=r['short_url'],
|
||||||
|
is_controlling=r['is_controlling'],
|
||||||
|
canonical_url=r['canonical_url'],
|
||||||
|
creation_height=r['creation_height'],
|
||||||
|
activation_height=r['activation_height'],
|
||||||
|
expiration_height=r['expiration_height'],
|
||||||
|
effective_amount=r['effective_amount'],
|
||||||
|
support_amount=r['support_amount'],
|
||||||
|
last_takeover_height=r['last_take_over_height'],
|
||||||
|
claims_in_channel=r['claims_in_channel'],
|
||||||
|
channel_hash=r['channel_hash'],
|
||||||
|
reposted_claim_hash=r['reposted_claim_hash'],
|
||||||
|
reposted=r['reposted'],
|
||||||
|
signature_valid=r['signature_valid']
|
||||||
|
) for r in await self._get_referenced_rows(total_referenced)
|
||||||
|
]
|
||||||
|
result = Outputs.to_base64(
|
||||||
|
response, extra, offset, total, censor
|
||||||
|
)
|
||||||
|
cache_item.result = result
|
||||||
|
return result
|
||||||
|
|
||||||
|
async def get_many(self, *claim_ids):
|
||||||
|
await self.populate_claim_cache(*claim_ids)
|
||||||
|
return filter(None, map(self.claim_cache.get, claim_ids))
|
||||||
|
|
||||||
|
async def populate_claim_cache(self, *claim_ids):
|
||||||
|
missing = [claim_id for claim_id in claim_ids if self.claim_cache.get(claim_id) is None]
|
||||||
|
if missing:
|
||||||
|
results = await self.search_client.mget(
|
||||||
|
index=self.index, body={"ids": missing}
|
||||||
|
)
|
||||||
|
for result in expand_result(filter(lambda doc: doc['found'], results["docs"])):
|
||||||
|
self.claim_cache.set(result['claim_id'], result)
|
||||||
|
|
||||||
|
|
||||||
|
async def search(self, **kwargs):
|
||||||
|
try:
|
||||||
|
return await self.search_ahead(**kwargs)
|
||||||
|
except NotFoundError:
|
||||||
|
return [], 0, 0
|
||||||
|
# return expand_result(result['hits']), 0, result.get('total', {}).get('value', 0)
|
||||||
|
|
||||||
|
async def search_ahead(self, **kwargs):
|
||||||
|
# 'limit_claims_per_channel' case. Fetch 1000 results, reorder, slice, inflate and return
|
||||||
|
per_channel_per_page = kwargs.pop('limit_claims_per_channel', 0) or 0
|
||||||
|
remove_duplicates = kwargs.pop('remove_duplicates', False)
|
||||||
|
page_size = kwargs.pop('limit', 10)
|
||||||
|
offset = kwargs.pop('offset', 0)
|
||||||
|
kwargs['limit'] = 1000
|
||||||
|
cache_item = ResultCacheItem.from_cache(f"ahead{per_channel_per_page}{kwargs}", self.search_cache)
|
||||||
|
if cache_item.result is not None:
|
||||||
|
reordered_hits = cache_item.result
|
||||||
|
else:
|
||||||
|
async with cache_item.lock:
|
||||||
|
if cache_item.result:
|
||||||
|
reordered_hits = cache_item.result
|
||||||
|
else:
|
||||||
|
query = expand_query(**kwargs)
|
||||||
|
search_hits = deque((await self.search_client.search(
|
||||||
|
query, index=self.index, track_total_hits=False,
|
||||||
|
_source_includes=['_id', 'channel_id', 'reposted_claim_id', 'creation_height']
|
||||||
|
))['hits']['hits'])
|
||||||
|
if remove_duplicates:
|
||||||
|
search_hits = self.__remove_duplicates(search_hits)
|
||||||
|
if per_channel_per_page > 0:
|
||||||
|
reordered_hits = self.__search_ahead(search_hits, page_size, per_channel_per_page)
|
||||||
|
else:
|
||||||
|
reordered_hits = [(hit['_id'], hit['_source']['channel_id']) for hit in search_hits]
|
||||||
|
cache_item.result = reordered_hits
|
||||||
|
result = list(await self.get_many(*(claim_id for claim_id, _ in reordered_hits[offset:(offset + page_size)])))
|
||||||
|
return result, 0, len(reordered_hits)
|
||||||
|
|
||||||
|
def __remove_duplicates(self, search_hits: deque) -> deque:
|
||||||
|
known_ids = {} # claim_id -> (creation_height, hit_id), where hit_id is either reposted claim id or original
|
||||||
|
dropped = set()
|
||||||
|
for hit in search_hits:
|
||||||
|
hit_height, hit_id = hit['_source']['creation_height'], hit['_source']['reposted_claim_id'] or hit['_id']
|
||||||
|
if hit_id not in known_ids:
|
||||||
|
known_ids[hit_id] = (hit_height, hit['_id'])
|
||||||
|
else:
|
||||||
|
previous_height, previous_id = known_ids[hit_id]
|
||||||
|
if hit_height < previous_height:
|
||||||
|
known_ids[hit_id] = (hit_height, hit['_id'])
|
||||||
|
dropped.add(previous_id)
|
||||||
|
else:
|
||||||
|
dropped.add(hit['_id'])
|
||||||
|
return deque(hit for hit in search_hits if hit['_id'] not in dropped)
|
||||||
|
|
||||||
|
def __search_ahead(self, search_hits: list, page_size: int, per_channel_per_page: int):
|
||||||
|
reordered_hits = []
|
||||||
|
channel_counters = Counter()
|
||||||
|
next_page_hits_maybe_check_later = deque()
|
||||||
|
while search_hits or next_page_hits_maybe_check_later:
|
||||||
|
if reordered_hits and len(reordered_hits) % page_size == 0:
|
||||||
|
channel_counters.clear()
|
||||||
|
elif not reordered_hits:
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
break # means last page was incomplete and we are left with bad replacements
|
||||||
|
for _ in range(len(next_page_hits_maybe_check_later)):
|
||||||
|
claim_id, channel_id = next_page_hits_maybe_check_later.popleft()
|
||||||
|
if per_channel_per_page > 0 and channel_counters[channel_id] < per_channel_per_page:
|
||||||
|
reordered_hits.append((claim_id, channel_id))
|
||||||
|
channel_counters[channel_id] += 1
|
||||||
|
else:
|
||||||
|
next_page_hits_maybe_check_later.append((claim_id, channel_id))
|
||||||
|
while search_hits:
|
||||||
|
hit = search_hits.popleft()
|
||||||
|
hit_id, hit_channel_id = hit['_id'], hit['_source']['channel_id']
|
||||||
|
if hit_channel_id is None or per_channel_per_page <= 0:
|
||||||
|
reordered_hits.append((hit_id, hit_channel_id))
|
||||||
|
elif channel_counters[hit_channel_id] < per_channel_per_page:
|
||||||
|
reordered_hits.append((hit_id, hit_channel_id))
|
||||||
|
channel_counters[hit_channel_id] += 1
|
||||||
|
if len(reordered_hits) % page_size == 0:
|
||||||
|
break
|
||||||
|
else:
|
||||||
|
next_page_hits_maybe_check_later.append((hit_id, hit_channel_id))
|
||||||
|
return reordered_hits
|
||||||
|
|
||||||
|
async def _get_referenced_rows(self, txo_rows: List[dict]):
|
||||||
|
txo_rows = [row for row in txo_rows if isinstance(row, dict)]
|
||||||
|
referenced_ids = set(filter(None, map(itemgetter('reposted_claim_id'), txo_rows)))
|
||||||
|
referenced_ids |= set(filter(None, (row['channel_id'] for row in txo_rows)))
|
||||||
|
referenced_ids |= set(filter(None, (row['censoring_channel_id'] for row in txo_rows)))
|
||||||
|
|
||||||
|
referenced_txos = []
|
||||||
|
if referenced_ids:
|
||||||
|
referenced_txos.extend(await self.get_many(*referenced_ids))
|
||||||
|
referenced_ids = set(filter(None, (row['channel_id'] for row in referenced_txos)))
|
||||||
|
|
||||||
|
if referenced_ids:
|
||||||
|
referenced_txos.extend(await self.get_many(*referenced_ids))
|
||||||
|
|
||||||
|
return referenced_txos
|
||||||
|
|
||||||
|
|
||||||
|
def expand_query(**kwargs):
|
||||||
|
if "amount_order" in kwargs:
|
||||||
|
kwargs["limit"] = 1
|
||||||
|
kwargs["order_by"] = "effective_amount"
|
||||||
|
kwargs["offset"] = int(kwargs["amount_order"]) - 1
|
||||||
|
if 'name' in kwargs:
|
||||||
|
kwargs['name'] = normalize_name(kwargs.pop('name'))
|
||||||
|
if kwargs.get('is_controlling') is False:
|
||||||
|
kwargs.pop('is_controlling')
|
||||||
|
query = {'must': [], 'must_not': []}
|
||||||
|
collapse = None
|
||||||
|
if 'fee_currency' in kwargs and kwargs['fee_currency'] is not None:
|
||||||
|
kwargs['fee_currency'] = kwargs['fee_currency'].upper()
|
||||||
|
for key, value in kwargs.items():
|
||||||
|
key = key.replace('claim.', '')
|
||||||
|
many = key.endswith('__in') or isinstance(value, list)
|
||||||
|
if many and len(value) > 2048:
|
||||||
|
raise TooManyClaimSearchParametersError(key, 2048)
|
||||||
|
if many:
|
||||||
|
key = key.replace('__in', '')
|
||||||
|
value = list(filter(None, value))
|
||||||
|
if value is None or isinstance(value, list) and len(value) == 0:
|
||||||
|
continue
|
||||||
|
key = REPLACEMENTS.get(key, key)
|
||||||
|
if key in FIELDS:
|
||||||
|
partial_id = False
|
||||||
|
if key == 'claim_type':
|
||||||
|
if isinstance(value, str):
|
||||||
|
value = CLAIM_TYPES[value]
|
||||||
|
else:
|
||||||
|
value = [CLAIM_TYPES[claim_type] for claim_type in value]
|
||||||
|
elif key == 'stream_type':
|
||||||
|
value = [STREAM_TYPES[value]] if isinstance(value, str) else list(map(STREAM_TYPES.get, value))
|
||||||
|
if key == '_id':
|
||||||
|
if isinstance(value, Iterable):
|
||||||
|
value = [item[::-1].hex() for item in value]
|
||||||
|
else:
|
||||||
|
value = value[::-1].hex()
|
||||||
|
if not many and key in ('_id', 'claim_id') and len(value) < 20:
|
||||||
|
partial_id = True
|
||||||
|
if key in ('signature_valid', 'has_source'):
|
||||||
|
continue # handled later
|
||||||
|
if key in TEXT_FIELDS:
|
||||||
|
key += '.keyword'
|
||||||
|
ops = {'<=': 'lte', '>=': 'gte', '<': 'lt', '>': 'gt'}
|
||||||
|
if partial_id:
|
||||||
|
query['must'].append({"prefix": {"claim_id": value}})
|
||||||
|
elif key in RANGE_FIELDS and isinstance(value, str) and value[0] in ops:
|
||||||
|
operator_length = 2 if value[:2] in ops else 1
|
||||||
|
operator, value = value[:operator_length], value[operator_length:]
|
||||||
|
if key == 'fee_amount':
|
||||||
|
value = str(Decimal(value)*1000)
|
||||||
|
query['must'].append({"range": {key: {ops[operator]: value}}})
|
||||||
|
elif many:
|
||||||
|
query['must'].append({"terms": {key: value}})
|
||||||
|
else:
|
||||||
|
if key == 'fee_amount':
|
||||||
|
value = str(Decimal(value)*1000)
|
||||||
|
query['must'].append({"term": {key: {"value": value}}})
|
||||||
|
elif key == 'not_channel_ids':
|
||||||
|
for channel_id in value:
|
||||||
|
query['must_not'].append({"term": {'channel_id.keyword': channel_id}})
|
||||||
|
query['must_not'].append({"term": {'_id': channel_id}})
|
||||||
|
elif key == 'channel_ids':
|
||||||
|
query['must'].append({"terms": {'channel_id.keyword': value}})
|
||||||
|
elif key == 'claim_ids':
|
||||||
|
query['must'].append({"terms": {'claim_id.keyword': value}})
|
||||||
|
elif key == 'media_types':
|
||||||
|
query['must'].append({"terms": {'media_type.keyword': value}})
|
||||||
|
elif key == 'any_languages':
|
||||||
|
query['must'].append({"terms": {'languages': clean_tags(value)}})
|
||||||
|
elif key == 'any_languages':
|
||||||
|
query['must'].append({"terms": {'languages': value}})
|
||||||
|
elif key == 'all_languages':
|
||||||
|
query['must'].extend([{"term": {'languages': tag}} for tag in value])
|
||||||
|
elif key == 'any_tags':
|
||||||
|
query['must'].append({"terms": {'tags.keyword': clean_tags(value)}})
|
||||||
|
elif key == 'all_tags':
|
||||||
|
query['must'].extend([{"term": {'tags.keyword': tag}} for tag in clean_tags(value)])
|
||||||
|
elif key == 'not_tags':
|
||||||
|
query['must_not'].extend([{"term": {'tags.keyword': tag}} for tag in clean_tags(value)])
|
||||||
|
elif key == 'not_claim_id':
|
||||||
|
query['must_not'].extend([{"term": {'claim_id.keyword': cid}} for cid in value])
|
||||||
|
elif key == 'limit_claims_per_channel':
|
||||||
|
collapse = ('channel_id.keyword', value)
|
||||||
|
if kwargs.get('has_channel_signature'):
|
||||||
|
query['must'].append({"exists": {"field": "signature"}})
|
||||||
|
if 'signature_valid' in kwargs:
|
||||||
|
query['must'].append({"term": {"is_signature_valid": bool(kwargs["signature_valid"])}})
|
||||||
|
elif 'signature_valid' in kwargs:
|
||||||
|
query.setdefault('should', [])
|
||||||
|
query["minimum_should_match"] = 1
|
||||||
|
query['should'].append({"bool": {"must_not": {"exists": {"field": "signature"}}}})
|
||||||
|
query['should'].append({"term": {"is_signature_valid": bool(kwargs["signature_valid"])}})
|
||||||
|
if 'has_source' in kwargs:
|
||||||
|
query.setdefault('should', [])
|
||||||
|
query["minimum_should_match"] = 1
|
||||||
|
is_stream_or_repost = {"terms": {"claim_type": [CLAIM_TYPES['stream'], CLAIM_TYPES['repost']]}}
|
||||||
|
query['should'].append(
|
||||||
|
{"bool": {"must": [{"match": {"has_source": kwargs['has_source']}}, is_stream_or_repost]}})
|
||||||
|
query['should'].append({"bool": {"must_not": [is_stream_or_repost]}})
|
||||||
|
query['should'].append({"bool": {"must": [{"term": {"reposted_claim_type": CLAIM_TYPES['channel']}}]}})
|
||||||
|
if kwargs.get('text'):
|
||||||
|
query['must'].append(
|
||||||
|
{"simple_query_string":
|
||||||
|
{"query": kwargs["text"], "fields": [
|
||||||
|
"claim_name^4", "channel_name^8", "title^1", "description^.5", "author^1", "tags^.5"
|
||||||
|
]}})
|
||||||
|
query = {
|
||||||
|
"_source": {"excludes": ["description", "title"]},
|
||||||
|
'query': {'bool': query},
|
||||||
|
"sort": [],
|
||||||
|
}
|
||||||
|
if "limit" in kwargs:
|
||||||
|
query["size"] = kwargs["limit"]
|
||||||
|
if 'offset' in kwargs:
|
||||||
|
query["from"] = kwargs["offset"]
|
||||||
|
if 'order_by' in kwargs:
|
||||||
|
if isinstance(kwargs["order_by"], str):
|
||||||
|
kwargs["order_by"] = [kwargs["order_by"]]
|
||||||
|
for value in kwargs['order_by']:
|
||||||
|
if 'trending_group' in value:
|
||||||
|
# fixme: trending_mixed is 0 for all records on variable decay, making sort slow.
|
||||||
|
continue
|
||||||
|
is_asc = value.startswith('^')
|
||||||
|
value = value[1:] if is_asc else value
|
||||||
|
value = REPLACEMENTS.get(value, value)
|
||||||
|
if value in TEXT_FIELDS:
|
||||||
|
value += '.keyword'
|
||||||
|
query['sort'].append({value: "asc" if is_asc else "desc"})
|
||||||
|
if collapse:
|
||||||
|
query["collapse"] = {
|
||||||
|
"field": collapse[0],
|
||||||
|
"inner_hits": {
|
||||||
|
"name": collapse[0],
|
||||||
|
"size": collapse[1],
|
||||||
|
"sort": query["sort"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return query
|
||||||
|
|
||||||
|
|
||||||
|
def expand_result(results):
|
||||||
|
inner_hits = []
|
||||||
|
expanded = []
|
||||||
|
for result in results:
|
||||||
|
if result.get("inner_hits"):
|
||||||
|
for _, inner_hit in result["inner_hits"].items():
|
||||||
|
inner_hits.extend(inner_hit["hits"]["hits"])
|
||||||
|
continue
|
||||||
|
result = result['_source']
|
||||||
|
result['claim_hash'] = unhexlify(result['claim_id'])[::-1]
|
||||||
|
if result['reposted_claim_id']:
|
||||||
|
result['reposted_claim_hash'] = unhexlify(result['reposted_claim_id'])[::-1]
|
||||||
|
else:
|
||||||
|
result['reposted_claim_hash'] = None
|
||||||
|
result['channel_hash'] = unhexlify(result['channel_id'])[::-1] if result['channel_id'] else None
|
||||||
|
result['txo_hash'] = unhexlify(result['tx_id'])[::-1] + struct.pack('<I', result['tx_nout'])
|
||||||
|
result['tx_hash'] = unhexlify(result['tx_id'])[::-1]
|
||||||
|
result['reposted'] = result.pop('repost_count')
|
||||||
|
result['signature_valid'] = result.pop('is_signature_valid')
|
||||||
|
# result['normalized'] = result.pop('normalized_name')
|
||||||
|
# if result['censoring_channel_hash']:
|
||||||
|
# result['censoring_channel_hash'] = unhexlify(result['censoring_channel_hash'])[::-1]
|
||||||
|
expanded.append(result)
|
||||||
|
if inner_hits:
|
||||||
|
return expand_result(inner_hits)
|
||||||
|
return expanded
|
||||||
|
|
||||||
|
|
||||||
|
class ResultCacheItem:
|
||||||
|
__slots__ = '_result', 'lock', 'has_result'
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
self.has_result = asyncio.Event()
|
||||||
|
self.lock = asyncio.Lock()
|
||||||
|
self._result = None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def result(self) -> str:
|
||||||
|
return self._result
|
||||||
|
|
||||||
|
@result.setter
|
||||||
|
def result(self, result: str):
|
||||||
|
self._result = result
|
||||||
|
if result is not None:
|
||||||
|
self.has_result.set()
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_cache(cls, cache_key, cache):
|
||||||
|
cache_item = cache.get(cache_key)
|
||||||
|
if cache_item is None:
|
||||||
|
cache_item = cache[cache_key] = ResultCacheItem()
|
||||||
|
return cache_item
|
88
lbry/wallet/server/db/elasticsearch/sync.py
Normal file
88
lbry/wallet/server/db/elasticsearch/sync.py
Normal file
|
@ -0,0 +1,88 @@
|
||||||
|
import argparse
|
||||||
|
import asyncio
|
||||||
|
import logging
|
||||||
|
from elasticsearch import AsyncElasticsearch
|
||||||
|
from elasticsearch.helpers import async_bulk
|
||||||
|
from lbry.wallet.server.env import Env
|
||||||
|
from lbry.wallet.server.coin import LBC
|
||||||
|
from lbry.wallet.server.leveldb import LevelDB
|
||||||
|
from lbry.wallet.server.db.elasticsearch.search import SearchIndex, IndexVersionMismatch
|
||||||
|
from lbry.wallet.server.db.elasticsearch.constants import ALL_FIELDS
|
||||||
|
|
||||||
|
|
||||||
|
async def get_all_claims(index_name='claims', db=None):
|
||||||
|
env = Env(LBC)
|
||||||
|
need_open = db is None
|
||||||
|
db = db or LevelDB(env)
|
||||||
|
if need_open:
|
||||||
|
await db.open_dbs()
|
||||||
|
try:
|
||||||
|
cnt = 0
|
||||||
|
async for claim in db.all_claims_producer():
|
||||||
|
yield {
|
||||||
|
'doc': {key: value for key, value in claim.items() if key in ALL_FIELDS},
|
||||||
|
'_id': claim['claim_id'],
|
||||||
|
'_index': index_name,
|
||||||
|
'_op_type': 'update',
|
||||||
|
'doc_as_upsert': True
|
||||||
|
}
|
||||||
|
cnt += 1
|
||||||
|
if cnt % 10000 == 0:
|
||||||
|
print(f"{cnt} claims sent")
|
||||||
|
finally:
|
||||||
|
if need_open:
|
||||||
|
db.close()
|
||||||
|
|
||||||
|
|
||||||
|
async def make_es_index(index=None):
|
||||||
|
env = Env(LBC)
|
||||||
|
if index is None:
|
||||||
|
index = SearchIndex('', elastic_host=env.elastic_host, elastic_port=env.elastic_port)
|
||||||
|
|
||||||
|
try:
|
||||||
|
return await index.start()
|
||||||
|
except IndexVersionMismatch as err:
|
||||||
|
logging.info(
|
||||||
|
"dropping ES search index (version %s) for upgrade to version %s", err.got_version, err.expected_version
|
||||||
|
)
|
||||||
|
await index.delete_index()
|
||||||
|
await index.stop()
|
||||||
|
return await index.start()
|
||||||
|
finally:
|
||||||
|
index.stop()
|
||||||
|
|
||||||
|
|
||||||
|
async def run_sync(index_name='claims', db=None, clients=32):
|
||||||
|
env = Env(LBC)
|
||||||
|
logging.info("ES sync host: %s:%i", env.elastic_host, env.elastic_port)
|
||||||
|
es = AsyncElasticsearch([{'host': env.elastic_host, 'port': env.elastic_port}])
|
||||||
|
claim_generator = get_all_claims(index_name=index_name, db=db)
|
||||||
|
|
||||||
|
try:
|
||||||
|
await async_bulk(es, claim_generator, request_timeout=600)
|
||||||
|
await es.indices.refresh(index=index_name)
|
||||||
|
finally:
|
||||||
|
await es.close()
|
||||||
|
|
||||||
|
|
||||||
|
def run_elastic_sync():
|
||||||
|
logging.basicConfig(level=logging.INFO)
|
||||||
|
logging.getLogger('aiohttp').setLevel(logging.WARNING)
|
||||||
|
logging.getLogger('elasticsearch').setLevel(logging.WARNING)
|
||||||
|
|
||||||
|
logging.info('lbry.server starting')
|
||||||
|
parser = argparse.ArgumentParser(prog="lbry-hub-elastic-sync")
|
||||||
|
# parser.add_argument("db_path", type=str)
|
||||||
|
parser.add_argument("-c", "--clients", type=int, default=32)
|
||||||
|
parser.add_argument("-b", "--blocks", type=int, default=0)
|
||||||
|
parser.add_argument("-f", "--force", default=False, action='store_true')
|
||||||
|
args = parser.parse_args()
|
||||||
|
|
||||||
|
# if not args.force and not os.path.exists(args.db_path):
|
||||||
|
# logging.info("DB path doesnt exist")
|
||||||
|
# return
|
||||||
|
|
||||||
|
if not args.force and not asyncio.run(make_es_index()):
|
||||||
|
logging.info("ES is already initialized")
|
||||||
|
return
|
||||||
|
asyncio.run(run_sync(clients=args.clients))
|
1652
lbry/wallet/server/db/prefixes.py
Normal file
1652
lbry/wallet/server/db/prefixes.py
Normal file
File diff suppressed because it is too large
Load diff
171
lbry/wallet/server/db/revertable.py
Normal file
171
lbry/wallet/server/db/revertable.py
Normal file
|
@ -0,0 +1,171 @@
|
||||||
|
import struct
|
||||||
|
import logging
|
||||||
|
from string import printable
|
||||||
|
from collections import defaultdict
|
||||||
|
from typing import Tuple, Iterable, Callable, Optional
|
||||||
|
from lbry.wallet.server.db import DB_PREFIXES
|
||||||
|
|
||||||
|
_OP_STRUCT = struct.Struct('>BLL')
|
||||||
|
log = logging.getLogger()
|
||||||
|
|
||||||
|
|
||||||
|
class RevertableOp:
|
||||||
|
__slots__ = [
|
||||||
|
'key',
|
||||||
|
'value',
|
||||||
|
]
|
||||||
|
is_put = 0
|
||||||
|
|
||||||
|
def __init__(self, key: bytes, value: bytes):
|
||||||
|
self.key = key
|
||||||
|
self.value = value
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_delete(self) -> bool:
|
||||||
|
return not self.is_put
|
||||||
|
|
||||||
|
def invert(self) -> 'RevertableOp':
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
def pack(self) -> bytes:
|
||||||
|
"""
|
||||||
|
Serialize to bytes
|
||||||
|
"""
|
||||||
|
return struct.pack(
|
||||||
|
f'>BLL{len(self.key)}s{len(self.value)}s', int(self.is_put), len(self.key), len(self.value), self.key,
|
||||||
|
self.value
|
||||||
|
)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def unpack(cls, packed: bytes) -> Tuple['RevertableOp', bytes]:
|
||||||
|
"""
|
||||||
|
Deserialize from bytes
|
||||||
|
|
||||||
|
:param packed: bytes containing at least one packed revertable op
|
||||||
|
:return: tuple of the deserialized op (a put or a delete) and the remaining serialized bytes
|
||||||
|
"""
|
||||||
|
is_put, key_len, val_len = _OP_STRUCT.unpack(packed[:9])
|
||||||
|
key = packed[9:9 + key_len]
|
||||||
|
value = packed[9 + key_len:9 + key_len + val_len]
|
||||||
|
if is_put == 1:
|
||||||
|
return RevertablePut(key, value), packed[9 + key_len + val_len:]
|
||||||
|
return RevertableDelete(key, value), packed[9 + key_len + val_len:]
|
||||||
|
|
||||||
|
def __eq__(self, other: 'RevertableOp') -> bool:
|
||||||
|
return (self.is_put, self.key, self.value) == (other.is_put, other.key, other.value)
|
||||||
|
|
||||||
|
def __repr__(self) -> str:
|
||||||
|
return str(self)
|
||||||
|
|
||||||
|
def __str__(self) -> str:
|
||||||
|
from lbry.wallet.server.db.prefixes import auto_decode_item
|
||||||
|
k, v = auto_decode_item(self.key, self.value)
|
||||||
|
key = ''.join(c if c in printable else '.' for c in str(k))
|
||||||
|
val = ''.join(c if c in printable else '.' for c in str(v))
|
||||||
|
return f"{'PUT' if self.is_put else 'DELETE'} {DB_PREFIXES(self.key[:1]).name}: {key} | {val}"
|
||||||
|
|
||||||
|
|
||||||
|
class RevertableDelete(RevertableOp):
|
||||||
|
def invert(self):
|
||||||
|
return RevertablePut(self.key, self.value)
|
||||||
|
|
||||||
|
|
||||||
|
class RevertablePut(RevertableOp):
|
||||||
|
is_put = True
|
||||||
|
|
||||||
|
def invert(self):
|
||||||
|
return RevertableDelete(self.key, self.value)
|
||||||
|
|
||||||
|
|
||||||
|
class OpStackIntegrity(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class RevertableOpStack:
|
||||||
|
def __init__(self, get_fn: Callable[[bytes], Optional[bytes]], unsafe_prefixes=None):
|
||||||
|
"""
|
||||||
|
This represents a sequence of revertable puts and deletes to a key-value database that checks for integrity
|
||||||
|
violations when applying the puts and deletes. The integrity checks assure that keys that do not exist
|
||||||
|
are not deleted, and that when keys are deleted the current value is correctly known so that the delete
|
||||||
|
may be undone. When putting values, the integrity checks assure that existing values are not overwritten
|
||||||
|
without first being deleted. Updates are performed by applying a delete op for the old value and a put op
|
||||||
|
for the new value.
|
||||||
|
|
||||||
|
:param get_fn: getter function from an object implementing `KeyValueStorage`
|
||||||
|
:param unsafe_prefixes: optional set of prefixes to ignore integrity errors for, violations are still logged
|
||||||
|
"""
|
||||||
|
self._get = get_fn
|
||||||
|
self._items = defaultdict(list)
|
||||||
|
self._unsafe_prefixes = unsafe_prefixes or set()
|
||||||
|
|
||||||
|
def append_op(self, op: RevertableOp):
|
||||||
|
"""
|
||||||
|
Apply a put or delete op, checking that it introduces no integrity errors
|
||||||
|
"""
|
||||||
|
|
||||||
|
inverted = op.invert()
|
||||||
|
if self._items[op.key] and inverted == self._items[op.key][-1]:
|
||||||
|
self._items[op.key].pop() # if the new op is the inverse of the last op, we can safely null both
|
||||||
|
return
|
||||||
|
elif self._items[op.key] and self._items[op.key][-1] == op: # duplicate of last op
|
||||||
|
return # raise an error?
|
||||||
|
stored_val = self._get(op.key)
|
||||||
|
has_stored_val = stored_val is not None
|
||||||
|
delete_stored_op = None if not has_stored_val else RevertableDelete(op.key, stored_val)
|
||||||
|
will_delete_existing_stored = False if delete_stored_op is None else (delete_stored_op in self._items[op.key])
|
||||||
|
try:
|
||||||
|
if op.is_put and has_stored_val and not will_delete_existing_stored:
|
||||||
|
raise OpStackIntegrity(
|
||||||
|
f"db op tries to add on top of existing key without deleting first: {op}"
|
||||||
|
)
|
||||||
|
elif op.is_delete and has_stored_val and stored_val != op.value and not will_delete_existing_stored:
|
||||||
|
# there is a value and we're not deleting it in this op
|
||||||
|
# check that a delete for the stored value is in the stack
|
||||||
|
raise OpStackIntegrity(f"delete {op}")
|
||||||
|
elif op.is_delete and not has_stored_val:
|
||||||
|
raise OpStackIntegrity(f"db op tries to delete nonexistent key: {op}")
|
||||||
|
elif op.is_delete and stored_val != op.value:
|
||||||
|
raise OpStackIntegrity(f"db op tries to delete with incorrect value: {op}")
|
||||||
|
except OpStackIntegrity as err:
|
||||||
|
if op.key[:1] in self._unsafe_prefixes:
|
||||||
|
log.error(f"skipping over integrity error: {err}")
|
||||||
|
else:
|
||||||
|
raise err
|
||||||
|
self._items[op.key].append(op)
|
||||||
|
|
||||||
|
def extend_ops(self, ops: Iterable[RevertableOp]):
|
||||||
|
"""
|
||||||
|
Apply a sequence of put or delete ops, checking that they introduce no integrity errors
|
||||||
|
"""
|
||||||
|
for op in ops:
|
||||||
|
self.append_op(op)
|
||||||
|
|
||||||
|
def clear(self):
|
||||||
|
self._items.clear()
|
||||||
|
|
||||||
|
def __len__(self):
|
||||||
|
return sum(map(len, self._items.values()))
|
||||||
|
|
||||||
|
def __iter__(self):
|
||||||
|
for key, ops in self._items.items():
|
||||||
|
for op in ops:
|
||||||
|
yield op
|
||||||
|
|
||||||
|
def __reversed__(self):
|
||||||
|
for key, ops in self._items.items():
|
||||||
|
for op in reversed(ops):
|
||||||
|
yield op
|
||||||
|
|
||||||
|
def get_undo_ops(self) -> bytes:
|
||||||
|
"""
|
||||||
|
Get the serialized bytes to undo all of the changes made by the pending ops
|
||||||
|
"""
|
||||||
|
return b''.join(op.invert().pack() for op in reversed(self))
|
||||||
|
|
||||||
|
def apply_packed_undo_ops(self, packed: bytes):
|
||||||
|
"""
|
||||||
|
Unpack and apply a sequence of undo ops from serialized undo bytes
|
||||||
|
"""
|
||||||
|
while packed:
|
||||||
|
op, packed = RevertableOp.unpack(packed)
|
||||||
|
self.append_op(op)
|
280
lbry/wallet/server/env.py
Normal file
280
lbry/wallet/server/env.py
Normal file
|
@ -0,0 +1,280 @@
|
||||||
|
# Copyright (c) 2016, Neil Booth
|
||||||
|
#
|
||||||
|
# All rights reserved.
|
||||||
|
#
|
||||||
|
# See the file "LICENCE" for information about the copyright
|
||||||
|
# and warranty status of this software.
|
||||||
|
|
||||||
|
import math
|
||||||
|
import re
|
||||||
|
import resource
|
||||||
|
from os import environ
|
||||||
|
from collections import namedtuple
|
||||||
|
from ipaddress import ip_address
|
||||||
|
|
||||||
|
from lbry.wallet.server.util import class_logger
|
||||||
|
from lbry.wallet.server.coin import Coin
|
||||||
|
import lbry.wallet.server.util as lib_util
|
||||||
|
|
||||||
|
|
||||||
|
NetIdentity = namedtuple('NetIdentity', 'host tcp_port ssl_port nick_suffix')
|
||||||
|
|
||||||
|
|
||||||
|
class Env:
|
||||||
|
|
||||||
|
# Peer discovery
|
||||||
|
PD_OFF, PD_SELF, PD_ON = range(3)
|
||||||
|
|
||||||
|
class Error(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def __init__(self, coin=None):
|
||||||
|
self.logger = class_logger(__name__, self.__class__.__name__)
|
||||||
|
self.allow_root = self.boolean('ALLOW_ROOT', False)
|
||||||
|
self.host = self.default('HOST', 'localhost')
|
||||||
|
self.rpc_host = self.default('RPC_HOST', 'localhost')
|
||||||
|
self.elastic_host = self.default('ELASTIC_HOST', 'localhost')
|
||||||
|
self.elastic_port = self.integer('ELASTIC_PORT', 9200)
|
||||||
|
self.loop_policy = self.set_event_loop_policy()
|
||||||
|
self.obsolete(['UTXO_MB', 'HIST_MB', 'NETWORK'])
|
||||||
|
self.db_dir = self.required('DB_DIRECTORY')
|
||||||
|
self.db_engine = self.default('DB_ENGINE', 'leveldb')
|
||||||
|
# self.trending_algorithms = [
|
||||||
|
# trending for trending in set(self.default('TRENDING_ALGORITHMS', 'zscore').split(' ')) if trending
|
||||||
|
# ]
|
||||||
|
self.trending_half_life = math.log2(0.1 ** (1 / (3 + self.integer('TRENDING_DECAY_RATE', 48)))) + 1
|
||||||
|
self.trending_whale_half_life = math.log2(0.1 ** (1 / (3 + self.integer('TRENDING_WHALE_DECAY_RATE', 24)))) + 1
|
||||||
|
self.trending_whale_threshold = float(self.integer('TRENDING_WHALE_THRESHOLD', 10000)) * 1E8
|
||||||
|
|
||||||
|
self.max_query_workers = self.integer('MAX_QUERY_WORKERS', 4)
|
||||||
|
self.individual_tag_indexes = self.boolean('INDIVIDUAL_TAG_INDEXES', True)
|
||||||
|
self.track_metrics = self.boolean('TRACK_METRICS', False)
|
||||||
|
self.websocket_host = self.default('WEBSOCKET_HOST', self.host)
|
||||||
|
self.websocket_port = self.integer('WEBSOCKET_PORT', None)
|
||||||
|
self.daemon_url = self.required('DAEMON_URL')
|
||||||
|
if coin is not None:
|
||||||
|
assert issubclass(coin, Coin)
|
||||||
|
self.coin = coin
|
||||||
|
else:
|
||||||
|
coin_name = self.required('COIN').strip()
|
||||||
|
network = self.default('NET', 'mainnet').strip()
|
||||||
|
self.coin = Coin.lookup_coin_class(coin_name, network)
|
||||||
|
self.es_index_prefix = self.default('ES_INDEX_PREFIX', '')
|
||||||
|
self.es_mode = self.default('ES_MODE', 'writer')
|
||||||
|
self.cache_MB = self.integer('CACHE_MB', 4096)
|
||||||
|
self.reorg_limit = self.integer('REORG_LIMIT', self.coin.REORG_LIMIT)
|
||||||
|
# Server stuff
|
||||||
|
self.tcp_port = self.integer('TCP_PORT', None)
|
||||||
|
self.udp_port = self.integer('UDP_PORT', self.tcp_port)
|
||||||
|
self.ssl_port = self.integer('SSL_PORT', None)
|
||||||
|
if self.ssl_port:
|
||||||
|
self.ssl_certfile = self.required('SSL_CERTFILE')
|
||||||
|
self.ssl_keyfile = self.required('SSL_KEYFILE')
|
||||||
|
self.rpc_port = self.integer('RPC_PORT', 8000)
|
||||||
|
self.prometheus_port = self.integer('PROMETHEUS_PORT', 0)
|
||||||
|
self.max_subscriptions = self.integer('MAX_SUBSCRIPTIONS', 10000)
|
||||||
|
self.banner_file = self.default('BANNER_FILE', None)
|
||||||
|
self.tor_banner_file = self.default('TOR_BANNER_FILE', self.banner_file)
|
||||||
|
self.anon_logs = self.boolean('ANON_LOGS', False)
|
||||||
|
self.log_sessions = self.integer('LOG_SESSIONS', 3600)
|
||||||
|
self.allow_lan_udp = self.boolean('ALLOW_LAN_UDP', False)
|
||||||
|
self.country = self.default('COUNTRY', 'US')
|
||||||
|
# Peer discovery
|
||||||
|
self.peer_discovery = self.peer_discovery_enum()
|
||||||
|
self.peer_announce = self.boolean('PEER_ANNOUNCE', True)
|
||||||
|
self.peer_hubs = self.extract_peer_hubs()
|
||||||
|
self.force_proxy = self.boolean('FORCE_PROXY', False)
|
||||||
|
self.tor_proxy_host = self.default('TOR_PROXY_HOST', 'localhost')
|
||||||
|
self.tor_proxy_port = self.integer('TOR_PROXY_PORT', None)
|
||||||
|
# The electrum client takes the empty string as unspecified
|
||||||
|
self.payment_address = self.default('PAYMENT_ADDRESS', '')
|
||||||
|
self.donation_address = self.default('DONATION_ADDRESS', '')
|
||||||
|
# Server limits to help prevent DoS
|
||||||
|
self.max_send = self.integer('MAX_SEND', 1000000)
|
||||||
|
self.max_receive = self.integer('MAX_RECEIVE', 1000000)
|
||||||
|
self.max_subs = self.integer('MAX_SUBS', 250000)
|
||||||
|
self.max_sessions = self.sane_max_sessions()
|
||||||
|
self.max_session_subs = self.integer('MAX_SESSION_SUBS', 50000)
|
||||||
|
self.session_timeout = self.integer('SESSION_TIMEOUT', 600)
|
||||||
|
self.drop_client = self.custom("DROP_CLIENT", None, re.compile)
|
||||||
|
self.description = self.default('DESCRIPTION', '')
|
||||||
|
self.daily_fee = self.string_amount('DAILY_FEE', '0')
|
||||||
|
|
||||||
|
# Identities
|
||||||
|
clearnet_identity = self.clearnet_identity()
|
||||||
|
tor_identity = self.tor_identity(clearnet_identity)
|
||||||
|
self.identities = [identity
|
||||||
|
for identity in (clearnet_identity, tor_identity)
|
||||||
|
if identity is not None]
|
||||||
|
self.database_query_timeout = float(self.integer('QUERY_TIMEOUT_MS', 3000)) / 1000.0
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def default(cls, envvar, default):
|
||||||
|
return environ.get(envvar, default)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def boolean(cls, envvar, default):
|
||||||
|
default = 'Yes' if default else ''
|
||||||
|
return bool(cls.default(envvar, default).strip())
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def required(cls, envvar):
|
||||||
|
value = environ.get(envvar)
|
||||||
|
if value is None:
|
||||||
|
raise cls.Error(f'required envvar {envvar} not set')
|
||||||
|
return value
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def string_amount(cls, envvar, default):
|
||||||
|
value = environ.get(envvar, default)
|
||||||
|
amount_pattern = re.compile("[0-9]{0,10}(\.[0-9]{1,8})?")
|
||||||
|
if len(value) > 0 and not amount_pattern.fullmatch(value):
|
||||||
|
raise cls.Error(f'{value} is not a valid amount for {envvar}')
|
||||||
|
return value
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def integer(cls, envvar, default):
|
||||||
|
value = environ.get(envvar)
|
||||||
|
if value is None:
|
||||||
|
return default
|
||||||
|
try:
|
||||||
|
return int(value)
|
||||||
|
except Exception:
|
||||||
|
raise cls.Error(f'cannot convert envvar {envvar} value {value} to an integer')
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def custom(cls, envvar, default, parse):
|
||||||
|
value = environ.get(envvar)
|
||||||
|
if value is None:
|
||||||
|
return default
|
||||||
|
try:
|
||||||
|
return parse(value)
|
||||||
|
except Exception as e:
|
||||||
|
raise cls.Error(f'cannot parse envvar {envvar} value {value}') from e
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def obsolete(cls, envvars):
|
||||||
|
bad = [envvar for envvar in envvars if environ.get(envvar)]
|
||||||
|
if bad:
|
||||||
|
raise cls.Error(f'remove obsolete environment variables {bad}')
|
||||||
|
|
||||||
|
def set_event_loop_policy(self):
|
||||||
|
policy_name = self.default('EVENT_LOOP_POLICY', None)
|
||||||
|
if not policy_name:
|
||||||
|
import asyncio
|
||||||
|
return asyncio.get_event_loop_policy()
|
||||||
|
elif policy_name == 'uvloop':
|
||||||
|
import uvloop
|
||||||
|
import asyncio
|
||||||
|
loop_policy = uvloop.EventLoopPolicy()
|
||||||
|
asyncio.set_event_loop_policy(loop_policy)
|
||||||
|
return loop_policy
|
||||||
|
raise self.Error(f'unknown event loop policy "{policy_name}"')
|
||||||
|
|
||||||
|
def cs_host(self, *, for_rpc):
|
||||||
|
"""Returns the 'host' argument to pass to asyncio's create_server
|
||||||
|
call. The result can be a single host name string, a list of
|
||||||
|
host name strings, or an empty string to bind to all interfaces.
|
||||||
|
|
||||||
|
If rpc is True the host to use for the RPC server is returned.
|
||||||
|
Otherwise the host to use for SSL/TCP servers is returned.
|
||||||
|
"""
|
||||||
|
host = self.rpc_host if for_rpc else self.host
|
||||||
|
result = [part.strip() for part in host.split(',')]
|
||||||
|
if len(result) == 1:
|
||||||
|
result = result[0]
|
||||||
|
# An empty result indicates all interfaces, which we do not
|
||||||
|
# permitted for an RPC server.
|
||||||
|
if for_rpc and not result:
|
||||||
|
result = 'localhost'
|
||||||
|
if result == 'localhost':
|
||||||
|
# 'localhost' resolves to ::1 (ipv6) on many systems, which fails on default setup of
|
||||||
|
# docker, using 127.0.0.1 instead forces ipv4
|
||||||
|
result = '127.0.0.1'
|
||||||
|
return result
|
||||||
|
|
||||||
|
def sane_max_sessions(self):
|
||||||
|
"""Return the maximum number of sessions to permit. Normally this
|
||||||
|
is MAX_SESSIONS. However, to prevent open file exhaustion, ajdust
|
||||||
|
downwards if running with a small open file rlimit."""
|
||||||
|
env_value = self.integer('MAX_SESSIONS', 1000)
|
||||||
|
nofile_limit = resource.getrlimit(resource.RLIMIT_NOFILE)[0]
|
||||||
|
# We give the DB 250 files; allow ElectrumX 100 for itself
|
||||||
|
value = max(0, min(env_value, nofile_limit - 350))
|
||||||
|
if value < env_value:
|
||||||
|
self.logger.warning(f'lowered maximum sessions from {env_value:,d} to {value:,d} '
|
||||||
|
f'because your open file limit is {nofile_limit:,d}')
|
||||||
|
return value
|
||||||
|
|
||||||
|
def clearnet_identity(self):
|
||||||
|
host = self.default('REPORT_HOST', None)
|
||||||
|
if host is None:
|
||||||
|
return None
|
||||||
|
try:
|
||||||
|
ip = ip_address(host)
|
||||||
|
except ValueError:
|
||||||
|
bad = (not lib_util.is_valid_hostname(host)
|
||||||
|
or host.lower() == 'localhost')
|
||||||
|
else:
|
||||||
|
bad = (ip.is_multicast or ip.is_unspecified
|
||||||
|
or (ip.is_private and self.peer_announce))
|
||||||
|
if bad:
|
||||||
|
raise self.Error(f'"{host}" is not a valid REPORT_HOST')
|
||||||
|
tcp_port = self.integer('REPORT_TCP_PORT', self.tcp_port) or None
|
||||||
|
ssl_port = self.integer('REPORT_SSL_PORT', self.ssl_port) or None
|
||||||
|
if tcp_port == ssl_port:
|
||||||
|
raise self.Error('REPORT_TCP_PORT and REPORT_SSL_PORT '
|
||||||
|
f'both resolve to {tcp_port}')
|
||||||
|
return NetIdentity(
|
||||||
|
host,
|
||||||
|
tcp_port,
|
||||||
|
ssl_port,
|
||||||
|
''
|
||||||
|
)
|
||||||
|
|
||||||
|
def tor_identity(self, clearnet):
|
||||||
|
host = self.default('REPORT_HOST_TOR', None)
|
||||||
|
if host is None:
|
||||||
|
return None
|
||||||
|
if not host.endswith('.onion'):
|
||||||
|
raise self.Error(f'tor host "{host}" must end with ".onion"')
|
||||||
|
|
||||||
|
def port(port_kind):
|
||||||
|
"""Returns the clearnet identity port, if any and not zero,
|
||||||
|
otherwise the listening port."""
|
||||||
|
result = 0
|
||||||
|
if clearnet:
|
||||||
|
result = getattr(clearnet, port_kind)
|
||||||
|
return result or getattr(self, port_kind)
|
||||||
|
|
||||||
|
tcp_port = self.integer('REPORT_TCP_PORT_TOR',
|
||||||
|
port('tcp_port')) or None
|
||||||
|
ssl_port = self.integer('REPORT_SSL_PORT_TOR',
|
||||||
|
port('ssl_port')) or None
|
||||||
|
if tcp_port == ssl_port:
|
||||||
|
raise self.Error('REPORT_TCP_PORT_TOR and REPORT_SSL_PORT_TOR '
|
||||||
|
f'both resolve to {tcp_port}')
|
||||||
|
|
||||||
|
return NetIdentity(
|
||||||
|
host,
|
||||||
|
tcp_port,
|
||||||
|
ssl_port,
|
||||||
|
'_tor',
|
||||||
|
)
|
||||||
|
|
||||||
|
def hosts_dict(self):
|
||||||
|
return {identity.host: {'tcp_port': identity.tcp_port,
|
||||||
|
'ssl_port': identity.ssl_port}
|
||||||
|
for identity in self.identities}
|
||||||
|
|
||||||
|
def peer_discovery_enum(self):
|
||||||
|
pd = self.default('PEER_DISCOVERY', 'on').strip().lower()
|
||||||
|
if pd in ('off', ''):
|
||||||
|
return self.PD_OFF
|
||||||
|
elif pd == 'self':
|
||||||
|
return self.PD_SELF
|
||||||
|
else:
|
||||||
|
return self.PD_ON
|
||||||
|
|
||||||
|
def extract_peer_hubs(self):
|
||||||
|
return [hub.strip() for hub in self.default('PEER_HUBS', '').split(',') if hub.strip()]
|
160
lbry/wallet/server/hash.py
Normal file
160
lbry/wallet/server/hash.py
Normal file
|
@ -0,0 +1,160 @@
|
||||||
|
# Copyright (c) 2016-2017, Neil Booth
|
||||||
|
#
|
||||||
|
# All rights reserved.
|
||||||
|
#
|
||||||
|
# The MIT License (MIT)
|
||||||
|
#
|
||||||
|
# Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
# a copy of this software and associated documentation files (the
|
||||||
|
# "Software"), to deal in the Software without restriction, including
|
||||||
|
# without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
# distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
# permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
# the following conditions:
|
||||||
|
#
|
||||||
|
# The above copyright notice and this permission notice shall be
|
||||||
|
# included in all copies or substantial portions of the Software.
|
||||||
|
#
|
||||||
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
|
"""Cryptograph hash functions and related classes."""
|
||||||
|
|
||||||
|
|
||||||
|
import hashlib
|
||||||
|
import hmac
|
||||||
|
|
||||||
|
from lbry.wallet.server.util import bytes_to_int, int_to_bytes, hex_to_bytes
|
||||||
|
|
||||||
|
_sha256 = hashlib.sha256
|
||||||
|
_sha512 = hashlib.sha512
|
||||||
|
_new_hash = hashlib.new
|
||||||
|
_new_hmac = hmac.new
|
||||||
|
HASHX_LEN = 11
|
||||||
|
CLAIM_HASH_LEN = 20
|
||||||
|
|
||||||
|
|
||||||
|
def sha256(x):
|
||||||
|
"""Simple wrapper of hashlib sha256."""
|
||||||
|
return _sha256(x).digest()
|
||||||
|
|
||||||
|
|
||||||
|
def ripemd160(x):
|
||||||
|
"""Simple wrapper of hashlib ripemd160."""
|
||||||
|
h = _new_hash('ripemd160')
|
||||||
|
h.update(x)
|
||||||
|
return h.digest()
|
||||||
|
|
||||||
|
|
||||||
|
def double_sha256(x):
|
||||||
|
"""SHA-256 of SHA-256, as used extensively in bitcoin."""
|
||||||
|
return sha256(sha256(x))
|
||||||
|
|
||||||
|
|
||||||
|
def hmac_sha512(key, msg):
|
||||||
|
"""Use SHA-512 to provide an HMAC."""
|
||||||
|
return _new_hmac(key, msg, _sha512).digest()
|
||||||
|
|
||||||
|
|
||||||
|
def hash160(x):
|
||||||
|
"""RIPEMD-160 of SHA-256.
|
||||||
|
|
||||||
|
Used to make bitcoin addresses from pubkeys."""
|
||||||
|
return ripemd160(sha256(x))
|
||||||
|
|
||||||
|
|
||||||
|
def hash_to_hex_str(x: bytes) -> str:
|
||||||
|
"""Convert a big-endian binary hash to displayed hex string.
|
||||||
|
|
||||||
|
Display form of a binary hash is reversed and converted to hex.
|
||||||
|
"""
|
||||||
|
return x[::-1].hex()
|
||||||
|
|
||||||
|
|
||||||
|
def hex_str_to_hash(x: str) -> bytes:
|
||||||
|
"""Convert a displayed hex string to a binary hash."""
|
||||||
|
return hex_to_bytes(x)[::-1]
|
||||||
|
|
||||||
|
|
||||||
|
class Base58Error(Exception):
|
||||||
|
"""Exception used for Base58 errors."""
|
||||||
|
|
||||||
|
|
||||||
|
class Base58:
|
||||||
|
"""Class providing base 58 functionality."""
|
||||||
|
|
||||||
|
chars = '123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz'
|
||||||
|
assert len(chars) == 58
|
||||||
|
cmap = {c: n for n, c in enumerate(chars)}
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def char_value(c):
|
||||||
|
val = Base58.cmap.get(c)
|
||||||
|
if val is None:
|
||||||
|
raise Base58Error(f'invalid base 58 character "{c}"')
|
||||||
|
return val
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def decode(txt):
|
||||||
|
"""Decodes txt into a big-endian bytearray."""
|
||||||
|
if not isinstance(txt, str):
|
||||||
|
raise TypeError('a string is required')
|
||||||
|
|
||||||
|
if not txt:
|
||||||
|
raise Base58Error('string cannot be empty')
|
||||||
|
|
||||||
|
value = 0
|
||||||
|
for c in txt:
|
||||||
|
value = value * 58 + Base58.char_value(c)
|
||||||
|
|
||||||
|
result = int_to_bytes(value)
|
||||||
|
|
||||||
|
# Prepend leading zero bytes if necessary
|
||||||
|
count = 0
|
||||||
|
for c in txt:
|
||||||
|
if c != '1':
|
||||||
|
break
|
||||||
|
count += 1
|
||||||
|
if count:
|
||||||
|
result = bytes(count) + result
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def encode(be_bytes):
|
||||||
|
"""Converts a big-endian bytearray into a base58 string."""
|
||||||
|
value = bytes_to_int(be_bytes)
|
||||||
|
|
||||||
|
txt = ''
|
||||||
|
while value:
|
||||||
|
value, mod = divmod(value, 58)
|
||||||
|
txt += Base58.chars[mod]
|
||||||
|
|
||||||
|
for byte in be_bytes:
|
||||||
|
if byte != 0:
|
||||||
|
break
|
||||||
|
txt += '1'
|
||||||
|
|
||||||
|
return txt[::-1]
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def decode_check(txt, *, hash_fn=double_sha256):
|
||||||
|
"""Decodes a Base58Check-encoded string to a payload. The version
|
||||||
|
prefixes it."""
|
||||||
|
be_bytes = Base58.decode(txt)
|
||||||
|
result, check = be_bytes[:-4], be_bytes[-4:]
|
||||||
|
if check != hash_fn(result)[:4]:
|
||||||
|
raise Base58Error(f'invalid base 58 checksum for {txt}')
|
||||||
|
return result
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def encode_check(payload, *, hash_fn=double_sha256):
|
||||||
|
"""Encodes a payload bytearray (which includes the version byte(s))
|
||||||
|
into a Base58Check string."""
|
||||||
|
be_bytes = payload + hash_fn(payload)[:4]
|
||||||
|
return Base58.encode(be_bytes)
|
1091
lbry/wallet/server/leveldb.py
Normal file
1091
lbry/wallet/server/leveldb.py
Normal file
File diff suppressed because it is too large
Load diff
360
lbry/wallet/server/mempool.py
Normal file
360
lbry/wallet/server/mempool.py
Normal file
|
@ -0,0 +1,360 @@
|
||||||
|
# Copyright (c) 2016-2018, Neil Booth
|
||||||
|
#
|
||||||
|
# All rights reserved.
|
||||||
|
#
|
||||||
|
# See the file "LICENCE" for information about the copyright
|
||||||
|
# and warranty status of this software.
|
||||||
|
|
||||||
|
"""Mempool handling."""
|
||||||
|
import asyncio
|
||||||
|
import itertools
|
||||||
|
import time
|
||||||
|
import attr
|
||||||
|
import typing
|
||||||
|
from typing import Set, Optional, Callable, Awaitable
|
||||||
|
from collections import defaultdict
|
||||||
|
from prometheus_client import Histogram
|
||||||
|
from lbry.wallet.server.hash import hash_to_hex_str, hex_str_to_hash
|
||||||
|
from lbry.wallet.server.util import class_logger, chunks
|
||||||
|
from lbry.wallet.server.leveldb import UTXO
|
||||||
|
if typing.TYPE_CHECKING:
|
||||||
|
from lbry.wallet.server.session import LBRYSessionManager
|
||||||
|
|
||||||
|
|
||||||
|
@attr.s(slots=True)
|
||||||
|
class MemPoolTx:
|
||||||
|
prevouts = attr.ib()
|
||||||
|
# A pair is a (hashX, value) tuple
|
||||||
|
in_pairs = attr.ib()
|
||||||
|
out_pairs = attr.ib()
|
||||||
|
fee = attr.ib()
|
||||||
|
size = attr.ib()
|
||||||
|
|
||||||
|
|
||||||
|
@attr.s(slots=True)
|
||||||
|
class MemPoolTxSummary:
|
||||||
|
hash = attr.ib()
|
||||||
|
fee = attr.ib()
|
||||||
|
has_unconfirmed_inputs = attr.ib()
|
||||||
|
|
||||||
|
|
||||||
|
NAMESPACE = "wallet_server"
|
||||||
|
HISTOGRAM_BUCKETS = (
|
||||||
|
.005, .01, .025, .05, .075, .1, .25, .5, .75, 1.0, 2.5, 5.0, 7.5, 10.0, 15.0, 20.0, 30.0, 60.0, float('inf')
|
||||||
|
)
|
||||||
|
mempool_process_time_metric = Histogram(
|
||||||
|
"processed_mempool", "Time to process mempool and notify touched addresses",
|
||||||
|
namespace=NAMESPACE, buckets=HISTOGRAM_BUCKETS
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class MemPool:
|
||||||
|
def __init__(self, coin, daemon, db, state_lock: asyncio.Lock, refresh_secs=1.0, log_status_secs=120.0):
|
||||||
|
self.coin = coin
|
||||||
|
self._daemon = daemon
|
||||||
|
self._db = db
|
||||||
|
self._touched_mp = {}
|
||||||
|
self._touched_bp = {}
|
||||||
|
self._highest_block = -1
|
||||||
|
|
||||||
|
self.logger = class_logger(__name__, self.__class__.__name__)
|
||||||
|
self.txs = {}
|
||||||
|
self.hashXs = defaultdict(set) # None can be a key
|
||||||
|
self.cached_compact_histogram = []
|
||||||
|
self.refresh_secs = refresh_secs
|
||||||
|
self.log_status_secs = log_status_secs
|
||||||
|
# Prevents mempool refreshes during fee histogram calculation
|
||||||
|
self.lock = state_lock
|
||||||
|
self.wakeup = asyncio.Event()
|
||||||
|
self.mempool_process_time_metric = mempool_process_time_metric
|
||||||
|
self.notified_mempool_txs = set()
|
||||||
|
self.notify_sessions: Optional[Callable[[int, Set[bytes], Set[bytes]], Awaitable[None]]] = None
|
||||||
|
|
||||||
|
async def _logging(self, synchronized_event):
|
||||||
|
"""Print regular logs of mempool stats."""
|
||||||
|
self.logger.info('beginning processing of daemon mempool. '
|
||||||
|
'This can take some time...')
|
||||||
|
start = time.perf_counter()
|
||||||
|
await synchronized_event.wait()
|
||||||
|
elapsed = time.perf_counter() - start
|
||||||
|
self.logger.info(f'synced in {elapsed:.2f}s')
|
||||||
|
while True:
|
||||||
|
self.logger.info(f'{len(self.txs):,d} txs '
|
||||||
|
f'touching {len(self.hashXs):,d} addresses')
|
||||||
|
await asyncio.sleep(self.log_status_secs)
|
||||||
|
await synchronized_event.wait()
|
||||||
|
|
||||||
|
def _accept_transactions(self, tx_map, utxo_map, touched):
|
||||||
|
"""Accept transactions in tx_map to the mempool if all their inputs
|
||||||
|
can be found in the existing mempool or a utxo_map from the
|
||||||
|
DB.
|
||||||
|
|
||||||
|
Returns an (unprocessed tx_map, unspent utxo_map) pair.
|
||||||
|
"""
|
||||||
|
hashXs = self.hashXs
|
||||||
|
txs = self.txs
|
||||||
|
|
||||||
|
deferred = {}
|
||||||
|
unspent = set(utxo_map)
|
||||||
|
# Try to find all prevouts so we can accept the TX
|
||||||
|
for hash, tx in tx_map.items():
|
||||||
|
in_pairs = []
|
||||||
|
try:
|
||||||
|
for prevout in tx.prevouts:
|
||||||
|
utxo = utxo_map.get(prevout)
|
||||||
|
if not utxo:
|
||||||
|
prev_hash, prev_index = prevout
|
||||||
|
# Raises KeyError if prev_hash is not in txs
|
||||||
|
utxo = txs[prev_hash].out_pairs[prev_index]
|
||||||
|
in_pairs.append(utxo)
|
||||||
|
except KeyError:
|
||||||
|
deferred[hash] = tx
|
||||||
|
continue
|
||||||
|
|
||||||
|
# Spend the prevouts
|
||||||
|
unspent.difference_update(tx.prevouts)
|
||||||
|
|
||||||
|
# Save the in_pairs, compute the fee and accept the TX
|
||||||
|
tx.in_pairs = tuple(in_pairs)
|
||||||
|
# Avoid negative fees if dealing with generation-like transactions
|
||||||
|
# because some in_parts would be missing
|
||||||
|
tx.fee = max(0, (sum(v for _, v in tx.in_pairs) -
|
||||||
|
sum(v for _, v in tx.out_pairs)))
|
||||||
|
txs[hash] = tx
|
||||||
|
|
||||||
|
for hashX, value in itertools.chain(tx.in_pairs, tx.out_pairs):
|
||||||
|
touched.add(hashX)
|
||||||
|
hashXs[hashX].add(hash)
|
||||||
|
|
||||||
|
return deferred, {prevout: utxo_map[prevout] for prevout in unspent}
|
||||||
|
|
||||||
|
async def _mempool_loop(self, synchronized_event):
|
||||||
|
try:
|
||||||
|
return await self._refresh_hashes(synchronized_event)
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
raise
|
||||||
|
except Exception as e:
|
||||||
|
self.logger.exception("MEMPOOL DIED")
|
||||||
|
raise e
|
||||||
|
|
||||||
|
async def _refresh_hashes(self, synchronized_event):
|
||||||
|
"""Refresh our view of the daemon's mempool."""
|
||||||
|
while True:
|
||||||
|
start = time.perf_counter()
|
||||||
|
height = self._daemon.cached_height()
|
||||||
|
hex_hashes = await self._daemon.mempool_hashes()
|
||||||
|
if height != await self._daemon.height():
|
||||||
|
continue
|
||||||
|
hashes = {hex_str_to_hash(hh) for hh in hex_hashes}
|
||||||
|
async with self.lock:
|
||||||
|
new_hashes = hashes.difference(self.notified_mempool_txs)
|
||||||
|
touched = await self._process_mempool(hashes)
|
||||||
|
self.notified_mempool_txs.update(new_hashes)
|
||||||
|
new_touched = {
|
||||||
|
touched_hashx for touched_hashx, txs in self.hashXs.items() if txs.intersection(new_hashes)
|
||||||
|
}
|
||||||
|
synchronized_event.set()
|
||||||
|
synchronized_event.clear()
|
||||||
|
await self.on_mempool(touched, new_touched, height)
|
||||||
|
duration = time.perf_counter() - start
|
||||||
|
self.mempool_process_time_metric.observe(duration)
|
||||||
|
try:
|
||||||
|
# we wait up to `refresh_secs` but go early if a broadcast happens (which triggers wakeup event)
|
||||||
|
await asyncio.wait_for(self.wakeup.wait(), timeout=self.refresh_secs)
|
||||||
|
except asyncio.TimeoutError:
|
||||||
|
pass
|
||||||
|
finally:
|
||||||
|
self.wakeup.clear()
|
||||||
|
|
||||||
|
async def _process_mempool(self, all_hashes):
|
||||||
|
# Re-sync with the new set of hashes
|
||||||
|
txs = self.txs
|
||||||
|
|
||||||
|
hashXs = self.hashXs # hashX: [tx_hash, ...]
|
||||||
|
touched = set()
|
||||||
|
|
||||||
|
# First handle txs that have disappeared
|
||||||
|
for tx_hash in set(txs).difference(all_hashes):
|
||||||
|
tx = txs.pop(tx_hash)
|
||||||
|
tx_hashXs = {hashX for hashX, value in tx.in_pairs}
|
||||||
|
tx_hashXs.update(hashX for hashX, value in tx.out_pairs)
|
||||||
|
for hashX in tx_hashXs:
|
||||||
|
hashXs[hashX].remove(tx_hash)
|
||||||
|
if not hashXs[hashX]:
|
||||||
|
del hashXs[hashX]
|
||||||
|
touched.update(tx_hashXs)
|
||||||
|
|
||||||
|
# Process new transactions
|
||||||
|
new_hashes = list(all_hashes.difference(txs))
|
||||||
|
if new_hashes:
|
||||||
|
fetches = []
|
||||||
|
for hashes in chunks(new_hashes, 200):
|
||||||
|
fetches.append(self._fetch_and_accept(hashes, all_hashes, touched))
|
||||||
|
tx_map = {}
|
||||||
|
utxo_map = {}
|
||||||
|
for fetch in asyncio.as_completed(fetches):
|
||||||
|
deferred, unspent = await fetch
|
||||||
|
tx_map.update(deferred)
|
||||||
|
utxo_map.update(unspent)
|
||||||
|
|
||||||
|
prior_count = 0
|
||||||
|
# FIXME: this is not particularly efficient
|
||||||
|
while tx_map and len(tx_map) != prior_count:
|
||||||
|
prior_count = len(tx_map)
|
||||||
|
tx_map, utxo_map = self._accept_transactions(tx_map, utxo_map, touched)
|
||||||
|
|
||||||
|
if tx_map:
|
||||||
|
self.logger.info(f'{len(tx_map)} txs dropped')
|
||||||
|
|
||||||
|
return touched
|
||||||
|
|
||||||
|
async def _fetch_and_accept(self, hashes, all_hashes, touched):
|
||||||
|
"""Fetch a list of mempool transactions."""
|
||||||
|
raw_txs = await self._daemon.getrawtransactions((hash_to_hex_str(hash) for hash in hashes))
|
||||||
|
|
||||||
|
to_hashX = self.coin.hashX_from_script
|
||||||
|
deserializer = self.coin.DESERIALIZER
|
||||||
|
|
||||||
|
tx_map = {}
|
||||||
|
for hash, raw_tx in zip(hashes, raw_txs):
|
||||||
|
# The daemon may have evicted the tx from its
|
||||||
|
# mempool or it may have gotten in a block
|
||||||
|
if not raw_tx:
|
||||||
|
continue
|
||||||
|
tx, tx_size = deserializer(raw_tx).read_tx_and_vsize()
|
||||||
|
# Convert the inputs and outputs into (hashX, value) pairs
|
||||||
|
# Drop generation-like inputs from MemPoolTx.prevouts
|
||||||
|
txin_pairs = tuple((txin.prev_hash, txin.prev_idx)
|
||||||
|
for txin in tx.inputs
|
||||||
|
if not txin.is_generation())
|
||||||
|
txout_pairs = tuple((to_hashX(txout.pk_script), txout.value)
|
||||||
|
for txout in tx.outputs)
|
||||||
|
tx_map[hash] = MemPoolTx(txin_pairs, None, txout_pairs,
|
||||||
|
0, tx_size)
|
||||||
|
|
||||||
|
# Determine all prevouts not in the mempool, and fetch the
|
||||||
|
# UTXO information from the database. Failed prevout lookups
|
||||||
|
# return None - concurrent database updates happen - which is
|
||||||
|
# relied upon by _accept_transactions. Ignore prevouts that are
|
||||||
|
# generation-like.
|
||||||
|
prevouts = tuple(prevout for tx in tx_map.values()
|
||||||
|
for prevout in tx.prevouts
|
||||||
|
if prevout[0] not in all_hashes)
|
||||||
|
utxos = await self._db.lookup_utxos(prevouts)
|
||||||
|
utxo_map = dict(zip(prevouts, utxos))
|
||||||
|
|
||||||
|
return self._accept_transactions(tx_map, utxo_map, touched)
|
||||||
|
|
||||||
|
#
|
||||||
|
# External interface
|
||||||
|
#
|
||||||
|
|
||||||
|
async def keep_synchronized(self, synchronized_event):
|
||||||
|
"""Keep the mempool synchronized with the daemon."""
|
||||||
|
await asyncio.wait([
|
||||||
|
self._mempool_loop(synchronized_event),
|
||||||
|
# self._refresh_histogram(synchronized_event),
|
||||||
|
self._logging(synchronized_event)
|
||||||
|
])
|
||||||
|
|
||||||
|
async def balance_delta(self, hashX):
|
||||||
|
"""Return the unconfirmed amount in the mempool for hashX.
|
||||||
|
|
||||||
|
Can be positive or negative.
|
||||||
|
"""
|
||||||
|
value = 0
|
||||||
|
if hashX in self.hashXs:
|
||||||
|
for hash in self.hashXs[hashX]:
|
||||||
|
tx = self.txs[hash]
|
||||||
|
value -= sum(v for h168, v in tx.in_pairs if h168 == hashX)
|
||||||
|
value += sum(v for h168, v in tx.out_pairs if h168 == hashX)
|
||||||
|
return value
|
||||||
|
|
||||||
|
def compact_fee_histogram(self):
|
||||||
|
"""Return a compact fee histogram of the current mempool."""
|
||||||
|
return self.cached_compact_histogram
|
||||||
|
|
||||||
|
async def potential_spends(self, hashX):
|
||||||
|
"""Return a set of (prev_hash, prev_idx) pairs from mempool
|
||||||
|
transactions that touch hashX.
|
||||||
|
|
||||||
|
None, some or all of these may be spends of the hashX, but all
|
||||||
|
actual spends of it (in the DB or mempool) will be included.
|
||||||
|
"""
|
||||||
|
result = set()
|
||||||
|
for tx_hash in self.hashXs.get(hashX, ()):
|
||||||
|
tx = self.txs[tx_hash]
|
||||||
|
result.update(tx.prevouts)
|
||||||
|
return result
|
||||||
|
|
||||||
|
def transaction_summaries(self, hashX):
|
||||||
|
"""Return a list of MemPoolTxSummary objects for the hashX."""
|
||||||
|
result = []
|
||||||
|
for tx_hash in self.hashXs.get(hashX, ()):
|
||||||
|
tx = self.txs[tx_hash]
|
||||||
|
has_ui = any(hash in self.txs for hash, idx in tx.prevouts)
|
||||||
|
result.append(MemPoolTxSummary(tx_hash, tx.fee, has_ui))
|
||||||
|
return result
|
||||||
|
|
||||||
|
async def unordered_UTXOs(self, hashX):
|
||||||
|
"""Return an unordered list of UTXO named tuples from mempool
|
||||||
|
transactions that pay to hashX.
|
||||||
|
|
||||||
|
This does not consider if any other mempool transactions spend
|
||||||
|
the outputs.
|
||||||
|
"""
|
||||||
|
utxos = []
|
||||||
|
for tx_hash in self.hashXs.get(hashX, ()):
|
||||||
|
tx = self.txs.get(tx_hash)
|
||||||
|
for pos, (hX, value) in enumerate(tx.out_pairs):
|
||||||
|
if hX == hashX:
|
||||||
|
utxos.append(UTXO(-1, pos, tx_hash, 0, value))
|
||||||
|
return utxos
|
||||||
|
|
||||||
|
def get_mempool_height(self, tx_hash):
|
||||||
|
# Height Progression
|
||||||
|
# -2: not broadcast
|
||||||
|
# -1: in mempool but has unconfirmed inputs
|
||||||
|
# 0: in mempool and all inputs confirmed
|
||||||
|
# +num: confirmed in a specific block (height)
|
||||||
|
if tx_hash not in self.txs:
|
||||||
|
return -2
|
||||||
|
tx = self.txs[tx_hash]
|
||||||
|
unspent_inputs = sum(1 if hash in self.txs else 0 for hash, idx in tx.prevouts)
|
||||||
|
if unspent_inputs:
|
||||||
|
return -1
|
||||||
|
return 0
|
||||||
|
|
||||||
|
async def _maybe_notify(self, new_touched):
|
||||||
|
tmp, tbp = self._touched_mp, self._touched_bp
|
||||||
|
common = set(tmp).intersection(tbp)
|
||||||
|
if common:
|
||||||
|
height = max(common)
|
||||||
|
elif tmp and max(tmp) == self._highest_block:
|
||||||
|
height = self._highest_block
|
||||||
|
else:
|
||||||
|
# Either we are processing a block and waiting for it to
|
||||||
|
# come in, or we have not yet had a mempool update for the
|
||||||
|
# new block height
|
||||||
|
return
|
||||||
|
touched = tmp.pop(height)
|
||||||
|
for old in [h for h in tmp if h <= height]:
|
||||||
|
del tmp[old]
|
||||||
|
for old in [h for h in tbp if h <= height]:
|
||||||
|
touched.update(tbp.pop(old))
|
||||||
|
# print("notify", height, len(touched), len(new_touched))
|
||||||
|
await self.notify_sessions(height, touched, new_touched)
|
||||||
|
|
||||||
|
async def start(self, height, session_manager: 'LBRYSessionManager'):
|
||||||
|
self._highest_block = height
|
||||||
|
self.notify_sessions = session_manager._notify_sessions
|
||||||
|
await self.notify_sessions(height, set(), set())
|
||||||
|
|
||||||
|
async def on_mempool(self, touched, new_touched, height):
|
||||||
|
self._touched_mp[height] = touched
|
||||||
|
await self._maybe_notify(new_touched)
|
||||||
|
|
||||||
|
async def on_block(self, touched, height):
|
||||||
|
self._touched_bp[height] = touched
|
||||||
|
self._highest_block = height
|
||||||
|
await self._maybe_notify(set())
|
258
lbry/wallet/server/merkle.py
Normal file
258
lbry/wallet/server/merkle.py
Normal file
|
@ -0,0 +1,258 @@
|
||||||
|
# Copyright (c) 2018, Neil Booth
|
||||||
|
#
|
||||||
|
# All rights reserved.
|
||||||
|
#
|
||||||
|
# The MIT License (MIT)
|
||||||
|
#
|
||||||
|
# Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
# a copy of this software and associated documentation files (the
|
||||||
|
# "Software"), to deal in the Software without restriction, including
|
||||||
|
# without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
# distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
# permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
# the following conditions:
|
||||||
|
#
|
||||||
|
# The above copyright notice and this permission notice shall be
|
||||||
|
# included in all copies or substantial portions of the Software.
|
||||||
|
#
|
||||||
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
# and warranty status of this software.
|
||||||
|
|
||||||
|
"""Merkle trees, branches, proofs and roots."""
|
||||||
|
|
||||||
|
from asyncio import Event
|
||||||
|
from math import ceil, log
|
||||||
|
|
||||||
|
from lbry.wallet.server.hash import double_sha256
|
||||||
|
|
||||||
|
|
||||||
|
class Merkle:
|
||||||
|
"""Perform merkle tree calculations on binary hashes using a given hash
|
||||||
|
function.
|
||||||
|
|
||||||
|
If the hash count is not even, the final hash is repeated when
|
||||||
|
calculating the next merkle layer up the tree.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, hash_func=double_sha256):
|
||||||
|
self.hash_func = hash_func
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def tree_depth(hash_count):
|
||||||
|
return Merkle.branch_length(hash_count) + 1
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def branch_length(hash_count):
|
||||||
|
"""Return the length of a merkle branch given the number of hashes."""
|
||||||
|
if not isinstance(hash_count, int):
|
||||||
|
raise TypeError('hash_count must be an integer')
|
||||||
|
if hash_count < 1:
|
||||||
|
raise ValueError('hash_count must be at least 1')
|
||||||
|
return ceil(log(hash_count, 2))
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def branch_and_root(hashes, index, length=None, hash_func=double_sha256):
|
||||||
|
"""Return a (merkle branch, merkle_root) pair given hashes, and the
|
||||||
|
index of one of those hashes.
|
||||||
|
"""
|
||||||
|
hashes = list(hashes)
|
||||||
|
if not isinstance(index, int):
|
||||||
|
raise TypeError('index must be an integer')
|
||||||
|
# This also asserts hashes is not empty
|
||||||
|
if not 0 <= index < len(hashes):
|
||||||
|
raise ValueError(f"index '{index}/{len(hashes)}' out of range")
|
||||||
|
natural_length = Merkle.branch_length(len(hashes))
|
||||||
|
if length is None:
|
||||||
|
length = natural_length
|
||||||
|
else:
|
||||||
|
if not isinstance(length, int):
|
||||||
|
raise TypeError('length must be an integer')
|
||||||
|
if length < natural_length:
|
||||||
|
raise ValueError('length out of range')
|
||||||
|
|
||||||
|
branch = []
|
||||||
|
for _ in range(length):
|
||||||
|
if len(hashes) & 1:
|
||||||
|
hashes.append(hashes[-1])
|
||||||
|
branch.append(hashes[index ^ 1])
|
||||||
|
index >>= 1
|
||||||
|
hashes = [hash_func(hashes[n] + hashes[n + 1])
|
||||||
|
for n in range(0, len(hashes), 2)]
|
||||||
|
|
||||||
|
return branch, hashes[0]
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def root(hashes, length=None):
|
||||||
|
"""Return the merkle root of a non-empty iterable of binary hashes."""
|
||||||
|
branch, root = Merkle.branch_and_root(hashes, 0, length)
|
||||||
|
return root
|
||||||
|
|
||||||
|
# @staticmethod
|
||||||
|
# def root_from_proof(hash, branch, index, hash_func=double_sha256):
|
||||||
|
# """Return the merkle root given a hash, a merkle branch to it, and
|
||||||
|
# its index in the hashes array.
|
||||||
|
#
|
||||||
|
# branch is an iterable sorted deepest to shallowest. If the
|
||||||
|
# returned root is the expected value then the merkle proof is
|
||||||
|
# verified.
|
||||||
|
#
|
||||||
|
# The caller should have confirmed the length of the branch with
|
||||||
|
# branch_length(). Unfortunately this is not easily done for
|
||||||
|
# bitcoin transactions as the number of transactions in a block
|
||||||
|
# is unknown to an SPV client.
|
||||||
|
# """
|
||||||
|
# for elt in branch:
|
||||||
|
# if index & 1:
|
||||||
|
# hash = hash_func(elt + hash)
|
||||||
|
# else:
|
||||||
|
# hash = hash_func(hash + elt)
|
||||||
|
# index >>= 1
|
||||||
|
# if index:
|
||||||
|
# raise ValueError('index out of range for branch')
|
||||||
|
# return hash
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def level(hashes, depth_higher):
|
||||||
|
"""Return a level of the merkle tree of hashes the given depth
|
||||||
|
higher than the bottom row of the original tree."""
|
||||||
|
size = 1 << depth_higher
|
||||||
|
root = Merkle.root
|
||||||
|
return [root(hashes[n: n + size], depth_higher)
|
||||||
|
for n in range(0, len(hashes), size)]
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def branch_and_root_from_level(level, leaf_hashes, index,
|
||||||
|
depth_higher):
|
||||||
|
"""Return a (merkle branch, merkle_root) pair when a merkle-tree has a
|
||||||
|
level cached.
|
||||||
|
|
||||||
|
To maximally reduce the amount of data hashed in computing a
|
||||||
|
markle branch, cache a tree of depth N at level N // 2.
|
||||||
|
|
||||||
|
level is a list of hashes in the middle of the tree (returned
|
||||||
|
by level())
|
||||||
|
|
||||||
|
leaf_hashes are the leaves needed to calculate a partial branch
|
||||||
|
up to level.
|
||||||
|
|
||||||
|
depth_higher is how much higher level is than the leaves of the tree
|
||||||
|
|
||||||
|
index is the index in the full list of hashes of the hash whose
|
||||||
|
merkle branch we want.
|
||||||
|
"""
|
||||||
|
if not isinstance(level, list):
|
||||||
|
raise TypeError("level must be a list")
|
||||||
|
if not isinstance(leaf_hashes, list):
|
||||||
|
raise TypeError("leaf_hashes must be a list")
|
||||||
|
leaf_index = (index >> depth_higher) << depth_higher
|
||||||
|
leaf_branch, leaf_root = Merkle.branch_and_root(
|
||||||
|
leaf_hashes, index - leaf_index, depth_higher)
|
||||||
|
index >>= depth_higher
|
||||||
|
level_branch, root = Merkle.branch_and_root(level, index)
|
||||||
|
# Check last so that we know index is in-range
|
||||||
|
if leaf_root != level[index]:
|
||||||
|
raise ValueError('leaf hashes inconsistent with level')
|
||||||
|
return leaf_branch + level_branch, root
|
||||||
|
|
||||||
|
|
||||||
|
class MerkleCache:
|
||||||
|
"""A cache to calculate merkle branches efficiently."""
|
||||||
|
|
||||||
|
def __init__(self, merkle, source_func):
|
||||||
|
"""Initialise a cache hashes taken from source_func:
|
||||||
|
|
||||||
|
async def source_func(index, count):
|
||||||
|
...
|
||||||
|
"""
|
||||||
|
self.merkle = merkle
|
||||||
|
self.source_func = source_func
|
||||||
|
self.length = 0
|
||||||
|
self.depth_higher = 0
|
||||||
|
self.initialized = Event()
|
||||||
|
|
||||||
|
def _segment_length(self):
|
||||||
|
return 1 << self.depth_higher
|
||||||
|
|
||||||
|
def _leaf_start(self, index):
|
||||||
|
"""Given a level's depth higher and a hash index, return the leaf
|
||||||
|
index and leaf hash count needed to calculate a merkle branch.
|
||||||
|
"""
|
||||||
|
depth_higher = self.depth_higher
|
||||||
|
return (index >> depth_higher) << depth_higher
|
||||||
|
|
||||||
|
def _level(self, hashes):
|
||||||
|
return self.merkle.level(hashes, self.depth_higher)
|
||||||
|
|
||||||
|
async def _extend_to(self, length):
|
||||||
|
"""Extend the length of the cache if necessary."""
|
||||||
|
if length <= self.length:
|
||||||
|
return
|
||||||
|
# Start from the beginning of any final partial segment.
|
||||||
|
# Retain the value of depth_higher; in practice this is fine
|
||||||
|
start = self._leaf_start(self.length)
|
||||||
|
hashes = await self.source_func(start, length - start)
|
||||||
|
self.level[start >> self.depth_higher:] = self._level(hashes)
|
||||||
|
self.length = length
|
||||||
|
|
||||||
|
async def _level_for(self, length):
|
||||||
|
"""Return a (level_length, final_hash) pair for a truncation
|
||||||
|
of the hashes to the given length."""
|
||||||
|
if length == self.length:
|
||||||
|
return self.level
|
||||||
|
level = self.level[:length >> self.depth_higher]
|
||||||
|
leaf_start = self._leaf_start(length)
|
||||||
|
count = min(self._segment_length(), length - leaf_start)
|
||||||
|
hashes = await self.source_func(leaf_start, count)
|
||||||
|
level += self._level(hashes)
|
||||||
|
return level
|
||||||
|
|
||||||
|
async def initialize(self, length):
|
||||||
|
"""Call to initialize the cache to a source of given length."""
|
||||||
|
self.length = length
|
||||||
|
self.depth_higher = self.merkle.tree_depth(length) // 2
|
||||||
|
self.level = self._level(await self.source_func(0, length))
|
||||||
|
self.initialized.set()
|
||||||
|
|
||||||
|
def truncate(self, length):
|
||||||
|
"""Truncate the cache so it covers no more than length underlying
|
||||||
|
hashes."""
|
||||||
|
if not isinstance(length, int):
|
||||||
|
raise TypeError('length must be an integer')
|
||||||
|
if length <= 0:
|
||||||
|
raise ValueError('length must be positive')
|
||||||
|
if length >= self.length:
|
||||||
|
return
|
||||||
|
length = self._leaf_start(length)
|
||||||
|
self.length = length
|
||||||
|
self.level[length >> self.depth_higher:] = []
|
||||||
|
|
||||||
|
async def branch_and_root(self, length, index):
|
||||||
|
"""Return a merkle branch and root. Length is the number of
|
||||||
|
hashes used to calculate the merkle root, index is the position
|
||||||
|
of the hash to calculate the branch of.
|
||||||
|
|
||||||
|
index must be less than length, which must be at least 1."""
|
||||||
|
if not isinstance(length, int):
|
||||||
|
raise TypeError('length must be an integer')
|
||||||
|
if not isinstance(index, int):
|
||||||
|
raise TypeError('index must be an integer')
|
||||||
|
if length <= 0:
|
||||||
|
raise ValueError('length must be positive')
|
||||||
|
if index >= length:
|
||||||
|
raise ValueError('index must be less than length')
|
||||||
|
await self.initialized.wait()
|
||||||
|
await self._extend_to(length)
|
||||||
|
leaf_start = self._leaf_start(index)
|
||||||
|
count = min(self._segment_length(), length - leaf_start)
|
||||||
|
leaf_hashes = await self.source_func(leaf_start, count)
|
||||||
|
if length < self._segment_length():
|
||||||
|
return self.merkle.branch_and_root(leaf_hashes, index)
|
||||||
|
level = await self._level_for(length)
|
||||||
|
return self.merkle.branch_and_root_from_level(
|
||||||
|
level, leaf_hashes, index, self.depth_higher)
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Reference in a new issue