SomberNight
7 years ago
3 changed files with 137 additions and 4 deletions
@ -0,0 +1,95 @@ |
|||
#!/usr/bin/env python3 |
|||
# |
|||
# Electrum - lightweight Bitcoin client |
|||
# Copyright (C) 2018 Thomas Voegtlin |
|||
# |
|||
# Permission is hereby granted, free of charge, to any person |
|||
# obtaining a copy of this software and associated documentation files |
|||
# (the "Software"), to deal in the Software without restriction, |
|||
# including without limitation the rights to use, copy, modify, merge, |
|||
# publish, distribute, sublicense, and/or sell copies of the Software, |
|||
# and to permit persons to whom the Software is furnished to do so, |
|||
# subject to the following conditions: |
|||
# |
|||
# The above copyright notice and this permission notice shall be |
|||
# included in all copies or substantial portions of the Software. |
|||
# |
|||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, |
|||
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF |
|||
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND |
|||
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS |
|||
# BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN |
|||
# ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN |
|||
# CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |
|||
# SOFTWARE. |
|||
|
|||
from jsonrpclib.SimpleJSONRPCServer import SimpleJSONRPCServer, SimpleJSONRPCRequestHandler |
|||
from base64 import b64decode |
|||
import time |
|||
|
|||
from . import util |
|||
|
|||
|
|||
class RPCAuthCredentialsInvalid(Exception): |
|||
def __str__(self): |
|||
return 'Authentication failed (bad credentials)' |
|||
|
|||
|
|||
class RPCAuthCredentialsMissing(Exception): |
|||
def __str__(self): |
|||
return 'Authentication failed (missing credentials)' |
|||
|
|||
|
|||
class RPCAuthUnsupportedType(Exception): |
|||
def __str__(self): |
|||
return 'Authentication failed (only basic auth is supported)' |
|||
|
|||
|
|||
# based on http://acooke.org/cute/BasicHTTPA0.html by andrew cooke |
|||
class VerifyingJSONRPCServer(SimpleJSONRPCServer): |
|||
|
|||
def __init__(self, *args, rpc_user, rpc_password, **kargs): |
|||
|
|||
self.rpc_user = rpc_user |
|||
self.rpc_password = rpc_password |
|||
|
|||
class VerifyingRequestHandler(SimpleJSONRPCRequestHandler): |
|||
def parse_request(myself): |
|||
# first, call the original implementation which returns |
|||
# True if all OK so far |
|||
if SimpleJSONRPCRequestHandler.parse_request(myself): |
|||
try: |
|||
self.authenticate(myself.headers) |
|||
return True |
|||
except (RPCAuthCredentialsInvalid, RPCAuthCredentialsMissing, |
|||
RPCAuthUnsupportedType) as e: |
|||
myself.send_error(401, str(e)) |
|||
except BaseException as e: |
|||
import traceback, sys |
|||
traceback.print_exc(file=sys.stderr) |
|||
myself.send_error(500, str(e)) |
|||
return False |
|||
|
|||
SimpleJSONRPCServer.__init__( |
|||
self, requestHandler=VerifyingRequestHandler, *args, **kargs) |
|||
|
|||
def authenticate(self, headers): |
|||
if self.rpc_password == '': |
|||
# RPC authentication is disabled |
|||
return |
|||
|
|||
auth_string = headers.get('Authorization', None) |
|||
if auth_string is None: |
|||
raise RPCAuthCredentialsMissing() |
|||
|
|||
(basic, _, encoded) = auth_string.partition(' ') |
|||
if basic != 'Basic': |
|||
raise RPCAuthUnsupportedType() |
|||
|
|||
encoded = util.to_bytes(encoded, 'utf8') |
|||
credentials = util.to_string(b64decode(encoded), 'utf8') |
|||
(username, _, password) = credentials.partition(':') |
|||
if not (util.constant_time_compare(username, self.rpc_user) |
|||
and util.constant_time_compare(password, self.rpc_password)): |
|||
time.sleep(0.050) |
|||
raise RPCAuthCredentialsInvalid() |
Loading…
Reference in new issue