|
|
from datetime import datetime
|
|
|
from decorator import decorator
|
|
|
from functools import wraps
|
|
|
from pylons import session, url
|
|
|
from pylons.controllers.util import abort, redirect
|
|
|
from pylons_app.model import meta
|
|
|
from pylons_app.model.db import Users, UserLogs
|
|
|
from sqlalchemy.exc import OperationalError
|
|
|
from sqlalchemy.orm.exc import NoResultFound, MultipleResultsFound
|
|
|
import crypt
|
|
|
import logging
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
|
def get_crypt_password(password):
|
|
|
"""
|
|
|
Cryptographic function used for password hashing
|
|
|
@param password: password to hash
|
|
|
"""
|
|
|
return crypt.crypt(password, '6a')
|
|
|
|
|
|
def authfunc(environ, username, password):
|
|
|
sa = meta.Session
|
|
|
password_crypt = get_crypt_password(password)
|
|
|
try:
|
|
|
user = sa.query(Users).filter(Users.username == username).one()
|
|
|
except (NoResultFound, MultipleResultsFound, OperationalError) as e:
|
|
|
log.error(e)
|
|
|
user = None
|
|
|
|
|
|
if user:
|
|
|
if user.active:
|
|
|
if user.username == username and user.password == password_crypt:
|
|
|
log.info('user %s authenticated correctly', username)
|
|
|
if environ:
|
|
|
http_accept = environ.get('HTTP_ACCEPT')
|
|
|
|
|
|
if http_accept.startswith('application/mercurial') or \
|
|
|
environ['PATH_INFO'].find('raw-file') != -1:
|
|
|
repo = environ['PATH_INFO']
|
|
|
for qry in environ['QUERY_STRING'].split('&'):
|
|
|
if qry.startswith('cmd'):
|
|
|
|
|
|
try:
|
|
|
user_log = UserLogs()
|
|
|
user_log.user_id = user.user_id
|
|
|
user_log.action = qry
|
|
|
user_log.repository = repo
|
|
|
user_log.action_date = datetime.now()
|
|
|
sa.add(user_log)
|
|
|
sa.commit()
|
|
|
log.info('Adding user %s, action %s', username, qry)
|
|
|
except Exception as e:
|
|
|
sa.rollback()
|
|
|
log.error(e)
|
|
|
|
|
|
return True
|
|
|
else:
|
|
|
log.error('user %s is disabled', username)
|
|
|
|
|
|
return False
|
|
|
|
|
|
class AuthUser(object):
|
|
|
"""
|
|
|
A simple object that handles a mercurial username for authentication
|
|
|
"""
|
|
|
username = 'Empty'
|
|
|
is_authenticated = False
|
|
|
is_admin = False
|
|
|
permissions = set()
|
|
|
group = set()
|
|
|
|
|
|
def __init__(self):
|
|
|
pass
|
|
|
|
|
|
#===============================================================================
|
|
|
# DECORATORS
|
|
|
#===============================================================================
|
|
|
class LoginRequired(object):
|
|
|
"""
|
|
|
Must be logged in to execute this function else redirect to login page
|
|
|
"""
|
|
|
def __init__(self):
|
|
|
pass
|
|
|
|
|
|
def __call__(self, func):
|
|
|
log.info('Checking login required')
|
|
|
|
|
|
@wraps(func)
|
|
|
def _wrapper(*fargs, **fkwargs):
|
|
|
user = session.get('hg_app_user', AuthUser())
|
|
|
if user.is_authenticated:
|
|
|
log.info('user %s is authenticated', user.username)
|
|
|
func(*fargs)
|
|
|
else:
|
|
|
logging.info('user %s not authenticated', user.username)
|
|
|
return redirect(url('login_home'))
|
|
|
|
|
|
return _wrapper
|
|
|
|