utils.py 5.0 KB
Newer Older
baltery's avatar
baltery 已提交
1 2
# ~*~ coding: utf-8 ~*~
#
baltery's avatar
baltery 已提交
3
from __future__ import unicode_literals
4
import base64
baltery's avatar
baltery 已提交
5
import logging
baltery's avatar
baltery 已提交
6
import uuid
baltery's avatar
baltery 已提交
7

baltery's avatar
baltery 已提交
8
from paramiko.rsakey import RSAKey
9
from django.conf import settings
baltery's avatar
baltery 已提交
10 11
from django.contrib.auth.mixins import UserPassesTestMixin
from django.urls import reverse_lazy
baltery's avatar
baltery 已提交
12
from django.utils.translation import ugettext as _
13
from django.core.cache import cache
baltery's avatar
baltery 已提交
14

baltery's avatar
baltery 已提交
15
from common.tasks import send_mail_async
baltery's avatar
baltery 已提交
16 17
from common.utils import reverse, get_object_or_none
from .models import User
baltery's avatar
baltery 已提交
18

baltery's avatar
baltery 已提交
19

baltery's avatar
baltery 已提交
20 21 22 23 24 25 26 27
try:
    import cStringIO as StringIO
except ImportError:
    import StringIO


logger = logging.getLogger('jumpserver')

baltery's avatar
baltery 已提交
28 29 30

class AdminUserRequiredMixin(UserPassesTestMixin):
    def test_func(self):
baltery's avatar
baltery 已提交
31 32 33 34 35 36
        if not self.request.user.is_authenticated:
            return False
        elif not self.request.user.is_superuser:
            self.raise_exception = True
            return False
        return True
baltery's avatar
baltery 已提交
37 38


baltery's avatar
baltery 已提交
39
def user_add_success_next(user):
baltery's avatar
baltery 已提交
40
    subject = _('Create account successfully')
baltery's avatar
baltery 已提交
41
    recipient_list = [user.email]
baltery's avatar
baltery 已提交
42 43
    message = _("""
    Hello %(name)s:
baltery's avatar
baltery 已提交
44
    </br>
baltery's avatar
baltery 已提交
45
    Your account has been created successfully
baltery's avatar
baltery 已提交
46
    </br>
baltery's avatar
baltery 已提交
47
    <a href="%(rest_password_url)s?token=%(rest_password_token)s">click here to set your password</a>
baltery's avatar
baltery 已提交
48
    </br>
baltery's avatar
baltery 已提交
49
    This link is valid for 1 hour. After it expires, <a href="%(forget_password_url)s?email=%(email)s">request new one</a>
baltery's avatar
baltery 已提交
50 51 52 53 54

    </br>
    ---

    </br>
baltery's avatar
baltery 已提交
55
    <a href="%(login_url)s">Login direct</a>
baltery's avatar
baltery 已提交
56 57

    </br>
baltery's avatar
baltery 已提交
58
    """) % {
baltery's avatar
baltery 已提交
59 60
        'name': user.name,
        'rest_password_url': reverse('users:reset-password', external=True),
baltery's avatar
baltery 已提交
61
        'rest_password_token': user.generate_reset_token(),
baltery's avatar
baltery 已提交
62
        'forget_password_url': reverse('users:forgot-password', external=True),
baltery's avatar
baltery 已提交
63 64 65 66 67 68 69 70
        'email': user.email,
        'login_url': reverse('users:login', external=True),
    }

    send_mail_async.delay(subject, message, recipient_list, html_message=message)


def send_reset_password_mail(user):
baltery's avatar
baltery 已提交
71
    subject = _('Reset password')
baltery's avatar
baltery 已提交
72
    recipient_list = [user.email]
baltery's avatar
baltery 已提交
73 74
    message = _("""
    Hello %(name)s:
baltery's avatar
baltery 已提交
75
    </br>
baltery's avatar
baltery 已提交
76
    Please click the link below to reset your password, if not your request, concern your account security
baltery's avatar
baltery 已提交
77
    </br>
baltery's avatar
baltery 已提交
78
    <a href="%(rest_password_url)s?token=%(rest_password_token)s">Click here reset password</a>
baltery's avatar
baltery 已提交
79
    </br>
baltery's avatar
baltery 已提交
80
    This link is valid for 1 hour. After it expires, <a href="%(forget_password_url)s?email=%(email)s">request new one</a>
baltery's avatar
baltery 已提交
81 82 83 84 85

    </br>
    ---

    </br>
baltery's avatar
baltery 已提交
86
    <a href="%(login_url)s">Login direct</a>
baltery's avatar
baltery 已提交
87 88

    </br>
baltery's avatar
baltery 已提交
89
    """) % {
baltery's avatar
baltery 已提交
90 91 92
        'name': user.name,
        'rest_password_url': reverse('users:reset-password', external=True),
        'rest_password_token': user.generate_reset_token(),
baltery's avatar
baltery 已提交
93
        'forget_password_url': reverse('users:forgot-password', external=True),
baltery's avatar
baltery 已提交
94 95 96
        'email': user.email,
        'login_url': reverse('users:login', external=True),
    }
97 98
    if settings.DEBUG:
        logger.debug(message)
baltery's avatar
baltery 已提交
99 100 101

    send_mail_async.delay(subject, message, recipient_list, html_message=message)

baltery's avatar
baltery 已提交
102

103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124
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)


baltery's avatar
baltery 已提交
125
def check_user_valid(**kwargs):
baltery's avatar
baltery 已提交
126 127
    password = kwargs.pop('password', None)
    public_key = kwargs.pop('public_key', None)
baltery's avatar
baltery 已提交
128 129
    email = kwargs.pop('email', None)
    username = kwargs.pop('username', None)
baltery's avatar
baltery 已提交
130 131 132 133 134 135 136 137 138 139 140 141

    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')
baltery's avatar
baltery 已提交
142

baltery's avatar
baltery 已提交
143
    if password and user.password and user.check_password(password):
baltery's avatar
baltery 已提交
144 145
        return user, ''

baltery's avatar
baltery 已提交
146
    if public_key and user.public_key:
baltery's avatar
baltery 已提交
147 148 149
        public_key_saved = user.public_key.split()
        if len(public_key_saved) == 1:
            if public_key == public_key_saved[0]:
baltery's avatar
baltery 已提交
150
                return user, ''
baltery's avatar
baltery 已提交
151 152
        elif len(public_key_saved) > 1:
            if public_key == public_key_saved[1]:
baltery's avatar
baltery 已提交
153
                return user, ''
baltery's avatar
baltery 已提交
154
    return None, _('Password or SSH public key invalid')
baltery's avatar
baltery 已提交
155 156


baltery's avatar
baltery 已提交
157
def refresh_token(token, user, expiration=3600):
158 159 160
    cache.set(token, user.id, expiration)


baltery's avatar
baltery 已提交
161
def generate_token(request, user):
162 163 164
    expiration = settings.CONFIG.TOKEN_EXPIRATION or 3600
    remote_addr = request.META.get('REMOTE_ADDR', '')
    remote_addr = base64.b16encode(remote_addr).replace('=', '')
baltery's avatar
baltery 已提交
165
    token = cache.get('%s_%s' % (user.id, remote_addr))
166 167
    if not token:
        token = uuid.uuid4().get_hex()
168
        print('Set cache: %s' % token)
baltery's avatar
baltery 已提交
169 170
        cache.set(token, user.id, expiration)
        cache.set('%s_%s' % (user.id, remote_addr), token, expiration)
171 172 173
    return token