Compare commits
1 commit
master
...
fix-tx-lis
Author | SHA1 | Date | |
---|---|---|---|
|
a9ba1fc1a7 |
441 changed files with 26927 additions and 68565 deletions
|
@ -2,6 +2,6 @@
|
||||||
.tox
|
.tox
|
||||||
__pycache__
|
__pycache__
|
||||||
dist
|
dist
|
||||||
lbry.egg-info
|
lbrynet.egg-info
|
||||||
docs
|
docs
|
||||||
tests
|
tests
|
1
.gitattributes
vendored
Normal file
1
.gitattributes
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
/CHANGELOG.md merge=union
|
74
.github/ISSUE_TEMPLATE.md
vendored
Normal file
74
.github/ISSUE_TEMPLATE.md
vendored
Normal file
|
@ -0,0 +1,74 @@
|
||||||
|
<!--
|
||||||
|
Thanks for reporting an issue to LBRY and helping us improve!
|
||||||
|
|
||||||
|
To make it possible for us to help you, please fill out below information carefully.
|
||||||
|
|
||||||
|
Before reporting any issues, please make sure that you're using the latest version.
|
||||||
|
- App: https://github.com/lbryio/lbry-desktop/releases
|
||||||
|
- Daemon: https://github.com/lbryio/lbry/releases
|
||||||
|
|
||||||
|
We are also available on Discord at https://chat.lbry.io
|
||||||
|
-->
|
||||||
|
|
||||||
|
|
||||||
|
## The Issue
|
||||||
|
|
||||||
|
In order to <achieve some value>,
|
||||||
|
as a <type of user>,
|
||||||
|
I want <some functionality>.
|
||||||
|
|
||||||
|
|
||||||
|
### Steps to reproduce
|
||||||
|
1.
|
||||||
|
2.
|
||||||
|
3.
|
||||||
|
|
||||||
|
### Expected behaviour
|
||||||
|
Tell us what should happen
|
||||||
|
|
||||||
|
### Actual behaviour
|
||||||
|
Tell us what happens instead
|
||||||
|
|
||||||
|
|
||||||
|
## System Configuration
|
||||||
|
|
||||||
|
<!-- For the app, this info is in the About section at the bottom of the Help page.
|
||||||
|
You can include a screenshot instead of typing it out -->
|
||||||
|
|
||||||
|
<!-- For the daemon, run:
|
||||||
|
curl 'http://localhost:5279' --data '{"method":"version"}'
|
||||||
|
and include the full output -->
|
||||||
|
|
||||||
|
- LBRY Daemon version:
|
||||||
|
- LBRY App version:
|
||||||
|
- LBRY Installation ID:
|
||||||
|
- Operating system:
|
||||||
|
|
||||||
|
|
||||||
|
## Anything Else
|
||||||
|
<!-- Include anything else that does not fit into the above sections -->
|
||||||
|
|
||||||
|
|
||||||
|
## Screenshots
|
||||||
|
<!-- If a screenshot would help explain the bug, please include one or two here -->
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## Internal Use
|
||||||
|
|
||||||
|
### Acceptance Criteria
|
||||||
|
1.
|
||||||
|
2.
|
||||||
|
3.
|
||||||
|
|
||||||
|
### Definition of Done
|
||||||
|
- [ ] Tested against acceptance criteria
|
||||||
|
- [ ] Tested against the assumptions of user story
|
||||||
|
- [ ] The project builds without errors
|
||||||
|
- [ ] Unit tests are written and passing
|
||||||
|
- [ ] Tests on devices/browsers listed in the issue have passed
|
||||||
|
- [ ] QA performed & issues resolved
|
||||||
|
- [ ] Refactoring completed
|
||||||
|
- [ ] Any configuration or build changes documented
|
||||||
|
- [ ] Documentation updated
|
||||||
|
- [ ] Peer Code Review performed
|
38
.github/PULL_REQUEST_TEMPLATE.md
vendored
Normal file
38
.github/PULL_REQUEST_TEMPLATE.md
vendored
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
## PR Checklist
|
||||||
|
Please check all that apply to this PR using "x":
|
||||||
|
|
||||||
|
- [ ] I have checked that this PR is not a duplicate of an existing PR (open, closed or merged)
|
||||||
|
- [ ] I have checked that this PR does not introduce a breaking change
|
||||||
|
- [ ] This PR introduces breaking changes and I have provided a detailed explanation below
|
||||||
|
|
||||||
|
|
||||||
|
## PR Type
|
||||||
|
What kind of change does this PR introduce?
|
||||||
|
|
||||||
|
Why is this change necessary?
|
||||||
|
|
||||||
|
<!-- Please check all that apply to this PR using "x". -->
|
||||||
|
|
||||||
|
- [ ] Bugfix
|
||||||
|
- [ ] Feature
|
||||||
|
- [ ] Breaking changes (bugfix or feature that introduces breaking changes)
|
||||||
|
- [ ] Code style update (formatting)
|
||||||
|
- [ ] Refactoring (no functional changes)
|
||||||
|
- [ ] Documentation changes
|
||||||
|
- [ ] Other - Please describe:
|
||||||
|
|
||||||
|
## Fixes
|
||||||
|
|
||||||
|
Issue Number: N/A
|
||||||
|
|
||||||
|
|
||||||
|
## What is the current behavior?
|
||||||
|
|
||||||
|
|
||||||
|
## What is the new behavior?
|
||||||
|
|
||||||
|
|
||||||
|
## Other information
|
||||||
|
|
||||||
|
|
||||||
|
<!-- If this PR contains a breaking change, please describe the impact and solution strategy for existing applications below. -->
|
206
.github/workflows/main.yml
vendored
206
.github/workflows/main.yml
vendored
|
@ -1,206 +0,0 @@
|
||||||
name: ci
|
|
||||||
on: ["push", "pull_request", "workflow_dispatch"]
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
|
|
||||||
lint:
|
|
||||||
name: lint
|
|
||||||
runs-on: ubuntu-20.04
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v3
|
|
||||||
- uses: actions/setup-python@v4
|
|
||||||
with:
|
|
||||||
python-version: '3.9'
|
|
||||||
- name: extract pip cache
|
|
||||||
uses: actions/cache@v3
|
|
||||||
with:
|
|
||||||
path: ~/.cache/pip
|
|
||||||
key: ${{ runner.os }}-pip-${{ hashFiles('setup.py') }}
|
|
||||||
restore-keys: ${{ runner.os }}-pip-
|
|
||||||
- run: pip install --user --upgrade pip wheel
|
|
||||||
- run: pip install -e .[lint]
|
|
||||||
- run: make lint
|
|
||||||
|
|
||||||
tests-unit:
|
|
||||||
name: "tests / unit"
|
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
os:
|
|
||||||
- ubuntu-20.04
|
|
||||||
- macos-latest
|
|
||||||
- windows-latest
|
|
||||||
runs-on: ${{ matrix.os }}
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v3
|
|
||||||
- uses: actions/setup-python@v4
|
|
||||||
with:
|
|
||||||
python-version: '3.9'
|
|
||||||
- name: set pip cache dir
|
|
||||||
shell: bash
|
|
||||||
run: echo "PIP_CACHE_DIR=$(pip cache dir)" >> $GITHUB_ENV
|
|
||||||
- name: extract pip cache
|
|
||||||
uses: actions/cache@v3
|
|
||||||
with:
|
|
||||||
path: ${{ env.PIP_CACHE_DIR }}
|
|
||||||
key: ${{ runner.os }}-pip-${{ hashFiles('setup.py') }}
|
|
||||||
restore-keys: ${{ runner.os }}-pip-
|
|
||||||
- id: os-name
|
|
||||||
uses: ASzc/change-string-case-action@v5
|
|
||||||
with:
|
|
||||||
string: ${{ runner.os }}
|
|
||||||
- run: python -m pip install --user --upgrade pip wheel
|
|
||||||
- if: startsWith(runner.os, 'linux')
|
|
||||||
run: pip install -e .[test]
|
|
||||||
- if: startsWith(runner.os, 'linux')
|
|
||||||
env:
|
|
||||||
HOME: /tmp
|
|
||||||
run: make test-unit-coverage
|
|
||||||
- if: startsWith(runner.os, 'linux') != true
|
|
||||||
run: pip install -e .[test]
|
|
||||||
- if: startsWith(runner.os, 'linux') != true
|
|
||||||
env:
|
|
||||||
HOME: /tmp
|
|
||||||
run: coverage run --source=lbry -m unittest tests/unit/test_conf.py
|
|
||||||
- name: submit coverage report
|
|
||||||
env:
|
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
COVERALLS_FLAG_NAME: tests-unit-${{ steps.os-name.outputs.lowercase }}
|
|
||||||
COVERALLS_PARALLEL: true
|
|
||||||
run: |
|
|
||||||
pip install coveralls
|
|
||||||
coveralls --service=github
|
|
||||||
|
|
||||||
tests-integration:
|
|
||||||
name: "tests / integration"
|
|
||||||
runs-on: ubuntu-20.04
|
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
test:
|
|
||||||
- datanetwork
|
|
||||||
- blockchain
|
|
||||||
- claims
|
|
||||||
- takeovers
|
|
||||||
- transactions
|
|
||||||
- other
|
|
||||||
steps:
|
|
||||||
- name: Configure sysctl limits
|
|
||||||
run: |
|
|
||||||
sudo swapoff -a
|
|
||||||
sudo sysctl -w vm.swappiness=1
|
|
||||||
sudo sysctl -w fs.file-max=262144
|
|
||||||
sudo sysctl -w vm.max_map_count=262144
|
|
||||||
- name: Runs Elasticsearch
|
|
||||||
uses: elastic/elastic-github-actions/elasticsearch@master
|
|
||||||
with:
|
|
||||||
stack-version: 7.12.1
|
|
||||||
- uses: actions/checkout@v3
|
|
||||||
- uses: actions/setup-python@v4
|
|
||||||
with:
|
|
||||||
python-version: '3.9'
|
|
||||||
- if: matrix.test == 'other'
|
|
||||||
run: |
|
|
||||||
sudo apt-get update
|
|
||||||
sudo apt-get install -y --no-install-recommends ffmpeg
|
|
||||||
- name: extract pip cache
|
|
||||||
uses: actions/cache@v3
|
|
||||||
with:
|
|
||||||
path: ./.tox
|
|
||||||
key: tox-integration-${{ matrix.test }}-${{ hashFiles('setup.py') }}
|
|
||||||
restore-keys: txo-integration-${{ matrix.test }}-
|
|
||||||
- run: pip install tox coverage coveralls
|
|
||||||
- if: matrix.test == 'claims'
|
|
||||||
run: rm -rf .tox
|
|
||||||
- run: tox -e ${{ matrix.test }}
|
|
||||||
- name: submit coverage report
|
|
||||||
env:
|
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
COVERALLS_FLAG_NAME: tests-integration-${{ matrix.test }}
|
|
||||||
COVERALLS_PARALLEL: true
|
|
||||||
run: |
|
|
||||||
coverage combine tests
|
|
||||||
coveralls --service=github
|
|
||||||
|
|
||||||
|
|
||||||
coverage:
|
|
||||||
needs: ["tests-unit", "tests-integration"]
|
|
||||||
runs-on: ubuntu-20.04
|
|
||||||
steps:
|
|
||||||
- name: finalize coverage report submission
|
|
||||||
env:
|
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
run: |
|
|
||||||
pip install coveralls
|
|
||||||
coveralls --service=github --finish
|
|
||||||
|
|
||||||
build:
|
|
||||||
needs: ["lint", "tests-unit", "tests-integration"]
|
|
||||||
name: "build / binary"
|
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
os:
|
|
||||||
- ubuntu-20.04
|
|
||||||
- macos-latest
|
|
||||||
- windows-latest
|
|
||||||
runs-on: ${{ matrix.os }}
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v3
|
|
||||||
- uses: actions/setup-python@v4
|
|
||||||
with:
|
|
||||||
python-version: '3.9'
|
|
||||||
- id: os-name
|
|
||||||
uses: ASzc/change-string-case-action@v5
|
|
||||||
with:
|
|
||||||
string: ${{ runner.os }}
|
|
||||||
- name: set pip cache dir
|
|
||||||
shell: bash
|
|
||||||
run: echo "PIP_CACHE_DIR=$(pip cache dir)" >> $GITHUB_ENV
|
|
||||||
- name: extract pip cache
|
|
||||||
uses: actions/cache@v3
|
|
||||||
with:
|
|
||||||
path: ${{ env.PIP_CACHE_DIR }}
|
|
||||||
key: ${{ runner.os }}-pip-${{ hashFiles('setup.py') }}
|
|
||||||
restore-keys: ${{ runner.os }}-pip-
|
|
||||||
- run: pip install pyinstaller==4.6
|
|
||||||
- run: pip install -e .
|
|
||||||
- if: startsWith(github.ref, 'refs/tags/v')
|
|
||||||
run: python docker/set_build.py
|
|
||||||
- if: startsWith(runner.os, 'linux') || startsWith(runner.os, 'mac')
|
|
||||||
name: Build & Run (Unix)
|
|
||||||
run: |
|
|
||||||
pyinstaller --onefile --name lbrynet lbry/extras/cli.py
|
|
||||||
dist/lbrynet --version
|
|
||||||
- if: startsWith(runner.os, 'windows')
|
|
||||||
name: Build & Run (Windows)
|
|
||||||
run: |
|
|
||||||
pip install pywin32==301
|
|
||||||
pyinstaller --additional-hooks-dir=scripts/. --icon=icons/lbry256.ico --onefile --name lbrynet lbry/extras/cli.py
|
|
||||||
dist/lbrynet.exe --version
|
|
||||||
- uses: actions/upload-artifact@v3
|
|
||||||
with:
|
|
||||||
name: lbrynet-${{ steps.os-name.outputs.lowercase }}
|
|
||||||
path: dist/
|
|
||||||
|
|
||||||
release:
|
|
||||||
name: "release"
|
|
||||||
if: startsWith(github.ref, 'refs/tags/v')
|
|
||||||
needs: ["build"]
|
|
||||||
runs-on: ubuntu-20.04
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v1
|
|
||||||
- uses: actions/download-artifact@v2
|
|
||||||
- name: upload binaries
|
|
||||||
env:
|
|
||||||
GITHUB_TOKEN: ${{ secrets.RELEASE_API_TOKEN }}
|
|
||||||
run: |
|
|
||||||
pip install githubrelease
|
|
||||||
chmod +x lbrynet-macos/lbrynet
|
|
||||||
chmod +x lbrynet-linux/lbrynet
|
|
||||||
zip --junk-paths lbrynet-mac.zip lbrynet-macos/lbrynet
|
|
||||||
zip --junk-paths lbrynet-linux.zip lbrynet-linux/lbrynet
|
|
||||||
zip --junk-paths lbrynet-windows.zip lbrynet-windows/lbrynet.exe
|
|
||||||
ls -lh
|
|
||||||
githubrelease release lbryio/lbry-sdk info ${GITHUB_REF#refs/tags/}
|
|
||||||
githubrelease asset lbryio/lbry-sdk upload ${GITHUB_REF#refs/tags/} \
|
|
||||||
lbrynet-mac.zip lbrynet-linux.zip lbrynet-windows.zip
|
|
||||||
githubrelease release lbryio/lbry-sdk publish ${GITHUB_REF#refs/tags/}
|
|
||||||
|
|
22
.github/workflows/release.yml
vendored
22
.github/workflows/release.yml
vendored
|
@ -1,22 +0,0 @@
|
||||||
name: slack
|
|
||||||
|
|
||||||
on:
|
|
||||||
release:
|
|
||||||
types: [published]
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
release:
|
|
||||||
name: "slack notification"
|
|
||||||
runs-on: ubuntu-20.04
|
|
||||||
steps:
|
|
||||||
- uses: LoveToKnow/slackify-markdown-action@v1.0.0
|
|
||||||
id: markdown
|
|
||||||
with:
|
|
||||||
text: "There is a new SDK release: ${{github.event.release.html_url}}\n${{ github.event.release.body }}"
|
|
||||||
- uses: slackapi/slack-github-action@v1.14.0
|
|
||||||
env:
|
|
||||||
CHANGELOG: '<!channel> ${{ steps.markdown.outputs.text }}'
|
|
||||||
SLACK_WEBHOOK_URL: ${{ secrets.SLACK_RELEASE_BOT_WEBHOOK }}
|
|
||||||
with:
|
|
||||||
payload: '{"type": "mrkdwn", "text": ${{ toJSON(env.CHANGELOG) }} }'
|
|
||||||
|
|
18
.gitignore
vendored
18
.gitignore
vendored
|
@ -1,22 +1,10 @@
|
||||||
/.idea
|
|
||||||
/.DS_Store
|
|
||||||
/build
|
/build
|
||||||
/dist
|
/dist
|
||||||
/.tox
|
/.tox
|
||||||
/.coverage*
|
/.idea
|
||||||
|
/.coverage
|
||||||
/lbry-venv
|
/lbry-venv
|
||||||
/venv
|
|
||||||
/lbry/blockchain
|
|
||||||
|
|
||||||
lbry.egg-info
|
lbrynet.egg-info
|
||||||
__pycache__
|
__pycache__
|
||||||
_trial_temp/
|
_trial_temp/
|
||||||
trending*.log
|
|
||||||
|
|
||||||
/tests/integration/claims/files
|
|
||||||
/tests/.coverage.*
|
|
||||||
|
|
||||||
/lbry/wallet/bin
|
|
||||||
|
|
||||||
/.vscode
|
|
||||||
/.gitignore
|
|
||||||
|
|
445
.pylintrc
Normal file
445
.pylintrc
Normal file
|
@ -0,0 +1,445 @@
|
||||||
|
[MASTER]
|
||||||
|
|
||||||
|
# Specify a configuration file.
|
||||||
|
#rcfile=
|
||||||
|
|
||||||
|
# Python code to execute, usually for sys.path manipulation such as
|
||||||
|
# pygtk.require().
|
||||||
|
#init-hook=
|
||||||
|
|
||||||
|
# Add files or directories to the blacklist. They should be base names, not
|
||||||
|
# paths.
|
||||||
|
ignore=CVS,schema
|
||||||
|
|
||||||
|
# Add files or directories matching the regex patterns to the
|
||||||
|
# blacklist. The regex matches against base names, not paths.
|
||||||
|
# `\.#.*` - add emacs tmp files to the blacklist
|
||||||
|
ignore-patterns=\.#.*
|
||||||
|
|
||||||
|
# Pickle collected data for later comparisons.
|
||||||
|
persistent=yes
|
||||||
|
|
||||||
|
# List of plugins (as comma separated values of python modules names) to load,
|
||||||
|
# usually to register additional checkers.
|
||||||
|
load-plugins=
|
||||||
|
|
||||||
|
# Use multiple processes to speed up Pylint.
|
||||||
|
jobs=1
|
||||||
|
|
||||||
|
# Allow loading of arbitrary C extensions. Extensions are imported into the
|
||||||
|
# active Python interpreter and may run arbitrary code.
|
||||||
|
unsafe-load-any-extension=no
|
||||||
|
|
||||||
|
# A comma-separated list of package or module names from where C extensions may
|
||||||
|
# be loaded. Extensions are loading into the active Python interpreter and may
|
||||||
|
# run arbitrary code
|
||||||
|
# extension-pkg-whitelist=
|
||||||
|
|
||||||
|
# Allow optimization of some AST trees. This will activate a peephole AST
|
||||||
|
# optimizer, which will apply various small optimizations. For instance, it can
|
||||||
|
# be used to obtain the result of joining multiple strings with the addition
|
||||||
|
# operator. Joining a lot of strings can lead to a maximum recursion error in
|
||||||
|
# Pylint and this flag can prevent that. It has one side effect, the resulting
|
||||||
|
# AST will be different than the one from reality.
|
||||||
|
optimize-ast=no
|
||||||
|
|
||||||
|
|
||||||
|
[MESSAGES CONTROL]
|
||||||
|
|
||||||
|
# Only show warnings with the listed confidence levels. Leave empty to show
|
||||||
|
# all. Valid levels: HIGH, INFERENCE, INFERENCE_FAILURE, UNDEFINED
|
||||||
|
confidence=
|
||||||
|
|
||||||
|
# Enable the message, report, category or checker with the given id(s). You can
|
||||||
|
# either give multiple identifier separated by comma (,) or put this option
|
||||||
|
# multiple time (only on the command line, not in the configuration file where
|
||||||
|
# it should appear only once). See also the "--disable" option for examples.
|
||||||
|
#enable=
|
||||||
|
|
||||||
|
# Disable the message, report, category or checker with the given id(s). You
|
||||||
|
# can either give multiple identifiers separated by comma (,) or put this
|
||||||
|
# option multiple times (only on the command line, not in the configuration
|
||||||
|
# file where it should appear only once).You can also use "--disable=all" to
|
||||||
|
# disable everything first and then re-enable specific checks. For example, if
|
||||||
|
# you want to run only the similarities checker, you can use "--disable=all
|
||||||
|
# --enable=similarities". If you want to run only the classes checker, but have
|
||||||
|
# no Warning level messages displayed, use"--disable=all --enable=classes
|
||||||
|
# --disable=W"
|
||||||
|
disable=
|
||||||
|
anomalous-backslash-in-string,
|
||||||
|
arguments-differ,
|
||||||
|
attribute-defined-outside-init,
|
||||||
|
bad-continuation,
|
||||||
|
bare-except,
|
||||||
|
broad-except,
|
||||||
|
cell-var-from-loop,
|
||||||
|
consider-iterating-dictionary,
|
||||||
|
dangerous-default-value,
|
||||||
|
duplicate-code,
|
||||||
|
fixme,
|
||||||
|
global-statement,
|
||||||
|
inherit-non-class,
|
||||||
|
invalid-name,
|
||||||
|
len-as-condition,
|
||||||
|
locally-disabled,
|
||||||
|
logging-not-lazy,
|
||||||
|
missing-docstring,
|
||||||
|
no-else-return,
|
||||||
|
no-init,
|
||||||
|
no-member,
|
||||||
|
no-self-use,
|
||||||
|
protected-access,
|
||||||
|
redefined-builtin,
|
||||||
|
redefined-outer-name,
|
||||||
|
redefined-variable-type,
|
||||||
|
relative-import,
|
||||||
|
signature-differs,
|
||||||
|
super-init-not-called,
|
||||||
|
too-few-public-methods,
|
||||||
|
too-many-arguments,
|
||||||
|
too-many-branches,
|
||||||
|
too-many-instance-attributes,
|
||||||
|
too-many-lines,
|
||||||
|
too-many-locals,
|
||||||
|
too-many-nested-blocks,
|
||||||
|
too-many-public-methods,
|
||||||
|
too-many-return-statements,
|
||||||
|
too-many-statements,
|
||||||
|
trailing-newlines,
|
||||||
|
undefined-loop-variable,
|
||||||
|
ungrouped-imports,
|
||||||
|
unnecessary-lambda,
|
||||||
|
unused-argument,
|
||||||
|
unused-variable,
|
||||||
|
wildcard-import,
|
||||||
|
wrong-import-order,
|
||||||
|
wrong-import-position,
|
||||||
|
deprecated-lambda,
|
||||||
|
simplifiable-if-statement,
|
||||||
|
unidiomatic-typecheck,
|
||||||
|
global-at-module-level,
|
||||||
|
inconsistent-return-statements,
|
||||||
|
keyword-arg-before-vararg,
|
||||||
|
assignment-from-no-return,
|
||||||
|
useless-return,
|
||||||
|
assignment-from-none,
|
||||||
|
stop-iteration-return
|
||||||
|
|
||||||
|
|
||||||
|
[REPORTS]
|
||||||
|
|
||||||
|
# Set the output format. Available formats are text, parseable, colorized, msvs
|
||||||
|
# (visual studio) and html. You can also give a reporter class, eg
|
||||||
|
# mypackage.mymodule.MyReporterClass.
|
||||||
|
output-format=text
|
||||||
|
|
||||||
|
# Put messages in a separate file for each module / package specified on the
|
||||||
|
# command line instead of printing them on stdout. Reports (if any) will be
|
||||||
|
# written in a file name "pylint_global.[txt|html]".
|
||||||
|
files-output=no
|
||||||
|
|
||||||
|
# Tells whether to display a full report or only the messages
|
||||||
|
reports=no
|
||||||
|
|
||||||
|
# Python expression which should return a note less than 10 (10 is the highest
|
||||||
|
# note). You have access to the variables errors warning, statement which
|
||||||
|
# respectively contain the number of errors / warnings messages and the total
|
||||||
|
# number of statements analyzed. This is used by the global evaluation report
|
||||||
|
# (RP0004).
|
||||||
|
evaluation=10.0 - ((float(5 * error + warning + refactor + convention) / statement) * 10)
|
||||||
|
|
||||||
|
# Template used to display messages. This is a python new-style format string
|
||||||
|
# used to format the message information. See doc for all details
|
||||||
|
#msg-template=
|
||||||
|
|
||||||
|
|
||||||
|
[VARIABLES]
|
||||||
|
|
||||||
|
# Tells whether we should check for unused import in __init__ files.
|
||||||
|
init-import=no
|
||||||
|
|
||||||
|
# A regular expression matching the name of dummy variables (i.e. expectedly
|
||||||
|
# not used).
|
||||||
|
dummy-variables-rgx=_$|dummy
|
||||||
|
|
||||||
|
# List of additional names supposed to be defined in builtins. Remember that
|
||||||
|
# you should avoid to define new builtins when possible.
|
||||||
|
additional-builtins=
|
||||||
|
|
||||||
|
# List of strings which can identify a callback function by name. A callback
|
||||||
|
# name must start or end with one of those strings.
|
||||||
|
callbacks=cb_,_cb
|
||||||
|
|
||||||
|
|
||||||
|
[LOGGING]
|
||||||
|
|
||||||
|
# Logging modules to check that the string format arguments are in logging
|
||||||
|
# function parameter format
|
||||||
|
logging-modules=logging
|
||||||
|
|
||||||
|
|
||||||
|
[BASIC]
|
||||||
|
|
||||||
|
# List of builtins function names that should not be used, separated by a comma
|
||||||
|
bad-functions=map,filter,input
|
||||||
|
|
||||||
|
# Good variable names which should always be accepted, separated by a comma
|
||||||
|
# allow `d` as its used frequently for deferred callback chains
|
||||||
|
good-names=i,j,k,ex,Run,_,d
|
||||||
|
|
||||||
|
# Bad variable names which should always be refused, separated by a comma
|
||||||
|
bad-names=foo,bar,baz,toto,tutu,tata
|
||||||
|
|
||||||
|
# Colon-delimited sets of names that determine each other's naming style when
|
||||||
|
# the name regexes allow several styles.
|
||||||
|
name-group=
|
||||||
|
|
||||||
|
# Include a hint for the correct naming format with invalid-name
|
||||||
|
include-naming-hint=no
|
||||||
|
|
||||||
|
# Regular expression matching correct function names
|
||||||
|
function-rgx=[a-z_][a-z0-9_]{2,30}$
|
||||||
|
|
||||||
|
# Naming hint for function names
|
||||||
|
function-name-hint=[a-z_][a-z0-9_]{2,30}$
|
||||||
|
|
||||||
|
# Regular expression matching correct variable names
|
||||||
|
variable-rgx=[a-z_][a-z0-9_]{2,30}$
|
||||||
|
|
||||||
|
# Naming hint for variable names
|
||||||
|
variable-name-hint=[a-z_][a-z0-9_]{2,30}$
|
||||||
|
|
||||||
|
# Regular expression matching correct constant names
|
||||||
|
const-rgx=(([A-Z_][A-Z0-9_]*)|(__.*__))$
|
||||||
|
|
||||||
|
# Naming hint for constant names
|
||||||
|
const-name-hint=(([A-Z_][A-Z0-9_]*)|(__.*__))$
|
||||||
|
|
||||||
|
# Regular expression matching correct attribute names
|
||||||
|
attr-rgx=[a-z_][a-z0-9_]{2,30}$
|
||||||
|
|
||||||
|
# Naming hint for attribute names
|
||||||
|
attr-name-hint=[a-z_][a-z0-9_]{2,30}$
|
||||||
|
|
||||||
|
# Regular expression matching correct argument names
|
||||||
|
argument-rgx=[a-z_][a-z0-9_]{2,30}$
|
||||||
|
|
||||||
|
# Naming hint for argument names
|
||||||
|
argument-name-hint=[a-z_][a-z0-9_]{2,30}$
|
||||||
|
|
||||||
|
# Regular expression matching correct class attribute names
|
||||||
|
class-attribute-rgx=([A-Za-z_][A-Za-z0-9_]{2,30}|(__.*__))$
|
||||||
|
|
||||||
|
# Naming hint for class attribute names
|
||||||
|
class-attribute-name-hint=([A-Za-z_][A-Za-z0-9_]{2,30}|(__.*__))$
|
||||||
|
|
||||||
|
# Regular expression matching correct inline iteration names
|
||||||
|
inlinevar-rgx=[A-Za-z_][A-Za-z0-9_]*$
|
||||||
|
|
||||||
|
# Naming hint for inline iteration names
|
||||||
|
inlinevar-name-hint=[A-Za-z_][A-Za-z0-9_]*$
|
||||||
|
|
||||||
|
# Regular expression matching correct class names
|
||||||
|
class-rgx=[A-Z_][a-zA-Z0-9]+$
|
||||||
|
|
||||||
|
# Naming hint for class names
|
||||||
|
class-name-hint=[A-Z_][a-zA-Z0-9]+$
|
||||||
|
|
||||||
|
# Regular expression matching correct module names
|
||||||
|
module-rgx=(([a-z_][a-z0-9_]*)|([A-Z][a-zA-Z0-9]+))$
|
||||||
|
|
||||||
|
# Naming hint for module names
|
||||||
|
module-name-hint=(([a-z_][a-z0-9_]*)|([A-Z][a-zA-Z0-9]+))$
|
||||||
|
|
||||||
|
# Regular expression matching correct method names
|
||||||
|
method-rgx=[a-z_][a-z0-9_]{2,30}$
|
||||||
|
|
||||||
|
# Naming hint for method names
|
||||||
|
method-name-hint=[a-z_][a-z0-9_]{2,30}$
|
||||||
|
|
||||||
|
# Regular expression which should only match function or class names that do
|
||||||
|
# not require a docstring.
|
||||||
|
no-docstring-rgx=^_
|
||||||
|
|
||||||
|
# Minimum line length for functions/classes that require docstrings, shorter
|
||||||
|
# ones are exempt.
|
||||||
|
docstring-min-length=-1
|
||||||
|
|
||||||
|
|
||||||
|
[ELIF]
|
||||||
|
|
||||||
|
# Maximum number of nested blocks for function / method body
|
||||||
|
max-nested-blocks=5
|
||||||
|
|
||||||
|
|
||||||
|
[SPELLING]
|
||||||
|
|
||||||
|
# Spelling dictionary name. Available dictionaries: none. To make it working
|
||||||
|
# install python-enchant package.
|
||||||
|
spelling-dict=
|
||||||
|
|
||||||
|
# List of comma separated words that should not be checked.
|
||||||
|
spelling-ignore-words=
|
||||||
|
|
||||||
|
# A path to a file that contains private dictionary; one word per line.
|
||||||
|
spelling-private-dict-file=
|
||||||
|
|
||||||
|
# Tells whether to store unknown words to indicated private dictionary in
|
||||||
|
# --spelling-private-dict-file option instead of raising a message.
|
||||||
|
spelling-store-unknown-words=no
|
||||||
|
|
||||||
|
|
||||||
|
[FORMAT]
|
||||||
|
|
||||||
|
# Maximum number of characters on a single line.
|
||||||
|
max-line-length=120
|
||||||
|
|
||||||
|
# Regexp for a line that is allowed to be longer than the limit.
|
||||||
|
ignore-long-lines=^\s*(# )?<?https?://\S+>?$
|
||||||
|
|
||||||
|
# Allow the body of an if to be on the same line as the test if there is no
|
||||||
|
# else.
|
||||||
|
single-line-if-stmt=no
|
||||||
|
|
||||||
|
# List of optional constructs for which whitespace checking is disabled. `dict-
|
||||||
|
# separator` is used to allow tabulation in dicts, etc.: {1 : 1,\n222: 2}.
|
||||||
|
# `trailing-comma` allows a space between comma and closing bracket: (a, ).
|
||||||
|
# `empty-line` allows space-only lines.
|
||||||
|
no-space-check=trailing-comma,dict-separator
|
||||||
|
|
||||||
|
# Maximum number of lines in a module
|
||||||
|
max-module-lines=1000
|
||||||
|
|
||||||
|
# String used as indentation unit. This is usually " " (4 spaces) or "\t" (1
|
||||||
|
# tab).
|
||||||
|
indent-string=' '
|
||||||
|
|
||||||
|
# Number of spaces of indent required inside a hanging or continued line.
|
||||||
|
indent-after-paren=4
|
||||||
|
|
||||||
|
# Expected format of line ending, e.g. empty (any line ending), LF or CRLF.
|
||||||
|
expected-line-ending-format=
|
||||||
|
|
||||||
|
|
||||||
|
[MISCELLANEOUS]
|
||||||
|
|
||||||
|
# List of note tags to take in consideration, separated by a comma.
|
||||||
|
notes=FIXME,XXX,TODO
|
||||||
|
|
||||||
|
|
||||||
|
[SIMILARITIES]
|
||||||
|
|
||||||
|
# Minimum lines number of a similarity.
|
||||||
|
min-similarity-lines=4
|
||||||
|
|
||||||
|
# Ignore comments when computing similarities.
|
||||||
|
ignore-comments=yes
|
||||||
|
|
||||||
|
# Ignore docstrings when computing similarities.
|
||||||
|
ignore-docstrings=yes
|
||||||
|
|
||||||
|
# Ignore imports when computing similarities.
|
||||||
|
ignore-imports=no
|
||||||
|
|
||||||
|
|
||||||
|
[TYPECHECK]
|
||||||
|
|
||||||
|
# Tells whether missing members accessed in mixin class should be ignored. A
|
||||||
|
# mixin class is detected if its name ends with "mixin" (case insensitive).
|
||||||
|
ignore-mixin-members=yes
|
||||||
|
|
||||||
|
# List of module names for which member attributes should not be checked
|
||||||
|
# (useful for modules/projects where namespaces are manipulated during runtime
|
||||||
|
# and thus existing member attributes cannot be deduced by static analysis. It
|
||||||
|
# supports qualified module names, as well as Unix pattern matching.
|
||||||
|
ignored-modules=leveldb,distutils
|
||||||
|
# Ignoring distutils because: https://github.com/PyCQA/pylint/issues/73
|
||||||
|
|
||||||
|
# List of classes names for which member attributes should not be checked
|
||||||
|
# (useful for classes with attributes dynamically set). This supports can work
|
||||||
|
# with qualified names.
|
||||||
|
# ignored-classes=
|
||||||
|
|
||||||
|
|
||||||
|
# List of members which are set dynamically and missed by pylint inference
|
||||||
|
# system, and so shouldn't trigger E1101 when accessed. Python regular
|
||||||
|
# expressions are accepted.
|
||||||
|
generated-members=lbrynet.lbrynet_daemon.LBRYDaemon.Parameters
|
||||||
|
|
||||||
|
|
||||||
|
[IMPORTS]
|
||||||
|
|
||||||
|
# Deprecated modules which should not be used, separated by a comma
|
||||||
|
deprecated-modules=regsub,TERMIOS,Bastion,rexec
|
||||||
|
|
||||||
|
# Create a graph of every (i.e. internal and external) dependencies in the
|
||||||
|
# given file (report RP0402 must not be disabled)
|
||||||
|
import-graph=
|
||||||
|
|
||||||
|
# Create a graph of external dependencies in the given file (report RP0402 must
|
||||||
|
# not be disabled)
|
||||||
|
ext-import-graph=
|
||||||
|
|
||||||
|
# Create a graph of internal dependencies in the given file (report RP0402 must
|
||||||
|
# not be disabled)
|
||||||
|
int-import-graph=
|
||||||
|
|
||||||
|
|
||||||
|
[DESIGN]
|
||||||
|
|
||||||
|
# Maximum number of arguments for function / method
|
||||||
|
max-args=10
|
||||||
|
|
||||||
|
# Argument names that match this expression will be ignored. Default to name
|
||||||
|
# with leading underscore
|
||||||
|
ignored-argument-names=_.*
|
||||||
|
|
||||||
|
# Maximum number of locals for function / method body
|
||||||
|
max-locals=15
|
||||||
|
|
||||||
|
# Maximum number of return / yield for function / method body
|
||||||
|
max-returns=6
|
||||||
|
|
||||||
|
# Maximum number of branch for function / method body
|
||||||
|
max-branches=12
|
||||||
|
|
||||||
|
# Maximum number of statements in function / method body
|
||||||
|
max-statements=50
|
||||||
|
|
||||||
|
# Maximum number of parents for a class (see R0901).
|
||||||
|
max-parents=8
|
||||||
|
|
||||||
|
# Maximum number of attributes for a class (see R0902).
|
||||||
|
max-attributes=7
|
||||||
|
|
||||||
|
# Minimum number of public methods for a class (see R0903).
|
||||||
|
min-public-methods=2
|
||||||
|
|
||||||
|
# Maximum number of public methods for a class (see R0904).
|
||||||
|
max-public-methods=20
|
||||||
|
|
||||||
|
# Maximum number of boolean expressions in a if statement
|
||||||
|
max-bool-expr=5
|
||||||
|
|
||||||
|
|
||||||
|
[CLASSES]
|
||||||
|
|
||||||
|
# List of method names used to declare (i.e. assign) instance attributes.
|
||||||
|
defining-attr-methods=__init__,__new__,setUp
|
||||||
|
|
||||||
|
# List of valid names for the first argument in a class method.
|
||||||
|
valid-classmethod-first-arg=cls
|
||||||
|
|
||||||
|
# List of valid names for the first argument in a metaclass class method.
|
||||||
|
valid-metaclass-classmethod-first-arg=mcs
|
||||||
|
|
||||||
|
# List of member names, which should be excluded from the protected access
|
||||||
|
# warning.
|
||||||
|
exclude-protected=_asdict,_fields,_replace,_source,_make
|
||||||
|
|
||||||
|
|
||||||
|
[EXCEPTIONS]
|
||||||
|
|
||||||
|
# Exceptions that will emit a warning when being caught. Defaults to
|
||||||
|
# "Exception"
|
||||||
|
overgeneral-exceptions=Exception
|
128
.travis.yml
Normal file
128
.travis.yml
Normal file
|
@ -0,0 +1,128 @@
|
||||||
|
sudo: required
|
||||||
|
dist: xenial
|
||||||
|
language: python
|
||||||
|
python: "3.7"
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
include:
|
||||||
|
|
||||||
|
- stage: code quality
|
||||||
|
name: "pylint lbrynet"
|
||||||
|
install:
|
||||||
|
- pip install astroid pylint
|
||||||
|
- pip install git+https://github.com/lbryio/torba.git#egg=torba
|
||||||
|
- pip install -e .
|
||||||
|
script: pylint lbrynet
|
||||||
|
|
||||||
|
- stage: test
|
||||||
|
name: "Unit Tests"
|
||||||
|
install:
|
||||||
|
- pip install coverage
|
||||||
|
- pip install git+https://github.com/lbryio/torba.git#egg=torba
|
||||||
|
- pip install -e .
|
||||||
|
script:
|
||||||
|
- HOME=/tmp coverage run -p --source=lbrynet -m unittest discover -vv tests.unit
|
||||||
|
after_success:
|
||||||
|
- coverage combine
|
||||||
|
- bash <(curl -s https://codecov.io/bash)
|
||||||
|
|
||||||
|
- name: "Integration Tests"
|
||||||
|
install:
|
||||||
|
- pip install tox-travis coverage
|
||||||
|
- pushd .. && git clone https://github.com/lbryio/torba.git && popd
|
||||||
|
script: tox
|
||||||
|
after_success:
|
||||||
|
- coverage combine tests/
|
||||||
|
- bash <(curl -s https://codecov.io/bash)
|
||||||
|
|
||||||
|
- stage: build
|
||||||
|
name: "Windows"
|
||||||
|
language: generic
|
||||||
|
services:
|
||||||
|
- docker
|
||||||
|
install:
|
||||||
|
- docker pull lbry/pyinstaller34_32bits:py371
|
||||||
|
script:
|
||||||
|
- python scripts/set_build.py
|
||||||
|
- docker run -v "$(pwd):/src/lbry" lbry/pyinstaller34_32bits:py371 lbry/scripts/wine_build.sh
|
||||||
|
- sudo zip -j dist/lbrynet-windows.zip dist/lbrynet.exe
|
||||||
|
deploy:
|
||||||
|
provider: releases
|
||||||
|
api_key: $GITHUB_OAUTH_TOKEN
|
||||||
|
file: dist/lbrynet-windows.zip
|
||||||
|
skip_cleanup: true
|
||||||
|
overwrite: true
|
||||||
|
draft: true
|
||||||
|
on:
|
||||||
|
tags: true
|
||||||
|
addons:
|
||||||
|
artifacts:
|
||||||
|
working_dir: dist
|
||||||
|
paths:
|
||||||
|
- lbrynet-windows.zip
|
||||||
|
target_paths:
|
||||||
|
- /daemon/build-${TRAVIS_BUILD_NUMBER}_commit-${TRAVIS_COMMIT:0:7}_branch-${TRAVIS_BRANCH}$([ ! -z ${TRAVIS_TAG} ] && echo _tag-${TRAVIS_TAG})
|
||||||
|
|
||||||
|
- &build
|
||||||
|
name: "Linux"
|
||||||
|
env: OS=linux
|
||||||
|
install:
|
||||||
|
- pip3 install pyinstaller
|
||||||
|
- pip3 install git+https://github.com/lbryio/torba.git
|
||||||
|
- python scripts/set_build.py
|
||||||
|
- pip3 install -e .
|
||||||
|
script:
|
||||||
|
- pyinstaller -F -n lbrynet lbrynet/extras/cli.py
|
||||||
|
- chmod +x dist/lbrynet
|
||||||
|
- zip -j dist/lbrynet-${OS}.zip dist/lbrynet
|
||||||
|
- ./dist/lbrynet --version
|
||||||
|
deploy:
|
||||||
|
provider: releases
|
||||||
|
api_key: $GITHUB_OAUTH_TOKEN
|
||||||
|
file: dist/lbrynet-${OS}.zip
|
||||||
|
skip_cleanup: true
|
||||||
|
overwrite: true
|
||||||
|
draft: true
|
||||||
|
on:
|
||||||
|
tags: true
|
||||||
|
addons:
|
||||||
|
artifacts:
|
||||||
|
working_dir: dist
|
||||||
|
paths:
|
||||||
|
- lbrynet-${OS}.zip
|
||||||
|
# artifact uploader thinks lbrynet is a directory, https://github.com/travis-ci/artifacts/issues/78
|
||||||
|
target_paths:
|
||||||
|
- /daemon/build-${TRAVIS_BUILD_NUMBER}_commit-${TRAVIS_COMMIT:0:7}_branch-${TRAVIS_BRANCH}$([ ! -z ${TRAVIS_TAG} ] && echo _tag-${TRAVIS_TAG})
|
||||||
|
|
||||||
|
- <<: *build
|
||||||
|
name: "Mac"
|
||||||
|
os: osx
|
||||||
|
osx_image: xcode8.3
|
||||||
|
language: generic
|
||||||
|
env: OS=mac
|
||||||
|
cache: false
|
||||||
|
before_install:
|
||||||
|
- brew upgrade python || true
|
||||||
|
- brew upgrade python || true
|
||||||
|
- if: tag IS present
|
||||||
|
stage: build
|
||||||
|
name: "Wallet Server Docker Image - Tagged Release"
|
||||||
|
script:
|
||||||
|
- set -e
|
||||||
|
- echo "$DOCKER_PASSWORD" | docker login --username "$DOCKER_USERNAME" --password-stdin
|
||||||
|
- travis_retry docker build -t lbry/wallet-server:$TRAVIS_TAG -f scripts/Dockerfile.wallet_server .
|
||||||
|
- docker push lbry/wallet-server:$TRAVIS_TAG
|
||||||
|
- if: tag IS blank AND branch = master
|
||||||
|
stage: build
|
||||||
|
name: "Wallet Server Docker Image - Master"
|
||||||
|
script:
|
||||||
|
- set -e
|
||||||
|
- echo "$DOCKER_PASSWORD" | docker login --username "$DOCKER_USERNAME" --password-stdin
|
||||||
|
- travis_retry docker build -t lbry/wallet-server:master -f scripts/Dockerfile.wallet_server .
|
||||||
|
- docker push lbry/wallet-server:master
|
||||||
|
|
||||||
|
cache:
|
||||||
|
directories:
|
||||||
|
- $HOME/.cache/pip
|
||||||
|
- $HOME/Library/Caches/pip
|
||||||
|
- $TRAVIS_BUILD_DIR/.tox
|
1019
CHANGELOG.md
1019
CHANGELOG.md
File diff suppressed because it is too large
Load diff
|
@ -1,3 +1,3 @@
|
||||||
## Contributing to LBRY
|
## Contributing to LBRY
|
||||||
|
|
||||||
https://lbry.tech/contribute
|
https://lbry.io/faq/contributing
|
||||||
|
|
6
Dangerfile
Normal file
6
Dangerfile
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
# Add a CHANGELOG entry for app changes
|
||||||
|
has_app_changes = !git.modified_files.grep(/lbrynet/).empty?
|
||||||
|
if !git.modified_files.include?("CHANGELOG.md") && has_app_changes
|
||||||
|
fail("Please include a CHANGELOG entry.")
|
||||||
|
message "See http://keepachangelog.com/en/0.3.0/ for details on good changelog guidelines"
|
||||||
|
end
|
156
INSTALL.md
156
INSTALL.md
|
@ -1,6 +1,6 @@
|
||||||
# Installing LBRY
|
# Installing LBRY
|
||||||
|
|
||||||
If only the JSON-RPC API server is needed, the recommended way to install LBRY is to use a pre-built binary. We provide binaries for all major operating systems. See the [README](README.md)!
|
If only the JSON-RPC API server is needed, the recommended way to install LBRY is to use a pre-built binary. We provide binaries for all major operating systems. See the [README](README.md).
|
||||||
|
|
||||||
These instructions are for installing LBRY from source, which is recommended if you are interested in doing development work or LBRY is not available on your operating system (godspeed, TempleOS users).
|
These instructions are for installing LBRY from source, which is recommended if you are interested in doing development work or LBRY is not available on your operating system (godspeed, TempleOS users).
|
||||||
|
|
||||||
|
@ -9,47 +9,36 @@ Here's a video walkthrough of this setup, which is itself hosted by the LBRY net
|
||||||
|
|
||||||
## Prerequisites
|
## Prerequisites
|
||||||
|
|
||||||
Running `lbrynet` from source requires Python 3.7. Get the installer for your OS [here](https://www.python.org/downloads/release/python-370/).
|
Running `lbrynet` from source requires Python 3.6 or higher (3.7 is preferred). Get the installer for your OS [here](https://www.python.org/downloads/release/python-370/)
|
||||||
|
|
||||||
After installing Python 3.7, you'll need to install some additional libraries depending on your operating system.
|
After installing python 3, you'll need to install some additional libraries depending on your operating system.
|
||||||
|
|
||||||
|
### Virtualenv
|
||||||
|
|
||||||
|
Once python 3 is installed run `python3 -m pip install virtualenv` to install virtualenv.
|
||||||
|
|
||||||
|
### Windows
|
||||||
|
|
||||||
|
Windows users will need to install `Visual C++ Build Tools`, which can be installed by [Microsoft Build Tools](Microsoft Build Tools 2015)
|
||||||
|
|
||||||
Because of [issue #2769](https://github.com/lbryio/lbry-sdk/issues/2769)
|
|
||||||
at the moment the `lbrynet` daemon will only work correctly with Python 3.7.
|
|
||||||
If Python 3.8+ is used, the daemon will start but the RPC server
|
|
||||||
may not accept messages, returning the following:
|
|
||||||
```
|
|
||||||
Could not connect to daemon. Are you sure it's running?
|
|
||||||
```
|
|
||||||
|
|
||||||
### macOS
|
### macOS
|
||||||
|
|
||||||
macOS users will need to install [xcode command line tools](https://developer.xamarin.com/guides/testcloud/calabash/configuring/osx/install-xcode-command-line-tools/) and [homebrew](http://brew.sh/).
|
macOS users will need to install [xcode command line tools](https://developer.xamarin.com/guides/testcloud/calabash/configuring/osx/install-xcode-command-line-tools/) and [homebrew](http://brew.sh/).
|
||||||
|
|
||||||
These environment variables also need to be set:
|
|
||||||
```
|
|
||||||
PYTHONUNBUFFERED=1
|
|
||||||
EVENT_NOKQUEUE=1
|
|
||||||
```
|
|
||||||
|
|
||||||
Remaining dependencies can then be installed by running:
|
Remaining dependencies can then be installed by running:
|
||||||
```
|
|
||||||
brew install python protobuf
|
|
||||||
```
|
|
||||||
|
|
||||||
Assistance installing Python3: https://docs.python-guide.org/starting/install3/osx/.
|
```
|
||||||
|
brew install python3 protobuf
|
||||||
|
```
|
||||||
|
|
||||||
### Linux
|
### Linux
|
||||||
|
|
||||||
On Ubuntu (we recommend 18.04 or 20.04), install the following:
|
On Ubuntu (we recommend 18.04), install the following:
|
||||||
```
|
|
||||||
sudo add-apt-repository ppa:deadsnakes/ppa
|
|
||||||
sudo apt-get update
|
|
||||||
sudo apt-get install build-essential python3.7 python3.7-dev git python3.7-venv libssl-dev python-protobuf
|
|
||||||
```
|
|
||||||
|
|
||||||
The [deadsnakes PPA](https://launchpad.net/~deadsnakes/+archive/ubuntu/ppa) provides Python 3.7
|
```
|
||||||
for those Ubuntu distributions that no longer have it in their
|
sudo apt-get install build-essential python3.7 python3.7-dev git python-virtualenv libssl-dev python-protobuf
|
||||||
official repositories.
|
```
|
||||||
|
|
||||||
On Raspbian, you will also need to install `python-pyparsing`.
|
On Raspbian, you will also need to install `python-pyparsing`.
|
||||||
|
|
||||||
|
@ -57,121 +46,32 @@ If you're running another Linux distro, install the equivalent of the above pack
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
### Linux/Mac
|
To install:
|
||||||
|
|
||||||
Clone the repository:
|
|
||||||
```bash
|
|
||||||
git clone https://github.com/lbryio/lbry-sdk.git
|
|
||||||
cd lbry-sdk
|
|
||||||
```
|
```
|
||||||
|
git clone https://github.com/lbryio/lbry.git
|
||||||
|
cd lbry
|
||||||
|
|
||||||
Create a Python virtual environment for lbry-sdk:
|
virtualenv lbry-venv --python=python3.7
|
||||||
```bash
|
|
||||||
python3.7 -m venv lbry-venv
|
|
||||||
```
|
|
||||||
|
|
||||||
Activate virtual environment:
|
|
||||||
```bash
|
|
||||||
source lbry-venv/bin/activate
|
source lbry-venv/bin/activate
|
||||||
```
|
|
||||||
|
|
||||||
Make sure you're on Python 3.7+ as default in the virtual environment:
|
python --version # Python 2 is not supported. Make sure you're on Python 3.7
|
||||||
```bash
|
|
||||||
python --version
|
|
||||||
```
|
|
||||||
|
|
||||||
Install packages:
|
|
||||||
```bash
|
|
||||||
make install
|
|
||||||
```
|
|
||||||
|
|
||||||
If you are on Linux and using PyCharm, generates initial configs:
|
|
||||||
```bash
|
|
||||||
make idea
|
|
||||||
```
|
|
||||||
|
|
||||||
To verify your installation, `which lbrynet` should return a path inside
|
|
||||||
of the `lbry-venv` folder.
|
|
||||||
```bash
|
|
||||||
(lbry-venv) $ which lbrynet
|
|
||||||
/opt/lbry-sdk/lbry-venv/bin/lbrynet
|
|
||||||
```
|
|
||||||
|
|
||||||
To exit the virtual environment simply use the command `deactivate`.
|
|
||||||
|
|
||||||
### Windows
|
|
||||||
|
|
||||||
Clone the repository:
|
|
||||||
```bash
|
|
||||||
git clone https://github.com/lbryio/lbry-sdk.git
|
|
||||||
cd lbry-sdk
|
|
||||||
```
|
|
||||||
|
|
||||||
Create a Python virtual environment for lbry-sdk:
|
|
||||||
```bash
|
|
||||||
python -m venv lbry-venv
|
|
||||||
```
|
|
||||||
|
|
||||||
Activate virtual environment:
|
|
||||||
```bash
|
|
||||||
lbry-venv\Scripts\activate
|
|
||||||
```
|
|
||||||
|
|
||||||
Install packages:
|
|
||||||
```bash
|
|
||||||
pip install -e .
|
pip install -e .
|
||||||
```
|
```
|
||||||
|
|
||||||
|
To verify your installation, `which lbrynet` should return a path inside of the `lbry-venv` folder created by the `virtualenv` command.
|
||||||
|
|
||||||
## Run the tests
|
## Run the tests
|
||||||
### Elasticsearch
|
To run the unit tests from the repo directory:
|
||||||
|
|
||||||
For running integration tests, Elasticsearch is required to be available at localhost:9200/
|
|
||||||
|
|
||||||
The easiest way to start it is using docker with:
|
|
||||||
```bash
|
|
||||||
make elastic-docker
|
|
||||||
```
|
```
|
||||||
|
trial --reactor=asyncio tests.unit
|
||||||
Alternative installation methods are available [at Elasticsearch website](https://www.elastic.co/guide/en/elasticsearch/reference/current/install-elasticsearch.html).
|
|
||||||
|
|
||||||
To run the unit and integration tests from the repo directory:
|
|
||||||
```
|
|
||||||
python -m unittest discover tests.unit
|
|
||||||
python -m unittest discover tests.integration
|
|
||||||
```
|
```
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
To start the API server:
|
To start the API server:
|
||||||
```
|
`lbrynet start`
|
||||||
lbrynet start
|
|
||||||
```
|
|
||||||
|
|
||||||
Whenever the code inside [lbry-sdk/lbry](./lbry)
|
|
||||||
is modified we should run `make install` to recompile the `lbrynet`
|
|
||||||
executable with the newest code.
|
|
||||||
|
|
||||||
## Development
|
|
||||||
|
|
||||||
When developing, remember to enter the environment,
|
|
||||||
and if you wish start the server interactively.
|
|
||||||
```bash
|
|
||||||
$ source lbry-venv/bin/activate
|
|
||||||
|
|
||||||
(lbry-venv) $ python lbry/extras/cli.py start
|
|
||||||
```
|
|
||||||
|
|
||||||
Parameters can be passed in the same way.
|
|
||||||
```bash
|
|
||||||
(lbry-venv) $ python lbry/extras/cli.py wallet balance
|
|
||||||
```
|
|
||||||
|
|
||||||
If a Python debugger (`pdb` or `ipdb`) is installed we can also start it
|
|
||||||
in this way, set up break points, and step through the code.
|
|
||||||
```bash
|
|
||||||
(lbry-venv) $ pip install ipdb
|
|
||||||
|
|
||||||
(lbry-venv) $ ipdb lbry/extras/cli.py
|
|
||||||
```
|
|
||||||
|
|
||||||
Happy hacking!
|
Happy hacking!
|
||||||
|
|
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-2019 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,
|
||||||
|
|
|
@ -1,4 +0,0 @@
|
||||||
include README.md
|
|
||||||
include CHANGELOG.md
|
|
||||||
include LICENSE
|
|
||||||
recursive-include lbry *.txt *.py
|
|
26
Makefile
26
Makefile
|
@ -1,26 +0,0 @@
|
||||||
.PHONY: install tools lint test test-unit test-unit-coverage test-integration idea
|
|
||||||
|
|
||||||
install:
|
|
||||||
pip install -e .
|
|
||||||
|
|
||||||
lint:
|
|
||||||
pylint --rcfile=setup.cfg lbry
|
|
||||||
#mypy --ignore-missing-imports lbry
|
|
||||||
|
|
||||||
test: test-unit test-integration
|
|
||||||
|
|
||||||
test-unit:
|
|
||||||
python -m unittest discover tests.unit
|
|
||||||
|
|
||||||
test-unit-coverage:
|
|
||||||
coverage run --source=lbry -m unittest discover -vv tests.unit
|
|
||||||
|
|
||||||
test-integration:
|
|
||||||
tox
|
|
||||||
|
|
||||||
idea:
|
|
||||||
mkdir -p .idea
|
|
||||||
cp -r scripts/idea/* .idea
|
|
||||||
|
|
||||||
elastic-docker:
|
|
||||||
docker run -d -v lbryhub:/usr/share/elasticsearch/data -p 9200:9200 -p 9300:9300 -e"ES_JAVA_OPTS=-Xms512m -Xmx512m" -e "discovery.type=single-node" docker.elastic.co/elasticsearch/elasticsearch:7.12.1
|
|
26
README.md
26
README.md
|
@ -1,19 +1,19 @@
|
||||||
# <img src="https://raw.githubusercontent.com/lbryio/lbry-sdk/master/lbry.png" alt="LBRY" width="48" height="36" /> LBRY SDK [![build](https://github.com/lbryio/lbry-sdk/actions/workflows/main.yml/badge.svg)](https://github.com/lbryio/lbry-sdk/actions/workflows/main.yml) [![coverage](https://coveralls.io/repos/github/lbryio/lbry-sdk/badge.svg)](https://coveralls.io/github/lbryio/lbry-sdk)
|
# <img src="https://raw.githubusercontent.com/lbryio/lbry/master/lbry.png" alt="LBRY" width="48" height="36" /> LBRY SDK [![Build Status](https://travis-ci.org/lbryio/lbry.svg?branch=master)](https://travis-ci.org/lbryio/lbry) [![Test Coverage](https://codecov.io/gh/lbryio/lbry/branch/master/graph/badge.svg)](https://codecov.io/gh/lbryio/lbry)
|
||||||
|
|
||||||
LBRY is a decentralized peer-to-peer protocol for publishing and accessing digital content. It utilizes the [LBRY blockchain](https://github.com/lbryio/lbrycrd) as a global namespace and database of digital content. Blockchain entries contain searchable content metadata, identities, rights and access rules. LBRY also provides a data network that consists of peers (seeders) uploading and downloading data from other peers, possibly in exchange for payments, as well as a distributed hash table used by peers to discover other peers.
|
LBRY is a decentralized peer-to-peer protocol for publishing and accessing digital content. It utilizes the [LBRY blockchain](https://github.com/lbryio/lbrycrd) as a global namespace and database of digital content. Blockchain entries contain searchable content metadata, identities, rights and access rules. LBRY also provides a data network that consists of peers (seeders) uploading and downloading data from other peers, possibly in exchange for payments, as well as a distributed hash table used by peers to discover other peers.
|
||||||
|
|
||||||
LBRY SDK for Python is currently the most fully featured implementation of the LBRY Network protocols and includes many useful components and tools for building decentralized applications. Primary features and components include:
|
LBRY SDK for Python is currently the most full featured implementation of the LBRY Network protocols and includes many useful components and tools for building decentralized applications. Primary features and components:
|
||||||
|
|
||||||
* Built on Python 3.7 and `asyncio`.
|
* Built on Python 3.7+ and `asyncio`.
|
||||||
* Kademlia DHT (Distributed Hash Table) implementation for finding peers to download from and announcing to peers what we have to host ([lbry.dht](https://github.com/lbryio/lbry-sdk/tree/master/lbry/dht)).
|
* Kademlia DHT (Distributed Hash Table) implementation for finding peers to download from and announcing to peers what we have to host ([lbrynet.dht](https://github.com/lbryio/lbry/tree/master/lbrynet/dht)).
|
||||||
* Blob exchange protocol for transferring encrypted blobs of content and negotiating payments ([lbry.blob_exchange](https://github.com/lbryio/lbry-sdk/tree/master/lbry/blob_exchange)).
|
* Blob exchange protocol for transferring encrypted blobs of content and negotiating payments ([lbrynet.blob_exchange](https://github.com/lbryio/lbry/tree/master/lbrynet/blob_exchange)).
|
||||||
* Protobuf schema for encoding and decoding metadata stored on the blockchain ([lbry.schema](https://github.com/lbryio/lbry-sdk/tree/master/lbry/schema)).
|
* Protobuf schema for encoding and decoding metadata stored on the blockchain ([lbrynet.schema](https://github.com/lbryio/lbry/tree/master/lbrynet/schema)).
|
||||||
* Wallet implementation for the LBRY blockchain ([lbry.wallet](https://github.com/lbryio/lbry-sdk/tree/master/lbry/wallet)).
|
* Wallet implementation for the LBRY blockchain ([lbrynet.extras.wallet](https://github.com/lbryio/lbry/tree/master/lbrynet/extras/wallet)).
|
||||||
* Daemon with a JSON-RPC API to ease building end user applications in any language and for automating various tasks ([lbry.extras.daemon](https://github.com/lbryio/lbry-sdk/tree/master/lbry/extras/daemon)).
|
* Daemon with a JSON-RPC API to ease building end user applications in any language and for automating various tasks ([lbrynet.extras.daemon](https://github.com/lbryio/lbry/tree/master/lbrynet/extras/daemon)).
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
Our [releases page](https://github.com/lbryio/lbry-sdk/releases) contains pre-built binaries of the latest release, pre-releases, and past releases for macOS, Debian-based Linux, and Windows. [Automated travis builds](http://build.lbry.io/daemon/) are also available for testing.
|
Our [releases page](https://github.com/lbryio/lbry/releases) contains pre-built binaries of the latest release, pre-releases, and past releases for macOS, Debian-based Linux, and Windows. [Automated travis builds](http://build.lbry.io/daemon/) are also available for testing.
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
|
@ -33,7 +33,7 @@ Installing from source is also relatively painless. Full instructions are in [IN
|
||||||
|
|
||||||
## Contributing
|
## Contributing
|
||||||
|
|
||||||
Contributions to this project are welcome, encouraged, and compensated. For more details, please check [this](https://lbry.tech/contribute) link.
|
Contributions to this project are welcome, encouraged, and compensated. For more details, please check [this](https://lbry.io/faq/contributing) link.
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
|
@ -41,11 +41,11 @@ This project is MIT licensed. For the full license, see [LICENSE](LICENSE).
|
||||||
|
|
||||||
## Security
|
## Security
|
||||||
|
|
||||||
We take security seriously. Please contact security@lbry.com regarding any security issues. [Our PGP key is here](https://lbry.com/faq/pgp-key) if you need it.
|
We take security seriously. Please contact security@lbry.io regarding any security issues. [Our GPG key is here](https://lbry.io/faq/gpg-key) if you need it.
|
||||||
|
|
||||||
## Contact
|
## Contact
|
||||||
|
|
||||||
The primary contact for this project is [@eukreign](mailto:lex@lbry.com).
|
The primary contact for this project is [@eukreign](mailto:lex@lbry.io).
|
||||||
|
|
||||||
## Additional information and links
|
## Additional information and links
|
||||||
|
|
||||||
|
@ -53,4 +53,4 @@ The documentation for the API can be found [here](https://lbry.tech/api/sdk).
|
||||||
|
|
||||||
Daemon defaults, ports, and other settings are documented [here](https://lbry.tech/resources/daemon-settings).
|
Daemon defaults, ports, and other settings are documented [here](https://lbry.tech/resources/daemon-settings).
|
||||||
|
|
||||||
Settings can be configured using a daemon-settings.yml file. An example can be found [here](https://github.com/lbryio/lbry-sdk/blob/master/example_daemon_settings.yml).
|
Settings can be configured using a daemon-settings.yml file. An example can be found [here](https://github.com/lbryio/lbry/blob/master/example_daemon_settings.yml).
|
||||||
|
|
|
@ -1,9 +0,0 @@
|
||||||
# Security Policy
|
|
||||||
|
|
||||||
## Supported Versions
|
|
||||||
|
|
||||||
While we are not at v1.0 yet, only the latest release will be supported.
|
|
||||||
|
|
||||||
## Reporting a Vulnerability
|
|
||||||
|
|
||||||
See https://lbry.com/faq/security
|
|
|
@ -1,43 +0,0 @@
|
||||||
FROM debian:10-slim
|
|
||||||
|
|
||||||
ARG user=lbry
|
|
||||||
ARG projects_dir=/home/$user
|
|
||||||
ARG db_dir=/database
|
|
||||||
|
|
||||||
ARG DOCKER_TAG
|
|
||||||
ARG DOCKER_COMMIT=docker
|
|
||||||
ENV DOCKER_TAG=$DOCKER_TAG DOCKER_COMMIT=$DOCKER_COMMIT
|
|
||||||
|
|
||||||
RUN apt-get update && \
|
|
||||||
apt-get -y --no-install-recommends install \
|
|
||||||
wget \
|
|
||||||
automake libtool \
|
|
||||||
tar unzip \
|
|
||||||
build-essential \
|
|
||||||
pkg-config \
|
|
||||||
libleveldb-dev \
|
|
||||||
python3.7 \
|
|
||||||
python3-dev \
|
|
||||||
python3-pip \
|
|
||||||
python3-wheel \
|
|
||||||
python3-setuptools && \
|
|
||||||
update-alternatives --install /usr/bin/pip pip /usr/bin/pip3 1 && \
|
|
||||||
rm -rf /var/lib/apt/lists/*
|
|
||||||
|
|
||||||
RUN groupadd -g 999 $user && useradd -m -u 999 -g $user $user
|
|
||||||
|
|
||||||
COPY . $projects_dir
|
|
||||||
RUN chown -R $user:$user $projects_dir
|
|
||||||
RUN mkdir -p $db_dir
|
|
||||||
RUN chown -R $user:$user $db_dir
|
|
||||||
|
|
||||||
USER $user
|
|
||||||
WORKDIR $projects_dir
|
|
||||||
|
|
||||||
RUN python3 -m pip install -U setuptools pip
|
|
||||||
RUN make install
|
|
||||||
RUN python3 docker/set_build.py
|
|
||||||
RUN rm ~/.cache -rf
|
|
||||||
VOLUME $db_dir
|
|
||||||
ENTRYPOINT ["python3", "scripts/dht_node.py"]
|
|
||||||
|
|
|
@ -1,56 +0,0 @@
|
||||||
FROM debian:10-slim
|
|
||||||
|
|
||||||
ARG user=lbry
|
|
||||||
ARG db_dir=/database
|
|
||||||
ARG projects_dir=/home/$user
|
|
||||||
|
|
||||||
ARG DOCKER_TAG
|
|
||||||
ARG DOCKER_COMMIT=docker
|
|
||||||
ENV DOCKER_TAG=$DOCKER_TAG DOCKER_COMMIT=$DOCKER_COMMIT
|
|
||||||
|
|
||||||
RUN apt-get update && \
|
|
||||||
apt-get -y --no-install-recommends install \
|
|
||||||
wget \
|
|
||||||
tar unzip \
|
|
||||||
build-essential \
|
|
||||||
automake libtool \
|
|
||||||
pkg-config \
|
|
||||||
libleveldb-dev \
|
|
||||||
python3.7 \
|
|
||||||
python3-dev \
|
|
||||||
python3-pip \
|
|
||||||
python3-wheel \
|
|
||||||
python3-cffi \
|
|
||||||
python3-setuptools && \
|
|
||||||
update-alternatives --install /usr/bin/pip pip /usr/bin/pip3 1 && \
|
|
||||||
rm -rf /var/lib/apt/lists/*
|
|
||||||
|
|
||||||
RUN groupadd -g 999 $user && useradd -m -u 999 -g $user $user
|
|
||||||
RUN mkdir -p $db_dir
|
|
||||||
RUN chown -R $user:$user $db_dir
|
|
||||||
|
|
||||||
COPY . $projects_dir
|
|
||||||
RUN chown -R $user:$user $projects_dir
|
|
||||||
|
|
||||||
USER $user
|
|
||||||
WORKDIR $projects_dir
|
|
||||||
|
|
||||||
RUN pip install uvloop
|
|
||||||
RUN make install
|
|
||||||
RUN python3 docker/set_build.py
|
|
||||||
RUN rm ~/.cache -rf
|
|
||||||
|
|
||||||
# entry point
|
|
||||||
ARG host=0.0.0.0
|
|
||||||
ARG tcp_port=50001
|
|
||||||
ARG daemon_url=http://lbry:lbry@localhost:9245/
|
|
||||||
VOLUME $db_dir
|
|
||||||
ENV TCP_PORT=$tcp_port
|
|
||||||
ENV HOST=$host
|
|
||||||
ENV DAEMON_URL=$daemon_url
|
|
||||||
ENV DB_DIRECTORY=$db_dir
|
|
||||||
ENV MAX_SESSIONS=1000000000
|
|
||||||
ENV MAX_SEND=1000000000000000000
|
|
||||||
ENV EVENT_LOOP_POLICY=uvloop
|
|
||||||
COPY ./docker/wallet_server_entrypoint.sh /entrypoint.sh
|
|
||||||
ENTRYPOINT ["/entrypoint.sh"]
|
|
|
@ -1,45 +0,0 @@
|
||||||
FROM debian:10-slim
|
|
||||||
|
|
||||||
ARG user=lbry
|
|
||||||
ARG downloads_dir=/database
|
|
||||||
ARG projects_dir=/home/$user
|
|
||||||
|
|
||||||
ARG DOCKER_TAG
|
|
||||||
ARG DOCKER_COMMIT=docker
|
|
||||||
ENV DOCKER_TAG=$DOCKER_TAG DOCKER_COMMIT=$DOCKER_COMMIT
|
|
||||||
|
|
||||||
RUN apt-get update && \
|
|
||||||
apt-get -y --no-install-recommends install \
|
|
||||||
wget \
|
|
||||||
automake libtool \
|
|
||||||
tar unzip \
|
|
||||||
build-essential \
|
|
||||||
pkg-config \
|
|
||||||
libleveldb-dev \
|
|
||||||
python3.7 \
|
|
||||||
python3-dev \
|
|
||||||
python3-pip \
|
|
||||||
python3-wheel \
|
|
||||||
python3-setuptools && \
|
|
||||||
update-alternatives --install /usr/bin/pip pip /usr/bin/pip3 1 && \
|
|
||||||
rm -rf /var/lib/apt/lists/*
|
|
||||||
|
|
||||||
RUN groupadd -g 999 $user && useradd -m -u 999 -g $user $user
|
|
||||||
RUN mkdir -p $downloads_dir
|
|
||||||
RUN chown -R $user:$user $downloads_dir
|
|
||||||
|
|
||||||
COPY . $projects_dir
|
|
||||||
RUN chown -R $user:$user $projects_dir
|
|
||||||
|
|
||||||
USER $user
|
|
||||||
WORKDIR $projects_dir
|
|
||||||
|
|
||||||
RUN pip install uvloop
|
|
||||||
RUN make install
|
|
||||||
RUN python3 docker/set_build.py
|
|
||||||
RUN rm ~/.cache -rf
|
|
||||||
|
|
||||||
# entry point
|
|
||||||
VOLUME $downloads_dir
|
|
||||||
COPY ./docker/webconf.yaml /webconf.yaml
|
|
||||||
ENTRYPOINT ["/home/lbry/.local/bin/lbrynet", "start", "--config=/webconf.yaml"]
|
|
|
@ -1,9 +0,0 @@
|
||||||
### How to run with docker-compose
|
|
||||||
1. Edit config file and after that fix permissions with
|
|
||||||
```
|
|
||||||
sudo chown -R 999:999 webconf.yaml
|
|
||||||
```
|
|
||||||
2. Start SDK with
|
|
||||||
```
|
|
||||||
docker-compose up -d
|
|
||||||
```
|
|
|
@ -1,49 +0,0 @@
|
||||||
version: "3"
|
|
||||||
|
|
||||||
volumes:
|
|
||||||
wallet_server:
|
|
||||||
es01:
|
|
||||||
|
|
||||||
services:
|
|
||||||
wallet_server:
|
|
||||||
depends_on:
|
|
||||||
- es01
|
|
||||||
image: lbry/wallet-server:${WALLET_SERVER_TAG:-latest-release}
|
|
||||||
restart: always
|
|
||||||
network_mode: host
|
|
||||||
ports:
|
|
||||||
- "50001:50001" # rpc port
|
|
||||||
- "2112:2112" # uncomment to enable prometheus
|
|
||||||
volumes:
|
|
||||||
- "wallet_server:/database"
|
|
||||||
environment:
|
|
||||||
- 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
|
|
||||||
- PROMETHEUS_PORT=2112
|
|
||||||
- FILTERING_CHANNEL_IDS=770bd7ecba84fd2f7607fb15aedd2b172c2e153f 95e5db68a3101df19763f3a5182e4b12ba393ee8
|
|
||||||
- BLOCKING_CHANNEL_IDS=dd687b357950f6f271999971f43c785e8067c3a9 06871aa438032244202840ec59a469b303257cad b4a2528f436eca1bf3bf3e10ff3f98c57bd6c4c6
|
|
||||||
es01:
|
|
||||||
image: docker.elastic.co/elasticsearch/elasticsearch:7.11.0
|
|
||||||
container_name: es01
|
|
||||||
environment:
|
|
||||||
- node.name=es01
|
|
||||||
- discovery.type=single-node
|
|
||||||
- indices.query.bool.max_clause_count=8192
|
|
||||||
- bootstrap.memory_lock=true
|
|
||||||
- "ES_JAVA_OPTS=-Xms4g -Xmx4g" # no more than 32, remember to disable swap
|
|
||||||
ulimits:
|
|
||||||
memlock:
|
|
||||||
soft: -1
|
|
||||||
hard: -1
|
|
||||||
volumes:
|
|
||||||
- es01:/usr/share/elasticsearch/data
|
|
||||||
ports:
|
|
||||||
- 127.0.0.1:9200:9200
|
|
|
@ -1,9 +0,0 @@
|
||||||
version: '3'
|
|
||||||
services:
|
|
||||||
websdk:
|
|
||||||
image: vshyba/websdk
|
|
||||||
ports:
|
|
||||||
- '5279:5279'
|
|
||||||
- '5280:5280'
|
|
||||||
volumes:
|
|
||||||
- ./webconf.yaml:/webconf.yaml
|
|
|
@ -1,7 +0,0 @@
|
||||||
#!/bin/bash
|
|
||||||
|
|
||||||
DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" >/dev/null 2>&1 && pwd )"
|
|
||||||
cd "$DIR/../.." ## make sure we're in the right place. Docker Hub screws this up sometimes
|
|
||||||
echo "docker build dir: $(pwd)"
|
|
||||||
|
|
||||||
docker build --build-arg DOCKER_TAG=$DOCKER_TAG --build-arg DOCKER_COMMIT=$SOURCE_COMMIT -f $DOCKERFILE_PATH -t $IMAGE_NAME .
|
|
|
@ -1,11 +0,0 @@
|
||||||
# requires powershell and .NET 4+. see https://chocolatey.org/install for more info.
|
|
||||||
|
|
||||||
$chocoVersion = powershell choco -v
|
|
||||||
if(-not($chocoVersion)){
|
|
||||||
Write-Output "Chocolatey is not installed, installing now"
|
|
||||||
Write-Output "IF YOU KEEP GETTING THIS MESSAGE ON EVERY BUILD, TRY RESTARTING THE GITLAB RUNNER SO IT GETS CHOCO INTO IT'S ENV"
|
|
||||||
Set-ExecutionPolicy Bypass -Scope Process -Force; iex ((New-Object System.Net.WebClient).DownloadString('https://chocolatey.org/install.ps1'))
|
|
||||||
}
|
|
||||||
else{
|
|
||||||
Write-Output "Chocolatey version $chocoVersion is already installed"
|
|
||||||
}
|
|
|
@ -1,44 +0,0 @@
|
||||||
import sys
|
|
||||||
import os
|
|
||||||
import re
|
|
||||||
import logging
|
|
||||||
import lbry.build_info as build_info_mod
|
|
||||||
|
|
||||||
log = logging.getLogger()
|
|
||||||
log.addHandler(logging.StreamHandler())
|
|
||||||
log.setLevel(logging.DEBUG)
|
|
||||||
|
|
||||||
|
|
||||||
def _check_and_set(d: dict, key: str, value: str):
|
|
||||||
try:
|
|
||||||
d[key]
|
|
||||||
except KeyError:
|
|
||||||
raise Exception(f"{key} var does not exist in {build_info_mod.__file__}")
|
|
||||||
d[key] = value
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
build_info = {item: build_info_mod.__dict__[item] for item in dir(build_info_mod) if not item.startswith("__")}
|
|
||||||
|
|
||||||
commit_hash = os.getenv('DOCKER_COMMIT', os.getenv('GITHUB_SHA'))
|
|
||||||
if commit_hash is None:
|
|
||||||
raise ValueError("Commit hash not found in env vars")
|
|
||||||
_check_and_set(build_info, "COMMIT_HASH", commit_hash[:6])
|
|
||||||
|
|
||||||
docker_tag = os.getenv('DOCKER_TAG')
|
|
||||||
if docker_tag:
|
|
||||||
_check_and_set(build_info, "DOCKER_TAG", docker_tag)
|
|
||||||
_check_and_set(build_info, "BUILD", "docker")
|
|
||||||
else:
|
|
||||||
if re.match(r'refs/tags/v\d+\.\d+\.\d+$', str(os.getenv('GITHUB_REF'))):
|
|
||||||
_check_and_set(build_info, "BUILD", "release")
|
|
||||||
else:
|
|
||||||
_check_and_set(build_info, "BUILD", "qa")
|
|
||||||
|
|
||||||
log.debug("build info: %s", ", ".join([f"{k}={v}" for k, v in build_info.items()]))
|
|
||||||
with open(build_info_mod.__file__, 'w') as f:
|
|
||||||
f.write("\n".join([f"{k} = \"{v}\"" for k, v in build_info.items()]) + "\n")
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
|
||||||
sys.exit(main())
|
|
|
@ -1,25 +0,0 @@
|
||||||
#!/bin/bash
|
|
||||||
|
|
||||||
# entrypoint for wallet server Docker image
|
|
||||||
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
SNAPSHOT_URL="${SNAPSHOT_URL:-}" #off by default. latest snapshot at https://lbry.com/snapshot/wallet
|
|
||||||
|
|
||||||
if [[ -n "$SNAPSHOT_URL" ]] && [[ ! -f /database/lbry-leveldb ]]; then
|
|
||||||
files="$(ls)"
|
|
||||||
echo "Downloading wallet snapshot from $SNAPSHOT_URL"
|
|
||||||
wget --no-verbose --trust-server-names --content-disposition "$SNAPSHOT_URL"
|
|
||||||
echo "Extracting snapshot..."
|
|
||||||
filename="$(grep -vf <(echo "$files") <(ls))" # finds the file that was not there before
|
|
||||||
case "$filename" in
|
|
||||||
*.tgz|*.tar.gz|*.tar.bz2 ) tar xvf "$filename" --directory /database ;;
|
|
||||||
*.zip ) unzip "$filename" -d /database ;;
|
|
||||||
* ) echo "Don't know how to extract ${filename}. SNAPSHOT COULD NOT BE LOADED" && exit 1 ;;
|
|
||||||
esac
|
|
||||||
rm "$filename"
|
|
||||||
fi
|
|
||||||
|
|
||||||
/home/lbry/.local/bin/lbry-hub-elastic-sync
|
|
||||||
echo 'starting server'
|
|
||||||
/home/lbry/.local/bin/lbry-hub "$@"
|
|
|
@ -1,9 +0,0 @@
|
||||||
allowed_origin: "*"
|
|
||||||
max_key_fee: "0.0 USD"
|
|
||||||
save_files: false
|
|
||||||
save_blobs: false
|
|
||||||
streaming_server: "0.0.0.0:5280"
|
|
||||||
api: "0.0.0.0:5279"
|
|
||||||
data_dir: /tmp
|
|
||||||
download_dir: /tmp
|
|
||||||
wallet_dir: /tmp
|
|
307
docs/404.html
Normal file
307
docs/404.html
Normal file
|
@ -0,0 +1,307 @@
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en" class="no-js">
|
||||||
|
<head>
|
||||||
|
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<meta name="viewport" content="width=device-width,initial-scale=1">
|
||||||
|
<meta http-equiv="x-ua-compatible" content="ie=edge">
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<meta name="lang:clipboard.copy" content="Copy to clipboard">
|
||||||
|
|
||||||
|
<meta name="lang:clipboard.copied" content="Copied to clipboard">
|
||||||
|
|
||||||
|
<meta name="lang:search.language" content="en">
|
||||||
|
|
||||||
|
<meta name="lang:search.pipeline.stopwords" content="True">
|
||||||
|
|
||||||
|
<meta name="lang:search.pipeline.trimmer" content="True">
|
||||||
|
|
||||||
|
<meta name="lang:search.result.none" content="No matching documents">
|
||||||
|
|
||||||
|
<meta name="lang:search.result.one" content="1 matching document">
|
||||||
|
|
||||||
|
<meta name="lang:search.result.other" content="# matching documents">
|
||||||
|
|
||||||
|
<meta name="lang:search.tokenizer" content="[\s\-]+">
|
||||||
|
|
||||||
|
<link rel="shortcut icon" href="/assets/images/favicon.png">
|
||||||
|
<meta name="generator" content="mkdocs-0.17.3, mkdocs-material-2.7.0">
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<title>LBRY</title>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<link rel="stylesheet" href="/assets/stylesheets/application.78aab2dc.css">
|
||||||
|
|
||||||
|
<link rel="stylesheet" href="/assets/stylesheets/application-palette.6079476c.css">
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<script src="/assets/javascripts/modernizr.1aa3b519.js"></script>
|
||||||
|
|
||||||
|
|
||||||
|
<link href="https://fonts.gstatic.com" rel="preconnect" crossorigin>
|
||||||
|
|
||||||
|
<link rel="stylesheet" href="https://fonts.googleapis.com/css?family=Roboto:300,400,400i,700|Roboto+Mono">
|
||||||
|
<style>body,input{font-family:"Roboto","Helvetica Neue",Helvetica,Arial,sans-serif}code,kbd,pre{font-family:"Roboto Mono","Courier New",Courier,monospace}</style>
|
||||||
|
|
||||||
|
<link rel="stylesheet" href="https://fonts.googleapis.com/icon?family=Material+Icons">
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
</head>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<body dir="ltr" data-md-color-primary="teal" data-md-color-accent="green">
|
||||||
|
|
||||||
|
<svg class="md-svg">
|
||||||
|
<defs>
|
||||||
|
|
||||||
|
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="416" height="448"
|
||||||
|
viewBox="0 0 416 448" id="github">
|
||||||
|
<path fill="currentColor" d="M160 304q0 10-3.125 20.5t-10.75 19-18.125
|
||||||
|
8.5-18.125-8.5-10.75-19-3.125-20.5 3.125-20.5 10.75-19 18.125-8.5
|
||||||
|
18.125 8.5 10.75 19 3.125 20.5zM320 304q0 10-3.125 20.5t-10.75
|
||||||
|
19-18.125 8.5-18.125-8.5-10.75-19-3.125-20.5 3.125-20.5 10.75-19
|
||||||
|
18.125-8.5 18.125 8.5 10.75 19 3.125 20.5zM360
|
||||||
|
304q0-30-17.25-51t-46.75-21q-10.25 0-48.75 5.25-17.75 2.75-39.25
|
||||||
|
2.75t-39.25-2.75q-38-5.25-48.75-5.25-29.5 0-46.75 21t-17.25 51q0 22 8
|
||||||
|
38.375t20.25 25.75 30.5 15 35 7.375 37.25 1.75h42q20.5 0
|
||||||
|
37.25-1.75t35-7.375 30.5-15 20.25-25.75 8-38.375zM416 260q0 51.75-15.25
|
||||||
|
82.75-9.5 19.25-26.375 33.25t-35.25 21.5-42.5 11.875-42.875 5.5-41.75
|
||||||
|
1.125q-19.5 0-35.5-0.75t-36.875-3.125-38.125-7.5-34.25-12.875-30.25-20.25-21.5-28.75q-15.5-30.75-15.5-82.75
|
||||||
|
0-59.25 34-99-6.75-20.5-6.75-42.5 0-29 12.75-54.5 27 0 47.5 9.875t47.25
|
||||||
|
30.875q36.75-8.75 77.25-8.75 37 0 70 8 26.25-20.5
|
||||||
|
46.75-30.25t47.25-9.75q12.75 25.5 12.75 54.5 0 21.75-6.75 42 34 40 34
|
||||||
|
99.5z" />
|
||||||
|
</svg>
|
||||||
|
|
||||||
|
</defs>
|
||||||
|
</svg>
|
||||||
|
<input class="md-toggle" data-md-toggle="drawer" type="checkbox" id="drawer">
|
||||||
|
<input class="md-toggle" data-md-toggle="search" type="checkbox" id="search">
|
||||||
|
<label class="md-overlay" data-md-component="overlay" for="drawer"></label>
|
||||||
|
|
||||||
|
|
||||||
|
<header class="md-header" data-md-component="header">
|
||||||
|
<nav class="md-header-nav md-grid">
|
||||||
|
<div class="md-flex">
|
||||||
|
<div class="md-flex__cell md-flex__cell--shrink">
|
||||||
|
<a href="/" title="LBRY" class="md-header-nav__button md-logo">
|
||||||
|
|
||||||
|
<img src="https://s3.amazonaws.com/files.lbry.io/logo-square-white-bookonly.png" alt="LBRY logo" width="24" height="24">
|
||||||
|
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="md-flex__cell md-flex__cell--shrink">
|
||||||
|
<label class="md-icon md-icon--menu md-header-nav__button" for="drawer"></label>
|
||||||
|
</div>
|
||||||
|
<div class="md-flex__cell md-flex__cell--stretch">
|
||||||
|
<div class="md-flex__ellipsis md-header-nav__title" data-md-component="title">
|
||||||
|
|
||||||
|
|
||||||
|
<span class="md-header-nav__topic">
|
||||||
|
LBRY
|
||||||
|
</span>
|
||||||
|
<span class="md-header-nav__topic">
|
||||||
|
|
||||||
|
</span>
|
||||||
|
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="md-flex__cell md-flex__cell--shrink">
|
||||||
|
|
||||||
|
|
||||||
|
<label class="md-icon md-icon--search md-header-nav__button" for="search"></label>
|
||||||
|
|
||||||
|
<div class="md-search" data-md-component="search" role="dialog">
|
||||||
|
<label class="md-search__overlay" for="search"></label>
|
||||||
|
<div class="md-search__inner" role="search">
|
||||||
|
<form class="md-search__form" name="search">
|
||||||
|
<input type="text" class="md-search__input" name="query" placeholder="Search" autocapitalize="off" autocorrect="off" autocomplete="off" spellcheck="false" data-md-component="query" data-md-state="active">
|
||||||
|
<label class="md-icon md-search__icon" for="search"></label>
|
||||||
|
<button type="reset" class="md-icon md-search__icon" data-md-component="reset" tabindex="-1">
|
||||||
|

|
||||||
|
</button>
|
||||||
|
</form>
|
||||||
|
<div class="md-search__output">
|
||||||
|
<div class="md-search__scrollwrap" data-md-scrollfix>
|
||||||
|
<div class="md-search-result" data-md-component="result">
|
||||||
|
<div class="md-search-result__meta">
|
||||||
|
Type to start searching
|
||||||
|
</div>
|
||||||
|
<ol class="md-search-result__list"></ol>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="md-flex__cell md-flex__cell--shrink">
|
||||||
|
<div class="md-header-nav__source">
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<a href="https://github.com/lbryio/lbry/" title="Go to repository" class="md-source" data-md-source="github">
|
||||||
|
|
||||||
|
<div class="md-source__icon">
|
||||||
|
<svg viewBox="0 0 24 24" width="24" height="24">
|
||||||
|
<use xlink:href="#github" width="24" height="24"></use>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="md-source__repository">
|
||||||
|
GitHub
|
||||||
|
</div>
|
||||||
|
</a>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<div class="md-container">
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<main class="md-main">
|
||||||
|
<div class="md-main__inner md-grid" data-md-component="container">
|
||||||
|
|
||||||
|
|
||||||
|
<div class="md-sidebar md-sidebar--primary" data-md-component="navigation">
|
||||||
|
<div class="md-sidebar__scrollwrap">
|
||||||
|
<div class="md-sidebar__inner">
|
||||||
|
<nav class="md-nav md-nav--primary" data-md-level="0">
|
||||||
|
<label class="md-nav__title md-nav__title--site" for="drawer">
|
||||||
|
<span class="md-nav__button md-logo">
|
||||||
|
|
||||||
|
<img src="https://s3.amazonaws.com/files.lbry.io/logo-square-white-bookonly.png" alt="LBRY logo" width="48" height="48">
|
||||||
|
|
||||||
|
</span>
|
||||||
|
LBRY
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<div class="md-nav__source">
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<a href="https://github.com/lbryio/lbry/" title="Go to repository" class="md-source" data-md-source="github">
|
||||||
|
|
||||||
|
<div class="md-source__icon">
|
||||||
|
<svg viewBox="0 0 24 24" width="24" height="24">
|
||||||
|
<use xlink:href="#github" width="24" height="24"></use>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="md-source__repository">
|
||||||
|
GitHub
|
||||||
|
</div>
|
||||||
|
</a>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<ul class="md-nav__list" data-md-scrollfix>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<li class="md-nav__item">
|
||||||
|
<a href="/" title="API" class="md-nav__link">
|
||||||
|
API
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<li class="md-nav__item">
|
||||||
|
<a href="/cli/" title="CLI" class="md-nav__link">
|
||||||
|
CLI
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
|
||||||
|
|
||||||
|
</ul>
|
||||||
|
</nav>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<div class="md-content">
|
||||||
|
<article class="md-content__inner md-typeset">
|
||||||
|
|
||||||
|
<h1>404 - Not found</h1>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
</article>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</main>
|
||||||
|
|
||||||
|
|
||||||
|
<footer class="md-footer">
|
||||||
|
|
||||||
|
<div class="md-footer-meta md-typeset">
|
||||||
|
<div class="md-footer-meta__inner md-grid">
|
||||||
|
<div class="md-footer-copyright">
|
||||||
|
|
||||||
|
powered by
|
||||||
|
<a href="http://www.mkdocs.org">MkDocs</a>
|
||||||
|
and
|
||||||
|
<a href="https://squidfunk.github.io/mkdocs-material/">
|
||||||
|
Material for MkDocs</a>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</footer>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script src="/assets/javascripts/application.8eb9be28.js"></script>
|
||||||
|
|
||||||
|
<script>app.initialize({version:"0.17.3",url:{base:""}})</script>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<script>!function(e,a,t,n,o,c,i){e.GoogleAnalyticsObject=o,e.ga=e.ga||function(){(e.ga.q=e.ga.q||[]).push(arguments)},e.ga.l=1*new Date,c=a.createElement(t),i=a.getElementsByTagName(t)[0],c.async=1,c.src="https://www.google-analytics.com/analytics.js",i.parentNode.insertBefore(c,i)}(window,document,"script",0,"ga"),ga("create","UA-60403362-1","auto"),ga("set","anonymizeIp",!0),ga("send","pageview");var links=document.getElementsByTagName("a");if(Array.prototype.map.call(links,function(e){e.host!=document.location.host&&e.addEventListener("click",function(){var a=e.getAttribute("data-md-action")||"follow";ga("send","event","outbound",a,e.href)})}),document.forms.search){var query=document.forms.search.query;query.addEventListener("blur",function(){if(this.value){var e=document.location.pathname;ga("send","pageview",e+"?q="+this.value)}})}</script>
|
||||||
|
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
4602
docs/api.json
4602
docs/api.json
File diff suppressed because one or more lines are too long
BIN
docs/assets/images/favicon.png
Normal file
BIN
docs/assets/images/favicon.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 521 B |
20
docs/assets/images/icons/bitbucket.4ebea66e.svg
Normal file
20
docs/assets/images/icons/bitbucket.4ebea66e.svg
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="352" height="448"
|
||||||
|
viewBox="0 0 352 448" id="bitbucket">
|
||||||
|
<path fill="currentColor" d="M203.75 214.75q2 15.75-12.625 25.25t-27.875
|
||||||
|
1.5q-9.75-4.25-13.375-14.5t-0.125-20.5 13-14.5q9-4.5 18.125-3t16 8.875
|
||||||
|
6.875 16.875zM231.5 209.5q-3.5-26.75-28.25-41t-49.25-3.25q-15.75
|
||||||
|
7-25.125 22.125t-8.625 32.375q1 22.75 19.375 38.75t41.375 14q22.75-2
|
||||||
|
38-21t12.5-42zM291.25
|
||||||
|
74q-5-6.75-14-11.125t-14.5-5.5-17.75-3.125q-72.75-11.75-141.5 0.5-10.75
|
||||||
|
1.75-16.5 3t-13.75 5.5-12.5 10.75q7.5 7 19 11.375t18.375 5.5 21.875
|
||||||
|
2.875q57 7.25 112 0.25 15.75-2 22.375-3t18.125-5.375 18.75-11.625zM305.5
|
||||||
|
332.75q-2 6.5-3.875 19.125t-3.5 21-7.125 17.5-14.5 14.125q-21.5
|
||||||
|
12-47.375 17.875t-50.5 5.5-50.375-4.625q-11.5-2-20.375-4.5t-19.125-6.75-18.25-10.875-13-15.375q-6.25-24-14.25-73l1.5-4
|
||||||
|
4.5-2.25q55.75 37 126.625 37t126.875-37q5.25 1.5 6 5.75t-1.25 11.25-2
|
||||||
|
9.25zM350.75 92.5q-6.5 41.75-27.75 163.75-1.25 7.5-6.75 14t-10.875
|
||||||
|
10-13.625 7.75q-63 31.5-152.5
|
||||||
|
22-62-6.75-98.5-34.75-3.75-3-6.375-6.625t-4.25-8.75-2.25-8.5-1.5-9.875-1.375-8.75q-2.25-12.5-6.625-37.5t-7-40.375-5.875-36.875-5.5-39.5q0.75-6.5
|
||||||
|
4.375-12.125t7.875-9.375 11.25-7.5 11.5-5.625 12-4.625q31.25-11.5
|
||||||
|
78.25-16 94.75-9.25 169 12.5 38.75 11.5 53.75 30.5 4 5 4.125
|
||||||
|
12.75t-1.375 13.5z" />
|
||||||
|
</svg>
|
After Width: | Height: | Size: 1.4 KiB |
18
docs/assets/images/icons/github.a4034fb1.svg
Normal file
18
docs/assets/images/icons/github.a4034fb1.svg
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="416" height="448"
|
||||||
|
viewBox="0 0 416 448" id="github">
|
||||||
|
<path fill="currentColor" d="M160 304q0 10-3.125 20.5t-10.75 19-18.125
|
||||||
|
8.5-18.125-8.5-10.75-19-3.125-20.5 3.125-20.5 10.75-19 18.125-8.5
|
||||||
|
18.125 8.5 10.75 19 3.125 20.5zM320 304q0 10-3.125 20.5t-10.75
|
||||||
|
19-18.125 8.5-18.125-8.5-10.75-19-3.125-20.5 3.125-20.5 10.75-19
|
||||||
|
18.125-8.5 18.125 8.5 10.75 19 3.125 20.5zM360
|
||||||
|
304q0-30-17.25-51t-46.75-21q-10.25 0-48.75 5.25-17.75 2.75-39.25
|
||||||
|
2.75t-39.25-2.75q-38-5.25-48.75-5.25-29.5 0-46.75 21t-17.25 51q0 22 8
|
||||||
|
38.375t20.25 25.75 30.5 15 35 7.375 37.25 1.75h42q20.5 0
|
||||||
|
37.25-1.75t35-7.375 30.5-15 20.25-25.75 8-38.375zM416 260q0 51.75-15.25
|
||||||
|
82.75-9.5 19.25-26.375 33.25t-35.25 21.5-42.5 11.875-42.875 5.5-41.75
|
||||||
|
1.125q-19.5 0-35.5-0.75t-36.875-3.125-38.125-7.5-34.25-12.875-30.25-20.25-21.5-28.75q-15.5-30.75-15.5-82.75
|
||||||
|
0-59.25 34-99-6.75-20.5-6.75-42.5 0-29 12.75-54.5 27 0 47.5 9.875t47.25
|
||||||
|
30.875q36.75-8.75 77.25-8.75 37 0 70 8 26.25-20.5
|
||||||
|
46.75-30.25t47.25-9.75q12.75 25.5 12.75 54.5 0 21.75-6.75 42 34 40 34
|
||||||
|
99.5z" />
|
||||||
|
</svg>
|
After Width: | Height: | Size: 1.2 KiB |
38
docs/assets/images/icons/gitlab.348cdb3a.svg
Normal file
38
docs/assets/images/icons/gitlab.348cdb3a.svg
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="500" height="500"
|
||||||
|
viewBox="0 0 500 500" id="gitlab">
|
||||||
|
<g transform="translate(156.197863, 1.160267)">
|
||||||
|
<path fill="currentColor"
|
||||||
|
d="M93.667,473.347L93.667,473.347l90.684-279.097H2.983L93.667,
|
||||||
|
473.347L93.667,473.347z" />
|
||||||
|
</g>
|
||||||
|
<g transform="translate(28.531199, 1.160800)" opacity="0.7">
|
||||||
|
<path fill="currentColor"
|
||||||
|
d="M221.333,473.345L130.649,194.25H3.557L221.333,473.345L221.333,
|
||||||
|
473.345z" />
|
||||||
|
</g>
|
||||||
|
<g transform="translate(0.088533, 0.255867)" opacity="0.5">
|
||||||
|
<path fill="currentColor"
|
||||||
|
d="M32,195.155L32,195.155L4.441,279.97c-2.513,7.735,0.24,16.21,6.821,
|
||||||
|
20.99l238.514,173.29 L32,195.155L32,195.155z" />
|
||||||
|
</g>
|
||||||
|
<g transform="translate(29.421866, 280.255593)">
|
||||||
|
<path fill="currentColor"
|
||||||
|
d="M2.667-84.844h127.092L75.14-252.942c-2.811-8.649-15.047-8.649-17.856,
|
||||||
|
0L2.667-84.844 L2.667-84.844z" />
|
||||||
|
</g>
|
||||||
|
<g transform="translate(247.197860, 1.160800)" opacity="0.7">
|
||||||
|
<path fill="currentColor"
|
||||||
|
d="M2.667,473.345L93.351,194.25h127.092L2.667,473.345L2.667,
|
||||||
|
473.345z" />
|
||||||
|
</g>
|
||||||
|
<g transform="translate(246.307061, 0.255867)" opacity="0.5">
|
||||||
|
<path fill="currentColor"
|
||||||
|
d="M221.334,195.155L221.334,195.155l27.559,84.815c2.514,7.735-0.24,
|
||||||
|
16.21-6.821,20.99 L3.557,474.25L221.334,195.155L221.334,195.155z" />
|
||||||
|
</g>
|
||||||
|
<g transform="translate(336.973725, 280.255593)">
|
||||||
|
<path fill="currentColor"
|
||||||
|
d="M130.667-84.844H3.575l54.618-168.098c2.811-8.649,15.047-8.649,
|
||||||
|
17.856,0L130.667-84.844 L130.667-84.844z" />
|
||||||
|
</g>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 1.6 KiB |
1
docs/assets/javascripts/application.8eb9be28.js
Normal file
1
docs/assets/javascripts/application.8eb9be28.js
Normal file
File diff suppressed because one or more lines are too long
1
docs/assets/javascripts/lunr/lunr.da.js
Normal file
1
docs/assets/javascripts/lunr/lunr.da.js
Normal file
|
@ -0,0 +1 @@
|
||||||
|
!function(e,r){"function"==typeof define&&define.amd?define(r):"object"==typeof exports?module.exports=r():r()(e.lunr)}(this,function(){return function(e){if(void 0===e)throw new Error("Lunr is not present. Please include / require Lunr before this script.");if(void 0===e.stemmerSupport)throw new Error("Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.");var r,i,n;e.da=function(){this.pipeline.reset(),this.pipeline.add(e.da.trimmer,e.da.stopWordFilter,e.da.stemmer),this.searchPipeline&&(this.searchPipeline.reset(),this.searchPipeline.add(e.da.stemmer))},e.da.wordCharacters="A-Za-zªºÀ-ÖØ-öø-ʸˠ-ˤᴀ-ᴥᴬ-ᵜᵢ-ᵥᵫ-ᵷᵹ-ᶾḀ-ỿⁱⁿₐ-ₜKÅℲⅎⅠ-ↈⱠ-ⱿꜢ-ꞇꞋ-ꞭꞰ-ꞷꟷ-ꟿꬰ-ꭚꭜ-ꭤff-stA-Za-z",e.da.trimmer=e.trimmerSupport.generateTrimmer(e.da.wordCharacters),e.Pipeline.registerFunction(e.da.trimmer,"trimmer-da"),e.da.stemmer=(r=e.stemmerSupport.Among,i=e.stemmerSupport.SnowballProgram,n=new function(){var e,n,t,s=[new r("hed",-1,1),new r("ethed",0,1),new r("ered",-1,1),new r("e",-1,1),new r("erede",3,1),new r("ende",3,1),new r("erende",5,1),new r("ene",3,1),new r("erne",3,1),new r("ere",3,1),new r("en",-1,1),new r("heden",10,1),new r("eren",10,1),new r("er",-1,1),new r("heder",13,1),new r("erer",13,1),new r("s",-1,2),new r("heds",16,1),new r("es",16,1),new r("endes",18,1),new r("erendes",19,1),new r("enes",18,1),new r("ernes",18,1),new r("eres",18,1),new r("ens",16,1),new r("hedens",24,1),new r("erens",24,1),new r("ers",16,1),new r("ets",16,1),new r("erets",28,1),new r("et",-1,1),new r("eret",30,1)],o=[new r("gd",-1,-1),new r("dt",-1,-1),new r("gt",-1,-1),new r("kt",-1,-1)],a=[new r("ig",-1,1),new r("lig",0,1),new r("elig",1,1),new r("els",-1,1),new r("løst",-1,2)],d=[17,65,16,1,0,0,0,0,0,0,0,0,0,0,0,0,48,0,128],u=[239,254,42,3,0,0,0,0,0,0,0,0,0,0,0,0,16],c=new i;function l(){var e,r=c.limit-c.cursor;c.cursor>=n&&(e=c.limit_backward,c.limit_backward=n,c.ket=c.cursor,c.find_among_b(o,4)?(c.bra=c.cursor,c.limit_backward=e,c.cursor=c.limit-r,c.cursor>c.limit_backward&&(c.cursor--,c.bra=c.cursor,c.slice_del())):c.limit_backward=e)}this.setCurrent=function(e){c.setCurrent(e)},this.getCurrent=function(){return c.getCurrent()},this.stem=function(){var r,i=c.cursor;return function(){var r,i=c.cursor+3;if(n=c.limit,0<=i&&i<=c.limit){for(e=i;;){if(r=c.cursor,c.in_grouping(d,97,248)){c.cursor=r;break}if(c.cursor=r,r>=c.limit)return;c.cursor++}for(;!c.out_grouping(d,97,248);){if(c.cursor>=c.limit)return;c.cursor++}(n=c.cursor)<e&&(n=e)}}(),c.limit_backward=i,c.cursor=c.limit,function(){var e,r;if(c.cursor>=n&&(r=c.limit_backward,c.limit_backward=n,c.ket=c.cursor,e=c.find_among_b(s,32),c.limit_backward=r,e))switch(c.bra=c.cursor,e){case 1:c.slice_del();break;case 2:c.in_grouping_b(u,97,229)&&c.slice_del()}}(),c.cursor=c.limit,l(),c.cursor=c.limit,function(){var e,r,i,t=c.limit-c.cursor;if(c.ket=c.cursor,c.eq_s_b(2,"st")&&(c.bra=c.cursor,c.eq_s_b(2,"ig")&&c.slice_del()),c.cursor=c.limit-t,c.cursor>=n&&(r=c.limit_backward,c.limit_backward=n,c.ket=c.cursor,e=c.find_among_b(a,5),c.limit_backward=r,e))switch(c.bra=c.cursor,e){case 1:c.slice_del(),i=c.limit-c.cursor,l(),c.cursor=c.limit-i;break;case 2:c.slice_from("løs")}}(),c.cursor=c.limit,c.cursor>=n&&(r=c.limit_backward,c.limit_backward=n,c.ket=c.cursor,c.out_grouping_b(d,97,248)?(c.bra=c.cursor,t=c.slice_to(t),c.limit_backward=r,c.eq_v_b(t)&&c.slice_del()):c.limit_backward=r),!0}},function(e){return"function"==typeof e.update?e.update(function(e){return n.setCurrent(e),n.stem(),n.getCurrent()}):(n.setCurrent(e),n.stem(),n.getCurrent())}),e.Pipeline.registerFunction(e.da.stemmer,"stemmer-da"),e.da.stopWordFilter=e.generateStopWordFilter("ad af alle alt anden at blev blive bliver da de dem den denne der deres det dette dig din disse dog du efter eller en end er et for fra ham han hans har havde have hende hendes her hos hun hvad hvis hvor i ikke ind jeg jer jo kunne man mange med meget men mig min mine mit mod ned noget nogle nu når og også om op os over på selv sig sin sine sit skal skulle som sådan thi til ud under var vi vil ville vor være været".split(" ")),e.Pipeline.registerFunction(e.da.stopWordFilter,"stopWordFilter-da")}});
|
1
docs/assets/javascripts/lunr/lunr.de.js
Normal file
1
docs/assets/javascripts/lunr/lunr.de.js
Normal file
File diff suppressed because one or more lines are too long
1
docs/assets/javascripts/lunr/lunr.du.js
Normal file
1
docs/assets/javascripts/lunr/lunr.du.js
Normal file
File diff suppressed because one or more lines are too long
1
docs/assets/javascripts/lunr/lunr.es.js
Normal file
1
docs/assets/javascripts/lunr/lunr.es.js
Normal file
File diff suppressed because one or more lines are too long
1
docs/assets/javascripts/lunr/lunr.fi.js
Normal file
1
docs/assets/javascripts/lunr/lunr.fi.js
Normal file
File diff suppressed because one or more lines are too long
1
docs/assets/javascripts/lunr/lunr.fr.js
Normal file
1
docs/assets/javascripts/lunr/lunr.fr.js
Normal file
File diff suppressed because one or more lines are too long
1
docs/assets/javascripts/lunr/lunr.hu.js
Normal file
1
docs/assets/javascripts/lunr/lunr.hu.js
Normal file
File diff suppressed because one or more lines are too long
1
docs/assets/javascripts/lunr/lunr.it.js
Normal file
1
docs/assets/javascripts/lunr/lunr.it.js
Normal file
File diff suppressed because one or more lines are too long
1
docs/assets/javascripts/lunr/lunr.jp.js
Normal file
1
docs/assets/javascripts/lunr/lunr.jp.js
Normal file
|
@ -0,0 +1 @@
|
||||||
|
!function(e,r){"function"==typeof define&&define.amd?define(r):"object"==typeof exports?module.exports=r():r()(e.lunr)}(this,function(){return function(e){if(void 0===e)throw new Error("Lunr is not present. Please include / require Lunr before this script.");if(void 0===e.stemmerSupport)throw new Error("Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.");var r="2"==e.version[0];e.jp=function(){this.pipeline.reset(),this.pipeline.add(e.jp.stopWordFilter,e.jp.stemmer),r?this.tokenizer=e.jp.tokenizer:(e.tokenizer&&(e.tokenizer=e.jp.tokenizer),this.tokenizerFn&&(this.tokenizerFn=e.jp.tokenizer))};var t=new e.TinySegmenter;e.jp.tokenizer=function(n){if(!arguments.length||null==n||null==n)return[];if(Array.isArray(n))return n.map(function(t){return r?new e.Token(t.toLowerCase()):t.toLowerCase()});for(var i=n.toString().toLowerCase().replace(/^\s+/,""),o=i.length-1;o>=0;o--)if(/\S/.test(i.charAt(o))){i=i.substring(0,o+1);break}return t.segment(i).filter(function(e){return!!e}).map(function(t){return r?new e.Token(t):t})},e.jp.stemmer=function(e){return e},e.Pipeline.registerFunction(e.jp.stemmer,"stemmer-jp"),e.jp.wordCharacters="一二三四五六七八九十百千万億兆一-龠々〆ヵヶぁ-んァ-ヴーア-ン゙a-zA-Za-zA-Z0-90-9",e.jp.stopWordFilter=function(t){if(-1===e.jp.stopWordFilter.stopWords.indexOf(r?t.toString():t))return t},e.jp.stopWordFilter=e.generateStopWordFilter("これ それ あれ この その あの ここ そこ あそこ こちら どこ だれ なに なん 何 私 貴方 貴方方 我々 私達 あの人 あのかた 彼女 彼 です あります おります います は が の に を で え から まで より も どの と し それで しかし".split(" ")),e.Pipeline.registerFunction(e.jp.stopWordFilter,"stopWordFilter-jp")}});
|
1
docs/assets/javascripts/lunr/lunr.multi.js
Normal file
1
docs/assets/javascripts/lunr/lunr.multi.js
Normal file
|
@ -0,0 +1 @@
|
||||||
|
!function(e,i){"function"==typeof define&&define.amd?define(i):"object"==typeof exports?module.exports=i():i()(e.lunr)}(this,function(){return function(e){e.multiLanguage=function(){for(var i=Array.prototype.slice.call(arguments),t=i.join("-"),r="",n=[],s=[],p=0;p<i.length;++p)"en"==i[p]?(r+="\\w",n.unshift(e.stopWordFilter),n.push(e.stemmer),s.push(e.stemmer)):(r+=e[i[p]].wordCharacters,n.unshift(e[i[p]].stopWordFilter),n.push(e[i[p]].stemmer),s.push(e[i[p]].stemmer));var o=e.trimmerSupport.generateTrimmer(r);return e.Pipeline.registerFunction(o,"lunr-multi-trimmer-"+t),n.unshift(o),function(){this.pipeline.reset(),this.pipeline.add.apply(this.pipeline,n),this.searchPipeline&&(this.searchPipeline.reset(),this.searchPipeline.add.apply(this.searchPipeline,s))}}}});
|
1
docs/assets/javascripts/lunr/lunr.no.js
Normal file
1
docs/assets/javascripts/lunr/lunr.no.js
Normal file
|
@ -0,0 +1 @@
|
||||||
|
!function(e,r){"function"==typeof define&&define.amd?define(r):"object"==typeof exports?module.exports=r():r()(e.lunr)}(this,function(){return function(e){if(void 0===e)throw new Error("Lunr is not present. Please include / require Lunr before this script.");if(void 0===e.stemmerSupport)throw new Error("Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.");var r,n,i;e.no=function(){this.pipeline.reset(),this.pipeline.add(e.no.trimmer,e.no.stopWordFilter,e.no.stemmer),this.searchPipeline&&(this.searchPipeline.reset(),this.searchPipeline.add(e.no.stemmer))},e.no.wordCharacters="A-Za-zªºÀ-ÖØ-öø-ʸˠ-ˤᴀ-ᴥᴬ-ᵜᵢ-ᵥᵫ-ᵷᵹ-ᶾḀ-ỿⁱⁿₐ-ₜKÅℲⅎⅠ-ↈⱠ-ⱿꜢ-ꞇꞋ-ꞭꞰ-ꞷꟷ-ꟿꬰ-ꭚꭜ-ꭤff-stA-Za-z",e.no.trimmer=e.trimmerSupport.generateTrimmer(e.no.wordCharacters),e.Pipeline.registerFunction(e.no.trimmer,"trimmer-no"),e.no.stemmer=(r=e.stemmerSupport.Among,n=e.stemmerSupport.SnowballProgram,i=new function(){var e,i,t=[new r("a",-1,1),new r("e",-1,1),new r("ede",1,1),new r("ande",1,1),new r("ende",1,1),new r("ane",1,1),new r("ene",1,1),new r("hetene",6,1),new r("erte",1,3),new r("en",-1,1),new r("heten",9,1),new r("ar",-1,1),new r("er",-1,1),new r("heter",12,1),new r("s",-1,2),new r("as",14,1),new r("es",14,1),new r("edes",16,1),new r("endes",16,1),new r("enes",16,1),new r("hetenes",19,1),new r("ens",14,1),new r("hetens",21,1),new r("ers",14,1),new r("ets",14,1),new r("et",-1,1),new r("het",25,1),new r("ert",-1,3),new r("ast",-1,1)],o=[new r("dt",-1,-1),new r("vt",-1,-1)],s=[new r("leg",-1,1),new r("eleg",0,1),new r("ig",-1,1),new r("eig",2,1),new r("lig",2,1),new r("elig",4,1),new r("els",-1,1),new r("lov",-1,1),new r("elov",7,1),new r("slov",7,1),new r("hetslov",9,1)],a=[17,65,16,1,0,0,0,0,0,0,0,0,0,0,0,0,48,0,128],m=[119,125,149,1],l=new n;this.setCurrent=function(e){l.setCurrent(e)},this.getCurrent=function(){return l.getCurrent()},this.stem=function(){var r,n,u,d,c=l.cursor;return function(){var r,n=l.cursor+3;if(i=l.limit,0<=n||n<=l.limit){for(e=n;;){if(r=l.cursor,l.in_grouping(a,97,248)){l.cursor=r;break}if(r>=l.limit)return;l.cursor=r+1}for(;!l.out_grouping(a,97,248);){if(l.cursor>=l.limit)return;l.cursor++}(i=l.cursor)<e&&(i=e)}}(),l.limit_backward=c,l.cursor=l.limit,function(){var e,r,n;if(l.cursor>=i&&(r=l.limit_backward,l.limit_backward=i,l.ket=l.cursor,e=l.find_among_b(t,29),l.limit_backward=r,e))switch(l.bra=l.cursor,e){case 1:l.slice_del();break;case 2:n=l.limit-l.cursor,l.in_grouping_b(m,98,122)?l.slice_del():(l.cursor=l.limit-n,l.eq_s_b(1,"k")&&l.out_grouping_b(a,97,248)&&l.slice_del());break;case 3:l.slice_from("er")}}(),l.cursor=l.limit,n=l.limit-l.cursor,l.cursor>=i&&(r=l.limit_backward,l.limit_backward=i,l.ket=l.cursor,l.find_among_b(o,2)?(l.bra=l.cursor,l.limit_backward=r,l.cursor=l.limit-n,l.cursor>l.limit_backward&&(l.cursor--,l.bra=l.cursor,l.slice_del())):l.limit_backward=r),l.cursor=l.limit,l.cursor>=i&&(d=l.limit_backward,l.limit_backward=i,l.ket=l.cursor,(u=l.find_among_b(s,11))?(l.bra=l.cursor,l.limit_backward=d,1==u&&l.slice_del()):l.limit_backward=d),!0}},function(e){return"function"==typeof e.update?e.update(function(e){return i.setCurrent(e),i.stem(),i.getCurrent()}):(i.setCurrent(e),i.stem(),i.getCurrent())}),e.Pipeline.registerFunction(e.no.stemmer,"stemmer-no"),e.no.stopWordFilter=e.generateStopWordFilter("alle at av bare begge ble blei bli blir blitt både båe da de deg dei deim deira deires dem den denne der dere deres det dette di din disse ditt du dykk dykkar då eg ein eit eitt eller elles en enn er et ett etter for fordi fra før ha hadde han hans har hennar henne hennes her hjå ho hoe honom hoss hossen hun hva hvem hver hvilke hvilken hvis hvor hvordan hvorfor i ikke ikkje ikkje ingen ingi inkje inn inni ja jeg kan kom korleis korso kun kunne kva kvar kvarhelst kven kvi kvifor man mange me med medan meg meget mellom men mi min mine mitt mot mykje ned no noe noen noka noko nokon nokor nokre nå når og også om opp oss over på samme seg selv si si sia sidan siden sin sine sitt sjøl skal skulle slik so som som somme somt så sånn til um upp ut uten var vart varte ved vere verte vi vil ville vore vors vort vår være være vært å".split(" ")),e.Pipeline.registerFunction(e.no.stopWordFilter,"stopWordFilter-no")}});
|
1
docs/assets/javascripts/lunr/lunr.pt.js
Normal file
1
docs/assets/javascripts/lunr/lunr.pt.js
Normal file
File diff suppressed because one or more lines are too long
1
docs/assets/javascripts/lunr/lunr.ro.js
Normal file
1
docs/assets/javascripts/lunr/lunr.ro.js
Normal file
File diff suppressed because one or more lines are too long
1
docs/assets/javascripts/lunr/lunr.ru.js
Normal file
1
docs/assets/javascripts/lunr/lunr.ru.js
Normal file
File diff suppressed because one or more lines are too long
1
docs/assets/javascripts/lunr/lunr.stemmer.support.js
Normal file
1
docs/assets/javascripts/lunr/lunr.stemmer.support.js
Normal file
|
@ -0,0 +1 @@
|
||||||
|
!function(r,t){"function"==typeof define&&define.amd?define(t):"object"==typeof exports?module.exports=t():t()(r.lunr)}(this,function(){return function(r){r.stemmerSupport={Among:function(r,t,i,s){if(this.toCharArray=function(r){for(var t=r.length,i=new Array(t),s=0;s<t;s++)i[s]=r.charCodeAt(s);return i},!r&&""!=r||!t&&0!=t||!i)throw"Bad Among initialisation: s:"+r+", substring_i: "+t+", result: "+i;this.s_size=r.length,this.s=this.toCharArray(r),this.substring_i=t,this.result=i,this.method=s},SnowballProgram:function(){var r;return{bra:0,ket:0,limit:0,cursor:0,limit_backward:0,setCurrent:function(t){r=t,this.cursor=0,this.limit=t.length,this.limit_backward=0,this.bra=this.cursor,this.ket=this.limit},getCurrent:function(){var t=r;return r=null,t},in_grouping:function(t,i,s){if(this.cursor<this.limit){var e=r.charCodeAt(this.cursor);if(e<=s&&e>=i&&t[(e-=i)>>3]&1<<(7&e))return this.cursor++,!0}return!1},in_grouping_b:function(t,i,s){if(this.cursor>this.limit_backward){var e=r.charCodeAt(this.cursor-1);if(e<=s&&e>=i&&t[(e-=i)>>3]&1<<(7&e))return this.cursor--,!0}return!1},out_grouping:function(t,i,s){if(this.cursor<this.limit){var e=r.charCodeAt(this.cursor);if(e>s||e<i)return this.cursor++,!0;if(!(t[(e-=i)>>3]&1<<(7&e)))return this.cursor++,!0}return!1},out_grouping_b:function(t,i,s){if(this.cursor>this.limit_backward){var e=r.charCodeAt(this.cursor-1);if(e>s||e<i)return this.cursor--,!0;if(!(t[(e-=i)>>3]&1<<(7&e)))return this.cursor--,!0}return!1},eq_s:function(t,i){if(this.limit-this.cursor<t)return!1;for(var s=0;s<t;s++)if(r.charCodeAt(this.cursor+s)!=i.charCodeAt(s))return!1;return this.cursor+=t,!0},eq_s_b:function(t,i){if(this.cursor-this.limit_backward<t)return!1;for(var s=0;s<t;s++)if(r.charCodeAt(this.cursor-t+s)!=i.charCodeAt(s))return!1;return this.cursor-=t,!0},find_among:function(t,i){for(var s=0,e=i,n=this.cursor,u=this.limit,o=0,h=0,c=!1;;){for(var a=s+(e-s>>1),f=0,l=o<h?o:h,_=t[a],m=l;m<_.s_size;m++){if(n+l==u){f=-1;break}if(f=r.charCodeAt(n+l)-_.s[m])break;l++}if(f<0?(e=a,h=l):(s=a,o=l),e-s<=1){if(s>0||e==s||c)break;c=!0}}for(;;){if(o>=(_=t[s]).s_size){if(this.cursor=n+_.s_size,!_.method)return _.result;var b=_.method();if(this.cursor=n+_.s_size,b)return _.result}if((s=_.substring_i)<0)return 0}},find_among_b:function(t,i){for(var s=0,e=i,n=this.cursor,u=this.limit_backward,o=0,h=0,c=!1;;){for(var a=s+(e-s>>1),f=0,l=o<h?o:h,_=(m=t[a]).s_size-1-l;_>=0;_--){if(n-l==u){f=-1;break}if(f=r.charCodeAt(n-1-l)-m.s[_])break;l++}if(f<0?(e=a,h=l):(s=a,o=l),e-s<=1){if(s>0||e==s||c)break;c=!0}}for(;;){var m;if(o>=(m=t[s]).s_size){if(this.cursor=n-m.s_size,!m.method)return m.result;var b=m.method();if(this.cursor=n-m.s_size,b)return m.result}if((s=m.substring_i)<0)return 0}},replace_s:function(t,i,s){var e=s.length-(i-t),n=r.substring(0,t),u=r.substring(i);return r=n+s+u,this.limit+=e,this.cursor>=i?this.cursor+=e:this.cursor>t&&(this.cursor=t),e},slice_check:function(){if(this.bra<0||this.bra>this.ket||this.ket>this.limit||this.limit>r.length)throw"faulty slice operation"},slice_from:function(r){this.slice_check(),this.replace_s(this.bra,this.ket,r)},slice_del:function(){this.slice_from("")},insert:function(r,t,i){var s=this.replace_s(r,t,i);r<=this.bra&&(this.bra+=s),r<=this.ket&&(this.ket+=s)},slice_to:function(){return this.slice_check(),r.substring(this.bra,this.ket)},eq_v_b:function(r){return this.eq_s_b(r.length,r)}}}},r.trimmerSupport={generateTrimmer:function(r){var t=new RegExp("^[^"+r+"]+"),i=new RegExp("[^"+r+"]+$");return function(r){return"function"==typeof r.update?r.update(function(r){return r.replace(t,"").replace(i,"")}):r.replace(t,"").replace(i,"")}}}}});
|
1
docs/assets/javascripts/lunr/lunr.sv.js
Normal file
1
docs/assets/javascripts/lunr/lunr.sv.js
Normal file
|
@ -0,0 +1 @@
|
||||||
|
!function(e,r){"function"==typeof define&&define.amd?define(r):"object"==typeof exports?module.exports=r():r()(e.lunr)}(this,function(){return function(e){if(void 0===e)throw new Error("Lunr is not present. Please include / require Lunr before this script.");if(void 0===e.stemmerSupport)throw new Error("Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.");var r,n,t;e.sv=function(){this.pipeline.reset(),this.pipeline.add(e.sv.trimmer,e.sv.stopWordFilter,e.sv.stemmer),this.searchPipeline&&(this.searchPipeline.reset(),this.searchPipeline.add(e.sv.stemmer))},e.sv.wordCharacters="A-Za-zªºÀ-ÖØ-öø-ʸˠ-ˤᴀ-ᴥᴬ-ᵜᵢ-ᵥᵫ-ᵷᵹ-ᶾḀ-ỿⁱⁿₐ-ₜKÅℲⅎⅠ-ↈⱠ-ⱿꜢ-ꞇꞋ-ꞭꞰ-ꞷꟷ-ꟿꬰ-ꭚꭜ-ꭤff-stA-Za-z",e.sv.trimmer=e.trimmerSupport.generateTrimmer(e.sv.wordCharacters),e.Pipeline.registerFunction(e.sv.trimmer,"trimmer-sv"),e.sv.stemmer=(r=e.stemmerSupport.Among,n=e.stemmerSupport.SnowballProgram,t=new function(){var e,t,i=[new r("a",-1,1),new r("arna",0,1),new r("erna",0,1),new r("heterna",2,1),new r("orna",0,1),new r("ad",-1,1),new r("e",-1,1),new r("ade",6,1),new r("ande",6,1),new r("arne",6,1),new r("are",6,1),new r("aste",6,1),new r("en",-1,1),new r("anden",12,1),new r("aren",12,1),new r("heten",12,1),new r("ern",-1,1),new r("ar",-1,1),new r("er",-1,1),new r("heter",18,1),new r("or",-1,1),new r("s",-1,2),new r("as",21,1),new r("arnas",22,1),new r("ernas",22,1),new r("ornas",22,1),new r("es",21,1),new r("ades",26,1),new r("andes",26,1),new r("ens",21,1),new r("arens",29,1),new r("hetens",29,1),new r("erns",21,1),new r("at",-1,1),new r("andet",-1,1),new r("het",-1,1),new r("ast",-1,1)],s=[new r("dd",-1,-1),new r("gd",-1,-1),new r("nn",-1,-1),new r("dt",-1,-1),new r("gt",-1,-1),new r("kt",-1,-1),new r("tt",-1,-1)],a=[new r("ig",-1,1),new r("lig",0,1),new r("els",-1,1),new r("fullt",-1,3),new r("löst",-1,2)],o=[17,65,16,1,0,0,0,0,0,0,0,0,0,0,0,0,24,0,32],u=[119,127,149],m=new n;this.setCurrent=function(e){m.setCurrent(e)},this.getCurrent=function(){return m.getCurrent()},this.stem=function(){var r,n=m.cursor;return function(){var r,n=m.cursor+3;if(t=m.limit,0<=n||n<=m.limit){for(e=n;;){if(r=m.cursor,m.in_grouping(o,97,246)){m.cursor=r;break}if(m.cursor=r,m.cursor>=m.limit)return;m.cursor++}for(;!m.out_grouping(o,97,246);){if(m.cursor>=m.limit)return;m.cursor++}(t=m.cursor)<e&&(t=e)}}(),m.limit_backward=n,m.cursor=m.limit,function(){var e,r=m.limit_backward;if(m.cursor>=t&&(m.limit_backward=t,m.cursor=m.limit,m.ket=m.cursor,e=m.find_among_b(i,37),m.limit_backward=r,e))switch(m.bra=m.cursor,e){case 1:m.slice_del();break;case 2:m.in_grouping_b(u,98,121)&&m.slice_del()}}(),m.cursor=m.limit,r=m.limit_backward,m.cursor>=t&&(m.limit_backward=t,m.cursor=m.limit,m.find_among_b(s,7)&&(m.cursor=m.limit,m.ket=m.cursor,m.cursor>m.limit_backward&&(m.bra=--m.cursor,m.slice_del())),m.limit_backward=r),m.cursor=m.limit,function(){var e,r;if(m.cursor>=t){if(r=m.limit_backward,m.limit_backward=t,m.cursor=m.limit,m.ket=m.cursor,e=m.find_among_b(a,5))switch(m.bra=m.cursor,e){case 1:m.slice_del();break;case 2:m.slice_from("lös");break;case 3:m.slice_from("full")}m.limit_backward=r}}(),!0}},function(e){return"function"==typeof e.update?e.update(function(e){return t.setCurrent(e),t.stem(),t.getCurrent()}):(t.setCurrent(e),t.stem(),t.getCurrent())}),e.Pipeline.registerFunction(e.sv.stemmer,"stemmer-sv"),e.sv.stopWordFilter=e.generateStopWordFilter("alla allt att av blev bli blir blivit de dem den denna deras dess dessa det detta dig din dina ditt du där då efter ej eller en er era ert ett från för ha hade han hans har henne hennes hon honom hur här i icke ingen inom inte jag ju kan kunde man med mellan men mig min mina mitt mot mycket ni nu när någon något några och om oss på samma sedan sig sin sina sitta själv skulle som så sådan sådana sådant till under upp ut utan vad var vara varför varit varje vars vart vem vi vid vilka vilkas vilken vilket vår våra vårt än är åt över".split(" ")),e.Pipeline.registerFunction(e.sv.stopWordFilter,"stopWordFilter-sv")}});
|
1
docs/assets/javascripts/lunr/lunr.tr.js
Normal file
1
docs/assets/javascripts/lunr/lunr.tr.js
Normal file
File diff suppressed because one or more lines are too long
1
docs/assets/javascripts/lunr/tinyseg.js
Normal file
1
docs/assets/javascripts/lunr/tinyseg.js
Normal file
File diff suppressed because one or more lines are too long
1
docs/assets/javascripts/modernizr.1aa3b519.js
Normal file
1
docs/assets/javascripts/modernizr.1aa3b519.js
Normal file
File diff suppressed because one or more lines are too long
2
docs/assets/stylesheets/application-palette.6079476c.css
Normal file
2
docs/assets/stylesheets/application-palette.6079476c.css
Normal file
File diff suppressed because one or more lines are too long
2
docs/assets/stylesheets/application.78aab2dc.css
Normal file
2
docs/assets/stylesheets/application.78aab2dc.css
Normal file
File diff suppressed because one or more lines are too long
2587
docs/cli/index.html
Normal file
2587
docs/cli/index.html
Normal file
File diff suppressed because it is too large
Load diff
2313
docs/index.html
Normal file
2313
docs/index.html
Normal file
File diff suppressed because it is too large
Load diff
584
docs/search/search_index.json
Normal file
584
docs/search/search_index.json
Normal file
File diff suppressed because one or more lines are too long
|
@ -4,22 +4,20 @@
|
||||||
share_usage_data: True
|
share_usage_data: True
|
||||||
|
|
||||||
lbryum_servers:
|
lbryum_servers:
|
||||||
- lbryumx1.lbry.com:50001
|
- lbryumx1.lbry.io:50001
|
||||||
- lbryumx2.lbry.com:50001
|
- lbryumx2.lbry.io:50001
|
||||||
- lbryumx4.lbry.com:50001
|
|
||||||
|
|
||||||
blockchain_name: lbrycrd_main
|
blockchain_name: lbrycrd_main
|
||||||
|
|
||||||
data_dir: /home/lbry/.lbrynet
|
data_dir: /home/lbry/.lbrynet
|
||||||
download_directory: /home/lbry/downloads
|
download_directory: /home/lbry/downloads
|
||||||
|
|
||||||
save_blobs: true
|
delete_blobs_on_remove: True
|
||||||
save_files: false
|
|
||||||
dht_node_port: 4444
|
dht_node_port: 4444
|
||||||
peer_port: 3333
|
peer_port: 3333
|
||||||
use_upnp: true
|
use_upnp: True
|
||||||
|
|
||||||
#components_to_skip:
|
#components_to_skip:
|
||||||
|
# - peer_protocol_server
|
||||||
# - hash_announcer
|
# - hash_announcer
|
||||||
# - blob_server
|
|
||||||
# - dht
|
# - dht
|
||||||
|
|
|
@ -1,2 +0,0 @@
|
||||||
__version__ = "0.113.0"
|
|
||||||
version = tuple(map(int, __version__.split('.'))) # pylint: disable=invalid-name
|
|
|
@ -1,6 +0,0 @@
|
||||||
from lbry.utils import get_lbry_hash_obj
|
|
||||||
|
|
||||||
MAX_BLOB_SIZE = 2 * 2 ** 20
|
|
||||||
|
|
||||||
# digest_size is in bytes, and blob hashes are hex encoded
|
|
||||||
BLOBHASH_LENGTH = get_lbry_hash_obj().digest_size * 2
|
|
|
@ -1,366 +0,0 @@
|
||||||
import os
|
|
||||||
import re
|
|
||||||
import time
|
|
||||||
import asyncio
|
|
||||||
import binascii
|
|
||||||
import logging
|
|
||||||
import typing
|
|
||||||
import contextlib
|
|
||||||
from io import BytesIO
|
|
||||||
from cryptography.hazmat.primitives.ciphers import Cipher, modes
|
|
||||||
from cryptography.hazmat.primitives.ciphers.algorithms import AES
|
|
||||||
from cryptography.hazmat.primitives.padding import PKCS7
|
|
||||||
from cryptography.hazmat.backends import default_backend
|
|
||||||
|
|
||||||
from lbry.utils import get_lbry_hash_obj
|
|
||||||
from lbry.error import DownloadCancelledError, InvalidBlobHashError, InvalidDataError
|
|
||||||
|
|
||||||
from lbry.blob import MAX_BLOB_SIZE, BLOBHASH_LENGTH
|
|
||||||
from lbry.blob.blob_info import BlobInfo
|
|
||||||
from lbry.blob.writer import HashBlobWriter
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
HEXMATCH = re.compile("^[a-f,0-9]+$")
|
|
||||||
BACKEND = default_backend()
|
|
||||||
|
|
||||||
|
|
||||||
def is_valid_blobhash(blobhash: str) -> bool:
|
|
||||||
"""Checks whether the blobhash is the correct length and contains only
|
|
||||||
valid characters (0-9, a-f)
|
|
||||||
|
|
||||||
@param blobhash: string, the blobhash to check
|
|
||||||
|
|
||||||
@return: True/False
|
|
||||||
"""
|
|
||||||
return len(blobhash) == BLOBHASH_LENGTH and HEXMATCH.match(blobhash)
|
|
||||||
|
|
||||||
|
|
||||||
def encrypt_blob_bytes(key: bytes, iv: bytes, unencrypted: bytes) -> typing.Tuple[bytes, str]:
|
|
||||||
cipher = Cipher(AES(key), modes.CBC(iv), backend=BACKEND)
|
|
||||||
padder = PKCS7(AES.block_size).padder()
|
|
||||||
encryptor = cipher.encryptor()
|
|
||||||
encrypted = encryptor.update(padder.update(unencrypted) + padder.finalize()) + encryptor.finalize()
|
|
||||||
digest = get_lbry_hash_obj()
|
|
||||||
digest.update(encrypted)
|
|
||||||
return encrypted, digest.hexdigest()
|
|
||||||
|
|
||||||
|
|
||||||
def decrypt_blob_bytes(data: bytes, length: int, key: bytes, iv: bytes) -> bytes:
|
|
||||||
if len(data) != length:
|
|
||||||
raise ValueError("unexpected length")
|
|
||||||
cipher = Cipher(AES(key), modes.CBC(iv), backend=BACKEND)
|
|
||||||
unpadder = PKCS7(AES.block_size).unpadder()
|
|
||||||
decryptor = cipher.decryptor()
|
|
||||||
return unpadder.update(decryptor.update(data) + decryptor.finalize()) + unpadder.finalize()
|
|
||||||
|
|
||||||
|
|
||||||
class AbstractBlob:
|
|
||||||
"""
|
|
||||||
A chunk of data (up to 2MB) available on the network which is specified by a sha384 hash
|
|
||||||
|
|
||||||
This class is non-io specific
|
|
||||||
"""
|
|
||||||
__slots__ = [
|
|
||||||
'loop',
|
|
||||||
'blob_hash',
|
|
||||||
'length',
|
|
||||||
'blob_completed_callback',
|
|
||||||
'blob_directory',
|
|
||||||
'writers',
|
|
||||||
'verified',
|
|
||||||
'writing',
|
|
||||||
'readers',
|
|
||||||
'added_on',
|
|
||||||
'is_mine',
|
|
||||||
]
|
|
||||||
|
|
||||||
def __init__(
|
|
||||||
self, loop: asyncio.AbstractEventLoop, blob_hash: str, length: typing.Optional[int] = None,
|
|
||||||
blob_completed_callback: typing.Optional[typing.Callable[['AbstractBlob'], asyncio.Task]] = None,
|
|
||||||
blob_directory: typing.Optional[str] = None, added_on: typing.Optional[int] = None, is_mine: bool = False,
|
|
||||||
):
|
|
||||||
self.loop = loop
|
|
||||||
self.blob_hash = blob_hash
|
|
||||||
self.length = length
|
|
||||||
self.blob_completed_callback = blob_completed_callback
|
|
||||||
self.blob_directory = blob_directory
|
|
||||||
self.writers: typing.Dict[typing.Tuple[typing.Optional[str], typing.Optional[int]], HashBlobWriter] = {}
|
|
||||||
self.verified: asyncio.Event = asyncio.Event()
|
|
||||||
self.writing: asyncio.Event = asyncio.Event()
|
|
||||||
self.readers: typing.List[typing.BinaryIO] = []
|
|
||||||
self.added_on = added_on or time.time()
|
|
||||||
self.is_mine = is_mine
|
|
||||||
|
|
||||||
if not is_valid_blobhash(blob_hash):
|
|
||||||
raise InvalidBlobHashError(blob_hash)
|
|
||||||
|
|
||||||
def __del__(self):
|
|
||||||
if self.writers or self.readers:
|
|
||||||
log.warning("%s not closed before being garbage collected", self.blob_hash)
|
|
||||||
self.close()
|
|
||||||
|
|
||||||
@contextlib.contextmanager
|
|
||||||
def _reader_context(self) -> typing.ContextManager[typing.BinaryIO]:
|
|
||||||
raise NotImplementedError()
|
|
||||||
|
|
||||||
@contextlib.contextmanager
|
|
||||||
def reader_context(self) -> typing.ContextManager[typing.BinaryIO]:
|
|
||||||
if not self.is_readable():
|
|
||||||
raise OSError(f"{str(type(self))} not readable, {len(self.readers)} readers {len(self.writers)} writers")
|
|
||||||
with self._reader_context() as reader:
|
|
||||||
try:
|
|
||||||
self.readers.append(reader)
|
|
||||||
yield reader
|
|
||||||
finally:
|
|
||||||
if reader in self.readers:
|
|
||||||
self.readers.remove(reader)
|
|
||||||
|
|
||||||
def _write_blob(self, blob_bytes: bytes) -> asyncio.Task:
|
|
||||||
raise NotImplementedError()
|
|
||||||
|
|
||||||
def set_length(self, length) -> None:
|
|
||||||
if self.length is not None and length == self.length:
|
|
||||||
return
|
|
||||||
if self.length is None and 0 <= length <= MAX_BLOB_SIZE:
|
|
||||||
self.length = length
|
|
||||||
return
|
|
||||||
log.warning("Got an invalid length. Previous length: %s, Invalid length: %s", self.length, length)
|
|
||||||
|
|
||||||
def get_length(self) -> typing.Optional[int]:
|
|
||||||
return self.length
|
|
||||||
|
|
||||||
def get_is_verified(self) -> bool:
|
|
||||||
return self.verified.is_set()
|
|
||||||
|
|
||||||
def is_readable(self) -> bool:
|
|
||||||
return self.verified.is_set()
|
|
||||||
|
|
||||||
def is_writeable(self) -> bool:
|
|
||||||
return not self.writing.is_set()
|
|
||||||
|
|
||||||
def write_blob(self, blob_bytes: bytes):
|
|
||||||
if not self.is_writeable():
|
|
||||||
raise OSError("cannot open blob for writing")
|
|
||||||
try:
|
|
||||||
self.writing.set()
|
|
||||||
self._write_blob(blob_bytes)
|
|
||||||
finally:
|
|
||||||
self.writing.clear()
|
|
||||||
|
|
||||||
def close(self):
|
|
||||||
while self.writers:
|
|
||||||
_, writer = self.writers.popitem()
|
|
||||||
if writer and writer.finished and not writer.finished.done() and not self.loop.is_closed():
|
|
||||||
writer.finished.cancel()
|
|
||||||
while self.readers:
|
|
||||||
reader = self.readers.pop()
|
|
||||||
if reader:
|
|
||||||
reader.close()
|
|
||||||
|
|
||||||
def delete(self):
|
|
||||||
self.close()
|
|
||||||
self.verified.clear()
|
|
||||||
self.length = None
|
|
||||||
|
|
||||||
async def sendfile(self, writer: asyncio.StreamWriter) -> int:
|
|
||||||
"""
|
|
||||||
Read and send the file to the writer and return the number of bytes sent
|
|
||||||
"""
|
|
||||||
|
|
||||||
if not self.is_readable():
|
|
||||||
raise OSError('blob files cannot be read')
|
|
||||||
with self.reader_context() as handle:
|
|
||||||
try:
|
|
||||||
return await self.loop.sendfile(writer.transport, handle, count=self.get_length())
|
|
||||||
except (ConnectionError, BrokenPipeError, RuntimeError, OSError, AttributeError):
|
|
||||||
return -1
|
|
||||||
|
|
||||||
def decrypt(self, key: bytes, iv: bytes) -> bytes:
|
|
||||||
"""
|
|
||||||
Decrypt a BlobFile to plaintext bytes
|
|
||||||
"""
|
|
||||||
|
|
||||||
with self.reader_context() as reader:
|
|
||||||
return decrypt_blob_bytes(reader.read(), self.length, key, iv)
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
async def create_from_unencrypted(
|
|
||||||
cls, loop: asyncio.AbstractEventLoop, blob_dir: typing.Optional[str], key: bytes, iv: bytes,
|
|
||||||
unencrypted: bytes, blob_num: int, added_on: int, is_mine: bool,
|
|
||||||
blob_completed_callback: typing.Optional[typing.Callable[['AbstractBlob'], None]] = None,
|
|
||||||
) -> BlobInfo:
|
|
||||||
"""
|
|
||||||
Create an encrypted BlobFile from plaintext bytes
|
|
||||||
"""
|
|
||||||
|
|
||||||
blob_bytes, blob_hash = encrypt_blob_bytes(key, iv, unencrypted)
|
|
||||||
length = len(blob_bytes)
|
|
||||||
blob = cls(loop, blob_hash, length, blob_completed_callback, blob_dir, added_on, is_mine)
|
|
||||||
writer = blob.get_blob_writer()
|
|
||||||
writer.write(blob_bytes)
|
|
||||||
await blob.verified.wait()
|
|
||||||
return BlobInfo(blob_num, length, binascii.hexlify(iv).decode(), added_on, blob_hash, is_mine)
|
|
||||||
|
|
||||||
def save_verified_blob(self, verified_bytes: bytes):
|
|
||||||
if self.verified.is_set():
|
|
||||||
return
|
|
||||||
|
|
||||||
def update_events(_):
|
|
||||||
self.verified.set()
|
|
||||||
self.writing.clear()
|
|
||||||
|
|
||||||
if self.is_writeable():
|
|
||||||
self.writing.set()
|
|
||||||
task = self._write_blob(verified_bytes)
|
|
||||||
task.add_done_callback(update_events)
|
|
||||||
if self.blob_completed_callback:
|
|
||||||
task.add_done_callback(lambda _: self.blob_completed_callback(self))
|
|
||||||
|
|
||||||
def get_blob_writer(self, peer_address: typing.Optional[str] = None,
|
|
||||||
peer_port: typing.Optional[int] = None) -> HashBlobWriter:
|
|
||||||
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}")
|
|
||||||
fut = asyncio.Future()
|
|
||||||
writer = HashBlobWriter(self.blob_hash, self.get_length, fut)
|
|
||||||
self.writers[(peer_address, peer_port)] = writer
|
|
||||||
|
|
||||||
def remove_writer(_):
|
|
||||||
if (peer_address, peer_port) in self.writers:
|
|
||||||
del self.writers[(peer_address, peer_port)]
|
|
||||||
|
|
||||||
fut.add_done_callback(remove_writer)
|
|
||||||
|
|
||||||
def writer_finished_callback(finished: asyncio.Future):
|
|
||||||
try:
|
|
||||||
err = finished.exception()
|
|
||||||
if err:
|
|
||||||
raise err
|
|
||||||
verified_bytes = finished.result()
|
|
||||||
while self.writers:
|
|
||||||
_, other = self.writers.popitem()
|
|
||||||
if other is not writer:
|
|
||||||
other.close_handle()
|
|
||||||
self.save_verified_blob(verified_bytes)
|
|
||||||
except (InvalidBlobHashError, InvalidDataError) as error:
|
|
||||||
log.warning("writer error downloading %s: %s", self.blob_hash[:8], str(error))
|
|
||||||
except (DownloadCancelledError, asyncio.CancelledError, asyncio.TimeoutError):
|
|
||||||
pass
|
|
||||||
|
|
||||||
fut.add_done_callback(writer_finished_callback)
|
|
||||||
return writer
|
|
||||||
|
|
||||||
|
|
||||||
class BlobBuffer(AbstractBlob):
|
|
||||||
"""
|
|
||||||
An in-memory only blob
|
|
||||||
"""
|
|
||||||
def __init__(
|
|
||||||
self, loop: asyncio.AbstractEventLoop, blob_hash: str, length: typing.Optional[int] = None,
|
|
||||||
blob_completed_callback: typing.Optional[typing.Callable[['AbstractBlob'], asyncio.Task]] = None,
|
|
||||||
blob_directory: typing.Optional[str] = None, added_on: typing.Optional[int] = None, is_mine: bool = False
|
|
||||||
):
|
|
||||||
self._verified_bytes: typing.Optional[BytesIO] = None
|
|
||||||
super().__init__(loop, blob_hash, length, blob_completed_callback, blob_directory, added_on, is_mine)
|
|
||||||
|
|
||||||
@contextlib.contextmanager
|
|
||||||
def _reader_context(self) -> typing.ContextManager[typing.BinaryIO]:
|
|
||||||
if not self.is_readable():
|
|
||||||
raise OSError("cannot open blob for reading")
|
|
||||||
try:
|
|
||||||
yield self._verified_bytes
|
|
||||||
finally:
|
|
||||||
if self._verified_bytes:
|
|
||||||
self._verified_bytes.close()
|
|
||||||
self._verified_bytes = None
|
|
||||||
self.verified.clear()
|
|
||||||
|
|
||||||
def _write_blob(self, blob_bytes: bytes):
|
|
||||||
async def write():
|
|
||||||
if self._verified_bytes:
|
|
||||||
raise OSError("already have bytes for blob")
|
|
||||||
self._verified_bytes = BytesIO(blob_bytes)
|
|
||||||
return self.loop.create_task(write())
|
|
||||||
|
|
||||||
def delete(self):
|
|
||||||
if self._verified_bytes:
|
|
||||||
self._verified_bytes.close()
|
|
||||||
self._verified_bytes = None
|
|
||||||
return super().delete()
|
|
||||||
|
|
||||||
def __del__(self):
|
|
||||||
super().__del__()
|
|
||||||
if self._verified_bytes:
|
|
||||||
self.delete()
|
|
||||||
|
|
||||||
|
|
||||||
class BlobFile(AbstractBlob):
|
|
||||||
"""
|
|
||||||
A blob existing on the local file system
|
|
||||||
"""
|
|
||||||
def __init__(
|
|
||||||
self, loop: asyncio.AbstractEventLoop, blob_hash: str, length: typing.Optional[int] = None,
|
|
||||||
blob_completed_callback: typing.Optional[typing.Callable[['AbstractBlob'], asyncio.Task]] = None,
|
|
||||||
blob_directory: typing.Optional[str] = None, added_on: typing.Optional[int] = None, is_mine: bool = False
|
|
||||||
):
|
|
||||||
super().__init__(loop, blob_hash, length, blob_completed_callback, blob_directory, added_on, is_mine)
|
|
||||||
if not blob_directory or not os.path.isdir(blob_directory):
|
|
||||||
raise OSError(f"invalid blob directory '{blob_directory}'")
|
|
||||||
self.file_path = os.path.join(self.blob_directory, self.blob_hash)
|
|
||||||
if self.file_exists:
|
|
||||||
file_size = int(os.stat(self.file_path).st_size)
|
|
||||||
if length and length != file_size:
|
|
||||||
log.warning("expected %s to be %s bytes, file has %s", self.blob_hash, length, file_size)
|
|
||||||
self.delete()
|
|
||||||
else:
|
|
||||||
self.length = file_size
|
|
||||||
self.verified.set()
|
|
||||||
|
|
||||||
@property
|
|
||||||
def file_exists(self):
|
|
||||||
return os.path.isfile(self.file_path)
|
|
||||||
|
|
||||||
def is_writeable(self) -> bool:
|
|
||||||
return super().is_writeable() and not os.path.isfile(self.file_path)
|
|
||||||
|
|
||||||
def get_blob_writer(self, peer_address: typing.Optional[str] = None,
|
|
||||||
peer_port: typing.Optional[str] = None) -> HashBlobWriter:
|
|
||||||
if self.file_exists:
|
|
||||||
raise OSError(f"File already exists '{self.file_path}'")
|
|
||||||
return super().get_blob_writer(peer_address, peer_port)
|
|
||||||
|
|
||||||
@contextlib.contextmanager
|
|
||||||
def _reader_context(self) -> typing.ContextManager[typing.BinaryIO]:
|
|
||||||
handle = open(self.file_path, 'rb')
|
|
||||||
try:
|
|
||||||
yield handle
|
|
||||||
finally:
|
|
||||||
handle.close()
|
|
||||||
|
|
||||||
def _write_blob(self, blob_bytes: bytes):
|
|
||||||
def _write_blob():
|
|
||||||
with open(self.file_path, 'wb') as f:
|
|
||||||
f.write(blob_bytes)
|
|
||||||
|
|
||||||
async def write_blob():
|
|
||||||
await self.loop.run_in_executor(None, _write_blob)
|
|
||||||
|
|
||||||
return self.loop.create_task(write_blob())
|
|
||||||
|
|
||||||
def delete(self):
|
|
||||||
super().delete()
|
|
||||||
if os.path.isfile(self.file_path):
|
|
||||||
os.remove(self.file_path)
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
async def create_from_unencrypted(
|
|
||||||
cls, loop: asyncio.AbstractEventLoop, blob_dir: typing.Optional[str], key: bytes, iv: bytes,
|
|
||||||
unencrypted: bytes, blob_num: int, added_on: float, is_mine: bool,
|
|
||||||
blob_completed_callback: typing.Optional[typing.Callable[['AbstractBlob'], asyncio.Task]] = None
|
|
||||||
) -> BlobInfo:
|
|
||||||
if not blob_dir or not os.path.isdir(blob_dir):
|
|
||||||
raise OSError(f"cannot create blob in directory: '{blob_dir}'")
|
|
||||||
return await super().create_from_unencrypted(
|
|
||||||
loop, blob_dir, key, iv, unencrypted, blob_num, added_on, is_mine, blob_completed_callback
|
|
||||||
)
|
|
|
@ -1,148 +0,0 @@
|
||||||
import os
|
|
||||||
import typing
|
|
||||||
import asyncio
|
|
||||||
import logging
|
|
||||||
from lbry.utils import LRUCacheWithMetrics
|
|
||||||
from lbry.blob.blob_file import is_valid_blobhash, BlobFile, BlobBuffer, AbstractBlob
|
|
||||||
from lbry.stream.descriptor import StreamDescriptor
|
|
||||||
from lbry.connection_manager import ConnectionManager
|
|
||||||
|
|
||||||
if typing.TYPE_CHECKING:
|
|
||||||
from lbry.conf import Config
|
|
||||||
from lbry.dht.protocol.data_store import DictDataStore
|
|
||||||
from lbry.extras.daemon.storage import SQLiteStorage
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class BlobManager:
|
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop, blob_dir: str, storage: 'SQLiteStorage', config: 'Config',
|
|
||||||
node_data_store: typing.Optional['DictDataStore'] = None):
|
|
||||||
"""
|
|
||||||
This class stores blobs on the hard disk
|
|
||||||
|
|
||||||
blob_dir - directory where blobs are stored
|
|
||||||
storage - SQLiteStorage object
|
|
||||||
"""
|
|
||||||
self.loop = loop
|
|
||||||
self.blob_dir = blob_dir
|
|
||||||
self.storage = storage
|
|
||||||
self._node_data_store = node_data_store
|
|
||||||
self.completed_blob_hashes: typing.Set[str] = set() if not self._node_data_store\
|
|
||||||
else self._node_data_store.completed_blobs
|
|
||||||
self.blobs: typing.Dict[str, AbstractBlob] = {}
|
|
||||||
self.config = config
|
|
||||||
self.decrypted_blob_lru_cache = None if not self.config.blob_lru_cache_size else LRUCacheWithMetrics(
|
|
||||||
self.config.blob_lru_cache_size)
|
|
||||||
self.connection_manager = ConnectionManager(loop)
|
|
||||||
|
|
||||||
def _get_blob(self, blob_hash: str, length: typing.Optional[int] = None, is_mine: bool = False):
|
|
||||||
if self.config.save_blobs or (
|
|
||||||
is_valid_blobhash(blob_hash) and os.path.isfile(os.path.join(self.blob_dir, blob_hash))):
|
|
||||||
return BlobFile(
|
|
||||||
self.loop, blob_hash, length, self.blob_completed, self.blob_dir, is_mine=is_mine
|
|
||||||
)
|
|
||||||
return BlobBuffer(
|
|
||||||
self.loop, blob_hash, length, self.blob_completed, self.blob_dir, is_mine=is_mine
|
|
||||||
)
|
|
||||||
|
|
||||||
def get_blob(self, blob_hash, length: typing.Optional[int] = None, is_mine: bool = False):
|
|
||||||
if blob_hash in self.blobs:
|
|
||||||
if self.config.save_blobs and isinstance(self.blobs[blob_hash], BlobBuffer):
|
|
||||||
buffer = self.blobs.pop(blob_hash)
|
|
||||||
if blob_hash in self.completed_blob_hashes:
|
|
||||||
self.completed_blob_hashes.remove(blob_hash)
|
|
||||||
self.blobs[blob_hash] = self._get_blob(blob_hash, length, is_mine)
|
|
||||||
if buffer.is_readable():
|
|
||||||
with buffer.reader_context() as reader:
|
|
||||||
self.blobs[blob_hash].write_blob(reader.read())
|
|
||||||
if length and self.blobs[blob_hash].length is None:
|
|
||||||
self.blobs[blob_hash].set_length(length)
|
|
||||||
else:
|
|
||||||
self.blobs[blob_hash] = self._get_blob(blob_hash, length, is_mine)
|
|
||||||
return self.blobs[blob_hash]
|
|
||||||
|
|
||||||
def is_blob_verified(self, blob_hash: str, length: typing.Optional[int] = None) -> bool:
|
|
||||||
if not is_valid_blobhash(blob_hash):
|
|
||||||
raise ValueError(blob_hash)
|
|
||||||
if not os.path.isfile(os.path.join(self.blob_dir, blob_hash)):
|
|
||||||
return False
|
|
||||||
if blob_hash in self.blobs:
|
|
||||||
return self.blobs[blob_hash].get_is_verified()
|
|
||||||
return self._get_blob(blob_hash, length).get_is_verified()
|
|
||||||
|
|
||||||
async def setup(self) -> bool:
|
|
||||||
def get_files_in_blob_dir() -> typing.Set[str]:
|
|
||||||
if not self.blob_dir:
|
|
||||||
return set()
|
|
||||||
return {
|
|
||||||
item.name for item in os.scandir(self.blob_dir) if is_valid_blobhash(item.name)
|
|
||||||
}
|
|
||||||
|
|
||||||
in_blobfiles_dir = await self.loop.run_in_executor(None, get_files_in_blob_dir)
|
|
||||||
to_add = await self.storage.sync_missing_blobs(in_blobfiles_dir)
|
|
||||||
if 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:
|
|
||||||
self.connection_manager.start()
|
|
||||||
return True
|
|
||||||
|
|
||||||
def stop(self):
|
|
||||||
self.connection_manager.stop()
|
|
||||||
while self.blobs:
|
|
||||||
_, blob = self.blobs.popitem()
|
|
||||||
blob.close()
|
|
||||||
self.completed_blob_hashes.clear()
|
|
||||||
|
|
||||||
def get_stream_descriptor(self, sd_hash):
|
|
||||||
return StreamDescriptor.from_stream_descriptor_blob(self.loop, self.blob_dir, self.get_blob(sd_hash))
|
|
||||||
|
|
||||||
def blob_completed(self, blob: AbstractBlob) -> asyncio.Task:
|
|
||||||
if blob.blob_hash is None:
|
|
||||||
raise Exception("Blob hash is None")
|
|
||||||
if not blob.length:
|
|
||||||
raise Exception("Blob has a length of 0")
|
|
||||||
if isinstance(blob, BlobFile):
|
|
||||||
if blob.blob_hash not in self.completed_blob_hashes:
|
|
||||||
self.completed_blob_hashes.add(blob.blob_hash)
|
|
||||||
return self.loop.create_task(self.storage.add_blobs(
|
|
||||||
(blob.blob_hash, blob.length, blob.added_on, blob.is_mine), finished=True)
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
return self.loop.create_task(self.storage.add_blobs(
|
|
||||||
(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]):
|
|
||||||
"""Ensures that completed blobs from a given list of blob hashes are set as 'finished' in the database."""
|
|
||||||
to_add = []
|
|
||||||
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):
|
|
||||||
if not is_valid_blobhash(blob_hash):
|
|
||||||
raise Exception("invalid blob hash to delete")
|
|
||||||
|
|
||||||
if blob_hash not in self.blobs:
|
|
||||||
if self.blob_dir and os.path.isfile(os.path.join(self.blob_dir, blob_hash)):
|
|
||||||
os.remove(os.path.join(self.blob_dir, blob_hash))
|
|
||||||
else:
|
|
||||||
self.blobs.pop(blob_hash).delete()
|
|
||||||
if blob_hash in self.completed_blob_hashes:
|
|
||||||
self.completed_blob_hashes.remove(blob_hash)
|
|
||||||
|
|
||||||
async def delete_blobs(self, blob_hashes: typing.List[str], delete_from_db: typing.Optional[bool] = True):
|
|
||||||
for blob_hash in blob_hashes:
|
|
||||||
self.delete_blob(blob_hash)
|
|
||||||
|
|
||||||
if delete_from_db:
|
|
||||||
await self.storage.delete_blobs_from_db(blob_hashes)
|
|
|
@ -1,77 +0,0 @@
|
||||||
import asyncio
|
|
||||||
import logging
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class DiskSpaceManager:
|
|
||||||
|
|
||||||
def __init__(self, config, db, blob_manager, cleaning_interval=30 * 60, analytics=None):
|
|
||||||
self.config = config
|
|
||||||
self.db = db
|
|
||||||
self.blob_manager = blob_manager
|
|
||||||
self.cleaning_interval = cleaning_interval
|
|
||||||
self.running = False
|
|
||||||
self.task = None
|
|
||||||
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):
|
|
||||||
self._used_space_bytes = await self.db.get_stored_blob_disk_usage()
|
|
||||||
return self._used_space_bytes
|
|
||||||
|
|
||||||
async def get_space_used_mb(self, cached=True):
|
|
||||||
cached = cached and self._used_space_bytes is not None
|
|
||||||
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):
|
|
||||||
await self._clean(False)
|
|
||||||
await self._clean(True)
|
|
||||||
|
|
||||||
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:
|
|
||||||
asyncio.create_task(
|
|
||||||
self.analytics.send_disk_space_used(space_used_mb, storage_limit_mb, is_network_blob)
|
|
||||||
)
|
|
||||||
delete = []
|
|
||||||
available = storage_limit_mb - space_used_mb
|
|
||||||
if storage_limit_mb == 0 if not is_network_blob else available >= 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.append(blob_hash)
|
|
||||||
available += int(file_size/1024.0/1024.0)
|
|
||||||
if available >= 0:
|
|
||||||
break
|
|
||||||
if delete:
|
|
||||||
await self.db.stop_all_files()
|
|
||||||
await self.blob_manager.delete_blobs(delete, delete_from_db=True)
|
|
||||||
self._used_space_bytes = None
|
|
||||||
return len(delete)
|
|
||||||
|
|
||||||
async def cleaning_loop(self):
|
|
||||||
while self.running:
|
|
||||||
await asyncio.sleep(self.cleaning_interval)
|
|
||||||
await self.clean()
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
self.running = True
|
|
||||||
self.task = asyncio.create_task(self.cleaning_loop())
|
|
||||||
self.task.add_done_callback(lambda _: log.info("Stopping blob cleanup service."))
|
|
||||||
|
|
||||||
async def stop(self):
|
|
||||||
if self.running:
|
|
||||||
self.running = False
|
|
||||||
self.task.cancel()
|
|
|
@ -1,141 +0,0 @@
|
||||||
import asyncio
|
|
||||||
import typing
|
|
||||||
import logging
|
|
||||||
from lbry.utils import cache_concurrent
|
|
||||||
from lbry.blob_exchange.client import request_blob
|
|
||||||
from lbry.dht.node import get_kademlia_peers_from_hosts
|
|
||||||
if typing.TYPE_CHECKING:
|
|
||||||
from lbry.conf import Config
|
|
||||||
from lbry.dht.node import Node
|
|
||||||
from lbry.dht.peer import KademliaPeer
|
|
||||||
from lbry.blob.blob_manager import BlobManager
|
|
||||||
from lbry.blob.blob_file import AbstractBlob
|
|
||||||
from lbry.blob_exchange.client import BlobExchangeClientProtocol
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class BlobDownloader:
|
|
||||||
BAN_FACTOR = 2.0 # fixme: when connection manager gets implemented, move it out from here
|
|
||||||
|
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop, config: 'Config', blob_manager: 'BlobManager',
|
|
||||||
peer_queue: asyncio.Queue):
|
|
||||||
self.loop = loop
|
|
||||||
self.config = config
|
|
||||||
self.blob_manager = blob_manager
|
|
||||||
self.peer_queue = peer_queue
|
|
||||||
self.active_connections: typing.Dict['KademliaPeer', asyncio.Task] = {} # active request_blob calls
|
|
||||||
self.ignored: typing.Dict['KademliaPeer', int] = {}
|
|
||||||
self.scores: typing.Dict['KademliaPeer', int] = {}
|
|
||||||
self.failures: typing.Dict['KademliaPeer', int] = {}
|
|
||||||
self.connection_failures: typing.Set['KademliaPeer'] = set()
|
|
||||||
self.connections: typing.Dict['KademliaPeer', 'BlobExchangeClientProtocol'] = {}
|
|
||||||
self.is_running = asyncio.Event()
|
|
||||||
|
|
||||||
def should_race_continue(self, blob: 'AbstractBlob'):
|
|
||||||
max_probes = self.config.max_connections_per_download * (1 if self.connections else 10)
|
|
||||||
if len(self.active_connections) >= max_probes:
|
|
||||||
return False
|
|
||||||
return not (blob.get_is_verified() or not blob.is_writeable())
|
|
||||||
|
|
||||||
async def request_blob_from_peer(self, blob: 'AbstractBlob', peer: 'KademliaPeer', connection_id: int = 0,
|
|
||||||
just_probe: bool = False):
|
|
||||||
if blob.get_is_verified():
|
|
||||||
return
|
|
||||||
start = self.loop.time()
|
|
||||||
bytes_received, protocol = await request_blob(
|
|
||||||
self.loop, blob if not just_probe else None, peer.address, peer.tcp_port, self.config.peer_connect_timeout,
|
|
||||||
self.config.blob_download_timeout, connected_protocol=self.connections.get(peer),
|
|
||||||
connection_id=connection_id, connection_manager=self.blob_manager.connection_manager
|
|
||||||
)
|
|
||||||
if not bytes_received and not protocol and peer not in self.connection_failures:
|
|
||||||
self.connection_failures.add(peer)
|
|
||||||
if not protocol and peer not in self.ignored:
|
|
||||||
self.ignored[peer] = self.loop.time()
|
|
||||||
log.debug("drop peer %s:%i", peer.address, peer.tcp_port)
|
|
||||||
self.failures[peer] = self.failures.get(peer, 0) + 1
|
|
||||||
if peer in self.connections:
|
|
||||||
del self.connections[peer]
|
|
||||||
elif protocol:
|
|
||||||
log.debug("keep peer %s:%i", peer.address, peer.tcp_port)
|
|
||||||
self.failures[peer] = 0
|
|
||||||
self.connections[peer] = protocol
|
|
||||||
elapsed = self.loop.time() - start
|
|
||||||
self.scores[peer] = bytes_received / elapsed if bytes_received and elapsed else 1
|
|
||||||
|
|
||||||
async def new_peer_or_finished(self):
|
|
||||||
active_tasks = list(self.active_connections.values()) + [asyncio.create_task(asyncio.sleep(1))]
|
|
||||||
await asyncio.wait(active_tasks, return_when='FIRST_COMPLETED')
|
|
||||||
|
|
||||||
def cleanup_active(self):
|
|
||||||
if not self.active_connections and not self.connections:
|
|
||||||
self.clearbanned()
|
|
||||||
to_remove = [peer for (peer, task) in self.active_connections.items() if task.done()]
|
|
||||||
for peer in to_remove:
|
|
||||||
del self.active_connections[peer]
|
|
||||||
|
|
||||||
def clearbanned(self):
|
|
||||||
now = self.loop.time()
|
|
||||||
self.ignored = {
|
|
||||||
peer: when for (peer, when) in self.ignored.items()
|
|
||||||
if (now - when) < min(30.0, (self.failures.get(peer, 0) ** self.BAN_FACTOR))
|
|
||||||
}
|
|
||||||
|
|
||||||
@cache_concurrent
|
|
||||||
async def download_blob(self, blob_hash: str, length: typing.Optional[int] = None,
|
|
||||||
connection_id: int = 0) -> 'AbstractBlob':
|
|
||||||
blob = self.blob_manager.get_blob(blob_hash, length)
|
|
||||||
if blob.get_is_verified():
|
|
||||||
return blob
|
|
||||||
self.is_running.set()
|
|
||||||
try:
|
|
||||||
while not blob.get_is_verified() and self.is_running.is_set():
|
|
||||||
batch: typing.Set['KademliaPeer'] = set(self.connections.keys())
|
|
||||||
while not self.peer_queue.empty():
|
|
||||||
batch.update(self.peer_queue.get_nowait())
|
|
||||||
log.debug(
|
|
||||||
"%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)
|
|
||||||
)
|
|
||||||
for peer in sorted(batch, key=lambda peer: self.scores.get(peer, 0), reverse=True):
|
|
||||||
if peer in self.ignored:
|
|
||||||
continue
|
|
||||||
if peer in self.active_connections or not self.should_race_continue(blob):
|
|
||||||
continue
|
|
||||||
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))
|
|
||||||
self.active_connections[peer] = t
|
|
||||||
self.peer_queue.put_nowait(list(batch))
|
|
||||||
await self.new_peer_or_finished()
|
|
||||||
self.cleanup_active()
|
|
||||||
log.debug("downloaded %s", blob_hash[:8])
|
|
||||||
return blob
|
|
||||||
finally:
|
|
||||||
blob.close()
|
|
||||||
if self.loop.is_running():
|
|
||||||
self.loop.call_soon(self.cleanup_active)
|
|
||||||
|
|
||||||
def close(self):
|
|
||||||
self.connection_failures.clear()
|
|
||||||
self.scores.clear()
|
|
||||||
self.ignored.clear()
|
|
||||||
self.is_running.clear()
|
|
||||||
for protocol in self.connections.values():
|
|
||||||
protocol.close()
|
|
||||||
|
|
||||||
|
|
||||||
async def download_blob(loop, config: 'Config', blob_manager: 'BlobManager', dht_node: 'Node',
|
|
||||||
blob_hash: str) -> 'AbstractBlob':
|
|
||||||
search_queue = asyncio.Queue(maxsize=config.max_connections_per_download)
|
|
||||||
search_queue.put_nowait(blob_hash)
|
|
||||||
peer_queue, accumulate_task = dht_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)
|
|
||||||
try:
|
|
||||||
return await downloader.download_blob(blob_hash)
|
|
||||||
finally:
|
|
||||||
if accumulate_task and not accumulate_task.done():
|
|
||||||
accumulate_task.cancel()
|
|
||||||
downloader.close()
|
|
|
@ -1,194 +0,0 @@
|
||||||
import asyncio
|
|
||||||
import binascii
|
|
||||||
import logging
|
|
||||||
import socket
|
|
||||||
import typing
|
|
||||||
from json.decoder import JSONDecodeError
|
|
||||||
from lbry.blob_exchange.serialization import BlobResponse, BlobRequest, blob_response_types
|
|
||||||
from lbry.blob_exchange.serialization import BlobAvailabilityResponse, BlobPriceResponse, BlobDownloadResponse, \
|
|
||||||
BlobPaymentAddressResponse
|
|
||||||
|
|
||||||
if typing.TYPE_CHECKING:
|
|
||||||
from lbry.blob.blob_manager import BlobManager
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
# a standard request will be 295 bytes
|
|
||||||
MAX_REQUEST_SIZE = 1200
|
|
||||||
|
|
||||||
|
|
||||||
class BlobServerProtocol(asyncio.Protocol):
|
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop, blob_manager: 'BlobManager', lbrycrd_address: str,
|
|
||||||
idle_timeout: float = 30.0, transfer_timeout: float = 60.0):
|
|
||||||
self.loop = loop
|
|
||||||
self.blob_manager = blob_manager
|
|
||||||
self.idle_timeout = idle_timeout
|
|
||||||
self.transfer_timeout = transfer_timeout
|
|
||||||
self.server_task: typing.Optional[asyncio.Task] = None
|
|
||||||
self.started_listening = asyncio.Event()
|
|
||||||
self.buf = b''
|
|
||||||
self.transport: typing.Optional[asyncio.Transport] = None
|
|
||||||
self.lbrycrd_address = lbrycrd_address
|
|
||||||
self.peer_address_and_port: typing.Optional[str] = None
|
|
||||||
self.started_transfer = asyncio.Event()
|
|
||||||
self.transfer_finished = asyncio.Event()
|
|
||||||
self.close_on_idle_task: typing.Optional[asyncio.Task] = None
|
|
||||||
|
|
||||||
async def close_on_idle(self):
|
|
||||||
while self.transport:
|
|
||||||
try:
|
|
||||||
await asyncio.wait_for(self.started_transfer.wait(), self.idle_timeout)
|
|
||||||
except asyncio.TimeoutError:
|
|
||||||
log.debug("closing idle connection from %s", self.peer_address_and_port)
|
|
||||||
return self.close()
|
|
||||||
self.started_transfer.clear()
|
|
||||||
await self.transfer_finished.wait()
|
|
||||||
self.transfer_finished.clear()
|
|
||||||
|
|
||||||
def close(self):
|
|
||||||
if self.transport:
|
|
||||||
self.transport.close()
|
|
||||||
|
|
||||||
def connection_made(self, transport):
|
|
||||||
self.transport = transport
|
|
||||||
self.close_on_idle_task = self.loop.create_task(self.close_on_idle())
|
|
||||||
self.peer_address_and_port = "%s:%i" % self.transport.get_extra_info('peername')
|
|
||||||
self.blob_manager.connection_manager.connection_received(self.peer_address_and_port)
|
|
||||||
log.debug("received connection from %s", self.peer_address_and_port)
|
|
||||||
|
|
||||||
def connection_lost(self, exc: typing.Optional[Exception]) -> None:
|
|
||||||
log.debug("lost connection from %s", self.peer_address_and_port)
|
|
||||||
self.blob_manager.connection_manager.incoming_connection_lost(self.peer_address_and_port)
|
|
||||||
self.transport = None
|
|
||||||
if self.close_on_idle_task and not self.close_on_idle_task.done():
|
|
||||||
self.close_on_idle_task.cancel()
|
|
||||||
self.close_on_idle_task = None
|
|
||||||
|
|
||||||
def send_response(self, responses: typing.List[blob_response_types]):
|
|
||||||
to_send = []
|
|
||||||
while responses:
|
|
||||||
to_send.append(responses.pop())
|
|
||||||
serialized = BlobResponse(to_send).serialize()
|
|
||||||
self.transport.write(serialized)
|
|
||||||
self.blob_manager.connection_manager.sent_data(self.peer_address_and_port, len(serialized))
|
|
||||||
|
|
||||||
async def handle_request(self, request: BlobRequest):
|
|
||||||
addr = self.transport.get_extra_info('peername')
|
|
||||||
peer_address, peer_port = addr
|
|
||||||
|
|
||||||
responses = []
|
|
||||||
address_request = request.get_address_request()
|
|
||||||
if address_request:
|
|
||||||
responses.append(BlobPaymentAddressResponse(lbrycrd_address=self.lbrycrd_address))
|
|
||||||
availability_request = request.get_availability_request()
|
|
||||||
if availability_request:
|
|
||||||
responses.append(BlobAvailabilityResponse(available_blobs=list(set(
|
|
||||||
filter(lambda blob_hash: blob_hash in self.blob_manager.completed_blob_hashes,
|
|
||||||
availability_request.requested_blobs)
|
|
||||||
))))
|
|
||||||
price_request = request.get_price_request()
|
|
||||||
if price_request:
|
|
||||||
responses.append(BlobPriceResponse(blob_data_payment_rate='RATE_ACCEPTED'))
|
|
||||||
download_request = request.get_blob_request()
|
|
||||||
|
|
||||||
if download_request:
|
|
||||||
blob = self.blob_manager.get_blob(download_request.requested_blob)
|
|
||||||
if blob.get_is_verified():
|
|
||||||
incoming_blob = {'blob_hash': blob.blob_hash, 'length': blob.length}
|
|
||||||
responses.append(BlobDownloadResponse(incoming_blob=incoming_blob))
|
|
||||||
self.send_response(responses)
|
|
||||||
blob_hash = blob.blob_hash[:8]
|
|
||||||
log.debug("send %s to %s:%i", blob_hash, peer_address, peer_port)
|
|
||||||
self.started_transfer.set()
|
|
||||||
try:
|
|
||||||
sent = await asyncio.wait_for(blob.sendfile(self), self.transfer_timeout)
|
|
||||||
if sent and sent > 0:
|
|
||||||
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)
|
|
||||||
else:
|
|
||||||
self.close()
|
|
||||||
log.debug("stopped sending %s to %s:%i", blob_hash, peer_address, peer_port)
|
|
||||||
return
|
|
||||||
except (OSError, ValueError, asyncio.TimeoutError) as err:
|
|
||||||
if isinstance(err, asyncio.TimeoutError):
|
|
||||||
log.debug("timed out sending blob %s to %s", blob_hash, peer_address)
|
|
||||||
else:
|
|
||||||
log.warning("could not read blob %s to send %s:%i", blob_hash, peer_address, peer_port)
|
|
||||||
self.close()
|
|
||||||
return
|
|
||||||
finally:
|
|
||||||
self.transfer_finished.set()
|
|
||||||
else:
|
|
||||||
log.info("don't have %s to send %s:%i", blob.blob_hash[:8], peer_address, peer_port)
|
|
||||||
if responses and not self.transport.is_closing():
|
|
||||||
self.send_response(responses)
|
|
||||||
|
|
||||||
def data_received(self, data):
|
|
||||||
request = None
|
|
||||||
if len(self.buf) + len(data or b'') >= MAX_REQUEST_SIZE:
|
|
||||||
log.warning("request from %s is too large", self.peer_address_and_port)
|
|
||||||
self.close()
|
|
||||||
return
|
|
||||||
if data:
|
|
||||||
self.blob_manager.connection_manager.received_data(self.peer_address_and_port, len(data))
|
|
||||||
_, separator, remainder = data.rpartition(b'}')
|
|
||||||
if not separator:
|
|
||||||
self.buf += data
|
|
||||||
return
|
|
||||||
try:
|
|
||||||
request = BlobRequest.deserialize(self.buf + data)
|
|
||||||
self.buf = remainder
|
|
||||||
except (UnicodeDecodeError, JSONDecodeError):
|
|
||||||
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())
|
|
||||||
self.close()
|
|
||||||
return
|
|
||||||
if not request.requests:
|
|
||||||
log.error("failed to decode request from %s (%i bytes): %s", self.peer_address_and_port,
|
|
||||||
len(self.buf + data), '' if not data else binascii.hexlify(self.buf + data).decode())
|
|
||||||
self.close()
|
|
||||||
return
|
|
||||||
self.loop.create_task(self.handle_request(request))
|
|
||||||
|
|
||||||
|
|
||||||
class BlobServer:
|
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop, blob_manager: 'BlobManager', lbrycrd_address: str,
|
|
||||||
idle_timeout: float = 30.0, transfer_timeout: float = 60.0):
|
|
||||||
self.loop = loop
|
|
||||||
self.blob_manager = blob_manager
|
|
||||||
self.server_task: typing.Optional[asyncio.Task] = None
|
|
||||||
self.started_listening = asyncio.Event()
|
|
||||||
self.lbrycrd_address = lbrycrd_address
|
|
||||||
self.idle_timeout = idle_timeout
|
|
||||||
self.transfer_timeout = transfer_timeout
|
|
||||||
self.server_protocol_class = BlobServerProtocol
|
|
||||||
|
|
||||||
def start_server(self, port: int, interface: typing.Optional[str] = '0.0.0.0'):
|
|
||||||
if self.server_task is not None:
|
|
||||||
raise Exception("already running")
|
|
||||||
|
|
||||||
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(
|
|
||||||
lambda: self.server_protocol_class(self.loop, self.blob_manager, self.lbrycrd_address,
|
|
||||||
self.idle_timeout, self.transfer_timeout),
|
|
||||||
interface, port
|
|
||||||
)
|
|
||||||
self.started_listening.set()
|
|
||||||
log.info("Blob server listening on TCP %s:%i", interface, port)
|
|
||||||
async with server:
|
|
||||||
await server.serve_forever()
|
|
||||||
|
|
||||||
self.server_task = self.loop.create_task(_start_server())
|
|
||||||
|
|
||||||
def stop_server(self):
|
|
||||||
if self.server_task:
|
|
||||||
self.server_task.cancel()
|
|
||||||
self.server_task = None
|
|
||||||
log.info("Stopped blob server")
|
|
|
@ -1,105 +0,0 @@
|
||||||
import time
|
|
||||||
import asyncio
|
|
||||||
import typing
|
|
||||||
import collections
|
|
||||||
import logging
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
CONNECTED_EVENT = "connected"
|
|
||||||
DISCONNECTED_EVENT = "disconnected"
|
|
||||||
TRANSFERRED_EVENT = "transferred"
|
|
||||||
|
|
||||||
|
|
||||||
class ConnectionManager:
|
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop):
|
|
||||||
self.loop = loop
|
|
||||||
self.incoming_connected: typing.Set[str] = set()
|
|
||||||
self.incoming: typing.DefaultDict[str, int] = collections.defaultdict(int)
|
|
||||||
self.outgoing_connected: typing.Set[str] = set()
|
|
||||||
self.outgoing: typing.DefaultDict[str, int] = collections.defaultdict(int)
|
|
||||||
self._max_incoming_mbs = 0.0
|
|
||||||
self._max_outgoing_mbs = 0.0
|
|
||||||
self._status = {}
|
|
||||||
self._running = False
|
|
||||||
self._task: typing.Optional[asyncio.Task] = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def status(self):
|
|
||||||
return self._status
|
|
||||||
|
|
||||||
def sent_data(self, host_and_port: str, size: int):
|
|
||||||
if self._running:
|
|
||||||
self.outgoing[host_and_port] += size
|
|
||||||
|
|
||||||
def received_data(self, host_and_port: str, size: int):
|
|
||||||
if self._running:
|
|
||||||
self.incoming[host_and_port] += size
|
|
||||||
|
|
||||||
def connection_made(self, host_and_port: str):
|
|
||||||
if self._running:
|
|
||||||
self.outgoing_connected.add(host_and_port)
|
|
||||||
|
|
||||||
def connection_received(self, host_and_port: str):
|
|
||||||
# self.incoming_connected.add(host_and_port)
|
|
||||||
pass
|
|
||||||
|
|
||||||
def outgoing_connection_lost(self, host_and_port: str):
|
|
||||||
if self._running and host_and_port in self.outgoing_connected:
|
|
||||||
self.outgoing_connected.remove(host_and_port)
|
|
||||||
|
|
||||||
def incoming_connection_lost(self, host_and_port: str):
|
|
||||||
if self._running and host_and_port in self.incoming_connected:
|
|
||||||
self.incoming_connected.remove(host_and_port)
|
|
||||||
|
|
||||||
async def _update(self):
|
|
||||||
self._status = {
|
|
||||||
'incoming_bps': {},
|
|
||||||
'outgoing_bps': {},
|
|
||||||
'total_incoming_mbs': 0.0,
|
|
||||||
'total_outgoing_mbs': 0.0,
|
|
||||||
'total_sent': 0,
|
|
||||||
'total_received': 0,
|
|
||||||
'max_incoming_mbs': 0.0,
|
|
||||||
'max_outgoing_mbs': 0.0
|
|
||||||
}
|
|
||||||
|
|
||||||
while True:
|
|
||||||
last = time.perf_counter()
|
|
||||||
await asyncio.sleep(0.1)
|
|
||||||
self._status['incoming_bps'].clear()
|
|
||||||
self._status['outgoing_bps'].clear()
|
|
||||||
now = time.perf_counter()
|
|
||||||
while self.outgoing:
|
|
||||||
k, sent = self.outgoing.popitem()
|
|
||||||
self._status['total_sent'] += sent
|
|
||||||
self._status['outgoing_bps'][k] = sent / (now - last)
|
|
||||||
while self.incoming:
|
|
||||||
k, received = self.incoming.popitem()
|
|
||||||
self._status['total_received'] += received
|
|
||||||
self._status['incoming_bps'][k] = received / (now - last)
|
|
||||||
self._status['total_outgoing_mbs'] = int(sum(list(self._status['outgoing_bps'].values())
|
|
||||||
)) / 1000000.0
|
|
||||||
self._status['total_incoming_mbs'] = int(sum(list(self._status['incoming_bps'].values())
|
|
||||||
)) / 1000000.0
|
|
||||||
self._max_incoming_mbs = max(self._max_incoming_mbs, self._status['total_incoming_mbs'])
|
|
||||||
self._max_outgoing_mbs = max(self._max_outgoing_mbs, self._status['total_outgoing_mbs'])
|
|
||||||
self._status['max_incoming_mbs'] = self._max_incoming_mbs
|
|
||||||
self._status['max_outgoing_mbs'] = self._max_outgoing_mbs
|
|
||||||
|
|
||||||
def stop(self):
|
|
||||||
if self._task:
|
|
||||||
self._task.cancel()
|
|
||||||
self._task = None
|
|
||||||
self.outgoing.clear()
|
|
||||||
self.outgoing_connected.clear()
|
|
||||||
self.incoming.clear()
|
|
||||||
self.incoming_connected.clear()
|
|
||||||
self._status.clear()
|
|
||||||
self._running = False
|
|
||||||
|
|
||||||
def start(self):
|
|
||||||
self.stop()
|
|
||||||
self._running = True
|
|
||||||
self._task = self.loop.create_task(self._update())
|
|
|
@ -1,2 +0,0 @@
|
||||||
CENT = 1000000
|
|
||||||
COIN = 100*CENT
|
|
|
@ -1,86 +0,0 @@
|
||||||
from lbry.crypto.hash import double_sha256
|
|
||||||
from lbry.crypto.util import bytes_to_int, int_to_bytes
|
|
||||||
|
|
||||||
|
|
||||||
class Base58Error(Exception):
|
|
||||||
""" Exception used for Base58 errors. """
|
|
||||||
|
|
||||||
|
|
||||||
class Base58:
|
|
||||||
""" Class providing base 58 functionality. """
|
|
||||||
|
|
||||||
chars = '123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz'
|
|
||||||
assert len(chars) == 58
|
|
||||||
char_map = {c: n for n, c in enumerate(chars)}
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def char_value(cls, c):
|
|
||||||
val = cls.char_map.get(c)
|
|
||||||
if val is None:
|
|
||||||
raise Base58Error(f'invalid base 58 character "{c}"')
|
|
||||||
return val
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def decode(cls, txt):
|
|
||||||
""" Decodes txt into a big-endian bytearray. """
|
|
||||||
if isinstance(txt, memoryview):
|
|
||||||
txt = str(txt)
|
|
||||||
|
|
||||||
if isinstance(txt, bytes):
|
|
||||||
txt = txt.decode()
|
|
||||||
|
|
||||||
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 + cls.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((0,)) * count + result
|
|
||||||
|
|
||||||
return result
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def encode(cls, 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 += cls.chars[mod]
|
|
||||||
|
|
||||||
for byte in be_bytes:
|
|
||||||
if byte != 0:
|
|
||||||
break
|
|
||||||
txt += '1'
|
|
||||||
|
|
||||||
return txt[::-1]
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def decode_check(cls, txt, hash_fn=double_sha256):
|
|
||||||
""" Decodes a Base58Check-encoded string to a payload. The version prefixes it. """
|
|
||||||
be_bytes = cls.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
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def encode_check(cls, 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 cls.encode(be_bytes)
|
|
|
@ -1,71 +0,0 @@
|
||||||
import os
|
|
||||||
import base64
|
|
||||||
import typing
|
|
||||||
from cryptography.hazmat.primitives.kdf.scrypt import Scrypt
|
|
||||||
from cryptography.hazmat.primitives.ciphers import Cipher, modes
|
|
||||||
from cryptography.hazmat.primitives.ciphers.algorithms import AES
|
|
||||||
from cryptography.hazmat.primitives.padding import PKCS7
|
|
||||||
from cryptography.hazmat.backends import default_backend
|
|
||||||
|
|
||||||
from lbry.error import InvalidPasswordError
|
|
||||||
from lbry.crypto.hash import double_sha256
|
|
||||||
|
|
||||||
|
|
||||||
def aes_encrypt(secret: str, value: str, init_vector: bytes = None) -> str:
|
|
||||||
if init_vector is not None:
|
|
||||||
assert len(init_vector) == 16
|
|
||||||
else:
|
|
||||||
init_vector = os.urandom(16)
|
|
||||||
key = double_sha256(secret.encode())
|
|
||||||
encryptor = Cipher(AES(key), modes.CBC(init_vector), default_backend()).encryptor()
|
|
||||||
padder = PKCS7(AES.block_size).padder()
|
|
||||||
padded_data = padder.update(value.encode()) + padder.finalize()
|
|
||||||
encrypted_data = encryptor.update(padded_data) + encryptor.finalize()
|
|
||||||
return base64.b64encode(init_vector + encrypted_data).decode()
|
|
||||||
|
|
||||||
|
|
||||||
def aes_decrypt(secret: str, value: str) -> typing.Tuple[str, bytes]:
|
|
||||||
try:
|
|
||||||
data = base64.b64decode(value.encode())
|
|
||||||
key = double_sha256(secret.encode())
|
|
||||||
init_vector, data = data[:16], data[16:]
|
|
||||||
decryptor = Cipher(AES(key), modes.CBC(init_vector), default_backend()).decryptor()
|
|
||||||
unpadder = PKCS7(AES.block_size).unpadder()
|
|
||||||
result = unpadder.update(decryptor.update(data)) + unpadder.finalize()
|
|
||||||
return result.decode(), init_vector
|
|
||||||
except UnicodeDecodeError:
|
|
||||||
raise InvalidPasswordError()
|
|
||||||
except ValueError as e:
|
|
||||||
if e.args[0] == 'Invalid padding bytes.':
|
|
||||||
raise InvalidPasswordError()
|
|
||||||
raise
|
|
||||||
|
|
||||||
|
|
||||||
def better_aes_encrypt(secret: str, value: bytes) -> bytes:
|
|
||||||
init_vector = os.urandom(16)
|
|
||||||
key = scrypt(secret.encode(), salt=init_vector)
|
|
||||||
encryptor = Cipher(AES(key), modes.CBC(init_vector), default_backend()).encryptor()
|
|
||||||
padder = PKCS7(AES.block_size).padder()
|
|
||||||
padded_data = padder.update(value) + padder.finalize()
|
|
||||||
encrypted_data = encryptor.update(padded_data) + encryptor.finalize()
|
|
||||||
return base64.b64encode(b's:8192:16:1:' + init_vector + encrypted_data)
|
|
||||||
|
|
||||||
|
|
||||||
def better_aes_decrypt(secret: str, value: bytes) -> bytes:
|
|
||||||
try:
|
|
||||||
data = base64.b64decode(value)
|
|
||||||
_, scryp_n, scrypt_r, scrypt_p, data = data.split(b':', maxsplit=4)
|
|
||||||
init_vector, data = data[:16], data[16:]
|
|
||||||
key = scrypt(secret.encode(), init_vector, int(scryp_n), int(scrypt_r), int(scrypt_p))
|
|
||||||
decryptor = Cipher(AES(key), modes.CBC(init_vector), default_backend()).decryptor()
|
|
||||||
unpadder = PKCS7(AES.block_size).unpadder()
|
|
||||||
return unpadder.update(decryptor.update(data)) + unpadder.finalize()
|
|
||||||
except ValueError as e:
|
|
||||||
if e.args[0] == 'Invalid padding bytes.':
|
|
||||||
raise InvalidPasswordError()
|
|
||||||
raise
|
|
||||||
|
|
||||||
|
|
||||||
def scrypt(passphrase, salt, scrypt_n=1<<13, scrypt_r=16, scrypt_p=1):
|
|
||||||
kdf = Scrypt(salt, length=32, n=scrypt_n, r=scrypt_r, p=scrypt_p, backend=default_backend())
|
|
||||||
return kdf.derive(passphrase)
|
|
|
@ -1,47 +0,0 @@
|
||||||
import hashlib
|
|
||||||
import hmac
|
|
||||||
from binascii import hexlify, unhexlify
|
|
||||||
|
|
||||||
|
|
||||||
def sha256(x):
|
|
||||||
""" Simple wrapper of hashlib sha256. """
|
|
||||||
return hashlib.sha256(x).digest()
|
|
||||||
|
|
||||||
|
|
||||||
def sha512(x):
|
|
||||||
""" Simple wrapper of hashlib sha512. """
|
|
||||||
return hashlib.sha512(x).digest()
|
|
||||||
|
|
||||||
|
|
||||||
def ripemd160(x):
|
|
||||||
""" Simple wrapper of hashlib ripemd160. """
|
|
||||||
h = hashlib.new('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 hmac.new(key, msg, hashlib.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):
|
|
||||||
""" Convert a big-endian binary hash to displayed hex string.
|
|
||||||
Display form of a binary hash is reversed and converted to hex. """
|
|
||||||
return hexlify(reversed(x))
|
|
||||||
|
|
||||||
|
|
||||||
def hex_str_to_hash(x):
|
|
||||||
""" Convert a displayed hex string to a binary hash. """
|
|
||||||
return reversed(unhexlify(x))
|
|
|
@ -1,13 +0,0 @@
|
||||||
from binascii import unhexlify, hexlify
|
|
||||||
|
|
||||||
|
|
||||||
def bytes_to_int(be_bytes):
|
|
||||||
""" Interprets a big-endian sequence of bytes as an integer. """
|
|
||||||
return int(hexlify(be_bytes), 16)
|
|
||||||
|
|
||||||
|
|
||||||
def int_to_bytes(value):
|
|
||||||
""" Converts an integer to a big-endian sequence of bytes. """
|
|
||||||
length = (value.bit_length() + 7) // 8
|
|
||||||
s = '%x' % value
|
|
||||||
return unhexlify(('0' * (len(s) % 2) + s).zfill(length * 2))
|
|
|
@ -1,78 +0,0 @@
|
||||||
import asyncio
|
|
||||||
import typing
|
|
||||||
import logging
|
|
||||||
|
|
||||||
from prometheus_client import Counter, Gauge
|
|
||||||
|
|
||||||
if typing.TYPE_CHECKING:
|
|
||||||
from lbry.dht.node import Node
|
|
||||||
from lbry.extras.daemon.storage import SQLiteStorage
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
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'):
|
|
||||||
self.loop = loop
|
|
||||||
self.node = node
|
|
||||||
self.storage = storage
|
|
||||||
self.announce_task: asyncio.Task = None
|
|
||||||
self.announce_queue: typing.List[str] = []
|
|
||||||
self._done = asyncio.Event()
|
|
||||||
self.announced = set()
|
|
||||||
|
|
||||||
async def _run_consumer(self):
|
|
||||||
while self.announce_queue:
|
|
||||||
try:
|
|
||||||
blob_hash = self.announce_queue.pop()
|
|
||||||
peers = len(await self.node.announce_blob(blob_hash))
|
|
||||||
self.announcements_sent_metric.labels(peers=peers, error=False).inc()
|
|
||||||
if peers > 4:
|
|
||||||
self.announced.add(blob_hash)
|
|
||||||
else:
|
|
||||||
log.debug("failed to announce %s, could only find %d peers, retrying soon.", blob_hash[:8], peers)
|
|
||||||
except Exception as err:
|
|
||||||
self.announcements_sent_metric.labels(peers=0, error=True).inc()
|
|
||||||
log.warning("error announcing %s: %s", blob_hash[:8], str(err))
|
|
||||||
|
|
||||||
async def _announce(self, batch_size: typing.Optional[int] = 10):
|
|
||||||
while batch_size:
|
|
||||||
if not self.node.joined.is_set():
|
|
||||||
await self.node.joined.wait()
|
|
||||||
await asyncio.sleep(60)
|
|
||||||
if not self.node.protocol.routing_table.get_peers():
|
|
||||||
log.warning("No peers in DHT, announce round skipped")
|
|
||||||
continue
|
|
||||||
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))
|
|
||||||
while len(self.announce_queue) > 0:
|
|
||||||
log.info("%i blobs to announce", len(self.announce_queue))
|
|
||||||
await asyncio.gather(*[self._run_consumer() for _ in range(batch_size)])
|
|
||||||
announced = list(filter(None, self.announced))
|
|
||||||
if announced:
|
|
||||||
await self.storage.update_last_announced_blobs(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):
|
|
||||||
assert not self.announce_task or self.announce_task.done(), "already running"
|
|
||||||
self.announce_task = self.loop.create_task(self._announce(batch_size))
|
|
||||||
|
|
||||||
def stop(self):
|
|
||||||
if self.announce_task and not self.announce_task.done():
|
|
||||||
self.announce_task.cancel()
|
|
||||||
|
|
||||||
def wait(self):
|
|
||||||
return self._done.wait()
|
|
|
@ -1,40 +0,0 @@
|
||||||
import hashlib
|
|
||||||
import os
|
|
||||||
|
|
||||||
HASH_CLASS = hashlib.sha384 # pylint: disable=invalid-name
|
|
||||||
HASH_LENGTH = HASH_CLASS().digest_size
|
|
||||||
HASH_BITS = HASH_LENGTH * 8
|
|
||||||
ALPHA = 5
|
|
||||||
K = 8
|
|
||||||
SPLIT_BUCKETS_UNDER_INDEX = 1
|
|
||||||
REPLACEMENT_CACHE_SIZE = 8
|
|
||||||
RPC_TIMEOUT = 5.0
|
|
||||||
RPC_ATTEMPTS = 5
|
|
||||||
RPC_ATTEMPTS_PRUNING_WINDOW = 600
|
|
||||||
ITERATIVE_LOOKUP_DELAY = RPC_TIMEOUT / 2.0 # TODO: use config val / 2 if rpc timeout is provided
|
|
||||||
REFRESH_INTERVAL = 3600 # 1 hour
|
|
||||||
REPLICATE_INTERVAL = REFRESH_INTERVAL
|
|
||||||
DATA_EXPIRATION = 86400 # 24 hours
|
|
||||||
TOKEN_SECRET_REFRESH_INTERVAL = 300 # 5 minutes
|
|
||||||
MAYBE_PING_DELAY = 300 # 5 minutes
|
|
||||||
CHECK_REFRESH_INTERVAL = REFRESH_INTERVAL / 5
|
|
||||||
RPC_ID_LENGTH = 20
|
|
||||||
PROTOCOL_VERSION = 1
|
|
||||||
MSG_SIZE_LIMIT = 1400
|
|
||||||
|
|
||||||
|
|
||||||
def digest(data: bytes) -> bytes:
|
|
||||||
h = HASH_CLASS()
|
|
||||||
h.update(data)
|
|
||||||
return h.digest()
|
|
||||||
|
|
||||||
|
|
||||||
def generate_id(num=None) -> bytes:
|
|
||||||
if num is not None:
|
|
||||||
return digest(str(num).encode())
|
|
||||||
else:
|
|
||||||
return digest(os.urandom(32))
|
|
||||||
|
|
||||||
|
|
||||||
def generate_rpc_id(num=None) -> bytes:
|
|
||||||
return generate_id(num)[:RPC_ID_LENGTH]
|
|
282
lbry/dht/node.py
282
lbry/dht/node.py
|
@ -1,282 +0,0 @@
|
||||||
import logging
|
|
||||||
import asyncio
|
|
||||||
import typing
|
|
||||||
import socket
|
|
||||||
|
|
||||||
from prometheus_client import Gauge
|
|
||||||
|
|
||||||
from lbry.utils import aclosing, resolve_host
|
|
||||||
from lbry.dht import constants
|
|
||||||
from lbry.dht.peer import make_kademlia_peer
|
|
||||||
from lbry.dht.protocol.distance import Distance
|
|
||||||
from lbry.dht.protocol.iterative_find import IterativeNodeFinder, IterativeValueFinder
|
|
||||||
from lbry.dht.protocol.protocol import KademliaProtocol
|
|
||||||
|
|
||||||
if typing.TYPE_CHECKING:
|
|
||||||
from lbry.dht.peer import PeerManager
|
|
||||||
from lbry.dht.peer import KademliaPeer
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
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,
|
|
||||||
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,
|
|
||||||
storage: typing.Optional['SQLiteStorage'] = None):
|
|
||||||
self.loop = loop
|
|
||||||
self.internal_udp_port = internal_udp_port
|
|
||||||
self.protocol = KademliaProtocol(loop, peer_manager, node_id, external_ip, udp_port, peer_port, rpc_timeout,
|
|
||||||
split_buckets_under_index, is_bootstrap_node)
|
|
||||||
self.listening_port: asyncio.DatagramTransport = None
|
|
||||||
self.joined = asyncio.Event()
|
|
||||||
self._join_task: asyncio.Task = None
|
|
||||||
self._refresh_task: asyncio.Task = None
|
|
||||||
self._storage = storage
|
|
||||||
|
|
||||||
@property
|
|
||||||
def stored_blob_hashes(self):
|
|
||||||
return self.protocol.data_store.keys()
|
|
||||||
|
|
||||||
async def refresh_node(self, force_once=False):
|
|
||||||
while True:
|
|
||||||
# remove peers with expired blob announcements from the datastore
|
|
||||||
self.protocol.data_store.removed_expired_peers()
|
|
||||||
|
|
||||||
total_peers: typing.List['KademliaPeer'] = []
|
|
||||||
# add all peers in the routing table
|
|
||||||
total_peers.extend(self.protocol.routing_table.get_peers())
|
|
||||||
# add all the peers who have announced blobs to us
|
|
||||||
storing_peers = 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
|
|
||||||
node_ids = self.protocol.routing_table.get_refresh_list(0, True)
|
|
||||||
|
|
||||||
if self.protocol.routing_table.get_peers():
|
|
||||||
# if we have node ids to look up, perform the iterative search until we have k results
|
|
||||||
while node_ids:
|
|
||||||
peers = await self.peer_search(node_ids.pop())
|
|
||||||
total_peers.extend(peers)
|
|
||||||
else:
|
|
||||||
if force_once:
|
|
||||||
break
|
|
||||||
fut = asyncio.Future()
|
|
||||||
self.loop.call_later(constants.REFRESH_INTERVAL // 4, fut.set_result, None)
|
|
||||||
await fut
|
|
||||||
continue
|
|
||||||
|
|
||||||
# ping the set of peers; upon success/failure the routing able and last replied/failed time will be updated
|
|
||||||
to_ping = [peer for peer in set(total_peers) if self.protocol.peer_manager.peer_is_good(peer) is not True]
|
|
||||||
if to_ping:
|
|
||||||
self.protocol.ping_queue.enqueue_maybe_ping(*to_ping, delay=0)
|
|
||||||
if self._storage:
|
|
||||||
await self._storage.save_kademlia_peers(self.protocol.routing_table.get_peers())
|
|
||||||
if force_once:
|
|
||||||
break
|
|
||||||
|
|
||||||
fut = asyncio.Future()
|
|
||||||
self.loop.call_later(constants.REFRESH_INTERVAL, fut.set_result, None)
|
|
||||||
await fut
|
|
||||||
|
|
||||||
async def announce_blob(self, blob_hash: str) -> typing.List[bytes]:
|
|
||||||
hash_value = bytes.fromhex(blob_hash)
|
|
||||||
assert len(hash_value) == constants.HASH_LENGTH
|
|
||||||
peers = await self.peer_search(hash_value)
|
|
||||||
|
|
||||||
if not self.protocol.external_ip:
|
|
||||||
raise Exception("Cannot determine external IP")
|
|
||||||
log.debug("Store to %i peers", len(peers))
|
|
||||||
for peer in peers:
|
|
||||||
log.debug("store to %s %s %s", peer.address, peer.udp_port, peer.tcp_port)
|
|
||||||
stored_to_tup = await asyncio.gather(
|
|
||||||
*(self.protocol.store_to_peer(hash_value, peer) for peer in peers)
|
|
||||||
)
|
|
||||||
stored_to = [node_id for node_id, contacted in stored_to_tup if contacted]
|
|
||||||
if stored_to:
|
|
||||||
log.debug(
|
|
||||||
"Stored %s to %i of %i attempted peers", hash_value.hex()[:8],
|
|
||||||
len(stored_to), len(peers)
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
log.debug("Failed announcing %s, stored to 0 peers", blob_hash[:8])
|
|
||||||
return stored_to
|
|
||||||
|
|
||||||
def stop(self) -> None:
|
|
||||||
if self.joined.is_set():
|
|
||||||
self.joined.clear()
|
|
||||||
if self._join_task:
|
|
||||||
self._join_task.cancel()
|
|
||||||
if self._refresh_task and not (self._refresh_task.done() or self._refresh_task.cancelled()):
|
|
||||||
self._refresh_task.cancel()
|
|
||||||
if self.protocol and self.protocol.ping_queue.running:
|
|
||||||
self.protocol.ping_queue.stop()
|
|
||||||
self.protocol.stop()
|
|
||||||
if self.listening_port is not None:
|
|
||||||
self.listening_port.close()
|
|
||||||
self._join_task = None
|
|
||||||
self.listening_port = None
|
|
||||||
log.info("Stopped DHT node")
|
|
||||||
|
|
||||||
async def start_listening(self, interface: str = '0.0.0.0') -> None:
|
|
||||||
if not self.listening_port:
|
|
||||||
self.listening_port, _ = await self.loop.create_datagram_endpoint(
|
|
||||||
lambda: self.protocol, (interface, self.internal_udp_port)
|
|
||||||
)
|
|
||||||
log.info("DHT node listening on UDP %s:%i", interface, self.internal_udp_port)
|
|
||||||
self.protocol.start()
|
|
||||||
else:
|
|
||||||
log.warning("Already bound to port %s", self.listening_port)
|
|
||||||
|
|
||||||
async def join_network(self, interface: str = '0.0.0.0',
|
|
||||||
known_node_urls: typing.Optional[typing.List[typing.Tuple[str, int]]] = None):
|
|
||||||
def peers_from_urls(urls: typing.Optional[typing.List[typing.Tuple[bytes, str, int, int]]]):
|
|
||||||
peer_addresses = []
|
|
||||||
for node_id, address, udp_port, tcp_port in urls:
|
|
||||||
if (node_id, address, udp_port, tcp_port) not in peer_addresses and \
|
|
||||||
(address, udp_port) != (self.protocol.external_ip, self.protocol.udp_port):
|
|
||||||
peer_addresses.append((node_id, address, udp_port, tcp_port))
|
|
||||||
return [make_kademlia_peer(*peer_address) for peer_address in peer_addresses]
|
|
||||||
|
|
||||||
if not self.listening_port:
|
|
||||||
await self.start_listening(interface)
|
|
||||||
self.protocol.ping_queue.start()
|
|
||||||
self._refresh_task = self.loop.create_task(self.refresh_node())
|
|
||||||
|
|
||||||
while True:
|
|
||||||
if self.protocol.routing_table.get_peers():
|
|
||||||
if not self.joined.is_set():
|
|
||||||
self.joined.set()
|
|
||||||
log.info(
|
|
||||||
"joined dht, %i peers known in %i buckets", len(self.protocol.routing_table.get_peers()),
|
|
||||||
self.protocol.routing_table.buckets_with_contacts()
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
if self.joined.is_set():
|
|
||||||
self.joined.clear()
|
|
||||||
seed_peers = peers_from_urls(
|
|
||||||
await self._storage.get_persisted_kademlia_peers()
|
|
||||||
) if self._storage else []
|
|
||||||
if not seed_peers:
|
|
||||||
try:
|
|
||||||
seed_peers.extend(peers_from_urls([
|
|
||||||
(None, await resolve_host(address, udp_port, 'udp'), udp_port, None)
|
|
||||||
for address, udp_port in known_node_urls or []
|
|
||||||
]))
|
|
||||||
except socket.gaierror:
|
|
||||||
await asyncio.sleep(30)
|
|
||||||
continue
|
|
||||||
|
|
||||||
self.protocol.peer_manager.reset()
|
|
||||||
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 asyncio.sleep(1)
|
|
||||||
|
|
||||||
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))
|
|
||||||
|
|
||||||
def get_iterative_node_finder(self, key: bytes, shortlist: typing.Optional[typing.List['KademliaPeer']] = None,
|
|
||||||
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)
|
|
||||||
|
|
||||||
def get_iterative_value_finder(self, key: bytes, shortlist: typing.Optional[typing.List['KademliaPeer']] = None,
|
|
||||||
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)
|
|
||||||
|
|
||||||
async def peer_search(self, node_id: bytes, count=constants.K, max_results=constants.K * 2,
|
|
||||||
shortlist: typing.Optional[typing.List['KademliaPeer']] = None
|
|
||||||
) -> typing.List['KademliaPeer']:
|
|
||||||
peers = []
|
|
||||||
async with aclosing(self.get_iterative_node_finder(
|
|
||||||
node_id, shortlist=shortlist, max_results=max_results)) as node_finder:
|
|
||||||
async for iteration_peers in node_finder:
|
|
||||||
peers.extend(iteration_peers)
|
|
||||||
distance = Distance(node_id)
|
|
||||||
peers.sort(key=lambda peer: distance(peer.node_id))
|
|
||||||
return peers[:count]
|
|
||||||
|
|
||||||
async def _accumulate_peers_for_value(self, search_queue: asyncio.Queue, result_queue: asyncio.Queue):
|
|
||||||
tasks = []
|
|
||||||
try:
|
|
||||||
while True:
|
|
||||||
blob_hash = await search_queue.get()
|
|
||||||
tasks.append(self.loop.create_task(self._peers_for_value_producer(blob_hash, result_queue)))
|
|
||||||
finally:
|
|
||||||
for task in tasks:
|
|
||||||
task.cancel()
|
|
||||||
|
|
||||||
async def _peers_for_value_producer(self, blob_hash: str, result_queue: asyncio.Queue):
|
|
||||||
async def put_into_result_queue_after_pong(_peer):
|
|
||||||
try:
|
|
||||||
await self.protocol.get_rpc_peer(_peer).ping()
|
|
||||||
result_queue.put_nowait([_peer])
|
|
||||||
log.debug("pong from %s:%i for %s", _peer.address, _peer.udp_port, blob_hash)
|
|
||||||
except asyncio.TimeoutError:
|
|
||||||
pass
|
|
||||||
|
|
||||||
# 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
|
|
||||||
async with aclosing(self.get_iterative_value_finder(bytes.fromhex(blob_hash))) as value_finder:
|
|
||||||
async for results in value_finder:
|
|
||||||
to_put = []
|
|
||||||
for peer in results:
|
|
||||||
if peer.address == self.protocol.external_ip and self.protocol.peer_port == peer.tcp_port:
|
|
||||||
continue
|
|
||||||
is_good = self.protocol.peer_manager.peer_is_good(peer)
|
|
||||||
if is_good:
|
|
||||||
# the peer has replied recently over UDP, it can probably be reached on the TCP port
|
|
||||||
to_put.append(peer)
|
|
||||||
elif is_good is None:
|
|
||||||
if not peer.udp_port:
|
|
||||||
# TODO: use the same port for TCP and UDP
|
|
||||||
# the udp port must be guessed
|
|
||||||
# default to the ports being the same. if the TCP port appears to be <=0.48.0 default,
|
|
||||||
# including on a network with several nodes, then assume the udp port is proportionately
|
|
||||||
# based on a starting port of 4444
|
|
||||||
udp_port_to_try = peer.tcp_port
|
|
||||||
if 3400 > peer.tcp_port > 3332:
|
|
||||||
udp_port_to_try = (peer.tcp_port - 3333) + 4444
|
|
||||||
self.loop.create_task(put_into_result_queue_after_pong(
|
|
||||||
make_kademlia_peer(peer.node_id, peer.address, udp_port_to_try, peer.tcp_port)
|
|
||||||
))
|
|
||||||
else:
|
|
||||||
self.loop.create_task(put_into_result_queue_after_pong(peer))
|
|
||||||
else:
|
|
||||||
# the peer is known to be bad/unreachable, skip trying to connect to it over TCP
|
|
||||||
log.debug("skip bad peer %s:%i for %s", peer.address, peer.tcp_port, blob_hash)
|
|
||||||
if to_put:
|
|
||||||
result_queue.put_nowait(to_put)
|
|
||||||
|
|
||||||
def accumulate_peers(self, search_queue: asyncio.Queue,
|
|
||||||
peer_queue: typing.Optional[asyncio.Queue] = None
|
|
||||||
) -> typing.Tuple[asyncio.Queue, asyncio.Task]:
|
|
||||||
queue = peer_queue or asyncio.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
|
|
|
@ -1,76 +0,0 @@
|
||||||
import asyncio
|
|
||||||
import typing
|
|
||||||
|
|
||||||
from lbry.dht import constants
|
|
||||||
if typing.TYPE_CHECKING:
|
|
||||||
from lbry.dht.peer import KademliaPeer, PeerManager
|
|
||||||
|
|
||||||
|
|
||||||
class DictDataStore:
|
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop, peer_manager: 'PeerManager'):
|
|
||||||
# Dictionary format:
|
|
||||||
# { <key>: [(<contact>, <age>), ...] }
|
|
||||||
self._data_store: typing.Dict[bytes, typing.List[typing.Tuple['KademliaPeer', float]]] = {}
|
|
||||||
|
|
||||||
self.loop = loop
|
|
||||||
self._peer_manager = peer_manager
|
|
||||||
self.completed_blobs: typing.Set[str] = set()
|
|
||||||
|
|
||||||
def keys(self):
|
|
||||||
return self._data_store.keys()
|
|
||||||
|
|
||||||
def __len__(self):
|
|
||||||
return self._data_store.__len__()
|
|
||||||
|
|
||||||
def removed_expired_peers(self):
|
|
||||||
now = self.loop.time()
|
|
||||||
keys = list(self._data_store.keys())
|
|
||||||
for key in keys:
|
|
||||||
to_remove = []
|
|
||||||
for (peer, ts) in self._data_store[key]:
|
|
||||||
if ts + constants.DATA_EXPIRATION < now or self._peer_manager.peer_is_good(peer) is False:
|
|
||||||
to_remove.append((peer, ts))
|
|
||||||
for item in to_remove:
|
|
||||||
self._data_store[key].remove(item)
|
|
||||||
if not self._data_store[key]:
|
|
||||||
del self._data_store[key]
|
|
||||||
|
|
||||||
def filter_bad_and_expired_peers(self, key: bytes) -> typing.Iterator['KademliaPeer']:
|
|
||||||
"""
|
|
||||||
Returns only non-expired and unknown/good peers
|
|
||||||
"""
|
|
||||||
for peer in self.filter_expired_peers(key):
|
|
||||||
if self._peer_manager.peer_is_good(peer) is not False:
|
|
||||||
yield peer
|
|
||||||
|
|
||||||
def filter_expired_peers(self, key: bytes) -> typing.Iterator['KademliaPeer']:
|
|
||||||
"""
|
|
||||||
Returns only non-expired peers
|
|
||||||
"""
|
|
||||||
now = self.loop.time()
|
|
||||||
for (peer, ts) in self._data_store.get(key, []):
|
|
||||||
if ts + constants.DATA_EXPIRATION > now:
|
|
||||||
yield peer
|
|
||||||
|
|
||||||
def has_peers_for_blob(self, key: bytes) -> bool:
|
|
||||||
return key in self._data_store
|
|
||||||
|
|
||||||
def add_peer_to_blob(self, contact: 'KademliaPeer', key: bytes) -> None:
|
|
||||||
now = self.loop.time()
|
|
||||||
if key in self._data_store:
|
|
||||||
current = list(filter(lambda x: x[0] == contact, self._data_store[key]))
|
|
||||||
if len(current) > 0:
|
|
||||||
self._data_store[key][self._data_store[key].index(current[0])] = contact, now
|
|
||||||
else:
|
|
||||||
self._data_store[key].append((contact, now))
|
|
||||||
else:
|
|
||||||
self._data_store[key] = [(contact, now)]
|
|
||||||
|
|
||||||
def get_peers_for_blob(self, key: bytes) -> typing.List['KademliaPeer']:
|
|
||||||
return list(self.filter_bad_and_expired_peers(key))
|
|
||||||
|
|
||||||
def get_storing_contacts(self) -> typing.List['KademliaPeer']:
|
|
||||||
peers = set()
|
|
||||||
for _, stored in self._data_store.items():
|
|
||||||
peers.update(set(map(lambda tup: tup[0], stored)))
|
|
||||||
return list(peers)
|
|
|
@ -1,361 +0,0 @@
|
||||||
import asyncio
|
|
||||||
from itertools import chain
|
|
||||||
from collections import defaultdict, OrderedDict
|
|
||||||
from collections.abc import AsyncIterator
|
|
||||||
import typing
|
|
||||||
import logging
|
|
||||||
from typing import TYPE_CHECKING
|
|
||||||
from lbry.dht import constants
|
|
||||||
from lbry.dht.error import RemoteException, TransportNotConnected
|
|
||||||
from lbry.dht.protocol.distance import Distance
|
|
||||||
from lbry.dht.peer import make_kademlia_peer, decode_tcp_peer_from_compact_address
|
|
||||||
from lbry.dht.serialization.datagram import PAGE_KEY
|
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
|
||||||
from lbry.dht.protocol.protocol import KademliaProtocol
|
|
||||||
from lbry.dht.peer import PeerManager, KademliaPeer
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class FindResponse:
|
|
||||||
@property
|
|
||||||
def found(self) -> bool:
|
|
||||||
raise NotImplementedError()
|
|
||||||
|
|
||||||
def get_close_triples(self) -> typing.List[typing.Tuple[bytes, str, int]]:
|
|
||||||
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):
|
|
||||||
def __init__(self, key: bytes, close_triples: typing.List[typing.Tuple[bytes, str, int]]):
|
|
||||||
self.key = key
|
|
||||||
self.close_triples = close_triples
|
|
||||||
|
|
||||||
@property
|
|
||||||
def found(self) -> bool:
|
|
||||||
return self.key in [triple[0] for triple in self.close_triples]
|
|
||||||
|
|
||||||
def get_close_triples(self) -> typing.List[typing.Tuple[bytes, str, int]]:
|
|
||||||
return self.close_triples
|
|
||||||
|
|
||||||
|
|
||||||
class FindValueResponse(FindResponse):
|
|
||||||
def __init__(self, key: bytes, result_dict: typing.Dict):
|
|
||||||
self.key = key
|
|
||||||
self.token = result_dict[b'token']
|
|
||||||
self.close_triples: typing.List[typing.Tuple[bytes, bytes, int]] = result_dict.get(b'contacts', [])
|
|
||||||
self.found_compact_addresses = result_dict.get(key, [])
|
|
||||||
self.pages = int(result_dict.get(PAGE_KEY, 0))
|
|
||||||
|
|
||||||
@property
|
|
||||||
def found(self) -> bool:
|
|
||||||
return len(self.found_compact_addresses) > 0
|
|
||||||
|
|
||||||
def get_close_triples(self) -> typing.List[typing.Tuple[bytes, str, int]]:
|
|
||||||
return [(node_id, address.decode(), port) for node_id, address, port in self.close_triples]
|
|
||||||
|
|
||||||
|
|
||||||
class IterativeFinder(AsyncIterator):
|
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop,
|
|
||||||
protocol: 'KademliaProtocol', key: bytes,
|
|
||||||
max_results: typing.Optional[int] = constants.K,
|
|
||||||
shortlist: typing.Optional[typing.List['KademliaPeer']] = None):
|
|
||||||
if len(key) != constants.HASH_LENGTH:
|
|
||||||
raise ValueError("invalid key length: %i" % len(key))
|
|
||||||
self.loop = loop
|
|
||||||
self.peer_manager = protocol.peer_manager
|
|
||||||
self.protocol = protocol
|
|
||||||
|
|
||||||
self.key = key
|
|
||||||
self.max_results = max(constants.K, max_results)
|
|
||||||
|
|
||||||
self.active: typing.Dict['KademliaPeer', int] = OrderedDict() # peer: distance, sorted
|
|
||||||
self.contacted: typing.Set['KademliaPeer'] = set()
|
|
||||||
self.distance = Distance(key)
|
|
||||||
|
|
||||||
self.iteration_queue = asyncio.Queue()
|
|
||||||
|
|
||||||
self.running_probes: typing.Dict['KademliaPeer', asyncio.Task] = {}
|
|
||||||
self.iteration_count = 0
|
|
||||||
self.running = False
|
|
||||||
self.tasks: typing.List[asyncio.Task] = []
|
|
||||||
for peer in shortlist:
|
|
||||||
if peer.node_id:
|
|
||||||
self._add_active(peer, force=True)
|
|
||||||
else:
|
|
||||||
# seed nodes
|
|
||||||
self._schedule_probe(peer)
|
|
||||||
|
|
||||||
async def send_probe(self, peer: 'KademliaPeer') -> FindResponse:
|
|
||||||
"""
|
|
||||||
Send the rpc request to the peer and return an object with the FindResponse interface
|
|
||||||
"""
|
|
||||||
raise NotImplementedError()
|
|
||||||
|
|
||||||
def search_exhausted(self):
|
|
||||||
"""
|
|
||||||
This method ends the iterator due no more peers to contact.
|
|
||||||
Override to provide last time results.
|
|
||||||
"""
|
|
||||||
self.iteration_queue.put_nowait(None)
|
|
||||||
|
|
||||||
def check_result_ready(self, response: FindResponse):
|
|
||||||
"""
|
|
||||||
Called after adding peers from an rpc result to the shortlist.
|
|
||||||
This method is responsible for putting a result for the generator into the Queue
|
|
||||||
"""
|
|
||||||
raise NotImplementedError()
|
|
||||||
|
|
||||||
def get_initial_result(self) -> typing.List['KademliaPeer']: #pylint: disable=no-self-use
|
|
||||||
"""
|
|
||||||
Get an initial or cached result to be put into the Queue. Used for findValue requests where the blob
|
|
||||||
has peers in the local data store of blobs announced to us
|
|
||||||
"""
|
|
||||||
return []
|
|
||||||
|
|
||||||
def _add_active(self, peer, force=False):
|
|
||||||
if not force and self.peer_manager.peer_is_good(peer) is False:
|
|
||||||
return
|
|
||||||
if peer in self.contacted:
|
|
||||||
return
|
|
||||||
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 = OrderedDict(sorted(self.active.items(), key=lambda item: item[1]))
|
|
||||||
|
|
||||||
async def _handle_probe_result(self, peer: 'KademliaPeer', response: FindResponse):
|
|
||||||
self._add_active(peer)
|
|
||||||
for new_peer in response.get_close_kademlia_peers(peer):
|
|
||||||
self._add_active(new_peer)
|
|
||||||
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'):
|
|
||||||
try:
|
|
||||||
response = await self.send_probe(peer)
|
|
||||||
except asyncio.TimeoutError:
|
|
||||||
self._reset_closest(peer)
|
|
||||||
return
|
|
||||||
except asyncio.CancelledError:
|
|
||||||
log.debug("%s[%x] cancelled probe",
|
|
||||||
type(self).__name__, id(self))
|
|
||||||
raise
|
|
||||||
except ValueError as err:
|
|
||||||
log.warning(str(err))
|
|
||||||
self._reset_closest(peer)
|
|
||||||
return
|
|
||||||
except TransportNotConnected:
|
|
||||||
await self._aclose(reason="not connected")
|
|
||||||
return
|
|
||||||
except RemoteException:
|
|
||||||
self._reset_closest(peer)
|
|
||||||
return
|
|
||||||
return await self._handle_probe_result(peer, response)
|
|
||||||
|
|
||||||
def _search_round(self):
|
|
||||||
"""
|
|
||||||
Send up to constants.alpha (5) probes to closest active peers
|
|
||||||
"""
|
|
||||||
|
|
||||||
added = 0
|
|
||||||
for index, peer in enumerate(self.active.keys()):
|
|
||||||
if index == 0:
|
|
||||||
log.debug("%s[%x] closest to probe: %s",
|
|
||||||
type(self).__name__, id(self),
|
|
||||||
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
|
|
||||||
origin_address = (peer.address, peer.udp_port)
|
|
||||||
if peer.node_id == self.protocol.node_id:
|
|
||||||
continue
|
|
||||||
if origin_address == (self.protocol.external_ip, self.protocol.udp_port):
|
|
||||||
continue
|
|
||||||
self._schedule_probe(peer)
|
|
||||||
added += 1
|
|
||||||
log.debug("%s[%x] running %d probes for key %s",
|
|
||||||
type(self).__name__, id(self),
|
|
||||||
len(self.running_probes), self.key.hex()[:8])
|
|
||||||
if not added and not self.running_probes:
|
|
||||||
log.debug("%s[%x] search for %s exhausted",
|
|
||||||
type(self).__name__, id(self),
|
|
||||||
self.key.hex()[:8])
|
|
||||||
self.search_exhausted()
|
|
||||||
|
|
||||||
def _schedule_probe(self, peer: 'KademliaPeer'):
|
|
||||||
self.contacted.add(peer)
|
|
||||||
|
|
||||||
t = self.loop.create_task(self._send_probe(peer))
|
|
||||||
|
|
||||||
def callback(_):
|
|
||||||
self.running_probes.pop(peer, None)
|
|
||||||
if self.running:
|
|
||||||
self._search_round()
|
|
||||||
|
|
||||||
t.add_done_callback(callback)
|
|
||||||
self.running_probes[peer] = t
|
|
||||||
|
|
||||||
def _log_state(self, reason="?"):
|
|
||||||
log.debug("%s[%x] [%s] %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())
|
|
||||||
|
|
||||||
def __aiter__(self):
|
|
||||||
if self.running:
|
|
||||||
raise Exception("already running")
|
|
||||||
self.running = True
|
|
||||||
self.loop.call_soon(self._search_round)
|
|
||||||
return self
|
|
||||||
|
|
||||||
async def __anext__(self) -> typing.List['KademliaPeer']:
|
|
||||||
try:
|
|
||||||
if self.iteration_count == 0:
|
|
||||||
result = self.get_initial_result() or await self.iteration_queue.get()
|
|
||||||
else:
|
|
||||||
result = await self.iteration_queue.get()
|
|
||||||
if not result:
|
|
||||||
raise StopAsyncIteration
|
|
||||||
self.iteration_count += 1
|
|
||||||
return result
|
|
||||||
except asyncio.CancelledError:
|
|
||||||
await self._aclose(reason="cancelled")
|
|
||||||
raise
|
|
||||||
except StopAsyncIteration:
|
|
||||||
await self._aclose(reason="no more results")
|
|
||||||
raise
|
|
||||||
|
|
||||||
async def _aclose(self, reason="?"):
|
|
||||||
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.iteration_queue.put_nowait(None)
|
|
||||||
for task in chain(self.tasks, self.running_probes.values()):
|
|
||||||
task.cancel()
|
|
||||||
self.tasks.clear()
|
|
||||||
self.running_probes.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):
|
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop,
|
|
||||||
protocol: 'KademliaProtocol', key: bytes,
|
|
||||||
max_results: typing.Optional[int] = constants.K,
|
|
||||||
shortlist: typing.Optional[typing.List['KademliaPeer']] = None):
|
|
||||||
super().__init__(loop, protocol, key, max_results, shortlist)
|
|
||||||
self.yielded_peers: typing.Set['KademliaPeer'] = set()
|
|
||||||
|
|
||||||
async def send_probe(self, peer: 'KademliaPeer') -> FindNodeResponse:
|
|
||||||
log.debug("probe %s:%d (%s) for NODE %s",
|
|
||||||
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)
|
|
||||||
return FindNodeResponse(self.key, response)
|
|
||||||
|
|
||||||
def search_exhausted(self):
|
|
||||||
self.put_result(self.active.keys(), finish=True)
|
|
||||||
|
|
||||||
def put_result(self, from_iter: typing.Iterable['KademliaPeer'], finish=False):
|
|
||||||
not_yet_yielded = [
|
|
||||||
peer for peer in from_iter
|
|
||||||
if peer not in self.yielded_peers
|
|
||||||
and peer.node_id != self.protocol.node_id
|
|
||||||
and self.peer_manager.peer_is_good(peer) is True # return only peers who answered
|
|
||||||
]
|
|
||||||
not_yet_yielded.sort(key=lambda peer: self.distance(peer.node_id))
|
|
||||||
to_yield = not_yet_yielded[:max(constants.K, self.max_results)]
|
|
||||||
if to_yield:
|
|
||||||
self.yielded_peers.update(to_yield)
|
|
||||||
self.iteration_queue.put_nowait(to_yield)
|
|
||||||
if finish:
|
|
||||||
self.iteration_queue.put_nowait(None)
|
|
||||||
|
|
||||||
def check_result_ready(self, response: FindNodeResponse):
|
|
||||||
found = response.found and self.key != self.protocol.node_id
|
|
||||||
|
|
||||||
if found:
|
|
||||||
log.debug("found")
|
|
||||||
return self.put_result(self.active.keys(), finish=True)
|
|
||||||
|
|
||||||
|
|
||||||
class IterativeValueFinder(IterativeFinder):
|
|
||||||
def __init__(self, loop: asyncio.AbstractEventLoop,
|
|
||||||
protocol: 'KademliaProtocol', key: bytes,
|
|
||||||
max_results: typing.Optional[int] = constants.K,
|
|
||||||
shortlist: typing.Optional[typing.List['KademliaPeer']] = None):
|
|
||||||
super().__init__(loop, protocol, key, max_results, shortlist)
|
|
||||||
self.blob_peers: typing.Set['KademliaPeer'] = set()
|
|
||||||
# this tracks the index of the most recent page we requested from each peer
|
|
||||||
self.peer_pages: typing.DefaultDict['KademliaPeer', int] = defaultdict(int)
|
|
||||||
# this tracks the set of blob peers returned by each peer
|
|
||||||
self.discovered_peers: typing.Dict['KademliaPeer', typing.Set['KademliaPeer']] = defaultdict(set)
|
|
||||||
|
|
||||||
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]
|
|
||||||
response = await self.protocol.get_rpc_peer(peer).find_value(self.key, page=page)
|
|
||||||
parsed = FindValueResponse(self.key, response)
|
|
||||||
if not parsed.found:
|
|
||||||
return parsed
|
|
||||||
already_known = len(self.discovered_peers[peer])
|
|
||||||
decoded_peers = set()
|
|
||||||
for compact_addr in parsed.found_compact_addresses:
|
|
||||||
try:
|
|
||||||
decoded_peers.add(decode_tcp_peer_from_compact_address(compact_addr))
|
|
||||||
except ValueError:
|
|
||||||
log.warning("misbehaving peer %s:%i returned invalid peer for blob",
|
|
||||||
peer.address, peer.udp_port)
|
|
||||||
self.peer_manager.report_failure(peer.address, peer.udp_port)
|
|
||||||
parsed.found_compact_addresses.clear()
|
|
||||||
return parsed
|
|
||||||
self.discovered_peers[peer].update(decoded_peers)
|
|
||||||
log.debug("probed %s:%i page %i, %i known", peer.address, peer.udp_port, page,
|
|
||||||
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)
|
|
||||||
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
|
|
||||||
self.peer_pages[peer] += 1
|
|
||||||
if peer in self.contacted:
|
|
||||||
# the peer must be removed from self.contacted so that it will be probed for the next page
|
|
||||||
self.contacted.remove(peer)
|
|
||||||
return parsed
|
|
||||||
|
|
||||||
def check_result_ready(self, response: FindValueResponse):
|
|
||||||
if response.found:
|
|
||||||
blob_peers = [decode_tcp_peer_from_compact_address(compact_addr)
|
|
||||||
for compact_addr in response.found_compact_addresses]
|
|
||||||
to_yield = []
|
|
||||||
for blob_peer in blob_peers:
|
|
||||||
if blob_peer not in self.blob_peers:
|
|
||||||
self.blob_peers.add(blob_peer)
|
|
||||||
to_yield.append(blob_peer)
|
|
||||||
if to_yield:
|
|
||||||
self.iteration_queue.put_nowait(to_yield)
|
|
||||||
|
|
||||||
def get_initial_result(self) -> typing.List['KademliaPeer']:
|
|
||||||
if self.protocol.data_store.has_peers_for_blob(self.key):
|
|
||||||
return self.protocol.data_store.get_peers_for_blob(self.key)
|
|
||||||
return []
|
|
|
@ -1,5 +0,0 @@
|
||||||
generate:
|
|
||||||
python generate.py generate > __init__.py
|
|
||||||
|
|
||||||
analyze:
|
|
||||||
python generate.py analyze
|
|
|
@ -1,95 +0,0 @@
|
||||||
# Exceptions
|
|
||||||
|
|
||||||
Exceptions in LBRY are defined and generated from the Markdown table at the end of this README.
|
|
||||||
|
|
||||||
## Guidelines
|
|
||||||
|
|
||||||
When possible, use [built-in Python exceptions](https://docs.python.org/3/library/exceptions.html) or `aiohttp` [general client](https://docs.aiohttp.org/en/latest/client_reference.html#client-exceptions) / [HTTP](https://docs.aiohttp.org/en/latest/web_exceptions.html) exceptions, unless:
|
|
||||||
1. You want to provide a better error message (extend the closest built-in/`aiohttp` exception in this case).
|
|
||||||
2. You need to represent a new situation.
|
|
||||||
|
|
||||||
When defining your own exceptions, consider:
|
|
||||||
1. Extending a built-in Python or `aiohttp` exception.
|
|
||||||
2. Using contextual variables in the error message.
|
|
||||||
|
|
||||||
## Table Column Definitions
|
|
||||||
|
|
||||||
Column | Meaning
|
|
||||||
---|---
|
|
||||||
Code | Codes are used only to define the hierarchy of exceptions and do not end up in the generated output, it is okay to re-number things as necessary at anytime to achieve the desired hierarchy.
|
|
||||||
Name | Becomes the class name of the exception with "Error" appended to the end. Changing names of existing exceptions makes the API backwards incompatible. When extending other exceptions you must specify the full class name, manually adding "Error" as necessary (if extending another SDK exception).
|
|
||||||
Message | User friendly error message explaining the exceptional event. Supports Python formatted strings: any variables used in the string will be generated as arguments in the `__init__` method. Use `--` to provide a doc string after the error message to be added to the class definition.
|
|
||||||
|
|
||||||
## Exceptions Table
|
|
||||||
|
|
||||||
Code | Name | Message
|
|
||||||
---:|---|---
|
|
||||||
**1xx** | UserInput | User input errors.
|
|
||||||
**10x** | Command | Errors preparing to execute commands.
|
|
||||||
101 | CommandDoesNotExist | Command '{command}' does not exist.
|
|
||||||
102 | CommandDeprecated | Command '{command}' is deprecated.
|
|
||||||
103 | CommandInvalidArgument | Invalid argument '{argument}' to command '{command}'.
|
|
||||||
104 | CommandTemporarilyUnavailable | Command '{command}' is temporarily unavailable. -- Such as waiting for required components to start.
|
|
||||||
105 | CommandPermanentlyUnavailable | Command '{command}' is permanently unavailable. -- such as when required component was intentionally configured not to start.
|
|
||||||
**11x** | InputValue(ValueError) | Invalid argument value provided to command.
|
|
||||||
111 | GenericInputValue | The value '{value}' for argument '{argument}' is not valid.
|
|
||||||
112 | InputValueIsNone | None or null is not valid value for argument '{argument}'.
|
|
||||||
113 | ConflictingInputValue | Only '{first_argument}' or '{second_argument}' is allowed, not both.
|
|
||||||
114 | InputStringIsBlank | {argument} cannot be blank.
|
|
||||||
115 | EmptyPublishedFile | Cannot publish empty file: {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.
|
|
||||||
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.
|
|
||||||
203 | ConfigParse | Failed to parse the configuration file '{path}'. -- Includes the syntax error / line number to help user fix it.
|
|
||||||
204 | ConfigMissing | Configuration file '{path}' is missing setting that has no default / fallback.
|
|
||||||
205 | ConfigInvalid | Configuration file '{path}' has setting with invalid value.
|
|
||||||
**3xx** | Network | **Networking**
|
|
||||||
301 | NoInternet | No internet connection.
|
|
||||||
302 | NoUPnPSupport | Router does not support UPnP.
|
|
||||||
**4xx** | Wallet | **Wallet Errors**
|
|
||||||
401 | TransactionRejected | Transaction rejected, unknown reason.
|
|
||||||
402 | TransactionFeeTooLow | Fee too low.
|
|
||||||
403 | TransactionInvalidSignature | Invalid signature.
|
|
||||||
404 | InsufficientFunds | Not enough funds to cover this transaction. -- determined by wallet prior to attempting to broadcast a tx; this is different for example from a TX being created and sent but then rejected by lbrycrd for unspendable utxos.
|
|
||||||
405 | ChannelKeyNotFound | Channel signing key not found.
|
|
||||||
406 | ChannelKeyInvalid | Channel signing key is out of date. -- For example, channel was updated but you don't have the updated key.
|
|
||||||
407 | DataDownload | Failed to download blob. *generic*
|
|
||||||
408 | PrivateKeyNotFound | Couldn't find private key for {key} '{value}'.
|
|
||||||
410 | Resolve | Failed to resolve '{url}'.
|
|
||||||
411 | ResolveTimeout | Failed to resolve '{url}' within the timeout.
|
|
||||||
411 | ResolveCensored | Resolve of '{url}' was censored by channel with claim id '{censor_id}'.
|
|
||||||
420 | KeyFeeAboveMaxAllowed | {message}
|
|
||||||
421 | InvalidPassword | Password is invalid.
|
|
||||||
422 | IncompatibleWalletServer | '{server}:{port}' has an incompatibly old version.
|
|
||||||
423 | TooManyClaimSearchParameters | {key} cant have more than {limit} items.
|
|
||||||
424 | AlreadyPurchased | You already have a purchase for claim_id '{claim_id_hex}'. Use --allow-duplicate-purchase flag to override.
|
|
||||||
431 | ServerPaymentInvalidAddress | Invalid address from wallet server: '{address}' - skipping payment round.
|
|
||||||
432 | ServerPaymentWalletLocked | Cannot spend funds with locked wallet, skipping payment round.
|
|
||||||
433 | ServerPaymentFeeAboveMaxAllowed | Daily server fee of {daily_fee} exceeds maximum configured of {max_fee} LBC.
|
|
||||||
434 | WalletNotLoaded | Wallet {wallet_id} is not loaded.
|
|
||||||
435 | WalletAlreadyLoaded | Wallet {wallet_path} is already loaded.
|
|
||||||
436 | WalletNotFound | Wallet not found at {wallet_path}.
|
|
||||||
437 | WalletAlreadyExists | Wallet {wallet_path} already exists, use `wallet_add` to load it.
|
|
||||||
**5xx** | Blob | **Blobs**
|
|
||||||
500 | BlobNotFound | Blob not found.
|
|
||||||
501 | BlobPermissionDenied | Permission denied to read blob.
|
|
||||||
502 | BlobTooBig | Blob is too big.
|
|
||||||
503 | BlobEmpty | Blob is empty.
|
|
||||||
510 | BlobFailedDecryption | Failed to decrypt blob.
|
|
||||||
511 | CorruptBlob | Blobs is corrupted.
|
|
||||||
520 | BlobFailedEncryption | Failed to encrypt blob.
|
|
||||||
531 | DownloadCancelled | Download was canceled.
|
|
||||||
532 | DownloadSDTimeout | Failed to download sd blob {download} within timeout.
|
|
||||||
533 | DownloadDataTimeout | Failed to download data blobs for sd hash {download} within timeout.
|
|
||||||
534 | InvalidStreamDescriptor | {message}
|
|
||||||
535 | InvalidData | {message}
|
|
||||||
536 | InvalidBlobHash | {message}
|
|
||||||
**6xx** | Component | **Components**
|
|
||||||
601 | ComponentStartConditionNotMet | Unresolved dependencies for: {components}
|
|
||||||
602 | ComponentsNotStarted | {message}
|
|
||||||
**7xx** | CurrencyExchange | **Currency Exchange**
|
|
||||||
701 | InvalidExchangeRateResponse | Failed to get exchange rate from {source}: {reason}
|
|
||||||
702 | CurrencyConversion | {message}
|
|
||||||
703 | InvalidCurrency | Invalid currency: {currency} is not a supported currency.
|
|
|
@ -1,494 +0,0 @@
|
||||||
from .base import BaseError, claim_id
|
|
||||||
|
|
||||||
|
|
||||||
class UserInputError(BaseError):
|
|
||||||
"""
|
|
||||||
User input errors.
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
class CommandError(UserInputError):
|
|
||||||
"""
|
|
||||||
Errors preparing to execute commands.
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
class CommandDoesNotExistError(CommandError):
|
|
||||||
|
|
||||||
def __init__(self, command):
|
|
||||||
self.command = command
|
|
||||||
super().__init__(f"Command '{command}' does not exist.")
|
|
||||||
|
|
||||||
|
|
||||||
class CommandDeprecatedError(CommandError):
|
|
||||||
|
|
||||||
def __init__(self, command):
|
|
||||||
self.command = command
|
|
||||||
super().__init__(f"Command '{command}' is deprecated.")
|
|
||||||
|
|
||||||
|
|
||||||
class CommandInvalidArgumentError(CommandError):
|
|
||||||
|
|
||||||
def __init__(self, argument, command):
|
|
||||||
self.argument = argument
|
|
||||||
self.command = command
|
|
||||||
super().__init__(f"Invalid argument '{argument}' to command '{command}'.")
|
|
||||||
|
|
||||||
|
|
||||||
class CommandTemporarilyUnavailableError(CommandError):
|
|
||||||
"""
|
|
||||||
Such as waiting for required components to start.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, command):
|
|
||||||
self.command = command
|
|
||||||
super().__init__(f"Command '{command}' is temporarily unavailable.")
|
|
||||||
|
|
||||||
|
|
||||||
class CommandPermanentlyUnavailableError(CommandError):
|
|
||||||
"""
|
|
||||||
such as when required component was intentionally configured not to start.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, command):
|
|
||||||
self.command = command
|
|
||||||
super().__init__(f"Command '{command}' is permanently unavailable.")
|
|
||||||
|
|
||||||
|
|
||||||
class InputValueError(UserInputError, ValueError):
|
|
||||||
"""
|
|
||||||
Invalid argument value provided to command.
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
class GenericInputValueError(InputValueError):
|
|
||||||
|
|
||||||
def __init__(self, value, argument):
|
|
||||||
self.value = value
|
|
||||||
self.argument = argument
|
|
||||||
super().__init__(f"The value '{value}' for argument '{argument}' is not valid.")
|
|
||||||
|
|
||||||
|
|
||||||
class InputValueIsNoneError(InputValueError):
|
|
||||||
|
|
||||||
def __init__(self, argument):
|
|
||||||
self.argument = argument
|
|
||||||
super().__init__(f"None or null is not valid value for argument '{argument}'.")
|
|
||||||
|
|
||||||
|
|
||||||
class ConflictingInputValueError(InputValueError):
|
|
||||||
|
|
||||||
def __init__(self, first_argument, second_argument):
|
|
||||||
self.first_argument = first_argument
|
|
||||||
self.second_argument = second_argument
|
|
||||||
super().__init__(f"Only '{first_argument}' or '{second_argument}' is allowed, not both.")
|
|
||||||
|
|
||||||
|
|
||||||
class InputStringIsBlankError(InputValueError):
|
|
||||||
|
|
||||||
def __init__(self, argument):
|
|
||||||
self.argument = argument
|
|
||||||
super().__init__(f"{argument} cannot be blank.")
|
|
||||||
|
|
||||||
|
|
||||||
class EmptyPublishedFileError(InputValueError):
|
|
||||||
|
|
||||||
def __init__(self, file_path):
|
|
||||||
self.file_path = file_path
|
|
||||||
super().__init__(f"Cannot publish empty file: {file_path}")
|
|
||||||
|
|
||||||
|
|
||||||
class MissingPublishedFileError(InputValueError):
|
|
||||||
|
|
||||||
def __init__(self, file_path):
|
|
||||||
self.file_path = 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):
|
|
||||||
"""
|
|
||||||
Configuration errors.
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
class ConfigWriteError(ConfigurationError):
|
|
||||||
"""
|
|
||||||
When writing the default config fails on startup, such as due to permission issues.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, path):
|
|
||||||
self.path = path
|
|
||||||
super().__init__(f"Cannot write configuration file '{path}'.")
|
|
||||||
|
|
||||||
|
|
||||||
class ConfigReadError(ConfigurationError):
|
|
||||||
"""
|
|
||||||
Can't open the config file user provided via command line args.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, path):
|
|
||||||
self.path = path
|
|
||||||
super().__init__(f"Cannot find provided configuration file '{path}'.")
|
|
||||||
|
|
||||||
|
|
||||||
class ConfigParseError(ConfigurationError):
|
|
||||||
"""
|
|
||||||
Includes the syntax error / line number to help user fix it.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, path):
|
|
||||||
self.path = path
|
|
||||||
super().__init__(f"Failed to parse the configuration file '{path}'.")
|
|
||||||
|
|
||||||
|
|
||||||
class ConfigMissingError(ConfigurationError):
|
|
||||||
|
|
||||||
def __init__(self, path):
|
|
||||||
self.path = path
|
|
||||||
super().__init__(f"Configuration file '{path}' is missing setting that has no default / fallback.")
|
|
||||||
|
|
||||||
|
|
||||||
class ConfigInvalidError(ConfigurationError):
|
|
||||||
|
|
||||||
def __init__(self, path):
|
|
||||||
self.path = path
|
|
||||||
super().__init__(f"Configuration file '{path}' has setting with invalid value.")
|
|
||||||
|
|
||||||
|
|
||||||
class NetworkError(BaseError):
|
|
||||||
"""
|
|
||||||
**Networking**
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
class NoInternetError(NetworkError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("No internet connection.")
|
|
||||||
|
|
||||||
|
|
||||||
class NoUPnPSupportError(NetworkError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Router does not support UPnP.")
|
|
||||||
|
|
||||||
|
|
||||||
class WalletError(BaseError):
|
|
||||||
"""
|
|
||||||
**Wallet Errors**
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
class TransactionRejectedError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Transaction rejected, unknown reason.")
|
|
||||||
|
|
||||||
|
|
||||||
class TransactionFeeTooLowError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Fee too low.")
|
|
||||||
|
|
||||||
|
|
||||||
class TransactionInvalidSignatureError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Invalid signature.")
|
|
||||||
|
|
||||||
|
|
||||||
class InsufficientFundsError(WalletError):
|
|
||||||
"""
|
|
||||||
determined by wallet prior to attempting to broadcast a tx; this is different for example from a TX
|
|
||||||
being created and sent but then rejected by lbrycrd for unspendable utxos.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Not enough funds to cover this transaction.")
|
|
||||||
|
|
||||||
|
|
||||||
class ChannelKeyNotFoundError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Channel signing key not found.")
|
|
||||||
|
|
||||||
|
|
||||||
class ChannelKeyInvalidError(WalletError):
|
|
||||||
"""
|
|
||||||
For example, channel was updated but you don't have the updated key.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Channel signing key is out of date.")
|
|
||||||
|
|
||||||
|
|
||||||
class DataDownloadError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Failed to download blob. *generic*")
|
|
||||||
|
|
||||||
|
|
||||||
class PrivateKeyNotFoundError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self, key, value):
|
|
||||||
self.key = key
|
|
||||||
self.value = value
|
|
||||||
super().__init__(f"Couldn't find private key for {key} '{value}'.")
|
|
||||||
|
|
||||||
|
|
||||||
class ResolveError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self, url):
|
|
||||||
self.url = url
|
|
||||||
super().__init__(f"Failed to resolve '{url}'.")
|
|
||||||
|
|
||||||
|
|
||||||
class ResolveTimeoutError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self, url):
|
|
||||||
self.url = url
|
|
||||||
super().__init__(f"Failed to resolve '{url}' within the timeout.")
|
|
||||||
|
|
||||||
|
|
||||||
class ResolveCensoredError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self, url, censor_id, censor_row):
|
|
||||||
self.url = url
|
|
||||||
self.censor_id = censor_id
|
|
||||||
self.censor_row = censor_row
|
|
||||||
super().__init__(f"Resolve of '{url}' was censored by channel with claim id '{censor_id}'.")
|
|
||||||
|
|
||||||
|
|
||||||
class KeyFeeAboveMaxAllowedError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self, message):
|
|
||||||
self.message = message
|
|
||||||
super().__init__(f"{message}")
|
|
||||||
|
|
||||||
|
|
||||||
class InvalidPasswordError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Password is invalid.")
|
|
||||||
|
|
||||||
|
|
||||||
class IncompatibleWalletServerError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self, server, port):
|
|
||||||
self.server = server
|
|
||||||
self.port = port
|
|
||||||
super().__init__(f"'{server}:{port}' has an incompatibly old version.")
|
|
||||||
|
|
||||||
|
|
||||||
class TooManyClaimSearchParametersError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self, key, limit):
|
|
||||||
self.key = key
|
|
||||||
self.limit = limit
|
|
||||||
super().__init__(f"{key} cant have more than {limit} items.")
|
|
||||||
|
|
||||||
|
|
||||||
class AlreadyPurchasedError(WalletError):
|
|
||||||
"""
|
|
||||||
allow-duplicate-purchase flag to override.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, claim_id_hex):
|
|
||||||
self.claim_id_hex = claim_id_hex
|
|
||||||
super().__init__(f"You already have a purchase for claim_id '{claim_id_hex}'. Use")
|
|
||||||
|
|
||||||
|
|
||||||
class ServerPaymentInvalidAddressError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self, address):
|
|
||||||
self.address = address
|
|
||||||
super().__init__(f"Invalid address from wallet server: '{address}' - skipping payment round.")
|
|
||||||
|
|
||||||
|
|
||||||
class ServerPaymentWalletLockedError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Cannot spend funds with locked wallet, skipping payment round.")
|
|
||||||
|
|
||||||
|
|
||||||
class ServerPaymentFeeAboveMaxAllowedError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self, daily_fee, max_fee):
|
|
||||||
self.daily_fee = daily_fee
|
|
||||||
self.max_fee = max_fee
|
|
||||||
super().__init__(f"Daily server fee of {daily_fee} exceeds maximum configured of {max_fee} LBC.")
|
|
||||||
|
|
||||||
|
|
||||||
class WalletNotLoadedError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self, wallet_id):
|
|
||||||
self.wallet_id = wallet_id
|
|
||||||
super().__init__(f"Wallet {wallet_id} is not loaded.")
|
|
||||||
|
|
||||||
|
|
||||||
class WalletAlreadyLoadedError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self, wallet_path):
|
|
||||||
self.wallet_path = wallet_path
|
|
||||||
super().__init__(f"Wallet {wallet_path} is already loaded.")
|
|
||||||
|
|
||||||
|
|
||||||
class WalletNotFoundError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self, wallet_path):
|
|
||||||
self.wallet_path = wallet_path
|
|
||||||
super().__init__(f"Wallet not found at {wallet_path}.")
|
|
||||||
|
|
||||||
|
|
||||||
class WalletAlreadyExistsError(WalletError):
|
|
||||||
|
|
||||||
def __init__(self, wallet_path):
|
|
||||||
self.wallet_path = wallet_path
|
|
||||||
super().__init__(f"Wallet {wallet_path} already exists, use `wallet_add` to load it.")
|
|
||||||
|
|
||||||
|
|
||||||
class BlobError(BaseError):
|
|
||||||
"""
|
|
||||||
**Blobs**
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
class BlobNotFoundError(BlobError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Blob not found.")
|
|
||||||
|
|
||||||
|
|
||||||
class BlobPermissionDeniedError(BlobError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Permission denied to read blob.")
|
|
||||||
|
|
||||||
|
|
||||||
class BlobTooBigError(BlobError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Blob is too big.")
|
|
||||||
|
|
||||||
|
|
||||||
class BlobEmptyError(BlobError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Blob is empty.")
|
|
||||||
|
|
||||||
|
|
||||||
class BlobFailedDecryptionError(BlobError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Failed to decrypt blob.")
|
|
||||||
|
|
||||||
|
|
||||||
class CorruptBlobError(BlobError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Blobs is corrupted.")
|
|
||||||
|
|
||||||
|
|
||||||
class BlobFailedEncryptionError(BlobError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Failed to encrypt blob.")
|
|
||||||
|
|
||||||
|
|
||||||
class DownloadCancelledError(BlobError):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
super().__init__("Download was canceled.")
|
|
||||||
|
|
||||||
|
|
||||||
class DownloadSDTimeoutError(BlobError):
|
|
||||||
|
|
||||||
def __init__(self, download):
|
|
||||||
self.download = download
|
|
||||||
super().__init__(f"Failed to download sd blob {download} within timeout.")
|
|
||||||
|
|
||||||
|
|
||||||
class DownloadDataTimeoutError(BlobError):
|
|
||||||
|
|
||||||
def __init__(self, download):
|
|
||||||
self.download = download
|
|
||||||
super().__init__(f"Failed to download data blobs for sd hash {download} within timeout.")
|
|
||||||
|
|
||||||
|
|
||||||
class InvalidStreamDescriptorError(BlobError):
|
|
||||||
|
|
||||||
def __init__(self, message):
|
|
||||||
self.message = message
|
|
||||||
super().__init__(f"{message}")
|
|
||||||
|
|
||||||
|
|
||||||
class InvalidDataError(BlobError):
|
|
||||||
|
|
||||||
def __init__(self, message):
|
|
||||||
self.message = message
|
|
||||||
super().__init__(f"{message}")
|
|
||||||
|
|
||||||
|
|
||||||
class InvalidBlobHashError(BlobError):
|
|
||||||
|
|
||||||
def __init__(self, message):
|
|
||||||
self.message = message
|
|
||||||
super().__init__(f"{message}")
|
|
||||||
|
|
||||||
|
|
||||||
class ComponentError(BaseError):
|
|
||||||
"""
|
|
||||||
**Components**
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
class ComponentStartConditionNotMetError(ComponentError):
|
|
||||||
|
|
||||||
def __init__(self, components):
|
|
||||||
self.components = components
|
|
||||||
super().__init__(f"Unresolved dependencies for: {components}")
|
|
||||||
|
|
||||||
|
|
||||||
class ComponentsNotStartedError(ComponentError):
|
|
||||||
|
|
||||||
def __init__(self, message):
|
|
||||||
self.message = message
|
|
||||||
super().__init__(f"{message}")
|
|
||||||
|
|
||||||
|
|
||||||
class CurrencyExchangeError(BaseError):
|
|
||||||
"""
|
|
||||||
**Currency Exchange**
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
class InvalidExchangeRateResponseError(CurrencyExchangeError):
|
|
||||||
|
|
||||||
def __init__(self, source, reason):
|
|
||||||
self.source = source
|
|
||||||
self.reason = reason
|
|
||||||
super().__init__(f"Failed to get exchange rate from {source}: {reason}")
|
|
||||||
|
|
||||||
|
|
||||||
class CurrencyConversionError(CurrencyExchangeError):
|
|
||||||
|
|
||||||
def __init__(self, message):
|
|
||||||
self.message = message
|
|
||||||
super().__init__(f"{message}")
|
|
||||||
|
|
||||||
|
|
||||||
class InvalidCurrencyError(CurrencyExchangeError):
|
|
||||||
|
|
||||||
def __init__(self, currency):
|
|
||||||
self.currency = currency
|
|
||||||
super().__init__(f"Invalid currency: {currency} is not a supported currency.")
|
|
|
@ -1,9 +0,0 @@
|
||||||
from binascii import hexlify
|
|
||||||
|
|
||||||
|
|
||||||
def claim_id(claim_hash):
|
|
||||||
return hexlify(claim_hash[::-1]).decode()
|
|
||||||
|
|
||||||
|
|
||||||
class BaseError(Exception):
|
|
||||||
pass
|
|
|
@ -1,167 +0,0 @@
|
||||||
import re
|
|
||||||
import sys
|
|
||||||
import argparse
|
|
||||||
from pathlib import Path
|
|
||||||
from textwrap import fill, indent
|
|
||||||
|
|
||||||
|
|
||||||
INDENT = ' ' * 4
|
|
||||||
|
|
||||||
CLASS = """
|
|
||||||
|
|
||||||
class {name}({parents}):{doc}
|
|
||||||
"""
|
|
||||||
|
|
||||||
INIT = """
|
|
||||||
def __init__({args}):{fields}
|
|
||||||
super().__init__({format}"{message}")
|
|
||||||
"""
|
|
||||||
|
|
||||||
FUNCTIONS = ['claim_id']
|
|
||||||
|
|
||||||
|
|
||||||
class ErrorClass:
|
|
||||||
|
|
||||||
def __init__(self, hierarchy, name, message):
|
|
||||||
self.hierarchy = hierarchy.replace('**', '')
|
|
||||||
self.other_parents = []
|
|
||||||
if '(' in name:
|
|
||||||
assert ')' in name, f"Missing closing parenthesis in '{name}'."
|
|
||||||
self.other_parents = name[name.find('(')+1:name.find(')')].split(',')
|
|
||||||
name = name[:name.find('(')]
|
|
||||||
self.name = name
|
|
||||||
self.class_name = name+'Error'
|
|
||||||
self.message = message
|
|
||||||
self.comment = ""
|
|
||||||
if '--' in message:
|
|
||||||
self.message, self.comment = message.split('--')
|
|
||||||
self.message = self.message.strip()
|
|
||||||
self.comment = self.comment.strip()
|
|
||||||
|
|
||||||
@property
|
|
||||||
def is_leaf(self):
|
|
||||||
return 'x' not in self.hierarchy
|
|
||||||
|
|
||||||
@property
|
|
||||||
def code(self):
|
|
||||||
return self.hierarchy.replace('x', '')
|
|
||||||
|
|
||||||
@property
|
|
||||||
def parent_codes(self):
|
|
||||||
return self.hierarchy[0:2], self.hierarchy[0]
|
|
||||||
|
|
||||||
def get_arguments(self):
|
|
||||||
args = ['self']
|
|
||||||
for arg in re.findall('{([a-z0-1_()]+)}', self.message):
|
|
||||||
for func in FUNCTIONS:
|
|
||||||
if arg.startswith(f'{func}('):
|
|
||||||
arg = arg[len(f'{func}('):-1]
|
|
||||||
break
|
|
||||||
args.append(arg)
|
|
||||||
return args
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def get_fields(args):
|
|
||||||
if len(args) > 1:
|
|
||||||
return ''.join(f'\n{INDENT*2}self.{field} = {field}' for field in args[1:])
|
|
||||||
return ''
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def get_doc_string(doc):
|
|
||||||
if doc:
|
|
||||||
return f'\n{INDENT}"""\n{indent(fill(doc, 100), INDENT)}\n{INDENT}"""'
|
|
||||||
return ""
|
|
||||||
|
|
||||||
def render(self, out, parent):
|
|
||||||
if not parent:
|
|
||||||
parents = ['BaseError']
|
|
||||||
else:
|
|
||||||
parents = [parent.class_name]
|
|
||||||
parents += self.other_parents
|
|
||||||
args = self.get_arguments()
|
|
||||||
if self.is_leaf:
|
|
||||||
out.write((CLASS + INIT).format(
|
|
||||||
name=self.class_name, parents=', '.join(parents),
|
|
||||||
args=', '.join(args), fields=self.get_fields(args),
|
|
||||||
message=self.message, doc=self.get_doc_string(self.comment), format='f' if len(args) > 1 else ''
|
|
||||||
))
|
|
||||||
else:
|
|
||||||
out.write(CLASS.format(
|
|
||||||
name=self.class_name, parents=', '.join(parents),
|
|
||||||
doc=self.get_doc_string(self.comment or self.message)
|
|
||||||
))
|
|
||||||
|
|
||||||
|
|
||||||
def get_errors():
|
|
||||||
with open('README.md', 'r') as readme:
|
|
||||||
lines = iter(readme.readlines())
|
|
||||||
for line in lines:
|
|
||||||
if line.startswith('## Exceptions Table'):
|
|
||||||
break
|
|
||||||
for line in lines:
|
|
||||||
if line.startswith('---:|'):
|
|
||||||
break
|
|
||||||
for line in lines:
|
|
||||||
if not line:
|
|
||||||
break
|
|
||||||
yield ErrorClass(*[c.strip() for c in line.split('|')])
|
|
||||||
|
|
||||||
|
|
||||||
def find_parent(stack, child):
|
|
||||||
for parent_code in child.parent_codes:
|
|
||||||
parent = stack.get(parent_code)
|
|
||||||
if parent:
|
|
||||||
return parent
|
|
||||||
|
|
||||||
|
|
||||||
def generate(out):
|
|
||||||
out.write(f"from .base import BaseError, {', '.join(FUNCTIONS)}\n")
|
|
||||||
stack = {}
|
|
||||||
for error in get_errors():
|
|
||||||
error.render(out, find_parent(stack, error))
|
|
||||||
if not error.is_leaf:
|
|
||||||
assert error.code not in stack, f"Duplicate code: {error.code}"
|
|
||||||
stack[error.code] = error
|
|
||||||
|
|
||||||
|
|
||||||
def analyze():
|
|
||||||
errors = {e.class_name: [] for e in get_errors() if e.is_leaf}
|
|
||||||
here = Path(__file__).absolute().parents[0]
|
|
||||||
module = here.parent
|
|
||||||
for file_path in module.glob('**/*.py'):
|
|
||||||
if here in file_path.parents:
|
|
||||||
continue
|
|
||||||
with open(file_path) as src_file:
|
|
||||||
src = src_file.read()
|
|
||||||
for error in errors.keys():
|
|
||||||
found = src.count(error)
|
|
||||||
if found > 0:
|
|
||||||
errors[error].append((file_path, found))
|
|
||||||
|
|
||||||
print('Unused Errors:\n')
|
|
||||||
for error, used in errors.items():
|
|
||||||
if used:
|
|
||||||
print(f' - {error}')
|
|
||||||
for use in used:
|
|
||||||
print(f' {use[0].relative_to(module.parent)} {use[1]}')
|
|
||||||
print('')
|
|
||||||
|
|
||||||
print('')
|
|
||||||
print('Unused Errors:')
|
|
||||||
for error, used in errors.items():
|
|
||||||
if not used:
|
|
||||||
print(f' - {error}')
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
parser = argparse.ArgumentParser()
|
|
||||||
parser.add_argument("action", choices=['generate', 'analyze'])
|
|
||||||
args = parser.parse_args()
|
|
||||||
if args.action == "analyze":
|
|
||||||
analyze()
|
|
||||||
elif args.action == "generate":
|
|
||||||
generate(sys.stdout)
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
main()
|
|
|
@ -1,6 +0,0 @@
|
||||||
from lbry.extras.cli import execute_command
|
|
||||||
from lbry.conf import Config
|
|
||||||
|
|
||||||
|
|
||||||
def daemon_rpc(conf: Config, method: str, **kwargs):
|
|
||||||
return execute_command(conf, method, kwargs, callback=lambda data: data)
|
|
|
@ -1,750 +0,0 @@
|
||||||
import math
|
|
||||||
import os
|
|
||||||
import asyncio
|
|
||||||
import logging
|
|
||||||
import binascii
|
|
||||||
import typing
|
|
||||||
|
|
||||||
import base58
|
|
||||||
|
|
||||||
from aioupnp import __version__ as aioupnp_version
|
|
||||||
from aioupnp.upnp import UPnP
|
|
||||||
from aioupnp.fault import UPnPError
|
|
||||||
|
|
||||||
from lbry import utils
|
|
||||||
from lbry.dht.node import Node
|
|
||||||
from lbry.dht.peer import is_valid_public_ipv4
|
|
||||||
from lbry.dht.blob_announcer import BlobAnnouncer
|
|
||||||
from lbry.blob.blob_manager import BlobManager
|
|
||||||
from lbry.blob.disk_space_manager import DiskSpaceManager
|
|
||||||
from lbry.blob_exchange.server import BlobServer
|
|
||||||
from lbry.stream.background_downloader import BackgroundDownloader
|
|
||||||
from lbry.stream.stream_manager import StreamManager
|
|
||||||
from lbry.file.file_manager import FileManager
|
|
||||||
from lbry.extras.daemon.component import Component
|
|
||||||
from lbry.extras.daemon.exchange_rate_manager import ExchangeRateManager
|
|
||||||
from lbry.extras.daemon.storage import SQLiteStorage
|
|
||||||
from lbry.torrent.torrent_manager import TorrentManager
|
|
||||||
from lbry.wallet import WalletManager
|
|
||||||
from lbry.wallet.usage_payment import WalletServerPayer
|
|
||||||
from lbry.torrent.tracker import TrackerClient
|
|
||||||
from lbry.torrent.session import TorrentSession
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
# settings must be initialized before this file is imported
|
|
||||||
|
|
||||||
DATABASE_COMPONENT = "database"
|
|
||||||
BLOB_COMPONENT = "blob_manager"
|
|
||||||
WALLET_COMPONENT = "wallet"
|
|
||||||
WALLET_SERVER_PAYMENTS_COMPONENT = "wallet_server_payments"
|
|
||||||
DHT_COMPONENT = "dht"
|
|
||||||
HASH_ANNOUNCER_COMPONENT = "hash_announcer"
|
|
||||||
FILE_MANAGER_COMPONENT = "file_manager"
|
|
||||||
DISK_SPACE_COMPONENT = "disk_space"
|
|
||||||
BACKGROUND_DOWNLOADER_COMPONENT = "background_downloader"
|
|
||||||
PEER_PROTOCOL_SERVER_COMPONENT = "peer_protocol_server"
|
|
||||||
UPNP_COMPONENT = "upnp"
|
|
||||||
EXCHANGE_RATE_MANAGER_COMPONENT = "exchange_rate_manager"
|
|
||||||
TRACKER_ANNOUNCER_COMPONENT = "tracker_announcer_component"
|
|
||||||
LIBTORRENT_COMPONENT = "libtorrent_component"
|
|
||||||
|
|
||||||
|
|
||||||
class DatabaseComponent(Component):
|
|
||||||
component_name = DATABASE_COMPONENT
|
|
||||||
|
|
||||||
def __init__(self, component_manager):
|
|
||||||
super().__init__(component_manager)
|
|
||||||
self.storage = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def component(self):
|
|
||||||
return self.storage
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def get_current_db_revision():
|
|
||||||
return 15
|
|
||||||
|
|
||||||
@property
|
|
||||||
def revision_filename(self):
|
|
||||||
return os.path.join(self.conf.data_dir, 'db_revision')
|
|
||||||
|
|
||||||
def _write_db_revision_file(self, version_num):
|
|
||||||
with open(self.revision_filename, mode='w') as db_revision:
|
|
||||||
db_revision.write(str(version_num))
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
# check directories exist, create them if they don't
|
|
||||||
log.info("Loading databases")
|
|
||||||
|
|
||||||
if not os.path.exists(self.revision_filename):
|
|
||||||
log.info("db_revision file not found. Creating it")
|
|
||||||
self._write_db_revision_file(self.get_current_db_revision())
|
|
||||||
|
|
||||||
# check the db migration and run any needed migrations
|
|
||||||
with open(self.revision_filename, "r") as revision_read_handle:
|
|
||||||
old_revision = int(revision_read_handle.read().strip())
|
|
||||||
|
|
||||||
if old_revision > self.get_current_db_revision():
|
|
||||||
raise Exception('This version of lbrynet is not compatible with the database\n'
|
|
||||||
'Your database is revision %i, expected %i' %
|
|
||||||
(old_revision, self.get_current_db_revision()))
|
|
||||||
if old_revision < self.get_current_db_revision():
|
|
||||||
from lbry.extras.daemon.migrator import dbmigrator # pylint: disable=import-outside-toplevel
|
|
||||||
log.info("Upgrading your databases (revision %i to %i)", old_revision, self.get_current_db_revision())
|
|
||||||
await asyncio.get_event_loop().run_in_executor(
|
|
||||||
None, dbmigrator.migrate_db, self.conf, old_revision, self.get_current_db_revision()
|
|
||||||
)
|
|
||||||
self._write_db_revision_file(self.get_current_db_revision())
|
|
||||||
log.info("Finished upgrading the databases.")
|
|
||||||
|
|
||||||
self.storage = SQLiteStorage(
|
|
||||||
self.conf, os.path.join(self.conf.data_dir, "lbrynet.sqlite")
|
|
||||||
)
|
|
||||||
await self.storage.open()
|
|
||||||
|
|
||||||
async def stop(self):
|
|
||||||
await self.storage.close()
|
|
||||||
self.storage = None
|
|
||||||
|
|
||||||
|
|
||||||
class WalletComponent(Component):
|
|
||||||
component_name = WALLET_COMPONENT
|
|
||||||
depends_on = [DATABASE_COMPONENT]
|
|
||||||
|
|
||||||
def __init__(self, component_manager):
|
|
||||||
super().__init__(component_manager)
|
|
||||||
self.wallet_manager = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def component(self):
|
|
||||||
return self.wallet_manager
|
|
||||||
|
|
||||||
async def get_status(self):
|
|
||||||
if self.wallet_manager is None:
|
|
||||||
return
|
|
||||||
is_connected = self.wallet_manager.ledger.network.is_connected
|
|
||||||
sessions = []
|
|
||||||
connected = None
|
|
||||||
if is_connected:
|
|
||||||
addr, port = self.wallet_manager.ledger.network.client.server
|
|
||||||
connected = f"{addr}:{port}"
|
|
||||||
sessions.append(self.wallet_manager.ledger.network.client)
|
|
||||||
|
|
||||||
result = {
|
|
||||||
'connected': connected,
|
|
||||||
'connected_features': self.wallet_manager.ledger.network.server_features,
|
|
||||||
'servers': [
|
|
||||||
{
|
|
||||||
'host': session.server[0],
|
|
||||||
'port': session.server[1],
|
|
||||||
'latency': session.connection_latency,
|
|
||||||
'availability': session.available,
|
|
||||||
} for session in sessions
|
|
||||||
],
|
|
||||||
'known_servers': len(self.wallet_manager.ledger.network.known_hubs),
|
|
||||||
'available_servers': 1 if is_connected else 0
|
|
||||||
}
|
|
||||||
|
|
||||||
if self.wallet_manager.ledger.network.remote_height:
|
|
||||||
local_height = self.wallet_manager.ledger.local_height_including_downloaded_height
|
|
||||||
disk_height = len(self.wallet_manager.ledger.headers)
|
|
||||||
remote_height = self.wallet_manager.ledger.network.remote_height
|
|
||||||
download_height, target_height = local_height - disk_height, remote_height - disk_height
|
|
||||||
if target_height > 0:
|
|
||||||
progress = min(max(math.ceil(float(download_height) / float(target_height) * 100), 0), 100)
|
|
||||||
else:
|
|
||||||
progress = 100
|
|
||||||
best_hash = await self.wallet_manager.get_best_blockhash()
|
|
||||||
result.update({
|
|
||||||
'headers_synchronization_progress': progress,
|
|
||||||
'blocks': max(local_height, 0),
|
|
||||||
'blocks_behind': max(remote_height - local_height, 0),
|
|
||||||
'best_blockhash': best_hash,
|
|
||||||
})
|
|
||||||
|
|
||||||
return result
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
log.info("Starting wallet")
|
|
||||||
self.wallet_manager = await WalletManager.from_lbrynet_config(self.conf)
|
|
||||||
await self.wallet_manager.start()
|
|
||||||
|
|
||||||
async def stop(self):
|
|
||||||
await self.wallet_manager.stop()
|
|
||||||
self.wallet_manager = None
|
|
||||||
|
|
||||||
|
|
||||||
class WalletServerPaymentsComponent(Component):
|
|
||||||
component_name = WALLET_SERVER_PAYMENTS_COMPONENT
|
|
||||||
depends_on = [WALLET_COMPONENT]
|
|
||||||
|
|
||||||
def __init__(self, component_manager):
|
|
||||||
super().__init__(component_manager)
|
|
||||||
self.usage_payment_service = WalletServerPayer(
|
|
||||||
max_fee=self.conf.max_wallet_server_fee, analytics_manager=self.component_manager.analytics_manager,
|
|
||||||
)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def component(self) -> typing.Optional[WalletServerPayer]:
|
|
||||||
return self.usage_payment_service
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
wallet_manager = self.component_manager.get_component(WALLET_COMPONENT)
|
|
||||||
await self.usage_payment_service.start(wallet_manager.ledger, wallet_manager.default_wallet)
|
|
||||||
|
|
||||||
async def stop(self):
|
|
||||||
await self.usage_payment_service.stop()
|
|
||||||
|
|
||||||
async def get_status(self):
|
|
||||||
return {
|
|
||||||
'max_fee': self.usage_payment_service.max_fee,
|
|
||||||
'running': self.usage_payment_service.running
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class BlobComponent(Component):
|
|
||||||
component_name = BLOB_COMPONENT
|
|
||||||
depends_on = [DATABASE_COMPONENT]
|
|
||||||
|
|
||||||
def __init__(self, component_manager):
|
|
||||||
super().__init__(component_manager)
|
|
||||||
self.blob_manager: typing.Optional[BlobManager] = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def component(self) -> typing.Optional[BlobManager]:
|
|
||||||
return self.blob_manager
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
storage = self.component_manager.get_component(DATABASE_COMPONENT)
|
|
||||||
data_store = None
|
|
||||||
if DHT_COMPONENT not in self.component_manager.skip_components:
|
|
||||||
dht_node: Node = self.component_manager.get_component(DHT_COMPONENT)
|
|
||||||
if dht_node:
|
|
||||||
data_store = dht_node.protocol.data_store
|
|
||||||
blob_dir = os.path.join(self.conf.data_dir, 'blobfiles')
|
|
||||||
if not os.path.isdir(blob_dir):
|
|
||||||
os.mkdir(blob_dir)
|
|
||||||
self.blob_manager = BlobManager(self.component_manager.loop, blob_dir, storage, self.conf, data_store)
|
|
||||||
return await self.blob_manager.setup()
|
|
||||||
|
|
||||||
async def stop(self):
|
|
||||||
self.blob_manager.stop()
|
|
||||||
|
|
||||||
async def get_status(self):
|
|
||||||
count = 0
|
|
||||||
if self.blob_manager:
|
|
||||||
count = len(self.blob_manager.completed_blob_hashes)
|
|
||||||
return {
|
|
||||||
'finished_blobs': count,
|
|
||||||
'connections': {} if not self.blob_manager else self.blob_manager.connection_manager.status
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class DHTComponent(Component):
|
|
||||||
component_name = DHT_COMPONENT
|
|
||||||
depends_on = [UPNP_COMPONENT, DATABASE_COMPONENT]
|
|
||||||
|
|
||||||
def __init__(self, component_manager):
|
|
||||||
super().__init__(component_manager)
|
|
||||||
self.dht_node: typing.Optional[Node] = None
|
|
||||||
self.external_udp_port = None
|
|
||||||
self.external_peer_port = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def component(self) -> typing.Optional[Node]:
|
|
||||||
return self.dht_node
|
|
||||||
|
|
||||||
async def get_status(self):
|
|
||||||
return {
|
|
||||||
'node_id': None if not self.dht_node else binascii.hexlify(self.dht_node.protocol.node_id),
|
|
||||||
'peers_in_routing_table': 0 if not self.dht_node else len(self.dht_node.protocol.routing_table.get_peers())
|
|
||||||
}
|
|
||||||
|
|
||||||
def get_node_id(self):
|
|
||||||
node_id_filename = os.path.join(self.conf.data_dir, "node_id")
|
|
||||||
if os.path.isfile(node_id_filename):
|
|
||||||
with open(node_id_filename, "r") as node_id_file:
|
|
||||||
return base58.b58decode(str(node_id_file.read()).strip())
|
|
||||||
node_id = utils.generate_id()
|
|
||||||
with open(node_id_filename, "w") as node_id_file:
|
|
||||||
node_id_file.write(base58.b58encode(node_id).decode())
|
|
||||||
return node_id
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
log.info("start the dht")
|
|
||||||
upnp_component = self.component_manager.get_component(UPNP_COMPONENT)
|
|
||||||
self.external_peer_port = upnp_component.upnp_redirects.get("TCP", self.conf.tcp_port)
|
|
||||||
self.external_udp_port = upnp_component.upnp_redirects.get("UDP", self.conf.udp_port)
|
|
||||||
external_ip = upnp_component.external_ip
|
|
||||||
storage = self.component_manager.get_component(DATABASE_COMPONENT)
|
|
||||||
if not external_ip:
|
|
||||||
external_ip, _ = await utils.get_external_ip(self.conf.lbryum_servers)
|
|
||||||
if not external_ip:
|
|
||||||
log.warning("failed to get external ip")
|
|
||||||
|
|
||||||
self.dht_node = Node(
|
|
||||||
self.component_manager.loop,
|
|
||||||
self.component_manager.peer_manager,
|
|
||||||
node_id=self.get_node_id(),
|
|
||||||
internal_udp_port=self.conf.udp_port,
|
|
||||||
udp_port=self.external_udp_port,
|
|
||||||
external_ip=external_ip,
|
|
||||||
peer_port=self.external_peer_port,
|
|
||||||
rpc_timeout=self.conf.node_rpc_timeout,
|
|
||||||
split_buckets_under_index=self.conf.split_buckets_under_index,
|
|
||||||
is_bootstrap_node=self.conf.is_bootstrap_node,
|
|
||||||
storage=storage
|
|
||||||
)
|
|
||||||
self.dht_node.start(self.conf.network_interface, self.conf.known_dht_nodes)
|
|
||||||
log.info("Started the dht")
|
|
||||||
|
|
||||||
async def stop(self):
|
|
||||||
self.dht_node.stop()
|
|
||||||
|
|
||||||
|
|
||||||
class HashAnnouncerComponent(Component):
|
|
||||||
component_name = HASH_ANNOUNCER_COMPONENT
|
|
||||||
depends_on = [DHT_COMPONENT, DATABASE_COMPONENT]
|
|
||||||
|
|
||||||
def __init__(self, component_manager):
|
|
||||||
super().__init__(component_manager)
|
|
||||||
self.hash_announcer: typing.Optional[BlobAnnouncer] = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def component(self) -> typing.Optional[BlobAnnouncer]:
|
|
||||||
return self.hash_announcer
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
storage = self.component_manager.get_component(DATABASE_COMPONENT)
|
|
||||||
dht_node = self.component_manager.get_component(DHT_COMPONENT)
|
|
||||||
self.hash_announcer = BlobAnnouncer(self.component_manager.loop, dht_node, storage)
|
|
||||||
self.hash_announcer.start(self.conf.concurrent_blob_announcers)
|
|
||||||
log.info("Started blob announcer")
|
|
||||||
|
|
||||||
async def stop(self):
|
|
||||||
self.hash_announcer.stop()
|
|
||||||
log.info("Stopped blob announcer")
|
|
||||||
|
|
||||||
async def get_status(self):
|
|
||||||
return {
|
|
||||||
'announce_queue_size': 0 if not self.hash_announcer else len(self.hash_announcer.announce_queue)
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class FileManagerComponent(Component):
|
|
||||||
component_name = FILE_MANAGER_COMPONENT
|
|
||||||
depends_on = [BLOB_COMPONENT, DATABASE_COMPONENT, WALLET_COMPONENT]
|
|
||||||
|
|
||||||
def __init__(self, component_manager):
|
|
||||||
super().__init__(component_manager)
|
|
||||||
self.file_manager: typing.Optional[FileManager] = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def component(self) -> typing.Optional[FileManager]:
|
|
||||||
return self.file_manager
|
|
||||||
|
|
||||||
async def get_status(self):
|
|
||||||
if not self.file_manager:
|
|
||||||
return
|
|
||||||
return {
|
|
||||||
'managed_files': len(self.file_manager.get_filtered()),
|
|
||||||
}
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
blob_manager = self.component_manager.get_component(BLOB_COMPONENT)
|
|
||||||
storage = self.component_manager.get_component(DATABASE_COMPONENT)
|
|
||||||
wallet = self.component_manager.get_component(WALLET_COMPONENT)
|
|
||||||
node = self.component_manager.get_component(DHT_COMPONENT) \
|
|
||||||
if self.component_manager.has_component(DHT_COMPONENT) else None
|
|
||||||
log.info('Starting the file manager')
|
|
||||||
loop = asyncio.get_event_loop()
|
|
||||||
self.file_manager = FileManager(
|
|
||||||
loop, self.conf, wallet, storage, self.component_manager.analytics_manager
|
|
||||||
)
|
|
||||||
self.file_manager.source_managers['stream'] = StreamManager(
|
|
||||||
loop, self.conf, blob_manager, wallet, storage, node,
|
|
||||||
)
|
|
||||||
if self.component_manager.has_component(LIBTORRENT_COMPONENT):
|
|
||||||
torrent = self.component_manager.get_component(LIBTORRENT_COMPONENT)
|
|
||||||
self.file_manager.source_managers['torrent'] = TorrentManager(
|
|
||||||
loop, self.conf, torrent, storage, self.component_manager.analytics_manager
|
|
||||||
)
|
|
||||||
await self.file_manager.start()
|
|
||||||
log.info('Done setting up file manager')
|
|
||||||
|
|
||||||
async def stop(self):
|
|
||||||
await 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):
|
|
||||||
component_name = DISK_SPACE_COMPONENT
|
|
||||||
depends_on = [DATABASE_COMPONENT, BLOB_COMPONENT]
|
|
||||||
|
|
||||||
def __init__(self, component_manager):
|
|
||||||
super().__init__(component_manager)
|
|
||||||
self.disk_space_manager: typing.Optional[DiskSpaceManager] = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def component(self) -> typing.Optional[DiskSpaceManager]:
|
|
||||||
return self.disk_space_manager
|
|
||||||
|
|
||||||
async def get_status(self):
|
|
||||||
if self.disk_space_manager:
|
|
||||||
space_used = await self.disk_space_manager.get_space_used_mb(cached=True)
|
|
||||||
return {
|
|
||||||
'total_used_mb': space_used['total'],
|
|
||||||
'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,
|
|
||||||
}
|
|
||||||
return {'space_used': '0', 'network_seeding_space_used': '0', 'running': False}
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
db = self.component_manager.get_component(DATABASE_COMPONENT)
|
|
||||||
blob_manager = self.component_manager.get_component(BLOB_COMPONENT)
|
|
||||||
self.disk_space_manager = DiskSpaceManager(
|
|
||||||
self.conf, db, blob_manager,
|
|
||||||
analytics=self.component_manager.analytics_manager
|
|
||||||
)
|
|
||||||
await self.disk_space_manager.start()
|
|
||||||
|
|
||||||
async def stop(self):
|
|
||||||
await self.disk_space_manager.stop()
|
|
||||||
|
|
||||||
|
|
||||||
class TorrentComponent(Component):
|
|
||||||
component_name = LIBTORRENT_COMPONENT
|
|
||||||
|
|
||||||
def __init__(self, component_manager):
|
|
||||||
super().__init__(component_manager)
|
|
||||||
self.torrent_session = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def component(self) -> typing.Optional[TorrentSession]:
|
|
||||||
return self.torrent_session
|
|
||||||
|
|
||||||
async def get_status(self):
|
|
||||||
if not self.torrent_session:
|
|
||||||
return
|
|
||||||
return {
|
|
||||||
'running': True, # TODO: what to return here?
|
|
||||||
}
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
self.torrent_session = TorrentSession(asyncio.get_event_loop(), None)
|
|
||||||
await self.torrent_session.bind() # TODO: specify host/port
|
|
||||||
|
|
||||||
async def stop(self):
|
|
||||||
if self.torrent_session:
|
|
||||||
await self.torrent_session.pause()
|
|
||||||
|
|
||||||
|
|
||||||
class PeerProtocolServerComponent(Component):
|
|
||||||
component_name = PEER_PROTOCOL_SERVER_COMPONENT
|
|
||||||
depends_on = [UPNP_COMPONENT, BLOB_COMPONENT, WALLET_COMPONENT]
|
|
||||||
|
|
||||||
def __init__(self, component_manager):
|
|
||||||
super().__init__(component_manager)
|
|
||||||
self.blob_server: typing.Optional[BlobServer] = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def component(self) -> typing.Optional[BlobServer]:
|
|
||||||
return self.blob_server
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
log.info("start blob server")
|
|
||||||
blob_manager: BlobManager = self.component_manager.get_component(BLOB_COMPONENT)
|
|
||||||
wallet: WalletManager = self.component_manager.get_component(WALLET_COMPONENT)
|
|
||||||
peer_port = self.conf.tcp_port
|
|
||||||
address = await wallet.get_unused_address()
|
|
||||||
self.blob_server = BlobServer(asyncio.get_event_loop(), blob_manager, address)
|
|
||||||
self.blob_server.start_server(peer_port, interface=self.conf.network_interface)
|
|
||||||
await self.blob_server.started_listening.wait()
|
|
||||||
|
|
||||||
async def stop(self):
|
|
||||||
if self.blob_server:
|
|
||||||
self.blob_server.stop_server()
|
|
||||||
|
|
||||||
|
|
||||||
class UPnPComponent(Component):
|
|
||||||
component_name = UPNP_COMPONENT
|
|
||||||
|
|
||||||
def __init__(self, component_manager):
|
|
||||||
super().__init__(component_manager)
|
|
||||||
self._int_peer_port = self.conf.tcp_port
|
|
||||||
self._int_dht_node_port = self.conf.udp_port
|
|
||||||
self.use_upnp = self.conf.use_upnp
|
|
||||||
self.upnp: typing.Optional[UPnP] = None
|
|
||||||
self.upnp_redirects = {}
|
|
||||||
self.external_ip: typing.Optional[str] = None
|
|
||||||
self._maintain_redirects_task = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def component(self) -> 'UPnPComponent':
|
|
||||||
return self
|
|
||||||
|
|
||||||
async def _repeatedly_maintain_redirects(self, now=True):
|
|
||||||
while True:
|
|
||||||
if now:
|
|
||||||
await self._maintain_redirects()
|
|
||||||
await asyncio.sleep(360)
|
|
||||||
|
|
||||||
async def _maintain_redirects(self):
|
|
||||||
# setup the gateway if necessary
|
|
||||||
if not self.upnp:
|
|
||||||
try:
|
|
||||||
self.upnp = await UPnP.discover(loop=self.component_manager.loop)
|
|
||||||
log.info("found upnp gateway: %s", self.upnp.gateway.manufacturer_string)
|
|
||||||
except Exception as err:
|
|
||||||
log.warning("upnp discovery failed: %s", err)
|
|
||||||
self.upnp = None
|
|
||||||
|
|
||||||
# update the external ip
|
|
||||||
external_ip = None
|
|
||||||
if self.upnp:
|
|
||||||
try:
|
|
||||||
external_ip = await self.upnp.get_external_ip()
|
|
||||||
if external_ip != "0.0.0.0" and not self.external_ip:
|
|
||||||
log.info("got external ip from UPnP: %s", external_ip)
|
|
||||||
except (asyncio.TimeoutError, UPnPError, NotImplementedError):
|
|
||||||
pass
|
|
||||||
if external_ip and not is_valid_public_ipv4(external_ip):
|
|
||||||
log.warning("UPnP returned a private/reserved ip - %s, checking lbry.com fallback", external_ip)
|
|
||||||
external_ip, _ = await utils.get_external_ip(self.conf.lbryum_servers)
|
|
||||||
if self.external_ip and self.external_ip != external_ip:
|
|
||||||
log.info("external ip changed from %s to %s", self.external_ip, external_ip)
|
|
||||||
if external_ip:
|
|
||||||
self.external_ip = external_ip
|
|
||||||
dht_component = self.component_manager.get_component(DHT_COMPONENT)
|
|
||||||
if dht_component:
|
|
||||||
dht_node = dht_component.component
|
|
||||||
dht_node.protocol.external_ip = external_ip
|
|
||||||
# assert self.external_ip is not None # TODO: handle going/starting offline
|
|
||||||
|
|
||||||
if not self.upnp_redirects and self.upnp: # setup missing redirects
|
|
||||||
log.info("add UPnP port mappings")
|
|
||||||
upnp_redirects = {}
|
|
||||||
if PEER_PROTOCOL_SERVER_COMPONENT not in self.component_manager.skip_components:
|
|
||||||
try:
|
|
||||||
upnp_redirects["TCP"] = await self.upnp.get_next_mapping(
|
|
||||||
self._int_peer_port, "TCP", "LBRY peer port", self._int_peer_port
|
|
||||||
)
|
|
||||||
except (UPnPError, asyncio.TimeoutError, NotImplementedError):
|
|
||||||
pass
|
|
||||||
if DHT_COMPONENT not in self.component_manager.skip_components:
|
|
||||||
try:
|
|
||||||
upnp_redirects["UDP"] = await self.upnp.get_next_mapping(
|
|
||||||
self._int_dht_node_port, "UDP", "LBRY DHT port", self._int_dht_node_port
|
|
||||||
)
|
|
||||||
except (UPnPError, asyncio.TimeoutError, NotImplementedError):
|
|
||||||
pass
|
|
||||||
if upnp_redirects:
|
|
||||||
log.info("set up redirects: %s", upnp_redirects)
|
|
||||||
self.upnp_redirects.update(upnp_redirects)
|
|
||||||
elif self.upnp: # check existing redirects are still active
|
|
||||||
found = set()
|
|
||||||
mappings = await self.upnp.get_redirects()
|
|
||||||
for mapping in mappings:
|
|
||||||
proto = mapping.protocol
|
|
||||||
if proto in self.upnp_redirects and mapping.external_port == self.upnp_redirects[proto]:
|
|
||||||
if mapping.lan_address == self.upnp.lan_address:
|
|
||||||
found.add(proto)
|
|
||||||
if 'UDP' not in found and DHT_COMPONENT not in self.component_manager.skip_components:
|
|
||||||
try:
|
|
||||||
udp_port = await self.upnp.get_next_mapping(self._int_dht_node_port, "UDP", "LBRY DHT port")
|
|
||||||
self.upnp_redirects['UDP'] = udp_port
|
|
||||||
log.info("refreshed upnp redirect for dht port: %i", udp_port)
|
|
||||||
except (asyncio.TimeoutError, UPnPError, NotImplementedError):
|
|
||||||
del self.upnp_redirects['UDP']
|
|
||||||
if 'TCP' not in found and PEER_PROTOCOL_SERVER_COMPONENT not in self.component_manager.skip_components:
|
|
||||||
try:
|
|
||||||
tcp_port = await self.upnp.get_next_mapping(self._int_peer_port, "TCP", "LBRY peer port")
|
|
||||||
self.upnp_redirects['TCP'] = tcp_port
|
|
||||||
log.info("refreshed upnp redirect for peer port: %i", tcp_port)
|
|
||||||
except (asyncio.TimeoutError, UPnPError, NotImplementedError):
|
|
||||||
del self.upnp_redirects['TCP']
|
|
||||||
if ('TCP' in self.upnp_redirects and
|
|
||||||
PEER_PROTOCOL_SERVER_COMPONENT not in self.component_manager.skip_components) and \
|
|
||||||
('UDP' in self.upnp_redirects and DHT_COMPONENT not in self.component_manager.skip_components):
|
|
||||||
if self.upnp_redirects:
|
|
||||||
log.debug("upnp redirects are still active")
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
log.info("detecting external ip")
|
|
||||||
if not self.use_upnp:
|
|
||||||
self.external_ip, _ = await utils.get_external_ip(self.conf.lbryum_servers)
|
|
||||||
return
|
|
||||||
success = False
|
|
||||||
await self._maintain_redirects()
|
|
||||||
if self.upnp:
|
|
||||||
if not self.upnp_redirects and not all(
|
|
||||||
x in self.component_manager.skip_components
|
|
||||||
for x in (DHT_COMPONENT, PEER_PROTOCOL_SERVER_COMPONENT)
|
|
||||||
):
|
|
||||||
log.error("failed to setup upnp")
|
|
||||||
else:
|
|
||||||
success = True
|
|
||||||
if self.upnp_redirects:
|
|
||||||
log.debug("set up upnp port redirects for gateway: %s", self.upnp.gateway.manufacturer_string)
|
|
||||||
else:
|
|
||||||
log.error("failed to setup upnp")
|
|
||||||
if not self.external_ip:
|
|
||||||
self.external_ip, probed_url = await utils.get_external_ip(self.conf.lbryum_servers)
|
|
||||||
if self.external_ip:
|
|
||||||
log.info("detected external ip using %s fallback", probed_url)
|
|
||||||
if self.component_manager.analytics_manager:
|
|
||||||
self.component_manager.loop.create_task(
|
|
||||||
self.component_manager.analytics_manager.send_upnp_setup_success_fail(
|
|
||||||
success, await self.get_status()
|
|
||||||
)
|
|
||||||
)
|
|
||||||
self._maintain_redirects_task = self.component_manager.loop.create_task(
|
|
||||||
self._repeatedly_maintain_redirects(now=False)
|
|
||||||
)
|
|
||||||
|
|
||||||
async def stop(self):
|
|
||||||
if self.upnp_redirects:
|
|
||||||
log.info("Removing upnp redirects: %s", self.upnp_redirects)
|
|
||||||
await asyncio.wait([
|
|
||||||
self.upnp.delete_port_mapping(port, protocol) for protocol, port in self.upnp_redirects.items()
|
|
||||||
])
|
|
||||||
if self._maintain_redirects_task and not self._maintain_redirects_task.done():
|
|
||||||
self._maintain_redirects_task.cancel()
|
|
||||||
|
|
||||||
async def get_status(self):
|
|
||||||
return {
|
|
||||||
'aioupnp_version': aioupnp_version,
|
|
||||||
'redirects': self.upnp_redirects,
|
|
||||||
'gateway': 'No gateway found' if not self.upnp else self.upnp.gateway.manufacturer_string,
|
|
||||||
'dht_redirect_set': 'UDP' in self.upnp_redirects,
|
|
||||||
'peer_redirect_set': 'TCP' in self.upnp_redirects,
|
|
||||||
'external_ip': self.external_ip
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class ExchangeRateManagerComponent(Component):
|
|
||||||
component_name = EXCHANGE_RATE_MANAGER_COMPONENT
|
|
||||||
|
|
||||||
def __init__(self, component_manager):
|
|
||||||
super().__init__(component_manager)
|
|
||||||
self.exchange_rate_manager = ExchangeRateManager()
|
|
||||||
|
|
||||||
@property
|
|
||||||
def component(self) -> ExchangeRateManager:
|
|
||||||
return self.exchange_rate_manager
|
|
||||||
|
|
||||||
async def start(self):
|
|
||||||
self.exchange_rate_manager.start()
|
|
||||||
|
|
||||||
async def stop(self):
|
|
||||||
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()
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,248 +0,0 @@
|
||||||
import json
|
|
||||||
import time
|
|
||||||
import asyncio
|
|
||||||
import logging
|
|
||||||
from statistics import median
|
|
||||||
from decimal import Decimal
|
|
||||||
from typing import Optional, Iterable, Type
|
|
||||||
from aiohttp.client_exceptions import ContentTypeError, ClientConnectionError
|
|
||||||
from lbry.error import InvalidExchangeRateResponseError, CurrencyConversionError
|
|
||||||
from lbry.utils import aiohttp_request
|
|
||||||
from lbry.wallet.dewies import lbc_to_dewies
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class ExchangeRate:
|
|
||||||
def __init__(self, market, spot, ts):
|
|
||||||
if not int(time.time()) - ts < 600:
|
|
||||||
raise ValueError('The timestamp is too dated.')
|
|
||||||
if not spot > 0:
|
|
||||||
raise ValueError('Spot must be greater than 0.')
|
|
||||||
self.currency_pair = (market[0:3], market[3:6])
|
|
||||||
self.spot = spot
|
|
||||||
self.ts = ts
|
|
||||||
|
|
||||||
def __repr__(self):
|
|
||||||
return f"Currency pair:{self.currency_pair}, spot:{self.spot}, ts:{self.ts}"
|
|
||||||
|
|
||||||
def as_dict(self):
|
|
||||||
return {'spot': self.spot, 'ts': self.ts}
|
|
||||||
|
|
||||||
|
|
||||||
class MarketFeed:
|
|
||||||
name: str = ""
|
|
||||||
market: str = ""
|
|
||||||
url: str = ""
|
|
||||||
params = {}
|
|
||||||
fee = 0
|
|
||||||
|
|
||||||
update_interval = 300
|
|
||||||
request_timeout = 50
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
self.rate: Optional[float] = None
|
|
||||||
self.last_check = 0
|
|
||||||
self._last_response = None
|
|
||||||
self._task: Optional[asyncio.Task] = None
|
|
||||||
self.event = asyncio.Event()
|
|
||||||
|
|
||||||
@property
|
|
||||||
def has_rate(self):
|
|
||||||
return self.rate is not None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def is_online(self):
|
|
||||||
return self.last_check+self.update_interval+self.request_timeout > time.time()
|
|
||||||
|
|
||||||
def get_rate_from_response(self, json_response):
|
|
||||||
raise NotImplementedError()
|
|
||||||
|
|
||||||
async def get_response(self):
|
|
||||||
async with aiohttp_request(
|
|
||||||
'get', self.url, params=self.params,
|
|
||||||
timeout=self.request_timeout, headers={"User-Agent": "lbrynet"}
|
|
||||||
) as response:
|
|
||||||
try:
|
|
||||||
self._last_response = await response.json(content_type=None)
|
|
||||||
except ContentTypeError as e:
|
|
||||||
self._last_response = {}
|
|
||||||
log.warning("Could not parse exchange rate response from %s: %s", self.name, e.message)
|
|
||||||
log.debug(await response.text())
|
|
||||||
return self._last_response
|
|
||||||
|
|
||||||
async def get_rate(self):
|
|
||||||
try:
|
|
||||||
data = await self.get_response()
|
|
||||||
rate = self.get_rate_from_response(data)
|
|
||||||
rate = rate / (1.0 - self.fee)
|
|
||||||
log.debug("Saving rate update %f for %s from %s", rate, self.market, self.name)
|
|
||||||
self.rate = ExchangeRate(self.market, rate, int(time.time()))
|
|
||||||
self.last_check = time.time()
|
|
||||||
return self.rate
|
|
||||||
except asyncio.TimeoutError:
|
|
||||||
log.warning("Timed out fetching exchange rate from %s.", self.name)
|
|
||||||
except json.JSONDecodeError as e:
|
|
||||||
msg = e.doc if '<html>' not in e.doc else 'unexpected content type.'
|
|
||||||
log.warning("Could not parse exchange rate response from %s: %s", self.name, msg)
|
|
||||||
log.debug(e.doc)
|
|
||||||
except InvalidExchangeRateResponseError as e:
|
|
||||||
log.warning(str(e))
|
|
||||||
except ClientConnectionError as e:
|
|
||||||
log.warning("Error trying to connect to exchange rate %s: %s", self.name, str(e))
|
|
||||||
except Exception as e:
|
|
||||||
log.exception("Exchange rate error (%s from %s):", self.market, self.name)
|
|
||||||
finally:
|
|
||||||
self.event.set()
|
|
||||||
|
|
||||||
async def keep_updated(self):
|
|
||||||
while True:
|
|
||||||
await self.get_rate()
|
|
||||||
await asyncio.sleep(self.update_interval)
|
|
||||||
|
|
||||||
def start(self):
|
|
||||||
if not self._task:
|
|
||||||
self._task = asyncio.create_task(self.keep_updated())
|
|
||||||
|
|
||||||
def stop(self):
|
|
||||||
if self._task and not self._task.done():
|
|
||||||
self._task.cancel()
|
|
||||||
self._task = None
|
|
||||||
self.event.clear()
|
|
||||||
|
|
||||||
|
|
||||||
class BaseBittrexFeed(MarketFeed):
|
|
||||||
name = "Bittrex"
|
|
||||||
market = None
|
|
||||||
url = None
|
|
||||||
fee = 0.0025
|
|
||||||
|
|
||||||
def get_rate_from_response(self, json_response):
|
|
||||||
if 'lastTradeRate' not in json_response:
|
|
||||||
raise InvalidExchangeRateResponseError(self.name, 'result not found')
|
|
||||||
return 1.0 / float(json_response['lastTradeRate'])
|
|
||||||
|
|
||||||
|
|
||||||
class BittrexBTCFeed(BaseBittrexFeed):
|
|
||||||
market = "BTCLBC"
|
|
||||||
url = "https://api.bittrex.com/v3/markets/LBC-BTC/ticker"
|
|
||||||
|
|
||||||
|
|
||||||
class BittrexUSDFeed(BaseBittrexFeed):
|
|
||||||
market = "USDLBC"
|
|
||||||
url = "https://api.bittrex.com/v3/markets/LBC-USD/ticker"
|
|
||||||
|
|
||||||
|
|
||||||
class BaseCoinExFeed(MarketFeed):
|
|
||||||
name = "CoinEx"
|
|
||||||
market = None
|
|
||||||
url = None
|
|
||||||
|
|
||||||
def get_rate_from_response(self, json_response):
|
|
||||||
if 'data' not in json_response or \
|
|
||||||
'ticker' not in json_response['data'] or \
|
|
||||||
'last' not in json_response['data']['ticker']:
|
|
||||||
raise InvalidExchangeRateResponseError(self.name, 'result not found')
|
|
||||||
return 1.0 / float(json_response['data']['ticker']['last'])
|
|
||||||
|
|
||||||
|
|
||||||
class CoinExBTCFeed(BaseCoinExFeed):
|
|
||||||
market = "BTCLBC"
|
|
||||||
url = "https://api.coinex.com/v1/market/ticker?market=LBCBTC"
|
|
||||||
|
|
||||||
|
|
||||||
class CoinExUSDFeed(BaseCoinExFeed):
|
|
||||||
market = "USDLBC"
|
|
||||||
url = "https://api.coinex.com/v1/market/ticker?market=LBCUSDT"
|
|
||||||
|
|
||||||
|
|
||||||
class BaseHotbitFeed(MarketFeed):
|
|
||||||
name = "hotbit"
|
|
||||||
market = None
|
|
||||||
url = "https://api.hotbit.io/api/v1/market.last"
|
|
||||||
|
|
||||||
def get_rate_from_response(self, json_response):
|
|
||||||
if 'result' not in json_response:
|
|
||||||
raise InvalidExchangeRateResponseError(self.name, 'result not found')
|
|
||||||
return 1.0 / float(json_response['result'])
|
|
||||||
|
|
||||||
|
|
||||||
class HotbitBTCFeed(BaseHotbitFeed):
|
|
||||||
market = "BTCLBC"
|
|
||||||
params = {"market": "LBC/BTC"}
|
|
||||||
|
|
||||||
|
|
||||||
class HotbitUSDFeed(BaseHotbitFeed):
|
|
||||||
market = "USDLBC"
|
|
||||||
params = {"market": "LBC/USDT"}
|
|
||||||
|
|
||||||
|
|
||||||
class UPbitBTCFeed(MarketFeed):
|
|
||||||
name = "UPbit"
|
|
||||||
market = "BTCLBC"
|
|
||||||
url = "https://api.upbit.com/v1/ticker"
|
|
||||||
params = {"markets": "BTC-LBC"}
|
|
||||||
|
|
||||||
def get_rate_from_response(self, json_response):
|
|
||||||
if "error" in json_response or len(json_response) != 1 or 'trade_price' not in json_response[0]:
|
|
||||||
raise InvalidExchangeRateResponseError(self.name, 'result not found')
|
|
||||||
return 1.0 / float(json_response[0]['trade_price'])
|
|
||||||
|
|
||||||
|
|
||||||
FEEDS: Iterable[Type[MarketFeed]] = (
|
|
||||||
BittrexBTCFeed,
|
|
||||||
BittrexUSDFeed,
|
|
||||||
CoinExBTCFeed,
|
|
||||||
CoinExUSDFeed,
|
|
||||||
# HotbitBTCFeed,
|
|
||||||
# HotbitUSDFeed,
|
|
||||||
# UPbitBTCFeed,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class ExchangeRateManager:
|
|
||||||
def __init__(self, feeds=FEEDS):
|
|
||||||
self.market_feeds = [Feed() for Feed in feeds]
|
|
||||||
|
|
||||||
def wait(self):
|
|
||||||
return asyncio.wait(
|
|
||||||
[feed.event.wait() for feed in self.market_feeds],
|
|
||||||
)
|
|
||||||
|
|
||||||
def start(self):
|
|
||||||
log.info("Starting exchange rate manager")
|
|
||||||
for feed in self.market_feeds:
|
|
||||||
feed.start()
|
|
||||||
|
|
||||||
def stop(self):
|
|
||||||
log.info("Stopping exchange rate manager")
|
|
||||||
for source in self.market_feeds:
|
|
||||||
source.stop()
|
|
||||||
|
|
||||||
def convert_currency(self, from_currency, to_currency, amount):
|
|
||||||
log.debug(
|
|
||||||
"Converting %f %s to %s, rates: %s",
|
|
||||||
amount, from_currency, to_currency,
|
|
||||||
[market.rate for market in self.market_feeds]
|
|
||||||
)
|
|
||||||
if from_currency == to_currency:
|
|
||||||
return round(amount, 8)
|
|
||||||
|
|
||||||
rates = []
|
|
||||||
for market in self.market_feeds:
|
|
||||||
if (market.has_rate and market.is_online and
|
|
||||||
market.rate.currency_pair == (from_currency, to_currency)):
|
|
||||||
rates.append(market.rate.spot)
|
|
||||||
|
|
||||||
if rates:
|
|
||||||
return round(amount * Decimal(median(rates)), 8)
|
|
||||||
|
|
||||||
raise CurrencyConversionError(
|
|
||||||
f'Unable to convert {amount} from {from_currency} to {to_currency}')
|
|
||||||
|
|
||||||
def to_dewies(self, currency, amount) -> int:
|
|
||||||
converted = self.convert_currency(currency, "LBC", amount)
|
|
||||||
return lbc_to_dewies(str(converted))
|
|
||||||
|
|
||||||
def fee_dict(self):
|
|
||||||
return {market: market.rate.as_dict() for market in self.market_feeds}
|
|
|
@ -1,361 +0,0 @@
|
||||||
import logging
|
|
||||||
from decimal import Decimal
|
|
||||||
from binascii import hexlify, unhexlify
|
|
||||||
from datetime import datetime
|
|
||||||
from json import JSONEncoder
|
|
||||||
|
|
||||||
from google.protobuf.message import DecodeError
|
|
||||||
|
|
||||||
from lbry.schema.claim import Claim
|
|
||||||
from lbry.schema.support import Support
|
|
||||||
from lbry.torrent.torrent_manager import TorrentSource
|
|
||||||
from lbry.wallet import Wallet, Ledger, Account, Transaction, Output
|
|
||||||
from lbry.wallet.bip32 import PublicKey
|
|
||||||
from lbry.wallet.dewies import dewies_to_lbc
|
|
||||||
from lbry.stream.managed_stream import ManagedStream
|
|
||||||
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
def encode_txo_doc():
|
|
||||||
return {
|
|
||||||
'txid': "hash of transaction in hex",
|
|
||||||
'nout': "position in the transaction",
|
|
||||||
'height': "block where transaction was recorded",
|
|
||||||
'amount': "value of the txo as a decimal",
|
|
||||||
'address': "address of who can spend the txo",
|
|
||||||
'confirmations': "number of confirmed blocks",
|
|
||||||
'is_change': "payment to change address, only available when it can be determined",
|
|
||||||
'is_received': "true if txo was sent from external account to this account",
|
|
||||||
'is_spent': "true if txo is spent",
|
|
||||||
'is_mine': "payment to one of your accounts, only available when it can be determined",
|
|
||||||
'type': "one of 'claim', 'support' or 'purchase'",
|
|
||||||
'name': "when type is 'claim' or 'support', this is the claim name",
|
|
||||||
'claim_id': "when type is 'claim', 'support' or 'purchase', this is the claim id",
|
|
||||||
'claim_op': "when type is 'claim', this determines if it is 'create' or 'update'",
|
|
||||||
'value': "when type is 'claim' or 'support' with payload, this is the decoded protobuf payload",
|
|
||||||
'value_type': "determines the type of the 'value' field: 'channel', 'stream', etc",
|
|
||||||
'protobuf': "hex encoded raw protobuf version of 'value' field",
|
|
||||||
'permanent_url': "when type is 'claim' or 'support', this is the long permanent claim URL",
|
|
||||||
'claim': "for purchase outputs only, metadata of purchased claim",
|
|
||||||
'reposted_claim': "for repost claims only, metadata of claim being reposted",
|
|
||||||
'signing_channel': "for signed claims only, metadata of signing channel",
|
|
||||||
'is_channel_signature_valid': "for signed claims only, whether signature is valid",
|
|
||||||
'purchase_receipt': "metadata for the purchase transaction associated with this claim"
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def encode_tx_doc():
|
|
||||||
return {
|
|
||||||
'txid': "hash of transaction in hex",
|
|
||||||
'height': "block where transaction was recorded",
|
|
||||||
'inputs': [encode_txo_doc()],
|
|
||||||
'outputs': [encode_txo_doc()],
|
|
||||||
'total_input': "sum of inputs as a decimal",
|
|
||||||
'total_output': "sum of outputs, sans fee, as a decimal",
|
|
||||||
'total_fee': "fee amount",
|
|
||||||
'hex': "entire transaction encoded in hex",
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def encode_account_doc():
|
|
||||||
return {
|
|
||||||
'id': 'account_id',
|
|
||||||
'is_default': 'this account is used by default',
|
|
||||||
'ledger': 'name of crypto currency and network',
|
|
||||||
'name': 'optional account name',
|
|
||||||
'seed': 'human friendly words from which account can be recreated',
|
|
||||||
'encrypted': 'if account is encrypted',
|
|
||||||
'private_key': 'extended private key',
|
|
||||||
'public_key': 'extended public key',
|
|
||||||
'address_generator': 'settings for generating addresses',
|
|
||||||
'modified_on': 'date of last modification to account settings'
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def encode_wallet_doc():
|
|
||||||
return {
|
|
||||||
'id': 'wallet_id',
|
|
||||||
'name': 'optional wallet name',
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def encode_file_doc():
|
|
||||||
return {
|
|
||||||
'streaming_url': '(str) url to stream the file using range requests',
|
|
||||||
'completed': '(bool) true if download is completed',
|
|
||||||
'file_name': '(str) name of file',
|
|
||||||
'download_directory': '(str) download directory',
|
|
||||||
'points_paid': '(float) credit paid to download file',
|
|
||||||
'stopped': '(bool) true if download is stopped',
|
|
||||||
'stream_hash': '(str) stream hash of file',
|
|
||||||
'stream_name': '(str) stream name',
|
|
||||||
'suggested_file_name': '(str) suggested file name',
|
|
||||||
'sd_hash': '(str) sd hash of file',
|
|
||||||
'download_path': '(str) download path of file',
|
|
||||||
'mime_type': '(str) mime type of file',
|
|
||||||
'key': '(str) key attached to file',
|
|
||||||
'total_bytes_lower_bound': '(int) lower bound file size in bytes',
|
|
||||||
'total_bytes': '(int) file upper bound size in bytes',
|
|
||||||
'written_bytes': '(int) written size in bytes',
|
|
||||||
'blobs_completed': '(int) number of fully downloaded blobs',
|
|
||||||
'blobs_in_stream': '(int) total blobs on stream',
|
|
||||||
'blobs_remaining': '(int) total blobs remaining to download',
|
|
||||||
'status': '(str) downloader status',
|
|
||||||
'claim_id': '(str) None if claim is not found else the claim id',
|
|
||||||
'txid': '(str) None if claim is not found else the transaction id',
|
|
||||||
'nout': '(int) None if claim is not found else the transaction output index',
|
|
||||||
'outpoint': '(str) None if claim is not found else the tx and output',
|
|
||||||
'metadata': '(dict) None if claim is not found else the claim metadata',
|
|
||||||
'channel_claim_id': '(str) None if claim is not found or not signed',
|
|
||||||
'channel_name': '(str) None if claim is not found or not signed',
|
|
||||||
'claim_name': '(str) None if claim is not found else the claim name',
|
|
||||||
'reflector_progress': '(int) reflector upload progress, 0 to 100',
|
|
||||||
'uploading_to_reflector': '(bool) set to True when currently uploading to reflector'
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class JSONResponseEncoder(JSONEncoder):
|
|
||||||
|
|
||||||
def __init__(self, *args, ledger: Ledger, include_protobuf=False, **kwargs):
|
|
||||||
super().__init__(*args, **kwargs)
|
|
||||||
self.ledger = ledger
|
|
||||||
self.include_protobuf = include_protobuf
|
|
||||||
|
|
||||||
def default(self, obj): # pylint: disable=method-hidden,arguments-renamed,too-many-return-statements
|
|
||||||
if isinstance(obj, Account):
|
|
||||||
return self.encode_account(obj)
|
|
||||||
if isinstance(obj, Wallet):
|
|
||||||
return self.encode_wallet(obj)
|
|
||||||
if isinstance(obj, (ManagedStream, TorrentSource)):
|
|
||||||
return self.encode_file(obj)
|
|
||||||
if isinstance(obj, Transaction):
|
|
||||||
return self.encode_transaction(obj)
|
|
||||||
if isinstance(obj, Output):
|
|
||||||
return self.encode_output(obj)
|
|
||||||
if isinstance(obj, Claim):
|
|
||||||
return self.encode_claim(obj)
|
|
||||||
if isinstance(obj, Support):
|
|
||||||
return obj.to_dict()
|
|
||||||
if isinstance(obj, PublicKey):
|
|
||||||
return obj.extended_key_string()
|
|
||||||
if isinstance(obj, datetime):
|
|
||||||
return obj.strftime("%Y%m%dT%H:%M:%S")
|
|
||||||
if isinstance(obj, Decimal):
|
|
||||||
return float(obj)
|
|
||||||
if isinstance(obj, bytes):
|
|
||||||
return obj.decode()
|
|
||||||
return super().default(obj)
|
|
||||||
|
|
||||||
def encode_transaction(self, tx):
|
|
||||||
return {
|
|
||||||
'txid': tx.id,
|
|
||||||
'height': tx.height,
|
|
||||||
'inputs': [self.encode_input(txo) for txo in tx.inputs],
|
|
||||||
'outputs': [self.encode_output(txo) for txo in tx.outputs],
|
|
||||||
'total_input': dewies_to_lbc(tx.input_sum),
|
|
||||||
'total_output': dewies_to_lbc(tx.input_sum - tx.fee),
|
|
||||||
'total_fee': dewies_to_lbc(tx.fee),
|
|
||||||
'hex': hexlify(tx.raw).decode(),
|
|
||||||
}
|
|
||||||
|
|
||||||
def encode_output(self, txo, check_signature=True):
|
|
||||||
if not txo:
|
|
||||||
return
|
|
||||||
tx_height = txo.tx_ref.height
|
|
||||||
best_height = self.ledger.headers.height
|
|
||||||
output = {
|
|
||||||
'txid': txo.tx_ref.id,
|
|
||||||
'nout': txo.position,
|
|
||||||
'height': tx_height,
|
|
||||||
'amount': dewies_to_lbc(txo.amount),
|
|
||||||
'address': txo.get_address(self.ledger) if txo.has_address else None,
|
|
||||||
'confirmations': (best_height+1) - tx_height if tx_height > 0 else tx_height,
|
|
||||||
'timestamp': self.ledger.headers.estimated_timestamp(tx_height)
|
|
||||||
}
|
|
||||||
if txo.is_spent is not None:
|
|
||||||
output['is_spent'] = txo.is_spent
|
|
||||||
if txo.is_my_output is not None:
|
|
||||||
output['is_my_output'] = txo.is_my_output
|
|
||||||
if txo.is_my_input is not None:
|
|
||||||
output['is_my_input'] = txo.is_my_input
|
|
||||||
if txo.sent_supports is not None:
|
|
||||||
output['sent_supports'] = dewies_to_lbc(txo.sent_supports)
|
|
||||||
if txo.sent_tips is not None:
|
|
||||||
output['sent_tips'] = dewies_to_lbc(txo.sent_tips)
|
|
||||||
if txo.received_tips is not None:
|
|
||||||
output['received_tips'] = dewies_to_lbc(txo.received_tips)
|
|
||||||
if txo.is_internal_transfer is not None:
|
|
||||||
output['is_internal_transfer'] = txo.is_internal_transfer
|
|
||||||
|
|
||||||
if txo.script.is_claim_name:
|
|
||||||
output['type'] = 'claim'
|
|
||||||
output['claim_op'] = 'create'
|
|
||||||
elif txo.script.is_update_claim:
|
|
||||||
output['type'] = 'claim'
|
|
||||||
output['claim_op'] = 'update'
|
|
||||||
elif txo.script.is_support_claim:
|
|
||||||
output['type'] = 'support'
|
|
||||||
elif txo.script.is_return_data:
|
|
||||||
output['type'] = 'data'
|
|
||||||
elif txo.purchase is not None:
|
|
||||||
output['type'] = 'purchase'
|
|
||||||
output['claim_id'] = txo.purchased_claim_id
|
|
||||||
if txo.purchased_claim is not None:
|
|
||||||
output['claim'] = self.encode_output(txo.purchased_claim)
|
|
||||||
else:
|
|
||||||
output['type'] = 'payment'
|
|
||||||
|
|
||||||
if txo.script.is_claim_involved:
|
|
||||||
output.update({
|
|
||||||
'name': txo.claim_name,
|
|
||||||
'normalized_name': txo.normalized_name,
|
|
||||||
'claim_id': txo.claim_id,
|
|
||||||
'permanent_url': txo.permanent_url,
|
|
||||||
'meta': self.encode_claim_meta(txo.meta.copy())
|
|
||||||
})
|
|
||||||
if 'short_url' in output['meta']:
|
|
||||||
output['short_url'] = output['meta'].pop('short_url')
|
|
||||||
if 'canonical_url' in output['meta']:
|
|
||||||
output['canonical_url'] = output['meta'].pop('canonical_url')
|
|
||||||
if txo.claims is not None:
|
|
||||||
output['claims'] = [self.encode_output(o) for o in txo.claims]
|
|
||||||
if txo.reposted_claim is not None:
|
|
||||||
output['reposted_claim'] = self.encode_output(txo.reposted_claim)
|
|
||||||
if txo.script.is_claim_name or txo.script.is_update_claim or txo.script.is_support_claim_data:
|
|
||||||
try:
|
|
||||||
output['value'] = txo.signable
|
|
||||||
if self.include_protobuf:
|
|
||||||
output['protobuf'] = hexlify(txo.signable.to_bytes())
|
|
||||||
if txo.purchase_receipt is not None:
|
|
||||||
output['purchase_receipt'] = self.encode_output(txo.purchase_receipt)
|
|
||||||
if txo.script.is_claim_name or txo.script.is_update_claim:
|
|
||||||
output['value_type'] = txo.claim.claim_type
|
|
||||||
if txo.claim.is_channel:
|
|
||||||
output['has_signing_key'] = txo.has_private_key
|
|
||||||
if check_signature and txo.signable.is_signed:
|
|
||||||
if txo.channel is not None:
|
|
||||||
output['signing_channel'] = self.encode_output(txo.channel)
|
|
||||||
output['is_channel_signature_valid'] = txo.is_signed_by(txo.channel, self.ledger)
|
|
||||||
else:
|
|
||||||
output['signing_channel'] = {'channel_id': txo.signable.signing_channel_id}
|
|
||||||
output['is_channel_signature_valid'] = False
|
|
||||||
except DecodeError:
|
|
||||||
pass
|
|
||||||
return output
|
|
||||||
|
|
||||||
def encode_claim_meta(self, meta):
|
|
||||||
for key, value in meta.items():
|
|
||||||
if key.endswith('_amount'):
|
|
||||||
if isinstance(value, int):
|
|
||||||
meta[key] = dewies_to_lbc(value)
|
|
||||||
if 0 < meta.get('creation_height', 0) <= self.ledger.headers.height:
|
|
||||||
meta['creation_timestamp'] = self.ledger.headers.estimated_timestamp(meta['creation_height'])
|
|
||||||
return meta
|
|
||||||
|
|
||||||
def encode_input(self, txi):
|
|
||||||
return self.encode_output(txi.txo_ref.txo, False) if txi.txo_ref.txo is not None else {
|
|
||||||
'txid': txi.txo_ref.tx_ref.id,
|
|
||||||
'nout': txi.txo_ref.position
|
|
||||||
}
|
|
||||||
|
|
||||||
def encode_account(self, account):
|
|
||||||
result = account.to_dict()
|
|
||||||
result['id'] = account.id
|
|
||||||
result.pop('certificates', None)
|
|
||||||
result['is_default'] = self.ledger.accounts[0] == account
|
|
||||||
return result
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def encode_wallet(wallet):
|
|
||||||
return {
|
|
||||||
'id': wallet.id,
|
|
||||||
'name': wallet.name
|
|
||||||
}
|
|
||||||
|
|
||||||
def encode_file(self, managed_stream):
|
|
||||||
output_exists = managed_stream.output_file_exists
|
|
||||||
tx_height = managed_stream.stream_claim_info.height
|
|
||||||
best_height = self.ledger.headers.height
|
|
||||||
is_stream = hasattr(managed_stream, 'stream_hash')
|
|
||||||
if is_stream:
|
|
||||||
total_bytes_lower_bound = managed_stream.descriptor.lower_bound_decrypted_length()
|
|
||||||
total_bytes = managed_stream.descriptor.upper_bound_decrypted_length()
|
|
||||||
else:
|
|
||||||
total_bytes_lower_bound = total_bytes = managed_stream.torrent_length
|
|
||||||
result = {
|
|
||||||
'streaming_url': None,
|
|
||||||
'completed': managed_stream.completed,
|
|
||||||
'file_name': None,
|
|
||||||
'download_directory': None,
|
|
||||||
'download_path': None,
|
|
||||||
'points_paid': 0.0,
|
|
||||||
'stopped': not managed_stream.running,
|
|
||||||
'stream_hash': None,
|
|
||||||
'stream_name': None,
|
|
||||||
'suggested_file_name': None,
|
|
||||||
'sd_hash': None,
|
|
||||||
'mime_type': None,
|
|
||||||
'key': None,
|
|
||||||
'total_bytes_lower_bound': total_bytes_lower_bound,
|
|
||||||
'total_bytes': total_bytes,
|
|
||||||
'written_bytes': managed_stream.written_bytes,
|
|
||||||
'blobs_completed': None,
|
|
||||||
'blobs_in_stream': None,
|
|
||||||
'blobs_remaining': None,
|
|
||||||
'status': managed_stream.status,
|
|
||||||
'claim_id': managed_stream.claim_id,
|
|
||||||
'txid': managed_stream.txid,
|
|
||||||
'nout': managed_stream.nout,
|
|
||||||
'outpoint': managed_stream.outpoint,
|
|
||||||
'metadata': managed_stream.metadata,
|
|
||||||
'protobuf': managed_stream.metadata_protobuf,
|
|
||||||
'channel_claim_id': managed_stream.channel_claim_id,
|
|
||||||
'channel_name': managed_stream.channel_name,
|
|
||||||
'claim_name': managed_stream.claim_name,
|
|
||||||
'content_fee': managed_stream.content_fee,
|
|
||||||
'purchase_receipt': self.encode_output(managed_stream.purchase_receipt),
|
|
||||||
'added_on': managed_stream.added_on,
|
|
||||||
'height': tx_height,
|
|
||||||
'confirmations': (best_height + 1) - tx_height if tx_height > 0 else tx_height,
|
|
||||||
'timestamp': self.ledger.headers.estimated_timestamp(tx_height),
|
|
||||||
'is_fully_reflected': False,
|
|
||||||
'reflector_progress': False,
|
|
||||||
'uploading_to_reflector': False
|
|
||||||
}
|
|
||||||
if is_stream:
|
|
||||||
result.update({
|
|
||||||
'streaming_url': managed_stream.stream_url,
|
|
||||||
'stream_hash': managed_stream.stream_hash,
|
|
||||||
'stream_name': managed_stream.stream_name,
|
|
||||||
'suggested_file_name': managed_stream.suggested_file_name,
|
|
||||||
'sd_hash': managed_stream.descriptor.sd_hash,
|
|
||||||
'mime_type': managed_stream.mime_type,
|
|
||||||
'key': managed_stream.descriptor.key,
|
|
||||||
'blobs_completed': managed_stream.blobs_completed,
|
|
||||||
'blobs_in_stream': managed_stream.blobs_in_stream,
|
|
||||||
'blobs_remaining': managed_stream.blobs_remaining,
|
|
||||||
'is_fully_reflected': managed_stream.is_fully_reflected,
|
|
||||||
'reflector_progress': managed_stream.reflector_progress,
|
|
||||||
'uploading_to_reflector': managed_stream.uploading_to_reflector
|
|
||||||
})
|
|
||||||
else:
|
|
||||||
result.update({
|
|
||||||
'streaming_url': f'file://{managed_stream.full_path}',
|
|
||||||
})
|
|
||||||
if output_exists:
|
|
||||||
result.update({
|
|
||||||
'file_name': managed_stream.file_name,
|
|
||||||
'download_directory': managed_stream.download_directory,
|
|
||||||
'download_path': managed_stream.full_path,
|
|
||||||
})
|
|
||||||
return result
|
|
||||||
|
|
||||||
def encode_claim(self, claim):
|
|
||||||
encoded = getattr(claim, claim.claim_type).to_dict()
|
|
||||||
if 'public_key' in encoded:
|
|
||||||
encoded['public_key_id'] = self.ledger.public_key_to_address(
|
|
||||||
unhexlify(encoded['public_key'])
|
|
||||||
)
|
|
||||||
return encoded
|
|
|
@ -1,74 +0,0 @@
|
||||||
# pylint: skip-file
|
|
||||||
import os
|
|
||||||
import sys
|
|
||||||
import logging
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
def migrate_db(conf, start, end):
|
|
||||||
current = start
|
|
||||||
while current < end:
|
|
||||||
if current == 1:
|
|
||||||
from .migrate1to2 import do_migration
|
|
||||||
elif current == 2:
|
|
||||||
from .migrate2to3 import do_migration
|
|
||||||
elif current == 3:
|
|
||||||
from .migrate3to4 import do_migration
|
|
||||||
elif current == 4:
|
|
||||||
from .migrate4to5 import do_migration
|
|
||||||
elif current == 5:
|
|
||||||
from .migrate5to6 import do_migration
|
|
||||||
elif current == 6:
|
|
||||||
from .migrate6to7 import do_migration
|
|
||||||
elif current == 7:
|
|
||||||
from .migrate7to8 import do_migration
|
|
||||||
elif current == 8:
|
|
||||||
from .migrate8to9 import do_migration
|
|
||||||
elif current == 9:
|
|
||||||
from .migrate9to10 import do_migration
|
|
||||||
elif current == 10:
|
|
||||||
from .migrate10to11 import do_migration
|
|
||||||
elif current == 11:
|
|
||||||
from .migrate11to12 import do_migration
|
|
||||||
elif current == 12:
|
|
||||||
from .migrate12to13 import do_migration
|
|
||||||
elif current == 13:
|
|
||||||
from .migrate13to14 import do_migration
|
|
||||||
elif current == 14:
|
|
||||||
from .migrate14to15 import do_migration
|
|
||||||
elif current == 15:
|
|
||||||
from .migrate15to16 import do_migration
|
|
||||||
else:
|
|
||||||
raise Exception(f"DB migration of version {current} to {current+1} is not available")
|
|
||||||
try:
|
|
||||||
do_migration(conf)
|
|
||||||
except Exception:
|
|
||||||
log.exception("failed to migrate database")
|
|
||||||
if os.path.exists(os.path.join(conf.data_dir, "lbrynet.sqlite")):
|
|
||||||
backup_name = f"rev_{current}_unmigrated_database"
|
|
||||||
count = 0
|
|
||||||
while os.path.exists(os.path.join(conf.data_dir, backup_name + ".sqlite")):
|
|
||||||
count += 1
|
|
||||||
backup_name = f"rev_{current}_unmigrated_database_{count}"
|
|
||||||
backup_path = os.path.join(conf.data_dir, backup_name + ".sqlite")
|
|
||||||
os.rename(os.path.join(conf.data_dir, "lbrynet.sqlite"), backup_path)
|
|
||||||
log.info("made a backup of the unmigrated database: %s", backup_path)
|
|
||||||
if os.path.isfile(os.path.join(conf.data_dir, "db_revision")):
|
|
||||||
os.remove(os.path.join(conf.data_dir, "db_revision"))
|
|
||||||
return None
|
|
||||||
current += 1
|
|
||||||
log.info("successfully migrated the database from revision %i to %i", current - 1, current)
|
|
||||||
return None
|
|
||||||
|
|
||||||
|
|
||||||
def run_migration_script():
|
|
||||||
log_format = "(%(asctime)s)[%(filename)s:%(lineno)s] %(funcName)s(): %(message)s"
|
|
||||||
logging.basicConfig(level=logging.DEBUG, format=log_format, filename="migrator.log")
|
|
||||||
sys.stdout = open("migrator.out.log", 'w')
|
|
||||||
sys.stderr = open("migrator.err.log", 'w')
|
|
||||||
migrate_db(sys.argv[1], int(sys.argv[2]), int(sys.argv[3]))
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
run_migration_script()
|
|
|
@ -1,54 +0,0 @@
|
||||||
import sqlite3
|
|
||||||
import os
|
|
||||||
import binascii
|
|
||||||
|
|
||||||
|
|
||||||
def do_migration(conf):
|
|
||||||
db_path = os.path.join(conf.data_dir, "lbrynet.sqlite")
|
|
||||||
connection = sqlite3.connect(db_path)
|
|
||||||
cursor = connection.cursor()
|
|
||||||
|
|
||||||
current_columns = []
|
|
||||||
for col_info in cursor.execute("pragma table_info('file');").fetchall():
|
|
||||||
current_columns.append(col_info[1])
|
|
||||||
if 'content_fee' in current_columns or 'saved_file' in current_columns:
|
|
||||||
connection.close()
|
|
||||||
print("already migrated")
|
|
||||||
return
|
|
||||||
|
|
||||||
cursor.execute(
|
|
||||||
"pragma foreign_keys=off;"
|
|
||||||
)
|
|
||||||
|
|
||||||
cursor.execute("""
|
|
||||||
create table if not exists new_file (
|
|
||||||
stream_hash text primary key not null references stream,
|
|
||||||
file_name text,
|
|
||||||
download_directory text,
|
|
||||||
blob_data_rate real not null,
|
|
||||||
status text not null,
|
|
||||||
saved_file integer not null,
|
|
||||||
content_fee text
|
|
||||||
);
|
|
||||||
""")
|
|
||||||
for (stream_hash, file_name, download_dir, data_rate, status) in cursor.execute("select * from file").fetchall():
|
|
||||||
saved_file = 0
|
|
||||||
if download_dir != '{stream}' and file_name != '{stream}':
|
|
||||||
try:
|
|
||||||
if os.path.isfile(os.path.join(binascii.unhexlify(download_dir).decode(),
|
|
||||||
binascii.unhexlify(file_name).decode())):
|
|
||||||
saved_file = 1
|
|
||||||
else:
|
|
||||||
download_dir, file_name = None, None
|
|
||||||
except Exception:
|
|
||||||
download_dir, file_name = None, None
|
|
||||||
else:
|
|
||||||
download_dir, file_name = None, None
|
|
||||||
cursor.execute(
|
|
||||||
"insert into new_file values (?, ?, ?, ?, ?, ?, NULL)",
|
|
||||||
(stream_hash, file_name, download_dir, data_rate, status, saved_file)
|
|
||||||
)
|
|
||||||
cursor.execute("drop table file")
|
|
||||||
cursor.execute("alter table new_file rename to file")
|
|
||||||
connection.commit()
|
|
||||||
connection.close()
|
|
|
@ -1,69 +0,0 @@
|
||||||
import sqlite3
|
|
||||||
import os
|
|
||||||
import time
|
|
||||||
|
|
||||||
|
|
||||||
def do_migration(conf):
|
|
||||||
db_path = os.path.join(conf.data_dir, 'lbrynet.sqlite')
|
|
||||||
connection = sqlite3.connect(db_path)
|
|
||||||
connection.row_factory = sqlite3.Row
|
|
||||||
cursor = connection.cursor()
|
|
||||||
|
|
||||||
current_columns = []
|
|
||||||
for col_info in cursor.execute("pragma table_info('file');").fetchall():
|
|
||||||
current_columns.append(col_info[1])
|
|
||||||
|
|
||||||
if 'added_on' in current_columns:
|
|
||||||
connection.close()
|
|
||||||
print('already migrated')
|
|
||||||
return
|
|
||||||
|
|
||||||
# follow 12 step schema change procedure
|
|
||||||
cursor.execute("pragma foreign_keys=off")
|
|
||||||
|
|
||||||
# we don't have any indexes, views or triggers, so step 3 is skipped.
|
|
||||||
cursor.execute("drop table if exists new_file")
|
|
||||||
cursor.execute("""
|
|
||||||
create table if not exists new_file (
|
|
||||||
stream_hash text not null primary key references stream,
|
|
||||||
file_name text,
|
|
||||||
download_directory text,
|
|
||||||
blob_data_rate text not null,
|
|
||||||
status text not null,
|
|
||||||
saved_file integer not null,
|
|
||||||
content_fee text,
|
|
||||||
added_on integer not null
|
|
||||||
);
|
|
||||||
|
|
||||||
|
|
||||||
""")
|
|
||||||
|
|
||||||
# step 5: transfer content from old to new
|
|
||||||
select = "select * from file"
|
|
||||||
for (stream_hash, file_name, download_dir, blob_rate, status, saved_file, fee) \
|
|
||||||
in cursor.execute(select).fetchall():
|
|
||||||
added_on = int(time.time())
|
|
||||||
cursor.execute(
|
|
||||||
"insert into new_file values (?, ?, ?, ?, ?, ?, ?, ?)",
|
|
||||||
(stream_hash, file_name, download_dir, blob_rate, status, saved_file, fee, added_on)
|
|
||||||
)
|
|
||||||
|
|
||||||
# step 6: drop old table
|
|
||||||
cursor.execute("drop table file")
|
|
||||||
|
|
||||||
# step 7: rename new table to old table
|
|
||||||
cursor.execute("alter table new_file rename to file")
|
|
||||||
|
|
||||||
# step 8: we aren't using indexes, views or triggers so skip
|
|
||||||
# step 9: no views so skip
|
|
||||||
# step 10: foreign key check
|
|
||||||
cursor.execute("pragma foreign_key_check;")
|
|
||||||
|
|
||||||
# step 11: commit transaction
|
|
||||||
connection.commit()
|
|
||||||
|
|
||||||
# step 12: re-enable foreign keys
|
|
||||||
connection.execute("pragma foreign_keys=on;")
|
|
||||||
|
|
||||||
# done :)
|
|
||||||
connection.close()
|
|
|
@ -1,80 +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()
|
|
||||||
|
|
||||||
current_columns = []
|
|
||||||
for col_info in cursor.execute("pragma table_info('file');").fetchall():
|
|
||||||
current_columns.append(col_info[1])
|
|
||||||
if 'bt_infohash' in current_columns:
|
|
||||||
connection.close()
|
|
||||||
print("already migrated")
|
|
||||||
return
|
|
||||||
|
|
||||||
cursor.executescript("""
|
|
||||||
pragma foreign_keys=off;
|
|
||||||
|
|
||||||
create table if not exists torrent (
|
|
||||||
bt_infohash char(20) not null primary key,
|
|
||||||
tracker text,
|
|
||||||
length integer not null,
|
|
||||||
name text not null
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists torrent_node ( -- BEP-0005
|
|
||||||
bt_infohash char(20) not null references torrent,
|
|
||||||
host text not null,
|
|
||||||
port integer not null
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists torrent_tracker ( -- BEP-0012
|
|
||||||
bt_infohash char(20) not null references torrent,
|
|
||||||
tracker text not null
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists torrent_http_seed ( -- BEP-0017
|
|
||||||
bt_infohash char(20) not null references torrent,
|
|
||||||
http_seed text not null
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists new_file (
|
|
||||||
stream_hash char(96) references stream,
|
|
||||||
bt_infohash char(20) references torrent,
|
|
||||||
file_name text,
|
|
||||||
download_directory text,
|
|
||||||
blob_data_rate real not null,
|
|
||||||
status text not null,
|
|
||||||
saved_file integer not null,
|
|
||||||
content_fee text,
|
|
||||||
added_on integer not null
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists new_content_claim (
|
|
||||||
stream_hash char(96) references stream,
|
|
||||||
bt_infohash char(20) references torrent,
|
|
||||||
claim_outpoint text unique not null references claim
|
|
||||||
);
|
|
||||||
|
|
||||||
insert into new_file (stream_hash, bt_infohash, file_name, download_directory, blob_data_rate, status,
|
|
||||||
saved_file, content_fee, added_on) select
|
|
||||||
stream_hash, NULL, file_name, download_directory, blob_data_rate, status, saved_file, content_fee,
|
|
||||||
added_on
|
|
||||||
from file;
|
|
||||||
|
|
||||||
insert or ignore into new_content_claim (stream_hash, bt_infohash, claim_outpoint)
|
|
||||||
select stream_hash, NULL, claim_outpoint from content_claim;
|
|
||||||
|
|
||||||
drop table file;
|
|
||||||
drop table content_claim;
|
|
||||||
alter table new_file rename to file;
|
|
||||||
alter table new_content_claim rename to content_claim;
|
|
||||||
|
|
||||||
pragma foreign_keys=on;
|
|
||||||
""")
|
|
||||||
|
|
||||||
connection.commit()
|
|
||||||
connection.close()
|
|
|
@ -1,21 +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("""
|
|
||||||
create table if not exists peer (
|
|
||||||
node_id char(96) not null primary key,
|
|
||||||
address text not null,
|
|
||||||
udp_port integer not null,
|
|
||||||
tcp_port integer,
|
|
||||||
unique (address, udp_port)
|
|
||||||
);
|
|
||||||
""")
|
|
||||||
|
|
||||||
connection.commit()
|
|
||||||
connection.close()
|
|
|
@ -1,16 +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("""
|
|
||||||
alter table blob add column added_on integer not null default 0;
|
|
||||||
alter table blob add column is_mine integer not null default 1;
|
|
||||||
""")
|
|
||||||
|
|
||||||
connection.commit()
|
|
||||||
connection.close()
|
|
|
@ -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()
|
|
|
@ -1,20 +0,0 @@
|
||||||
import sqlite3
|
|
||||||
import os
|
|
||||||
|
|
||||||
|
|
||||||
def do_migration(conf):
|
|
||||||
db_path = os.path.join(conf.data_dir, "lbrynet.sqlite")
|
|
||||||
connection = sqlite3.connect(db_path)
|
|
||||||
cursor = connection.cursor()
|
|
||||||
|
|
||||||
query = "select stream_hash, sd_hash from main.stream"
|
|
||||||
for stream_hash, sd_hash in cursor.execute(query).fetchall():
|
|
||||||
head_blob_hash = cursor.execute(
|
|
||||||
"select blob_hash from stream_blob where position = 0 and stream_hash = ?",
|
|
||||||
(stream_hash,)
|
|
||||||
).fetchone()
|
|
||||||
if not head_blob_hash:
|
|
||||||
continue
|
|
||||||
cursor.execute("update blob set should_announce=1 where blob_hash in (?, ?)", (sd_hash, head_blob_hash[0],))
|
|
||||||
connection.commit()
|
|
||||||
connection.close()
|
|
|
@ -1,31 +0,0 @@
|
||||||
import logging
|
|
||||||
from aiohttp import web
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
def ensure_request_allowed(request, conf):
|
|
||||||
if is_request_allowed(request, conf):
|
|
||||||
return
|
|
||||||
if conf.allowed_origin:
|
|
||||||
log.warning(
|
|
||||||
"API requests with Origin '%s' are not allowed, "
|
|
||||||
"configuration 'allowed_origin' limits requests to: '%s'",
|
|
||||||
request.headers.get('Origin'), conf.allowed_origin
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
log.warning(
|
|
||||||
"API requests with Origin '%s' are not allowed, "
|
|
||||||
"update configuration 'allowed_origin' to enable this origin.",
|
|
||||||
request.headers.get('Origin')
|
|
||||||
)
|
|
||||||
raise web.HTTPForbidden()
|
|
||||||
|
|
||||||
|
|
||||||
def is_request_allowed(request, conf) -> bool:
|
|
||||||
origin = request.headers.get('Origin')
|
|
||||||
return (
|
|
||||||
origin is None or
|
|
||||||
origin == conf.allowed_origin or
|
|
||||||
conf.allowed_origin == '*'
|
|
||||||
)
|
|
|
@ -1,936 +0,0 @@
|
||||||
import os
|
|
||||||
import logging
|
|
||||||
import sqlite3
|
|
||||||
import typing
|
|
||||||
import asyncio
|
|
||||||
import binascii
|
|
||||||
import time
|
|
||||||
from typing import Optional
|
|
||||||
from lbry.wallet import SQLiteMixin
|
|
||||||
from lbry.conf import Config
|
|
||||||
from lbry.wallet.dewies import dewies_to_lbc, lbc_to_dewies
|
|
||||||
from lbry.wallet.transaction import Transaction, Output
|
|
||||||
from lbry.schema.claim import Claim
|
|
||||||
from lbry.dht.constants import DATA_EXPIRATION
|
|
||||||
from lbry.blob.blob_info import BlobInfo
|
|
||||||
|
|
||||||
if typing.TYPE_CHECKING:
|
|
||||||
from lbry.blob.blob_file import BlobFile
|
|
||||||
from lbry.stream.descriptor import StreamDescriptor
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
def calculate_effective_amount(amount: str, supports: typing.Optional[typing.List[typing.Dict]] = None) -> str:
|
|
||||||
return dewies_to_lbc(
|
|
||||||
lbc_to_dewies(amount) + sum([lbc_to_dewies(support['amount']) for support in supports])
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class StoredContentClaim:
|
|
||||||
def __init__(self, outpoint: Optional[str] = None, claim_id: Optional[str] = None, name: Optional[str] = None,
|
|
||||||
amount: Optional[int] = None, height: Optional[int] = None, serialized: Optional[str] = None,
|
|
||||||
channel_claim_id: Optional[str] = None, address: Optional[str] = None,
|
|
||||||
claim_sequence: Optional[int] = None, channel_name: Optional[str] = None):
|
|
||||||
self.claim_id = claim_id
|
|
||||||
self.outpoint = outpoint
|
|
||||||
self.claim_name = name
|
|
||||||
self.amount = amount
|
|
||||||
self.height = height
|
|
||||||
self.claim: typing.Optional[Claim] = None if not serialized else Claim.from_bytes(
|
|
||||||
binascii.unhexlify(serialized)
|
|
||||||
)
|
|
||||||
self.claim_address = address
|
|
||||||
self.claim_sequence = claim_sequence
|
|
||||||
self.channel_claim_id = channel_claim_id
|
|
||||||
self.channel_name = channel_name
|
|
||||||
|
|
||||||
@property
|
|
||||||
def txid(self) -> typing.Optional[str]:
|
|
||||||
return None if not self.outpoint else self.outpoint.split(":")[0]
|
|
||||||
|
|
||||||
@property
|
|
||||||
def nout(self) -> typing.Optional[int]:
|
|
||||||
return None if not self.outpoint else int(self.outpoint.split(":")[1])
|
|
||||||
|
|
||||||
def as_dict(self) -> typing.Dict:
|
|
||||||
return {
|
|
||||||
"name": self.claim_name,
|
|
||||||
"claim_id": self.claim_id,
|
|
||||||
"address": self.claim_address,
|
|
||||||
"claim_sequence": self.claim_sequence,
|
|
||||||
"value": self.claim,
|
|
||||||
"height": self.height,
|
|
||||||
"amount": dewies_to_lbc(self.amount),
|
|
||||||
"nout": self.nout,
|
|
||||||
"txid": self.txid,
|
|
||||||
"channel_claim_id": self.channel_claim_id,
|
|
||||||
"channel_name": self.channel_name
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def _get_content_claims(transaction: sqlite3.Connection, query: str,
|
|
||||||
source_hashes: typing.List[str]) -> typing.Dict[str, StoredContentClaim]:
|
|
||||||
claims = {}
|
|
||||||
for claim_info in _batched_select(transaction, query, source_hashes):
|
|
||||||
claims[claim_info[0]] = StoredContentClaim(*claim_info[1:])
|
|
||||||
return claims
|
|
||||||
|
|
||||||
|
|
||||||
def get_claims_from_stream_hashes(transaction: sqlite3.Connection,
|
|
||||||
stream_hashes: typing.List[str]) -> typing.Dict[str, StoredContentClaim]:
|
|
||||||
query = (
|
|
||||||
"select content_claim.stream_hash, c.*, case when c.channel_claim_id is not null then "
|
|
||||||
" (select claim_name from claim where claim_id==c.channel_claim_id) "
|
|
||||||
" else null end as channel_name "
|
|
||||||
" from content_claim "
|
|
||||||
" inner join claim c on c.claim_outpoint=content_claim.claim_outpoint and content_claim.stream_hash in {}"
|
|
||||||
" order by c.rowid desc"
|
|
||||||
)
|
|
||||||
return _get_content_claims(transaction, query, stream_hashes)
|
|
||||||
|
|
||||||
|
|
||||||
def get_claims_from_torrent_info_hashes(transaction: sqlite3.Connection,
|
|
||||||
info_hashes: typing.List[str]) -> typing.Dict[str, StoredContentClaim]:
|
|
||||||
query = (
|
|
||||||
"select content_claim.bt_infohash, c.*, case when c.channel_claim_id is not null then "
|
|
||||||
" (select claim_name from claim where claim_id==c.channel_claim_id) "
|
|
||||||
" else null end as channel_name "
|
|
||||||
" from content_claim "
|
|
||||||
" inner join claim c on c.claim_outpoint=content_claim.claim_outpoint and content_claim.bt_infohash in {}"
|
|
||||||
" order by c.rowid desc"
|
|
||||||
)
|
|
||||||
return _get_content_claims(transaction, query, info_hashes)
|
|
||||||
|
|
||||||
|
|
||||||
def _batched_select(transaction, query, parameters, batch_size=900):
|
|
||||||
for start_index in range(0, len(parameters), batch_size):
|
|
||||||
current_batch = parameters[start_index:start_index+batch_size]
|
|
||||||
bind = "({})".format(','.join(['?'] * len(current_batch)))
|
|
||||||
yield from transaction.execute(query.format(bind), current_batch)
|
|
||||||
|
|
||||||
|
|
||||||
def _get_lbry_file_stream_dict(rowid, added_on, stream_hash, file_name, download_dir, data_rate, status,
|
|
||||||
sd_hash, stream_key, stream_name, suggested_file_name, claim, saved_file,
|
|
||||||
raw_content_fee, fully_reflected):
|
|
||||||
return {
|
|
||||||
"rowid": rowid,
|
|
||||||
"added_on": added_on,
|
|
||||||
"stream_hash": stream_hash,
|
|
||||||
"file_name": file_name, # hex
|
|
||||||
"download_directory": download_dir, # hex
|
|
||||||
"blob_data_rate": data_rate,
|
|
||||||
"status": status,
|
|
||||||
"sd_hash": sd_hash,
|
|
||||||
"key": stream_key,
|
|
||||||
"stream_name": stream_name, # hex
|
|
||||||
"suggested_file_name": suggested_file_name, # hex
|
|
||||||
"claim": claim,
|
|
||||||
"saved_file": bool(saved_file),
|
|
||||||
"content_fee": None if not raw_content_fee else Transaction(
|
|
||||||
binascii.unhexlify(raw_content_fee)
|
|
||||||
),
|
|
||||||
"fully_reflected": fully_reflected
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def get_all_lbry_files(transaction: sqlite3.Connection) -> typing.List[typing.Dict]:
|
|
||||||
files = []
|
|
||||||
signed_claims = {}
|
|
||||||
for (rowid, stream_hash, _, file_name, download_dir, data_rate, status, saved_file, raw_content_fee,
|
|
||||||
added_on, _, sd_hash, stream_key, stream_name, suggested_file_name, *claim_args) in transaction.execute(
|
|
||||||
"select file.rowid, file.*, stream.*, c.*, "
|
|
||||||
" case when (SELECT 1 FROM reflected_stream r WHERE r.sd_hash=stream.sd_hash) "
|
|
||||||
" is null then 0 else 1 end as fully_reflected "
|
|
||||||
"from file inner join stream on file.stream_hash=stream.stream_hash "
|
|
||||||
"inner join content_claim cc on file.stream_hash=cc.stream_hash "
|
|
||||||
"inner join claim c on cc.claim_outpoint=c.claim_outpoint "
|
|
||||||
"order by c.rowid desc").fetchall():
|
|
||||||
claim_args, fully_reflected = tuple(claim_args[:-1]), claim_args[-1]
|
|
||||||
claim = StoredContentClaim(*claim_args)
|
|
||||||
if claim.channel_claim_id:
|
|
||||||
if claim.channel_claim_id not in signed_claims:
|
|
||||||
signed_claims[claim.channel_claim_id] = []
|
|
||||||
signed_claims[claim.channel_claim_id].append(claim)
|
|
||||||
files.append(
|
|
||||||
_get_lbry_file_stream_dict(
|
|
||||||
rowid, added_on, stream_hash, file_name, download_dir, data_rate, status,
|
|
||||||
sd_hash, stream_key, stream_name, suggested_file_name, claim, saved_file,
|
|
||||||
raw_content_fee, fully_reflected
|
|
||||||
)
|
|
||||||
)
|
|
||||||
for claim_name, claim_id in _batched_select(
|
|
||||||
transaction, "select c.claim_name, c.claim_id from claim c where c.claim_id in {}",
|
|
||||||
tuple(signed_claims.keys())):
|
|
||||||
for claim in signed_claims[claim_id]:
|
|
||||||
claim.channel_name = claim_name
|
|
||||||
return files
|
|
||||||
|
|
||||||
|
|
||||||
def store_stream(transaction: sqlite3.Connection, sd_blob: 'BlobFile', descriptor: 'StreamDescriptor'):
|
|
||||||
# add all blobs, except the last one, which is empty
|
|
||||||
transaction.executemany(
|
|
||||||
"insert or ignore into blob values (?, ?, ?, ?, ?, ?, ?, ?, ?)",
|
|
||||||
((blob.blob_hash, blob.length, 0, 0, "pending", 0, 0, blob.added_on, blob.is_mine)
|
|
||||||
for blob in (descriptor.blobs[:-1] if len(descriptor.blobs) > 1 else descriptor.blobs) + [sd_blob])
|
|
||||||
).fetchall()
|
|
||||||
# associate the blobs to the stream
|
|
||||||
transaction.execute("insert or ignore into stream values (?, ?, ?, ?, ?)",
|
|
||||||
(descriptor.stream_hash, sd_blob.blob_hash, descriptor.key,
|
|
||||||
binascii.hexlify(descriptor.stream_name.encode()).decode(),
|
|
||||||
binascii.hexlify(descriptor.suggested_file_name.encode()).decode())).fetchall()
|
|
||||||
# add the stream
|
|
||||||
transaction.executemany(
|
|
||||||
"insert or ignore into stream_blob values (?, ?, ?, ?)",
|
|
||||||
((descriptor.stream_hash, blob.blob_hash, blob.blob_num, blob.iv)
|
|
||||||
for blob in descriptor.blobs)
|
|
||||||
).fetchall()
|
|
||||||
# ensure should_announce is set regardless if insert was ignored
|
|
||||||
transaction.execute(
|
|
||||||
"update blob set should_announce=1 where blob_hash in (?)",
|
|
||||||
(sd_blob.blob_hash,)
|
|
||||||
).fetchall()
|
|
||||||
|
|
||||||
|
|
||||||
def delete_stream(transaction: sqlite3.Connection, descriptor: 'StreamDescriptor'):
|
|
||||||
blob_hashes = [(blob.blob_hash, ) for blob in descriptor.blobs[:-1]]
|
|
||||||
blob_hashes.append((descriptor.sd_hash, ))
|
|
||||||
transaction.execute("delete from content_claim where stream_hash=? ", (descriptor.stream_hash,)).fetchall()
|
|
||||||
transaction.execute("delete from file where stream_hash=? ", (descriptor.stream_hash,)).fetchall()
|
|
||||||
transaction.execute("delete from stream_blob where stream_hash=?", (descriptor.stream_hash,)).fetchall()
|
|
||||||
transaction.execute("delete from stream where stream_hash=? ", (descriptor.stream_hash,)).fetchall()
|
|
||||||
transaction.executemany("delete from blob where blob_hash=?", blob_hashes).fetchall()
|
|
||||||
|
|
||||||
|
|
||||||
def delete_torrent(transaction: sqlite3.Connection, bt_infohash: str):
|
|
||||||
transaction.execute("delete from content_claim where bt_infohash=?", (bt_infohash, )).fetchall()
|
|
||||||
transaction.execute("delete from torrent_tracker where bt_infohash=?", (bt_infohash,)).fetchall()
|
|
||||||
transaction.execute("delete from torrent_node where bt_infohash=?", (bt_infohash,)).fetchall()
|
|
||||||
transaction.execute("delete from torrent_http_seed where bt_infohash=?", (bt_infohash,)).fetchall()
|
|
||||||
transaction.execute("delete from file where bt_infohash=?", (bt_infohash,)).fetchall()
|
|
||||||
transaction.execute("delete from torrent where bt_infohash=?", (bt_infohash,)).fetchall()
|
|
||||||
|
|
||||||
|
|
||||||
def store_file(transaction: sqlite3.Connection, stream_hash: str, file_name: typing.Optional[str],
|
|
||||||
download_directory: typing.Optional[str], data_payment_rate: float, status: str,
|
|
||||||
content_fee: typing.Optional[Transaction], added_on: typing.Optional[int] = None) -> int:
|
|
||||||
if not file_name and not download_directory:
|
|
||||||
encoded_file_name, encoded_download_dir = None, None
|
|
||||||
else:
|
|
||||||
encoded_file_name = binascii.hexlify(file_name.encode()).decode()
|
|
||||||
encoded_download_dir = binascii.hexlify(download_directory.encode()).decode()
|
|
||||||
time_added = added_on or int(time.time())
|
|
||||||
transaction.execute(
|
|
||||||
"insert or replace into file values (?, NULL, ?, ?, ?, ?, ?, ?, ?)",
|
|
||||||
(stream_hash, encoded_file_name, encoded_download_dir, data_payment_rate, status,
|
|
||||||
1 if (file_name and download_directory and os.path.isfile(os.path.join(download_directory, file_name))) else 0,
|
|
||||||
None if not content_fee else binascii.hexlify(content_fee.raw).decode(), time_added)
|
|
||||||
).fetchall()
|
|
||||||
|
|
||||||
return transaction.execute("select rowid from file where stream_hash=?", (stream_hash, )).fetchone()[0]
|
|
||||||
|
|
||||||
|
|
||||||
class SQLiteStorage(SQLiteMixin):
|
|
||||||
CREATE_TABLES_QUERY = """
|
|
||||||
pragma foreign_keys=on;
|
|
||||||
pragma journal_mode=WAL;
|
|
||||||
|
|
||||||
create table if not exists blob (
|
|
||||||
blob_hash char(96) primary key not null,
|
|
||||||
blob_length integer not null,
|
|
||||||
next_announce_time integer not null,
|
|
||||||
should_announce integer not null default 0,
|
|
||||||
status text not null,
|
|
||||||
last_announced_time integer,
|
|
||||||
single_announce integer,
|
|
||||||
added_on integer not null,
|
|
||||||
is_mine integer not null default 0
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists stream (
|
|
||||||
stream_hash char(96) not null primary key,
|
|
||||||
sd_hash char(96) not null references blob,
|
|
||||||
stream_key text not null,
|
|
||||||
stream_name text not null,
|
|
||||||
suggested_filename text not null
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists stream_blob (
|
|
||||||
stream_hash char(96) not null references stream,
|
|
||||||
blob_hash char(96) references blob,
|
|
||||||
position integer not null,
|
|
||||||
iv char(32) not null,
|
|
||||||
primary key (stream_hash, blob_hash)
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists claim (
|
|
||||||
claim_outpoint text not null primary key,
|
|
||||||
claim_id char(40) not null,
|
|
||||||
claim_name text not null,
|
|
||||||
amount integer not null,
|
|
||||||
height integer not null,
|
|
||||||
serialized_metadata blob not null,
|
|
||||||
channel_claim_id text,
|
|
||||||
address text not null,
|
|
||||||
claim_sequence integer not null
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists torrent (
|
|
||||||
bt_infohash char(20) not null primary key,
|
|
||||||
tracker text,
|
|
||||||
length integer not null,
|
|
||||||
name text not null
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists torrent_node ( -- BEP-0005
|
|
||||||
bt_infohash char(20) not null references torrent,
|
|
||||||
host text not null,
|
|
||||||
port integer not null
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists torrent_tracker ( -- BEP-0012
|
|
||||||
bt_infohash char(20) not null references torrent,
|
|
||||||
tracker text not null
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists torrent_http_seed ( -- BEP-0017
|
|
||||||
bt_infohash char(20) not null references torrent,
|
|
||||||
http_seed text not null
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists file (
|
|
||||||
stream_hash char(96) references stream,
|
|
||||||
bt_infohash char(20) references torrent,
|
|
||||||
file_name text,
|
|
||||||
download_directory text,
|
|
||||||
blob_data_rate real not null,
|
|
||||||
status text not null,
|
|
||||||
saved_file integer not null,
|
|
||||||
content_fee text,
|
|
||||||
added_on integer not null
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists content_claim (
|
|
||||||
stream_hash char(96) references stream,
|
|
||||||
bt_infohash char(20) references torrent,
|
|
||||||
claim_outpoint text unique not null references claim
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists support (
|
|
||||||
support_outpoint text not null primary key,
|
|
||||||
claim_id text not null,
|
|
||||||
amount integer not null,
|
|
||||||
address text not null
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists reflected_stream (
|
|
||||||
sd_hash text not null,
|
|
||||||
reflector_address text not null,
|
|
||||||
timestamp integer,
|
|
||||||
primary key (sd_hash, reflector_address)
|
|
||||||
);
|
|
||||||
|
|
||||||
create table if not exists peer (
|
|
||||||
node_id char(96) not null primary key,
|
|
||||||
address text not null,
|
|
||||||
udp_port integer not null,
|
|
||||||
tcp_port integer,
|
|
||||||
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):
|
|
||||||
super().__init__(path)
|
|
||||||
self.conf = conf
|
|
||||||
self.content_claim_callbacks = {}
|
|
||||||
self.loop = loop or asyncio.get_event_loop()
|
|
||||||
self.time_getter = time_getter or time.time
|
|
||||||
|
|
||||||
async def run_and_return_one_or_none(self, query, *args):
|
|
||||||
for row in await self.db.execute_fetchall(query, args):
|
|
||||||
if len(row) == 1:
|
|
||||||
return row[0]
|
|
||||||
return row
|
|
||||||
|
|
||||||
async def run_and_return_list(self, query, *args):
|
|
||||||
rows = list(await self.db.execute_fetchall(query, args))
|
|
||||||
return [col[0] for col in rows] if rows else []
|
|
||||||
|
|
||||||
# # # # # # # # # blob functions # # # # # # # # #
|
|
||||||
|
|
||||||
async def add_blobs(self, *blob_hashes_and_lengths: typing.Tuple[str, int, int, int], finished=False):
|
|
||||||
def _add_blobs(transaction: sqlite3.Connection):
|
|
||||||
transaction.executemany(
|
|
||||||
"insert or ignore into blob values (?, ?, ?, ?, ?, ?, ?, ?, ?)",
|
|
||||||
(
|
|
||||||
(blob_hash, length, 0, 0, "pending" if not finished else "finished", 0, 0, added_on, is_mine)
|
|
||||||
for blob_hash, length, added_on, is_mine in blob_hashes_and_lengths
|
|
||||||
)
|
|
||||||
).fetchall()
|
|
||||||
if finished:
|
|
||||||
transaction.executemany(
|
|
||||||
"update blob set status='finished' where blob.blob_hash=?", (
|
|
||||||
(blob_hash, ) for blob_hash, _, _, _ in blob_hashes_and_lengths
|
|
||||||
)
|
|
||||||
).fetchall()
|
|
||||||
return await self.db.run(_add_blobs)
|
|
||||||
|
|
||||||
def get_blob_status(self, blob_hash: str):
|
|
||||||
return self.run_and_return_one_or_none(
|
|
||||||
"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(transaction: sqlite3.Connection):
|
|
||||||
last_announced = self.time_getter()
|
|
||||||
return transaction.executemany(
|
|
||||||
"update blob set next_announce_time=?, last_announced_time=?, single_announce=0 "
|
|
||||||
"where blob_hash=?",
|
|
||||||
((int(last_announced + (DATA_EXPIRATION / 2)), int(last_announced), blob_hash)
|
|
||||||
for blob_hash in blob_hashes)
|
|
||||||
).fetchall()
|
|
||||||
return self.db.run(_update_last_announced_blobs)
|
|
||||||
|
|
||||||
def should_single_announce_blobs(self, blob_hashes, immediate=False):
|
|
||||||
def set_single_announce(transaction):
|
|
||||||
now = int(self.time_getter())
|
|
||||||
for blob_hash in blob_hashes:
|
|
||||||
if immediate:
|
|
||||||
transaction.execute(
|
|
||||||
"update blob set single_announce=1, next_announce_time=? "
|
|
||||||
"where blob_hash=? and status='finished'", (int(now), blob_hash)
|
|
||||||
).fetchall()
|
|
||||||
else:
|
|
||||||
transaction.execute(
|
|
||||||
"update blob set single_announce=1 where blob_hash=? and status='finished'", (blob_hash,)
|
|
||||||
).fetchall()
|
|
||||||
return self.db.run(set_single_announce)
|
|
||||||
|
|
||||||
def get_blobs_to_announce(self):
|
|
||||||
def get_and_update(transaction):
|
|
||||||
timestamp = int(self.time_getter())
|
|
||||||
if self.conf.announce_head_and_sd_only:
|
|
||||||
r = transaction.execute(
|
|
||||||
"select blob_hash from blob "
|
|
||||||
"where blob_hash is not null and "
|
|
||||||
"(should_announce=1 or single_announce=1) and next_announce_time<? and status='finished' "
|
|
||||||
"order by next_announce_time asc limit ?",
|
|
||||||
(timestamp, int(self.conf.concurrent_blob_announcers * 10))
|
|
||||||
).fetchall()
|
|
||||||
else:
|
|
||||||
r = transaction.execute(
|
|
||||||
"select blob_hash from blob where blob_hash is not null "
|
|
||||||
"and next_announce_time<? and status='finished' "
|
|
||||||
"order by next_announce_time asc limit ?",
|
|
||||||
(timestamp, int(self.conf.concurrent_blob_announcers * 10))
|
|
||||||
).fetchall()
|
|
||||||
return [b[0] for b in r]
|
|
||||||
return self.db.run(get_and_update)
|
|
||||||
|
|
||||||
def delete_blobs_from_db(self, blob_hashes):
|
|
||||||
def delete_blobs(transaction):
|
|
||||||
transaction.executemany(
|
|
||||||
"delete from blob where blob_hash=?;", ((blob_hash,) for blob_hash in blob_hashes)
|
|
||||||
).fetchall()
|
|
||||||
return self.db.run_with_foreign_keys_disabled(delete_blobs)
|
|
||||||
|
|
||||||
def get_all_blob_hashes(self):
|
|
||||||
return self.run_and_return_list("select blob_hash from blob")
|
|
||||||
|
|
||||||
async def get_stored_blobs(self, is_mine: bool, is_network_blob=False):
|
|
||||||
is_mine = 1 if is_mine else 0
|
|
||||||
if is_network_blob:
|
|
||||||
return await self.db.execute_fetchall(
|
|
||||||
"select blob.blob_hash, blob.blob_length, blob.added_on "
|
|
||||||
"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,)
|
|
||||||
)
|
|
||||||
|
|
||||||
sd_blobs = await self.db.execute_fetchall(
|
|
||||||
"select blob.blob_hash, blob.blob_length, blob.added_on "
|
|
||||||
"from blob join stream on blob.blob_hash=stream.sd_hash join file using (stream_hash) "
|
|
||||||
"where blob.is_mine=? order by blob.added_on asc",
|
|
||||||
(is_mine,)
|
|
||||||
)
|
|
||||||
content_blobs = await self.db.execute_fetchall(
|
|
||||||
"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):
|
|
||||||
is_mine = 1 if is_mine else 0
|
|
||||||
await self.db.execute_fetchall(
|
|
||||||
"update blob set is_mine = ? where blob_hash in ("
|
|
||||||
" select blob_hash from blob natural join stream_blob natural join stream where sd_hash = ?"
|
|
||||||
") OR blob_hash = ?", (is_mine, sd_hash, sd_hash)
|
|
||||||
)
|
|
||||||
|
|
||||||
def sync_missing_blobs(self, blob_files: typing.Set[str]) -> typing.Awaitable[typing.Set[str]]:
|
|
||||||
def _sync_blobs(transaction: sqlite3.Connection) -> typing.Set[str]:
|
|
||||||
finished_blob_hashes = tuple(
|
|
||||||
blob_hash for (blob_hash, ) in transaction.execute(
|
|
||||||
"select blob_hash from blob where status='finished'"
|
|
||||||
).fetchall()
|
|
||||||
)
|
|
||||||
finished_blobs_set = set(finished_blob_hashes)
|
|
||||||
to_update_set = finished_blobs_set.difference(blob_files)
|
|
||||||
transaction.executemany(
|
|
||||||
"update blob set status='pending' where blob_hash=?",
|
|
||||||
((blob_hash, ) for blob_hash in to_update_set)
|
|
||||||
).fetchall()
|
|
||||||
return blob_files.intersection(finished_blobs_set)
|
|
||||||
return self.db.run(_sync_blobs)
|
|
||||||
|
|
||||||
# # # # # # # # # stream functions # # # # # # # # #
|
|
||||||
|
|
||||||
async def stream_exists(self, sd_hash: str) -> bool:
|
|
||||||
streams = await self.run_and_return_one_or_none("select stream_hash from stream where sd_hash=?", sd_hash)
|
|
||||||
return streams is not None
|
|
||||||
|
|
||||||
async def file_exists(self, sd_hash: str) -> bool:
|
|
||||||
streams = await self.run_and_return_one_or_none("select f.stream_hash from file f "
|
|
||||||
"inner join stream s on "
|
|
||||||
"s.stream_hash=f.stream_hash and s.sd_hash=?", sd_hash)
|
|
||||||
return streams is not None
|
|
||||||
|
|
||||||
def store_stream(self, sd_blob: 'BlobFile', descriptor: 'StreamDescriptor'):
|
|
||||||
return self.db.run(store_stream, sd_blob, descriptor)
|
|
||||||
|
|
||||||
def get_blobs_for_stream(self, stream_hash, only_completed=False) -> typing.Awaitable[typing.List[BlobInfo]]:
|
|
||||||
def _get_blobs_for_stream(transaction):
|
|
||||||
crypt_blob_infos = []
|
|
||||||
stream_blobs = transaction.execute(
|
|
||||||
"select s.blob_hash, s.position, s.iv, b.added_on "
|
|
||||||
"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, )
|
|
||||||
).fetchall()
|
|
||||||
if only_completed:
|
|
||||||
lengths = transaction.execute(
|
|
||||||
"select b.blob_hash, b.blob_length from blob b "
|
|
||||||
"inner join stream_blob s ON b.blob_hash=s.blob_hash and b.status='finished' and s.stream_hash=?",
|
|
||||||
(stream_hash, )
|
|
||||||
).fetchall()
|
|
||||||
else:
|
|
||||||
lengths = transaction.execute(
|
|
||||||
"select b.blob_hash, b.blob_length from blob b "
|
|
||||||
"inner join stream_blob s ON b.blob_hash=s.blob_hash and s.stream_hash=?",
|
|
||||||
(stream_hash, )
|
|
||||||
).fetchall()
|
|
||||||
|
|
||||||
blob_length_dict = {}
|
|
||||||
for blob_hash, length in lengths:
|
|
||||||
blob_length_dict[blob_hash] = length
|
|
||||||
|
|
||||||
current_time = time.time()
|
|
||||||
for blob_hash, position, iv, added_on in stream_blobs:
|
|
||||||
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))
|
|
||||||
if not blob_hash:
|
|
||||||
break
|
|
||||||
return crypt_blob_infos
|
|
||||||
return self.db.run(_get_blobs_for_stream)
|
|
||||||
|
|
||||||
def get_sd_blob_hash_for_stream(self, stream_hash):
|
|
||||||
return self.run_and_return_one_or_none(
|
|
||||||
"select sd_hash from stream where stream_hash=?", stream_hash
|
|
||||||
)
|
|
||||||
|
|
||||||
def get_stream_hash_for_sd_hash(self, sd_blob_hash):
|
|
||||||
return self.run_and_return_one_or_none(
|
|
||||||
"select stream_hash from stream where sd_hash = ?", sd_blob_hash
|
|
||||||
)
|
|
||||||
|
|
||||||
def delete_stream(self, descriptor: 'StreamDescriptor'):
|
|
||||||
return self.db.run_with_foreign_keys_disabled(delete_stream, descriptor)
|
|
||||||
|
|
||||||
async def delete_torrent(self, bt_infohash: str):
|
|
||||||
return await self.db.run(delete_torrent, bt_infohash)
|
|
||||||
|
|
||||||
# # # # # # # # # file stuff # # # # # # # # #
|
|
||||||
|
|
||||||
def save_downloaded_file(self, stream_hash: str, file_name: typing.Optional[str],
|
|
||||||
download_directory: typing.Optional[str], data_payment_rate: float,
|
|
||||||
content_fee: typing.Optional[Transaction] = None,
|
|
||||||
added_on: typing.Optional[int] = None) -> typing.Awaitable[int]:
|
|
||||||
return self.save_published_file(
|
|
||||||
stream_hash, file_name, download_directory, data_payment_rate, status="running",
|
|
||||||
content_fee=content_fee, added_on=added_on
|
|
||||||
)
|
|
||||||
|
|
||||||
def save_published_file(self, stream_hash: str, file_name: typing.Optional[str],
|
|
||||||
download_directory: typing.Optional[str], data_payment_rate: float,
|
|
||||||
status: str = "finished",
|
|
||||||
content_fee: typing.Optional[Transaction] = None,
|
|
||||||
added_on: typing.Optional[int] = None) -> typing.Awaitable[int]:
|
|
||||||
return self.db.run(store_file, stream_hash, file_name, download_directory, data_payment_rate, status,
|
|
||||||
content_fee, added_on)
|
|
||||||
|
|
||||||
async def update_manually_removed_files_since_last_run(self):
|
|
||||||
"""
|
|
||||||
Update files that have been removed from the downloads directory since the last run
|
|
||||||
"""
|
|
||||||
def update_manually_removed_files(transaction: sqlite3.Connection):
|
|
||||||
files = {}
|
|
||||||
query = "select stream_hash, download_directory, file_name from file where saved_file=1 " \
|
|
||||||
"and stream_hash is not null"
|
|
||||||
for (stream_hash, download_directory, file_name) in transaction.execute(query).fetchall():
|
|
||||||
if download_directory and file_name:
|
|
||||||
files[stream_hash] = download_directory, file_name
|
|
||||||
return files
|
|
||||||
|
|
||||||
def detect_removed(files):
|
|
||||||
return [
|
|
||||||
stream_hash for stream_hash, (download_directory, file_name) in files.items()
|
|
||||||
if not os.path.isfile(os.path.join(binascii.unhexlify(download_directory).decode(),
|
|
||||||
binascii.unhexlify(file_name).decode()))
|
|
||||||
]
|
|
||||||
|
|
||||||
def update_db_removed(transaction: sqlite3.Connection, removed):
|
|
||||||
query = "update file set file_name=null, download_directory=null, saved_file=0 where stream_hash in {}"
|
|
||||||
for cur in _batched_select(transaction, query, removed):
|
|
||||||
cur.fetchall()
|
|
||||||
|
|
||||||
stream_and_file = await self.db.run(update_manually_removed_files)
|
|
||||||
removed = await self.loop.run_in_executor(None, detect_removed, stream_and_file)
|
|
||||||
if removed:
|
|
||||||
await self.db.run(update_db_removed, removed)
|
|
||||||
|
|
||||||
def get_all_lbry_files(self) -> typing.Awaitable[typing.List[typing.Dict]]:
|
|
||||||
return self.db.run(get_all_lbry_files)
|
|
||||||
|
|
||||||
def change_file_status(self, stream_hash: str, new_status: str):
|
|
||||||
log.debug("update file status %s -> %s", stream_hash, new_status)
|
|
||||||
return self.db.execute_fetchall("update file set status=? where stream_hash=?", (new_status, stream_hash))
|
|
||||||
|
|
||||||
def stop_all_files(self):
|
|
||||||
log.debug("stopping all files")
|
|
||||||
return self.db.execute_fetchall("update file set status=?", ("stopped",))
|
|
||||||
|
|
||||||
async def change_file_download_dir_and_file_name(self, stream_hash: str, download_dir: typing.Optional[str],
|
|
||||||
file_name: typing.Optional[str]):
|
|
||||||
if not file_name or not download_dir:
|
|
||||||
encoded_file_name, encoded_download_dir = None, None
|
|
||||||
else:
|
|
||||||
encoded_file_name = binascii.hexlify(file_name.encode()).decode()
|
|
||||||
encoded_download_dir = binascii.hexlify(download_dir.encode()).decode()
|
|
||||||
return await self.db.execute_fetchall("update file set download_directory=?, file_name=? where stream_hash=?", (
|
|
||||||
encoded_download_dir, encoded_file_name, stream_hash,
|
|
||||||
))
|
|
||||||
|
|
||||||
async def save_content_fee(self, stream_hash: str, content_fee: Transaction):
|
|
||||||
return await self.db.execute_fetchall("update file set content_fee=? where stream_hash=?", (
|
|
||||||
binascii.hexlify(content_fee.raw), stream_hash,
|
|
||||||
))
|
|
||||||
|
|
||||||
async def set_saved_file(self, stream_hash: str):
|
|
||||||
return await self.db.execute_fetchall("update file set saved_file=1 where stream_hash=?", (
|
|
||||||
stream_hash,
|
|
||||||
))
|
|
||||||
|
|
||||||
async def clear_saved_file(self, stream_hash: str):
|
|
||||||
return await self.db.execute_fetchall("update file set saved_file=0 where stream_hash=?", (
|
|
||||||
stream_hash,
|
|
||||||
))
|
|
||||||
|
|
||||||
async def recover_streams(self, descriptors_and_sds: typing.List[typing.Tuple['StreamDescriptor', 'BlobFile',
|
|
||||||
typing.Optional[Transaction]]],
|
|
||||||
download_directory: str):
|
|
||||||
def _recover(transaction: sqlite3.Connection):
|
|
||||||
stream_hashes = [x[0].stream_hash for x in descriptors_and_sds]
|
|
||||||
for descriptor, sd_blob, content_fee in descriptors_and_sds:
|
|
||||||
content_claim = transaction.execute(
|
|
||||||
"select * from content_claim where stream_hash=?", (descriptor.stream_hash, )
|
|
||||||
).fetchone()
|
|
||||||
delete_stream(transaction, descriptor) # this will also delete the content claim
|
|
||||||
store_stream(transaction, sd_blob, descriptor)
|
|
||||||
store_file(transaction, descriptor.stream_hash, os.path.basename(descriptor.suggested_file_name),
|
|
||||||
download_directory, 0.0, 'stopped', content_fee=content_fee)
|
|
||||||
if content_claim:
|
|
||||||
transaction.execute("insert or ignore into content_claim values (?, ?, ?)", content_claim)
|
|
||||||
transaction.executemany(
|
|
||||||
"update file set status='stopped' where stream_hash=?",
|
|
||||||
((stream_hash, ) for stream_hash in stream_hashes)
|
|
||||||
).fetchall()
|
|
||||||
download_dir = binascii.hexlify(self.conf.download_dir.encode()).decode()
|
|
||||||
transaction.executemany(
|
|
||||||
"update file set download_directory=? where stream_hash=?",
|
|
||||||
((download_dir, stream_hash) for stream_hash in stream_hashes)
|
|
||||||
).fetchall()
|
|
||||||
await self.db.run_with_foreign_keys_disabled(_recover)
|
|
||||||
|
|
||||||
def get_all_stream_hashes(self):
|
|
||||||
return self.run_and_return_list("select stream_hash from stream")
|
|
||||||
|
|
||||||
# # # # # # # # # support functions # # # # # # # # #
|
|
||||||
|
|
||||||
def save_supports(self, claim_id_to_supports: dict):
|
|
||||||
# TODO: add 'address' to support items returned for a claim from lbrycrdd and lbryum-server
|
|
||||||
def _save_support(transaction):
|
|
||||||
bind = "({})".format(','.join(['?'] * len(claim_id_to_supports)))
|
|
||||||
transaction.execute(
|
|
||||||
f"delete from support where claim_id in {bind}", tuple(claim_id_to_supports.keys())
|
|
||||||
).fetchall()
|
|
||||||
for claim_id, supports in claim_id_to_supports.items():
|
|
||||||
for support in supports:
|
|
||||||
transaction.execute(
|
|
||||||
"insert into support values (?, ?, ?, ?)",
|
|
||||||
("%s:%i" % (support['txid'], support['nout']), claim_id, lbc_to_dewies(support['amount']),
|
|
||||||
support.get('address', ""))
|
|
||||||
).fetchall()
|
|
||||||
return self.db.run(_save_support)
|
|
||||||
|
|
||||||
def get_supports(self, *claim_ids):
|
|
||||||
def _format_support(outpoint, supported_id, amount, address):
|
|
||||||
return {
|
|
||||||
"txid": outpoint.split(":")[0],
|
|
||||||
"nout": int(outpoint.split(":")[1]),
|
|
||||||
"claim_id": supported_id,
|
|
||||||
"amount": dewies_to_lbc(amount),
|
|
||||||
"address": address,
|
|
||||||
}
|
|
||||||
|
|
||||||
def _get_supports(transaction):
|
|
||||||
return [
|
|
||||||
_format_support(*support_info)
|
|
||||||
for support_info in _batched_select(
|
|
||||||
transaction,
|
|
||||||
"select * from support where claim_id in {}",
|
|
||||||
claim_ids
|
|
||||||
)
|
|
||||||
]
|
|
||||||
|
|
||||||
return self.db.run(_get_supports)
|
|
||||||
|
|
||||||
# # # # # # # # # claim functions # # # # # # # # #
|
|
||||||
|
|
||||||
async def save_claims(self, claim_infos):
|
|
||||||
claim_id_to_supports = {}
|
|
||||||
update_file_callbacks = []
|
|
||||||
|
|
||||||
def _save_claims(transaction):
|
|
||||||
content_claims_to_update = []
|
|
||||||
for claim_info in claim_infos:
|
|
||||||
outpoint = "%s:%i" % (claim_info['txid'], claim_info['nout'])
|
|
||||||
claim_id = claim_info['claim_id']
|
|
||||||
name = claim_info['name']
|
|
||||||
amount = lbc_to_dewies(claim_info['amount'])
|
|
||||||
height = claim_info['height']
|
|
||||||
address = claim_info['address']
|
|
||||||
sequence = claim_info['claim_sequence']
|
|
||||||
certificate_id = claim_info['value'].signing_channel_id
|
|
||||||
try:
|
|
||||||
source_hash = claim_info['value'].stream.source.sd_hash
|
|
||||||
except (AttributeError, ValueError):
|
|
||||||
source_hash = None
|
|
||||||
serialized = binascii.hexlify(claim_info['value'].to_bytes())
|
|
||||||
transaction.execute(
|
|
||||||
"insert or replace into claim values (?, ?, ?, ?, ?, ?, ?, ?, ?)",
|
|
||||||
(outpoint, claim_id, name, amount, height, serialized, certificate_id, address, sequence)
|
|
||||||
).fetchall()
|
|
||||||
# if this response doesn't have support info don't overwrite the existing
|
|
||||||
# support info
|
|
||||||
if 'supports' in claim_info:
|
|
||||||
claim_id_to_supports[claim_id] = claim_info['supports']
|
|
||||||
if not source_hash:
|
|
||||||
continue
|
|
||||||
stream_hash = transaction.execute(
|
|
||||||
"select file.stream_hash from stream "
|
|
||||||
"inner join file on file.stream_hash=stream.stream_hash where sd_hash=?", (source_hash,)
|
|
||||||
).fetchone()
|
|
||||||
if not stream_hash:
|
|
||||||
continue
|
|
||||||
stream_hash = stream_hash[0]
|
|
||||||
known_outpoint = transaction.execute(
|
|
||||||
"select claim_outpoint from content_claim where stream_hash=?", (stream_hash,)
|
|
||||||
).fetchone()
|
|
||||||
known_claim_id = transaction.execute(
|
|
||||||
"select claim_id from claim "
|
|
||||||
"inner join content_claim c3 ON claim.claim_outpoint=c3.claim_outpoint "
|
|
||||||
"where c3.stream_hash=?", (stream_hash,)
|
|
||||||
).fetchone()
|
|
||||||
if not known_claim_id:
|
|
||||||
content_claims_to_update.append((stream_hash, outpoint))
|
|
||||||
elif known_outpoint != outpoint:
|
|
||||||
content_claims_to_update.append((stream_hash, outpoint))
|
|
||||||
for stream_hash, outpoint in content_claims_to_update:
|
|
||||||
self._save_content_claim(transaction, outpoint, stream_hash)
|
|
||||||
if stream_hash in self.content_claim_callbacks:
|
|
||||||
update_file_callbacks.append(self.content_claim_callbacks[stream_hash]())
|
|
||||||
|
|
||||||
await self.db.run(_save_claims)
|
|
||||||
if update_file_callbacks:
|
|
||||||
await asyncio.wait(map(asyncio.create_task, update_file_callbacks))
|
|
||||||
if claim_id_to_supports:
|
|
||||||
await self.save_supports(claim_id_to_supports)
|
|
||||||
|
|
||||||
def save_claim_from_output(self, ledger, *outputs: Output):
|
|
||||||
return self.save_claims([{
|
|
||||||
"claim_id": output.claim_id,
|
|
||||||
"name": output.claim_name,
|
|
||||||
"amount": dewies_to_lbc(output.amount),
|
|
||||||
"address": output.get_address(ledger),
|
|
||||||
"txid": output.tx_ref.id,
|
|
||||||
"nout": output.position,
|
|
||||||
"value": output.claim,
|
|
||||||
"height": output.tx_ref.height,
|
|
||||||
"claim_sequence": -1,
|
|
||||||
} for output in outputs])
|
|
||||||
|
|
||||||
def save_claims_for_resolve(self, claim_infos):
|
|
||||||
to_save = {}
|
|
||||||
for info in claim_infos:
|
|
||||||
if 'value' in info:
|
|
||||||
if info['value']:
|
|
||||||
to_save[info['claim_id']] = info
|
|
||||||
else:
|
|
||||||
for key in ('certificate', 'claim'):
|
|
||||||
if info.get(key, {}).get('value'):
|
|
||||||
to_save[info[key]['claim_id']] = info[key]
|
|
||||||
return self.save_claims(to_save.values())
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def _save_content_claim(transaction, claim_outpoint, stream_hash=None, bt_infohash=None):
|
|
||||||
assert stream_hash or bt_infohash
|
|
||||||
# get the claim id and serialized metadata
|
|
||||||
claim_info = transaction.execute(
|
|
||||||
"select claim_id, serialized_metadata from claim where claim_outpoint=?", (claim_outpoint,)
|
|
||||||
).fetchone()
|
|
||||||
if not claim_info:
|
|
||||||
raise Exception("claim not found")
|
|
||||||
new_claim_id, claim = claim_info[0], Claim.from_bytes(binascii.unhexlify(claim_info[1]))
|
|
||||||
|
|
||||||
# certificate claims should not be in the content_claim table
|
|
||||||
if not claim.is_stream:
|
|
||||||
raise Exception("claim does not contain a stream")
|
|
||||||
|
|
||||||
# get the known sd hash for this stream
|
|
||||||
known_sd_hash = transaction.execute(
|
|
||||||
"select sd_hash from stream where stream_hash=?", (stream_hash,)
|
|
||||||
).fetchone()
|
|
||||||
if not known_sd_hash:
|
|
||||||
raise Exception("stream not found")
|
|
||||||
# check the claim contains the same sd hash
|
|
||||||
if known_sd_hash[0] != claim.stream.source.sd_hash:
|
|
||||||
raise Exception("stream mismatch")
|
|
||||||
|
|
||||||
# if there is a current claim associated to the file, check that the new claim is an update to it
|
|
||||||
current_associated_content = transaction.execute(
|
|
||||||
"select claim_outpoint from content_claim where stream_hash=?", (stream_hash,)
|
|
||||||
).fetchone()
|
|
||||||
if current_associated_content:
|
|
||||||
current_associated_claim_id = transaction.execute(
|
|
||||||
"select claim_id from claim where claim_outpoint=?", current_associated_content
|
|
||||||
).fetchone()[0]
|
|
||||||
if current_associated_claim_id != new_claim_id:
|
|
||||||
raise Exception(
|
|
||||||
f"mismatching claim ids when updating stream {current_associated_claim_id} vs {new_claim_id}"
|
|
||||||
)
|
|
||||||
|
|
||||||
# update the claim associated to the file
|
|
||||||
transaction.execute("delete from content_claim where stream_hash=?", (stream_hash, )).fetchall()
|
|
||||||
transaction.execute(
|
|
||||||
"insert into content_claim values (?, NULL, ?)", (stream_hash, claim_outpoint)
|
|
||||||
).fetchall()
|
|
||||||
|
|
||||||
async def save_content_claim(self, stream_hash, claim_outpoint):
|
|
||||||
await self.db.run(self._save_content_claim, claim_outpoint, stream_hash)
|
|
||||||
# update corresponding ManagedEncryptedFileDownloader object
|
|
||||||
if stream_hash in self.content_claim_callbacks:
|
|
||||||
await self.content_claim_callbacks[stream_hash]()
|
|
||||||
|
|
||||||
async def save_torrent_content_claim(self, bt_infohash, claim_outpoint, length, name):
|
|
||||||
def _save_torrent(transaction):
|
|
||||||
transaction.execute(
|
|
||||||
"insert or replace into torrent values (?, NULL, ?, ?)", (bt_infohash, length, name)
|
|
||||||
).fetchall()
|
|
||||||
transaction.execute(
|
|
||||||
"insert or replace into content_claim values (NULL, ?, ?)", (bt_infohash, claim_outpoint)
|
|
||||||
).fetchall()
|
|
||||||
await self.db.run(_save_torrent)
|
|
||||||
# update corresponding ManagedEncryptedFileDownloader object
|
|
||||||
if bt_infohash in self.content_claim_callbacks:
|
|
||||||
await self.content_claim_callbacks[bt_infohash]()
|
|
||||||
|
|
||||||
async def get_content_claim(self, stream_hash: str, include_supports: typing.Optional[bool] = True) -> typing.Dict:
|
|
||||||
claims = await self.db.run(get_claims_from_stream_hashes, [stream_hash])
|
|
||||||
claim = None
|
|
||||||
if claims:
|
|
||||||
claim = claims[stream_hash].as_dict()
|
|
||||||
if include_supports:
|
|
||||||
supports = await self.get_supports(claim['claim_id'])
|
|
||||||
claim['supports'] = supports
|
|
||||||
claim['effective_amount'] = calculate_effective_amount(claim['amount'], supports)
|
|
||||||
return claim
|
|
||||||
|
|
||||||
async def get_content_claim_for_torrent(self, bt_infohash):
|
|
||||||
claims = await self.db.run(get_claims_from_torrent_info_hashes, [bt_infohash])
|
|
||||||
return claims[bt_infohash].as_dict() if claims else None
|
|
||||||
|
|
||||||
# # # # # # # # # reflector functions # # # # # # # # #
|
|
||||||
|
|
||||||
def update_reflected_stream(self, sd_hash, reflector_address, success=True):
|
|
||||||
if success:
|
|
||||||
return self.db.execute_fetchall(
|
|
||||||
"insert or replace into reflected_stream values (?, ?, ?)",
|
|
||||||
(sd_hash, reflector_address, self.time_getter())
|
|
||||||
)
|
|
||||||
return self.db.execute_fetchall(
|
|
||||||
"delete from reflected_stream where sd_hash=? and reflector_address=?",
|
|
||||||
(sd_hash, reflector_address)
|
|
||||||
)
|
|
||||||
|
|
||||||
def get_streams_to_re_reflect(self):
|
|
||||||
return self.run_and_return_list(
|
|
||||||
"select s.sd_hash from stream s "
|
|
||||||
"left outer join reflected_stream r on s.sd_hash=r.sd_hash "
|
|
||||||
"where r.timestamp is null or r.timestamp < ?",
|
|
||||||
int(self.time_getter()) - 86400
|
|
||||||
)
|
|
||||||
|
|
||||||
# # # # # # # # # # dht functions # # # # # # # # # # #
|
|
||||||
async def get_persisted_kademlia_peers(self) -> typing.List[typing.Tuple[bytes, str, int, int]]:
|
|
||||||
query = 'select node_id, address, udp_port, tcp_port from peer'
|
|
||||||
return [(binascii.unhexlify(n), a, u, t) for n, a, u, t in await self.db.execute_fetchall(query)]
|
|
||||||
|
|
||||||
async def save_kademlia_peers(self, peers: typing.List['KademliaPeer']):
|
|
||||||
def _save_kademlia_peers(transaction: sqlite3.Connection):
|
|
||||||
transaction.execute('delete from peer').fetchall()
|
|
||||||
transaction.executemany(
|
|
||||||
'insert into peer(node_id, address, udp_port, tcp_port) values (?, ?, ?, ?)',
|
|
||||||
((binascii.hexlify(p.node_id), p.address, p.udp_port, p.tcp_port) for p in peers)
|
|
||||||
).fetchall()
|
|
||||||
return await self.db.run(_save_kademlia_peers)
|
|
|
@ -1,29 +0,0 @@
|
||||||
import platform
|
|
||||||
import os
|
|
||||||
import logging.handlers
|
|
||||||
|
|
||||||
from lbry import build_info, __version__ as lbrynet_version
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
def get_platform() -> dict:
|
|
||||||
os_system = platform.system()
|
|
||||||
if os.environ and 'ANDROID_ARGUMENT' in os.environ:
|
|
||||||
os_system = 'android'
|
|
||||||
d = {
|
|
||||||
"processor": platform.processor(),
|
|
||||||
"python_version": platform.python_version(),
|
|
||||||
"platform": platform.platform(),
|
|
||||||
"os_release": platform.release(),
|
|
||||||
"os_system": os_system,
|
|
||||||
"lbrynet_version": lbrynet_version,
|
|
||||||
"version": lbrynet_version,
|
|
||||||
"build": build_info.BUILD, # CI server sets this during build step
|
|
||||||
}
|
|
||||||
if d["os_system"] == "Linux":
|
|
||||||
import distro # pylint: disable=import-outside-toplevel
|
|
||||||
d["distro"] = distro.info()
|
|
||||||
d["desktop"] = os.environ.get('XDG_CURRENT_DESKTOP', 'Unknown')
|
|
||||||
|
|
||||||
return d
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Reference in a new issue