2021-01-20 01:41:54 -03:00
|
|
|
import argparse
|
|
|
|
import asyncio
|
2021-02-12 14:41:03 -03:00
|
|
|
import logging
|
2021-02-16 12:52:32 -03:00
|
|
|
import os
|
2021-01-20 01:41:54 -03:00
|
|
|
from collections import namedtuple
|
2021-01-27 01:43:06 -03:00
|
|
|
from multiprocessing import Process
|
2021-01-20 01:41:54 -03:00
|
|
|
|
2021-06-15 16:12:23 -04:00
|
|
|
import sqlite3
|
2021-01-20 01:41:54 -03:00
|
|
|
from elasticsearch import AsyncElasticsearch
|
|
|
|
from elasticsearch.helpers import async_bulk
|
2021-03-29 13:16:49 -04:00
|
|
|
from lbry.wallet.server.env import Env
|
|
|
|
from lbry.wallet.server.coin import LBC
|
2021-05-07 12:42:52 -04:00
|
|
|
from lbry.wallet.server.db.elasticsearch.search import extract_doc, SearchIndex, IndexVersionMismatch
|
2021-01-20 01:41:54 -03:00
|
|
|
|
|
|
|
|
2021-05-11 21:38:05 -03:00
|
|
|
async def get_all(db, shard_num, shards_total, limit=0, index_name='claims'):
|
2021-02-12 14:41:03 -03:00
|
|
|
logging.info("shard %d starting", shard_num)
|
2021-01-20 01:41:54 -03:00
|
|
|
|
2021-06-15 16:12:23 -04:00
|
|
|
def namedtuple_factory(cursor, row):
|
|
|
|
Row = namedtuple('Row', (d[0] for d in cursor.description))
|
|
|
|
return Row(*row)
|
|
|
|
db.row_factory = namedtuple_factory
|
2021-02-02 17:11:13 -03:00
|
|
|
total = db.execute(f"select count(*) as total from claim where height % {shards_total} = {shard_num};").fetchone()[0]
|
2021-01-20 01:41:54 -03:00
|
|
|
for num, claim in enumerate(db.execute(f"""
|
2021-08-09 15:55:43 -03:00
|
|
|
SELECT claimtrie.claim_hash as is_controlling,
|
|
|
|
claimtrie.last_take_over_height,
|
|
|
|
(select group_concat(tag, ',,') from tag where tag.claim_hash in (claim.claim_hash, claim.reposted_claim_hash)) as tags,
|
|
|
|
(select group_concat(language, ' ') from language where language.claim_hash in (claim.claim_hash, claim.reposted_claim_hash)) as languages,
|
|
|
|
cr.has_source as reposted_has_source,
|
|
|
|
cr.claim_type as reposted_claim_type,
|
|
|
|
cr.stream_type as reposted_stream_type,
|
|
|
|
cr.media_type as reposted_media_type,
|
|
|
|
cr.duration as reposted_duration,
|
|
|
|
cr.fee_amount as reposted_fee_amount,
|
|
|
|
cr.fee_currency as reposted_fee_currency,
|
|
|
|
claim.*
|
|
|
|
FROM claim LEFT JOIN claimtrie USING (claim_hash) LEFT JOIN claim cr ON cr.claim_hash=claim.reposted_claim_hash
|
|
|
|
WHERE claim.height % {shards_total} = {shard_num}
|
|
|
|
ORDER BY claim.height desc
|
2021-01-20 01:41:54 -03:00
|
|
|
""")):
|
|
|
|
claim = dict(claim._asdict())
|
2021-03-25 04:46:21 -03:00
|
|
|
claim['has_source'] = bool(claim.pop('reposted_has_source') or claim['has_source'])
|
2021-08-09 15:55:43 -03:00
|
|
|
claim['stream_type'] = claim.pop('reposted_stream_type') or claim['stream_type']
|
|
|
|
claim['media_type'] = claim.pop('reposted_media_type') or claim['media_type']
|
|
|
|
claim['fee_amount'] = claim.pop('reposted_fee_amount') or claim['fee_amount']
|
|
|
|
claim['fee_currency'] = claim.pop('reposted_fee_currency') or claim['fee_currency']
|
|
|
|
claim['duration'] = claim.pop('reposted_duration') or claim['duration']
|
2021-01-20 01:41:54 -03:00
|
|
|
claim['censor_type'] = 0
|
2021-07-20 18:09:39 -03:00
|
|
|
claim['censoring_channel_id'] = None
|
2021-01-26 22:33:17 -03:00
|
|
|
claim['tags'] = claim['tags'].split(',,') if claim['tags'] else []
|
2021-01-20 01:41:54 -03:00
|
|
|
claim['languages'] = claim['languages'].split(' ') if claim['languages'] else []
|
2021-01-26 22:26:45 -03:00
|
|
|
if num % 10_000 == 0:
|
2021-02-12 14:41:03 -03:00
|
|
|
logging.info("%d/%d", num, total)
|
2021-05-11 21:38:05 -03:00
|
|
|
yield extract_doc(claim, index_name)
|
2021-02-17 01:09:12 -03:00
|
|
|
if 0 < limit <= num:
|
|
|
|
break
|
2021-01-20 01:41:54 -03:00
|
|
|
|
|
|
|
|
2021-05-11 21:38:05 -03:00
|
|
|
async def consume(producer, index_name):
|
2021-03-29 13:16:49 -04:00
|
|
|
env = Env(LBC)
|
|
|
|
logging.info("ES sync host: %s:%i", env.elastic_host, env.elastic_port)
|
|
|
|
es = AsyncElasticsearch([{'host': env.elastic_host, 'port': env.elastic_port}])
|
2021-02-11 23:10:30 -05:00
|
|
|
try:
|
|
|
|
await async_bulk(es, producer, request_timeout=120)
|
2021-05-12 00:21:03 -03:00
|
|
|
await es.indices.refresh(index=index_name)
|
2021-02-11 23:10:30 -05:00
|
|
|
finally:
|
|
|
|
await es.close()
|
|
|
|
|
|
|
|
|
2021-05-11 21:38:05 -03:00
|
|
|
async def make_es_index(index=None):
|
2021-03-29 13:16:49 -04:00
|
|
|
env = Env(LBC)
|
2021-05-11 21:38:05 -03:00
|
|
|
if index is None:
|
|
|
|
index = SearchIndex('', elastic_host=env.elastic_host, elastic_port=env.elastic_port)
|
2021-05-07 12:42:52 -04:00
|
|
|
|
2021-02-11 23:10:30 -05:00
|
|
|
try:
|
2021-02-12 14:41:03 -03:00
|
|
|
return await index.start()
|
2021-05-07 12:42:52 -04:00
|
|
|
except IndexVersionMismatch as err:
|
|
|
|
logging.info(
|
|
|
|
"dropping ES search index (version %s) for upgrade to version %s", err.got_version, err.expected_version
|
|
|
|
)
|
|
|
|
await index.delete_index()
|
2021-05-12 00:21:03 -03:00
|
|
|
await index.stop()
|
2021-05-07 12:42:52 -04:00
|
|
|
return await index.start()
|
2021-02-11 23:10:30 -05:00
|
|
|
finally:
|
2021-02-12 14:41:03 -03:00
|
|
|
index.stop()
|
2021-01-24 23:19:28 -03:00
|
|
|
|
|
|
|
|
2021-05-11 21:38:05 -03:00
|
|
|
async def run(db_path, clients, blocks, shard, index_name='claims'):
|
2021-06-15 16:51:50 -04:00
|
|
|
db = sqlite3.connect(db_path, isolation_level=None, check_same_thread=False, uri=True)
|
2021-06-15 16:12:23 -04:00
|
|
|
db.execute('pragma journal_mode=wal;')
|
|
|
|
db.execute('pragma temp_store=memory;')
|
|
|
|
producer = get_all(db, shard, clients, limit=blocks, index_name=index_name)
|
2021-05-11 21:38:05 -03:00
|
|
|
await asyncio.gather(*(consume(producer, index_name=index_name) for _ in range(min(8, clients))))
|
2021-01-27 01:43:06 -03:00
|
|
|
|
2021-02-11 23:10:30 -05:00
|
|
|
|
2021-01-27 01:43:06 -03:00
|
|
|
def __run(args, shard):
|
2021-05-11 21:38:05 -03:00
|
|
|
asyncio.run(run(args.db_path, args.clients, args.blocks, shard))
|
2021-01-27 01:43:06 -03:00
|
|
|
|
|
|
|
|
2021-02-11 23:10:30 -05:00
|
|
|
def run_elastic_sync():
|
2021-02-12 14:41:03 -03:00
|
|
|
logging.basicConfig(level=logging.INFO)
|
2021-03-29 13:16:49 -04:00
|
|
|
logging.getLogger('aiohttp').setLevel(logging.WARNING)
|
|
|
|
logging.getLogger('elasticsearch').setLevel(logging.WARNING)
|
|
|
|
|
2021-02-12 14:41:03 -03:00
|
|
|
logging.info('lbry.server starting')
|
2021-03-24 17:07:17 -03:00
|
|
|
parser = argparse.ArgumentParser(prog="lbry-hub-elastic-sync")
|
2021-01-27 01:43:06 -03:00
|
|
|
parser.add_argument("db_path", type=str)
|
|
|
|
parser.add_argument("-c", "--clients", type=int, default=16)
|
2021-02-17 01:09:12 -03:00
|
|
|
parser.add_argument("-b", "--blocks", type=int, default=0)
|
|
|
|
parser.add_argument("-f", "--force", default=False, action='store_true')
|
2021-01-27 01:43:06 -03:00
|
|
|
args = parser.parse_args()
|
|
|
|
processes = []
|
2021-02-11 23:10:30 -05:00
|
|
|
|
2021-02-17 01:09:12 -03:00
|
|
|
if not args.force and not os.path.exists(args.db_path):
|
2021-02-16 12:52:32 -03:00
|
|
|
logging.info("DB path doesnt exist")
|
|
|
|
return
|
|
|
|
|
2021-02-17 01:09:12 -03:00
|
|
|
if not args.force and not asyncio.run(make_es_index()):
|
2021-02-12 14:41:03 -03:00
|
|
|
logging.info("ES is already initialized")
|
2021-02-11 23:10:30 -05:00
|
|
|
return
|
2021-01-27 01:43:06 -03:00
|
|
|
for i in range(args.clients):
|
|
|
|
processes.append(Process(target=__run, args=(args, i)))
|
|
|
|
processes[-1].start()
|
|
|
|
for process in processes:
|
|
|
|
process.join()
|
|
|
|
process.close()
|