2016-07-20 12:00:34 -05:00
|
|
|
import base64
|
2018-07-17 21:35:53 -03:00
|
|
|
import codecs
|
2016-10-17 20:00:24 -05:00
|
|
|
import datetime
|
2015-08-20 11:27:15 -04:00
|
|
|
import random
|
2016-10-17 20:00:24 -05:00
|
|
|
import socket
|
2016-12-30 12:35:17 -06:00
|
|
|
import string
|
2018-07-21 14:12:29 -04:00
|
|
|
import json
|
2019-01-22 17:44:17 -05:00
|
|
|
import typing
|
|
|
|
import asyncio
|
2018-06-07 12:18:07 -04:00
|
|
|
import logging
|
2016-11-19 16:58:40 -06:00
|
|
|
import pkg_resources
|
2018-09-17 17:31:44 -03:00
|
|
|
from lbrynet.schema.claim import ClaimDict
|
2018-11-07 15:15:05 -05:00
|
|
|
from lbrynet.cryptoutils import get_lbry_hash_obj
|
2016-07-25 16:09:13 -05:00
|
|
|
|
2019-01-22 17:44:17 -05:00
|
|
|
|
2018-06-07 12:18:07 -04:00
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
2016-10-17 20:00:24 -05:00
|
|
|
|
2016-10-05 14:16:20 -05:00
|
|
|
# defining these time functions here allows for easier overriding in testing
|
2016-09-29 23:06:07 -05:00
|
|
|
def now():
|
|
|
|
return datetime.datetime.now()
|
|
|
|
|
2016-10-05 14:16:20 -05:00
|
|
|
|
2016-09-29 23:06:07 -05:00
|
|
|
def utcnow():
|
|
|
|
return datetime.datetime.utcnow()
|
|
|
|
|
2016-10-05 14:16:20 -05:00
|
|
|
|
2016-09-29 23:06:07 -05:00
|
|
|
def isonow():
|
|
|
|
"""Return utc now in isoformat with timezone"""
|
|
|
|
return utcnow().isoformat() + 'Z'
|
|
|
|
|
2016-10-31 16:19:19 -05:00
|
|
|
|
2016-09-29 23:06:07 -05:00
|
|
|
def today():
|
|
|
|
return datetime.datetime.today()
|
|
|
|
|
|
|
|
|
2017-01-02 14:52:24 -05:00
|
|
|
def timedelta(**kwargs):
|
|
|
|
return datetime.timedelta(**kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
def datetime_obj(*args, **kwargs):
|
|
|
|
return datetime.datetime(*args, **kwargs)
|
|
|
|
|
|
|
|
|
2015-08-20 11:27:15 -04:00
|
|
|
def generate_id(num=None):
|
|
|
|
h = get_lbry_hash_obj()
|
|
|
|
if num is not None:
|
2018-06-12 11:54:01 -04:00
|
|
|
h.update(str(num).encode())
|
2015-08-20 11:27:15 -04:00
|
|
|
else:
|
2018-06-12 11:54:01 -04:00
|
|
|
h.update(str(random.getrandbits(512)).encode())
|
2015-08-20 11:27:15 -04:00
|
|
|
return h.digest()
|
|
|
|
|
|
|
|
|
2016-07-25 16:09:13 -05:00
|
|
|
def version_is_greater_than(a, b):
|
|
|
|
"""Returns True if version a is more recent than version b"""
|
2016-11-19 16:58:40 -06:00
|
|
|
return pkg_resources.parse_version(a) > pkg_resources.parse_version(b)
|
2016-07-20 12:00:34 -05:00
|
|
|
|
|
|
|
|
2018-07-17 21:35:53 -03:00
|
|
|
def rot13(some_str):
|
|
|
|
return codecs.encode(some_str, 'rot_13')
|
|
|
|
|
|
|
|
|
2016-07-20 12:00:34 -05:00
|
|
|
def deobfuscate(obfustacated):
|
2018-10-18 15:57:15 -04:00
|
|
|
return base64.b64decode(rot13(obfustacated)).decode()
|
2016-07-20 12:00:34 -05:00
|
|
|
|
|
|
|
|
|
|
|
def obfuscate(plain):
|
2018-07-21 21:12:33 -04:00
|
|
|
return rot13(base64.b64encode(plain).decode())
|
2016-09-16 00:14:25 -04:00
|
|
|
|
|
|
|
|
2018-08-05 02:49:10 +05:30
|
|
|
def check_connection(server="lbry.io", port=80, timeout=5):
|
2016-10-17 20:00:24 -05:00
|
|
|
"""Attempts to open a socket to server:port and returns True if successful."""
|
2017-09-20 22:52:16 +02:00
|
|
|
log.debug('Checking connection to %s:%s', server, port)
|
2016-10-17 20:00:24 -05:00
|
|
|
try:
|
2017-10-31 12:18:47 -04:00
|
|
|
server = socket.gethostbyname(server)
|
2018-10-16 11:03:56 -04:00
|
|
|
conn = socket.create_connection((server, port), timeout)
|
|
|
|
conn.close()
|
2016-11-11 10:10:00 -06:00
|
|
|
log.debug('Connection successful')
|
2016-10-17 20:00:24 -05:00
|
|
|
return True
|
2017-09-20 22:52:16 +02:00
|
|
|
except (socket.gaierror, socket.herror) as ex:
|
2017-10-31 12:18:55 -04:00
|
|
|
log.warning("Failed to connect to %s:%s. Unable to resolve domain. Trying to bypass DNS",
|
|
|
|
server, port)
|
2017-09-20 22:52:16 +02:00
|
|
|
try:
|
|
|
|
server = "8.8.8.8"
|
|
|
|
port = 53
|
|
|
|
socket.create_connection((server, port), timeout)
|
|
|
|
log.debug('Connection successful')
|
|
|
|
return True
|
|
|
|
except Exception as ex:
|
2017-10-31 12:18:55 -04:00
|
|
|
log.error("Failed to connect to %s:%s. Maybe the internet connection is not working",
|
|
|
|
server, port)
|
2017-09-20 22:52:16 +02:00
|
|
|
return False
|
2016-10-17 20:00:24 -05:00
|
|
|
except Exception as ex:
|
2017-10-31 12:18:55 -04:00
|
|
|
log.error("Failed to connect to %s:%s. Maybe the internet connection is not working",
|
|
|
|
server, port)
|
2016-10-17 20:00:24 -05:00
|
|
|
return False
|
2016-10-21 15:26:36 -07:00
|
|
|
|
|
|
|
|
2016-12-30 12:35:17 -06:00
|
|
|
def random_string(length=10, chars=string.ascii_lowercase):
|
|
|
|
return ''.join([random.choice(chars) for _ in range(length)])
|
2017-02-16 09:09:21 -05:00
|
|
|
|
|
|
|
|
|
|
|
def short_hash(hash_str):
|
|
|
|
return hash_str[:6]
|
2017-03-09 10:39:17 -05:00
|
|
|
|
|
|
|
|
|
|
|
def get_sd_hash(stream_info):
|
|
|
|
if not stream_info:
|
|
|
|
return None
|
2017-04-06 20:37:10 -04:00
|
|
|
if isinstance(stream_info, ClaimDict):
|
|
|
|
return stream_info.source_hash
|
2018-02-06 01:16:10 -05:00
|
|
|
result = stream_info.get('claim', {}).\
|
|
|
|
get('value', {}).\
|
|
|
|
get('stream', {}).\
|
|
|
|
get('source', {}).\
|
|
|
|
get('source')
|
2018-02-03 23:08:15 -05:00
|
|
|
if not result:
|
2018-07-21 19:08:28 -04:00
|
|
|
log.warning("Unable to get sd_hash")
|
2018-02-03 23:08:15 -05:00
|
|
|
return result
|
2017-03-15 16:19:11 -04:00
|
|
|
|
|
|
|
|
|
|
|
def json_dumps_pretty(obj, **kwargs):
|
|
|
|
return json.dumps(obj, sort_keys=True, indent=2, separators=(',', ': '), **kwargs)
|
2018-02-28 14:59:12 -05:00
|
|
|
|
|
|
|
|
2019-01-22 17:44:17 -05:00
|
|
|
def cancel_task(task: typing.Optional[asyncio.Task]):
|
|
|
|
if task and not task.done():
|
|
|
|
task.cancel()
|
|
|
|
|
|
|
|
|
|
|
|
def cancel_tasks(tasks: typing.List[typing.Optional[asyncio.Task]]):
|
|
|
|
for task in tasks:
|
|
|
|
cancel_task(task)
|
|
|
|
|
|
|
|
|
|
|
|
def drain_tasks(tasks: typing.List[typing.Optional[asyncio.Task]]):
|
|
|
|
while tasks:
|
|
|
|
cancel_task(tasks.pop())
|