mirror of
https://github.com/espocrm/espocrm.git
synced 2026-03-03 20:27:01 +00:00
Compare commits
263 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f896a2d71a | ||
|
|
dd6704ace5 | ||
|
|
1dc4d44a65 | ||
|
|
bfb28ea178 | ||
|
|
ec6f3a22f2 | ||
|
|
5a0c7c330c | ||
|
|
f64df5af87 | ||
|
|
d4dc7a4051 | ||
|
|
d676f85c8f | ||
|
|
ab382f2387 | ||
|
|
206219c738 | ||
|
|
37d1c707cb | ||
|
|
93af1c9bfc | ||
|
|
a021c4c8d5 | ||
|
|
a125244cdf | ||
|
|
1cfd251c4c | ||
|
|
d2f4f312e5 | ||
|
|
c468b061d9 | ||
|
|
7bf945f0b6 | ||
|
|
fecbb26cbf | ||
|
|
a5ae33ab81 | ||
|
|
c6fa0e464e | ||
|
|
38bae6238a | ||
|
|
79de4c874f | ||
|
|
814748ec61 | ||
|
|
1f0ad0cbec | ||
|
|
7224f566d6 | ||
|
|
eefb01ec4f | ||
|
|
24d46ed81d | ||
|
|
37c749faf8 | ||
|
|
4306a3131e | ||
|
|
8fa95fcce3 | ||
|
|
b7c41ce640 | ||
|
|
a21be94ed3 | ||
|
|
9d59edcae2 | ||
|
|
deaa26a355 | ||
|
|
25d6fb6d82 | ||
|
|
c1bcc44f04 | ||
|
|
fdf8183385 | ||
|
|
db40426f00 | ||
|
|
eb92e648a0 | ||
|
|
5df26c324f | ||
|
|
c36a064fdc | ||
|
|
a28baa6a75 | ||
|
|
27aed29ddf | ||
|
|
744fb176cc | ||
|
|
5f8a0736e4 | ||
|
|
7fb179e769 | ||
|
|
48239c53ca | ||
|
|
4e18afb80f | ||
|
|
e9d3d7c807 | ||
|
|
60923197e5 | ||
|
|
a9f7c90323 | ||
|
|
cc723095c2 | ||
|
|
373b77f83f | ||
|
|
68ab589f3e | ||
|
|
23ccbb226f | ||
|
|
48edf2a2b5 | ||
|
|
082c65ef05 | ||
|
|
6553d8ec6c | ||
|
|
c54b6fcc7c | ||
|
|
655ee740e0 | ||
|
|
b02f77b8f9 | ||
|
|
351c46af06 | ||
|
|
d3db25d98a | ||
|
|
ef1fe1bd1d | ||
|
|
6c2cd93826 | ||
|
|
4859f54f42 | ||
|
|
749c2dc1e9 | ||
|
|
3d13026084 | ||
|
|
6cf66aa88e | ||
|
|
d10ef7038b | ||
|
|
cda612810d | ||
|
|
a7aaac513d | ||
|
|
329dbdf408 | ||
|
|
007f705904 | ||
|
|
36cddbe0cf | ||
|
|
e9409ccf72 | ||
|
|
b5a1ede962 | ||
|
|
2dc6951ef9 | ||
|
|
66f686c013 | ||
|
|
937aab2b1c | ||
|
|
43bee97055 | ||
|
|
a31f5ea87a | ||
|
|
0c4d5f3405 | ||
|
|
4e70ea0586 | ||
|
|
7736f6cd76 | ||
|
|
2db6eaa344 | ||
|
|
38b6cbda0e | ||
|
|
9292f18b5b | ||
|
|
193cf2438a | ||
|
|
155e0e3841 | ||
|
|
019c03ed2f | ||
|
|
316e54df92 | ||
|
|
40691ae899 | ||
|
|
ab8ba168ac | ||
|
|
b066d91cf5 | ||
|
|
2833581cde | ||
|
|
279c8b3188 | ||
|
|
eca6408415 | ||
|
|
35cfb1a480 | ||
|
|
30f3d4ab5e | ||
|
|
7e5424e40a | ||
|
|
82612326bf | ||
|
|
da2e5c835a | ||
|
|
af8eb51c76 | ||
|
|
7253e257f1 | ||
|
|
37e93edf25 | ||
|
|
36a56f050f | ||
|
|
31c2d1360d | ||
|
|
4addc48339 | ||
|
|
a7267bc920 | ||
|
|
93e94f40c3 | ||
|
|
bd75790c29 | ||
|
|
f1cffcae38 | ||
|
|
9111bce47a | ||
|
|
73ec161ac4 | ||
|
|
495a07639e | ||
|
|
044223e70e | ||
|
|
42c9250995 | ||
|
|
8b4a1d96fd | ||
|
|
fff06d7e92 | ||
|
|
726adc550c | ||
|
|
f5bde670f6 | ||
|
|
050873dd3a | ||
|
|
d3886c1c94 | ||
|
|
84f03cf3f9 | ||
|
|
f8953d56f4 | ||
|
|
ebbf14160b | ||
|
|
4df0a09de0 | ||
|
|
c485aa9a62 | ||
|
|
ae474022b9 | ||
|
|
f994cdf7c4 | ||
|
|
7ff514e5a7 | ||
|
|
344403de54 | ||
|
|
836243d170 | ||
|
|
0dc9129d79 | ||
|
|
61c9d07ad8 | ||
|
|
a238295ba8 | ||
|
|
dbabd0dd45 | ||
|
|
f87ff8db8a | ||
|
|
60d121f1e1 | ||
|
|
1497abff07 | ||
|
|
935ba7d4e4 | ||
|
|
fba79cde8f | ||
|
|
ce79d14a29 | ||
|
|
14b2083226 | ||
|
|
779a841817 | ||
|
|
3647523150 | ||
|
|
eb5b941fe6 | ||
|
|
c542d99536 | ||
|
|
918031a9c6 | ||
|
|
163e0cac2a | ||
|
|
95096dd8fd | ||
|
|
67bd98bf4b | ||
|
|
ac434c00d6 | ||
|
|
0fb8330ecf | ||
|
|
3f969cee5b | ||
|
|
ea5f6300f3 | ||
|
|
0d5e75db23 | ||
|
|
7d667a1d1b | ||
|
|
610922c8e0 | ||
|
|
36e0882542 | ||
|
|
b9979e87af | ||
|
|
bcd38dd853 | ||
|
|
b018580c0b | ||
|
|
38a7f42a1e | ||
|
|
1387d856ba | ||
|
|
f2bbc872d6 | ||
|
|
c74e7b416e | ||
|
|
f008688c14 | ||
|
|
dff0d5a992 | ||
|
|
6fe6f8960f | ||
|
|
0b0184098d | ||
|
|
647515d21a | ||
|
|
2ee944ef7b | ||
|
|
f4d2325b8b | ||
|
|
c0904125eb | ||
|
|
7dad30ad0c | ||
|
|
d6acb6dfdc | ||
|
|
b6da94fe2e | ||
|
|
dc5e292a02 | ||
|
|
4bf938107d | ||
|
|
ec7d49cdbc | ||
|
|
3c73062b91 | ||
|
|
5122f112d1 | ||
|
|
5e20fa6717 | ||
|
|
d4599d9377 | ||
|
|
adb9ce4d7e | ||
|
|
e777413a7b | ||
|
|
2cb0ac6221 | ||
|
|
8f7fafb990 | ||
|
|
158c911787 | ||
|
|
aa67575ecb | ||
|
|
81cf82c99f | ||
|
|
e638bf2eec | ||
|
|
6c9d1dbb3d | ||
|
|
fbb034ef92 | ||
|
|
335601d6b8 | ||
|
|
f6ad51ca74 | ||
|
|
e119f8b008 | ||
|
|
5af0eeff3b | ||
|
|
f685feb312 | ||
|
|
c78254a8fa | ||
|
|
9c3758b92c | ||
|
|
4ca2a7fa1a | ||
|
|
d6814b1601 | ||
|
|
55e5a21dcd | ||
|
|
792f422f76 | ||
|
|
8cac546087 | ||
|
|
0cfe91d960 | ||
|
|
4dea1762e8 | ||
|
|
b1a9b8d8b1 | ||
|
|
6787acce61 | ||
|
|
eb0adf7c28 | ||
|
|
783cccaa1b | ||
|
|
36a0a22996 | ||
|
|
a3f3357b2e | ||
|
|
cbbfa44174 | ||
|
|
812d2ec9bc | ||
|
|
b51ba96bb7 | ||
|
|
4c60cf79f6 | ||
|
|
8451949805 | ||
|
|
af5750cdf6 | ||
|
|
1a4dfd6f67 | ||
|
|
6aa800d453 | ||
|
|
82aa3b9508 | ||
|
|
6910a113da | ||
|
|
bd20aaa577 | ||
|
|
0aef3c0b04 | ||
|
|
2f97010b54 | ||
|
|
5bcdad2996 | ||
|
|
1e8a8d94c2 | ||
|
|
9aeefd7685 | ||
|
|
f72385471c | ||
|
|
f7a542560a | ||
|
|
400f43447d | ||
|
|
866593a831 | ||
|
|
c90cdc62fb | ||
|
|
d5c93f21b5 | ||
|
|
b58d78a29e | ||
|
|
f5b41eb78b | ||
|
|
c2a7d90944 | ||
|
|
b2a4ec238c | ||
|
|
df69584c7d | ||
|
|
574da55be7 | ||
|
|
52b808b902 | ||
|
|
714c7b0a33 | ||
|
|
3d774e3afa | ||
|
|
c91db1699b | ||
|
|
2b5695d8dc | ||
|
|
b562fc33bd | ||
|
|
425414b8f3 | ||
|
|
3b363d5ee3 | ||
|
|
4fc3b9a99e | ||
|
|
ab2b9bafeb | ||
|
|
e99fc6302f | ||
|
|
82996a807f | ||
|
|
86b31b1f26 | ||
|
|
19c9b38f82 | ||
|
|
b1d0d1cd27 | ||
|
|
c916011530 | ||
|
|
5db6327272 |
@@ -29,12 +29,12 @@
|
||||
|
||||
namespace Espo\Acl;
|
||||
|
||||
use \Espo\Entities\User;
|
||||
use \Espo\Entities\User as EntityUser;
|
||||
use \Espo\ORM\Entity;
|
||||
|
||||
class Attachment extends \Espo\Core\Acl\Base
|
||||
{
|
||||
public function checkEntityRead(User $user, Entity $entity, $data)
|
||||
public function checkEntityRead(EntityUser $user, Entity $entity, $data)
|
||||
{
|
||||
if ($user->isAdmin()) {
|
||||
return true;
|
||||
@@ -82,7 +82,7 @@ class Attachment extends \Espo\Core\Acl\Base
|
||||
return false;
|
||||
}
|
||||
|
||||
public function checkIsOwner(User $user, Entity $entity)
|
||||
public function checkIsOwner(EntityUser $user, Entity $entity)
|
||||
{
|
||||
if ($user->id === $entity->get('createdById')) {
|
||||
return true;
|
||||
|
||||
@@ -29,13 +29,13 @@
|
||||
|
||||
namespace Espo\Acl;
|
||||
|
||||
use \Espo\Entities\User;
|
||||
use \Espo\Entities\User as EntityUser;
|
||||
use \Espo\ORM\Entity;
|
||||
|
||||
class Email extends \Espo\Core\Acl\Base
|
||||
{
|
||||
|
||||
public function checkEntityRead(User $user, Entity $entity, $data)
|
||||
public function checkEntityRead(EntityUser $user, Entity $entity, $data)
|
||||
{
|
||||
if ($this->checkEntity($user, $entity, $data, 'read')) {
|
||||
return true;
|
||||
@@ -60,7 +60,7 @@ class Email extends \Espo\Core\Acl\Base
|
||||
return false;
|
||||
}
|
||||
|
||||
public function checkIsOwner(User $user, Entity $entity)
|
||||
public function checkIsOwner(EntityUser $user, Entity $entity)
|
||||
{
|
||||
if ($user->id === $entity->get('assignedUserId')) {
|
||||
return true;
|
||||
@@ -76,5 +76,46 @@ class Email extends \Espo\Core\Acl\Base
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
public function checkEntityDelete(EntityUser $user, Entity $entity, $data)
|
||||
{
|
||||
if ($user->isAdmin()) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if ($data === false) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ($data->delete === 'own') {
|
||||
if ($user->id === $entity->get('assignedUserId')) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if ($user->id === $entity->get('createdById')) {
|
||||
return true;
|
||||
}
|
||||
|
||||
$assignedUserIdList = $entity->getLinkMultipleIdList('assignedUsers');
|
||||
if (count($assignedUserIdList) === 1 && $entity->hasLinkMultipleId('assignedUsers', $user->id)) {
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
if ($this->checkEntity($user, $entity, $data, 'delete')) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if ($data->edit !== 'no' || $data->create !== 'no') {
|
||||
if ($entity->get('createdById') === $user->id) {
|
||||
if ($entity->get('status') !== 'Sent' && $entity->get('status') !== 'Archived') {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -29,12 +29,12 @@
|
||||
|
||||
namespace Espo\Acl;
|
||||
|
||||
use \Espo\Entities\User;
|
||||
use \Espo\Entities\User as EntityUser;
|
||||
use \Espo\ORM\Entity;
|
||||
|
||||
class EmailFilter extends \Espo\Core\Acl\Base
|
||||
{
|
||||
public function checkIsOwner(User $user, Entity $entity)
|
||||
public function checkIsOwner(EntityUser $user, Entity $entity)
|
||||
{
|
||||
if ($entity->has('parentId') && $entity->has('parentType')) {
|
||||
$parentType = $entity->get('parentType');
|
||||
@@ -42,11 +42,14 @@ class EmailFilter extends \Espo\Core\Acl\Base
|
||||
if (!$parentType || !$parentId) return;
|
||||
|
||||
$parent = $this->getEntityManager()->getEntity($parentType, $parentId);
|
||||
|
||||
if ($parent->getEntityType() === 'User') {
|
||||
return $parent->id === $user->id;
|
||||
}
|
||||
if ($parent && $parent->has('assignedUserId') && $parent->get('assignedUserId') === $user->id) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -29,12 +29,12 @@
|
||||
|
||||
namespace Espo\Acl;
|
||||
|
||||
use \Espo\Entities\User;
|
||||
use \Espo\Entities\User as EntityUser;
|
||||
use \Espo\ORM\Entity;
|
||||
|
||||
class Notification extends \Espo\Core\Acl\Base
|
||||
{
|
||||
public function checkIsOwner(User $user, Entity $entity)
|
||||
public function checkIsOwner(EntityUser $user, Entity $entity)
|
||||
{
|
||||
if ($user->id === $entity->get('userId')) {
|
||||
return true;
|
||||
|
||||
@@ -29,12 +29,12 @@
|
||||
|
||||
namespace Espo\AclPortal;
|
||||
|
||||
use \Espo\Entities\User;
|
||||
use \Espo\Entities\User as EntityUser;
|
||||
use \Espo\ORM\Entity;
|
||||
|
||||
class Attachment extends \Espo\Core\AclPortal\Base
|
||||
{
|
||||
public function checkEntityRead(User $user, Entity $entity, $data)
|
||||
public function checkEntityRead(EntityUser $user, Entity $entity, $data)
|
||||
{
|
||||
if ($user->isAdmin()) {
|
||||
return true;
|
||||
@@ -82,7 +82,7 @@ class Attachment extends \Espo\Core\AclPortal\Base
|
||||
return false;
|
||||
}
|
||||
|
||||
public function checkIsOwner(User $user, Entity $entity)
|
||||
public function checkIsOwner(EntityUser $user, Entity $entity)
|
||||
{
|
||||
if ($user->id === $entity->get('createdById')) {
|
||||
return true;
|
||||
|
||||
@@ -29,13 +29,13 @@
|
||||
|
||||
namespace Espo\AclPortal;
|
||||
|
||||
use \Espo\Entities\User;
|
||||
use \Espo\Entities\User as EntityUser;
|
||||
use \Espo\ORM\Entity;
|
||||
|
||||
class Email extends \Espo\Core\AclPortal\Base
|
||||
{
|
||||
|
||||
public function checkEntityRead(User $user, Entity $entity, $data)
|
||||
public function checkEntityRead(EntityUser $user, Entity $entity, $data)
|
||||
{
|
||||
if ($this->checkEntity($user, $entity, $data, 'read')) {
|
||||
return true;
|
||||
@@ -60,7 +60,7 @@ class Email extends \Espo\Core\AclPortal\Base
|
||||
return false;
|
||||
}
|
||||
|
||||
public function checkIsOwner(User $user, Entity $entity)
|
||||
public function checkIsOwner(EntityUser $user, Entity $entity)
|
||||
{
|
||||
if ($user->id === $entity->get('createdById')) {
|
||||
return true;
|
||||
|
||||
@@ -29,12 +29,12 @@
|
||||
|
||||
namespace Espo\AclPortal;
|
||||
|
||||
use \Espo\Entities\User;
|
||||
use \Espo\Entities\User as EntityUser;
|
||||
use \Espo\ORM\Entity;
|
||||
|
||||
class Notification extends \Espo\Core\AclPortal\Base
|
||||
{
|
||||
public function checkIsOwner(User $user, Entity $entity)
|
||||
public function checkIsOwner(EntityUser $user, Entity $entity)
|
||||
{
|
||||
if ($user->id === $entity->get('userId')) {
|
||||
return true;
|
||||
|
||||
@@ -47,8 +47,21 @@ class App extends \Espo\Core\Controllers\Base
|
||||
$user->loadLinkMultipleField('accounts');
|
||||
}
|
||||
|
||||
$userData = $user->getValues();
|
||||
|
||||
$emailAddressList = [];
|
||||
foreach ($user->get('emailAddresses') as $emailAddress) {
|
||||
if ($emailAddress->get('invalid')) continue;
|
||||
if ($user->get('emailAddrses') === $emailAddress->get('name')) continue;
|
||||
$emailAddressList[] = $emailAddress->get('name');
|
||||
}
|
||||
if ($user->get('emailAddrses')) {
|
||||
array_unshift($emailAddressList, $user->get('emailAddrses'));
|
||||
}
|
||||
$userData['emailAddressList'] = $emailAddressList;
|
||||
|
||||
return array(
|
||||
'user' => $user->getValues(),
|
||||
'user' => $userData,
|
||||
'acl' => $this->getAcl()->getMap(),
|
||||
'preferences' => $preferences,
|
||||
'token' => $this->getUser()->get('token')
|
||||
|
||||
@@ -32,6 +32,7 @@ namespace Espo\Controllers;
|
||||
use \Espo\Core\Exceptions\BadRequest;
|
||||
use \Espo\Core\Exceptions\Forbidden;
|
||||
use \Espo\Core\Exceptions\Error;
|
||||
use \Espo\Core\Exceptions\NotFound;
|
||||
|
||||
class Email extends \Espo\Core\Controllers\Record
|
||||
{
|
||||
@@ -51,22 +52,48 @@ class Email extends \Espo\Core\Controllers\Record
|
||||
throw new BadRequest();
|
||||
}
|
||||
|
||||
if (!$this->getAcl()->checkScope('Email')) {
|
||||
throw new Forbidden();
|
||||
}
|
||||
|
||||
if (is_null($data['password'])) {
|
||||
if ($data['type'] == 'preferences') {
|
||||
if (!$this->getUser()->isAdmin() && $data['id'] != $this->getUser()->id) {
|
||||
if (!$this->getUser()->isAdmin() && $data['id'] !== $this->getUser()->id) {
|
||||
throw new Forbidden();
|
||||
}
|
||||
$preferences = $this->getEntityManager()->getEntity('Preferences', $data['id']);
|
||||
if (!$preferences) {
|
||||
throw new Error();
|
||||
throw new NotFound();
|
||||
}
|
||||
|
||||
$data['password'] = $this->getContainer()->get('crypt')->decrypt($preferences->get('smtpPassword'));
|
||||
if (is_null($data['password'])) {
|
||||
$data['password'] = $this->getContainer()->get('crypt')->decrypt($preferences->get('smtpPassword'));
|
||||
}
|
||||
} else if ($data['type'] == 'emailAccount') {
|
||||
if (!$this->getAcl()->checkScope('EmailAccount')) {
|
||||
throw new Forbidden();
|
||||
}
|
||||
if (!empty($data['id'])) {
|
||||
$emailAccount = $this->getEntityManager()->getEntity('EmailAccount', $data['id']);
|
||||
if (!$emailAccount) {
|
||||
throw new NotFound();
|
||||
}
|
||||
if (!$this->getUser()->isAdmin()) {
|
||||
if ($emailAccount->get('assigniedUserId') !== $this->getUser()->id) {
|
||||
throw new Forbidden();
|
||||
}
|
||||
}
|
||||
if (is_null($data['password'])) {
|
||||
$data['password'] = $this->getContainer()->get('crypt')->decrypt($emailAccount->get('smtpPassword'));
|
||||
}
|
||||
}
|
||||
} else {
|
||||
if (!$this->getUser()->isAdmin()) {
|
||||
throw new Forbidden();
|
||||
}
|
||||
$data['password'] = $this->getConfig()->get('smtpPassword');
|
||||
if (is_null($data['password'])) {
|
||||
$data['password'] = $this->getConfig()->get('smtpPassword');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -161,5 +188,38 @@ class Email extends \Espo\Core\Controllers\Record
|
||||
}
|
||||
return $this->getRecordService()->retrieveFromTrashByIdList($ids);
|
||||
}
|
||||
|
||||
public function getActionGetFoldersNotReadCounts(&$params, $request, $data)
|
||||
{
|
||||
return $this->getRecordService()->getFoldersNotReadCounts();
|
||||
}
|
||||
|
||||
protected function fetchListParamsFromRequest(&$params, $request, $data)
|
||||
{
|
||||
parent::fetchListParamsFromRequest($params, $request, $data);
|
||||
|
||||
$folderId = $request->get('folderId');
|
||||
if ($folderId) {
|
||||
$params['folderId'] = $request->get('folderId');
|
||||
}
|
||||
}
|
||||
|
||||
public function postActionMoveToFolder($params, $data)
|
||||
{
|
||||
if (!empty($data['ids'])) {
|
||||
$ids = $data['ids'];
|
||||
} else {
|
||||
if (!empty($data['id'])) {
|
||||
$ids = [$data['id']];
|
||||
} else {
|
||||
throw new BadRequest();
|
||||
}
|
||||
}
|
||||
|
||||
if (empty($data['folderId'])) {
|
||||
throw new BadRequest();
|
||||
}
|
||||
return $this->getRecordService()->moveToFolderByIdList($ids, $data['folderId']);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
63
application/Espo/Controllers/EmailFolder.php
Normal file
63
application/Espo/Controllers/EmailFolder.php
Normal file
@@ -0,0 +1,63 @@
|
||||
<?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/.
|
||||
*
|
||||
* The interactive user interfaces in modified source and object code versions
|
||||
* of this program must display Appropriate Legal Notices, as required under
|
||||
* Section 5 of the GNU General Public License version 3.
|
||||
*
|
||||
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
||||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Controllers;
|
||||
|
||||
use \Espo\Core\Exceptions\BadRequest;
|
||||
|
||||
class EmailFolder extends \Espo\Core\Controllers\Record
|
||||
{
|
||||
public function postActionMoveUp($params, $data, $request)
|
||||
{
|
||||
if (empty($data['id'])) {
|
||||
throw new BadRequest();
|
||||
}
|
||||
|
||||
$this->getRecordService()->moveUp($data['id']);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public function postActionMoveDown($params, $data, $request)
|
||||
{
|
||||
if (empty($data['id'])) {
|
||||
throw new BadRequest();
|
||||
}
|
||||
|
||||
$this->getRecordService()->moveDown($data['id']);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public function getActionListAll()
|
||||
{
|
||||
return $this->getRecordService()->listAll();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -86,9 +86,12 @@ class EntityManager extends \Espo\Core\Controllers\Base
|
||||
|
||||
if ($result) {
|
||||
$tabList = $this->getConfig()->get('tabList', []);
|
||||
$tabList[] = $name;
|
||||
$this->getConfig()->set('tabList', $tabList);
|
||||
$this->getConfig()->save();
|
||||
|
||||
if (!in_array($name, $tabList)) {
|
||||
$tabList[] = $name;
|
||||
$this->getConfig()->set('tabList', $tabList);
|
||||
$this->getConfig()->save();
|
||||
}
|
||||
|
||||
$this->getContainer()->get('dataManager')->rebuild();
|
||||
} else {
|
||||
|
||||
@@ -43,12 +43,12 @@ class Import extends \Espo\Core\Controllers\Record
|
||||
}
|
||||
}
|
||||
|
||||
public function actionPatch($params, $data)
|
||||
public function actionPatch($params, $data, $request)
|
||||
{
|
||||
throw new BadRequest();
|
||||
}
|
||||
|
||||
public function actionUpdate($params, $data)
|
||||
public function actionUpdate($params, $data, $request)
|
||||
{
|
||||
throw new BadRequest();
|
||||
}
|
||||
@@ -58,12 +58,12 @@ class Import extends \Espo\Core\Controllers\Record
|
||||
throw new BadRequest();
|
||||
}
|
||||
|
||||
public function actionCreateLink($params, $data)
|
||||
public function actionCreateLink($params, $data, $request)
|
||||
{
|
||||
throw new BadRequest();
|
||||
}
|
||||
|
||||
public function actionRemoveLink($params, $data)
|
||||
public function actionRemoveLink($params, $data, $request)
|
||||
{
|
||||
throw new BadRequest();
|
||||
}
|
||||
|
||||
@@ -132,6 +132,7 @@ class Preferences extends \Espo\Core\Controllers\Base
|
||||
|
||||
$entity->set('smtpEmailAddress', $user->get('emailAddress'));
|
||||
$entity->set('name', $user->get('name'));
|
||||
$entity->set('isPortalUser', $user->get('isPortalUser'));
|
||||
|
||||
$entity->clear('smtpPassword');
|
||||
|
||||
|
||||
@@ -90,4 +90,23 @@ class Settings extends \Espo\Core\Controllers\Base
|
||||
|
||||
return $this->getConfigData();
|
||||
}
|
||||
|
||||
public function postActionTestLdapConnection($params, $data)
|
||||
{
|
||||
if (!$this->getUser()->isAdmin()) {
|
||||
throw new Forbidden();
|
||||
}
|
||||
|
||||
if (!isset($data['password'])) {
|
||||
$data['password'] = $this->getConfig()->get('ldapPassword');
|
||||
}
|
||||
|
||||
$ldapUtils = new \Espo\Core\Utils\Authentication\LDAP\Utils();
|
||||
$options = $ldapUtils->normalizeOptions($data);
|
||||
|
||||
$ldapClient = new \Espo\Core\Utils\Authentication\LDAP\Client($options);
|
||||
$ldapClient->bind(); //an exception if no connection
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -118,5 +118,15 @@ class Acl
|
||||
{
|
||||
return $this->getAclManager()->getScopeForbiddenFieldList($this->getUser(), $scope, $action, $thresholdLevel);
|
||||
}
|
||||
|
||||
public function checkUserPermission($target, $permissionType = 'userPermission')
|
||||
{
|
||||
return $this->getAclManager()->checkUserPermission($this->getUser(), $target, $permissionType);
|
||||
}
|
||||
|
||||
public function checkAssignmentPermission($target)
|
||||
{
|
||||
return $this->getAclManager()->checkAssignmentPermission($this->getUser(), $target);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -66,6 +66,13 @@ class Base implements Injectable
|
||||
return $this->injections[$name];
|
||||
}
|
||||
|
||||
protected function addDependencyList(array $list)
|
||||
{
|
||||
foreach ($list as $item) {
|
||||
$this->addDependency($item);
|
||||
}
|
||||
}
|
||||
|
||||
protected function addDependency($name)
|
||||
{
|
||||
$this->dependencies[] = $name;
|
||||
|
||||
@@ -207,6 +207,7 @@ class Table
|
||||
$this->applyDefault($aclTable, $fieldTable);
|
||||
$this->applyDisabled($aclTable, $fieldTable);
|
||||
$this->applyMandatory($aclTable, $fieldTable);
|
||||
$this->applyAdditional($aclTable, $fieldTable, $valuePermissionLists);
|
||||
} else {
|
||||
$aclTable = (object) [];
|
||||
foreach ($this->getScopeList() as $scope) {
|
||||
@@ -524,6 +525,19 @@ class Table
|
||||
}
|
||||
}
|
||||
|
||||
protected function applyAdditional(&$table, &$fieldTable, &$valuePermissionLists)
|
||||
{
|
||||
if ($this->getUser()->get('isPortalUser')) {
|
||||
foreach ($this->getScopeList() as $scope) {
|
||||
$table->$scope = false;
|
||||
unset($fieldTable->$scope);
|
||||
}
|
||||
foreach ($this->valuePermissionList as $permission) {
|
||||
$valuePermissionLists->{$permission}[] = 'no';
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private function mergeValueList(array $list, $defaultValue)
|
||||
{
|
||||
$result = null;
|
||||
|
||||
@@ -184,7 +184,7 @@ class AclManager
|
||||
return $this->getImplementation($entity->getEntityType())->checkIsOwner($user, $entity);
|
||||
}
|
||||
|
||||
public function checkInTeam(User $user, Entity $entity, $action)
|
||||
public function checkInTeam(User $user, Entity $entity)
|
||||
{
|
||||
return $this->getImplementation($entity->getEntityType())->checkInTeam($user, $entity);
|
||||
}
|
||||
@@ -235,5 +235,40 @@ class AclManager
|
||||
if ($user->isAdmin()) return [];
|
||||
return $this->getTable($user)->getScopeForbiddenFieldList($scope, $action, $thresholdLevel);
|
||||
}
|
||||
|
||||
public function checkUserPermission(User $user, $target, $permissionType = 'userPermission')
|
||||
{
|
||||
$permission = $this->get($user, $permissionType);
|
||||
|
||||
if (is_object($target)) {
|
||||
$userId = $target->id;
|
||||
} else {
|
||||
$userId = $target;
|
||||
}
|
||||
|
||||
if ($user->id === $userId) return true;
|
||||
|
||||
if ($permission === 'no') {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ($permission === 'yes') {
|
||||
return true;
|
||||
}
|
||||
|
||||
if ($permission === 'team') {
|
||||
$teamIdList = $user->getLinkMultipleIdList('teams');
|
||||
if (!$this->getContainer()->get('entityManager')->getRepository('User')->checkBelongsToAnyOfTeams($userId, $teamIdList)) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public function checkAssignmentPermission(User $user, $target)
|
||||
{
|
||||
return $this->checkUserPermission($user, $target, 'assignmentPermission');
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -127,5 +127,9 @@ class Table extends \Espo\Core\Acl\Table
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
protected function applyAdditional(&$table, &$fieldTable, &$valuePermissionLists)
|
||||
{
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -315,7 +315,7 @@ class Application
|
||||
return $_GET['portalId'];
|
||||
}
|
||||
if (!empty($_COOKIE['auth-token'])) {
|
||||
$token = $this->getContainer()->get('entityManager')->getRepository('AuthToken')->where(array('token'=>$_COOKIE['auth-token']))->findOne();
|
||||
$token = $this->getContainer()->get('entityManager')->getRepository('AuthToken')->where(array('token' => $_COOKIE['auth-token']))->findOne();
|
||||
|
||||
if ($token && $token->get('portalId')) {
|
||||
return $token->get('portalId');
|
||||
|
||||
@@ -295,7 +295,8 @@ class Container
|
||||
{
|
||||
return new \Espo\Core\Utils\FieldManager(
|
||||
$this->get('metadata'),
|
||||
$this->get('language')
|
||||
$this->get('language'),
|
||||
$this
|
||||
);
|
||||
}
|
||||
|
||||
|
||||
@@ -131,9 +131,7 @@ class Record extends Base
|
||||
$asc = $request->get('asc', 'true') === 'true';
|
||||
$sortBy = $request->get('sortBy');
|
||||
$q = $request->get('q');
|
||||
$primaryFilter = $request->get('primaryFilter');
|
||||
$textFilter = $request->get('textFilter');
|
||||
$boolFilterList = $request->get('boolFilterList');
|
||||
|
||||
if (empty($maxSize)) {
|
||||
$maxSize = self::MAX_SIZE_LIMIT;
|
||||
@@ -151,12 +149,8 @@ class Record extends Base
|
||||
'q' => $q,
|
||||
'textFilter' => $textFilter
|
||||
);
|
||||
if ($request->get('primaryFilter')) {
|
||||
$params['primaryFilter'] = $request->get('primaryFilter');
|
||||
}
|
||||
if ($request->get('boolFilterList')) {
|
||||
$params['boolFilterList'] = $request->get('boolFilterList');
|
||||
}
|
||||
|
||||
$this->fetchListParamsFromRequest($params, $request, $data);
|
||||
|
||||
$result = $this->getRecordService()->findEntities($params);
|
||||
|
||||
@@ -166,6 +160,16 @@ class Record extends Base
|
||||
);
|
||||
}
|
||||
|
||||
protected function fetchListParamsFromRequest(&$params, $request, $data)
|
||||
{
|
||||
if ($request->get('primaryFilter')) {
|
||||
$params['primaryFilter'] = $request->get('primaryFilter');
|
||||
}
|
||||
if ($request->get('boolFilterList')) {
|
||||
$params['boolFilterList'] = $request->get('boolFilterList');
|
||||
}
|
||||
}
|
||||
|
||||
public function actionListLinked($params, $data, $request)
|
||||
{
|
||||
$id = $params['id'];
|
||||
@@ -195,12 +199,8 @@ class Record extends Base
|
||||
'q' => $q,
|
||||
'textFilter' => $textFilter
|
||||
);
|
||||
if ($request->get('primaryFilter')) {
|
||||
$params['primaryFilter'] = $request->get('primaryFilter');
|
||||
}
|
||||
if ($request->get('boolFilterList')) {
|
||||
$params['boolFilterList'] = $request->get('boolFilterList');
|
||||
}
|
||||
|
||||
$this->fetchListParamsFromRequest($params, $request, $data);
|
||||
|
||||
$result = $this->getRecordService()->findLinkedEntities($id, $link, $params);
|
||||
|
||||
@@ -423,5 +423,21 @@ class Record extends Base
|
||||
|
||||
return $this->getRecordService()->merge($targetId, $sourceIds, $attributes);
|
||||
}
|
||||
|
||||
public function postActionGetDuplicateAttributes($params, $data, $request)
|
||||
{
|
||||
if (empty($data['id'])) {
|
||||
throw new BadRequest();
|
||||
}
|
||||
|
||||
if (!$this->getAcl()->check($this->name, 'create')) {
|
||||
throw new Forbidden();
|
||||
}
|
||||
if (!$this->getAcl()->check($this->name, 'read')) {
|
||||
throw new Forbidden();
|
||||
}
|
||||
|
||||
return $this->getRecordService()->getDuplicateAttributes($data['id']);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -274,12 +274,18 @@ class CronManager
|
||||
$createdJobIdList = array();
|
||||
foreach ($activeScheduledJobList as $scheduledJob) {
|
||||
$scheduling = $scheduledJob['scheduling'];
|
||||
$cronExpression = \Cron\CronExpression::factory($scheduling);
|
||||
|
||||
try {
|
||||
$cronExpression = \Cron\CronExpression::factory($scheduling);
|
||||
} catch (\Exception $e) {
|
||||
$GLOBALS['log']->error('CronManager (ScheduledJob ['.$scheduledJob['id'].']): Scheduling string error - '. $e->getMessage() . '.');
|
||||
continue;
|
||||
}
|
||||
|
||||
try {
|
||||
$previousDate = $cronExpression->getPreviousRunDate()->format('Y-m-d H:i:s');
|
||||
} catch (\Exception $e) {
|
||||
$GLOBALS['log']->error('CronManager: ScheduledJob ['.$scheduledJob['id'].']: CronExpression - Impossible CRON expression ['.$scheduling.']');
|
||||
$GLOBALS['log']->error('CronManager (ScheduledJob ['.$scheduledJob['id'].']): Unsupported CRON expression ['.$scheduling.']');
|
||||
continue;
|
||||
}
|
||||
|
||||
@@ -287,6 +293,9 @@ class CronManager
|
||||
$previousDate = date('Y-m-d H:i:s');
|
||||
}
|
||||
|
||||
$existingJob = $this->getCronJob()->getJobByScheduledJob($scheduledJob['id'], $previousDate);
|
||||
if ($existingJob) continue;
|
||||
|
||||
$className = $this->getScheduledJobUtil()->get($scheduledJob['job']);
|
||||
if ($className) {
|
||||
if (method_exists($className, 'prepare')) {
|
||||
@@ -300,9 +309,6 @@ class CronManager
|
||||
continue;
|
||||
}
|
||||
|
||||
$existingJob = $this->getCronJob()->getJobByScheduledJob($scheduledJob['id'], $previousDate);
|
||||
if ($existingJob) continue;
|
||||
|
||||
$jobEntity = $this->getEntityManager()->getEntity('Job');
|
||||
$jobEntity->set(array(
|
||||
'name' => $scheduledJob['name'],
|
||||
|
||||
@@ -131,7 +131,7 @@ class HookManager
|
||||
}
|
||||
return $hook;
|
||||
}
|
||||
$GLOBALS['log']->error("Hook class '{$name}' does not exist.");
|
||||
$GLOBALS['log']->error("Hook class '{$className}' does not exist.");
|
||||
}
|
||||
|
||||
/**
|
||||
|
||||
@@ -29,15 +29,16 @@
|
||||
|
||||
namespace Espo\Core\Hooks;
|
||||
|
||||
use \Espo\Core\Interfaces\Injectable;
|
||||
use Espo\Core\Interfaces\Injectable;
|
||||
|
||||
abstract class Base implements Injectable
|
||||
{
|
||||
protected $dependencies = array(
|
||||
'container',
|
||||
'entityManager',
|
||||
'config',
|
||||
'metadata',
|
||||
'acl',
|
||||
'aclManager',
|
||||
'user',
|
||||
);
|
||||
|
||||
@@ -59,6 +60,13 @@ abstract class Base implements Injectable
|
||||
return $this->dependencies;
|
||||
}
|
||||
|
||||
protected function addDependencyList(array $list)
|
||||
{
|
||||
foreach ($list as $item) {
|
||||
$this->addDependency($item);
|
||||
}
|
||||
}
|
||||
|
||||
protected function addDependency($name)
|
||||
{
|
||||
$this->dependencies[] = $name;
|
||||
@@ -74,29 +82,39 @@ abstract class Base implements Injectable
|
||||
$this->injections[$name] = $object;
|
||||
}
|
||||
|
||||
protected function getContainer()
|
||||
{
|
||||
return $this->getInjection('container');
|
||||
}
|
||||
|
||||
protected function getEntityManager()
|
||||
{
|
||||
return $this->injections['entityManager'];
|
||||
return $this->getInjection('entityManager');
|
||||
}
|
||||
|
||||
protected function getUser()
|
||||
{
|
||||
return $this->injections['user'];
|
||||
return $this->getInjection('user');
|
||||
}
|
||||
|
||||
protected function getAcl()
|
||||
{
|
||||
return $this->injections['acl'];
|
||||
return $this->getContainer()->get('acl');
|
||||
}
|
||||
|
||||
protected function getAclManager()
|
||||
{
|
||||
return $this->getInjection('aclManager');
|
||||
}
|
||||
|
||||
protected function getConfig()
|
||||
{
|
||||
return $this->injections['config'];
|
||||
return $this->getInjection('config');
|
||||
}
|
||||
|
||||
protected function getMetadata()
|
||||
{
|
||||
return $this->injections['metadata'];
|
||||
return $this->getInjection('metadata');
|
||||
}
|
||||
|
||||
protected function getRepository()
|
||||
|
||||
@@ -46,11 +46,19 @@ class Htmlizer
|
||||
|
||||
protected $config;
|
||||
|
||||
public function __construct(FileManager $fileManager, DateTime $dateTime, Number $number)
|
||||
protected $acl;
|
||||
|
||||
public function __construct(FileManager $fileManager, DateTime $dateTime, Number $number, $acl = null)
|
||||
{
|
||||
$this->fileManager = $fileManager;
|
||||
$this->dateTime = $dateTime;
|
||||
$this->number = $number;
|
||||
$this->acl = $acl;
|
||||
}
|
||||
|
||||
protected function getAcl()
|
||||
{
|
||||
return $this->acl;
|
||||
}
|
||||
|
||||
protected function formatNumber($value)
|
||||
@@ -68,20 +76,25 @@ class Htmlizer
|
||||
return $value;
|
||||
}
|
||||
|
||||
protected function getDataFromEntity(Entity $entity)
|
||||
protected function getDataFromEntity(Entity $entity, $skipLinks = false)
|
||||
{
|
||||
$data = $entity->toArray();
|
||||
|
||||
|
||||
|
||||
$fieldDefs = $entity->getFields();
|
||||
$fieldList = array_keys($fieldDefs);
|
||||
|
||||
$forbidenAttributeList = [];
|
||||
|
||||
if ($this->getAcl()) {
|
||||
$forbidenAttributeList = $this->getAcl()->getScopeForbiddenAttributeList($entity->getEntityType(), 'read');
|
||||
}
|
||||
|
||||
foreach ($fieldList as $field) {
|
||||
$type = null;
|
||||
if (!empty($fieldDefs[$field]['type'])) {
|
||||
$type = $fieldDefs[$field]['type'];
|
||||
}
|
||||
if (in_array($field, $forbidenAttributeList)) continue;
|
||||
|
||||
|
||||
$type = $entity->getAttributeType($field);
|
||||
|
||||
if ($type == Entity::DATETIME) {
|
||||
if (!empty($data[$field])) {
|
||||
$data[$field] = $this->dateTime->convertSystemDateTime($data[$field]);
|
||||
@@ -116,6 +129,8 @@ class Htmlizer
|
||||
$data[$field][$k] = $this->format($data[$field][$k]);
|
||||
}
|
||||
}
|
||||
} else if ($type === Entity::PASSWORD) {
|
||||
unset($data[$field]);
|
||||
}
|
||||
|
||||
if (array_key_exists($field, $data)) {
|
||||
@@ -123,19 +138,52 @@ class Htmlizer
|
||||
}
|
||||
}
|
||||
|
||||
if (!$skipLinks) {
|
||||
$relationDefs = $entity->getRelations();
|
||||
foreach ($entity->getRelationList() as $relation) {
|
||||
if (
|
||||
!empty($relationDefs[$relation]['type'])
|
||||
&&
|
||||
($entity->getRelationType($relation) === 'belongsTo' || $entity->getRelationType($relation) === 'belongsToParent')
|
||||
) {
|
||||
$relatedEntity = $entity->get($relation);
|
||||
if (!$relatedEntity) continue;
|
||||
if ($this->getAcl()) {
|
||||
if (!$this->getAcl()->check($relatedEntity, 'read')) continue;
|
||||
}
|
||||
|
||||
$data[$relation] = $this->getDataFromEntity($relatedEntity, true);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return $data;
|
||||
}
|
||||
|
||||
public function render(Entity $entity, $template)
|
||||
public function render(Entity $entity, $template, $id = null, $additionalData = array(), $skipLinks = false)
|
||||
{
|
||||
$code = \LightnCandy::compile($template);
|
||||
$id = uniqid('', true);
|
||||
$fileName = 'data/cache/template-' . $id;
|
||||
$this->fileManager->putContents($fileName, $code);
|
||||
$renderer = include($fileName);
|
||||
$this->fileManager->removeFile($fileName);
|
||||
|
||||
$data = $this->getDataFromEntity($entity);
|
||||
$toRemove = false;
|
||||
if ($id === null) {
|
||||
$id = uniqid('', true);
|
||||
$toRemove = true;
|
||||
}
|
||||
|
||||
$fileName = 'data/cache/templates/' . $id . '.php';
|
||||
|
||||
$this->fileManager->putContents($fileName, $code);
|
||||
$renderer = $this->fileManager->getPhpContents($fileName);
|
||||
|
||||
if ($toRemove) {
|
||||
$this->fileManager->removeFile($fileName);
|
||||
}
|
||||
|
||||
$data = $this->getDataFromEntity($entity, $skipLinks);
|
||||
|
||||
foreach ($additionalData as $k => $value) {
|
||||
$data[$k] = $value;
|
||||
}
|
||||
|
||||
$html = $renderer($data);
|
||||
|
||||
|
||||
@@ -25,14 +25,14 @@
|
||||
*
|
||||
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
||||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||||
************************************************************************/
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Core\Interfaces;
|
||||
|
||||
interface Injectable
|
||||
{
|
||||
public function getDependencyList();
|
||||
|
||||
|
||||
public function inject($name, $object);
|
||||
}
|
||||
|
||||
|
||||
43
application/Espo/Core/Loaders/EmailFilterManager.php
Normal file
43
application/Espo/Core/Loaders/EmailFilterManager.php
Normal file
@@ -0,0 +1,43 @@
|
||||
<?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/.
|
||||
*
|
||||
* The interactive user interfaces in modified source and object code versions
|
||||
* of this program must display Appropriate Legal Notices, as required under
|
||||
* Section 5 of the GNU General Public License version 3.
|
||||
*
|
||||
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
||||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Core\Loaders;
|
||||
|
||||
class EmailFilterManager extends Base
|
||||
{
|
||||
public function load()
|
||||
{
|
||||
$emailFilterManager = new \Espo\Core\Utils\EmailFilterManager(
|
||||
$this->getContainer()->get('entityManager')
|
||||
);
|
||||
|
||||
return $emailFilterManager;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -40,6 +40,7 @@ class EntityManager extends Base
|
||||
'port' => $config->get('database.port'),
|
||||
'dbname' => $config->get('database.dbname'),
|
||||
'user' => $config->get('database.user'),
|
||||
'charset' => $config->get('database.charset', 'utf8'),
|
||||
'password' => $config->get('database.password'),
|
||||
'metadata' => $this->getContainer()->get('metadata')->getOrmMetadata(),
|
||||
'repositoryFactoryClassName' => '\\Espo\\Core\\ORM\\RepositoryFactory',
|
||||
|
||||
@@ -39,8 +39,14 @@ class FiltersMatcher
|
||||
|
||||
}
|
||||
|
||||
public function match(Email $email, $filterList = [])
|
||||
public function match(Email $email, $subject, $skipBody = false)
|
||||
{
|
||||
if (is_array($subject) || $subject instanceof \Traversable) {
|
||||
$filterList = $subject;
|
||||
} else {
|
||||
$filterList = [$subject];
|
||||
}
|
||||
|
||||
foreach ($filterList as $filter) {
|
||||
if ($filter->get('from')) {
|
||||
if ($this->matchString(strtolower($filter->get('from')), strtolower($email->get('from')))) {
|
||||
@@ -63,11 +69,24 @@ class FiltersMatcher
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (!$skipBody) {
|
||||
if ($this->matchBody($email, $filterList)) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
public function matchBody(Email $email, $filterList = [])
|
||||
public function matchBody(Email $email, $subject)
|
||||
{
|
||||
if (is_array($subject) || $subject instanceof \Traversable) {
|
||||
$filterList = $subject;
|
||||
} else {
|
||||
$filterList = [$subject];
|
||||
}
|
||||
|
||||
foreach ($filterList as $filter) {
|
||||
if ($filter->get('bodyContains')) {
|
||||
$phraseList = $filter->get('bodyContains');
|
||||
|
||||
@@ -38,16 +38,13 @@ class Importer
|
||||
{
|
||||
private $entityManager;
|
||||
|
||||
private $fileManager;
|
||||
|
||||
private $config;
|
||||
|
||||
private $filtersMatcher;
|
||||
|
||||
public function __construct($entityManager, $fileManager, $config)
|
||||
public function __construct($entityManager, $config)
|
||||
{
|
||||
$this->entityManager = $entityManager;
|
||||
$this->fileManager = $fileManager;
|
||||
$this->config = $config;
|
||||
$this->filtersMatcher = new FiltersMatcher();
|
||||
}
|
||||
@@ -56,27 +53,28 @@ class Importer
|
||||
{
|
||||
return $this->entityManager;
|
||||
}
|
||||
|
||||
protected function getConfig()
|
||||
{
|
||||
return $this->config;
|
||||
}
|
||||
|
||||
protected function getFileManager()
|
||||
{
|
||||
return $this->fileManager;
|
||||
}
|
||||
|
||||
protected function getFiltersMatcher()
|
||||
{
|
||||
return $this->filtersMatcher;
|
||||
}
|
||||
|
||||
public function importMessage($message, $assignedUserId = null, $teamsIdList = [], $userIdList = [], $filterList = [], $fetchOnlyHeader = false)
|
||||
public function importMessage($message, $assignedUserId = null, $teamsIdList = [], $userIdList = [], $filterList = [], $fetchOnlyHeader = false, $folderData = null)
|
||||
{
|
||||
try {
|
||||
$email = $this->getEntityManager()->getEntity('Email');
|
||||
|
||||
$email->set('isBeingImported', true);
|
||||
|
||||
$subject = $message->subject;
|
||||
if (!empty($subject) && is_string($subject)) {
|
||||
$subject = trim($subject);
|
||||
}
|
||||
if ($subject !== '0' && empty($subject)) {
|
||||
$subject = '(No Subject)';
|
||||
}
|
||||
@@ -87,12 +85,14 @@ class Importer
|
||||
$email->set('attachmentsIds', []);
|
||||
if ($assignedUserId) {
|
||||
$email->set('assignedUserId', $assignedUserId);
|
||||
$email->set('assignedUsersIds', [$assignedUserId]);
|
||||
$email->addLinkMultipleId('assignedUsers', $assignedUserId);
|
||||
}
|
||||
$email->set('teamsIds', $teamsIdList);
|
||||
|
||||
if (!empty($userIdList)) {
|
||||
$email->set('usersIds', $userIdList);
|
||||
foreach ($userIdList as $uId) {
|
||||
$email->addLinkMultipleId('users', $uId);
|
||||
}
|
||||
}
|
||||
|
||||
$fromArr = $this->getAddressListFromMessage($message, 'from');
|
||||
@@ -112,7 +112,13 @@ class Importer
|
||||
$email->set('cc', implode(';', $ccArr));
|
||||
$email->set('replyTo', implode(';', $replyToArr));
|
||||
|
||||
if ($this->getFiltersMatcher()->match($email, $filterList)) {
|
||||
if ($folderData) {
|
||||
foreach ($folderData as $uId => $folderId) {
|
||||
$email->setLinkMultipleColumn('users', 'folderId', $uId, $folderId);
|
||||
}
|
||||
}
|
||||
|
||||
if ($this->getFiltersMatcher()->match($email, $filterList, true)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
@@ -137,6 +143,14 @@ class Importer
|
||||
}
|
||||
}
|
||||
|
||||
if ($folderData) {
|
||||
foreach ($folderData as $uId => $folderId) {
|
||||
$email->setLinkMultipleColumn('users', 'folderId', $uId, $folderId);
|
||||
}
|
||||
}
|
||||
|
||||
$duplicate->set('isBeingImported', true);
|
||||
|
||||
$this->getEntityManager()->saveEntity($duplicate);
|
||||
|
||||
if (!empty($teamsIdList)) {
|
||||
@@ -201,6 +215,7 @@ class Importer
|
||||
|
||||
$parentFound = false;
|
||||
|
||||
$replied = null;
|
||||
if (isset($message->inReplyTo) && !empty($message->inReplyTo)) {
|
||||
$arr = explode(' ', $message->inReplyTo);
|
||||
$inReplyTo = $arr[0];
|
||||
@@ -250,6 +265,15 @@ class Importer
|
||||
}
|
||||
}
|
||||
|
||||
if (!$parentFound) {
|
||||
if ($replied && $replied->get('parentId') && $replied->get('parentType')) {
|
||||
$parentFound = $this->getEntityManager()->getEntity($replied->get('parentType'), $replied->get('parentId'));
|
||||
if ($parentFound) {
|
||||
$email->set('parentType', $replied->get('parentType'));
|
||||
$email->set('parentId', $replied->get('parentId'));
|
||||
}
|
||||
}
|
||||
}
|
||||
if (!$parentFound) {
|
||||
$from = $email->get('from');
|
||||
if ($from) {
|
||||
@@ -446,13 +470,10 @@ class Importer
|
||||
$content = base64_decode($content);
|
||||
}
|
||||
|
||||
$attachment->set('size', strlen($content));
|
||||
$attachment->set('contents', $content);
|
||||
|
||||
$this->getEntityManager()->saveEntity($attachment);
|
||||
|
||||
$path = $this->getEntityManager()->getRepository('Attachment')->getFilePath($attachment);
|
||||
$this->getFileManager()->putContents($path, $content);
|
||||
|
||||
if ($disposition == 'attachment') {
|
||||
$attachmentsIds = $email->get('attachmentsIds');
|
||||
$attachmentsIds[] = $attachment->id;
|
||||
|
||||
@@ -34,6 +34,7 @@ use Zend\Mail\Header\HeaderInterface;
|
||||
use Zend\Mime;
|
||||
use Zend\Mail\Storage\Exception;
|
||||
use Zend\Mail\Storage\AbstractStorage;
|
||||
use Zend\Stdlib\ErrorHandler;
|
||||
|
||||
class Message extends \Zend\Mail\Storage\Message
|
||||
{
|
||||
|
||||
@@ -104,10 +104,10 @@ class Sender
|
||||
$opts['connection_config']['ssl'] = strtolower($params['security']);
|
||||
}
|
||||
|
||||
if (in_array('fromName', $params)) {
|
||||
if (array_key_exists('fromName', $params)) {
|
||||
$this->params['fromName'] = $params['fromName'];
|
||||
}
|
||||
if (in_array('fromAddress', $params)) {
|
||||
if (array_key_exists('fromAddress', $params)) {
|
||||
$this->params['fromAddress'] = $params['fromAddress'];
|
||||
}
|
||||
|
||||
@@ -340,15 +340,12 @@ class Sender
|
||||
$message->setEncoding('UTF-8');
|
||||
|
||||
try {
|
||||
$rand = mt_rand(1000, 9999);
|
||||
|
||||
if ($email->get('parentType') && $email->get('parentId')) {
|
||||
$messageId = '' . $email->get('parentType') .'/' . $email->get('parentId') . '/' . time() . '/' . $rand . '@espo';
|
||||
$messageId = $email->get('messageId');
|
||||
if (empty($messageId) || !is_string($messageId) || strlen($messageId) < 4) {
|
||||
$messageId = $this->generateMessageId($email);
|
||||
$email->set('messageId', '<' . $messageId . '>');
|
||||
} else {
|
||||
$messageId = '' . md5($email->get('name')) . '/' . time() . '/' . $rand . '@espo';
|
||||
}
|
||||
if ($email->get('isSystem')) {
|
||||
$messageId .= '-system';
|
||||
$messageId = substr($messageId, 1, strlen($messageId) - 2);
|
||||
}
|
||||
|
||||
$messageIdHeader = new \Zend\Mail\Header\MessageId();
|
||||
@@ -357,7 +354,6 @@ class Sender
|
||||
|
||||
$this->transport->send($message);
|
||||
|
||||
$email->set('messageId', '<' . $messageId . '>');
|
||||
$email->set('status', 'Sent');
|
||||
$email->set('dateSent', date("Y-m-d H:i:s"));
|
||||
} catch (\Exception $e) {
|
||||
@@ -366,5 +362,21 @@ class Sender
|
||||
|
||||
$this->useGlobal();
|
||||
}
|
||||
|
||||
static public function generateMessageId(Email $email)
|
||||
{
|
||||
$rand = mt_rand(1000, 9999);
|
||||
|
||||
if ($email->get('parentType') && $email->get('parentId')) {
|
||||
$messageId = '' . $email->get('parentType') .'/' . $email->get('parentId') . '/' . time() . '/' . $rand . '@espo';
|
||||
} else {
|
||||
$messageId = '' . md5($email->get('name')) . '/' . time() . '/' . $rand . '@espo';
|
||||
}
|
||||
if ($email->get('isSystem')) {
|
||||
$messageId .= '-system';
|
||||
}
|
||||
|
||||
return $messageId;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -53,6 +53,13 @@ class Base implements Injectable
|
||||
{
|
||||
}
|
||||
|
||||
protected function addDependencyList(array $list)
|
||||
{
|
||||
foreach ($list as $item) {
|
||||
$this->addDependency($item);
|
||||
}
|
||||
}
|
||||
|
||||
protected function addDependency($name)
|
||||
{
|
||||
$this->dependencies[] = $name;
|
||||
|
||||
@@ -105,7 +105,27 @@ class Entity extends \Espo\ORM\Entity
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public function setLinkMultipleColumn($field, $column, $id, $value)
|
||||
{
|
||||
$columnsField = $field . 'Columns';
|
||||
if (!$this->hasField($columnsField)) {
|
||||
return;
|
||||
}
|
||||
$object = $this->get($columnsField);
|
||||
if (!isset($object) || !($object instanceof \StdClass)) {
|
||||
$object = (object) [];
|
||||
}
|
||||
if (!isset($object->$id)) {
|
||||
$object->$id = (object) [];
|
||||
}
|
||||
if (!isset($object->$id->$column)) {
|
||||
$object->$id->$column = (object) [];
|
||||
}
|
||||
|
||||
$object->$id->$column = $value;
|
||||
$this->set($columnsField, $object);
|
||||
}
|
||||
|
||||
public function setLinkMultipleIdList($field, array $idList)
|
||||
|
||||
@@ -52,6 +52,13 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
$this->dependencies[] = $name;
|
||||
}
|
||||
|
||||
protected function addDependencyList(array $list)
|
||||
{
|
||||
foreach ($list as $item) {
|
||||
$this->addDependency($item);
|
||||
}
|
||||
}
|
||||
|
||||
public function inject($name, $object)
|
||||
{
|
||||
$this->injections[$name] = $object;
|
||||
@@ -72,6 +79,16 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
return $this->getInjection('metadata');
|
||||
}
|
||||
|
||||
public function __construct($entityType, EntityManager $entityManager, EntityFactory $entityFactory)
|
||||
{
|
||||
parent::__construct($entityType, $entityManager, $entityFactory);
|
||||
$this->init();
|
||||
}
|
||||
|
||||
protected function init()
|
||||
{
|
||||
}
|
||||
|
||||
public function handleSelectParams(&$params)
|
||||
{
|
||||
$this->handleEmailAddressParams($params);
|
||||
@@ -81,7 +98,7 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
|
||||
protected function handleCurrencyParams(&$params)
|
||||
{
|
||||
$entityName = $this->entityName;
|
||||
$entityType = $this->entityType;
|
||||
|
||||
$metadata = $this->getMetadata();
|
||||
|
||||
@@ -89,7 +106,7 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
return;
|
||||
}
|
||||
|
||||
$defs = $metadata->get('entityDefs.' . $entityName);
|
||||
$defs = $metadata->get('entityDefs.' . $entityType);
|
||||
|
||||
foreach ($defs['fields'] as $field => $d) {
|
||||
if (isset($d['type']) && $d['type'] == 'currency') {
|
||||
@@ -101,7 +118,7 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
}
|
||||
$alias = Util::toUnderScore($field) . "_currency_alias";
|
||||
$params['customJoin'] .= "
|
||||
LEFT JOIN currency AS `{$alias}` ON {$alias}.id = ".Util::toUnderScore($entityName).".".Util::toUnderScore($field)."_currency
|
||||
LEFT JOIN currency AS `{$alias}` ON {$alias}.id = ".Util::toUnderScore($entityType).".".Util::toUnderScore($field)."_currency
|
||||
";
|
||||
}
|
||||
}
|
||||
@@ -110,9 +127,9 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
|
||||
protected function handleEmailAddressParams(&$params)
|
||||
{
|
||||
$entityName = $this->entityName;
|
||||
$entityType = $this->entityType;
|
||||
|
||||
$defs = $this->getEntityManager()->getMetadata()->get($entityName);
|
||||
$defs = $this->getEntityManager()->getMetadata()->get($entityType);
|
||||
if (!empty($defs['relations']) && array_key_exists('emailAddresses', $defs['relations'])) {
|
||||
if (empty($params['leftJoins'])) {
|
||||
$params['leftJoins'] = array();
|
||||
@@ -132,9 +149,9 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
|
||||
protected function handlePhoneNumberParams(&$params)
|
||||
{
|
||||
$entityName = $this->entityName;
|
||||
$entityType = $this->entityType;
|
||||
|
||||
$defs = $this->getEntityManager()->getMetadata()->get($entityName);
|
||||
$defs = $this->getEntityManager()->getMetadata()->get($entityType);
|
||||
if (!empty($defs['relations']) && array_key_exists('phoneNumbers', $defs['relations'])) {
|
||||
if (empty($params['leftJoins'])) {
|
||||
$params['leftJoins'] = array();
|
||||
@@ -155,7 +172,7 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
protected function beforeRemove(Entity $entity, array $options = array())
|
||||
{
|
||||
parent::beforeRemove($entity, $options);
|
||||
$this->getEntityManager()->getHookManager()->process($this->entityName, 'beforeRemove', $entity, $options);
|
||||
$this->getEntityManager()->getHookManager()->process($this->entityType, 'beforeRemove', $entity, $options);
|
||||
|
||||
$nowString = date('Y-m-d H:i:s', time());
|
||||
if ($entity->hasAttribute('modifiedAt')) {
|
||||
@@ -169,14 +186,14 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
protected function afterRemove(Entity $entity, array $options = array())
|
||||
{
|
||||
parent::afterRemove($entity, $options);
|
||||
$this->getEntityManager()->getHookManager()->process($this->entityName, 'afterRemove', $entity, $options);
|
||||
$this->getEntityManager()->getHookManager()->process($this->entityType, 'afterRemove', $entity, $options);
|
||||
}
|
||||
|
||||
public function remove(Entity $entity, array $options = array())
|
||||
{
|
||||
$result = parent::remove($entity, $options);
|
||||
if ($result) {
|
||||
$this->getEntityManager()->getHookManager()->process($this->entityName, 'afterRemove', $entity, $options);
|
||||
$this->getEntityManager()->getHookManager()->process($this->entityType, 'afterRemove', $entity, $options);
|
||||
}
|
||||
return $result;
|
||||
}
|
||||
@@ -185,7 +202,7 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
{
|
||||
parent::beforeSave($entity, $options);
|
||||
|
||||
$this->getEntityManager()->getHookManager()->process($this->entityName, 'beforeSave', $entity, $options);
|
||||
$this->getEntityManager()->getHookManager()->process($this->entityType, 'beforeSave', $entity, $options);
|
||||
}
|
||||
|
||||
protected function afterSave(Entity $entity, array $options = array())
|
||||
@@ -196,12 +213,12 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
}
|
||||
parent::afterSave($entity, $options);
|
||||
|
||||
$this->handleEmailAddressSave($entity);
|
||||
$this->handlePhoneNumberSave($entity);
|
||||
$this->handleSpecifiedRelations($entity);
|
||||
$this->handleFileFields($entity);
|
||||
$this->processEmailAddressSave($entity);
|
||||
$this->processPhoneNumberSave($entity);
|
||||
$this->processSpecifiedRelationsSave($entity);
|
||||
$this->processFileFieldsSave($entity);
|
||||
|
||||
$this->getEntityManager()->getHookManager()->process($this->entityName, 'afterSave', $entity, $options);
|
||||
$this->getEntityManager()->getHookManager()->process($this->entityType, 'afterSave', $entity, $options);
|
||||
}
|
||||
|
||||
public function save(Entity $entity, array $options = array())
|
||||
@@ -215,13 +232,17 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
}
|
||||
|
||||
if ($entity->hasAttribute('createdAt')) {
|
||||
$entity->set('createdAt', $nowString);
|
||||
if (empty($options['import']) || !$entity->has('createdAt')) {
|
||||
$entity->set('createdAt', $nowString);
|
||||
}
|
||||
}
|
||||
if ($entity->hasAttribute('modifiedAt')) {
|
||||
$entity->set('modifiedAt', $nowString);
|
||||
}
|
||||
if ($entity->hasAttribute('createdById')) {
|
||||
$entity->set('createdById', $this->entityManager->getUser()->id);
|
||||
if (empty($options['import']) || !$entity->has('createdById')) {
|
||||
$entity->set('createdById', $this->entityManager->getUser()->id);
|
||||
}
|
||||
}
|
||||
|
||||
if ($entity->has('modifiedById')) {
|
||||
@@ -242,13 +263,18 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
}
|
||||
|
||||
if ($entity->has('createdById')) {
|
||||
$restoreData['createdById'] = $entity->get('createdById');
|
||||
if (empty($options['import'])) {
|
||||
$restoreData['createdById'] = $entity->get('createdById');
|
||||
$entity->clear('createdById');
|
||||
}
|
||||
}
|
||||
|
||||
if ($entity->has('createdAt')) {
|
||||
$restoreData['createdAt'] = $entity->get('createdAt');
|
||||
if (empty($options['import'])) {
|
||||
$restoreData['createdAt'] = $entity->get('createdAt');
|
||||
$entity->clear('createdAt');
|
||||
}
|
||||
}
|
||||
$entity->clear('createdById');
|
||||
$entity->clear('createdAt');
|
||||
}
|
||||
$this->restoreData = $restoreData;
|
||||
|
||||
@@ -257,7 +283,7 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
return $result;
|
||||
}
|
||||
|
||||
protected function handleFileFields(Entity $entity)
|
||||
protected function processFileFieldsSave(Entity $entity)
|
||||
{
|
||||
foreach ($entity->getRelations() as $name => $defs) {
|
||||
if (!isset($defs['type']) || !isset($defs['entity'])) continue;
|
||||
@@ -278,23 +304,22 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
}
|
||||
}
|
||||
|
||||
protected function handleEmailAddressSave(Entity $entity)
|
||||
protected function processEmailAddressSave(Entity $entity)
|
||||
{
|
||||
if ($entity->hasRelation('emailAddresses') && $entity->hasAttribute('emailAddress')) {
|
||||
$emailAddressRepository = $this->getEntityManager()->getRepository('EmailAddress')->storeEntityEmailAddress($entity);
|
||||
}
|
||||
}
|
||||
|
||||
protected function handlePhoneNumberSave(Entity $entity)
|
||||
protected function processPhoneNumberSave(Entity $entity)
|
||||
{
|
||||
if ($entity->hasRelation('phoneNumbers') && $entity->hasAttribute('phoneNumber')) {
|
||||
$emailAddressRepository = $this->getEntityManager()->getRepository('PhoneNumber')->storeEntityPhoneNumber($entity);
|
||||
}
|
||||
}
|
||||
|
||||
protected function handleSpecifiedRelations(Entity $entity)
|
||||
protected function processSpecifiedRelationsSave(Entity $entity)
|
||||
{
|
||||
|
||||
$relationTypeList = [$entity::HAS_MANY, $entity::MANY_MANY, $entity::HAS_CHILDREN];
|
||||
foreach ($entity->getRelations() as $name => $defs) {
|
||||
if (in_array($defs['type'], $relationTypeList)) {
|
||||
@@ -358,15 +383,16 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
|
||||
} else {
|
||||
if (!empty($columns)) {
|
||||
foreach ($columns as $columnName => $columnField) {
|
||||
if ($columnData->$id->$columnName != $existingColumnsData->$id->$columnName) {
|
||||
$toUpdateIds[] = $id;
|
||||
if (isset($columnData->$id)) {
|
||||
if ($columnData->$id->$columnName !== $existingColumnsData->$id->$columnName) {
|
||||
$toUpdateIds[] = $id;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
foreach ($specifiedIds as $id) {
|
||||
if (!in_array($id, $existingIds)) {
|
||||
$data = null;
|
||||
|
||||
@@ -31,12 +31,18 @@ namespace Espo\Core\ORM;
|
||||
|
||||
use \Espo\Core\Interfaces\Injectable;
|
||||
|
||||
use \Espo\ORM\EntityFactory;
|
||||
|
||||
abstract class Repository extends \Espo\ORM\Repository implements Injectable
|
||||
{
|
||||
protected $dependencies = array();
|
||||
|
||||
protected $injections = array();
|
||||
|
||||
protected function init()
|
||||
{
|
||||
}
|
||||
|
||||
public function inject($name, $object)
|
||||
{
|
||||
$this->injections[$name] = $object;
|
||||
@@ -51,5 +57,23 @@ abstract class Repository extends \Espo\ORM\Repository implements Injectable
|
||||
{
|
||||
return $this->dependencies;
|
||||
}
|
||||
|
||||
protected function addDependencyList(array $list)
|
||||
{
|
||||
foreach ($list as $item) {
|
||||
$this->addDependency($item);
|
||||
}
|
||||
}
|
||||
|
||||
protected function addDependency($name)
|
||||
{
|
||||
$this->dependencies[] = $name;
|
||||
}
|
||||
|
||||
public function __construct($entityType, EntityManager $entityManager, EntityFactory $entityFactory)
|
||||
{
|
||||
parent::__construct($entityType, $entityManager, $entityFactory);
|
||||
$this->init();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -37,6 +37,10 @@ class AclManager extends \Espo\Core\AclManager
|
||||
{
|
||||
protected $tableClassName = '\\Espo\\Core\\AclPortal\\Table';
|
||||
|
||||
private $mainManager = null;
|
||||
|
||||
private $portal = null;
|
||||
|
||||
public function getImplementation($scope)
|
||||
{
|
||||
if (empty($this->implementationHashMap[$scope])) {
|
||||
@@ -70,6 +74,29 @@ class AclManager extends \Espo\Core\AclManager
|
||||
return $this->implementationHashMap[$scope];
|
||||
}
|
||||
|
||||
public function setMainManager($mainManager)
|
||||
{
|
||||
$this->mainManager = $mainManager;
|
||||
}
|
||||
|
||||
protected function getMainManager()
|
||||
{
|
||||
return $this->mainManager;
|
||||
}
|
||||
|
||||
public function setPortal($portal)
|
||||
{
|
||||
$this->portal = $portal;
|
||||
}
|
||||
|
||||
protected function getPortal()
|
||||
{
|
||||
if ($this->portal) {
|
||||
return $this->portal;
|
||||
}
|
||||
return $this->getContainer()->get('portal');
|
||||
}
|
||||
|
||||
protected function getTable(User $user)
|
||||
{
|
||||
$key = $user->id;
|
||||
@@ -82,7 +109,7 @@ class AclManager extends \Espo\Core\AclManager
|
||||
$fileManager = $this->getContainer()->get('fileManager');
|
||||
$metadata = $this->getContainer()->get('metadata');
|
||||
$fieldManager = $this->getContainer()->get('fieldManager');
|
||||
$portal = $this->getContainer()->get('portal');
|
||||
$portal = $this->getPortal();
|
||||
|
||||
$this->tableHashMap[$key] = new $this->tableClassName($user, $portal, $config, $fileManager, $metadata, $fieldManager);
|
||||
}
|
||||
@@ -118,5 +145,114 @@ class AclManager extends \Espo\Core\AclManager
|
||||
return $this->getImplementation($entity->getEntityType())->checkIsOwnContact($user, $entity);
|
||||
}
|
||||
|
||||
public function getMap(User $user)
|
||||
{
|
||||
if ($this->checkUserIsNotPortal($user)) {
|
||||
return $this->getMainManager()->getMap($user);
|
||||
}
|
||||
return parent::getMap($user);
|
||||
}
|
||||
|
||||
public function getLevel(User $user, $scope, $action)
|
||||
{
|
||||
if ($this->checkUserIsNotPortal($user)) {
|
||||
return $this->getMainManager()->getLevel($user, $scope, $action);
|
||||
}
|
||||
return parent::getLevel($user, $scope, $action);
|
||||
}
|
||||
|
||||
public function get(User $user, $permission)
|
||||
{
|
||||
if ($this->checkUserIsNotPortal($user)) {
|
||||
return $this->getMainManager()->get($user, $permission);
|
||||
}
|
||||
return parent::get($user, $permission);
|
||||
}
|
||||
|
||||
public function checkReadOnlyTeam(User $user, $permission)
|
||||
{
|
||||
if ($this->checkUserIsNotPortal($user)) {
|
||||
return $this->getMainManager()->checkReadOnlyTeam($user, $permission);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public function checkReadOnlyOwn(User $user, $permission)
|
||||
{
|
||||
if ($this->checkUserIsNotPortal($user)) {
|
||||
return $this->getMainManager()->checkReadOnlyOwn($user, $permission);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public function check(User $user, $subject, $action = null)
|
||||
{
|
||||
if ($this->checkUserIsNotPortal($user)) {
|
||||
return $this->getMainManager()->check($user, $subject, $action);
|
||||
}
|
||||
return parent::check($user, $subject, $action);
|
||||
}
|
||||
|
||||
public function checkEntity(User $user, Entity $entity, $action = 'read')
|
||||
{
|
||||
if ($this->checkUserIsNotPortal($user)) {
|
||||
return $this->getMainManager()->checkEntity($user, $entity, $action);
|
||||
}
|
||||
return parent::checkEntity($user, $entity, $action);
|
||||
}
|
||||
|
||||
public function checkIsOwner(User $user, Entity $entity)
|
||||
{
|
||||
if ($this->checkUserIsNotPortal($user)) {
|
||||
return $this->getMainManager()->checkIsOwner($user, $entity);
|
||||
}
|
||||
return parent::checkIsOwner($user, $entity);
|
||||
}
|
||||
|
||||
public function checkInTeam(User $user, Entity $entity)
|
||||
{
|
||||
if ($this->checkUserIsNotPortal($user)) {
|
||||
return $this->getMainManager()->checkInTeam($user, $entity);
|
||||
}
|
||||
return parent::checkInTeam($user, $entity);
|
||||
}
|
||||
|
||||
public function checkScope(User $user, $scope, $action = null)
|
||||
{
|
||||
if ($this->checkUserIsNotPortal($user)) {
|
||||
return $this->getMainManager()->checkScope($user, $scope, $action);
|
||||
}
|
||||
return parent::checkScope($user, $scope, $action);
|
||||
}
|
||||
|
||||
public function checkUser(User $user, $permission, User $entity)
|
||||
{
|
||||
if ($this->checkUserIsNotPortal($user)) {
|
||||
return $this->getMainManager()->checkUser($user, $permission, $entity);
|
||||
}
|
||||
return parent::checkUser($user, $permission, $entity);
|
||||
}
|
||||
|
||||
public function getScopeForbiddenAttributeList(User $user, $scope, $action = 'read', $thresholdLevel = 'no')
|
||||
{
|
||||
if ($this->checkUserIsNotPortal($user)) {
|
||||
return $this->getMainManager()->getScopeForbiddenAttributeList($user, $scope, $action, $thresholdLevel);
|
||||
}
|
||||
return parent::getScopeForbiddenAttributeList($user, $scope, $action, $thresholdLevel);
|
||||
}
|
||||
|
||||
public function getScopeForbiddenFieldList(User $user, $scope, $action = 'read', $thresholdLevel = 'no')
|
||||
{
|
||||
if ($this->checkUserIsNotPortal($user)) {
|
||||
return $this->getMainManager()->getScopeForbiddenFieldList($user, $scope, $action, $thresholdLevel);
|
||||
}
|
||||
return parent::getScopeForbiddenFieldList($user, $scope, $action, $thresholdLevel);
|
||||
}
|
||||
|
||||
protected function checkUserIsNotPortal($user)
|
||||
{
|
||||
return !$user->get('isPortalUser');
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
@@ -49,6 +49,12 @@ class Application extends \Espo\Core\Application
|
||||
|
||||
$portal = $this->getContainer()->get('entityManager')->getEntity('Portal', $portalId);
|
||||
|
||||
if (!$portal) {
|
||||
$portal = $this->getContainer()->get('entityManager')->getRepository('Portal')->where(array(
|
||||
'customId' => $portalId
|
||||
))->findOne();
|
||||
}
|
||||
|
||||
if (!$portal) {
|
||||
throw new NotFound();
|
||||
}
|
||||
|
||||
@@ -38,12 +38,27 @@ class Container extends \Espo\Core\Container
|
||||
return $className;
|
||||
}
|
||||
|
||||
protected function getServiceMainClassName($name, $default)
|
||||
{
|
||||
$metadata = $this->get('metadata');
|
||||
$className = $metadata->get('app.serviceContainer.classNames.' . $name, $default);
|
||||
return $className;
|
||||
}
|
||||
|
||||
protected function loadAclManager()
|
||||
{
|
||||
$className = $this->getServiceClassName('aclManager', '\\Espo\\Core\\Portal\\AclManager');
|
||||
return new $className(
|
||||
$mainClassName = $this->getServiceMainClassName('aclManager', '\\Espo\\Core\\AclManager');
|
||||
|
||||
$obj = new $className(
|
||||
$this->get('container')
|
||||
);
|
||||
$objMain = new $mainClassName(
|
||||
$this->get('container')
|
||||
);
|
||||
$obj->setMainManager($objMain);
|
||||
|
||||
return $obj;
|
||||
}
|
||||
|
||||
protected function loadAcl()
|
||||
|
||||
@@ -77,6 +77,11 @@ class Base
|
||||
return $this->entityManager;
|
||||
}
|
||||
|
||||
protected function getMetadata()
|
||||
{
|
||||
return $this->metadata;
|
||||
}
|
||||
|
||||
protected function getUser()
|
||||
{
|
||||
return $this->user;
|
||||
@@ -117,18 +122,30 @@ class Base
|
||||
}
|
||||
}
|
||||
|
||||
protected function order($sortBy, $asc, &$result)
|
||||
protected function order($sortBy, $desc = false, &$result)
|
||||
{
|
||||
if (!empty($sortBy)) {
|
||||
$result['orderBy'] = $sortBy;
|
||||
$type = $this->metadata->get("entityDefs.{$this->entityType}.fields." . $result['orderBy'] . ".type");
|
||||
if ($type == 'link') {
|
||||
$type = $this->getMetadata()->get(['entityDefs', $this->getEntityType(), 'fields', $sortBy, 'type']);
|
||||
if ($type === 'link') {
|
||||
$result['orderBy'] .= 'Name';
|
||||
} else if ($type == 'linkParent') {
|
||||
} else if ($type === 'linkParent') {
|
||||
$result['orderBy'] .= 'Type';
|
||||
} else if ($type === 'enum') {
|
||||
$list = $this->getMetadata()->get(['entityDefs', $this->getEntityType(), 'fields', $sortBy, 'options']);
|
||||
if ($list && is_array($list) && count($list)) {
|
||||
if ($this->getMetadata()->get(['entityDefs', $this->getEntityType(), 'fields', $sortBy, 'isSorted'])) {
|
||||
$list = asort($list);
|
||||
}
|
||||
if ($desc) {
|
||||
$list = array_reverse($list);
|
||||
}
|
||||
$result['orderBy'] = 'LIST:' . $sortBy . ':' . implode(',', $list);
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
if ($asc) {
|
||||
if (!$desc) {
|
||||
$result['order'] = 'ASC';
|
||||
} else {
|
||||
$result['order'] = 'DESC';
|
||||
@@ -137,7 +154,7 @@ class Base
|
||||
|
||||
protected function getTextFilterFieldList()
|
||||
{
|
||||
return $this->metadata->get("entityDefs.{$this->entityType}.collection.textFilterFields", ['name']);
|
||||
return $this->getMetadata()->get("entityDefs.{$this->entityType}.collection.textFilterFields", ['name']);
|
||||
}
|
||||
|
||||
protected function getSeed()
|
||||
@@ -612,6 +629,11 @@ class Base
|
||||
return $result;
|
||||
}
|
||||
|
||||
public function buildSelectParams(array $params, $withAcl = false, $checkWherePermission = false)
|
||||
{
|
||||
return $this->getSelectParams($params, $withAcl, $checkWherePermission);
|
||||
}
|
||||
|
||||
public function getSelectParams(array $params, $withAcl = false, $checkWherePermission = false)
|
||||
{
|
||||
$result = array();
|
||||
@@ -621,7 +643,7 @@ class Base
|
||||
if (!array_key_exists('asc', $params)) {
|
||||
$params['asc'] = true;
|
||||
}
|
||||
$this->order($params['sortBy'], $params['asc'], $result);
|
||||
$this->order($params['sortBy'], !$params['asc'], $result);
|
||||
}
|
||||
|
||||
if (!isset($params['offset'])) {
|
||||
@@ -684,18 +706,22 @@ class Base
|
||||
}
|
||||
}
|
||||
|
||||
protected function getUserTimeZone()
|
||||
public function getUserTimeZone()
|
||||
{
|
||||
if (empty($this->userTimeZone)) {
|
||||
$preferences = $this->getEntityManager()->getEntity('Preferences', $this->getUser()->id);
|
||||
$timeZone = $preferences->get('timeZone');
|
||||
$this->userTimeZone = $timeZone;
|
||||
if ($preferences) {
|
||||
$timeZone = $preferences->get('timeZone');
|
||||
$this->userTimeZone = $timeZone;
|
||||
} else {
|
||||
$this->userTimeZone = 'UTC';
|
||||
}
|
||||
}
|
||||
|
||||
return $this->userTimeZone;
|
||||
}
|
||||
|
||||
protected function convertDateTimeWhere($item)
|
||||
public function convertDateTimeWhere($item)
|
||||
{
|
||||
$format = 'Y-m-d H:i:s';
|
||||
|
||||
@@ -1039,10 +1065,10 @@ class Base
|
||||
return $part;
|
||||
}
|
||||
|
||||
public function applyOrder($sortBy, $asc, &$result)
|
||||
public function applyOrder($sortBy, $desc, &$result)
|
||||
{
|
||||
$this->prepareResult($result);
|
||||
$this->order($sortBy, $asc, $result);
|
||||
$this->order($sortBy, $desc, $result);
|
||||
}
|
||||
|
||||
public function applyLimit($offset, $maxSize, &$result)
|
||||
@@ -1084,12 +1110,36 @@ class Base
|
||||
|
||||
public function hasJoin($join, &$result)
|
||||
{
|
||||
return in_array($join, $result['joins']);
|
||||
if (in_array($join, $result['joins'])) {
|
||||
return true;
|
||||
}
|
||||
|
||||
foreach ($result['joins'] as $item) {
|
||||
if (is_array($item) && count($item) > 1) {
|
||||
if ($item[1] == $join) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
public function hasLeftJoin($leftJoin, &$result)
|
||||
{
|
||||
return in_array($leftJoin, $result['leftJoins']);
|
||||
if (in_array($leftJoin, $result['leftJoins'])) {
|
||||
return true;
|
||||
}
|
||||
|
||||
foreach ($result['leftJoins'] as $item) {
|
||||
if (is_array($item) && count($item) > 1) {
|
||||
if ($item[1] == $leftJoin) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
public function addJoin($join, &$result)
|
||||
|
||||
@@ -65,6 +65,13 @@ abstract class Base implements Injectable
|
||||
$this->dependencies[] = $name;
|
||||
}
|
||||
|
||||
protected function addDependencyList(array $list)
|
||||
{
|
||||
foreach ($list as $item) {
|
||||
$this->addDependency($item);
|
||||
}
|
||||
}
|
||||
|
||||
public function getDependencyList()
|
||||
{
|
||||
return $this->dependencies;
|
||||
|
||||
36
application/Espo/Core/Templates/Controllers/Event.php
Normal file
36
application/Espo/Core/Templates/Controllers/Event.php
Normal file
@@ -0,0 +1,36 @@
|
||||
<?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/.
|
||||
*
|
||||
* The interactive user interfaces in modified source and object code versions
|
||||
* of this program must display Appropriate Legal Notices, as required under
|
||||
* Section 5 of the GNU General Public License version 3.
|
||||
*
|
||||
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
||||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Core\Templates\Controllers;
|
||||
|
||||
class Event extends \Espo\Core\Controllers\Record
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
36
application/Espo/Core/Templates/Entities/Event.php
Normal file
36
application/Espo/Core/Templates/Entities/Event.php
Normal file
@@ -0,0 +1,36 @@
|
||||
<?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/.
|
||||
*
|
||||
* The interactive user interfaces in modified source and object code versions
|
||||
* of this program must display Appropriate Legal Notices, as required under
|
||||
* Section 5 of the GNU General Public License version 3.
|
||||
*
|
||||
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
||||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Core\Templates\Entities;
|
||||
|
||||
class Event extends \Espo\Core\ORM\Entity
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
39
application/Espo/Core/Templates/Layouts/Event/detail.json
Normal file
39
application/Espo/Core/Templates/Layouts/Event/detail.json
Normal file
@@ -0,0 +1,39 @@
|
||||
[
|
||||
{
|
||||
"label": "Overview",
|
||||
"rows": [
|
||||
[
|
||||
{
|
||||
"name": "name"
|
||||
},
|
||||
false
|
||||
],
|
||||
[
|
||||
{
|
||||
"name": "status"
|
||||
},
|
||||
false
|
||||
],
|
||||
[
|
||||
{
|
||||
"name": "dateStart"
|
||||
},
|
||||
{
|
||||
"name": "dateEnd"
|
||||
}
|
||||
],
|
||||
[
|
||||
{
|
||||
"name": "duration"
|
||||
},
|
||||
false
|
||||
],
|
||||
[
|
||||
{
|
||||
"name": "description",
|
||||
"fullWidth": true
|
||||
}
|
||||
]
|
||||
]
|
||||
}
|
||||
]
|
||||
@@ -0,0 +1,43 @@
|
||||
[
|
||||
{
|
||||
"label": "",
|
||||
"rows": [
|
||||
[
|
||||
{
|
||||
"name": "name",
|
||||
"fullWidth": true
|
||||
}
|
||||
],
|
||||
[
|
||||
{
|
||||
"name": "status",
|
||||
"fullWidth": true
|
||||
}
|
||||
],
|
||||
[
|
||||
{
|
||||
"name": "dateStart",
|
||||
"fullWidth": true
|
||||
}
|
||||
],
|
||||
[
|
||||
{
|
||||
"name": "duration",
|
||||
"fullWidth": true
|
||||
}
|
||||
],
|
||||
[
|
||||
{
|
||||
"name": "dateEnd",
|
||||
"fullWidth": true
|
||||
}
|
||||
],
|
||||
[
|
||||
{
|
||||
"name": "description",
|
||||
"fullWidth": true
|
||||
}
|
||||
]
|
||||
]
|
||||
}
|
||||
]
|
||||
@@ -1,4 +1,4 @@
|
||||
{
|
||||
"controller": "Controllers.Record",
|
||||
"controller": "controllers/record",
|
||||
"boolFilterList": ["onlyMy"]
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"controller": "Controllers.RecordTree",
|
||||
"collection": "Collections.Tree",
|
||||
"controller": "controllers/record-tree",
|
||||
"collection": "collections/tree",
|
||||
"menu": {
|
||||
"listTree": {
|
||||
"buttons": [
|
||||
|
||||
@@ -0,0 +1,4 @@
|
||||
{
|
||||
"controller": "controllers/record",
|
||||
"boolFilterList": ["onlyMy"]
|
||||
}
|
||||
116
application/Espo/Core/Templates/Metadata/Event/entityDefs.json
Normal file
116
application/Espo/Core/Templates/Metadata/Event/entityDefs.json
Normal file
@@ -0,0 +1,116 @@
|
||||
{
|
||||
"fields": {
|
||||
"name": {
|
||||
"type": "varchar",
|
||||
"required": true,
|
||||
"trim": true
|
||||
},
|
||||
"status": {
|
||||
"type": "enum",
|
||||
"options": ["Planned", "Held", "Not Held"],
|
||||
"default": "Planned",
|
||||
"view": "views/fields/enum-styled",
|
||||
"style": {
|
||||
"Held": "success"
|
||||
},
|
||||
"audited": true
|
||||
},
|
||||
"dateStart": {
|
||||
"type": "datetime",
|
||||
"required": true,
|
||||
"default": "javascript: return this.dateTime.getNow(15);",
|
||||
"audited": true
|
||||
},
|
||||
"dateEnd": {
|
||||
"type": "datetime",
|
||||
"required": true,
|
||||
"after": "dateStart"
|
||||
},
|
||||
"duration": {
|
||||
"type": "duration",
|
||||
"start": "dateStart",
|
||||
"end": "dateEnd",
|
||||
"options": [300, 600, 900, 1800, 2700, 3600, 7200],
|
||||
"default": 300,
|
||||
"notStorable": true
|
||||
},
|
||||
"parent": {
|
||||
"type": "linkParent",
|
||||
"entityList": ["Account", "Lead"]
|
||||
},
|
||||
"description": {
|
||||
"type": "text"
|
||||
},
|
||||
"createdAt": {
|
||||
"type": "datetime",
|
||||
"readOnly": true
|
||||
},
|
||||
"modifiedAt": {
|
||||
"type": "datetime",
|
||||
"readOnly": true
|
||||
},
|
||||
"createdBy": {
|
||||
"type": "link",
|
||||
"readOnly": true,
|
||||
"view": "views/fields/user"
|
||||
},
|
||||
"modifiedBy": {
|
||||
"type": "link",
|
||||
"readOnly": true,
|
||||
"view": "views/fields/user"
|
||||
},
|
||||
"assignedUser": {
|
||||
"type": "link",
|
||||
"required": false,
|
||||
"view": "views/fields/assigned-user"
|
||||
},
|
||||
"teams": {
|
||||
"type": "linkMultiple",
|
||||
"view": "views/fields/teams"
|
||||
}
|
||||
},
|
||||
"links": {
|
||||
"parent": {
|
||||
"type": "belongsToParent"
|
||||
},
|
||||
"createdBy": {
|
||||
"type": "belongsTo",
|
||||
"entity": "User"
|
||||
},
|
||||
"modifiedBy": {
|
||||
"type": "belongsTo",
|
||||
"entity": "User"
|
||||
},
|
||||
"assignedUser": {
|
||||
"type": "belongsTo",
|
||||
"entity": "User"
|
||||
},
|
||||
"teams": {
|
||||
"type": "hasMany",
|
||||
"entity": "Team",
|
||||
"relationName": "EntityTeam",
|
||||
"layoutRelationshipsDisabled": true
|
||||
}
|
||||
},
|
||||
"collection": {
|
||||
"sortBy": "dateStart",
|
||||
"asc": false
|
||||
},
|
||||
"indexes": {
|
||||
"dateStartStatus": {
|
||||
"columns": ["dateStart", "status"]
|
||||
},
|
||||
"dateStart": {
|
||||
"columns": ["dateStart", "deleted"]
|
||||
},
|
||||
"status": {
|
||||
"columns": ["status", "deleted"]
|
||||
},
|
||||
"assignedUser": {
|
||||
"columns": ["assignedUserId", "deleted"]
|
||||
},
|
||||
"assignedUserStatus": {
|
||||
"columns": ["assignedUserId", "status"]
|
||||
}
|
||||
}
|
||||
}
|
||||
11
application/Espo/Core/Templates/Metadata/Event/scopes.json
Normal file
11
application/Espo/Core/Templates/Metadata/Event/scopes.json
Normal file
@@ -0,0 +1,11 @@
|
||||
{
|
||||
"entity": true,
|
||||
"layouts": true,
|
||||
"tab": true,
|
||||
"acl": true,
|
||||
"aclPortal": true,
|
||||
"customizable": true,
|
||||
"importable": true,
|
||||
"calendar": true,
|
||||
"notifications": true
|
||||
}
|
||||
@@ -1,4 +1,4 @@
|
||||
{
|
||||
"controller": "Controllers.Record",
|
||||
"controller": "controllers/record",
|
||||
"boolFilterList": ["onlyMy"]
|
||||
}
|
||||
|
||||
@@ -21,10 +21,6 @@
|
||||
"description": {
|
||||
"type": "text"
|
||||
},
|
||||
"createdAt": {
|
||||
"type": "datetime",
|
||||
"readOnly": true
|
||||
},
|
||||
"emailAddress": {
|
||||
"type": "email"
|
||||
},
|
||||
@@ -53,6 +49,10 @@
|
||||
"addressPostalCode": {
|
||||
"type": "varchar"
|
||||
},
|
||||
"createdAt": {
|
||||
"type": "datetime",
|
||||
"readOnly": true
|
||||
},
|
||||
"modifiedAt": {
|
||||
"type": "datetime",
|
||||
"readOnly": true
|
||||
|
||||
36
application/Espo/Core/Templates/Repositories/Event.php
Normal file
36
application/Espo/Core/Templates/Repositories/Event.php
Normal file
@@ -0,0 +1,36 @@
|
||||
<?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/.
|
||||
*
|
||||
* The interactive user interfaces in modified source and object code versions
|
||||
* of this program must display Appropriate Legal Notices, as required under
|
||||
* Section 5 of the GNU General Public License version 3.
|
||||
*
|
||||
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
||||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Core\Templates\Repositories;
|
||||
|
||||
class Event extends \Espo\Core\ORM\Repositories\RDB
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
36
application/Espo/Core/Templates/Services/Event.php
Normal file
36
application/Espo/Core/Templates/Services/Event.php
Normal file
@@ -0,0 +1,36 @@
|
||||
<?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/.
|
||||
*
|
||||
* The interactive user interfaces in modified source and object code versions
|
||||
* of this program must display Appropriate Legal Notices, as required under
|
||||
* Section 5 of the GNU General Public License version 3.
|
||||
*
|
||||
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
||||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Core\Templates\Services;
|
||||
|
||||
class Event extends \Espo\Services\Record
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
@@ -32,6 +32,39 @@ namespace Espo\Core\Templates\Services;
|
||||
|
||||
class Person extends \Espo\Services\Record
|
||||
{
|
||||
protected function getDuplicateWhereClause(Entity $entity, $data = array())
|
||||
{
|
||||
$data = array(
|
||||
'OR' => array(
|
||||
array(
|
||||
'firstName' => $entity->get('firstName'),
|
||||
'lastName' => $entity->get('lastName'),
|
||||
)
|
||||
)
|
||||
);
|
||||
if (
|
||||
($entity->get('emailAddress') || $entity->get('emailAddressData'))
|
||||
&&
|
||||
($entity->isNew() || $entity->isFieldChanged('emailAddress') || $entity->isFieldChanged('emailAddressData'))
|
||||
) {
|
||||
if ($entity->get('emailAddress')) {
|
||||
$list = [$entity->get('emailAddress')];
|
||||
}
|
||||
if ($entity->get('emailAddressData')) {
|
||||
foreach ($entity->get('emailAddressData') as $row) {
|
||||
if (!in_array($row->emailAddress, $list)) {
|
||||
$list[] = $row->emailAddress;
|
||||
}
|
||||
}
|
||||
}
|
||||
foreach ($list as $emailAddress) {
|
||||
$data['OR'][] = array(
|
||||
'emailAddress' => $emailAddress
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
return $data;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -28,49 +28,68 @@
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Core\Utils\Authentication;
|
||||
use Espo\Core\Exceptions\Error,
|
||||
Espo\Core\Utils\Config,
|
||||
Espo\Core\ORM\EntityManager,
|
||||
Espo\Core\Utils\Auth;
|
||||
|
||||
use Espo\Core\Exceptions\Error;
|
||||
use Espo\Core\Utils\Config;
|
||||
use Espo\Core\ORM\EntityManager;
|
||||
use Espo\Core\Utils\Auth;
|
||||
|
||||
class LDAP extends Base
|
||||
{
|
||||
private $utils;
|
||||
|
||||
private $zendLdap;
|
||||
private $ldapClient;
|
||||
|
||||
/**
|
||||
* Espo => LDAP name
|
||||
* User field name => option name (LDAP attribute)
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
private $fields = array(
|
||||
'userName' => 'cn',
|
||||
'firstName' => 'givenname',
|
||||
'lastName' => 'sn',
|
||||
'title' => 'title',
|
||||
'emailAddress' => 'mail',
|
||||
'phoneNumber' => 'telephonenumber',
|
||||
protected $ldapFieldMap = array(
|
||||
'userName' => 'userNameAttribute',
|
||||
'firstName' => 'userFirstNameAttribute',
|
||||
'lastName' => 'userLastNameAttribute',
|
||||
'title' => 'userTitleAttribute',
|
||||
'emailAddress' => 'userEmailAddressAttribute',
|
||||
'phoneNumber' => 'userPhoneNumberAttribute',
|
||||
);
|
||||
|
||||
/**
|
||||
* User field name => option name
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
protected $userFieldMap = array(
|
||||
'teamsIds' => 'userTeamsIds',
|
||||
'defaultTeamId' => 'userDefaultTeamId',
|
||||
);
|
||||
|
||||
public function __construct(Config $config, EntityManager $entityManager, Auth $auth)
|
||||
{
|
||||
parent::__construct($config, $entityManager, $auth);
|
||||
|
||||
$this->zendLdap = new LDAP\LDAP();
|
||||
$this->utils = new LDAP\Utils($config);
|
||||
}
|
||||
|
||||
protected function getZendLdap()
|
||||
{
|
||||
return $this->zendLdap;
|
||||
}
|
||||
|
||||
protected function getUtils()
|
||||
{
|
||||
return $this->utils;
|
||||
}
|
||||
|
||||
protected function getLdapClient()
|
||||
{
|
||||
if (!isset($this->ldapClient)) {
|
||||
$options = $this->getUtils()->getLdapClientOptions();
|
||||
|
||||
try {
|
||||
$this->ldapClient = new LDAP\Client($options);
|
||||
} catch (\Exception $e) {
|
||||
$GLOBALS['log']->error('LDAP error: ' . $e->getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
return $this->ldapClient;
|
||||
}
|
||||
|
||||
/**
|
||||
* LDAP login
|
||||
@@ -78,6 +97,7 @@ class LDAP extends Base
|
||||
* @param string $username
|
||||
* @param string $password
|
||||
* @param \Espo\Entities\AuthToken $authToken
|
||||
*
|
||||
* @return \Espo\Entities\User | null
|
||||
*/
|
||||
public function login($username, $password, \Espo\Entities\AuthToken $authToken = null)
|
||||
@@ -86,28 +106,36 @@ class LDAP extends Base
|
||||
return $this->loginByToken($username, $authToken);
|
||||
}
|
||||
|
||||
$options = $this->getUtils()->getZendOptions();
|
||||
|
||||
$ldap = $this->getZendLdap();
|
||||
$ldap = $ldap->setOptions($options);
|
||||
$ldapClient = $this->getLdapClient();
|
||||
|
||||
//login LDAP system user (ldapUsername, ldapPassword)
|
||||
try {
|
||||
$ldap->bind($username, $password);
|
||||
$ldapClient->bind();
|
||||
} catch (\Exception $e) {
|
||||
$options = $this->getUtils()->getLdapClientOptions();
|
||||
$GLOBALS['log']->error('LDAP: Could not connect to LDAP server ['.$options['host'].'], details: ' . $e->getMessage());
|
||||
|
||||
$dn = $ldap->getDn($username);
|
||||
|
||||
$loginFilter = $this->getUtils()->getOption('userLoginFilter');
|
||||
$userData = $ldap->searchByLoginFilter($loginFilter, $dn, 3);
|
||||
|
||||
} catch (\Zend\Ldap\Exception\LdapException $zle) {
|
||||
|
||||
$admin = $this->adminLogin($username, $password);
|
||||
if (!isset($admin)) {
|
||||
$GLOBALS['log']->info('LDAP Authentication: ' . $zle->getMessage());
|
||||
$adminUser = $this->adminLogin($username, $password);
|
||||
if (!isset($adminUser)) {
|
||||
return null;
|
||||
}
|
||||
$GLOBALS['log']->info('LDAP: Administrator ['.$username.'] was logged in by Espo method.');
|
||||
}
|
||||
|
||||
$GLOBALS['log']->info('LDAP Authentication: Administrator login by username ['.$username.']');
|
||||
if (!isset($adminUser)) {
|
||||
$userDn = $this->findLdapUserDnByUsername($username);
|
||||
$GLOBALS['log']->debug('Found DN for ['.$username.']: ['.$userDn.'].');
|
||||
if (!isset($userDn)) {
|
||||
$GLOBALS['log']->error('LDAP: Authentication failed for user ['.$username.'], details: user is not found.');
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
$ldapClient->bind($userDn, $password);
|
||||
} catch (\Exception $e) {
|
||||
$GLOBALS['log']->error('LDAP: Authentication failed for user ['.$username.'], details: ' . $e->getMessage());
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
$user = $this->getEntityManager()->getRepository('User')->findOne(array(
|
||||
@@ -118,7 +146,7 @@ class LDAP extends Base
|
||||
|
||||
$isCreateUser = $this->getUtils()->getOption('createEspoUser');
|
||||
if (!isset($user) && $isCreateUser) {
|
||||
$this->getAuth()->useNoAuth(); /** Required to fix Acl "isFetched()" error */
|
||||
$userData = $ldapClient->getEntry($userDn);
|
||||
$user = $this->createUser($userData);
|
||||
}
|
||||
|
||||
@@ -130,6 +158,7 @@ class LDAP extends Base
|
||||
*
|
||||
* @param string $username
|
||||
* @param \Espo\Entities\AuthToken $authToken
|
||||
*
|
||||
* @return \Espo\Entities\User | null
|
||||
*/
|
||||
protected function loginByToken($username, \Espo\Entities\AuthToken $authToken = null)
|
||||
@@ -182,26 +211,106 @@ class LDAP extends Base
|
||||
* Create Espo user with data gets from LDAP server
|
||||
*
|
||||
* @param array $userData LDAP entity data
|
||||
*
|
||||
* @return \Espo\Entities\User
|
||||
*/
|
||||
protected function createUser(array $userData)
|
||||
{
|
||||
$GLOBALS['log']->info('Creating new user ...');
|
||||
$data = array();
|
||||
foreach ($this->fields as $espo => $ldap) {
|
||||
|
||||
// show full array of the LDAP user
|
||||
$GLOBALS['log']->debug('LDAP: user data: ' .print_r($userData, true));
|
||||
|
||||
//set values from ldap server
|
||||
$ldapFields = $this->loadFields('ldap');
|
||||
foreach ($ldapFields as $espo => $ldap) {
|
||||
$ldap = strtolower($ldap);
|
||||
if (isset($userData[$ldap][0])) {
|
||||
$GLOBALS['log']->debug('LDAP: Create a user wtih ['.$espo.'] = ['.$userData[$ldap][0].'].');
|
||||
$data[$espo] = $userData[$ldap][0];
|
||||
}
|
||||
}
|
||||
|
||||
//set user fields
|
||||
$userFields = $this->loadFields('user');
|
||||
foreach ($userFields as $fieldName => $fieldValue) {
|
||||
$data[$fieldName] = $fieldValue;
|
||||
}
|
||||
|
||||
$user = $this->getEntityManager()->getEntity('User');
|
||||
$user->set($data);
|
||||
|
||||
$this->getEntityManager()->saveEntity($user);
|
||||
|
||||
return $user;
|
||||
return $this->getEntityManager()->getEntity('User', $user->id);
|
||||
}
|
||||
|
||||
/**
|
||||
* Find LDAP user DN by his username
|
||||
*
|
||||
* @param string $username
|
||||
*
|
||||
* @return string | null
|
||||
*/
|
||||
protected function findLdapUserDnByUsername($username)
|
||||
{
|
||||
$ldapClient = $this->getLdapClient();
|
||||
$options = $this->getUtils()->getOptions();
|
||||
|
||||
$loginFilterString = '';
|
||||
if (!empty($options['userLoginFilter'])) {
|
||||
$loginFilterString = $this->convertToFilterFormat($options['userLoginFilter']);
|
||||
}
|
||||
|
||||
}
|
||||
$searchString = '(&(objectClass='.$options['userObjectClass'].')('.$options['userNameAttribute'].'='.$username.')'.$loginFilterString.')';
|
||||
$result = $ldapClient->search($searchString, null, LDAP\Client::SEARCH_SCOPE_SUB);
|
||||
$GLOBALS['log']->debug('LDAP: user search string: "' . $searchString . '"');
|
||||
|
||||
foreach ($result as $item) {
|
||||
return $item["dn"];
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Check and convert filter item into LDAP format
|
||||
*
|
||||
* @param string $filter E.g. "memberof=CN=externalTesters,OU=groups,DC=espo,DC=local"
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
protected function convertToFilterFormat($filter)
|
||||
{
|
||||
$filter = trim($filter);
|
||||
if (substr($filter, 0, 1) != '(') {
|
||||
$filter = '(' . $filter;
|
||||
}
|
||||
if (substr($filter, -1) != ')') {
|
||||
$filter = $filter . ')';
|
||||
}
|
||||
return $filter;
|
||||
}
|
||||
|
||||
/**
|
||||
* Load fields for a user
|
||||
*
|
||||
* @param string $type
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
protected function loadFields($type)
|
||||
{
|
||||
$options = $this->getUtils()->getOptions();
|
||||
|
||||
$typeMap = $type . 'FieldMap';
|
||||
|
||||
$fields = array();
|
||||
foreach ($this->$typeMap as $fieldName => $fieldValue) {
|
||||
if (isset($options[$fieldValue])) {
|
||||
$fields[$fieldName] = $options[$fieldValue];
|
||||
}
|
||||
}
|
||||
|
||||
return $fields;
|
||||
}
|
||||
}
|
||||
35
application/Espo/Core/Utils/Authentication/LDAP/Client.php
Normal file
35
application/Espo/Core/Utils/Authentication/LDAP/Client.php
Normal file
@@ -0,0 +1,35 @@
|
||||
<?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/.
|
||||
*
|
||||
* The interactive user interfaces in modified source and object code versions
|
||||
* of this program must display Appropriate Legal Notices, as required under
|
||||
* Section 5 of the GNU General Public License version 3.
|
||||
*
|
||||
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
||||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Core\Utils\Authentication\LDAP;
|
||||
|
||||
class Client extends \Zend\Ldap\Ldap
|
||||
{
|
||||
|
||||
}
|
||||
@@ -1,129 +0,0 @@
|
||||
<?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/.
|
||||
*
|
||||
* The interactive user interfaces in modified source and object code versions
|
||||
* of this program must display Appropriate Legal Notices, as required under
|
||||
* Section 5 of the GNU General Public License version 3.
|
||||
*
|
||||
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
||||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Core\Utils\Authentication\LDAP;
|
||||
class LDAP extends \Zend\Ldap\Ldap
|
||||
{
|
||||
protected $usernameAttribute = 'cn';
|
||||
|
||||
|
||||
/**
|
||||
* Get DN depends on options, ex. "cn=test,ou=People,dc=maxcrc,dc=com"
|
||||
*
|
||||
* @return string DN format
|
||||
*/
|
||||
public function getDn($acctname)
|
||||
{
|
||||
return $this->getAccountDn($acctname, \Zend\Ldap\Ldap::ACCTNAME_FORM_DN);
|
||||
}
|
||||
|
||||
/**
|
||||
* Fix a bug, ex. CN=Alice Baker,CN=Users,DC=example,DC=com
|
||||
*
|
||||
* @param string $acctname
|
||||
* @return string - Account DN
|
||||
*/
|
||||
protected function getAccountDn($acctname)
|
||||
{
|
||||
$baseDn = $this->getBaseDn();
|
||||
|
||||
if ($this->getBindRequiresDn() && isset($baseDn)) {
|
||||
try {
|
||||
return parent::getAccountDn($acctname);
|
||||
} catch (\Zend\Ldap\Exception\LdapException $zle) {
|
||||
if ($zle->getCode() != \Zend\Ldap\Exception\LdapException::LDAP_NO_SUCH_OBJECT) {
|
||||
throw $zle;
|
||||
}
|
||||
}
|
||||
|
||||
$acctname = $this->usernameAttribute . '=' . \Zend\Ldap\Filter\AbstractFilter::escapeValue($acctname) . ',' . $baseDn;
|
||||
}
|
||||
|
||||
return parent::getAccountDn($acctname);
|
||||
}
|
||||
|
||||
/**
|
||||
* Search a user using userLoginFilter
|
||||
*
|
||||
* @param string $filter
|
||||
* @param string $basedn
|
||||
* @param int $scope
|
||||
* @param array $attributes
|
||||
* @return array
|
||||
*/
|
||||
public function searchByLoginFilter($filter, $basedn = null, $scope = self::SEARCH_SCOPE_SUB, array $attributes = array())
|
||||
{
|
||||
$filter = $this->getLoginFilter($filter);
|
||||
|
||||
$result = $this->search($filter, $basedn, $scope, $attributes);
|
||||
|
||||
if ($result->count() > 0) {
|
||||
return $result->getFirst();
|
||||
}
|
||||
|
||||
throw new \Zend\Ldap\Exception\LdapException($this, 'searching: ' . $filter);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get login filter in LDAP format
|
||||
*
|
||||
* @param string $filter
|
||||
* @return string
|
||||
*/
|
||||
protected function getLoginFilter($filter)
|
||||
{
|
||||
$baseFilter = '(objectClass=*)';
|
||||
|
||||
if (!empty($filter)) {
|
||||
$baseFilter = '(&' . $baseFilter . $this->convertToFilterFormat($filter). ')';
|
||||
}
|
||||
|
||||
return $baseFilter;
|
||||
}
|
||||
|
||||
/**
|
||||
* Check and convert filter item in LDAP format
|
||||
*
|
||||
* @param string $filter [description]
|
||||
* @return string
|
||||
*/
|
||||
protected function convertToFilterFormat($filter)
|
||||
{
|
||||
$filter = trim($filter);
|
||||
if (substr($filter, 0, 1) != '(') {
|
||||
$filter = '(' . $filter;
|
||||
}
|
||||
|
||||
if (substr($filter, -1) != ')') {
|
||||
$filter = $filter . ')';
|
||||
}
|
||||
|
||||
return $filter;
|
||||
}
|
||||
}
|
||||
@@ -57,7 +57,16 @@ class Utils
|
||||
'tryUsernameSplit' => 'ldapTryUsernameSplit',
|
||||
'networkTimeout' => 'ldapNetworkTimeout',
|
||||
'createEspoUser' => 'ldapCreateEspoUser',
|
||||
'userNameAttribute' => 'ldapUserNameAttribute',
|
||||
'userTitleAttribute' => 'ldapUserTitleAttribute',
|
||||
'userFirstNameAttribute' => 'ldapUserFirstNameAttribute',
|
||||
'userLastNameAttribute' => 'ldapUserLastNameAttribute',
|
||||
'userEmailAddressAttribute' => 'ldapUserEmailAddressAttribute',
|
||||
'userPhoneNumberAttribute' => 'ldapUserPhoneNumberAttribute',
|
||||
'userLoginFilter' => 'ldapUserLoginFilter',
|
||||
'userTeamsIds' => 'ldapUserTeamsIds',
|
||||
'userDefaultTeamId' => 'ldapUserDefaultTeamId',
|
||||
'userObjectClass' => 'ldapUserObjectClass',
|
||||
);
|
||||
|
||||
/**
|
||||
@@ -66,8 +75,17 @@ class Utils
|
||||
* @var array
|
||||
*/
|
||||
protected $permittedEspoOptions = array(
|
||||
'createEspoUser' => false,
|
||||
'userLoginFilter' => null,
|
||||
'createEspoUser',
|
||||
'userNameAttribute',
|
||||
'userObjectClass',
|
||||
'userTitleAttribute',
|
||||
'userFirstNameAttribute',
|
||||
'userLastNameAttribute',
|
||||
'userEmailAddressAttribute',
|
||||
'userPhoneNumberAttribute',
|
||||
'userLoginFilter',
|
||||
'userTeamsIds',
|
||||
'userDefaultTeamId',
|
||||
);
|
||||
|
||||
/**
|
||||
@@ -83,9 +101,11 @@ class Utils
|
||||
);
|
||||
|
||||
|
||||
public function __construct(Config $config)
|
||||
public function __construct(Config $config = null)
|
||||
{
|
||||
$this->config = $config;
|
||||
if (isset($config)) {
|
||||
$this->config = $config;
|
||||
}
|
||||
}
|
||||
|
||||
protected function getConfig()
|
||||
@@ -113,14 +133,25 @@ class Utils
|
||||
}
|
||||
}
|
||||
|
||||
/** peculiar fields */
|
||||
$this->options = $this->normalizeOptions($options);
|
||||
|
||||
return $this->options;
|
||||
}
|
||||
|
||||
/**
|
||||
* Normalize options to LDAP client format
|
||||
*
|
||||
* @param array $options
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function normalizeOptions(array $options)
|
||||
{
|
||||
$options['useSsl'] = (bool) ($options['useSsl'] == 'SSL');
|
||||
$options['useStartTls'] = (bool) ($options['useStartTls'] == 'TLS');
|
||||
$options['accountCanonicalForm'] = $this->accountCanonicalFormMap[ $options['accountCanonicalForm'] ];
|
||||
|
||||
$this->options = $options;
|
||||
|
||||
return $this->options;
|
||||
return $options;
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -148,12 +179,10 @@ class Utils
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function getZendOptions()
|
||||
public function getLdapClientOptions()
|
||||
{
|
||||
$options = $this->getOptions();
|
||||
$espoOptions = array_keys($this->permittedEspoOptions);
|
||||
|
||||
$zendOptions = array_diff_key($options, array_flip($espoOptions));
|
||||
$zendOptions = array_diff_key($options, array_flip($this->permittedEspoOptions));
|
||||
|
||||
return $zendOptions;
|
||||
}
|
||||
|
||||
@@ -96,8 +96,9 @@ class ClientManager
|
||||
foreach ($vars as $key => $value) {
|
||||
$html = str_replace('{{'.$key.'}}', $value, $html);
|
||||
}
|
||||
$html = str_replace('{{applicationName}}', $this->getConfig()->get('applicationName', 'EspoCRM'), $html);
|
||||
$html = str_replace('{{cacheTimestamp}}', $this->getCacheTimestamp(), $html);
|
||||
$html = str_replace('{{useCache}}', $this->getConfig()->get('useCache') ? 'true' : 'false' , $html);
|
||||
$html = str_replace('{{useCache}}', $this->getConfig()->get('useCache') ? 'true' : 'false', $html);
|
||||
$html = str_replace('{{stylesheet}}', $this->getThemeManager()->getStylesheet(), $html);
|
||||
$html = str_replace('{{runScript}}', $runScript , $html);
|
||||
$html = str_replace('{{basePath}}', $this->basePath , $html);
|
||||
|
||||
@@ -316,6 +316,11 @@ class Config
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public function getSiteUrl()
|
||||
{
|
||||
return rtrim($this->get('siteUrl'), '/');
|
||||
}
|
||||
}
|
||||
|
||||
?>
|
||||
|
||||
@@ -245,7 +245,7 @@ class Job
|
||||
deleted = 0
|
||||
GROUP BY scheduled_job_id
|
||||
HAVING count( * ) > 1
|
||||
ORDER BY execute_time ASC
|
||||
ORDER BY MAX(execute_time) ASC
|
||||
";
|
||||
$sth = $pdo->prepare($query);
|
||||
$sth->execute();
|
||||
|
||||
@@ -82,15 +82,11 @@ class Converter
|
||||
*/
|
||||
public function process()
|
||||
{
|
||||
$GLOBALS['log']->debug('Orm\Converter - Start: orm convertation');
|
||||
|
||||
$ormMeta = $this->getOrmConverter()->process();
|
||||
|
||||
//save database meta to a file espoMetadata.php
|
||||
$result = $this->getMetadata()->setOrmMetadata($ormMeta);
|
||||
|
||||
$GLOBALS['log']->debug('Orm\Converter - End: orm convertation, result=['.$result.']');
|
||||
|
||||
return $result;
|
||||
}
|
||||
}
|
||||
@@ -116,6 +116,11 @@ class DateTime
|
||||
return null;
|
||||
}
|
||||
|
||||
public function setTimezone($timezone)
|
||||
{
|
||||
$this->timezone = new \DateTimeZone($timezone);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
78
application/Espo/Core/Utils/EmailFilterManager.php
Normal file
78
application/Espo/Core/Utils/EmailFilterManager.php
Normal file
@@ -0,0 +1,78 @@
|
||||
<?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/.
|
||||
*
|
||||
* The interactive user interfaces in modified source and object code versions
|
||||
* of this program must display Appropriate Legal Notices, as required under
|
||||
* Section 5 of the GNU General Public License version 3.
|
||||
*
|
||||
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
||||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Core\Utils;
|
||||
|
||||
use \Espo\Core\ORM\EntityManager;
|
||||
use \Espo\Entities\Email;
|
||||
|
||||
class EmailFilterManager
|
||||
{
|
||||
private $entityManager;
|
||||
|
||||
private $data = array();
|
||||
|
||||
protected $filtersMatcher = null;
|
||||
|
||||
public function __construct(EntityManager $entityManager)
|
||||
{
|
||||
$this->entityManager = $entityManager;
|
||||
}
|
||||
|
||||
protected function getEntityManager()
|
||||
{
|
||||
return $this->entityManager;
|
||||
}
|
||||
|
||||
protected function getFiltersMatcher()
|
||||
{
|
||||
if (!$this->filtersMatcher) {
|
||||
$this->filtersMatcher = new \Espo\Core\Mail\FiltersMatcher();
|
||||
}
|
||||
return $this->filtersMatcher;
|
||||
}
|
||||
|
||||
public function getMatchingFilter(Email $email, $userId)
|
||||
{
|
||||
if (!array_key_exists($userId, $this->data)) {
|
||||
$emailFilterList = $this->getEntityManager()->getRepository('EmailFilter')->where(array(
|
||||
'parentId' => $userId,
|
||||
'parentType' => 'User'
|
||||
))->order('LIST:action:Skip;Move to Folder')->find();
|
||||
$this->data[$userId] = $emailFilterList;
|
||||
}
|
||||
foreach ($this->data[$userId] as $emailFilter) {
|
||||
if ($this->getFiltersMatcher()->match($email, $emailFilter)) {
|
||||
return $emailFilter;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -31,6 +31,8 @@ namespace Espo\Core\Utils;
|
||||
use \Espo\Core\Exceptions\Error,
|
||||
\Espo\Core\Exceptions\Conflict;
|
||||
|
||||
use \Espo\Core\Container;
|
||||
|
||||
class FieldManager
|
||||
{
|
||||
private $metadata;
|
||||
@@ -41,14 +43,17 @@ class FieldManager
|
||||
|
||||
protected $isChanged = null;
|
||||
|
||||
private $container;
|
||||
|
||||
protected $metadataType = 'entityDefs';
|
||||
|
||||
protected $customOptionName = 'isCustom';
|
||||
|
||||
public function __construct(Metadata $metadata, Language $language)
|
||||
public function __construct(Metadata $metadata, Language $language, Container $container = null)
|
||||
{
|
||||
$this->metadata = $metadata;
|
||||
$this->language = $language;
|
||||
$this->container = $container;
|
||||
|
||||
$this->metadataHelper = new \Espo\Core\Utils\Metadata\Helper($this->metadata);
|
||||
}
|
||||
@@ -74,6 +79,10 @@ class FieldManager
|
||||
|
||||
$fieldDefs['label'] = $this->getLanguage()->translate($name, 'fields', $scope);
|
||||
|
||||
$type = $this->getMetadata()->get(['entityDefs', $scope, 'fields', $name, 'type']);
|
||||
|
||||
$this->processHook('onRead', $type, $scope, $name, $fieldDefs);
|
||||
|
||||
return $fieldDefs;
|
||||
}
|
||||
|
||||
@@ -100,7 +109,11 @@ class FieldManager
|
||||
$this->setLabel($name, $fieldDefs['label'], $scope);
|
||||
}
|
||||
|
||||
if (isset($fieldDefs['type']) && ($fieldDefs['type'] == 'enum' || $fieldDefs['type'] == 'phone')) {
|
||||
$type = isset($fieldDefs['type']) ? $fieldDefs['type'] : $type = $this->getMetadata()->get(['entityDefs', $scope, 'fields', $name, 'type']);
|
||||
|
||||
$this->processHook('beforeSave', $type, $scope, $name, $fieldDefs);
|
||||
|
||||
if ($this->getMetadata()->get(['fields', $type, 'translatedOptions'])) {
|
||||
if (isset($fieldDefs['translatedOptions'])) {
|
||||
$this->setTranslatedOptions($name, $fieldDefs['translatedOptions'], $scope);
|
||||
}
|
||||
@@ -108,6 +121,8 @@ class FieldManager
|
||||
|
||||
if (isset($fieldDefs['label']) || isset($fieldDefs['translatedOptions'])) {
|
||||
$res &= $this->getLanguage()->save();
|
||||
|
||||
$this->processHook('afterSave', $type, $scope, $name, $fieldDefs);
|
||||
}
|
||||
|
||||
if ($this->isDefsChanged($name, $fieldDefs, $scope)) {
|
||||
@@ -123,6 +138,10 @@ class FieldManager
|
||||
throw new Error('Cannot delete core field ['.$name.'] in '.$scope);
|
||||
}
|
||||
|
||||
$type = $this->getMetadata()->get(['entityDefs', $scope, 'fields', $name, 'type']);
|
||||
|
||||
$this->processHook('beforeRemove', $type, $scope, $name);
|
||||
|
||||
$unsets = array(
|
||||
'fields.'.$name,
|
||||
'links.'.$name,
|
||||
@@ -132,6 +151,8 @@ class FieldManager
|
||||
$res = $this->getMetadata()->save();
|
||||
$res &= $this->deleteLabel($name, $scope);
|
||||
|
||||
$this->processHook('afterRemove', $type, $scope, $name);
|
||||
|
||||
return (bool) $res;
|
||||
}
|
||||
|
||||
@@ -337,4 +358,31 @@ class FieldManager
|
||||
{
|
||||
return array_merge($this->getActualAttributeList($scope, $name), $this->getNotActualAttributeList($scope, $name));
|
||||
}
|
||||
|
||||
protected function processHook($methodName, $type, $scope, $name, &$defs = null)
|
||||
{
|
||||
$hook = $this->getHook($type);
|
||||
if (!$hook) return;
|
||||
|
||||
if (!method_exists($hook, $methodName)) return;
|
||||
|
||||
$hook->$methodName($scope, $name, $defs);
|
||||
}
|
||||
|
||||
protected function getHook($type)
|
||||
{
|
||||
$className = $this->getMetadata()->get(['fields', $type, 'hookClassName']);
|
||||
|
||||
if (!$className) return;
|
||||
|
||||
if (class_exists($className)) {
|
||||
$hook = new $className();
|
||||
foreach ($hook->getDependencyList() as $name) {
|
||||
$hook->inject($name, $this->container->get($name));
|
||||
}
|
||||
return $hook;
|
||||
}
|
||||
$GLOBALS['log']->error("Field Manager hook class '{$className}' does not exist.");
|
||||
return;
|
||||
}
|
||||
}
|
||||
65
application/Espo/Core/Utils/FieldManager/Hooks/Base.php
Normal file
65
application/Espo/Core/Utils/FieldManager/Hooks/Base.php
Normal file
@@ -0,0 +1,65 @@
|
||||
<?php
|
||||
|
||||
namespace Espo\Core\Utils\FieldManager\Hooks;
|
||||
|
||||
abstract class Base
|
||||
{
|
||||
protected $dependencyList = [
|
||||
'entityManager',
|
||||
'config',
|
||||
'metadata',
|
||||
];
|
||||
|
||||
protected $injections = array();
|
||||
|
||||
public function __construct()
|
||||
{
|
||||
$this->init();
|
||||
}
|
||||
|
||||
protected function init()
|
||||
{
|
||||
}
|
||||
|
||||
public function getDependencyList()
|
||||
{
|
||||
return $this->dependencyList;
|
||||
}
|
||||
|
||||
protected function addDependencyList(array $list)
|
||||
{
|
||||
foreach ($list as $item) {
|
||||
$this->addDependency($item);
|
||||
}
|
||||
}
|
||||
|
||||
protected function addDependency($name)
|
||||
{
|
||||
$this->dependencies[] = $name;
|
||||
}
|
||||
|
||||
protected function getInjection($name)
|
||||
{
|
||||
return $this->injections[$name];
|
||||
}
|
||||
|
||||
public function inject($name, $object)
|
||||
{
|
||||
$this->injections[$name] = $object;
|
||||
}
|
||||
|
||||
protected function getMetadata()
|
||||
{
|
||||
return $this->getInjection('metadata');
|
||||
}
|
||||
|
||||
protected function getConfig()
|
||||
{
|
||||
return $this->getInjection('config');
|
||||
}
|
||||
|
||||
protected function getEntityManager()
|
||||
{
|
||||
return $this->getInjection('entityManager');
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,61 @@
|
||||
<?php
|
||||
|
||||
namespace Espo\Core\Utils\FieldManager\Hooks;
|
||||
|
||||
class NumberType extends Base
|
||||
{
|
||||
public function onRead($scope, $name, &$defs)
|
||||
{
|
||||
$number = $this->getEntityManager()->getRepository('NextNumber')->where(array(
|
||||
'entityType' => $scope,
|
||||
'fieldName' => $name
|
||||
))->findOne();
|
||||
|
||||
$value = null;
|
||||
if (!$number) {
|
||||
$value = 1;
|
||||
} else {
|
||||
if (!$number->get('value')) {
|
||||
$value = 1;
|
||||
}
|
||||
}
|
||||
|
||||
if (!$value && $number) {
|
||||
$value = $number->get('value');
|
||||
}
|
||||
|
||||
$defs['nextNumber'] = $value;
|
||||
}
|
||||
|
||||
public function afterSave($scope, $name, $defs)
|
||||
{
|
||||
if (!isset($defs['nextNumber'])) return;
|
||||
|
||||
$number = $this->getEntityManager()->getRepository('NextNumber')->where(array(
|
||||
'entityType' => $scope,
|
||||
'fieldName' => $name
|
||||
))->findOne();
|
||||
|
||||
if (!$number) {
|
||||
$number = $this->getEntityManager()->getEntity('NextNumber');
|
||||
|
||||
$number->set('entityType', $scope);
|
||||
$number->set('fieldName', $name);
|
||||
}
|
||||
|
||||
$number->set('value', $defs['nextNumber']);
|
||||
$this->getEntityManager()->saveEntity($number);
|
||||
}
|
||||
|
||||
public function afterRemove($scope, $name)
|
||||
{
|
||||
$number = $this->getEntityManager()->getRepository('NextNumber')->where(array(
|
||||
'entityType' => $scope,
|
||||
'fieldName' => $name
|
||||
))->findOne();
|
||||
|
||||
if (!$number) return;
|
||||
|
||||
$this->getEntityManager()->removeEntity($number);
|
||||
}
|
||||
}
|
||||
@@ -188,9 +188,7 @@ class Manager
|
||||
|
||||
if (file_exists($fullPath) && strtolower(substr($fullPath, -4)) == '.php') {
|
||||
$phpContents = include($fullPath);
|
||||
if (is_array($phpContents)) {
|
||||
return $phpContents;
|
||||
}
|
||||
return $phpContents;
|
||||
}
|
||||
|
||||
return false;
|
||||
@@ -336,7 +334,7 @@ class Manager
|
||||
public function unsetContents($path, $unsets, $isJSON = true)
|
||||
{
|
||||
$currentData = $this->getContents($path);
|
||||
if ($currentData == false) {
|
||||
if (!isset($currentData) || !$currentData) {
|
||||
$GLOBALS['log']->notice('FileManager::unsetContents: File ['.$this->concatPaths($path).'] does not exist.');
|
||||
return false;
|
||||
}
|
||||
|
||||
@@ -373,7 +373,7 @@ class Permission
|
||||
protected function chmodReal($filename, $mode)
|
||||
{
|
||||
try {
|
||||
$result = chmod($filename, $mode);
|
||||
$result = @chmod($filename, $mode);
|
||||
} catch (\Exception $e) {
|
||||
$result = false;
|
||||
}
|
||||
@@ -383,7 +383,7 @@ class Permission
|
||||
$this->chgrp($filename, $this->getDefaultGroup(true));
|
||||
|
||||
try {
|
||||
$result = chmod($filename, $mode);
|
||||
$result = @chmod($filename, $mode);
|
||||
} catch (\Exception $e) {
|
||||
throw new Error($e->getMessage());
|
||||
}
|
||||
@@ -395,7 +395,7 @@ class Permission
|
||||
protected function chownReal($path, $user)
|
||||
{
|
||||
try {
|
||||
$result = chown($path, $user);
|
||||
$result = @chown($path, $user);
|
||||
} catch (\Exception $e) {
|
||||
throw new Error($e->getMessage());
|
||||
}
|
||||
@@ -406,7 +406,7 @@ class Permission
|
||||
protected function chgrpReal($path, $group)
|
||||
{
|
||||
try {
|
||||
$result = chgrp($path, $group);
|
||||
$result = @chgrp($path, $group);
|
||||
} catch (\Exception $e) {
|
||||
throw new Error($e->getMessage());
|
||||
}
|
||||
|
||||
@@ -28,6 +28,7 @@
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Core\Utils;
|
||||
|
||||
use Espo\Core\Exceptions\Error;
|
||||
|
||||
class Metadata
|
||||
|
||||
@@ -394,33 +394,35 @@ class Util
|
||||
$unsets = (array) $unsets;
|
||||
}
|
||||
|
||||
foreach($unsets as $rootKey => $unsetItem){
|
||||
foreach ($unsets as $rootKey => $unsetItem) {
|
||||
$unsetItem = is_array($unsetItem) ? $unsetItem : (array) $unsetItem;
|
||||
|
||||
foreach($unsetItem as $unsetSett){
|
||||
if (!empty($unsetSett)){
|
||||
$keyItems = explode('.', $unsetSett);
|
||||
$currVal = isset($content[$rootKey]) ? "\$content['{$rootKey}']" : "\$content";
|
||||
foreach ($unsetItem as $unsetString) {
|
||||
if (is_string($rootKey)) {
|
||||
$unsetString = $rootKey . '.' . $unsetString;
|
||||
}
|
||||
|
||||
$lastKey = array_pop($keyItems);
|
||||
foreach($keyItems as $keyItem){
|
||||
$currVal .= "['{$keyItem}']";
|
||||
}
|
||||
$keyСhain = explode('.', $unsetString);
|
||||
$keyChainCount = count($keyСhain) - 1;
|
||||
|
||||
$unsetElem = $currVal . "['{$lastKey}']";
|
||||
$elem = & $content;
|
||||
for ($i = 0; $i <= $keyChainCount; $i++) {
|
||||
|
||||
$evalString = "
|
||||
if (isset({$unsetElem}) || ( is_array({$currVal}) && array_key_exists('{$lastKey}', {$currVal}) )) {
|
||||
unset({$unsetElem});
|
||||
} ";
|
||||
eval($evalString);
|
||||
if (is_array($elem) && array_key_exists($keyСhain[$i], $elem)) {
|
||||
|
||||
if ($i == $keyChainCount) {
|
||||
|
||||
unset($elem[$keyСhain[$i]]);
|
||||
|
||||
if ($unsetParentEmptyArray && is_array($elem) && empty($elem)) {
|
||||
unset($keyСhain[$i]);
|
||||
$content = static::unsetInArray($content, implode('.', $keyСhain), false);
|
||||
}
|
||||
|
||||
} else if (is_array($elem[$keyСhain[$i]])) {
|
||||
$elem = & $elem[$keyСhain[$i]];
|
||||
}
|
||||
|
||||
if ($unsetParentEmptyArray) {
|
||||
$evalString = "
|
||||
if (is_array({$currVal}) && empty({$currVal})) {
|
||||
unset({$currVal});
|
||||
} ";
|
||||
eval($evalString);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -32,6 +32,7 @@ return array (
|
||||
'driver' => 'pdo_mysql',
|
||||
'host' => 'localhost',
|
||||
'port' => '',
|
||||
'charset' => 'utf8',
|
||||
'dbname' => '',
|
||||
'user' => '',
|
||||
'password' => '',
|
||||
@@ -67,6 +68,8 @@ return array (
|
||||
'de_DE',
|
||||
'es_ES',
|
||||
'fr_FR',
|
||||
'id_ID',
|
||||
'it_IT',
|
||||
'nl_NL',
|
||||
'tr_TR',
|
||||
'ro_RO',
|
||||
@@ -94,11 +97,12 @@ return array (
|
||||
),
|
||||
"tabList" => ["Account", "Contact", "Lead", "Opportunity", "Calendar", "Meeting", "Call", "Task", "Case", "Email", "Document", "Campaign", "KnowledgeBaseArticle"],
|
||||
"quickCreateList" => ["Account", "Contact", "Lead", "Opportunity", "Meeting", "Call", "Task", "Case", "Email"],
|
||||
'calendarDefaultEntity' => 'Meeting',
|
||||
'exportDisabled' => false,
|
||||
'assignmentEmailNotifications' => false,
|
||||
'assignmentEmailNotificationsEntityList' => ['Lead', 'Opportunity', 'Task', 'Case'],
|
||||
'assignmentNotificationsEntityList' => ['Meeting', 'Call', 'Task', 'Email'],
|
||||
"portalStreamEmailNotifications" => true,
|
||||
'streamEmailNotificationsEntityList' => ['Case'],
|
||||
'emailMessageMaxSize' => 10,
|
||||
'notificationsCheckInterval' => 10,
|
||||
'disabledCountQueryEntityList' => ['Email'],
|
||||
@@ -146,6 +150,7 @@ return array (
|
||||
]
|
||||
]
|
||||
],
|
||||
'isInstalled' => false
|
||||
"calendarEntityList" => ["Meeting", "Call", "Task"],
|
||||
'isInstalled' => false,
|
||||
);
|
||||
|
||||
|
||||
@@ -123,6 +123,15 @@ return array ( 'defaultPermissions' =>
|
||||
'ldapTryUsernameSplit',
|
||||
'ldapOptReferrals',
|
||||
'ldapCreateEspoUser',
|
||||
'ldapAccountDomainName',
|
||||
'ldapAccountDomainNameShort',
|
||||
'ldapUserNameAttribute',
|
||||
'ldapUserFirstNameAttribute',
|
||||
'ldapUserLastNameAttribute',
|
||||
'ldapUserTitleAttribute',
|
||||
'ldapUserEmailAddressAttribute',
|
||||
'ldapUserPhoneNumberAttribute',
|
||||
'ldapUserObjectClass',
|
||||
'maxEmailAccountCount',
|
||||
'massEmailMaxPerHourCount',
|
||||
'personalEmailMaxPortionSize',
|
||||
@@ -131,5 +140,12 @@ return array ( 'defaultPermissions' =>
|
||||
'authTokenMaxIdleTime'
|
||||
),
|
||||
'isInstalled' => false,
|
||||
'ldapUserNameAttribute' => 'sAMAccountName',
|
||||
'ldapUserFirstNameAttribute' => 'givenName',
|
||||
'ldapUserLastNameAttribute' => 'sn',
|
||||
'ldapUserTitleAttribute' => 'title',
|
||||
'ldapUserEmailAddressAttribute' => 'mail',
|
||||
'ldapUserPhoneNumberAttribute' => 'telephoneNumber',
|
||||
'ldapUserObjectClass' => 'person',
|
||||
);
|
||||
|
||||
|
||||
@@ -51,6 +51,11 @@ class Email extends \Espo\Core\ORM\Entity
|
||||
}
|
||||
}
|
||||
|
||||
public function isManuallyArchived()
|
||||
{
|
||||
return $this->get('status') === 'Archived' && $this->get('createdById') !== 'system';
|
||||
}
|
||||
|
||||
public function addAttachment(\Espo\Entities\Attachment $attachment)
|
||||
{
|
||||
if (!empty($this->id)) {
|
||||
|
||||
36
application/Espo/Entities/EmailFolder.php
Normal file
36
application/Espo/Entities/EmailFolder.php
Normal file
@@ -0,0 +1,36 @@
|
||||
<?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/.
|
||||
*
|
||||
* The interactive user interfaces in modified source and object code versions
|
||||
* of this program must display Appropriate Legal Notices, as required under
|
||||
* Section 5 of the GNU General Public License version 3.
|
||||
*
|
||||
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
||||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Entities;
|
||||
|
||||
class EmailFolder extends \Espo\Core\ORM\Entity
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
35
application/Espo/Entities/NextNumber.php
Normal file
35
application/Espo/Entities/NextNumber.php
Normal file
@@ -0,0 +1,35 @@
|
||||
<?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/.
|
||||
*
|
||||
* The interactive user interfaces in modified source and object code versions
|
||||
* of this program must display Appropriate Legal Notices, as required under
|
||||
* Section 5 of the GNU General Public License version 3.
|
||||
*
|
||||
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
||||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Entities;
|
||||
|
||||
class NextNumber extends \Espo\Core\ORM\Entity
|
||||
{
|
||||
|
||||
}
|
||||
@@ -66,8 +66,6 @@ class Attachment extends \Espo\Core\EntryPoints\Base
|
||||
|
||||
header('Pragma: public');
|
||||
header('Content-Length: ' . filesize($fileName));
|
||||
ob_clean();
|
||||
flush();
|
||||
readfile($fileName);
|
||||
exit;
|
||||
}
|
||||
|
||||
@@ -92,8 +92,7 @@ class Download extends \Espo\Core\EntryPoints\Base
|
||||
header('Cache-Control: must-revalidate');
|
||||
header('Pragma: public');
|
||||
header('Content-Length: ' . filesize($fileName));
|
||||
ob_clean();
|
||||
flush();
|
||||
|
||||
readfile($fileName);
|
||||
exit;
|
||||
}
|
||||
|
||||
@@ -143,8 +143,6 @@ class Image extends \Espo\Core\EntryPoints\Base
|
||||
if ($fileSize) {
|
||||
header('Content-Length: ' . $fileSize);
|
||||
}
|
||||
ob_clean();
|
||||
flush();
|
||||
readfile($filePath);
|
||||
exit;
|
||||
}
|
||||
|
||||
@@ -44,7 +44,10 @@ class Portal extends \Espo\Core\EntryPoints\Base
|
||||
} else if (!empty($data['id'])) {
|
||||
$id = $data['id'];
|
||||
} else {
|
||||
$id = $this->getConfig()->get('defaultPortalId');
|
||||
$id = explode('/', $_SERVER['REQUEST_URI'])[count(explode('/', $_SERVER['SCRIPT_NAME'])) - 1];
|
||||
if (!$id) {
|
||||
$id = $this->getConfig()->get('defaultPortalId');
|
||||
}
|
||||
if (!$id) {
|
||||
throw new NotFound();
|
||||
}
|
||||
|
||||
111
application/Espo/Hooks/Common/NextNumber.php
Normal file
111
application/Espo/Hooks/Common/NextNumber.php
Normal file
@@ -0,0 +1,111 @@
|
||||
<?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/.
|
||||
*
|
||||
* The interactive user interfaces in modified source and object code versions
|
||||
* of this program must display Appropriate Legal Notices, as required under
|
||||
* Section 5 of the GNU General Public License version 3.
|
||||
*
|
||||
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
||||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Hooks\Common;
|
||||
|
||||
use Espo\ORM\Entity;
|
||||
use Espo\Core\Utils\Util;
|
||||
|
||||
class NextNumber extends \Espo\Core\Hooks\Base
|
||||
{
|
||||
public static $order = 10;
|
||||
|
||||
protected function init()
|
||||
{
|
||||
$this->addDependency('metadata');
|
||||
}
|
||||
|
||||
protected function getMetadata()
|
||||
{
|
||||
return $this->getInjection('metadata');
|
||||
}
|
||||
|
||||
protected function composeNumberAttribute(Entity $nextNumber)
|
||||
{
|
||||
$entityType = $nextNumber->get('entityType');
|
||||
$fieldName = $nextNumber->get('fieldName');
|
||||
$value = $nextNumber->get('value');
|
||||
|
||||
$prefix = $this->getMetadata()->get(['entityDefs', $entityType, 'fields', $fieldName, 'prefix'], '');
|
||||
$padLength = $this->getMetadata()->get(['entityDefs', $entityType, 'fields', $fieldName, 'padLength'], 0);
|
||||
|
||||
return $prefix . str_pad(strval($value), $padLength, '0', \STR_PAD_LEFT);
|
||||
}
|
||||
|
||||
public function beforeSave(Entity $entity, array $options = array())
|
||||
{
|
||||
$fieldDefs = $this->getMetadata()->get(['entityDefs', $entity->getEntityType(), 'fields'], array());
|
||||
|
||||
foreach ($fieldDefs as $fieldName => $defs) {
|
||||
if (isset($defs['type']) && $defs['type'] === 'number') {
|
||||
if (!$entity->isNew()) {
|
||||
if ($entity->isAttributeChanged($fieldName)) {
|
||||
$entity->set($fieldName, $entity->getFetched($fieldName));
|
||||
}
|
||||
continue;
|
||||
}
|
||||
$nextNumber = $this->getEntityManager()->getRepository('NextNumber')->where(array(
|
||||
'fieldName' => $fieldName,
|
||||
'entityType' => $entity->getEntityType()
|
||||
))->findOne();
|
||||
if (!$nextNumber) continue;
|
||||
$entity->set($fieldName, $this->composeNumberAttribute($nextNumber));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public function afterSave(Entity $entity, array $options = array())
|
||||
{
|
||||
if (!$entity->isNew()) return;
|
||||
|
||||
$fieldDefs = $this->getMetadata()->get(['entityDefs', $entity->getEntityType(), 'fields'], array());
|
||||
|
||||
foreach ($fieldDefs as $fieldName => $defs) {
|
||||
if (isset($defs['type']) && $defs['type'] === 'number') {
|
||||
$nextNumber = $this->getEntityManager()->getRepository('NextNumber')->where(array(
|
||||
'fieldName' => $fieldName,
|
||||
'entityType' => $entity->getEntityType()
|
||||
))->findOne();
|
||||
if (!$nextNumber) continue;
|
||||
|
||||
$value = $nextNumber->get('value');
|
||||
if (!$value) {
|
||||
$value = 1;
|
||||
}
|
||||
$value++;
|
||||
|
||||
$nextNumber->set('value', $value);
|
||||
$this->getEntityManager()->saveEntity($nextNumber);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
|
||||
@@ -40,29 +40,13 @@ class Notifications extends \Espo\Core\Hooks\Base
|
||||
|
||||
private $streamService;
|
||||
|
||||
protected function init()
|
||||
{
|
||||
$this->dependencies[] = 'container';
|
||||
$this->dependencies[] = 'metadata';
|
||||
}
|
||||
|
||||
private $hasStreamCache = array();
|
||||
|
||||
protected function getContainer()
|
||||
{
|
||||
return $this->getInjection('container');
|
||||
}
|
||||
|
||||
protected function getServiceFactory()
|
||||
{
|
||||
return $this->getContainer()->get('serviceFactory');
|
||||
}
|
||||
|
||||
protected function getMetadata()
|
||||
{
|
||||
return $this->getInjection('metadata');
|
||||
}
|
||||
|
||||
protected function checkHasStream($entityType)
|
||||
{
|
||||
if (!array_key_exists($entityType, $this->hasStreamCache)) {
|
||||
|
||||
@@ -47,7 +47,8 @@ class Stream extends \Espo\Core\Hooks\Base
|
||||
|
||||
protected function init()
|
||||
{
|
||||
$this->dependencies[] = 'serviceFactory';
|
||||
parent::init();
|
||||
$this->addDependency('serviceFactory');
|
||||
}
|
||||
|
||||
protected function getServiceFactory()
|
||||
@@ -57,11 +58,11 @@ class Stream extends \Espo\Core\Hooks\Base
|
||||
|
||||
protected function checkHasStream(Entity $entity)
|
||||
{
|
||||
$entityName = $entity->getEntityName();
|
||||
if (!array_key_exists($entityName, $this->hasStreamCache)) {
|
||||
$this->hasStreamCache[$entityName] = $this->getMetadata()->get("scopes.{$entityName}.stream");
|
||||
$entityType = $entity->getEntityType();
|
||||
if (!array_key_exists($entityType, $this->hasStreamCache)) {
|
||||
$this->hasStreamCache[$entityType] = $this->getMetadata()->get("scopes.{$entityType}.stream");
|
||||
}
|
||||
return $this->hasStreamCache[$entityName];
|
||||
return $this->hasStreamCache[$entityType];
|
||||
}
|
||||
|
||||
protected function isLinkObservableInStream($scope, $link)
|
||||
@@ -79,6 +80,8 @@ class Stream extends \Espo\Core\Hooks\Base
|
||||
{
|
||||
if ($this->checkHasStream($entity)) {
|
||||
$this->getStreamService()->unfollowAllUsersFromEntity($entity);
|
||||
echo "---";
|
||||
die;
|
||||
}
|
||||
$query = $this->getEntityManager()->getQuery();
|
||||
$sql = "
|
||||
@@ -176,7 +179,11 @@ class Stream extends \Espo\Core\Hooks\Base
|
||||
$assignedUserId = $entity->get('assignedUserId');
|
||||
$createdById = $entity->get('createdById');
|
||||
|
||||
if ($this->getConfig()->get('followCreatedEntities') && !empty($createdById)) {
|
||||
if (
|
||||
($this->getConfig()->get('followCreatedEntities') || $this->getUser()->get('isPortalUser'))
|
||||
&&
|
||||
!empty($createdById)
|
||||
) {
|
||||
$userIdList[] = $createdById;
|
||||
}
|
||||
if (!empty($assignedUserId) && !in_array($assignedUserId, $userIdList)) {
|
||||
@@ -242,6 +249,36 @@ class Stream extends \Espo\Core\Hooks\Base
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
$methodName = 'isChangedWithAclAffect';
|
||||
if (
|
||||
(
|
||||
method_exists($entity, $methodName) && $entity->$methodName()
|
||||
)
|
||||
||
|
||||
(
|
||||
!method_exists($entity, $methodName)
|
||||
&&
|
||||
(
|
||||
$entity->isAttributeChanged('assignedUserId')
|
||||
||
|
||||
$entity->isAttributeChanged('teamsIds')
|
||||
||
|
||||
$entity->isAttributeChanged('assignedUsersIds')
|
||||
)
|
||||
)
|
||||
) {
|
||||
$job = $this->getEntityManager()->getEntity('Job');
|
||||
$job->set(array(
|
||||
'serviceName' => 'Stream',
|
||||
'method' => 'controlFollowersJob',
|
||||
'data' => array(
|
||||
'entityType' => $entity->getEntityType(),
|
||||
'entityId' => $entity->id
|
||||
)
|
||||
));
|
||||
$this->getEntityManager()->saveEntity($job);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
48
application/Espo/Hooks/Integration/GoogleMaps.php
Normal file
48
application/Espo/Hooks/Integration/GoogleMaps.php
Normal file
@@ -0,0 +1,48 @@
|
||||
<?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/.
|
||||
*
|
||||
* The interactive user interfaces in modified source and object code versions
|
||||
* of this program must display Appropriate Legal Notices, as required under
|
||||
* Section 5 of the GNU General Public License version 3.
|
||||
*
|
||||
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
||||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Hooks\Integration;
|
||||
|
||||
use Espo\ORM\Entity;
|
||||
|
||||
class GoogleMaps extends \Espo\Core\Hooks\Base
|
||||
{
|
||||
public function afterSave(Entity $entity)
|
||||
{
|
||||
if ($entity->id === 'GoogleMaps') {
|
||||
if (!$entity->get('enabled') || !$entity->get('apiKey')) {
|
||||
$this->getConfig()->set('googleMapsApiKey', null);
|
||||
$this->getConfig()->save();
|
||||
return;
|
||||
}
|
||||
$this->getConfig()->set('googleMapsApiKey', $entity->get('apiKey'));
|
||||
$this->getConfig()->save();
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -39,7 +39,7 @@ class Mentions extends \Espo\Core\Hooks\Base
|
||||
|
||||
protected function init()
|
||||
{
|
||||
$this->dependencies[] = 'serviceFactory';
|
||||
$this->addDependency('serviceFactory');
|
||||
}
|
||||
|
||||
protected function getServiceFactory()
|
||||
@@ -65,7 +65,13 @@ class Mentions extends \Espo\Core\Hooks\Base
|
||||
|
||||
$mentionCount = 0;
|
||||
|
||||
|
||||
|
||||
if (is_array($matches) && !empty($matches[0]) && is_array($matches[0])) {
|
||||
$parent = null;
|
||||
if ($entity->get('parentId') && $entity->get('parentType')) {
|
||||
$parent = $this->getEntityManager()->getEntity($entity->get('parentType'), $entity->get('parentId'));
|
||||
}
|
||||
foreach ($matches[0] as $item) {
|
||||
$userName = substr($item, 1);
|
||||
$user = $this->getEntityManager()->getRepository('User')->where(array('userName' => $userName))->findOne();
|
||||
@@ -85,7 +91,7 @@ class Mentions extends \Espo\Core\Hooks\Base
|
||||
if ($user->id == $this->getUser()->id) {
|
||||
continue;
|
||||
}
|
||||
$this->notifyAboutMention($entity, $user);
|
||||
$this->notifyAboutMention($entity, $user, $parent);
|
||||
$entity->addNotifiedUserId($user->id);
|
||||
}
|
||||
}
|
||||
@@ -114,8 +120,12 @@ class Mentions extends \Espo\Core\Hooks\Base
|
||||
}
|
||||
}
|
||||
|
||||
protected function notifyAboutMention(Entity $entity, \Espo\Entities\User $user)
|
||||
protected function notifyAboutMention(Entity $entity, \Espo\Entities\User $user, Entity $parent = null)
|
||||
{
|
||||
if ($user->get('isPortalUser')) return;
|
||||
if ($parent) {
|
||||
if (!$this->getAclManager()->check($user, $parent, 'stream')) return;
|
||||
}
|
||||
$this->getNotificationService()->notifyAboutMentionInPost($user->id, $entity->id);
|
||||
}
|
||||
|
||||
@@ -127,4 +137,3 @@ class Mentions extends \Espo\Core\Hooks\Base
|
||||
return $this->notificationService;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -39,7 +39,7 @@ class Notifications extends \Espo\Core\Hooks\Base
|
||||
|
||||
protected function init()
|
||||
{
|
||||
$this->dependencies[] = 'serviceFactory';
|
||||
$this->addDependency('serviceFactory');
|
||||
}
|
||||
|
||||
protected function getServiceFactory()
|
||||
@@ -60,13 +60,26 @@ class Notifications extends \Espo\Core\Hooks\Base
|
||||
return $mentionedUserList;
|
||||
}
|
||||
|
||||
protected function getSubscriberIdList($parentType, $parentId)
|
||||
protected function getSubscriberIdList($parentType, $parentId, $isInternal = false)
|
||||
{
|
||||
$pdo = $this->getEntityManager()->getPDO();
|
||||
$sql = "
|
||||
SELECT user_id AS userId
|
||||
FROM subscription
|
||||
WHERE entity_id = " . $pdo->quote($parentId) . " AND entity_type = " . $pdo->quote($parentType);
|
||||
|
||||
if (!$isInternal) {
|
||||
$sql = "
|
||||
SELECT user_id AS userId
|
||||
FROM subscription
|
||||
WHERE entity_id = " . $pdo->quote($parentId) . " AND entity_type = " . $pdo->quote($parentType) . "
|
||||
";
|
||||
} else {
|
||||
$sql = "
|
||||
SELECT subscription.user_id AS userId
|
||||
FROM subscription
|
||||
JOIN user ON user.id = subscription.user_id
|
||||
WHERE
|
||||
entity_id = " . $pdo->quote($parentId) . " AND entity_type = " . $pdo->quote($parentType) . " AND
|
||||
user.is_portal_user = 0
|
||||
";
|
||||
}
|
||||
$sth = $pdo->prepare($sql);
|
||||
$sth->execute();
|
||||
$userIdList = [];
|
||||
@@ -89,9 +102,9 @@ class Notifications extends \Espo\Core\Hooks\Base
|
||||
$userIdList = [];
|
||||
|
||||
if ($parentType && $parentId) {
|
||||
$userIdList = array_merge($userIdList, $this->getSubscriberIdList($parentType, $parentId));
|
||||
$userIdList = array_merge($userIdList, $this->getSubscriberIdList($parentType, $parentId, $entity->get('isInternal')));
|
||||
if ($superParentType && $superParentId) {
|
||||
$userIdList = array_merge($userIdList, $this->getSubscriberIdList($superParentType, $superParentId));
|
||||
$userIdList = array_merge($userIdList, $this->getSubscriberIdList($superParentType, $superParentId, $entity->get('isInternal')));
|
||||
}
|
||||
} else {
|
||||
$targetType = $entity->get('targetType');
|
||||
|
||||
42
application/Espo/Jobs/SendEmailNotifications.php
Normal file
42
application/Espo/Jobs/SendEmailNotifications.php
Normal file
@@ -0,0 +1,42 @@
|
||||
<?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/.
|
||||
*
|
||||
* The interactive user interfaces in modified source and object code versions
|
||||
* of this program must display Appropriate Legal Notices, as required under
|
||||
* Section 5 of the GNU General Public License version 3.
|
||||
*
|
||||
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
||||
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
||||
************************************************************************/
|
||||
|
||||
namespace Espo\Jobs;
|
||||
|
||||
use \Espo\Core\Exceptions;
|
||||
|
||||
class SendEmailNotifications extends \Espo\Core\Jobs\Base
|
||||
{
|
||||
public function run()
|
||||
{
|
||||
$service = $this->getServiceFactory()->create('EmailNotification');
|
||||
$service->process();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -47,7 +47,7 @@ class Invitations
|
||||
|
||||
protected $ics;
|
||||
|
||||
public function __construct($entityManager, $smtpParams, $mailSender, $config, $dateTime, $language)
|
||||
public function __construct($entityManager, $smtpParams, $mailSender, $config, $fileManager, $dateTime, $number, $language)
|
||||
{
|
||||
$this->entityManager = $entityManager;
|
||||
$this->smtpParams = $smtpParams;
|
||||
@@ -55,6 +55,8 @@ class Invitations
|
||||
$this->config = $config;
|
||||
$this->dateTime = $dateTime;
|
||||
$this->language = $language;
|
||||
$this->number = $number;
|
||||
$this->fileManager = $fileManager;
|
||||
}
|
||||
|
||||
protected function getEntityManager()
|
||||
@@ -62,78 +64,24 @@ class Invitations
|
||||
return $this->entityManager;
|
||||
}
|
||||
|
||||
protected function parseInvitationTemplate($contents, $entity, $invitee = null, $uid = null)
|
||||
protected function getConfig()
|
||||
{
|
||||
|
||||
$contents = str_replace('{eventType}', strtolower($this->language->translate($entity->getEntityType(), 'scopeNames')), $contents);
|
||||
|
||||
foreach ($entity->getAttributes() as $field => $d) {
|
||||
if (empty($d['type'])) continue;
|
||||
$key = '{'.$field.'}';
|
||||
switch ($d['type']) {
|
||||
case 'datetime':
|
||||
$value = $entity->get($field);
|
||||
if ($value) {
|
||||
$value = $this->dateTime->convertSystemDateTime($value);
|
||||
}
|
||||
$contents = str_replace($key, $value, $contents);
|
||||
break;
|
||||
case 'date':
|
||||
$value = $entity->get($field);
|
||||
if ($value) {
|
||||
$value = $this->dateTime->convertSystemDate($value);
|
||||
}
|
||||
$contents = str_replace($key, $value, $contents);
|
||||
break;
|
||||
case 'jsonArray':
|
||||
break;
|
||||
case 'jsonObject':
|
||||
break;
|
||||
default:
|
||||
$value = $entity->get($field);
|
||||
if (is_string($value) || $value === null || is_scalar($value) || is_callable([$value, '__toString'])) {
|
||||
$contents = str_replace($key, $value, $contents);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if ($invitee) {
|
||||
$contents = str_replace('{inviteeName}', $invitee->get('name'), $contents);
|
||||
}
|
||||
|
||||
$siteUrl = rtrim($this->config->get('siteUrl'), '/');
|
||||
|
||||
$url = $siteUrl . '/#' . $entity->getEntityType() . '/view/' . $entity->id;
|
||||
$contents = str_replace('{url}', $url, $contents);
|
||||
|
||||
if ($invitee && $invitee->getEntityType() != 'User') {
|
||||
$contents = preg_replace('/\{#userOnly\}(.*?)\{\/userOnly\}/s', '', $contents);
|
||||
}
|
||||
|
||||
$contents = str_replace('{#userOnly}', '', $contents);
|
||||
$contents = str_replace('{/userOnly}', '', $contents);
|
||||
|
||||
if ($uid) {
|
||||
$contents = str_replace('{acceptLink}', $siteUrl . '?entryPoint=eventConfirmation&action=accept&uid=' . $uid->get('name'), $contents);
|
||||
$contents = str_replace('{declineLink}', $siteUrl . '?entryPoint=eventConfirmation&action=decline&uid=' . $uid->get('name'), $contents);
|
||||
$contents = str_replace('{tentativeLink}', $siteUrl . '?entryPoint=eventConfirmation&action=tentative&uid=' . $uid->get('name'), $contents);
|
||||
}
|
||||
return $contents;
|
||||
return $this->config;
|
||||
}
|
||||
|
||||
protected function getTemplate($name)
|
||||
{
|
||||
$systemLanguage = $this->config->get('language');
|
||||
|
||||
$fileName = 'custom/Espo/Custom/Resources/templates/'.$name.'.'.$systemLanguage.'.tpl';
|
||||
$fileName = "custom/Espo/Custom/Resources/templates/invitation/{$systemLanguage}/{$name}.tpl";
|
||||
if (!file_exists($fileName)) {
|
||||
$fileName = 'application/Espo/Modules/Crm/Resources/templates/'.$name.'.'.$systemLanguage.'.tpl';
|
||||
$fileName = "application/Espo/Modules/Crm/Resources/templates/invitation/{$systemLanguage}/{$name}.tpl";
|
||||
}
|
||||
if (!file_exists($fileName)) {
|
||||
$fileName = 'custom/Espo/Custom/Resources/templates/'.$name.'.en_US.tpl';
|
||||
$fileName = "custom/Espo/Custom/Resources/templates/invitation/en_US/{$name}.tpl";
|
||||
}
|
||||
if (!file_exists($fileName)) {
|
||||
$fileName = 'application/Espo/Modules/Crm/Resources/templates/'.$name.'.en_US.tpl';
|
||||
$fileName = "application/Espo/Modules/Crm/Resources/templates/invitation/en_US/{$name}.tpl";
|
||||
}
|
||||
|
||||
return file_get_contents($fileName);
|
||||
@@ -159,13 +107,44 @@ class Invitations
|
||||
$email = $this->getEntityManager()->getEntity('Email');
|
||||
$email->set('to', $emailAddress);
|
||||
|
||||
$subjectTpl = $this->getTemplate('InvitationSubject');
|
||||
$bodyTpl = $this->getTemplate('InvitationBody');
|
||||
$subjectTpl = $this->getTemplate('subject');
|
||||
$bodyTpl = $this->getTemplate('body');
|
||||
$subjectTpl = str_replace(array("\n", "\r"), '', $subjectTpl);
|
||||
|
||||
$subject = $this->parseInvitationTemplate($subjectTpl, $entity, $invitee, $uid);
|
||||
$subject = str_replace(array("\n", "\r"), '', $subject);
|
||||
$data = array();
|
||||
|
||||
$body = $this->parseInvitationTemplate($bodyTpl, $entity, $invitee, $uid);
|
||||
$siteUrl = rtrim($this->getConfig()->get('siteUrl'), '/');
|
||||
$recordUrl = $siteUrl . '/#' . $entity->getEntityType() . '/view/' . $entity->id;
|
||||
$data['recordUrl'] = $recordUrl;
|
||||
|
||||
$data['acceptLink'] = $siteUrl . '?entryPoint=eventConfirmation&action=accept&uid=' . $uid->get('name');
|
||||
$data['declineLink'] = $siteUrl . '?entryPoint=eventConfirmation&action=decline&uid=' . $uid->get('name');
|
||||
$data['tentativeLink'] = $siteUrl . '?entryPoint=eventConfirmation&action=tentative&uid=' . $uid->get('name');
|
||||
|
||||
if ($invitee && $invitee->getEntityType() === 'User') {
|
||||
$data['isUser'] = true;
|
||||
|
||||
$preferences = $this->getEntityManager()->getEntity('Preferences', $invitee->id);
|
||||
$timezone = $preferences->get('timeZone');
|
||||
$dateTime = clone($this->dateTime);
|
||||
if ($timezone) {
|
||||
$dateTime->setTimezone($timezone);
|
||||
}
|
||||
} else {
|
||||
$dateTime = $this->dateTime;
|
||||
}
|
||||
|
||||
if ($invitee) {
|
||||
$data['inviteeName'] = $invitee->get('name');
|
||||
}
|
||||
|
||||
$data['entityType'] = $this->language->translate($entity->getEntityType(), 'scopeNames');
|
||||
$data['entityTypeLowerFirst'] = lcfirst($data['entityType']);
|
||||
|
||||
$htmlizer = new \Espo\Core\Htmlizer\Htmlizer($this->fileManager, $dateTime, $this->number, null);
|
||||
|
||||
$subject = $htmlizer->render($entity, $subjectTpl, 'invitation-email-subject-' . $entity->getEntityType(), $data, true);
|
||||
$body = $htmlizer->render($entity, $bodyTpl, 'invitation-email-body-' . $entity->getEntityType(), $data, true);
|
||||
|
||||
$email->set('subject', $subject);
|
||||
$email->set('body', $body);
|
||||
|
||||
@@ -43,14 +43,15 @@ class EmailReminder
|
||||
|
||||
protected $language;
|
||||
|
||||
|
||||
public function __construct($entityManager, $mailSender, $config, $dateTime, $language)
|
||||
public function __construct($entityManager, $mailSender, $config, $fileManager, $dateTime, $number, $language)
|
||||
{
|
||||
$this->entityManager = $entityManager;
|
||||
$this->mailSender = $mailSender;
|
||||
$this->config = $config;
|
||||
$this->dateTime = $dateTime;
|
||||
$this->language = $language;
|
||||
$this->number = $number;
|
||||
$this->fileManager = $fileManager;
|
||||
}
|
||||
|
||||
protected function getEntityManager()
|
||||
@@ -58,6 +59,16 @@ class EmailReminder
|
||||
return $this->entityManager;
|
||||
}
|
||||
|
||||
protected function getConfig()
|
||||
{
|
||||
return $this->config;
|
||||
}
|
||||
|
||||
protected function getLanguage()
|
||||
{
|
||||
return $this->language;
|
||||
}
|
||||
|
||||
protected function parseInvitationTemplate($contents, $entity, $user = null)
|
||||
{
|
||||
|
||||
@@ -97,15 +108,15 @@ class EmailReminder
|
||||
{
|
||||
$systemLanguage = $this->config->get('language');
|
||||
|
||||
$fileName = 'custom/Espo/Custom/Resources/templates/'.$name.'.'.$systemLanguage.'.tpl';
|
||||
$fileName = "custom/Espo/Custom/Resources/templates/reminder/{$systemLanguage}/{$name}.tpl";
|
||||
if (!file_exists($fileName)) {
|
||||
$fileName = 'application/Espo/Modules/Crm/Resources/templates/'.$name.'.'.$systemLanguage.'.tpl';
|
||||
$fileName = "application/Espo/Modules/Crm/Resources/templates/reminder/{$systemLanguage}/{$name}.tpl";
|
||||
}
|
||||
if (!file_exists($fileName)) {
|
||||
$fileName = 'custom/Espo/Custom/Resources/templates/'.$name.'.en_US.tpl';
|
||||
$fileName = "custom/Espo/Custom/Resources/templates/reminder/en_US/{$name}.tpl";
|
||||
}
|
||||
if (!file_exists($fileName)) {
|
||||
$fileName = 'application/Espo/Modules/Crm/Resources/templates/'.$name.'.en_US.tpl';
|
||||
$fileName = "application/Espo/Modules/Crm/Resources/templates/reminder/en_US/{$name}.tpl";
|
||||
}
|
||||
|
||||
return file_get_contents($fileName);
|
||||
@@ -125,13 +136,34 @@ class EmailReminder
|
||||
$email = $this->getEntityManager()->getEntity('Email');
|
||||
$email->set('to', $emailAddress);
|
||||
|
||||
$subjectTpl = $this->getTemplate('ReminderSubject');
|
||||
$bodyTpl = $this->getTemplate('ReminderBody');
|
||||
$subjectTpl = $this->getTemplate('subject');
|
||||
$bodyTpl = $this->getTemplate('body');
|
||||
$subjectTpl = str_replace(array("\n", "\r"), '', $subjectTpl);
|
||||
|
||||
$subject = $this->parseInvitationTemplate($subjectTpl, $entity, $user);
|
||||
$subject = str_replace(array("\n", "\r"), '', $subject);
|
||||
$data = array();
|
||||
|
||||
$body = $this->parseInvitationTemplate($bodyTpl, $entity, $user);
|
||||
$siteUrl = rtrim($this->getConfig()->get('siteUrl'), '/');
|
||||
$recordUrl = $siteUrl . '/#' . $entity->getEntityType() . '/view/' . $entity->id;
|
||||
$data['recordUrl'] = $recordUrl;
|
||||
|
||||
$data['entityType'] = $this->getLanguage()->translate($entity->getEntityType(), 'scopeNames');
|
||||
$data['entityTypeLowerFirst'] = lcfirst($data['entityType']);
|
||||
|
||||
if ($user) {
|
||||
$data['userName'] = $user->get('name');
|
||||
}
|
||||
|
||||
$preferences = $this->getEntityManager()->getEntity('Preferences', $user->id);
|
||||
$timezone = $preferences->get('timeZone');
|
||||
$dateTime = clone($this->dateTime);
|
||||
if ($timezone) {
|
||||
$dateTime->setTimezone($timezone);
|
||||
}
|
||||
|
||||
$htmlizer = new \Espo\Core\Htmlizer\Htmlizer($this->fileManager, $dateTime, $this->number, null);
|
||||
|
||||
$subject = $htmlizer->render($entity, $subjectTpl, 'reminder-email-subject-' . $entity->getEntityType(), $data, true);
|
||||
$body = $htmlizer->render($entity, $bodyTpl, 'reminder-email-body-' . $entity->getEntityType(), $data, true);
|
||||
|
||||
$email->set('subject', $subject);
|
||||
$email->set('body', $body);
|
||||
|
||||
@@ -40,4 +40,36 @@ class KnowledgeBaseArticle extends \Espo\Core\Controllers\Record
|
||||
|
||||
return $this->getRecordService()->getCopiedAttachments($id);
|
||||
}
|
||||
|
||||
public function postActionMoveUp($params, $data, $request)
|
||||
{
|
||||
if (empty($data['id'])) {
|
||||
throw new BadRequest();
|
||||
}
|
||||
$where = null;
|
||||
if (!empty($data['where'])) {
|
||||
$where = $data['where'];
|
||||
$where = json_decode(json_encode($where), true);
|
||||
}
|
||||
|
||||
$this->getRecordService()->moveUp($data['id'], $where);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public function postActionMoveDown($params, $data, $request)
|
||||
{
|
||||
if (empty($data['id'])) {
|
||||
throw new BadRequest();
|
||||
}
|
||||
$where = null;
|
||||
if (!empty($data['where'])) {
|
||||
$where = $data['where'];
|
||||
$where = json_decode(json_encode($where), true);
|
||||
}
|
||||
|
||||
$this->getRecordService()->moveDown($data['id'], $where);
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -91,7 +91,7 @@ class CampaignTrackOpened extends \Espo\Core\EntryPoints\Base
|
||||
|
||||
imagepng($img);
|
||||
imagecolordeallocate($background);
|
||||
imagedestroy( $tt_image );
|
||||
imagedestroy($img);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -45,7 +45,7 @@ class SendEmailReminders extends \Espo\Core\Jobs\Base
|
||||
$collection = $this->getEntityManager()->getRepository('Reminder')->where(array(
|
||||
'type' => 'Email',
|
||||
'remindAt<=' => $now,
|
||||
'startAt>' => $nowShifted,
|
||||
'startAt>' => $nowShifted
|
||||
))->find();
|
||||
|
||||
if (!empty($collection)) {
|
||||
@@ -53,8 +53,11 @@ class SendEmailReminders extends \Espo\Core\Jobs\Base
|
||||
$this->getEntityManager(),
|
||||
$this->getContainer()->get('mailSender'),
|
||||
$this->getConfig(),
|
||||
$this->getContainer()->get('fileManager'),
|
||||
$this->getContainer()->get('dateTime'),
|
||||
$this->getContainer()->get('number'),
|
||||
$this->getContainer()->get('language')
|
||||
|
||||
);
|
||||
$pdo = $this->getEntityManager()->getPDO();
|
||||
}
|
||||
|
||||
@@ -87,4 +87,19 @@ class KnowledgeBaseArticle extends \Espo\Core\ORM\Repositories\RDB
|
||||
|
||||
$this->getEntityManager()->removeEntity($note);
|
||||
}
|
||||
|
||||
protected function beforeSave(Entity $entity, array $options = array())
|
||||
{
|
||||
parent::beforeSave($entity, $options);
|
||||
$order = $entity->get('order');
|
||||
if (is_null($order)) {
|
||||
$order = $this->min('order');
|
||||
if (!$order) {
|
||||
$order = 9999;
|
||||
}
|
||||
$order--;
|
||||
$entity->set('order', $order);
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
@@ -61,19 +61,23 @@ class Meeting extends \Espo\Core\ORM\Repositories\RDB
|
||||
}
|
||||
|
||||
$assignedUserId = $entity->get('assignedUserId');
|
||||
if ($assignedUserId && $entity->has('usersIds')) {
|
||||
$usersIds = $entity->get('usersIds');
|
||||
if (!is_array($usersIds)) {
|
||||
$usersIds = array();
|
||||
}
|
||||
if (!in_array($assignedUserId, $usersIds)) {
|
||||
$usersIds[] = $assignedUserId;
|
||||
$entity->set('usersIds', $usersIds);
|
||||
$hash = $entity->get('usersNames');
|
||||
if ($hash instanceof \StdClass) {
|
||||
$hash->$assignedUserId = $entity->get('assignedUserName');
|
||||
$entity->set('usersNames', $hash);
|
||||
if ($assignedUserId) {
|
||||
if ($entity->has('usersIds')) {
|
||||
$usersIds = $entity->get('usersIds');
|
||||
if (!is_array($usersIds)) {
|
||||
$usersIds = [];
|
||||
}
|
||||
if (!in_array($assignedUserId, $usersIds)) {
|
||||
$usersIds[] = $assignedUserId;
|
||||
$entity->set('usersIds', $usersIds);
|
||||
$hash = $entity->get('usersNames');
|
||||
if ($hash instanceof \StdClass) {
|
||||
$hash->$assignedUserId = $entity->get('assignedUserName');
|
||||
$entity->set('usersNames', $hash);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
$entity->addLinkMultipleId('users', $assignedUserId);
|
||||
}
|
||||
if ($entity->isNew()) {
|
||||
$currentUserId = $this->getEntityManager()->getUser()->id;
|
||||
@@ -114,19 +118,19 @@ class Meeting extends \Espo\Core\ORM\Repositories\RDB
|
||||
}
|
||||
}
|
||||
|
||||
public function getEntityReminders(Entity $entity)
|
||||
public function getEntityReminderList(Entity $entity)
|
||||
{
|
||||
$pdo = $this->getEntityManager()->getPDO();
|
||||
$reminders = array();
|
||||
$reminderList = [];
|
||||
|
||||
$sql = "
|
||||
SELECT id, `seconds`, `type`
|
||||
SELECT DISTINCT `seconds`, `type`
|
||||
FROM `reminder`
|
||||
WHERE
|
||||
`entity_type` = ".$pdo->quote($entity->getEntityType())." AND
|
||||
`entity_id` = ".$pdo->quote($entity->id)." AND
|
||||
`deleted` = 0
|
||||
ORDER BY `seconds` ASC
|
||||
ORDER BY `seconds` ASC
|
||||
";
|
||||
|
||||
$sth = $pdo->prepare($sql);
|
||||
@@ -137,10 +141,10 @@ class Meeting extends \Espo\Core\ORM\Repositories\RDB
|
||||
$o = new \StdClass();
|
||||
$o->seconds = intval($row['seconds']);
|
||||
$o->type = $row['type'];
|
||||
$reminders[] = $o;
|
||||
$reminderList[] = $o;
|
||||
}
|
||||
|
||||
return $reminders;
|
||||
return $reminderList;
|
||||
}
|
||||
|
||||
protected function afterSave(Entity $entity, array $options = array())
|
||||
@@ -150,6 +154,7 @@ class Meeting extends \Espo\Core\ORM\Repositories\RDB
|
||||
if (
|
||||
$entity->isNew() ||
|
||||
$entity->isFieldChanged('assignedUserId') ||
|
||||
$entity->isFieldChanged('usersIds') ||
|
||||
$entity->isFieldChanged('dateStart') ||
|
||||
$entity->has('reminders')
|
||||
) {
|
||||
@@ -158,9 +163,9 @@ class Meeting extends \Espo\Core\ORM\Repositories\RDB
|
||||
$reminderTypeList = $this->getMetadata()->get('entityDefs.Reminder.fields.type.options');
|
||||
|
||||
if (!$entity->has('reminders')) {
|
||||
$reminders = $this->getEntityReminders($entity);
|
||||
$reminderList = $this->getEntityReminderList($entity);
|
||||
} else {
|
||||
$reminders = $entity->get('reminders');
|
||||
$reminderList = $entity->get('reminders');
|
||||
}
|
||||
|
||||
if (!$entity->isNew()) {
|
||||
@@ -174,32 +179,28 @@ class Meeting extends \Espo\Core\ORM\Repositories\RDB
|
||||
$pdo->query($sql);
|
||||
}
|
||||
|
||||
if (empty($reminders) || !is_array($reminders)) return;
|
||||
if (empty($reminderList) || !is_array($reminderList)) return;
|
||||
|
||||
$entityType = $entity->getEntityName();
|
||||
|
||||
$dateStart = $entity->get('dateStart');
|
||||
$assignedUserId = $entity->get('assignedUserId');
|
||||
|
||||
if (!$dateStart || !$assignedUserId) {
|
||||
if (!$dateStart) {
|
||||
$e = $this->get($entity->id);
|
||||
if ($e) {
|
||||
$dateStart = $e->get('dateStart');
|
||||
$assignedUserId = $e->get('assignedUserId');
|
||||
}
|
||||
}
|
||||
|
||||
if (!$dateStart || !$assignedUserId) {
|
||||
return;
|
||||
}
|
||||
$userIdList = $entity->getLinkMultipleIdList('users');
|
||||
|
||||
if (!$dateStart) return;
|
||||
if (empty($userIdList)) return;
|
||||
|
||||
$dateStartObj = new \DateTime($dateStart);
|
||||
if (!$dateStartObj) {
|
||||
return;
|
||||
}
|
||||
if (!$dateStartObj) return;
|
||||
|
||||
foreach ($reminders as $item) {
|
||||
foreach ($reminderList as $item) {
|
||||
$remindAt = clone $dateStartObj;
|
||||
$seconds = intval($item->seconds);
|
||||
$type = $item->type;
|
||||
@@ -208,25 +209,26 @@ class Meeting extends \Espo\Core\ORM\Repositories\RDB
|
||||
|
||||
$remindAt->sub(new \DateInterval('PT' . $seconds . 'S'));
|
||||
|
||||
$id = uniqid();
|
||||
|
||||
$sql = "
|
||||
INSERT
|
||||
INTO `reminder`
|
||||
(id, entity_id, entity_type, `type`, user_id, remind_at, start_at, `seconds`)
|
||||
VALUES (
|
||||
".$pdo->quote($id).",
|
||||
".$pdo->quote($entity->id).",
|
||||
".$pdo->quote($entityType).",
|
||||
".$pdo->quote($type).",
|
||||
".$pdo->quote($assignedUserId).",
|
||||
".$pdo->quote($remindAt->format('Y-m-d H:i:s')).",
|
||||
".$pdo->quote($dateStart).",
|
||||
".$pdo->quote($seconds)."
|
||||
)
|
||||
";
|
||||
$pdo->query($sql);
|
||||
foreach ($userIdList as $userId) {
|
||||
$id = uniqid(true);
|
||||
|
||||
$sql = "
|
||||
INSERT
|
||||
INTO `reminder`
|
||||
(id, entity_id, entity_type, `type`, user_id, remind_at, start_at, `seconds`)
|
||||
VALUES (
|
||||
".$pdo->quote($id).",
|
||||
".$pdo->quote($entity->id).",
|
||||
".$pdo->quote($entityType).",
|
||||
".$pdo->quote($type).",
|
||||
".$pdo->quote($userId).",
|
||||
".$pdo->quote($remindAt->format('Y-m-d H:i:s')).",
|
||||
".$pdo->quote($dateStart).",
|
||||
".$pdo->quote($seconds)."
|
||||
)
|
||||
";
|
||||
$pdo->query($sql);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -35,8 +35,9 @@ class Task extends \Espo\Core\ORM\Repositories\RDB
|
||||
{
|
||||
protected function init()
|
||||
{
|
||||
$this->dependencies[] = 'dateTime';
|
||||
$this->dependencies[] = 'config';
|
||||
parent::init();
|
||||
$this->addDependency('dateTime');
|
||||
$this->addDependency('config');
|
||||
}
|
||||
|
||||
protected function getConfig()
|
||||
|
||||
@@ -1,79 +1,80 @@
|
||||
{
|
||||
"fields": {
|
||||
"name": "Name",
|
||||
"emailAddress": "E-Mail",
|
||||
"website": "Webseite",
|
||||
"phoneNumber": "Telefon",
|
||||
"billingAddress": "Rechnungsadresse",
|
||||
"shippingAddress": "Lieferadresse",
|
||||
"description": "Beschreibung",
|
||||
"sicCode": "WKN Nummer",
|
||||
"industry": "Branche",
|
||||
"type": "Typ",
|
||||
"contactRole": "Rolle",
|
||||
"campaign": "Kampagne",
|
||||
"targetLists": "Kontaktlisten",
|
||||
"targetList": "Kontaktliste"
|
||||
"fields": {
|
||||
"name": "Betreff",
|
||||
"emailAddress": "E-Mail",
|
||||
"website": "Webseite",
|
||||
"phoneNumber": "Telefon",
|
||||
"billingAddress": "Rechnungsadresse",
|
||||
"shippingAddress": "Lieferadresse",
|
||||
"description": "Beschreibung",
|
||||
"sicCode": "WKN Nummer",
|
||||
"industry": "Branche",
|
||||
"type": "Typ",
|
||||
"contactRole": "Rolle",
|
||||
"campaign": "Kampagne",
|
||||
"targetLists": "Kontaktlisten",
|
||||
"targetList": "Kontaktliste"
|
||||
},
|
||||
"links": {
|
||||
"contacts": "Kontakte",
|
||||
"opportunities": "Verkaufschancen",
|
||||
"cases": "Fälle",
|
||||
"documents": "Dokumente",
|
||||
"meetingsPrimary": "Meetings (erweitert)",
|
||||
"callsPrimary": "Anrufe (erweitert)",
|
||||
"tasksPrimary": "Aufgaben (erweitert)",
|
||||
"emailsPrimary": "E-Mails (erweitert)",
|
||||
"targetLists": "Kontaktlisten",
|
||||
"campaignLogRecords": "Kampagnen Log",
|
||||
"campaign": "Kampagne",
|
||||
"portalUsers": "Portal Benutzer"
|
||||
},
|
||||
"options": {
|
||||
"type": {
|
||||
"Customer": "Kunde",
|
||||
"Investor": "Investor",
|
||||
"Partner": "Partner",
|
||||
"Reseller": "Wiederverkäufer"
|
||||
},
|
||||
"links": {
|
||||
"contacts": "Kontakte",
|
||||
"opportunities": "Verkaufschancen",
|
||||
"cases": "Fälle",
|
||||
"documents": "Dokumente",
|
||||
"meetingsPrimary": "Meetings (erweitert)",
|
||||
"callsPrimary": "Anrufe (erweitert)",
|
||||
"tasksPrimary": "Aufgaben (erweitert)",
|
||||
"emailsPrimary": "E-Mails (erweitert)",
|
||||
"targetLists": "Kontaktlisten",
|
||||
"campaignLogRecords": "Kampagnen Log",
|
||||
"campaign": "Kampagne"
|
||||
},
|
||||
"options": {
|
||||
"type": {
|
||||
"Customer": "Kunde",
|
||||
"Investor": "Investor",
|
||||
"Partner": "Partner",
|
||||
"Reseller": "Wiederverkäufer"
|
||||
},
|
||||
"industry": {
|
||||
"Agriculture": "Landwirtschaft",
|
||||
"Advertising": "Werbewirtschaft",
|
||||
"Apparel & Accessories": "Bekleidungsindustrie",
|
||||
"Automotive": "Automobilindustrie",
|
||||
"Banking": "Bankwesen",
|
||||
"Biotechnology": "Biotechnologie",
|
||||
"Building Materials & Equipment": "Baumaterial & -ausstattung",
|
||||
"Chemical": "Chemieindustrie",
|
||||
"Computer": "Informationstechnologie",
|
||||
"Education": "Bildungswesen",
|
||||
"Electronics": "Elektronik",
|
||||
"Energy": "Energieerzeuger",
|
||||
"Entertainment & Leisure": "Freizeit- und Unterhaltungsindustrie",
|
||||
"Finance": "Finanzsektor",
|
||||
"Food & Beverage": "Speisen und Getränke",
|
||||
"Grocery": "Einzelhandel",
|
||||
"Healthcare": "Gesundheitswesen",
|
||||
"Insurance": "Versicherung",
|
||||
"Legal": "Rechtswesen",
|
||||
"Manufacturing": "Produktion",
|
||||
"Publishing": "Medien",
|
||||
"Real Estate": "Immobilien",
|
||||
"Service": "Service",
|
||||
"Sports": "Sport",
|
||||
"Software": "Software",
|
||||
"Technology": "Technologie",
|
||||
"Telecommunications": "Telekommunikation",
|
||||
"Television": "Fernsehen",
|
||||
"Transportation": "Transportwesen",
|
||||
"Venture Capital": "Risikokapital"
|
||||
}
|
||||
},
|
||||
"labels": {
|
||||
"Create Account": "Firma erstellen",
|
||||
"Copy Billing": "Rechnungsadresse kopieren"
|
||||
},
|
||||
"presetFilters": {
|
||||
"customers": "Kunden",
|
||||
"partners": "Partner"
|
||||
"industry": {
|
||||
"Agriculture": "Landwirtschaft",
|
||||
"Advertising": "Werbewirtschaft",
|
||||
"Apparel & Accessories": "Bekleidungsindustrie",
|
||||
"Automotive": "Automobilindustrie",
|
||||
"Banking": "Bankwesen",
|
||||
"Biotechnology": "Biotechnologie",
|
||||
"Building Materials & Equipment": "Baumaterial & -ausstattung",
|
||||
"Chemical": "Chemieindustrie",
|
||||
"Computer": "Informationstechnologie",
|
||||
"Education": "Bildungswesen",
|
||||
"Electronics": "Elektronik",
|
||||
"Energy": "Energieerzeuger",
|
||||
"Entertainment & Leisure": "Freizeit- und Unterhaltungsindustrie",
|
||||
"Finance": "Finanzsektor",
|
||||
"Food & Beverage": "Speisen und Getränke",
|
||||
"Grocery": "Einzelhandel",
|
||||
"Healthcare": "Gesundheitswesen",
|
||||
"Insurance": "Versicherung",
|
||||
"Legal": "Rechtswesen",
|
||||
"Manufacturing": "Produktion",
|
||||
"Publishing": "Medien",
|
||||
"Real Estate": "Immobilien",
|
||||
"Service": "Service",
|
||||
"Sports": "Sport",
|
||||
"Software": "Software",
|
||||
"Technology": "Technologie",
|
||||
"Telecommunications": "Telekommunikation",
|
||||
"Television": "Fernsehen",
|
||||
"Transportation": "Transportwesen",
|
||||
"Venture Capital": "Risikokapital"
|
||||
}
|
||||
}
|
||||
},
|
||||
"labels": {
|
||||
"Create Account": "Firma erstellen",
|
||||
"Copy Billing": "Rechnungsadresse kopieren"
|
||||
},
|
||||
"presetFilters": {
|
||||
"customers": "Kunden",
|
||||
"partners": "Partner"
|
||||
}
|
||||
}
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"layouts": {
|
||||
"detailConvert": "Interessent umwandeln",
|
||||
"listForAccount": "Liste (für Firma)"
|
||||
}
|
||||
}
|
||||
"layouts": {
|
||||
"detailConvert": "Interessent umwandeln",
|
||||
"listForAccount": "Liste (für Firma)"
|
||||
}
|
||||
}
|
||||
@@ -1,13 +1,18 @@
|
||||
{
|
||||
"modes": {
|
||||
"month": "Monat",
|
||||
"week": "Woche",
|
||||
"day": "Tag",
|
||||
"agendaWeek": "Woche",
|
||||
"agendaDay": "Tag"
|
||||
},
|
||||
"labels": {
|
||||
"Today": "Heute",
|
||||
"Create": "Erstellen"
|
||||
}
|
||||
}
|
||||
"modes": {
|
||||
"month": "Monat",
|
||||
"week": "Woche",
|
||||
"agendaWeek": "Woche",
|
||||
"day": "Tag",
|
||||
"agendaDay": "Tag",
|
||||
"timeline": "Zeitachse"
|
||||
},
|
||||
"labels": {
|
||||
"Today": "Heute",
|
||||
"Create": "Erstellen",
|
||||
"Shared": "Gemeinsam",
|
||||
"Add User": "Benutzer hinzufügen",
|
||||
"current": "aktuell",
|
||||
"time": "Zeit"
|
||||
}
|
||||
}
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user