jumpserver/apps/users/utils.py

209 lines
6.1 KiB
Python
Raw Normal View History

2016-08-09 09:27:37 +00:00
# ~*~ coding: utf-8 ~*~
#
2016-08-31 17:12:02 +00:00
from __future__ import unicode_literals
import base64
2016-08-30 17:00:20 +00:00
import logging
2016-10-31 10:58:23 +00:00
import uuid
2016-08-30 17:00:20 +00:00
2017-12-04 12:15:47 +00:00
import requests
import ipaddress
from django.conf import settings
2016-08-28 15:58:22 +00:00
from django.contrib.auth.mixins import UserPassesTestMixin
2017-12-05 04:24:31 +00:00
from django.contrib.auth import authenticate, login as auth_login
2016-09-03 16:51:36 +00:00
from django.utils.translation import ugettext as _
from django.core.cache import cache
2016-08-28 15:58:22 +00:00
2016-08-31 17:12:02 +00:00
from common.tasks import send_mail_async
2016-09-24 16:11:31 +00:00
from common.utils import reverse, get_object_or_none
2017-12-04 12:15:47 +00:00
from .models import User, LoginLog
2016-08-30 17:00:20 +00:00
logger = logging.getLogger('jumpserver')
2016-08-28 15:58:22 +00:00
class AdminUserRequiredMixin(UserPassesTestMixin):
def test_func(self):
2017-03-31 03:25:25 +00:00
if not self.request.user.is_authenticated:
return False
elif not self.request.user.is_superuser:
self.raise_exception = True
return False
return True
2016-08-30 17:00:20 +00:00
2017-12-12 04:19:45 +00:00
def send_user_created_mail(user):
2016-09-03 16:51:36 +00:00
subject = _('Create account successfully')
2016-08-31 17:12:02 +00:00
recipient_list = [user.email]
2016-09-03 16:51:36 +00:00
message = _("""
Hello %(name)s:
2016-08-31 17:12:02 +00:00
</br>
2016-09-03 16:51:36 +00:00
Your account has been created successfully
2016-08-31 17:12:02 +00:00
</br>
2016-09-03 16:51:36 +00:00
<a href="%(rest_password_url)s?token=%(rest_password_token)s">click here to set your password</a>
2016-08-31 17:12:02 +00:00
</br>
2016-09-03 16:51:36 +00:00
This link is valid for 1 hour. After it expires, <a href="%(forget_password_url)s?email=%(email)s">request new one</a>
2016-08-31 17:12:02 +00:00
</br>
---
</br>
2016-09-03 16:51:36 +00:00
<a href="%(login_url)s">Login direct</a>
2016-08-31 17:12:02 +00:00
</br>
2016-09-03 16:51:36 +00:00
""") % {
2016-08-31 17:12:02 +00:00
'name': user.name,
'rest_password_url': reverse('users:reset-password', external=True),
2016-09-01 15:09:58 +00:00
'rest_password_token': user.generate_reset_token(),
2016-09-06 07:09:00 +00:00
'forget_password_url': reverse('users:forgot-password', external=True),
2016-09-01 15:09:58 +00:00
'email': user.email,
'login_url': reverse('users:login', external=True),
}
2017-12-12 04:19:45 +00:00
if settings.DEBUG:
print(message)
2016-09-01 15:09:58 +00:00
send_mail_async.delay(subject, message, recipient_list, html_message=message)
def send_reset_password_mail(user):
2016-09-03 16:51:36 +00:00
subject = _('Reset password')
2016-09-01 15:09:58 +00:00
recipient_list = [user.email]
2016-09-03 16:51:36 +00:00
message = _("""
Hello %(name)s:
2016-09-01 15:09:58 +00:00
</br>
2016-09-03 16:51:36 +00:00
Please click the link below to reset your password, if not your request, concern your account security
2016-09-01 15:09:58 +00:00
</br>
2016-09-03 16:51:36 +00:00
<a href="%(rest_password_url)s?token=%(rest_password_token)s">Click here reset password</a>
2016-09-01 15:09:58 +00:00
</br>
2016-11-09 11:29:15 +00:00
This link is valid for 1 hour. After it expires, <a href="%(forget_password_url)s?email=%(email)s">request new one</a>
2016-09-01 15:09:58 +00:00
</br>
---
</br>
2016-09-03 16:51:36 +00:00
<a href="%(login_url)s">Login direct</a>
2016-09-01 15:09:58 +00:00
</br>
2016-09-03 16:51:36 +00:00
""") % {
2016-09-01 15:09:58 +00:00
'name': user.name,
'rest_password_url': reverse('users:reset-password', external=True),
'rest_password_token': user.generate_reset_token(),
2016-09-06 07:09:00 +00:00
'forget_password_url': reverse('users:forgot-password', external=True),
2016-08-31 17:12:02 +00:00
'email': user.email,
'login_url': reverse('users:login', external=True),
}
if settings.DEBUG:
logger.debug(message)
2016-08-31 17:12:02 +00:00
send_mail_async.delay(subject, message, recipient_list, html_message=message)
2016-08-30 17:00:20 +00:00
def send_reset_ssh_key_mail(user):
subject = _('SSH Key Reset')
recipient_list = [user.email]
message = _("""
Hello %(name)s:
</br>
Your ssh public key has been reset by site administrator.
Please login and reset your ssh public key.
</br>
<a href="%(login_url)s">Login direct</a>
</br>
""") % {
'name': user.name,
'login_url': reverse('users:login', external=True),
}
if settings.DEBUG:
logger.debug(message)
send_mail_async.delay(subject, message, recipient_list, html_message=message)
2016-10-31 10:58:23 +00:00
def check_user_valid(**kwargs):
2016-09-24 16:11:31 +00:00
password = kwargs.pop('password', None)
public_key = kwargs.pop('public_key', None)
2016-12-29 11:17:00 +00:00
email = kwargs.pop('email', None)
username = kwargs.pop('username', None)
2016-12-28 16:29:59 +00:00
if username:
user = get_object_or_none(User, username=username)
elif email:
user = get_object_or_none(User, email=email)
else:
user = None
if user is None:
return None, _('User not exist')
elif not user.is_valid:
return None, _('Disabled or expired')
2016-09-24 16:11:31 +00:00
2017-10-10 06:18:08 +00:00
if password and authenticate(username=username, password=password):
2016-12-28 16:29:59 +00:00
return user, ''
2017-01-02 16:11:44 +00:00
if public_key and user.public_key:
2016-11-07 08:59:52 +00:00
public_key_saved = user.public_key.split()
if len(public_key_saved) == 1:
if public_key == public_key_saved[0]:
2016-12-28 16:29:59 +00:00
return user, ''
2016-11-07 08:59:52 +00:00
elif len(public_key_saved) > 1:
if public_key == public_key_saved[1]:
2016-12-28 16:29:59 +00:00
return user, ''
2016-12-29 11:17:00 +00:00
return None, _('Password or SSH public key invalid')
2016-09-24 16:11:31 +00:00
2017-07-28 12:42:16 +00:00
def refresh_token(token, user, expiration=settings.CONFIG.TOKEN_EXPIRATION or 3600):
cache.set(token, user.id, expiration)
2016-12-29 11:17:00 +00:00
def generate_token(request, user):
expiration = settings.CONFIG.TOKEN_EXPIRATION or 3600
remote_addr = request.META.get('REMOTE_ADDR', '')
2017-05-15 15:39:54 +00:00
if not isinstance(remote_addr, bytes):
remote_addr = remote_addr.encode("utf-8")
remote_addr = base64.b16encode(remote_addr) #.replace(b'=', '')
2016-12-29 11:17:00 +00:00
token = cache.get('%s_%s' % (user.id, remote_addr))
if not token:
2017-05-15 15:39:54 +00:00
token = uuid.uuid4().hex
2016-12-29 11:17:00 +00:00
cache.set(token, user.id, expiration)
cache.set('%s_%s' % (user.id, remote_addr), token, expiration)
return token
2017-12-04 12:15:47 +00:00
def validate_ip(ip):
try:
ipaddress.ip_address(ip)
return True
except ValueError:
pass
return False
def write_login_log(username, type='', ip='', user_agent=''):
if not (ip and validate_ip(ip)):
ip = '0.0.0.0'
city = get_ip_city(ip)
LoginLog.objects.create(
username=username, type=type,
ip=ip, city=city, user_agent=user_agent
)
def get_ip_city(ip, timeout=10):
# Taobao ip api: http://ip.taobao.com//service/getIpInfo.php?ip=8.8.8.8
# Sina ip api: http://int.dpool.sina.com.cn/iplookup/iplookup.php?ip=8.8.8.8&format=json
url = 'http://int.dpool.sina.com.cn/iplookup/iplookup.php?ip=%s&format=json' % ip
try:
r = requests.get(url, timeout=timeout)
except requests.Timeout:
r = None
city = 'Unknown'
if r and r.status_code == 200:
try:
data = r.json()
if not isinstance(data, int) and data['ret'] == 1:
city = data['country'] + ' ' + data['city']
except ValueError:
pass
return city