lbry-sdk/lbrynet/utils.py

186 lines
5.1 KiB
Python
Raw Permalink Normal View History

import base64
import codecs
import datetime
2015-08-20 11:27:15 -04:00
import random
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
2019-02-28 12:40:11 -05:00
import ssl
2018-06-07 12:18:07 -04:00
import logging
import ipaddress
2016-11-19 16:58:40 -06:00
import pkg_resources
2019-02-28 12:40:11 -05:00
import contextlib
import certifi
import aiohttp
from lbrynet.schema.claim import ClaimDict
2018-11-07 15:15:05 -05:00
from lbrynet.cryptoutils import get_lbry_hash_obj
2019-01-22 17:44:17 -05:00
2018-06-07 12:18:07 -04:00
log = logging.getLogger(__name__)
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-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()
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)
def rot13(some_str):
return codecs.encode(some_str, 'rot_13')
def deobfuscate(obfustacated):
2018-10-18 15:57:15 -04:00
return base64.b64decode(rot13(obfustacated)).decode()
def obfuscate(plain):
return rot13(base64.b64encode(plain).decode())
2019-02-03 16:19:29 -05:00
def check_connection(server="lbry.io", port=80, timeout=5) -> bool:
"""Attempts to open a socket to server:port and returns True if successful."""
log.debug('Checking connection to %s:%s', server, port)
try:
2017-10-31 12:18:47 -04:00
server = socket.gethostbyname(server)
2019-02-03 16:19:29 -05:00
socket.create_connection((server, port), timeout).close()
2016-11-11 10:10:00 -06:00
log.debug('Connection successful')
return True
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)
try:
server = "8.8.8.8"
port = 53
2019-02-03 16:19:29 -05:00
socket.create_connection((server, port), timeout).close()
log.debug('Connection successful')
return True
2019-02-03 16:19:29 -05:00
except Exception:
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)
return False
2019-02-03 16:19:29 -05:00
except Exception:
2017-10-31 12:18:55 -04:00
log.error("Failed to connect to %s:%s. Maybe the internet connection is not working",
2019-02-03 16:19:29 -05:00
server, port)
return False
2019-02-03 16:19:29 -05:00
async def async_check_connection(server="lbry.io", port=80, timeout=5) -> bool:
return await asyncio.get_event_loop().run_in_executor(None, check_connection, server, port, timeout)
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)])
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
if isinstance(stream_info, ClaimDict):
return stream_info.source_hash
result = stream_info.get('claim', {}).\
get('value', {}).\
get('stream', {}).\
get('source', {}).\
get('source')
if not result:
2018-07-21 19:08:28 -04:00
log.warning("Unable to get sd_hash")
return result
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())
2019-02-05 13:31:57 -05:00
async def resolve_host(url: str, port: int, proto: str) -> str:
if proto not in ['udp', 'tcp']:
raise Exception("invalid protocol")
try:
if ipaddress.ip_address(url):
return url
except ValueError:
pass
loop = asyncio.get_running_loop()
return (await loop.getaddrinfo(
2019-02-05 13:31:57 -05:00
url, port,
2019-02-05 13:36:25 -05:00
proto=socket.IPPROTO_TCP if proto == 'tcp' else socket.IPPROTO_UDP,
type=socket.SOCK_STREAM if proto == 'tcp' else socket.SOCK_DGRAM
))[0][4][0]
2019-02-28 12:40:11 -05:00
2019-02-28 12:42:23 -05:00
def get_ssl_context() -> ssl.SSLContext:
2019-02-28 12:40:11 -05:00
return ssl.create_default_context(
2019-02-28 12:45:56 -05:00
purpose=ssl.Purpose.CLIENT_AUTH, capath=certifi.where()
2019-02-28 12:40:11 -05:00
)
@contextlib.asynccontextmanager
2019-02-28 12:42:23 -05:00
async def aiohttp_request(method, url, **kwargs) -> typing.AsyncContextManager[aiohttp.ClientResponse]:
2019-02-28 12:40:11 -05:00
async with aiohttp.ClientSession() as session:
async with session.request(method, url, ssl=get_ssl_context(), **kwargs) as response:
yield response
2019-03-10 21:55:33 -04:00
async def get_external_ip() -> typing.Optional[str]: # used if upnp is disabled or non-functioning
try:
async with aiohttp_request("get", "https://api.lbry.io/ip") as resp:
response = await resp.json()
if response['success']:
return response['data']['ip']
except Exception as e:
return