2014-03-10 05:18:05 +00:00
|
|
|
import cherrypy
|
|
|
|
from cherrypy.lib import httpauth
|
|
|
|
|
|
|
|
|
|
|
|
def check_auth(users, encrypt=None, realm=None):
|
2014-06-05 01:28:59 +00:00
|
|
|
"""If an authorization header contains credentials, return True or False.
|
|
|
|
"""
|
2014-03-10 05:18:05 +00:00
|
|
|
request = cherrypy.serving.request
|
|
|
|
if 'authorization' in request.headers:
|
|
|
|
# make sure the provided credentials are correctly set
|
|
|
|
ah = httpauth.parseAuthorization(request.headers['authorization'])
|
|
|
|
if ah is None:
|
|
|
|
raise cherrypy.HTTPError(400, 'Bad Request')
|
2014-06-05 01:28:59 +00:00
|
|
|
|
2014-03-10 05:18:05 +00:00
|
|
|
if not encrypt:
|
|
|
|
encrypt = httpauth.DIGEST_AUTH_ENCODERS[httpauth.MD5]
|
2014-06-05 01:28:59 +00:00
|
|
|
|
2014-03-10 05:18:05 +00:00
|
|
|
if hasattr(users, '__call__'):
|
|
|
|
try:
|
|
|
|
# backward compatibility
|
2014-06-05 01:28:59 +00:00
|
|
|
users = users() # expect it to return a dictionary
|
|
|
|
|
2014-03-10 05:18:05 +00:00
|
|
|
if not isinstance(users, dict):
|
2014-06-05 01:28:59 +00:00
|
|
|
raise ValueError(
|
|
|
|
"Authentication users must be a dictionary")
|
|
|
|
|
2014-03-10 05:18:05 +00:00
|
|
|
# fetch the user password
|
|
|
|
password = users.get(ah["username"], None)
|
|
|
|
except TypeError:
|
|
|
|
# returns a password (encrypted or clear text)
|
|
|
|
password = users(ah["username"])
|
|
|
|
else:
|
|
|
|
if not isinstance(users, dict):
|
|
|
|
raise ValueError("Authentication users must be a dictionary")
|
2014-06-05 01:28:59 +00:00
|
|
|
|
2014-03-10 05:18:05 +00:00
|
|
|
# fetch the user password
|
|
|
|
password = users.get(ah["username"], None)
|
2014-06-05 01:28:59 +00:00
|
|
|
|
2014-03-10 05:18:05 +00:00
|
|
|
# validate the authorization by re-computing it here
|
|
|
|
# and compare it with what the user-agent provided
|
|
|
|
if httpauth.checkResponse(ah, password, method=request.method,
|
|
|
|
encrypt=encrypt, realm=realm):
|
|
|
|
request.login = ah["username"]
|
|
|
|
return True
|
2014-06-05 01:28:59 +00:00
|
|
|
|
2014-03-10 05:18:05 +00:00
|
|
|
request.login = False
|
|
|
|
return False
|
|
|
|
|
2014-06-05 01:28:59 +00:00
|
|
|
|
2014-03-10 05:18:05 +00:00
|
|
|
def basic_auth(realm, users, encrypt=None, debug=False):
|
|
|
|
"""If auth fails, raise 401 with a basic authentication header.
|
2014-06-05 01:28:59 +00:00
|
|
|
|
|
|
|
realm
|
|
|
|
A string containing the authentication realm.
|
|
|
|
|
|
|
|
users
|
|
|
|
A dict of the form: {username: password} or a callable returning
|
|
|
|
a dict.
|
|
|
|
|
|
|
|
encrypt
|
|
|
|
callable used to encrypt the password returned from the user-agent.
|
|
|
|
if None it defaults to a md5 encryption.
|
|
|
|
|
2014-03-10 05:18:05 +00:00
|
|
|
"""
|
|
|
|
if check_auth(users, encrypt):
|
|
|
|
if debug:
|
|
|
|
cherrypy.log('Auth successful', 'TOOLS.BASIC_AUTH')
|
|
|
|
return
|
2014-06-05 01:28:59 +00:00
|
|
|
|
2014-03-10 05:18:05 +00:00
|
|
|
# inform the user-agent this path is protected
|
2014-06-05 01:28:59 +00:00
|
|
|
cherrypy.serving.response.headers[
|
|
|
|
'www-authenticate'] = httpauth.basicAuth(realm)
|
|
|
|
|
|
|
|
raise cherrypy.HTTPError(
|
|
|
|
401, "You are not authorized to access that resource")
|
|
|
|
|
2014-03-10 05:18:05 +00:00
|
|
|
|
|
|
|
def digest_auth(realm, users, debug=False):
|
|
|
|
"""If auth fails, raise 401 with a digest authentication header.
|
2014-06-05 01:28:59 +00:00
|
|
|
|
|
|
|
realm
|
|
|
|
A string containing the authentication realm.
|
|
|
|
users
|
|
|
|
A dict of the form: {username: password} or a callable returning
|
|
|
|
a dict.
|
2014-03-10 05:18:05 +00:00
|
|
|
"""
|
|
|
|
if check_auth(users, realm=realm):
|
|
|
|
if debug:
|
|
|
|
cherrypy.log('Auth successful', 'TOOLS.DIGEST_AUTH')
|
|
|
|
return
|
2014-06-05 01:28:59 +00:00
|
|
|
|
2014-03-10 05:18:05 +00:00
|
|
|
# inform the user-agent this path is protected
|
2014-06-05 01:28:59 +00:00
|
|
|
cherrypy.serving.response.headers[
|
|
|
|
'www-authenticate'] = httpauth.digestAuth(realm)
|
|
|
|
|
|
|
|
raise cherrypy.HTTPError(
|
|
|
|
401, "You are not authorized to access that resource")
|