Compare commits

...

263 Commits
4.1.2 ... 4.2.3

Author SHA1 Message Date
yuri
f896a2d71a fix calendar range issue 2016-08-18 11:13:24 +03:00
yuri
dd6704ace5 fix moving to trash 2016-08-18 10:40:16 +03:00
yuri
1dc4d44a65 es_ES lang fix 2016-08-17 17:52:22 +03:00
yuri
bfb28ea178 Merge branch 'hotfix/4.2.3' of ssh://172.20.0.1/var/git/espo/backend into hotfix/4.2.3 2016-08-17 17:43:43 +03:00
yuri
ec6f3a22f2 version 2016-08-17 17:30:34 +03:00
yuri
5a0c7c330c fix email imported error in php4 2016-08-17 17:28:17 +03:00
Taras Machyshyn
f64df5af87 LDAP fixed admin login 2016-08-17 11:54:59 +03:00
Taras Machyshyn
d4dc7a4051 LDAP fixes 2016-08-16 16:53:40 +03:00
Taras Machyshyn
d676f85c8f LDAP fixes 2016-08-16 16:01:32 +03:00
Taras Machyshyn
ab382f2387 LDAP: changed ldapUserObjectClass attribute. 2016-08-15 16:46:55 +03:00
yuri
206219c738 Merge branch 'hotfix/4.2.2' of ssh://172.20.0.1/var/git/espo/backend into hotfix/4.2.2 2016-08-15 15:51:40 +03:00
Taras Machyshyn
37d1c707cb LDAP: label corrections 2016-08-15 15:51:10 +03:00
Taras Machyshyn
93af1c9bfc LDAP improvements: added possibility to define user objectClass 2016-08-15 15:47:25 +03:00
yuri
a021c4c8d5 open attachments in new window 2016-08-15 11:08:45 +03:00
yuri
a125244cdf Merge branch 'hotfix/4.2.2' of ssh://172.20.0.1/var/git/espo/backend into hotfix/4.2.2 2016-08-15 10:50:10 +03:00
yuri
1cfd251c4c copy attachments for duplicate 2016-08-12 12:51:02 +03:00
Taras Machyshyn
d2f4f312e5 Improvements 2016-08-11 16:21:35 +03:00
Taras Machyshyn
c468b061d9 Bug fixes for installation 2016-08-11 16:21:03 +03:00
yuri
7bf945f0b6 v 2016-08-11 15:31:19 +03:00
yuri
fecbb26cbf email acl fix 2016-08-11 15:30:47 +03:00
yuri
a5ae33ab81 fix dateTime exception 2016-08-11 13:01:25 +03:00
yuri
c6fa0e464e prevent stream request after remove record 2016-08-11 12:49:56 +03:00
yuri
38bae6238a portal: disable teams field 2016-08-11 12:28:14 +03:00
yuri
79de4c874f portal: follow created 2016-08-11 12:16:51 +03:00
yuri
814748ec61 fix map 2016-08-11 11:48:46 +03:00
yuri
1f0ad0cbec fix inbound email duplicate 2016-08-11 11:12:36 +03:00
yuri
7224f566d6 email import: bad date catch 2016-08-10 16:50:22 +03:00
yuri
eefb01ec4f fix salesByMonth 2016-08-10 16:22:17 +03:00
yuri
24d46ed81d fix currency rates 2016-08-10 15:51:20 +03:00
yuri
37c749faf8 remove messageIdInternal index 2016-08-10 15:34:23 +03:00
yuri
4306a3131e fix inbound email ui 2016-08-10 15:32:59 +03:00
yuri
8fa95fcce3 email: fix whitespace subject 2016-08-09 11:12:30 +03:00
yuri
b7c41ce640 email fix 2016-08-08 13:09:51 +03:00
yuri
a21be94ed3 fix acl 2016-08-08 13:06:40 +03:00
yuri
9d59edcae2 v 2016-08-08 11:51:54 +03:00
yuri
deaa26a355 es_ES lang fix 2016-08-08 10:29:49 +03:00
yuri
25d6fb6d82 theme fixes 2016-08-08 10:18:06 +03:00
yuri
c1bcc44f04 fix error 2016-08-08 10:03:00 +03:00
yuri
fdf8183385 fix massRemove acl check 2016-08-08 09:57:06 +03:00
yuri
db40426f00 lang fix 2016-08-05 11:00:24 +03:00
yuri
eb92e648a0 dashlet action url 2016-08-04 12:52:37 +03:00
yuri
5df26c324f fix scheduled job running 2016-08-04 12:30:26 +03:00
yuri
c36a064fdc lang 2016-08-04 11:01:39 +03:00
yuri
a28baa6a75 fix install footer date 2016-08-03 17:29:19 +03:00
yuri
27aed29ddf Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2016-08-03 17:26:43 +03:00
yuri
744fb176cc fix code to meet standards 2016-08-03 17:25:29 +03:00
Taras Machyshyn
5f8a0736e4 LDAP corrections 2016-08-03 16:25:33 +03:00
yuri
7fb179e769 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2016-08-03 16:09:55 +03:00
Taras Machyshyn
48239c53ca Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2016-08-03 12:41:50 +03:00
Taras Machyshyn
4e18afb80f LDAP: added 'Test Connection' button 2016-08-03 12:41:33 +03:00
yuri
e9d3d7c807 fix date stringify 2016-08-03 12:23:58 +03:00
Taras Machyshyn
60923197e5 Ldap: added user teams, user default team 2016-08-03 10:56:08 +03:00
Taras Machyshyn
a9f7c90323 Improvements 2016-08-03 10:37:48 +03:00
yuri
cc723095c2 fix file manager 2016-08-02 15:04:53 +03:00
yuri
373b77f83f fix tests 2016-08-02 14:57:47 +03:00
Taras Machyshyn
68ab589f3e Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2016-08-02 13:09:26 +03:00
Taras Machyshyn
23ccbb226f Test fixes 2016-08-02 13:09:14 +03:00
yuri
48edf2a2b5 selectManager test 2016-08-02 13:03:56 +03:00
yuri
082c65ef05 fix unit test 2016-08-02 10:54:12 +03:00
yuri
6553d8ec6c Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2016-08-01 17:45:07 +03:00
yuri
c54b6fcc7c test addition 2016-08-01 17:44:43 +03:00
yuri
655ee740e0 importer test 2016-08-01 17:40:47 +03:00
Taras Machyshyn
b02f77b8f9 LDAP improvements 2016-08-01 16:29:26 +03:00
yuri
351c46af06 fix kb order 2016-08-01 15:36:24 +03:00
yuri
d3db25d98a fix portal user password email 2016-08-01 15:24:01 +03:00
yuri
ef1fe1bd1d Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2016-08-01 11:12:51 +03:00
Taras Machyshyn
6c2cd93826 Default config 2016-08-01 11:12:26 +03:00
yuri
4859f54f42 email import: get rid of file manager 2016-08-01 10:16:09 +03:00
Taras Machyshyn
749c2dc1e9 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2016-07-29 17:54:02 +03:00
Taras Machyshyn
3d13026084 LDAP fixes 2016-07-29 17:53:49 +03:00
yuri
6cf66aa88e account: recently created filter 2016-07-29 11:51:23 +03:00
yuri
d10ef7038b meetings and assignmentPermissions 2016-07-29 11:41:37 +03:00
yuri
cda612810d Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2016-07-28 11:31:42 +03:00
yuri
a7aaac513d composer.lock 2016-07-28 11:30:57 +03:00
yuri
329dbdf408 it_IT lang 2016-07-28 11:28:47 +03:00
Taras Machyshyn
007f705904 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2016-07-27 16:53:24 +03:00
Taras Machyshyn
36cddbe0cf Fixed LDAP authorization 2016-07-27 16:53:11 +03:00
yuri
e9409ccf72 remove tab 2016-07-27 16:44:46 +03:00
yuri
b5a1ede962 Merge branch 'hotfix/4.1.7' 2016-07-27 11:26:28 +03:00
yuri
2dc6951ef9 v 2016-07-27 11:26:07 +03:00
yuri
66f686c013 fix kb order 2016-07-27 10:56:20 +03:00
yuri
937aab2b1c massAction successMessage 2016-07-26 15:31:49 +03:00
yuri
43bee97055 massAction fix 2016-07-26 12:22:36 +03:00
yuri
a31f5ea87a email address book filter by portal permissions 2016-07-26 12:06:50 +03:00
yuri
0c4d5f3405 mass action defs 2016-07-26 11:31:07 +03:00
yuri
4e70ea0586 fix email 2016-07-25 17:13:34 +03:00
yuri
7736f6cd76 fix email update 2016-07-25 17:07:14 +03:00
yuri
2db6eaa344 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2016-07-25 14:48:08 +03:00
Taras Machyshyn
38b6cbda0e Merge pull request #159 from ecm4u/master
Support AD with LDAP Auth
2016-07-25 14:47:36 +03:00
Taras Machyshyn
9292f18b5b WARNING fixes 2016-07-25 12:35:52 +03:00
Taras Machyshyn
193cf2438a Cron manager bug fixes 2016-07-25 12:31:27 +03:00
yuri
155e0e3841 import: allow created at and created by 2016-07-25 12:01:03 +03:00
yuri
019c03ed2f email-to-task name 2016-07-25 11:24:23 +03:00
yuri
316e54df92 fix message id issue 2016-07-25 11:17:00 +03:00
yuri
40691ae899 fix entity manager text filter list 2016-07-25 11:16:53 +03:00
yuri
ab8ba168ac move kb articles 2016-07-22 14:50:09 +03:00
yuri
b066d91cf5 fix warning 2016-07-21 15:24:01 +03:00
yuri
2833581cde email folders side panel: remove edit link 2016-07-21 13:28:53 +03:00
yuri
279c8b3188 email folder and email filter email link 2016-07-21 13:24:20 +03:00
yuri
eca6408415 prevent changing users email address and phone numbers for non admins 2016-07-21 13:07:13 +03:00
yuri
35cfb1a480 scheduled job mass actions 2016-07-21 10:58:09 +03:00
yuri
30f3d4ab5e stream notifications 2016-07-20 16:40:21 +03:00
yuri
7e5424e40a activities and history listen to save 2016-07-19 12:23:46 +03:00
Heiko Robert
82612326bf fixed wrong method syntax 2016-07-18 09:04:41 +02:00
Heiko Robert
da2e5c835a removed unnecessary comments 2016-07-14 15:06:31 +02:00
Heiko Robert
af8eb51c76 fix: setting ldap user fields array after constructor in createUser to support getConfig() 2016-07-14 14:51:26 +02:00
Heiko Robert
7253e257f1 moved hard coded ldap config to conf.php, added default config for Active Directory (cn --> samaccountname) 2016-07-13 17:50:43 +02:00
Heiko Robert
37e93edf25 Merge pull request #2 from espocrm/master
Update from original
2016-07-13 17:40:48 +02:00
yuri
36a56f050f multiple smtp 2016-07-08 16:25:27 +03:00
yuri
31c2d1360d fix warnings 2016-07-08 11:18:28 +03:00
yuri
4addc48339 fix import 2016-07-08 11:00:19 +03:00
yuri
a7267bc920 fix import of noJoin relations 2016-07-08 10:49:06 +03:00
yuri
93e94f40c3 fix move to folder 2016-07-07 17:36:58 +03:00
yuri
bd75790c29 change email list layout 2016-07-07 17:26:43 +03:00
yuri
f1cffcae38 email actions 2016-07-07 17:08:05 +03:00
yuri
9111bce47a move to folder email 2016-07-07 16:30:08 +03:00
yuri
73ec161ac4 fix RDB findRelated if new 2016-07-07 15:55:53 +03:00
yuri
495a07639e email folders dev 2016-07-07 15:55:37 +03:00
yuri
044223e70e fix RDB findRelated if new 2016-07-07 15:27:53 +03:00
yuri
42c9250995 email folders url 2016-07-07 12:07:51 +03:00
yuri
8b4a1d96fd replyInHtml 2016-07-07 11:32:42 +03:00
yuri
fff06d7e92 email folder dev 2016-07-06 17:14:08 +03:00
yuri
726adc550c email folders 2016-07-06 16:38:29 +03:00
yuri
f5bde670f6 fix mapper fatal error 2016-07-06 10:26:22 +03:00
yuri
050873dd3a Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2016-07-06 10:12:50 +03:00
yuri
d3886c1c94 dev 2016-07-06 10:12:40 +03:00
yuri
84f03cf3f9 email improvements 2 2016-07-05 12:41:57 +03:00
yuri
f8953d56f4 fix link multiple with role 2016-07-05 10:23:01 +03:00
yuri
ebbf14160b email folder changes 2016-07-04 17:29:46 +03:00
yuri
4df0a09de0 lang 2016-07-04 10:22:58 +03:00
yuri
c485aa9a62 email folder entity 2016-07-01 17:18:32 +03:00
yuri
ae474022b9 email filters skip 2016-07-01 15:23:41 +03:00
yuri
f994cdf7c4 email filter changes 2016-06-30 16:56:04 +03:00
yuri
7ff514e5a7 email sentBy field 2016-06-30 11:48:53 +03:00
yuri
344403de54 fix email sent filter 2016-06-30 11:18:15 +03:00
yuri
836243d170 fix email sent filter 2016-06-30 11:17:31 +03:00
yuri
0dc9129d79 notification improvements 2 2016-06-29 16:54:23 +03:00
yuri
61c9d07ad8 notifications improvements 2016-06-29 16:50:01 +03:00
yuri
a238295ba8 control followers after reassignment 2016-06-29 16:23:30 +03:00
yuri
dbabd0dd45 Merge branch 'hotfix/4.1.6' 2016-06-29 16:03:10 +03:00
yuri
f87ff8db8a fix 2016-06-29 15:38:21 +03:00
yuri
60d121f1e1 portal acl fixes 2016-06-29 15:17:52 +03:00
yuri
1497abff07 v 2016-06-29 13:12:09 +03:00
yuri
935ba7d4e4 fix 2016-06-29 13:10:36 +03:00
yuri
fba79cde8f notification chanfes 2016-06-29 11:51:30 +03:00
yuri
ce79d14a29 duplicate by default 2016-06-28 15:53:18 +03:00
yuri
14b2083226 google maps key 2016-06-28 15:28:50 +03:00
yuri
779a841817 kba ordering 2016-06-28 11:56:29 +03:00
yuri
3647523150 naming fix 2016-06-28 10:59:32 +03:00
yuri
eb5b941fe6 email accounts massUpdate 2016-06-28 10:56:25 +03:00
yuri
c542d99536 fix stream header long texts 2016-06-27 10:50:48 +03:00
yuri
918031a9c6 fix long texts in stream 2016-06-27 10:49:00 +03:00
yuri
163e0cac2a drag and drop additional 2016-06-24 17:08:49 +03:00
Taras Machyshyn
95096dd8fd Correct translations for installer 2016-06-24 15:28:28 +03:00
yuri
67bd98bf4b iso week numbers 2016-06-24 15:22:03 +03:00
yuri
ac434c00d6 stream drag and drop 2016-06-24 15:04:54 +03:00
yuri
0fb8330ecf preferenses: hide notifications if not enabled 2016-06-24 12:36:55 +03:00
yuri
3f969cee5b Merge branch 'hotfix/4.1.6' 2016-06-24 12:15:59 +03:00
yuri
ea5f6300f3 fix text overflow in list expanded 2016-06-24 12:15:51 +03:00
yuri
0d5e75db23 activities dashlet: show only todays and next day 2016-06-23 17:18:49 +03:00
yuri
7d667a1d1b notification about mention 2016-06-23 16:18:32 +03:00
yuri
610922c8e0 fix client notification 2016-06-23 15:20:17 +03:00
yuri
36e0882542 rename method 2016-06-23 11:44:13 +03:00
yuri
b9979e87af email notifications: dont notify old emails 2016-06-23 11:21:38 +03:00
yuri
bcd38dd853 fix email filters 2016-06-23 10:52:22 +03:00
yuri
b018580c0b fix email reminder 2016-06-23 10:36:08 +03:00
yuri
38a7f42a1e remove ob_clean and flush 2016-06-22 16:26:52 +03:00
yuri
1387d856ba remove ob_clean from installer 2016-06-22 16:20:11 +03:00
yuri
f2bbc872d6 lang fixes 2016-06-22 16:00:45 +03:00
yuri
c74e7b416e lang 2016-06-22 15:03:33 +03:00
yuri
f008688c14 orm: fix order by list 2016-06-22 12:14:15 +03:00
yuri
dff0d5a992 fix email notification from person 2016-06-22 12:01:38 +03:00
yuri
6fe6f8960f fix stream note.js 2016-06-22 10:28:25 +03:00
yuri
0b0184098d fix gender in stream 2016-06-21 17:25:18 +03:00
yuri
647515d21a gender support 2016-06-21 17:21:45 +03:00
yuri
2ee944ef7b change assignment templates 2016-06-21 12:23:51 +03:00
yuri
f4d2325b8b reminder and invitations change 2016-06-21 12:08:55 +03:00
yuri
c0904125eb notification changes and stream message changes 2016-06-20 17:35:57 +03:00
yuri
7dad30ad0c fix notices 2016-06-20 11:24:15 +03:00
yuri
d6acb6dfdc fix notice 2016-06-20 11:06:26 +03:00
yuri
b6da94fe2e fix modal backdrop close 2016-06-17 11:54:09 +03:00
yuri
dc5e292a02 fix modal backdrop close 2016-06-17 11:53:30 +03:00
yuri
4bf938107d portal id 2016-06-16 16:59:10 +03:00
yuri
ec7d49cdbc fix validate message if empty element 2016-06-15 17:20:19 +03:00
yuri
3c73062b91 email: is replied 2016-06-15 16:38:01 +03:00
yuri
5122f112d1 task dashlet: dont show if date start is future 2016-06-15 12:43:47 +03:00
yuri
5e20fa6717 orm: skip text fields param 2016-06-15 11:41:52 +03:00
yuri
d4599d9377 fix mapper test 2016-06-15 11:40:26 +03:00
yuri
adb9ce4d7e email: rename name label 2016-06-15 11:17:16 +03:00
yuri
e777413a7b enum order 2016-06-15 11:15:02 +03:00
yuri
2cb0ac6221 email: forward info 2016-06-14 17:30:24 +03:00
yuri
8f7fafb990 email import: fetch parent from replied 2016-06-14 16:37:54 +03:00
yuri
158c911787 fix label 2016-06-14 13:01:58 +03:00
yuri
aa67575ecb internal post dev 2016-06-14 12:56:18 +03:00
yuri
81cf82c99f fix stream posting 2016-06-14 11:56:42 +03:00
yuri
e638bf2eec Merge branch 'hotfix/4.1.6' 2016-06-14 11:44:55 +03:00
yuri
6c9d1dbb3d fix notification sounds 2016-06-14 11:43:31 +03:00
yuri
fbb034ef92 intenal post dev 2016-06-14 11:24:42 +03:00
yuri
335601d6b8 fix multienum validate message 2016-06-13 15:57:37 +03:00
yuri
f6ad51ca74 fix calendar colors 2016-06-13 15:30:16 +03:00
yuri
e119f8b008 fix entity manager tabList 2016-06-13 15:23:29 +03:00
yuri
5af0eeff3b entity manager: event type 2016-06-13 13:12:41 +03:00
yuri
f685feb312 Merge branch 'hotfix/4.1.6' 2016-06-13 10:56:38 +03:00
yuri
c78254a8fa no dashboard layout for portal useres 2016-06-13 10:52:20 +03:00
yuri
9c3758b92c Merge branch 'stable' 2016-06-10 16:10:48 +03:00
yuri
4ca2a7fa1a fix notice 2016-06-10 16:03:10 +03:00
yuri
d6814b1601 Merge branch 'hotfix/4.1.5' 2016-06-10 15:55:57 +03:00
yuri
55e5a21dcd Merge branch 'hotfix/4.1.5' of ssh://172.20.0.1/var/git/espo/backend into hotfix/4.1.5 2016-06-10 15:54:14 +03:00
Taras Machyshyn
792f422f76 Bug fixing 2016-06-10 15:53:37 +03:00
yuri
8cac546087 merge with hotfix/4.1.5 2016-06-10 15:05:41 +03:00
yuri
0cfe91d960 loadAdditionalFieldsForPdf 2016-06-10 14:44:09 +03:00
yuri
4dea1762e8 stream textarea fix 2016-06-10 13:08:22 +03:00
Taras Machyshyn
b1a9b8d8b1 Code improvement 2016-06-10 10:57:56 +03:00
yuri
6787acce61 fix next number 2016-06-10 10:50:40 +03:00
yuri
eb0adf7c28 fix afterMassRemove 2016-06-09 14:03:23 +03:00
yuri
783cccaa1b Number field type 2016-06-09 12:47:17 +03:00
yuri
36a0a22996 naming fix 2016-06-09 10:49:56 +03:00
yuri
a3f3357b2e Merge branch 'hotfix/4.1.5' 2016-06-08 17:58:29 +03:00
yuri
cbbfa44174 fix field manager ui 2016-06-08 17:58:18 +03:00
yuri
812d2ec9bc fix field manager hook 2016-06-08 17:52:20 +03:00
yuri
b51ba96bb7 number field type prepare 2016-06-08 17:38:12 +03:00
yuri
4c60cf79f6 rdb: rename methods 2016-06-08 15:56:04 +03:00
yuri
8451949805 database charset param 2016-06-08 11:47:52 +03:00
yuri
af5750cdf6 color change 2016-06-07 19:00:57 +03:00
yuri
1a4dfd6f67 version 2016-06-07 12:29:29 +03:00
yuri
6aa800d453 fix fetchOnModelAfterRelate 2016-06-07 12:26:20 +03:00
yuri
82aa3b9508 fix massRelate 2016-06-07 12:15:04 +03:00
yuri
6910a113da email filter: filters 2016-06-06 16:49:07 +03:00
yuri
bd20aaa577 fix htmlizer 2016-06-06 12:55:04 +03:00
yuri
0aef3c0b04 htmlizer and password 2016-06-06 12:53:10 +03:00
yuri
2f97010b54 htmlizer improvements 2016-06-06 12:31:26 +03:00
yuri
5bcdad2996 wysiwyg htmlToPlain fix 2016-06-06 11:59:49 +03:00
yuri
1e8a8d94c2 fix timeline timezone 2016-06-06 11:51:21 +03:00
yuri
9aeefd7685 contact: fix accounts field 2016-06-03 11:29:46 +03:00
yuri
f72385471c naming fix 2016-06-03 11:27:41 +03:00
yuri
f7a542560a check duplicates: all email 2016-06-03 11:19:03 +03:00
yuri
400f43447d v 2016-06-02 17:22:15 +03:00
yuri
866593a831 Storage Message: define ErrorHandler 2016-06-02 17:15:30 +03:00
yuri
c90cdc62fb global search changes 2016-06-02 12:39:06 +03:00
yuri
d5c93f21b5 audited for miltiEnum and arrays 2016-06-02 12:26:47 +03:00
yuri
b58d78a29e preferences: dashboardLayout 2016-05-31 12:05:37 +03:00
yuri
f5b41eb78b field manager: translatedOptions param 2016-05-31 11:01:17 +03:00
yuri
c2a7d90944 fix view-helper options 2016-05-27 13:17:51 +03:00
yuri
b2a4ec238c fix preferences fields 2016-05-27 12:11:17 +03:00
yuri
df69584c7d v 2016-05-27 11:46:31 +03:00
yuri
574da55be7 Merge branch 'hotfix/4.1.3' of ssh://172.20.0.1/var/git/espo/backend into hotfix/4.1.3 2016-05-26 17:15:40 +03:00
Taras Machyshyn
52b808b902 Removed unnecessary debugging messages 2016-05-26 16:53:52 +03:00
Taras Machyshyn
714c7b0a33 Fixed an issue for MySQL 5.7 2016-05-26 16:41:50 +03:00
yuri
3d774e3afa fix wysywyg modals 2016-05-26 16:26:16 +03:00
yuri
c91db1699b upgrade bootstrap 2016-05-26 15:39:04 +03:00
yuri
2b5695d8dc fix wysywig field detail view height 2016-05-26 12:23:52 +03:00
yuri
b562fc33bd dont allow email to modify 2016-05-26 11:43:52 +03:00
yuri
425414b8f3 fix add email address 2016-05-26 11:24:21 +03:00
yuri
3b363d5ee3 cleanup 2016-05-26 11:10:34 +03:00
yuri
4fc3b9a99e fix kb massUpdate layout 2016-05-25 15:23:12 +03:00
yuri
ab2b9bafeb fix meeting/call select manager 2016-05-24 15:53:09 +03:00
yuri
e99fc6302f reminders for all users 2016-05-24 12:35:46 +03:00
yuri
82996a807f calendar: getCalendarSelectParams method 2016-05-24 11:38:14 +03:00
yuri
86b31b1f26 clear role cach if isAdmin changed 2016-05-23 17:25:58 +03:00
yuri
19c9b38f82 applicationName in settings 2016-05-23 12:24:18 +03:00
yuri
b1d0d1cd27 activities: status not required 2016-05-23 11:47:04 +03:00
yuri
c916011530 add id_ID language 2016-05-23 11:36:58 +03:00
yuri
5db6327272 de_DE 2016-05-23 11:10:03 +03:00
700 changed files with 21538 additions and 6626 deletions

View File

@@ -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;

View File

@@ -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;
}
}

View File

@@ -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;
}
}

View File

@@ -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;

View File

@@ -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;

View File

@@ -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;

View File

@@ -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;

View File

@@ -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')

View File

@@ -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']);
}
}

View 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();
}
}

View File

@@ -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 {

View File

@@ -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();
}

View File

@@ -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');

View File

@@ -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;
}
}

View File

@@ -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);
}
}

View File

@@ -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;

View File

@@ -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;

View File

@@ -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');
}
}

View File

@@ -127,5 +127,9 @@ class Table extends \Espo\Core\Acl\Table
}
}
}
protected function applyAdditional(&$table, &$fieldTable, &$valuePermissionLists)
{
}
}

View File

@@ -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');

View File

@@ -295,7 +295,8 @@ class Container
{
return new \Espo\Core\Utils\FieldManager(
$this->get('metadata'),
$this->get('language')
$this->get('language'),
$this
);
}

View File

@@ -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']);
}
}

View File

@@ -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'],

View File

@@ -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.");
}
/**

View File

@@ -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()

View File

@@ -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);

View File

@@ -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);
}

View 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;
}
}

View File

@@ -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',

View File

@@ -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');

View File

@@ -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;

View File

@@ -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
{

View File

@@ -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;
}
}

View File

@@ -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;

View File

@@ -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)

View File

@@ -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;

View File

@@ -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();
}
}

View File

@@ -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');
}
}

View File

@@ -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();
}

View File

@@ -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()

View File

@@ -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)

View File

@@ -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;

View 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
{
}

View 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
{
}

View 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
}
]
]
}
]

View File

@@ -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
}
]
]
}
]

View File

@@ -1,4 +1,4 @@
{
"controller": "Controllers.Record",
"controller": "controllers/record",
"boolFilterList": ["onlyMy"]
}

View File

@@ -1,6 +1,6 @@
{
"controller": "Controllers.RecordTree",
"collection": "Collections.Tree",
"controller": "controllers/record-tree",
"collection": "collections/tree",
"menu": {
"listTree": {
"buttons": [

View File

@@ -0,0 +1,4 @@
{
"controller": "controllers/record",
"boolFilterList": ["onlyMy"]
}

View 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"]
}
}
}

View File

@@ -0,0 +1,11 @@
{
"entity": true,
"layouts": true,
"tab": true,
"acl": true,
"aclPortal": true,
"customizable": true,
"importable": true,
"calendar": true,
"notifications": true
}

View File

@@ -1,4 +1,4 @@
{
"controller": "Controllers.Record",
"controller": "controllers/record",
"boolFilterList": ["onlyMy"]
}

View File

@@ -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

View 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
{
}

View 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
{
}

View File

@@ -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;
}
}

View File

@@ -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;
}
}

View 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
{
}

View File

@@ -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;
}
}

View File

@@ -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;
}

View File

@@ -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);

View File

@@ -316,6 +316,11 @@ class Config
return true;
}
public function getSiteUrl()
{
return rtrim($this->get('siteUrl'), '/');
}
}
?>

View File

@@ -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();

View File

@@ -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;
}
}

View File

@@ -116,6 +116,11 @@ class DateTime
return null;
}
public function setTimezone($timezone)
{
$this->timezone = new \DateTimeZone($timezone);
}
}

View 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;
}
}
}
}

View File

@@ -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;
}
}

View 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');
}
}

View File

@@ -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);
}
}

View File

@@ -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;
}

View File

@@ -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());
}

View File

@@ -28,6 +28,7 @@
************************************************************************/
namespace Espo\Core\Utils;
use Espo\Core\Exceptions\Error;
class Metadata

View File

@@ -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);
}
}
}

View File

@@ -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,
);

View File

@@ -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',
);

View File

@@ -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)) {

View 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
{
}

View 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
{
}

View File

@@ -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;
}

View File

@@ -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;
}

View File

@@ -143,8 +143,6 @@ class Image extends \Espo\Core\EntryPoints\Base
if ($fileSize) {
header('Content-Length: ' . $fileSize);
}
ob_clean();
flush();
readfile($filePath);
exit;
}

View File

@@ -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();
}

View 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);
}
}
}
}

View File

@@ -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)) {

View File

@@ -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);
}
}
}

View 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();
}
}
}

View File

@@ -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;
}
}

View File

@@ -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');

View 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();
}
}

View File

@@ -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);

View File

@@ -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);

View File

@@ -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;
}
}

View File

@@ -91,7 +91,7 @@ class CampaignTrackOpened extends \Espo\Core\EntryPoints\Base
imagepng($img);
imagecolordeallocate($background);
imagedestroy( $tt_image );
imagedestroy($img);
}
}

View File

@@ -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();
}

View File

@@ -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);
}
}
}

View File

@@ -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);
}
}
}
}

View File

@@ -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()

View File

@@ -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"
}
}

View File

@@ -1,6 +1,6 @@
{
"layouts": {
"detailConvert": "Interessent umwandeln",
"listForAccount": "Liste (für Firma)"
}
}
"layouts": {
"detailConvert": "Interessent umwandeln",
"listForAccount": "Liste (für Firma)"
}
}

View File

@@ -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