api accepts urls, resolves them, returns seeder count
This commit is contained in:
parent
9ed985d6ec
commit
e2634974e7
3 changed files with 93 additions and 26 deletions
|
@ -286,7 +286,7 @@ class TreeRoutingTable:
|
||||||
to_pop = [i for i, bucket in enumerate(self.buckets) if len(bucket) == 0]
|
to_pop = [i for i, bucket in enumerate(self.buckets) if len(bucket) == 0]
|
||||||
if not to_pop:
|
if not to_pop:
|
||||||
return
|
return
|
||||||
log.info("%s: join buckets %i", bytes.hex(self._parent_node_id)[:8], len(to_pop))
|
log.debug("%s: join buckets %i", bytes.hex(self._parent_node_id)[:8], len(to_pop))
|
||||||
bucket_index_to_pop = to_pop[0]
|
bucket_index_to_pop = to_pop[0]
|
||||||
assert len(self.buckets[bucket_index_to_pop]) == 0
|
assert len(self.buckets[bucket_index_to_pop]) == 0
|
||||||
can_go_lower = bucket_index_to_pop - 1 >= 0
|
can_go_lower = bucket_index_to_pop - 1 >= 0
|
||||||
|
|
|
@ -5,8 +5,10 @@ from binascii import hexlify
|
||||||
from itertools import chain
|
from itertools import chain
|
||||||
|
|
||||||
from lbry.error import ResolveCensoredError
|
from lbry.error import ResolveCensoredError
|
||||||
from lbry.schema.types.v2.result_pb2 import Outputs as OutputsMessage
|
from lbry.schema.types.v2.result_pb2 import \
|
||||||
from lbry.schema.types.v2.result_pb2 import Error as ErrorMessage
|
Outputs as OutputsMessage, \
|
||||||
|
Output as OutputMessage, \
|
||||||
|
Error as ErrorMessage
|
||||||
|
|
||||||
INVALID = ErrorMessage.Code.Name(ErrorMessage.INVALID)
|
INVALID = ErrorMessage.Code.Name(ErrorMessage.INVALID)
|
||||||
NOT_FOUND = ErrorMessage.Code.Name(ErrorMessage.NOT_FOUND)
|
NOT_FOUND = ErrorMessage.Code.Name(ErrorMessage.NOT_FOUND)
|
||||||
|
@ -70,7 +72,7 @@ class Outputs:
|
||||||
|
|
||||||
__slots__ = 'txos', 'extra_txos', 'txs', 'offset', 'total', 'blocked', 'blocked_total'
|
__slots__ = 'txos', 'extra_txos', 'txs', 'offset', 'total', 'blocked', 'blocked_total'
|
||||||
|
|
||||||
def __init__(self, txos: List, extra_txos: List, txs: set,
|
def __init__(self, txos: List[OutputMessage], extra_txos: List, txs: set,
|
||||||
offset: int, total: int, blocked: List, blocked_total: int):
|
offset: int, total: int, blocked: List, blocked_total: int):
|
||||||
self.txos = txos
|
self.txos = txos
|
||||||
self.txs = txs
|
self.txs = txs
|
||||||
|
|
|
@ -1,18 +1,23 @@
|
||||||
#!/usr/bin/env python
|
#!/usr/bin/env python
|
||||||
|
|
||||||
import asyncio
|
import asyncio
|
||||||
|
import json
|
||||||
import logging
|
import logging
|
||||||
import signal
|
|
||||||
import time
|
|
||||||
import sqlite3
|
|
||||||
import pickle
|
import pickle
|
||||||
|
import signal
|
||||||
|
import sqlite3
|
||||||
|
import time
|
||||||
from os import path
|
from os import path
|
||||||
from pprint import pprint
|
from pprint import pprint
|
||||||
|
from urllib.parse import unquote
|
||||||
|
|
||||||
from aioupnp import upnp, fault as upnpfault
|
from aioupnp import upnp, fault as upnpfault
|
||||||
from aiohttp import web
|
from aiohttp import web
|
||||||
import json
|
|
||||||
|
|
||||||
|
|
||||||
|
from lbry.wallet.network import ClientSession
|
||||||
|
from lbry.schema.result import Outputs
|
||||||
|
from lbry.wallet.transaction import Transaction
|
||||||
|
from binascii import hexlify, unhexlify
|
||||||
from lbry.dht import node, peer
|
from lbry.dht import node, peer
|
||||||
|
|
||||||
log = logging.getLogger("lbry")
|
log = logging.getLogger("lbry")
|
||||||
|
@ -35,7 +40,11 @@ async def main():
|
||||||
db.execute('CREATE TABLE IF NOT EXISTS announce (local_id TEXT, hash TEXT, node_id TEXT, ip TEXT, port INT, timestamp INT)')
|
db.execute('CREATE TABLE IF NOT EXISTS announce (local_id TEXT, hash TEXT, node_id TEXT, ip TEXT, port INT, timestamp INT)')
|
||||||
db.execute('CREATE UNIQUE INDEX IF NOT EXISTS node_id_hash_idx ON announce (node_id, hash)')
|
db.execute('CREATE UNIQUE INDEX IF NOT EXISTS node_id_hash_idx ON announce (node_id, hash)')
|
||||||
|
|
||||||
asyncio.create_task(run_web_api(loop, db))
|
spv_host = 'spv13.lbry.com'
|
||||||
|
wallet_client = ClientSession(network=None, server=(spv_host, 50001))
|
||||||
|
await wallet_client.create_connection()
|
||||||
|
|
||||||
|
asyncio.create_task(run_web_api(loop, db, wallet_client))
|
||||||
|
|
||||||
num_nodes = 128
|
num_nodes = 128
|
||||||
u = await upnp.UPnP.discover()
|
u = await upnp.UPnP.discover()
|
||||||
|
@ -58,6 +67,11 @@ async def main():
|
||||||
# print(f"{local_node_id[:8]}: {method} from {bytes.hex(node_id)} ({ip})")
|
# print(f"{local_node_id[:8]}: {method} from {bytes.hex(node_id)} ({ip})")
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
if len(args)< 5:
|
||||||
|
print(f'malformed args to Store')
|
||||||
|
pprint(args)
|
||||||
|
continue
|
||||||
|
|
||||||
blob_hash, token, port, original_publisher_id, age = args[:5]
|
blob_hash, token, port, original_publisher_id, age = args[:5]
|
||||||
print(f"STORE to {local_node_id[:8]} from {bytes.hex(node_id)[:8]} ({ip}) for blob {bytes.hex(blob_hash)[:8]}")
|
print(f"STORE to {local_node_id[:8]} from {bytes.hex(node_id)[:8]} ({ip}) for blob {bytes.hex(blob_hash)[:8]}")
|
||||||
|
|
||||||
|
@ -74,7 +88,7 @@ async def main():
|
||||||
db.commit()
|
db.commit()
|
||||||
cur.close()
|
cur.close()
|
||||||
except sqlite3.Error as err:
|
except sqlite3.Error as err:
|
||||||
print("failed insert", err)
|
print("failed sqlite insert", err)
|
||||||
finally:
|
finally:
|
||||||
print("shutting down")
|
print("shutting down")
|
||||||
for n in nodes:
|
for n in nodes:
|
||||||
|
@ -99,6 +113,7 @@ async def main():
|
||||||
# pprint(state.datastore)
|
# pprint(state.datastore)
|
||||||
print(f'{node_id[:8]}: saved {len(state.routing_table_peers)} rt peers, {len(state.datastore)} in store')
|
print(f'{node_id[:8]}: saved {len(state.routing_table_peers)} rt peers, {len(state.datastore)} in store')
|
||||||
pickle.dump(state, f)
|
pickle.dump(state, f)
|
||||||
|
await wallet_client.close()
|
||||||
db.close()
|
db.close()
|
||||||
|
|
||||||
|
|
||||||
|
@ -159,7 +174,7 @@ async def start_nodes(loop, num_nodes, external_ip, state_dir):
|
||||||
|
|
||||||
|
|
||||||
async def drain_events(n, q):
|
async def drain_events(n, q):
|
||||||
print(f'drain started on {bytes.hex(n.protocol.node_id)[:8]}')
|
# print(f'drain started on {bytes.hex(n.protocol.node_id)[:8]}')
|
||||||
while True:
|
while True:
|
||||||
(node_id, ip, method, args) = await n.protocol.event_queue.get()
|
(node_id, ip, method, args) = await n.protocol.event_queue.get()
|
||||||
try:
|
try:
|
||||||
|
@ -168,12 +183,15 @@ async def drain_events(n, q):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
async def run_web_api(loop, db):
|
async def run_web_api(loop, db, wallet_client):
|
||||||
|
try:
|
||||||
app = web.Application(loop=loop)
|
app = web.Application(loop=loop)
|
||||||
app['db'] = db
|
app['db'] = db
|
||||||
|
app['wallet_client'] = wallet_client
|
||||||
app.add_routes([
|
app.add_routes([
|
||||||
web.get('/', api_handler),
|
web.get('/', api_handler),
|
||||||
web.get('/seeds/{hash}', seeds_handler),
|
web.get('/seeds/hash/{hash}', seeds_handler),
|
||||||
|
web.get('/seeds/url/{url}', url_handler),
|
||||||
])
|
])
|
||||||
# server = web.Server(api_handler, loop=loop)
|
# server = web.Server(api_handler, loop=loop)
|
||||||
# runner = web.ServerRunner(server)
|
# runner = web.ServerRunner(server)
|
||||||
|
@ -181,7 +199,9 @@ async def run_web_api(loop, db):
|
||||||
await runner.setup()
|
await runner.setup()
|
||||||
site = web.TCPSite(runner, 'localhost', 8080)
|
site = web.TCPSite(runner, 'localhost', 8080)
|
||||||
await site.start()
|
await site.start()
|
||||||
# web.run_app(app)
|
except Exception as err:
|
||||||
|
pprint(err)
|
||||||
|
await runner.cleanup()
|
||||||
|
|
||||||
|
|
||||||
async def seeds_handler(request):
|
async def seeds_handler(request):
|
||||||
|
@ -198,9 +218,54 @@ async def seeds_handler(request):
|
||||||
return web.Response(text=json.dumps({'error': err})+"\n")
|
return web.Response(text=json.dumps({'error': err})+"\n")
|
||||||
|
|
||||||
|
|
||||||
|
async def url_handler(request):
|
||||||
|
url = unquote(request.match_info['url'])
|
||||||
|
log.warning(url)
|
||||||
|
db = request.app['db']
|
||||||
|
wallet_client = request.app['wallet_client']
|
||||||
|
|
||||||
|
try:
|
||||||
|
sd_hash = await get_sd_hash(wallet_client, url)
|
||||||
|
if sd_hash is None:
|
||||||
|
return web.Response(text=json.dumps({'error': 'Could not get sd hash for url', 'url': url})+"\n")
|
||||||
|
seeds = get_seeds(db, sd_hash)
|
||||||
|
return web.Response(text=json.dumps({'url': url, 'sd_hash': sd_hash, 'seeds': seeds})+"\n")
|
||||||
|
except Exception as err:
|
||||||
|
return web.Response(text=json.dumps({'error': err})+"\n")
|
||||||
|
|
||||||
|
|
||||||
|
def get_seeds(db, blobhash):
|
||||||
|
cur = db.cursor()
|
||||||
|
c = cur.execute(
|
||||||
|
"select count(distinct(node_id)) from announce where hash = ? and timestamp > strftime('%s','now','-1 day')",
|
||||||
|
(blobhash,)
|
||||||
|
).fetchone()[0]
|
||||||
|
cur.close()
|
||||||
|
return c
|
||||||
|
|
||||||
|
|
||||||
|
async def get_sd_hash(wallet_client, url):
|
||||||
|
try:
|
||||||
|
resolved_txos = Outputs.from_base64(await wallet_client.send_request('blockchain.claimtrie.resolve', [url]))
|
||||||
|
if not resolved_txos.txos:
|
||||||
|
return None
|
||||||
|
raw_txs = await wallet_client.send_request('blockchain.transaction.get_batch', [txid for (txid, height) in resolved_txos.txs])
|
||||||
|
txo_proto = resolved_txos.txos[0]
|
||||||
|
txid = txo_proto.tx_hash[::-1].hex()
|
||||||
|
raw_tx_hex, _ = raw_txs[txid]
|
||||||
|
txo = Transaction(bytes.fromhex(raw_tx_hex)).outputs[txo_proto.nout]
|
||||||
|
return txo.claim.stream.source.sd_hash
|
||||||
|
except Exception as err: # claim is not a stream, stream has no source, protobuf err, etc
|
||||||
|
if isinstance(err, asyncio.CancelledError):
|
||||||
|
raise err
|
||||||
|
log.exception("failed to get sd_hash")
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
async def api_handler(request):
|
async def api_handler(request):
|
||||||
return web.Response(text="tracker OK")
|
return web.Response(text="tracker OK")
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
try:
|
try:
|
||||||
asyncio.run(main())
|
asyncio.run(main())
|
||||||
|
|
Loading…
Add table
Reference in a new issue