PasswordTool.py 8.78 KB
Newer Older
1
# -*- coding: utf-8 -*-
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
##############################################################################
#
# Copyright (c) 2008 Nexedi SARL and Contributors. All Rights Reserved.
#                    Aurelien Calonne <aurel@nexedi.com>
#
# WARNING: This program as such is intended to be used by professional
# programmers who take the whole responsability of assessing all potential
# consequences resulting from its eventual inadequacies and bugs
# End users who are looking for a ready-to-use solution with commercial
# garantees and support are strongly adviced to contract a Free Software
# Service Company
#
# This program is Free Software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
#
##############################################################################

30
import socket
31 32

from AccessControl import ClassSecurityInfo
33
from Products.ERP5Type.Globals import InitializeClass, DTMLFile, get_request
34 35 36
from Products.ERP5Type.Tool.BaseTool import BaseTool
from Products.ERP5Type import Permissions
from Products.ERP5 import _dtmldir
37
from zLOG import LOG, INFO
38 39
import time, random, md5
from DateTime import DateTime
40
from Products.ERP5Type.Message import translateString
41
from Acquisition import aq_base
42
from Products.ERP5Type.Globals import PersistentMapping
43
from urllib import urlencode
44 45 46

class PasswordTool(BaseTool):
  """
Jérome Perrin's avatar
Jérome Perrin committed
47
    PasswordTool is used to allow a user to change its password
48 49 50 51 52 53 54 55 56 57 58 59 60 61 62
  """
  title = 'Password Tool'
  id = 'portal_password'
  meta_type = 'ERP5 Password Tool'
  portal_type = 'Password Tool'
  allowed_types = ()

  # Declarative Security
  security = ClassSecurityInfo()

  security.declareProtected(Permissions.ManagePortal, 'manage_overview' )
  manage_overview = DTMLFile( 'explainPasswordTool', _dtmldir )


  _expiration_day = 1
63 64
  _password_request_dict = {}

65
  def __init__(self):
66
    self._password_request_dict = PersistentMapping()
67 68 69

  def mailPasswordResetRequest(self, user_login=None, REQUEST=None):
    """
Jérome Perrin's avatar
Jérome Perrin committed
70
    Create a random string and expiration date for request
71 72 73 74 75
    """
    if user_login is None:
      user_login = REQUEST["user_login"]

    # check user exists
76 77
    user_list = self.getPortalObject().acl_users.\
                      erp5_users.getUserByLogin(user_login)
78
    if len(user_list) == 0:
Yusei Tahara's avatar
Yusei Tahara committed
79
      msg = translateString("User ${user} does not exist.",
80
                            mapping={'user':user_login})
81
      if REQUEST is not None:
82 83 84 85
        parameter = urlencode(dict(portal_status_message=msg))
        ret_url = '%s/login_form?%s' % \
                  (self.getPortalObject().absolute_url(),
                  parameter)
86 87 88 89 90
        return REQUEST.RESPONSE.redirect( ret_url )
      else:
        return msg

    user = user_list[0].getObject()
Jérome Perrin's avatar
Jérome Perrin committed
91
    # generate a random string
92
    random_url = self._generateUUID()
93 94 95 96 97
    parameter = urlencode(dict(reset_key=random_url))
    url = "%s/portal_password/%s?%s" % (
                                self.getPortalObject().absolute_url(),
                                'PasswordTool_viewResetPassword',
                                parameter)
98 99
    # generate expiration date
    expiration_date = DateTime() + self._expiration_day
100 101

    # XXX before r26093, _password_request_dict was initialized by an OOBTree and
102 103
    # replaced by a dict on each request, so if it's data structure is not up
    # to date, we update it if needed
104
    if not isinstance(self._password_request_dict, PersistentMapping):
105 106
      LOG('ERP5.PasswordTool', INFO, 'Updating password_request_dict to'
                                     ' PersistentMapping')
107 108
      self._password_request_dict = PersistentMapping()

109
    # register request
110
    self._password_request_dict[random_url] = (user_login, expiration_date)
111 112 113 114 115 116 117 118

    # send mail
    subject = "[%s] Reset of your password" %(self.getPortalObject().getTitle())
    message = "\nYou requested to reset your %s account password.\n\n" \
              "Please copy and paste the following link into your browser: \n%s\n\n" \
              "Please note that this link will be valid only one time, until %s.\n" \
              "After this date, or after having used this link, you will have to make " \
              "a new request\n\n" \
119 120
              "Thank you" %(self.getPortalObject().getTitle(), url, expiration_date)
    self.getPortalObject().portal_notifications.sendMessage(sender=None, recipient=[user,], subject=subject, message=message)
121
    if REQUEST is not None:
122
      msg = translateString("An email has been sent to you.")
123 124 125
      parameter = urlencode(dict(portal_status_message=msg))
      ret_url = '%s/login_form?%s' % (self.getPortalObject().absolute_url(),
                                      parameter)
126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142
      return REQUEST.RESPONSE.redirect( ret_url )

  def _generateUUID(self, args=""):
    """
    Generate a unique id that will be used as url for password
    """
    # this code is based on
    # http://aspn.activestate.com/ASPN/Cookbook/Python/Recipe/213761
    # by Carl Free Jr
    # as uuid module is only available in pyhton 2.5
    t = long( time.time() * 1000 )
    r = long( random.random()*100000000000000000L )
    try:
      a = socket.gethostbyname( socket.gethostname() )
    except:
      # if we can't get a network address, just imagine one
      a = random.random()*100000000000000000L
143
    data = ' '.join((str(t), str(r), str(a), str(args)))
144 145 146 147
    data = md5.md5(data).hexdigest()
    return data


148
  def resetPassword(self, reset_key=None, REQUEST=None):
149 150 151 152
    """
    """
    if REQUEST is None:
      REQUEST = get_request()
153
    user_login, expiration_date = self._password_request_dict.get(reset_key, (None, None))
154
    if reset_key is None or user_login is None:
155 156 157 158 159 160
      ret_url = '%s/login_form' % self.getPortalObject().absolute_url()
      return REQUEST.RESPONSE.redirect( ret_url )

    # check date
    current_date = DateTime()
    if current_date > expiration_date:
161
      msg = translateString("Date has expire.")
162 163 164
      parameter = urlencode(dict(portal_status_message=msg))
      ret_url = '%s/login_form?%s' % (self.getPortalObject().absolute_url(),
                                      parameter)
165
      return REQUEST.RESPONSE.redirect( ret_url )
166

167
    # redirect to form as all is ok
168
    REQUEST.set("password_key", reset_key)
169
    return self.reset_password_form(REQUEST=REQUEST)
170 171 172 173 174 175 176


  def removeExpiredRequests(self, **kw):
    """
    Browse dict and remove expired request
    """
    current_date = DateTime()
177
    for key, (login, date) in self._password_request_dict.items():
178
      if date < current_date:
179 180 181 182 183
        self._password_request_dict.pop(key)


  def changeUserPassword(self, user_login, password, password_confirmation,
                         password_key, REQUEST=None):
184 185 186 187
    """
    Reset the password for a given login    
    """
    # check the key
188 189
    register_user_login, expiration_date = self._password_request_dict.get(
                                                    password_key, (None, None))
190 191 192 193

    current_date = DateTime()
    msg = None
    if register_user_login is None:
194
      msg = "Key not known. Please ask reset password."
195
    elif register_user_login != user_login:
196
      msg = translateString("Bad login provided.")
197
    elif current_date > expiration_date:
198
      msg = translateString("Date has expire.")
199
    elif not password:
Jérome Perrin's avatar
Jérome Perrin committed
200
      msg = translateString("Password must be entered.")
201
    elif password != password_confirmation:
Yusei Tahara's avatar
Yusei Tahara committed
202
      msg = translateString("Passwords do not match.")
203 204
    if msg is not None:
      if REQUEST is not None:
205 206 207
        parameter = urlencode(dict(portal_status_message=msg))
        ret_url = '%s/login_form?%s' % (self.getPortalObject().absolute_url(),
                                        parameter)
208 209 210 211 212
        return REQUEST.RESPONSE.redirect( ret_url )
      else:
        return msg

    # all is OK, change password and remove it from request dict
213 214
    self._password_request_dict.pop(password_key)
    persons = self.getPortalObject().acl_users.erp5_users.getUserByLogin(user_login)
215
    person = persons[0]
216
    person._forceSetPassword(password)
217 218
    person.reindexObject()
    if REQUEST is not None:
219
      msg = translateString("Password changed.")
220 221 222
      parameter = urlencode(dict(portal_status_message=msg))
      ret_url = '%s/login_form?%s' % (self.getPortalObject().absolute_url(),
                                      parameter)
223
      return REQUEST.RESPONSE.redirect( ret_url )
224

225
InitializeClass(PasswordTool)