2016-09-21 03:49:52 -04:00
|
|
|
import urlparse
|
2016-09-20 16:58:30 -04:00
|
|
|
import logging
|
|
|
|
import requests
|
|
|
|
import os
|
|
|
|
import base64
|
|
|
|
import json
|
|
|
|
|
2017-06-25 21:04:33 -04:00
|
|
|
from lbrynet.daemon.auth.util import load_api_keys, APIKey, API_KEY_NAME, get_auth_message
|
2016-12-21 11:55:43 -08:00
|
|
|
from lbrynet import conf
|
2016-10-26 03:16:33 -04:00
|
|
|
from jsonrpc.proxy import JSONRPCProxy
|
2016-09-20 16:58:30 -04:00
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
USER_AGENT = "AuthServiceProxy/0.1"
|
2016-09-21 03:49:52 -04:00
|
|
|
TWISTED_SESSION = "TWISTED_SESSION"
|
|
|
|
LBRY_SECRET = "LBRY_SECRET"
|
2016-09-20 16:58:30 -04:00
|
|
|
HTTP_TIMEOUT = 30
|
|
|
|
|
|
|
|
|
|
|
|
class JSONRPCException(Exception):
|
|
|
|
def __init__(self, rpc_error):
|
|
|
|
Exception.__init__(self)
|
|
|
|
self.error = rpc_error
|
|
|
|
|
|
|
|
|
2016-10-26 03:16:33 -04:00
|
|
|
class AuthAPIClient(object):
|
2017-01-03 14:13:01 -05:00
|
|
|
def __init__(self, key, timeout, connection, count, cookies, auth, url, login_url):
|
2016-09-21 03:49:52 -04:00
|
|
|
self.__api_key = key
|
|
|
|
self.__service_url = login_url
|
2016-09-20 16:58:30 -04:00
|
|
|
self.__id_count = count
|
2016-09-21 03:49:52 -04:00
|
|
|
self.__url = url
|
|
|
|
self.__auth_header = auth
|
|
|
|
self.__conn = connection
|
|
|
|
self.__cookies = cookies
|
2016-09-20 16:58:30 -04:00
|
|
|
|
|
|
|
def __getattr__(self, name):
|
|
|
|
if name.startswith('__') and name.endswith('__'):
|
2017-01-03 14:13:01 -05:00
|
|
|
raise AttributeError # Python internal stuff
|
|
|
|
|
|
|
|
def f(*args):
|
2017-01-09 19:31:06 -05:00
|
|
|
return self.call(name, args[0] if args else {})
|
2017-01-03 14:13:01 -05:00
|
|
|
|
|
|
|
return f
|
|
|
|
|
|
|
|
def call(self, method, params={}):
|
2016-09-20 16:58:30 -04:00
|
|
|
self.__id_count += 1
|
2017-01-09 19:31:06 -05:00
|
|
|
pre_auth_post_data = {
|
2017-03-24 17:18:17 -04:00
|
|
|
'version': '2',
|
2017-01-03 14:13:01 -05:00
|
|
|
'method': method,
|
2017-01-30 18:46:06 -05:00
|
|
|
'params': params,
|
2017-01-03 14:13:01 -05:00
|
|
|
'id': self.__id_count
|
|
|
|
}
|
2017-01-09 19:31:06 -05:00
|
|
|
to_auth = get_auth_message(pre_auth_post_data)
|
2016-09-21 21:36:06 -04:00
|
|
|
token = self.__api_key.get_hmac(to_auth)
|
2017-01-09 19:31:06 -05:00
|
|
|
pre_auth_post_data.update({'hmac': token})
|
|
|
|
post_data = json.dumps(pre_auth_post_data)
|
2016-09-20 16:58:30 -04:00
|
|
|
service_url = self.__service_url
|
|
|
|
auth_header = self.__auth_header
|
|
|
|
cookies = self.__cookies
|
|
|
|
host = self.__url.hostname
|
|
|
|
|
|
|
|
req = requests.Request(method='POST',
|
|
|
|
url=service_url,
|
2017-01-09 19:31:06 -05:00
|
|
|
data=post_data,
|
2017-01-03 14:13:01 -05:00
|
|
|
headers={
|
|
|
|
'Host': host,
|
|
|
|
'User-Agent': USER_AGENT,
|
|
|
|
'Authorization': auth_header,
|
|
|
|
'Content-type': 'application/json'
|
|
|
|
},
|
2016-09-20 16:58:30 -04:00
|
|
|
cookies=cookies)
|
|
|
|
r = req.prepare()
|
|
|
|
http_response = self.__conn.send(r)
|
2016-09-21 03:49:52 -04:00
|
|
|
cookies = http_response.cookies
|
2016-09-20 16:58:30 -04:00
|
|
|
headers = http_response.headers
|
2016-09-21 03:49:52 -04:00
|
|
|
next_secret = headers.get(LBRY_SECRET, False)
|
2016-09-20 16:58:30 -04:00
|
|
|
if next_secret:
|
2016-09-21 03:49:52 -04:00
|
|
|
self.__api_key.secret = next_secret
|
|
|
|
self.__cookies = cookies
|
2016-09-20 16:58:30 -04:00
|
|
|
|
|
|
|
if http_response is None:
|
|
|
|
raise JSONRPCException({
|
|
|
|
'code': -342, 'message': 'missing HTTP response from server'})
|
|
|
|
|
|
|
|
http_response.raise_for_status()
|
|
|
|
|
|
|
|
response = http_response.json()
|
|
|
|
|
|
|
|
if response['error'] is not None:
|
|
|
|
raise JSONRPCException(response['error'])
|
|
|
|
elif 'result' not in response:
|
|
|
|
raise JSONRPCException({
|
|
|
|
'code': -343, 'message': 'missing JSON-RPC result'})
|
|
|
|
else:
|
2016-09-21 03:49:52 -04:00
|
|
|
return response['result']
|
|
|
|
|
|
|
|
@classmethod
|
2016-11-30 14:20:45 -06:00
|
|
|
def config(cls, key_name=None, key=None, pw_path=None,
|
|
|
|
timeout=HTTP_TIMEOUT,
|
|
|
|
connection=None, count=0,
|
|
|
|
cookies=None, auth=None,
|
|
|
|
url=None, login_url=None):
|
2016-09-21 21:36:06 -04:00
|
|
|
|
2016-09-21 03:49:52 -04:00
|
|
|
api_key_name = API_KEY_NAME if not key_name else key_name
|
2017-01-16 22:23:20 -05:00
|
|
|
pw_path = os.path.join(conf.settings['data_dir'], ".api_keys") if not pw_path else pw_path
|
2016-09-21 03:49:52 -04:00
|
|
|
if not key:
|
|
|
|
keys = load_api_keys(pw_path)
|
|
|
|
api_key = keys.get(api_key_name, False)
|
|
|
|
else:
|
|
|
|
api_key = APIKey(name=api_key_name, secret=key)
|
|
|
|
if login_url is None:
|
2016-10-19 00:12:44 -04:00
|
|
|
service_url = "http://%s:%s@%s:%i/%s" % (api_key_name,
|
|
|
|
api_key.secret,
|
2017-01-16 22:23:20 -05:00
|
|
|
conf.settings['api_host'],
|
|
|
|
conf.settings['api_port'],
|
|
|
|
conf.settings['API_ADDRESS'])
|
2016-09-21 03:49:52 -04:00
|
|
|
else:
|
|
|
|
service_url = login_url
|
|
|
|
id_count = count
|
|
|
|
|
|
|
|
if auth is None and connection is None and cookies is None and url is None:
|
|
|
|
# This is a new client instance, initialize the auth header and start a session
|
|
|
|
url = urlparse.urlparse(service_url)
|
|
|
|
(user, passwd) = (url.username, url.password)
|
|
|
|
try:
|
|
|
|
user = user.encode('utf8')
|
|
|
|
except AttributeError:
|
|
|
|
pass
|
|
|
|
try:
|
|
|
|
passwd = passwd.encode('utf8')
|
|
|
|
except AttributeError:
|
|
|
|
pass
|
|
|
|
authpair = user + b':' + passwd
|
|
|
|
auth_header = b'Basic ' + base64.b64encode(authpair)
|
|
|
|
conn = requests.Session()
|
|
|
|
conn.auth = (user, passwd)
|
|
|
|
req = requests.Request(method='POST',
|
|
|
|
url=service_url,
|
|
|
|
auth=conn.auth,
|
|
|
|
headers={'Host': url.hostname,
|
|
|
|
'User-Agent': USER_AGENT,
|
|
|
|
'Authorization': auth_header,
|
|
|
|
'Content-type': 'application/json'},)
|
|
|
|
r = req.prepare()
|
|
|
|
http_response = conn.send(r)
|
|
|
|
cookies = http_response.cookies
|
|
|
|
uid = cookies.get(TWISTED_SESSION)
|
|
|
|
api_key = APIKey.new(seed=uid)
|
|
|
|
else:
|
|
|
|
# This is a client that already has a session, use it
|
|
|
|
auth_header = auth
|
|
|
|
conn = connection
|
|
|
|
assert cookies.get(LBRY_SECRET, False), "Missing cookie"
|
|
|
|
secret = cookies.get(LBRY_SECRET)
|
|
|
|
api_key = APIKey(secret, api_key_name)
|
2017-01-09 19:31:06 -05:00
|
|
|
return cls(api_key, timeout, conn, id_count, cookies, auth_header, url, service_url)
|
2016-10-26 03:16:33 -04:00
|
|
|
|
|
|
|
|
|
|
|
class LBRYAPIClient(object):
|
|
|
|
@staticmethod
|
2017-01-03 14:13:01 -05:00
|
|
|
def get_client():
|
2017-06-25 21:23:48 -04:00
|
|
|
if not conf.settings:
|
|
|
|
conf.initialize_settings()
|
2017-01-16 22:23:20 -05:00
|
|
|
return AuthAPIClient.config() if conf.settings['use_auth_http'] else \
|
|
|
|
JSONRPCProxy.from_url(conf.settings.get_api_connection_string())
|