2021-10-20 19:34:55 -04:00
|
|
|
import os
|
2021-01-20 01:41:54 -03:00
|
|
|
import argparse
|
|
|
|
import asyncio
|
2021-02-12 14:41:03 -03:00
|
|
|
import logging
|
2021-01-20 01:41:54 -03:00
|
|
|
from elasticsearch import AsyncElasticsearch
|
2021-10-19 14:00:39 -04:00
|
|
|
from elasticsearch.helpers import async_streaming_bulk
|
2021-03-29 13:16:49 -04:00
|
|
|
from lbry.wallet.server.env import Env
|
2021-06-17 21:22:23 -04:00
|
|
|
from lbry.wallet.server.leveldb import LevelDB
|
2021-08-06 14:11:28 -04:00
|
|
|
from lbry.wallet.server.db.elasticsearch.search import SearchIndex, IndexVersionMismatch
|
|
|
|
from lbry.wallet.server.db.elasticsearch.constants import ALL_FIELDS
|
2021-01-20 01:41:54 -03:00
|
|
|
|
|
|
|
|
2021-10-20 19:34:55 -04:00
|
|
|
async def get_recent_claims(env, index_name='claims', db=None):
|
2021-10-19 14:00:39 -04:00
|
|
|
need_open = db is None
|
|
|
|
db = db or LevelDB(env)
|
|
|
|
try:
|
2021-10-20 19:34:55 -04:00
|
|
|
if need_open:
|
|
|
|
await db.open_dbs()
|
|
|
|
db_state = db.prefix_db.db_state.get()
|
|
|
|
if db_state.es_sync_height == db_state.height:
|
|
|
|
return
|
2021-10-19 14:00:39 -04:00
|
|
|
cnt = 0
|
|
|
|
touched_claims = set()
|
|
|
|
deleted_claims = set()
|
2021-10-20 19:34:55 -04:00
|
|
|
for height in range(db_state.es_sync_height, db_state.height + 1):
|
2021-10-19 14:00:39 -04:00
|
|
|
touched_or_deleted = db.prefix_db.touched_or_deleted.get(height)
|
|
|
|
touched_claims.update(touched_or_deleted.touched_claims)
|
|
|
|
deleted_claims.update(touched_or_deleted.deleted_claims)
|
|
|
|
touched_claims.difference_update(deleted_claims)
|
|
|
|
|
|
|
|
for deleted in deleted_claims:
|
|
|
|
yield {
|
|
|
|
'_index': index_name,
|
|
|
|
'_op_type': 'delete',
|
|
|
|
'_id': deleted.hex()
|
|
|
|
}
|
|
|
|
for touched in touched_claims:
|
|
|
|
claim = db.claim_producer(touched)
|
|
|
|
if claim:
|
|
|
|
yield {
|
|
|
|
'doc': {key: value for key, value in claim.items() if key in ALL_FIELDS},
|
|
|
|
'_id': claim['claim_id'],
|
|
|
|
'_index': index_name,
|
|
|
|
'_op_type': 'update',
|
|
|
|
'doc_as_upsert': True
|
|
|
|
}
|
|
|
|
cnt += 1
|
|
|
|
else:
|
|
|
|
logging.warning("could not sync claim %s", touched.hex())
|
|
|
|
if cnt % 10000 == 0:
|
2021-10-20 11:40:15 -04:00
|
|
|
logging.info("%i claims sent to ES", cnt)
|
2021-10-20 19:34:55 -04:00
|
|
|
|
|
|
|
db.es_sync_height = db.db_height
|
|
|
|
db.write_db_state()
|
|
|
|
db.prefix_db.unsafe_commit()
|
|
|
|
db.assert_db_state()
|
|
|
|
|
2021-10-20 11:40:15 -04:00
|
|
|
logging.info("finished sending %i claims to ES, deleted %i", cnt, len(touched_claims), len(deleted_claims))
|
2021-10-19 14:00:39 -04:00
|
|
|
finally:
|
|
|
|
if need_open:
|
|
|
|
db.close()
|
|
|
|
|
|
|
|
|
2021-10-20 19:34:55 -04:00
|
|
|
async def get_all_claims(env, index_name='claims', db=None):
|
2021-06-17 21:22:23 -04:00
|
|
|
need_open = db is None
|
|
|
|
db = db or LevelDB(env)
|
|
|
|
if need_open:
|
|
|
|
await db.open_dbs()
|
2021-10-20 11:40:15 -04:00
|
|
|
logging.info("Fetching claims to send ES from leveldb")
|
2021-06-17 21:30:31 -04:00
|
|
|
try:
|
|
|
|
cnt = 0
|
2021-07-27 16:11:27 -04:00
|
|
|
async for claim in db.all_claims_producer():
|
2021-08-06 14:11:28 -04:00
|
|
|
yield {
|
|
|
|
'doc': {key: value for key, value in claim.items() if key in ALL_FIELDS},
|
|
|
|
'_id': claim['claim_id'],
|
|
|
|
'_index': index_name,
|
|
|
|
'_op_type': 'update',
|
|
|
|
'doc_as_upsert': True
|
|
|
|
}
|
2021-06-17 21:30:31 -04:00
|
|
|
cnt += 1
|
|
|
|
if cnt % 10000 == 0:
|
2021-10-20 11:40:15 -04:00
|
|
|
logging.info("sent %i claims to ES", cnt)
|
2021-06-17 21:30:31 -04:00
|
|
|
finally:
|
|
|
|
if need_open:
|
|
|
|
db.close()
|
2021-02-11 23:10:30 -05:00
|
|
|
|
|
|
|
|
2021-10-20 19:34:55 -04:00
|
|
|
async def make_es_index_and_run_sync(env: Env, clients=32, force=False, db=None, index_name='claims'):
|
|
|
|
index = SearchIndex(env.es_index_prefix, elastic_host=env.elastic_host, elastic_port=env.elastic_port)
|
|
|
|
logging.info("ES sync host: %s:%i", env.elastic_host, env.elastic_port)
|
2021-02-11 23:10:30 -05:00
|
|
|
try:
|
2021-10-20 19:34:55 -04:00
|
|
|
created = 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-10-20 19:34:55 -04:00
|
|
|
created = 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-06-17 21:22:23 -04:00
|
|
|
es = AsyncElasticsearch([{'host': env.elastic_host, 'port': env.elastic_port}])
|
2021-10-20 19:34:55 -04:00
|
|
|
if force or created:
|
|
|
|
claim_generator = get_all_claims(env, index_name=index_name, db=db)
|
2021-10-19 14:00:39 -04:00
|
|
|
else:
|
2021-10-20 19:34:55 -04:00
|
|
|
claim_generator = get_recent_claims(env, index_name=index_name, db=db)
|
2021-06-17 21:22:23 -04:00
|
|
|
try:
|
2021-10-19 14:00:39 -04:00
|
|
|
async for ok, item in async_streaming_bulk(es, claim_generator, request_timeout=600, raise_on_error=False):
|
|
|
|
if not ok:
|
|
|
|
logging.warning("indexing failed for an item: %s", item)
|
2021-06-17 21:22:23 -04:00
|
|
|
await es.indices.refresh(index=index_name)
|
|
|
|
finally:
|
|
|
|
await es.close()
|
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-07-21 12:54:10 -04:00
|
|
|
parser.add_argument("-c", "--clients", type=int, default=32)
|
2021-02-17 01:09:12 -03:00
|
|
|
parser.add_argument("-f", "--force", default=False, action='store_true')
|
2021-10-20 19:34:55 -04:00
|
|
|
Env.contribute_to_arg_parser(parser)
|
2021-01-27 01:43:06 -03:00
|
|
|
args = parser.parse_args()
|
2021-10-20 19:34:55 -04:00
|
|
|
env = Env.from_arg_parser(args)
|
2021-02-11 23:10:30 -05:00
|
|
|
|
2021-10-20 19:34:55 -04:00
|
|
|
if not os.path.exists(os.path.join(args.db_dir, 'lbry-leveldb')):
|
|
|
|
logging.info("DB path doesnt exist, nothing to sync to ES")
|
2021-02-11 23:10:30 -05:00
|
|
|
return
|
2021-10-20 19:34:55 -04:00
|
|
|
|
|
|
|
asyncio.run(make_es_index_and_run_sync(env, clients=args.clients, force=args.force))
|