mirror of
https://github.com/espocrm/espocrm.git
synced 2026-06-28 06:56:05 +00:00
310 lines
8.6 KiB
PHP
310 lines
8.6 KiB
PHP
<?php
|
|
/************************************************************************
|
|
* This file is part of EspoCRM.
|
|
*
|
|
* EspoCRM - Open Source CRM application.
|
|
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
|
|
* Website: http://www.espocrm.com
|
|
*
|
|
* EspoCRM 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 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* EspoCRM 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 EspoCRM. If not, see http://www.gnu.org/licenses/.
|
|
************************************************************************/
|
|
|
|
namespace Espo\Services;
|
|
|
|
use \Espo\Core\Exceptions\Forbidden;
|
|
use \Espo\Core\Exceptions\Error;
|
|
use \Espo\Core\Exceptions\NotFound;
|
|
|
|
use \Espo\ORM\Entity;
|
|
|
|
class User extends Record
|
|
{
|
|
const PASSWORD_CHANGE_REQUEST_LIFETIME = 360; // minutes
|
|
|
|
protected function init()
|
|
{
|
|
$this->dependencies[] = 'container';
|
|
}
|
|
|
|
protected $internalFields = array('password');
|
|
|
|
protected function getMailSender()
|
|
{
|
|
return $this->getContainer()->get('mailSender');
|
|
}
|
|
|
|
protected function getLanguage()
|
|
{
|
|
return $this->getContainer()->get('language');
|
|
}
|
|
|
|
protected function getContainer()
|
|
{
|
|
return $this->injections['container'];
|
|
}
|
|
|
|
public function getEntity($id = null)
|
|
{
|
|
if (isset($id) && $id == 'system') {
|
|
throw new Forbidden();
|
|
}
|
|
|
|
$entity = parent::getEntity($id);
|
|
return $entity;
|
|
}
|
|
|
|
public function findEntities($params)
|
|
{
|
|
if (empty($params['where'])) {
|
|
$params['where'] = array();
|
|
}
|
|
$params['where'][] = array(
|
|
'type' => 'notEquals',
|
|
'field' => 'id',
|
|
'value' => 'system'
|
|
);
|
|
|
|
$result = parent::findEntities($params);
|
|
return $result;
|
|
}
|
|
|
|
public function changePassword($userId, $password, $checkCurrentPassword = false, $currentPassword)
|
|
{
|
|
$user = $this->getEntityManager()->getEntity('User', $userId);
|
|
if (!$user) {
|
|
throw new NotFound();
|
|
}
|
|
|
|
if (empty($password)) {
|
|
throw new Error('Password can\'t be empty.');
|
|
}
|
|
|
|
if ($checkCurrentPassword) {
|
|
$passwordHash = new \Espo\Core\Utils\PasswordHash($this->getConfig());
|
|
$u = $this->getEntityManager()->getRepository('User')->where(array(
|
|
'id' => $user->id,
|
|
'password' => $passwordHash->hash($currentPassword)
|
|
))->findOne();
|
|
if (!$u) {
|
|
throw new Forbidden();
|
|
}
|
|
}
|
|
|
|
$user->set('password', $this->hashPassword($password));
|
|
|
|
$this->getEntityManager()->saveEntity($user);
|
|
|
|
return true;
|
|
}
|
|
|
|
public function passwordChangeRequest($userName, $emailAddress)
|
|
{
|
|
$user = $this->getEntityManager()->getRepository('User')->where(array(
|
|
'userName' => $userName,
|
|
'emailAddress' => $emailAddress
|
|
))->findOne();
|
|
|
|
if (empty($user)) {
|
|
throw new NotFound();
|
|
}
|
|
|
|
if (!$user->isActive()) {
|
|
throw new NotFound();
|
|
}
|
|
|
|
$userId = $user->id;
|
|
|
|
$passwordChangeRequest = $this->getEntityManager()->getRepository('PasswordChangeRequest')->where(array(
|
|
'userId' => $userId
|
|
))->findOne();
|
|
if ($passwordChangeRequest) {
|
|
throw new Forbidden();
|
|
}
|
|
|
|
$requestId = uniqid();
|
|
|
|
$passwordChangeRequest = $this->getEntityManager()->getEntity('PasswordChangeRequest');
|
|
$passwordChangeRequest->set(array(
|
|
'userId' => $userId,
|
|
'requestId' => $requestId
|
|
));
|
|
|
|
$this->sendChangePasswordLink($requestId, $emailAddress);
|
|
|
|
$this->getEntityManager()->saveEntity($passwordChangeRequest);
|
|
|
|
if (!$passwordChangeRequest->id) {
|
|
throw new Error();
|
|
}
|
|
|
|
$dt = new \DateTime();
|
|
$dt->add(new \DateInterval('PT'. self::PASSWORD_CHANGE_REQUEST_LIFETIME . 'M'));
|
|
|
|
$job = $this->getEntityManager()->getEntity('Job');
|
|
|
|
$job->set(array(
|
|
'serviceName' => 'User',
|
|
'method' => 'removeChangePasswordRequestJob',
|
|
'data' => json_encode(array(
|
|
'id' => $passwordChangeRequest->id,
|
|
)),
|
|
'executeTime' => $dt->format('Y-m-d H:i:s') ,
|
|
));
|
|
|
|
$this->getEntityManager()->saveEntity($job);
|
|
|
|
return true;
|
|
}
|
|
|
|
public function removeChangePasswordRequestJob($data)
|
|
{
|
|
if (empty($data['id'])) {
|
|
return;
|
|
}
|
|
$id = $data['id'];
|
|
|
|
$p = $this->getEntityManager()->getEntity('PasswordChangeRequest', $id);
|
|
if ($p) {
|
|
$this->getEntityManager()->removeEntity($p);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
protected function hashPassword($password)
|
|
{
|
|
$config = $this->getConfig();
|
|
$passwordHash = new \Espo\Core\Utils\PasswordHash($config);
|
|
|
|
return $passwordHash->hash($password);
|
|
}
|
|
|
|
public function createEntity($data)
|
|
{
|
|
$newPassword = null;
|
|
if (array_key_exists('password', $data)) {
|
|
$newPassword = $data['password'];
|
|
$data['password'] = $this->hashPassword($data['password']);
|
|
}
|
|
$user = parent::createEntity($data);
|
|
|
|
if (!is_null($newPassword)) {
|
|
if ($user->isActive()) {
|
|
$this->sendPassword($user, $newPassword);
|
|
}
|
|
}
|
|
|
|
return $user;
|
|
}
|
|
|
|
public function updateEntity($id, $data)
|
|
{
|
|
if ($id == 'system') {
|
|
throw new Forbidden();
|
|
}
|
|
$newPassword = null;
|
|
if (array_key_exists('password', $data)) {
|
|
$newPassword = $data['password'];
|
|
$data['password'] = $this->hashPassword($data['password']);
|
|
}
|
|
|
|
if ($id == $this->getUser()->id) {
|
|
unset($data['isActive']);
|
|
}
|
|
$user = parent::updateEntity($id, $data);
|
|
|
|
if (!is_null($newPassword)) {
|
|
try {
|
|
if ($user->isActive()) {
|
|
$this->sendPassword($user, $newPassword);
|
|
}
|
|
} catch (\Exception $e) {}
|
|
}
|
|
|
|
return $user;
|
|
}
|
|
|
|
protected function sendPassword(Entity $user, $password)
|
|
{
|
|
$emailAddress = $user->get('emailAddress');
|
|
|
|
if (empty($emailAddress)) {
|
|
return;
|
|
}
|
|
|
|
$email = $this->getEntityManager()->getEntity('Email');
|
|
|
|
if (!$this->getConfig()->get('smtpServer')) {
|
|
return;
|
|
}
|
|
|
|
|
|
$subject = $this->getLanguage()->translate('accountInfoEmailSubject', 'messages', 'User');
|
|
$body = $this->getLanguage()->translate('accountInfoEmailBody', 'messages', 'User');
|
|
|
|
$body = str_replace('{userName}', $user->get('userName'), $body);
|
|
$body = str_replace('{password}', $password, $body);
|
|
$body = str_replace('{siteUrl}', $this->getConfig()->get('siteUrl'), $body);
|
|
|
|
$email->set(array(
|
|
'subject' => $subject,
|
|
'body' => $body,
|
|
'isHtml' => false,
|
|
'to' => $emailAddress
|
|
));
|
|
|
|
$this->getMailSender()->send($email);
|
|
}
|
|
|
|
protected function sendChangePasswordLink($requestId, $emailAddress, Entity $user = null)
|
|
{
|
|
if (empty($emailAddress)) {
|
|
return;
|
|
}
|
|
|
|
$email = $this->getEntityManager()->getEntity('Email');
|
|
|
|
if (!$this->getConfig()->get('smtpServer')) {
|
|
return;
|
|
}
|
|
|
|
$subject = $this->getLanguage()->translate('passwordChangeLinkEmailSubject', 'messages', 'User');
|
|
$body = $this->getLanguage()->translate('passwordChangeLinkEmailBody', 'messages', 'User');
|
|
|
|
$link = $this->getConfig()->get('siteUrl') . '?entryPoint=changePassword&id=' . $requestId;
|
|
|
|
$body = str_replace('{link}', $link, $body);
|
|
|
|
$email->set(array(
|
|
'subject' => $subject,
|
|
'body' => $body,
|
|
'isHtml' => false,
|
|
'to' => $emailAddress
|
|
));
|
|
|
|
$this->getMailSender()->send($email);
|
|
}
|
|
|
|
public function deleteEntity($id)
|
|
{
|
|
if ($id == 'system') {
|
|
throw new Forbidden();
|
|
}
|
|
if ($id == $this->getUser()->id) {
|
|
throw new Forbidden();
|
|
}
|
|
return parent::deleteEntity($id);
|
|
}
|
|
}
|
|
|