Compare commits

..

702 Commits
5.6.2 ... 5.7.7

Author SHA1 Message Date
Yuri Kuznetsov
b1f272215a email index skip mysql 5.7 2019-11-12 11:08:46 +02:00
Taras Machyshyn
032199d9c7 Save database type and version while rebuild 2019-11-12 10:56:12 +02:00
Yuri Kuznetsov
cce78ac258 fix metadata and language acl 2019-11-11 10:22:22 +02:00
Yuri Kuznetsov
67074f2b0a dashlet options button 2019-11-06 15:29:00 +02:00
Yuri Kuznetsov
ea3189ebc3 metadata & language acl dependencies 2019-11-04 14:04:10 +02:00
Yuri Kuznetsov
83e7e0b517 fix link multiple with role acl 2019-11-04 13:36:12 +02:00
Yuri Kuznetsov
2c1fed8adf fix email header 2019-11-04 10:33:42 +02:00
Yuri Kuznetsov
45e6f3bb34 array field: display as list param 2019-11-01 16:46:27 +02:00
Yuri Kuznetsov
f53708c38e remove no joins 2019-11-01 14:34:57 +02:00
Yuri Kuznetsov
d1aa3f721d no full text threshold for emails 2019-11-01 13:16:08 +02:00
Yuri Kuznetsov
0afe715d9a full text search threshold 2019-11-01 13:08:40 +02:00
Yuri Kuznetsov
9cbe31300a lock user table before checking duplicate 2019-11-01 11:59:33 +02:00
Yuri Kuznetsov
dde139f0a1 full text changes 2019-10-31 16:53:26 +02:00
Yuri Kuznetsov
5fb8b2c71c orm match test 2019-10-31 15:56:21 +02:00
Yuri Kuznetsov
63c07a1f17 fulltext search improvements 2019-10-31 15:42:08 +02:00
Yuri Kuznetsov
a338bbca59 integration language 2019-10-31 13:38:34 +02:00
Yuri Kuznetsov
157bf87d40 ouathcallback message 2019-10-31 12:49:53 +02:00
Yuri Kuznetsov
911a9f80de bool search change 2019-10-29 15:56:19 +02:00
Yuri Kuznetsov
c9bbbeb2c4 fix history email list 2019-10-29 12:03:52 +02:00
Yuri Kuznetsov
57803c0828 v 2019-10-28 12:24:03 +02:00
Yuri Kuznetsov
3d8be61fbf amend 2019-10-28 12:05:33 +02:00
Yuri Kuznetsov
690e87a3fb select manager error if order by non existing field 2019-10-28 12:05:00 +02:00
Yuri Kuznetsov
cd45f07e54 fix account description 2019-10-24 14:34:46 +03:00
Yuri Kuznetsov
a35f9625af fix email reply id 2019-10-24 13:33:07 +03:00
Yuri Kuznetsov
f5245ef3eb email import message id fix 2019-10-24 13:06:11 +03:00
yuri
4ad9ef770b pdf: not latin filename support 2019-10-23 11:05:26 +03:00
yuri
acbc74d858 fix entity plural label 2019-10-22 15:42:41 +03:00
yuri
8da3350523 base plus emails link 2019-10-22 15:40:29 +03:00
yuri
496879dd79 pdf link support 2019-10-22 15:30:49 +03:00
yuri
84cda80fd8 forbid base entity type name 2019-10-22 13:13:13 +03:00
yuri
50a81b1247 date picker disable keyboard navigation 2019-10-22 12:56:57 +03:00
yuri
f0f225f349 propagate remove event for stored main view 2019-10-22 12:25:40 +03:00
yuri
d49fff1289 cleanup 2019-10-22 11:44:26 +03:00
yuri
abaa1b302b email filters from to 2019-10-22 11:38:44 +03:00
yuri
0eaa7825ea email filters layout change 2019-10-22 10:40:34 +03:00
yuri
1028b7c5b3 v 2019-10-22 10:32:36 +03:00
yuri
f319e5219d Merge branch 'hotfix/5.7.6' of ssh://172.20.0.1/var/git/espo/backend into hotfix/5.7.6 2019-10-22 10:05:32 +03:00
Taras Machyshyn
0daae5ced5 Integration tests changes 2019-10-21 16:38:46 +03:00
Taras Machyshyn
abee63d269 FieldManager fix 2019-10-21 16:38:15 +03:00
yuri
be33d90986 all of filter 2019-10-21 15:44:41 +03:00
yuri
3a7cac824d formula comments 2019-10-21 13:44:16 +03:00
yuri
7d616b075f fix formula sumRelated 2019-10-21 12:22:05 +03:00
yuri
b718765138 system requirements tpl fix 2019-10-21 11:46:27 +03:00
yuri
3104be7c99 link multiple default 2019-10-21 10:46:29 +03:00
yuri
a8d182b4cd Merge branch 'hotfix/5.7.5' into stable 2019-10-18 16:14:48 +03:00
yuri
3bfc99c88b entity array clone 2019-10-18 14:46:18 +03:00
yuri
ac3300a5cf fix password labels 2019-10-18 12:58:49 +03:00
yuri
636587a22c Merge branch 'hotfix/5.7.5' of ssh://172.20.0.1/var/git/espo/backend into hotfix/5.7.5 2019-10-18 12:48:48 +03:00
Taras Machyshyn
4ed2c71dfd Integration tests fix 2019-10-18 12:48:30 +03:00
yuri
723229d8e6 Merge branch 'hotfix/5.7.5' of ssh://172.20.0.1/var/git/espo/backend into hotfix/5.7.5 2019-10-18 12:12:31 +03:00
yuri
080ab488fa fix json object save 2019-10-18 11:11:55 +03:00
Taras Machyshyn
fd32696c5a Integration tests optimization 2019-10-18 10:40:28 +03:00
Taras Machyshyn
9a56858f12 Label changes 2019-10-18 10:40:20 +03:00
Taras Machyshyn
5923757810 Database indexes changes 2019-10-18 10:40:07 +03:00
yuri
62948a7740 v 2019-10-17 17:21:22 +03:00
yuri
821d57ce00 fix stream 2019-10-17 17:20:47 +03:00
yuri
efd0505ead fix 2019-10-17 16:59:37 +03:00
yuri
91c55965a7 fix 2019-10-17 16:43:16 +03:00
yuri
974305e152 v 2019-10-17 16:42:57 +03:00
yuri
016abaf7b6 fix 2019-10-17 16:16:14 +03:00
yuri
f3ec50dcbd fix 2019-10-17 16:10:39 +03:00
yuri
baeee7caf5 timestamp diff orm functions 2019-10-17 16:07:14 +03:00
yuri
56280224f9 access control disabled 2019-10-17 12:14:47 +03:00
yuri
4b3e341606 fix 2019-10-17 11:38:06 +03:00
yuri
58fad48d5e languageAclDisabled param 2019-10-17 11:03:28 +03:00
yuri
841a73b886 email address lookup fix 2019-10-17 10:17:03 +03:00
yuri
f74ae46ec9 orm additionalSelect 2019-10-16 16:32:00 +03:00
yuri
542d78425c fix email address lookup 2019-10-16 15:39:23 +03:00
yuri
7b241b90a0 fix email search 2019-10-16 11:27:12 +03:00
yuri
de77b1b1d0 v 2019-10-16 10:34:13 +03:00
yuri
8199692df7 calendar ui fixes 2019-10-15 16:38:08 +03:00
yuri
1d3a340d3a duplicate returnUrl 2019-10-15 15:19:53 +03:00
yuri
d664f29388 fix 2019-10-15 14:04:00 +03:00
yuri
4d0e1af000 save and continue action 2019-10-15 14:00:55 +03:00
yuri
97ea0c71e1 fix handle action 2019-10-15 14:00:45 +03:00
yuri
d9fbcda231 lanf 2019-10-15 11:57:28 +03:00
yuri
f4b5cfa5b6 add global search test 2019-10-15 11:43:37 +03:00
yuri
4391c7a7ac fix test 2019-10-15 11:43:27 +03:00
yuri
6c14f390f6 fix global search 2019-10-15 11:31:08 +03:00
yuri
4a6829cf10 fix output 2019-10-15 11:15:45 +03:00
yuri
2b6c493be5 fix link parent 2019-10-14 13:24:42 +03:00
yuri
597406f70d fix test 2019-10-14 13:00:31 +03:00
yuri
110e2fbc37 string pos formula function 2019-10-14 12:32:30 +03:00
yuri
3e1fab487a htmlizer ifInArray 2019-10-14 11:30:50 +03:00
yuri
c76e34fe81 fix enum 2019-10-11 15:49:53 +03:00
yuri
eb922103a8 v 2019-10-11 13:02:04 +03:00
yuri
9b946c6a1f compose email layout columns 2019-10-11 13:01:45 +03:00
yuri
e2df819e57 call allow 0 duration 2019-10-11 11:17:21 +03:00
yuri
9b00d50079 fix select manager user from team 2019-10-11 10:50:25 +03:00
yuri
814f79d53b Merge branch 'hotfix/5.7.2' of github.com:espocrm/espocrm into hotfix/5.7.2 2019-10-10 15:17:42 +03:00
yuri
0d6482374e Merge branch 'hotfix/5.7.2' of ssh://172.20.0.1/var/git/espo/backend into hotfix/5.7.2 2019-10-10 15:06:34 +03:00
Karthik Bhat K
427e3d466a count(): Parameter must be an array or an object that implements Countable (#1460) 2019-10-10 10:13:27 +03:00
yuri
ae72768f8c v 2019-10-09 11:14:28 +03:00
yuri
bd4e92d892 multi-enum display as list 2019-10-09 11:06:34 +03:00
yuri
afc17ac54c fix filter by foreign array 2019-10-09 10:48:21 +03:00
Taras Machyshyn
52dda066cd Index name consists of a list of fields for relation tables 2019-10-09 10:34:26 +03:00
yuri
29bec8fcd6 fix email storing 2019-10-09 10:15:27 +03:00
yuri
5c4fd344e4 smtp default tls 2019-10-08 17:54:12 +03:00
yuri
e936ad907b outbound email settings fixes 2019-10-08 17:50:45 +03:00
yuri
bc59dab9ec fix lang acl 2019-10-08 17:40:17 +03:00
yuri
48652e35ee fix integrations 2019-10-08 17:04:35 +03:00
yuri
381a51b836 Merge branch 'hotfix/5.7.1' of ssh://172.20.0.1/var/git/espo/backend into hotfix/5.7.1 2019-10-08 15:10:15 +03:00
Taras Machyshyn
a9f36a352d Orm: defined index name for relations 2019-10-08 15:07:55 +03:00
yuri
6c8a0bbb66 orm useIndex change 2019-10-08 13:43:31 +03:00
yuri
b21b69eed5 cs fix 2019-10-08 13:43:09 +03:00
yuri
ef0d3febd8 buttons small fix 2019-10-08 11:54:15 +03:00
yuri
2d6db7570c fix record next/prev 2019-10-08 11:44:47 +03:00
yuri
a7804742e7 fix select manager order 2019-10-08 11:18:50 +03:00
yuri
09301e9d7c fix reminders 2019-10-08 10:45:07 +03:00
yuri
5d10da3b78 orm fix 2019-10-07 14:42:37 +03:00
yuri
dcc4cce872 update reserved words 2019-10-07 14:23:53 +03:00
yuri
02fbc20838 orm join only middle 2019-10-07 13:37:59 +03:00
yuri
6641075043 orm fix 2019-10-07 13:22:30 +03:00
yuri
6624b178c0 fix select manager is not linked filter 2019-10-07 13:08:19 +03:00
yuri
4ec6708cb4 v 2019-10-07 12:41:17 +03:00
yuri
f9752eab02 orm more sanitizing 2019-10-07 12:28:10 +03:00
yuri
3de816e03a formula entity get link column 2019-10-07 12:27:42 +03:00
yuri
b854ebab43 lead capture fix 2019-10-07 11:23:41 +03:00
yuri
ffab7c3e6b external account fix 2019-10-07 10:26:47 +03:00
yuri
b6e09ec75a stream fix 2019-10-04 15:24:06 +03:00
yuri
083c768af9 globas search optimization 2019-10-04 12:38:39 +03:00
yuri
fd052c1cdb email select manager fix 2019-10-04 10:35:36 +03:00
yuri
41e0e29500 select manager apply default order 2019-10-03 18:03:57 +03:00
yuri
3cbd918507 full text change 2019-10-03 17:57:07 +03:00
yuri
94c9db2fde full text changes 2019-10-03 17:43:20 +03:00
yuri
da29d6149b Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-10-03 17:29:18 +03:00
Taras Machyshyn
abd887363f Label changes 2019-10-03 17:26:45 +03:00
Taras Machyshyn
8e2fe95e0c Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-10-03 17:11:09 +03:00
Taras Machyshyn
75e5349475 DBAL bug fixes 2019-10-03 17:02:22 +03:00
yuri
9a081d784e fix 2019-10-03 14:56:08 +03:00
yuri
960446f9fa enity manager: disable count 2019-10-03 14:44:10 +03:00
yuri
7cac38f39d Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-10-03 14:30:22 +03:00
yuri
98e69436d9 sum related support children 2019-10-03 13:42:17 +03:00
yuri
415eff0486 fix stream portal user 2019-10-03 13:41:53 +03:00
yuri
850f6ad41e cs fix 2019-10-03 13:12:51 +03:00
yuri
a5433ec82a stream use index 2019-10-03 13:09:54 +03:00
yuri
9b2ac56563 email use index fix 2019-10-03 10:27:29 +03:00
yuri
43fed19d07 formula password functions 2019-10-02 14:58:20 +03:00
yuri
2f93dfd653 array displayAsList 2019-10-02 12:18:27 +03:00
yuri
c752e8fc91 job front end fixes 2019-10-02 11:44:28 +03:00
yuri
00230f50d6 remove notifications sound disabled 2019-10-02 10:46:19 +03:00
yuri
3b880de16d preferences fix 2019-10-02 10:43:42 +03:00
Taras Machyshyn
6acba3efb8 Fixes for 'indexes' attribute for manyMany relationships 2019-10-01 17:17:39 +03:00
yuri
57843f5f86 calendar apply color 2019-10-01 17:13:12 +03:00
yuri
73b00190f7 colorpicker fix 2019-10-01 17:06:32 +03:00
yuri
20ede11bdd css fix 2019-10-01 16:40:03 +03:00
yuri
9d485624a8 fix 2019-10-01 16:00:21 +03:00
yuri
3fe4bc34ef fix 2019-10-01 15:51:02 +03:00
yuri
a38fd487f4 lang fix 2019-10-01 15:43:22 +03:00
yuri
91506a8ca6 unsubscribe without mass email 2019-10-01 14:21:51 +03:00
yuri
d01748a24c hasher 2019-10-01 14:21:38 +03:00
yuri
a3f7f4e2d2 fix smtp account field 2019-10-01 11:19:01 +03:00
yuri
2df476d734 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-10-01 10:16:48 +03:00
yuri
bf5fe492b3 user data fix 2019-10-01 10:10:08 +03:00
Taras Machyshyn
87036f83e1 Added 'indexes' attribute for manyMany relationships 2019-09-30 17:47:55 +03:00
yuri
cf65e40b35 fix case portal acl 2019-09-30 16:20:54 +03:00
yuri
5fe98cbbbc fix validation 2019-09-30 16:19:22 +03:00
yuri
1b94573502 fix acl portal 2019-09-30 16:12:04 +03:00
yuri
d28638136a fix 2019-09-30 16:00:40 +03:00
yuri
ca05495e9e fix metadata filter 2019-09-30 15:48:40 +03:00
yuri
c572d0110f dashboard fix 2019-09-30 15:36:26 +03:00
yuri
3057f749c4 fix language filter 2019-09-30 14:16:04 +03:00
yuri
62c865bdc3 subscription index 2019-09-30 13:28:25 +03:00
yuri
7e2bd60382 filter relationships layout by acl 2019-09-30 12:39:25 +03:00
yuri
84434fa14a acl frontend fix 2019-09-30 12:38:36 +03:00
yuri
1c1b8b3985 restrict settings lists 2019-09-28 15:13:47 +03:00
yuri
ea15452db4 restrict user data and acl 2019-09-28 14:52:39 +03:00
yuri
b39d2b34e2 acl portal additions 2019-09-28 14:52:15 +03:00
yuri
e6d79e0847 dashboard layout fix 2019-09-28 14:14:00 +03:00
yuri
1f334c2690 clear frontend cache on role change 2019-09-28 13:46:46 +03:00
yuri
6cfd5db8ac fix portal 2019-09-28 13:46:32 +03:00
yuri
3478aa772c metadata hide by acl 2019-09-28 13:04:48 +03:00
yuri
f2c769a062 fix acl 2019-09-28 12:23:59 +03:00
yuri
6e1d905db9 team role by default 2019-09-28 12:19:19 +03:00
yuri
882eeffdd5 modal backdrop fix 2019-09-28 12:16:17 +03:00
yuri
1e4e8b3fcf follow access 2019-09-28 12:07:08 +03:00
yuri
4b25456acd fix teams select manager 2019-09-28 11:55:31 +03:00
yuri
68615ba8bb language hide labels by acl 2019-09-28 11:33:20 +03:00
yuri
8bafcded50 portal role scope 2019-09-28 11:20:42 +03:00
Taras Machyshyn
2e7dca6540 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-09-27 16:34:16 +03:00
Taras Machyshyn
d25b4a4e3a Bug fixes 2019-09-27 16:34:08 +03:00
yuri
fbf6b39dae Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-09-27 16:21:39 +03:00
yuri
60bb06c6af dashlets records number change 2019-09-27 16:21:33 +03:00
Taras Machyshyn
a169aca4f0 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-09-27 16:19:02 +03:00
Taras Machyshyn
3e83e97197 Apache / nginx rewrite changes 2019-09-27 16:18:50 +03:00
yuri
70cdbe6020 stream union all 2019-09-27 16:17:22 +03:00
yuri
36d089cfe7 maintainence mode close button 2019-09-27 15:21:50 +03:00
yuri
0ea1be6307 modal buttons 2019-09-27 15:17:53 +03:00
yuri
73cebf7e5f msg change 2019-09-27 15:02:23 +03:00
yuri
bbd943eb93 fix confirm 2019-09-27 14:54:55 +03:00
yuri
2a891c01e7 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-09-27 13:10:38 +03:00
yuri
c6b1bb9c57 header popover touchscreen 2019-09-27 13:09:44 +03:00
yuri
03d06a5919 ui popover 2019-09-27 13:09:28 +03:00
Taras Machyshyn
c02f39cf93 Upgrade: bug fixes 2019-09-27 13:06:22 +03:00
yuri
d064d3ea1c Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-09-27 12:03:20 +03:00
yuri
670cbc4971 maintenance mode message 2019-09-27 12:01:33 +03:00
yuri
2eedc93d27 modal improvements 2019-09-27 12:01:15 +03:00
yuri
8e80153a1a confirm fix 2019-09-27 11:37:40 +03:00
yuri
a80f1f4662 cs fix 2019-09-27 11:12:43 +03:00
yuri
0602074554 fix stream 2019-09-27 11:06:15 +03:00
Taras Machyshyn
8f49f25057 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-09-27 10:36:22 +03:00
Taras Machyshyn
374d827edd Upgrade: bug fixes 2019-09-27 10:36:14 +03:00
yuri
6648e9ba02 stream skip own param 2019-09-27 10:36:03 +03:00
yuri
45773c5fce stream performance fix 2019-09-27 10:26:22 +03:00
yuri
c9336b882c Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-09-26 15:43:18 +03:00
Taras Machyshyn
b39d220f82 Added database version checking for upgrades, extensions 2019-09-26 15:42:38 +03:00
yuri
064d63abed acceptance status modal improvement 2019-09-26 15:40:12 +03:00
yuri
104fc41895 css fix 2019-09-26 12:04:34 +03:00
yuri
b0fc7ffd0a job count disabled 2019-09-26 11:38:11 +03:00
yuri
ce2861b337 Merge branch 'master' of github.com:espocrm/espocrm 2019-09-26 11:29:33 +03:00
yuri
ccb5baf025 drop mysql 5.5 2019-09-26 11:29:22 +03:00
Yuri Kuznetsov
ff227c9813 Update README.md 2019-09-26 11:17:51 +03:00
yuri
a3a60713ef lang fix 2019-09-26 10:46:57 +03:00
yuri
5bae4c7968 small screen list wider 2019-09-26 10:25:06 +03:00
yuri
57e7556424 layout manager link param tooltip 2019-09-25 16:57:14 +03:00
yuri
de7e573dbe link manager label fix 2019-09-25 16:43:37 +03:00
yuri
27579a0111 fix webhook 2019-09-25 16:27:44 +03:00
yuri
9933e2bff2 stream user fix 2019-09-25 12:47:06 +03:00
yuri
061aeaeecc stream service use orm 2019-09-25 12:35:56 +03:00
yuri
07eb8daee2 aclStrictMode system param 2019-09-25 11:46:09 +03:00
yuri
1d0fdbd73b aclStrictMode true if not set 2019-09-25 11:39:54 +03:00
yuri
f9a9f60ad4 small screen footer to the bottom 2019-09-25 11:37:35 +03:00
yuri
21bf0953e6 lang fix 2019-09-25 11:27:34 +03:00
yuri
a6b31c48af close notification panel in mobile mode 2019-09-25 10:51:49 +03:00
yuri
62ac499867 css fix 2019-09-24 17:33:52 +03:00
yuri
d7d0e79e4b cs fix 2019-09-24 17:25:03 +03:00
yuri
4667d62ad5 email store when message id is set 2019-09-24 17:24:05 +03:00
yuri
017ac9c547 email store right after send 2019-09-24 17:20:36 +03:00
yuri
b73db26a41 webhook fix 2019-09-24 16:29:57 +03:00
yuri
3634872a33 import support link multiple ids 2019-09-24 15:55:22 +03:00
yuri
cbe6711a83 scheduled job status default 2019-09-24 12:06:56 +03:00
yuri
bcad4a9e88 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-09-24 11:28:31 +03:00
Taras Machyshyn
4b6e6928c6 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-09-24 11:28:17 +03:00
Taras Machyshyn
fe32fd86c5 Fix upgrade tests 2019-09-24 11:28:08 +03:00
yuri
b704730e18 stream dashlet view label 2019-09-24 11:15:35 +03:00
yuri
11562ff515 fix notice 2019-09-24 11:07:03 +03:00
yuri
bf3abeaa1d Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-09-24 11:04:38 +03:00
yuri
3731a5558b note notification number bigint 2019-09-24 10:42:05 +03:00
yuri
66e15a4404 email address lookup access check 2019-09-24 10:36:10 +03:00
yuri
23dc2f09a2 email address search optimization 2019-09-24 10:33:06 +03:00
yuri
925702516c upgrade strong text warning 2019-09-24 10:26:37 +03:00
yuri
5bcea44ecb fix user activities 2019-09-24 10:18:58 +03:00
Taras Machyshyn
054f3851be Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-09-23 16:52:35 +03:00
Taras Machyshyn
fffa95a206 Improved Upgrade process 2019-09-23 16:50:33 +03:00
yuri
9d9e13a07d Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-09-23 16:08:27 +03:00
yuri
afeff90096 fix activities 2019-09-23 16:06:32 +03:00
yuri
fffb214e5d user history performance 2019-09-23 15:48:07 +03:00
Taras Machyshyn
acb8f95929 Fix file permission after copy 2019-09-23 15:42:36 +03:00
yuri
3323e947b6 relation panel set filter ui improvement 2019-09-23 14:48:38 +03:00
yuri
efc4bf2a89 use history query optimization 2019-09-23 14:20:59 +03:00
yuri
47618a3555 lang 2019-09-23 12:23:01 +03:00
yuri
e89694c8b9 user history no count 2019-09-23 11:13:45 +03:00
yuri
69c6b2db1c scheduled job log perfrormance 2019-09-20 16:53:01 +03:00
yuri
89818b2eea scheduled job log index 2019-09-20 16:33:21 +03:00
yuri
806461750a layout manager button group 2019-09-20 15:13:44 +03:00
yuri
8ac7a855cc merge buttons fix 2019-09-20 15:11:12 +03:00
yuri
e85f7a533f select multiple records button style 2019-09-20 15:09:26 +03:00
yuri
b58561e46f stickButtonsFormBottomSelector 2019-09-20 14:54:41 +03:00
yuri
21909f985b stick buttons z-index 5 2019-09-20 14:49:05 +03:00
yuri
a1c0269443 formula field change 2019-09-19 16:58:42 +03:00
yuri
96bdd54782 fix diff 2019-09-19 15:50:31 +03:00
yuri
8d68305e38 v 2019-09-19 15:19:30 +03:00
yuri
554d105aac update mail-mime-parser 2019-09-19 15:16:24 +03:00
yuri
f3681e8ad7 naming fix 2019-09-19 13:54:11 +03:00
yuri
d088225542 bool filter OR 2019-09-19 13:54:01 +03:00
yuri
e4be1156db followed filter use orm 2019-09-19 12:59:51 +03:00
yuri
76599130cf Merge branch 'stable' 2019-09-19 12:31:35 +03:00
yuri
e9208602f6 css fix 2019-09-19 12:00:03 +03:00
yuri
e2f57d4869 formula record attribute 2019-09-18 16:21:22 +03:00
yuri
98dd4bc51c formula record findOne 2019-09-18 16:20:57 +03:00
yuri
cb4356c689 afterClick hook 2019-09-18 13:11:15 +03:00
yuri
99e8642ef6 fix 2019-09-18 12:59:27 +03:00
yuri
e5575ee673 generate password method fix 2019-09-18 12:34:57 +03:00
yuri
e1d7733848 meeting/call do not show send invitees if only own user and accepted 2019-09-18 11:07:10 +03:00
yuri
b08f7de0cb email address fix double escaping 2019-09-18 10:48:12 +03:00
yuri
d83f02010a fix inbound email from address 2019-09-18 10:26:10 +03:00
yuri
02b66272bb markdown hint fix 2019-09-17 17:42:17 +03:00
yuri
4ef3b2975e Merge branch 'hotfix/5.6.14' 2019-09-17 17:36:52 +03:00
yuri
88bc592fcd Merge branch 'hotfix/5.6.14' of ssh://172.20.0.1/var/git/espo/backend into hotfix/5.6.14 2019-09-17 17:35:54 +03:00
yuri
2fe1d892c5 password strength params 2019-09-17 17:05:07 +03:00
Taras Machyshyn
9c2a125365 Field Manager fix 2019-09-17 17:04:36 +03:00
yuri
d21e61bdf5 2fa labels fixes 2019-09-17 14:00:49 +03:00
yuri
a2e6bb6fd9 fix label 2019-09-17 13:37:39 +03:00
yuri
381f4886a9 passwordGenerateLength param 2019-09-17 13:35:31 +03:00
yuri
ab538e1901 password both cases 2019-09-17 13:19:05 +03:00
yuri
506200901d generate and send password 2019-09-17 12:25:13 +03:00
yuri
764b0c6705 fix generate password 2019-09-17 11:59:11 +03:00
yuri
34209c0811 fix passwordRecoveryEnabled 2019-09-17 11:43:29 +03:00
yuri
0e3b94e1bb confirm return promise 2019-09-17 11:28:15 +03:00
yuri
fd95fa73ce generate password improvement 2019-09-17 11:02:26 +03:00
yuri
3f056c8472 ability to disable password recovery 2019-09-16 15:47:06 +03:00
yuri
ef48999614 hide forgot password if smtp is not configured 2019-09-16 14:04:50 +03:00
yuri
087ab2b954 Merge branch 'hotfix/5.6.14' 2019-09-16 12:33:06 +03:00
yuri
e79e75c0d5 text field remove default rows param 2019-09-16 12:32:14 +03:00
yuri
cb37ee88b2 lang fix 2019-09-16 12:14:23 +03:00
yuri
9ead5dad66 stream performance fix 2019-09-16 12:06:04 +03:00
yuri
4b39626b83 stream onlyNotified param 2019-09-16 11:59:19 +03:00
yuri
2b5b7eb601 varchar: support conditional options dynamic logic 2019-09-16 10:50:57 +03:00
yuri
0785e2a059 Merge branch 'hotfix/5.6.14' 2019-09-16 10:48:40 +03:00
yuri
2187073e2b css fix side navbar tab 2019-09-16 10:45:54 +03:00
yuri
3c36e81502 add ouath-callback to build 2019-09-16 10:34:39 +03:00
yuri
fe6b6bc869 enum fix 2019-09-13 16:35:15 +03:00
yuri
d9cdcfa332 markdown in tooltips 2019-09-13 15:38:23 +03:00
yuri
0db2dbc0f3 field manager custom field tooltip 2019-09-13 14:57:47 +03:00
yuri
e319ead6a7 tooltip ascept custom string 2019-09-13 14:50:33 +03:00
yuri
0dae38943f file accept param 2019-09-13 13:58:14 +03:00
yuri
f9e4ae7b71 orm attribute changed check for json object 2019-09-13 13:26:01 +03:00
yuri
2f4f4dc023 Merge branch 'hotfix/5.6.14' 2019-09-13 13:05:21 +03:00
yuri
f8bafa5221 fix orm order 2019-09-13 12:58:44 +03:00
yuri
00a0cb1394 fix test 2019-09-13 12:58:31 +03:00
yuri
3a37f8421d fix action handler 2019-09-13 12:25:50 +03:00
yuri
2dbde4e2c8 fix detail action label 2019-09-13 11:58:04 +03:00
yuri
98084e9314 formula addLinkMultipleId do not throw error 2019-09-13 11:37:55 +03:00
yuri
2946969e3a detail custom actions 2019-09-13 11:36:01 +03:00
yuri
75c3891654 action history user type 2019-09-12 17:21:51 +03:00
yuri
c0866c8fea fix email template insert 2019-09-12 16:57:19 +03:00
yuri
5d3a785a49 v 2019-09-12 16:41:20 +03:00
yuri
4bcf88dcee fix mail sender 2019-09-12 16:39:32 +03:00
yuri
889f976c5f Merge branch 'stable' 2019-09-12 11:21:51 +03:00
yuri
022d9a42ec transfromMarkdownText 2019-09-12 11:06:57 +03:00
yuri
ac80f67533 fix view helper 2019-09-11 15:55:02 +03:00
yuri
1ffbf63731 dynamicLogicOptions field param 2019-09-11 15:26:14 +03:00
yuri
83fa22852a lang no quote escaping 2019-09-10 16:52:42 +03:00
yuri
7ffd8a54a1 v 2019-09-10 11:07:44 +03:00
yuri
739cb5e98f emailFoldersDisabled param 2019-09-10 11:04:13 +03:00
yuri
f5e1b7175e utils checkActionAvailability 2019-09-10 11:03:56 +03:00
yuri
81cb2aaeb8 fix navbar tpl 2019-09-09 13:28:53 +03:00
yuri
bc54bafc3a lang 2019-09-09 13:26:56 +03:00
yuri
fe635064d5 Merge branch 'stable' 2019-09-06 16:36:18 +03:00
yuri
01ba132a62 field manager fix 2019-09-06 16:19:49 +03:00
yuri
ac318bb302 Merge branch 'hotfix/5.6.12' 2019-09-06 15:21:57 +03:00
yuri
4b31173355 multi enum fix 2019-09-06 15:21:39 +03:00
yuri
34cd265fc1 afterRelate hook for link multiple save 2019-09-04 11:27:27 +03:00
yuri
22f19c0ac9 Merge branch 'hotfix/5.6.12' 2019-09-03 11:05:06 +03:00
yuri
93f8ef9daa Merge branch 'hotfix/5.6.12' of ssh://172.20.0.1/var/git/espo/backend into hotfix/5.6.12 2019-09-03 11:04:15 +03:00
Taras Machyshyn
81ffbc8e6d Merge branch 'hotfix/5.6.12' of ssh://172.20.0.1/var/git/espo/backend into hotfix/5.6.12 2019-09-02 18:18:40 +03:00
Taras Machyshyn
061dda1a1e Fixed index name issue 2019-09-02 18:18:30 +03:00
yuri
c83a24794d hide console commands metadata 2019-09-02 14:52:44 +03:00
yuri
5ab6d25e82 fix dashboard fallback css 2019-09-02 10:31:19 +03:00
yuri
36aab22272 metadata loaders hidden 2019-09-02 10:27:08 +03:00
yuri
a21d5dba9f fix detail nav buttons 2019-08-30 16:33:08 +03:00
yuri
bb8e543509 preferences: hide smtp if email accoint is available 2019-08-30 16:01:31 +03:00
yuri
b93112aaff group email accont refactoring 2019-08-30 15:44:25 +03:00
yuri
c303ec9479 smtp send test fix 2019-08-30 14:04:39 +03:00
yuri
76f10fefa3 css fix 2019-08-30 12:42:44 +03:00
yuri
becd564420 user calendar button icon 2019-08-30 12:41:29 +03:00
yuri
c4f3c1dbd1 email account ui improvement 2019-08-30 12:36:41 +03:00
yuri
f4d1b4a5b8 email account: smtp auth mechanism param 2019-08-30 12:22:01 +03:00
yuri
045264d95f formula tests 2019-08-29 12:49:57 +03:00
yuri
797fab769f Merge branch 'hotfix/5.6.12' 2019-08-29 10:46:42 +03:00
yuri
41d8605e17 fix typo 2019-08-29 10:41:56 +03:00
yuri
18bcaf7e6e Merge branch 'hotfix/5.6.12' 2019-08-28 16:59:30 +03:00
yuri
13a2ba47e7 fix wysiwyg 2019-08-28 16:58:33 +03:00
yuri
1ef60781a0 custom navbar view 2019-08-28 16:43:04 +03:00
yuri
64dfad8cc1 orm binary function 2019-08-28 16:14:25 +03:00
yuri
9862c6d960 list layout fix not sortable fields 2019-08-28 15:39:59 +03:00
yuri
1cfa5ce2b3 fix link multiple list overflow 2019-08-28 13:51:24 +03:00
yuri
4a09baef3a fix layout manager 2019-08-28 12:50:21 +03:00
yuri
5893243ae3 Merge branch 'hotfix/5.6.12' 2019-08-28 12:13:44 +03:00
yuri
4c598d19b5 link one empty filters 2019-08-28 12:13:27 +03:00
yuri
dbc5a47d79 link one empty filters 2019-08-28 12:13:10 +03:00
yuri
0fda920610 acl portal changes and tests 2019-08-28 11:57:12 +03:00
yuri
a6a6ce6722 fix layout 2019-08-28 10:37:23 +03:00
yuri
4db421a75f stream note changes 2019-08-27 17:09:32 +03:00
yuri
3f6cfe3749 stream notes created modified by 2019-08-27 17:00:09 +03:00
yuri
ec9653ff48 fix typo 2019-08-27 15:51:09 +03:00
yuri
24c19f12e5 convert currency changes 2019-08-27 15:45:06 +03:00
yuri
172bba9de5 record service setUser 2019-08-27 15:27:56 +03:00
yuri
7d4227c98d Merge branch 'hotfix/5.6.12' of ssh://172.20.0.1/var/git/espo/backend into hotfix/5.6.12 2019-08-27 15:19:15 +03:00
yuri
4e64ebb346 record service setAcl 2019-08-27 15:18:56 +03:00
Taras Machyshyn
1102e9f1e8 Added support PHPUnit 8 2019-08-27 14:57:31 +03:00
yuri
f5f16a7eb3 convert currency fixes 2019-08-27 12:28:51 +03:00
yuri
7901df4221 covnert currency all fields 2019-08-27 12:09:50 +03:00
yuri
41b114684e mass update currency stubs 2019-08-27 11:59:58 +03:00
yuri
03788be209 client field manager fix 2019-08-27 11:56:34 +03:00
yuri
685226c4d4 fix convert currency 2019-08-27 11:50:01 +03:00
yuri
f8d9548ed5 convert currency detail view action and fixes 2019-08-27 11:40:03 +03:00
yuri
49798c8f7f convert currency fix 2019-08-27 10:56:56 +03:00
yuri
ab4e85ea45 client field manager improvements 2019-08-27 10:56:44 +03:00
yuri
7d7d2d4276 convert currency fix 2019-08-27 10:28:34 +03:00
Taras Machyshyn
607ee9a080 Fix errors in a test 2019-08-26 17:13:33 +03:00
yuri
3f3b65ad90 revert 2019-08-23 16:42:22 +03:00
yuri
354866a809 fix convert currency 2019-08-23 16:41:14 +03:00
yuri
58cda4f31a convert currency mass action 2019-08-23 16:15:28 +03:00
yuri
454c388d10 grid layout highlight droppable 2019-08-23 14:53:10 +03:00
yuri
e0125fa8f9 layout manager css load 2019-08-23 14:44:21 +03:00
yuri
540c594b1c loader fix 2019-08-23 14:44:09 +03:00
yuri
025ac8fe34 update bull js 2019-08-23 14:43:47 +03:00
yuri
6d18cac9a9 contact side panel layout 2019-08-23 13:33:37 +03:00
yuri
f555ee9f9d side panel fields layout 2019-08-23 13:20:48 +03:00
yuri
5ac2e9b516 grid layout with 3 and 4 cells in row 2019-08-22 16:38:02 +03:00
yuri
2cc8c3710f not actual statuses 2019-08-22 13:09:17 +03:00
yuri
72d93cbae4 Merge branch 'hotfix/5.6.12' 2019-08-22 12:15:40 +03:00
yuri
e30838cf27 link fields empty name fix 2019-08-22 12:10:08 +03:00
yuri
790355fb63 lead converted at 2019-08-22 11:46:46 +03:00
yuri
371fbf9864 email skip index if filter 2019-08-21 16:35:45 +03:00
yuri
6e157db23d fix preferences cancel 2019-08-21 15:26:22 +03:00
yuri
85c3d6c4e5 assignment notifications in preferences 2019-08-21 13:53:50 +03:00
yuri
d947236bcc checklist field 2019-08-21 12:44:51 +03:00
yuri
98343cb099 Merge branch 'hotfix/5.6.12' 2019-08-21 12:44:27 +03:00
Taras Machyshyn
e3ac8c3f67 Bug fixes for url params util 2019-08-20 13:58:37 +03:00
Taras Machyshyn
2087894c3a Added 'binary' option for a field 2019-08-19 17:21:00 +03:00
yuri
5c00a7ad43 fix email fitler duplicate 2019-08-19 11:44:25 +03:00
yuri
fe0dcc5342 remove duplicate label 2019-08-19 11:24:34 +03:00
Taras Machyshyn
f05381cc43 Corrected file permissions 2019-08-15 17:18:08 +03:00
yuri
a9a30d7237 duplicate silent confict exception 2019-08-14 13:09:22 +03:00
yuri
5312c1d141 image original link 2019-08-14 12:07:51 +03:00
yuri
a97d516c4f restrict common entity type name 2019-08-13 17:03:28 +03:00
Taras Machyshyn
16c24abb55 Integration tests: create database if not exists 2019-08-13 16:09:02 +03:00
Taras Machyshyn
ed897e9b97 Util changes 2019-08-13 11:56:15 +03:00
Taras Machyshyn
b4b222c1d7 Unit tests for sanitizeHtml() 2019-08-12 15:23:29 +03:00
yuri
a68d711249 summernote fix fullscreen 2019-08-12 15:15:57 +03:00
Taras Machyshyn
d6834f7f97 Installer changes 2019-08-12 14:53:11 +03:00
Taras Machyshyn
20818659e8 Installer changes 2019-08-12 14:49:25 +03:00
yuri
1c668ce983 signature toolbar 2019-08-12 11:39:59 +03:00
yuri
a9dacd74c6 v 2019-08-12 10:56:29 +03:00
yuri
b7c9ddc4e6 auth token secret fixes 2019-08-12 10:56:11 +03:00
yuri
d5ee599f9f language fix 2019-08-12 10:25:12 +03:00
yuri
252d6bccd3 Merge branch 'stable' 2019-08-10 19:16:13 +03:00
yuri
742f4e21b5 comment out auth token secret 2019-08-10 19:11:03 +03:00
yuri
78fc80a619 v 2019-08-10 19:08:57 +03:00
yuri
5a30c6a229 Merge branch 'stable' 2019-08-09 15:26:37 +03:00
yuri
e74fdce568 notifications lower case first 2019-08-09 15:17:16 +03:00
yuri
20139a88af email reply fix 2019-08-09 12:55:54 +03:00
yuri
3d1f632a73 fix lead capture 2019-08-09 12:29:00 +03:00
yuri
9e459e1287 escape error message 2019-08-09 12:22:10 +03:00
yuri
7b9cf6c039 wysiwyg fixes 2019-08-09 12:00:04 +03:00
yuri
1cebef107e dompurify target blank 2019-08-09 11:59:19 +03:00
yuri
47fe232c78 summernote css fixes 2019-08-09 11:31:32 +03:00
yuri
2c458a69a1 skipReRender and summernote fixes 2019-08-09 10:47:09 +03:00
yuri
54a290121e summernote fixes 2019-08-08 19:06:53 +03:00
yuri
11420dc0af summernote updates 2019-08-08 18:51:52 +03:00
yuri
957680a3d0 wysiwyg custom dialogs 2019-08-08 18:18:09 +03:00
yuri
3c6e19f4c0 wysiwyg codeviewFilter 2019-08-08 15:04:33 +03:00
yuri
4253915eed update summernote 2019-08-08 14:56:12 +03:00
yuri
da34789020 remove not needed escaping 2019-08-08 13:10:53 +03:00
yuri
cc057af59a filename escaping 2019-08-08 12:46:31 +03:00
yuri
ebaa3ed958 Merge branch 'hotfix/5.6.10' of ssh://172.20.0.1/var/git/espo/backend into hotfix/5.6.10 2019-08-08 11:49:05 +03:00
yuri
307c045c0e css fixes 2019-08-08 11:44:24 +03:00
Taras Machyshyn
8e09b81cf0 Fix instalation warnings 2019-08-08 10:16:08 +03:00
yuri
f40aeb0830 keep admin view on return 2019-08-07 14:31:42 +03:00
yuri
135b56001c admin icons 2019-08-07 12:38:54 +03:00
yuri
cc97454c58 Merge branch 'hotfix/5.6.10' 2019-08-07 12:00:19 +03:00
yuri
9d3e3e282f more admin icons 2019-08-07 11:56:53 +03:00
yuri
1bcd54dfe1 wysiwyg dom purify for edit except pdf templates 2019-08-07 11:19:41 +03:00
yuri
727f3a4707 attachment entry point allow only image types 2019-08-07 10:49:28 +03:00
yuri
d5473f9985 css fix 2019-08-06 17:54:39 +03:00
yuri
85c06f7ca3 admin icons 2019-08-06 17:42:41 +03:00
yuri
1d4d9882b6 portal users default order by date 2019-08-06 16:00:54 +03:00
yuri
e633984f7e fix tests 2019-08-06 15:15:09 +03:00
yuri
2dabecbae3 acl strict mode by default 2019-08-06 14:59:55 +03:00
yuri
1a4e40d0cd login 2nd step send header 2019-08-06 13:56:19 +03:00
yuri
6ea5ed4f55 merge manual 2019-08-06 13:55:59 +03:00
yuri
c0800a4d16 auth token secret 2019-08-06 13:37:34 +03:00
yuri
85c62c0789 dashlets wo scroll on small screen 2019-08-06 12:48:50 +03:00
yuri
551e82ad8b css fixes 2019-08-06 11:11:30 +03:00
yuri
7f4328499b task priority on activities dashlet 2019-08-06 11:05:29 +03:00
yuri
de4d794d72 mention de_DE template 2019-08-06 10:31:28 +03:00
yuri
e2e374301a escape string 2019-08-05 16:58:22 +03:00
yuri
7a919dedfd lead capture improvements 2019-08-05 16:47:12 +03:00
yuri
06c8cbac52 mail sender fix 2019-08-05 16:29:52 +03:00
yuri
f54ad3ab0d lead capture dont log twice 2019-08-05 15:43:42 +03:00
yuri
197fb26beb campaign log filter 2019-08-05 15:30:32 +03:00
yuri
9e4582af17 langauge sanitize 2019-08-05 15:25:52 +03:00
yuri
b399c4444d global restriction fix 2019-08-05 14:05:04 +03:00
yuri
b7e9e69f35 confirm leave out backdrop 2019-08-05 10:27:49 +03:00
yuri
105b6444ed fix popovers 2019-08-02 17:16:16 +03:00
yuri
3559ff9f4e de lang fix 2019-08-02 16:54:04 +03:00
yuri
4be2962567 remove html from language 2019-08-02 16:46:48 +03:00
yuri
1cce748af5 fix test 2019-08-02 15:22:12 +03:00
yuri
64d3e3b5dd acl globalrestriction test 2019-08-02 15:18:47 +03:00
yuri
2e0233a239 task priority label 2019-08-02 15:07:26 +03:00
yuri
5b4bd88907 case priority label 2019-08-02 15:05:40 +03:00
yuri
6176ead935 GlobalRestricton cache fix 2019-08-02 14:12:07 +03:00
yuri
46ddf3b127 language fixes 2019-08-02 13:58:25 +03:00
yuri
2944e64de5 css fix 2019-08-02 12:23:44 +03:00
yuri
372cf20145 css fixes 2019-08-01 18:38:57 +03:00
yuri
de68227d49 Merge branch 'hotfix/5.6.10' 2019-08-01 17:14:26 +03:00
yuri
633cc12650 css fixes 2019-08-01 16:51:39 +03:00
yuri
202d799436 header changes 2019-08-01 16:44:59 +03:00
yuri
f14d12853b header grid auto fit 2019-08-01 16:13:49 +03:00
yuri
a98f7f5068 lead capture optInDateTime placeholder 2019-08-01 13:55:47 +03:00
yuri
ebe249bd6b mobile ui fixes 2019-08-01 12:31:04 +03:00
yuri
7957296479 fix 2019-07-31 16:48:08 +03:00
yuri
b7eac5ba42 flexible titles 2019-07-31 16:43:54 +03:00
yuri
14cab0dbde dashboard changes 2019-07-31 15:26:41 +03:00
yuri
5db39d14e7 Merge branch 'hotfix/5.6.10' of github.com:espocrm/espocrm into hotfix/5.6.10 2019-07-31 13:06:49 +03:00
yuri
cf473d6f16 ui improvements 2019-07-31 13:06:40 +03:00
Karthik Bhat K
a082ea6acc added use Forbidden and removed unused class (#1372) 2019-07-31 09:06:47 +03:00
yuri
f770074be2 v 2019-07-30 18:53:33 +03:00
yuri
106e7726ba xs view navbar improvements 2019-07-30 18:50:50 +03:00
yuri
ed0ca19ed5 Merge branch 'hotfix/5.6.10' 2019-07-30 15:50:36 +03:00
yuri
e680be54ad fix mapper 2019-07-30 14:58:56 +03:00
yuri
3224795368 sth collection fix 2019-07-30 13:41:09 +03:00
yuri
ec5bd78a33 access modal backdrop 2019-07-30 12:43:36 +03:00
yuri
d0c5d2aace lead capture hooks 2019-07-30 11:26:16 +03:00
yuri
f8d0688f2f fix invalid json 2019-07-30 11:07:31 +03:00
yuri
2b2b2b9faa Merge branch 'hotfix/5.6.9' 2019-07-30 11:07:02 +03:00
yuri
ed4ce1a36e user disable mass update emailAddress and unset password 2019-07-30 11:06:42 +03:00
yuri
a6d7f91a58 disable inline edit for user type 2019-07-30 11:03:06 +03:00
yuri
f6f6c2b2ac email signature fit height 2019-07-30 11:01:51 +03:00
yuri
8c628df639 fix typo 2019-07-30 10:59:06 +03:00
yuri
ffd3f762ce xss fixes 2019-07-30 10:57:44 +03:00
yuri
24628a8487 returnSthCollection param 2019-07-29 17:56:15 +03:00
yuri
573d147dad v 2019-07-29 13:13:45 +03:00
yuri
4ab767e95b hook service factrory 2019-07-29 12:50:04 +03:00
yuri
18b64238ee fix list expanded overflow 2019-07-29 12:48:34 +03:00
yuri
65ded811b4 calendar height fix 2019-07-26 16:50:39 +03:00
yuri
57f454693a lead capture addition 2019-07-26 10:59:07 +03:00
yuri
b6409b5cba panel list border top 2019-07-26 10:42:47 +03:00
yuri
e6a83f7d19 reset button text 2019-07-26 10:35:16 +03:00
yuri
c39e1d140f css flip class 2019-07-26 10:35:10 +03:00
yuri
1c69f1ca95 categories fix 2019-07-25 17:50:30 +03:00
yuri
a21c071349 categories panel style change 2019-07-25 17:46:12 +03:00
yuri
d8d8e68914 Merge branch 'stable' 2019-07-25 17:35:47 +03:00
yuri
741a6d5dab wysiwyg fixes 2019-07-25 16:25:18 +03:00
yuri
0c0a602330 list w categoties expand when text search 2019-07-25 13:06:30 +03:00
yuri
7d13018eba fix category expanding storing 2019-07-25 12:38:20 +03:00
yuri
44c65c0117 kb min body height in portal 2019-07-25 11:59:00 +03:00
yuri
e9e758c4b3 v 2019-07-25 11:26:02 +03:00
yuri
7e7acb8d28 fix template comments stripped 2019-07-25 11:25:48 +03:00
yuri
4b28687c37 gMerge branch 'stable' 2019-07-24 16:27:31 +03:00
yuri
1ab897db28 fix email address field 2019-07-24 16:17:22 +03:00
yuri
1a990850ca add MMK currency 2019-07-24 16:05:37 +03:00
yuri
36e360e167 v 2019-07-24 15:49:55 +03:00
yuri
9ffdb1a1f0 title for some fields in list mode 2019-07-24 13:00:08 +03:00
yuri
64b15f9282 stream dont show create post for portal 2019-07-24 12:54:37 +03:00
yuri
e8ebe51f1c note create acl 2019-07-24 12:53:03 +03:00
yuri
ec2a7d2f48 fix no join name null 2019-07-24 12:25:24 +03:00
yuri
6782e7c15c dashboard fixes 2019-07-24 12:20:02 +03:00
yuri
3c73e3e8cf more string escaping 2019-07-24 12:04:24 +03:00
yuri
321306c42d Merge branch 'stable' 2019-07-23 18:22:53 +03:00
yuri
4ab7d19776 xss fixes 2019-07-23 17:50:02 +03:00
yuri
92864d9133 fix typo 2019-07-23 14:21:39 +03:00
yuri
34e33bd13a searchPanelInPortalDisabled 2019-07-23 13:29:01 +03:00
yuri
49fa22fa3d v 2019-07-23 11:37:07 +03:00
yuri
10fcd79155 portal home tab 2019-07-23 11:36:39 +03:00
yuri
00598170af fix user dashboard template 2019-07-22 18:33:44 +03:00
yuri
ac5b2f38d4 portal user dashboard template 2019-07-22 18:29:21 +03:00
yuri
82d486a2b4 Merge branch 'hotfix/5.6.6' 2019-07-22 18:27:25 +03:00
yuri
6cdc8f2823 mass update for portal users 2019-07-22 15:07:40 +03:00
yuri
d611ebfc86 cleanup try catch 2019-07-22 14:04:21 +03:00
yuri
599a7c6080 cleanup record service method 2019-07-22 13:53:02 +03:00
yuri
4c21f1192d email send: dont fail if attachment deleted 2019-07-22 13:00:25 +03:00
yuri
63e78baf21 fix orm join conditions 2019-07-22 12:46:14 +03:00
yuri
fa2c689a34 email skip index when search by link 2019-07-19 17:37:05 +03:00
yuri
a53b440b8b mass update hide not accessible fields 2019-07-19 17:13:30 +03:00
yuri
8189832af2 style option color 2019-07-19 17:07:58 +03:00
yuri
135f869e1a dynamic logix support currency 2019-07-19 13:12:18 +03:00
yuri
2d56525a25 fix autocomplete empty name 2019-07-19 12:57:48 +03:00
yuri
9e78276f3d Merge branch 'hotfix/5.6.5' 2019-07-18 17:23:53 +03:00
yuri
03773dd929 fix xss document file 2019-07-18 17:10:20 +03:00
yuri
5b8dba68f3 fix orm getAllAttributesFromComplexExpression 2019-07-18 16:55:55 +03:00
yuri
f63c75e18d v 2019-07-18 16:37:09 +03:00
yuri
6dd0bd8b90 sanitize wysiwyg 2019-07-18 16:36:00 +03:00
yuri
94e86f875a lead capture changes 2019-07-18 13:43:35 +03:00
yuri
b582b20003 array field improvement 2019-07-17 13:02:24 +03:00
yuri
e58e82eea1 entry point dont log stop error 2019-07-17 12:14:11 +03:00
yuri
dddc4feda8 image silent error not found 2019-07-17 12:13:52 +03:00
yuri
a65421a268 pt br lang 2019-07-17 12:01:45 +03:00
yuri
b6696dcc26 ru lang 2019-07-17 11:59:35 +03:00
yuri
4e3d8fb98f pl lang 2019-07-17 11:58:53 +03:00
yuri
e0aaff932e nl lang 2019-07-17 11:58:08 +03:00
yuri
bd61c12ea2 Merge branch 'stable' 2019-07-17 11:26:23 +03:00
yuri
17ea760851 layout fix 2019-07-16 17:55:57 +03:00
yuri
f3d11aede3 disable inlide edit for email accounts 2019-07-16 14:23:56 +03:00
yuri
8053d65f33 dynamic logic in options 2019-07-16 11:00:48 +03:00
yuri
01809d2cc3 confirm focus 2019-07-16 10:47:21 +03:00
yuri
ac8e3d1a69 fix portal base path 2019-07-16 10:41:47 +03:00
yuri
b669ccf733 fix portal isCustom undefined 2019-07-16 10:22:26 +03:00
yuri
8c42c6bf0b fix bool opted out search 2019-07-16 10:17:06 +03:00
yuri
97bdd22795 sanitize complex text 2019-07-15 12:37:23 +03:00
yuri
0f3cd7913c link skip acl param 2019-07-11 16:54:49 +03:00
yuri
502cfc8d76 fix foreign enum 2019-07-11 10:40:52 +03:00
yuri
010f757452 totp fix input max length 2019-07-10 13:26:03 +03:00
yuri
4230f768e5 login form fix 2019-07-10 13:22:36 +03:00
yuri
4197f30ad5 frontendHiddenPathList in metadata 2019-07-10 12:27:13 +03:00
yuri
b4eecbd0af fix 2fa 2019-07-10 12:14:24 +03:00
yuri
6560c0e0bd 2fa totp 2019-07-10 12:04:18 +03:00
yuri
d31e826305 fix websocket 2019-07-08 16:18:22 +03:00
yuri
a9f2fe8590 Merge branch 'hotfix/5.6.4' 2019-07-05 11:31:42 +03:00
yuri
28a62a2581 auth method list from metadata 2019-07-05 11:31:29 +03:00
yuri
0d4b548160 auth prepare for 2 factor 2019-07-04 16:44:41 +03:00
yuri
2240819930 Merge branch 'hotfix/5.6.4' 2019-07-04 14:39:09 +03:00
yuri
deff6bad36 login page changes 2019-07-04 13:30:36 +03:00
yuri
f2bc80b7d9 auth settings changes 2019-07-04 12:15:29 +03:00
yuri
b5b5cac0ac calendar mode buttons improvement 2019-07-03 13:25:54 +03:00
yuri
308f30510b calendar css fixes 2019-07-03 11:51:35 +03:00
yuri
574513a9c8 fix navbar more scroll 2019-07-03 11:27:52 +03:00
yuri
f37bc7d46b external account token renewal fix 2019-07-02 16:09:25 +03:00
yuri
9f5db434df fix attachment multiple 2019-07-02 15:54:41 +03:00
yuri
1c8abd6e36 mutliple attachment multiple fields 2019-07-02 15:46:10 +03:00
yuri
c95fcd6fbd action configCheck 2019-07-01 16:31:02 +03:00
yuri
e6009366e8 version 2019-07-01 12:47:27 +03:00
yuri
286502d872 cssList fix 2019-07-01 12:47:07 +03:00
yuri
0749bdc006 cs fix 2019-07-01 12:45:54 +03:00
yuri
72ef137dbe fix app js logout request url 2019-07-01 12:45:50 +03:00
yuri
f624bdef5c settings global params 2019-07-01 12:30:44 +03:00
yuri
9730680e41 fix htmlizer 2019-07-01 10:30:23 +03:00
yuri
5d857f6a86 install css font weight fix 2019-06-27 12:21:52 +03:00
yuri
44947f8844 Merge branch 'hotfix/5.6.4' 2019-06-27 12:18:50 +03:00
yuri
c59fa40615 log exception message line and file 2019-06-27 11:51:34 +03:00
yuri
4dc08dd37c custom login view 2019-06-27 11:34:35 +03:00
yuri
e76c7564dc custom unsubscribe template 2019-06-27 11:05:44 +03:00
yuri
802bac82f0 fix create activity from panel 2019-06-26 16:37:58 +03:00
yuri
172af61b6b Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2019-06-26 15:23:10 +03:00
yuri
e6bdaa1c86 Merge branch 'stable' 2019-06-26 15:22:52 +03:00
yuri
c64cc13e4c link additional select params 2019-06-26 12:06:30 +03:00
yuri
30be62eabe fix google maps 2019-06-26 10:37:16 +03:00
yuri
1c1703b349 fix pdf 2019-06-25 13:35:50 +03:00
yuri
6916f3242c external account after connect hook 2019-06-25 11:07:32 +03:00
yuri
d2630a5c3f fix deleted user view 2019-06-24 14:25:52 +03:00
yuri
3e02776fcd email fixes 2019-06-24 13:29:45 +03:00
yuri
376bfe63e0 fix meeting set held 2019-06-20 16:33:09 +03:00
yuri
950a3b4703 calendar fix 2019-06-20 13:35:02 +03:00
yuri
2f5ffb7421 external acount acl 2019-06-18 13:32:18 +03:00
yuri
043ef8ef3e oauth client support header string 2019-06-13 16:30:10 +03:00
yuri
a959a2deaf view followers all users 2019-06-12 15:33:08 +03:00
yuri
65a3f4e5f3 external account return uri path 2019-06-12 13:36:34 +03:00
yuri
97355ad024 fix redirect uri 2019-06-12 11:26:50 +03:00
yuri
99300d2d61 integration redirect uri custom 2019-06-12 11:24:11 +03:00
yuri
e070d9d9f1 oauth-callback file 2019-06-12 11:11:43 +03:00
yuri
77a884c951 add field in modal 2019-06-10 15:54:28 +03:00
yuri
6a2974845a dashboard template append 2019-06-10 14:51:08 +03:00
yuri
0095cfd68c dashboard tab id 2019-06-10 13:36:25 +03:00
yuri
22f0e5b399 dashboard layout fix 2019-06-10 12:58:14 +03:00
yuri
1abc853bc5 dashboard templates 2019-06-10 12:37:35 +03:00
yuri
6f9dc66104 entity manager fetchEntity function 2019-06-10 12:37:23 +03:00
yuri
d9094238c1 edit for modal 2019-06-10 12:14:23 +03:00
yuri
25ea320042 Merge branch 'hotfix/5.6.3' 2019-06-10 10:41:07 +03:00
yuri
8f9bb79978 button handler init 2019-06-07 16:37:02 +03:00
yuri
405a5c29df layout manager css fix 2019-06-07 14:24:20 +03:00
yuri
4f2a2b290b fix calendar custom event 2019-06-07 11:32:30 +03:00
yuri
fd5218c10b btn-text color 2019-06-06 16:05:22 +03:00
yuri
73ab4aa42e meeting modal acceptance right 2019-06-06 13:16:48 +03:00
yuri
66a60b1257 timeline tpl change 2019-06-06 12:53:39 +03:00
yuri
c975954944 stream page btn-text 2019-06-06 12:42:47 +03:00
yuri
86799ed742 btn-text normal weight 2019-06-06 12:41:34 +03:00
yuri
0c143eb8a1 Merge branch 'hotfix/5.6.3' 2019-06-06 11:45:40 +03:00
yuri
1da6eca3be update npm packages 2019-06-06 11:45:31 +03:00
yuri
a462158582 category folder style fix 2019-06-05 16:22:35 +03:00
yuri
152b8991d3 email apply filters on sending 2019-06-05 14:15:36 +03:00
yuri
5adcf013d7 fix client manager 2019-06-05 12:56:02 +03:00
yuri
9d6401cbd3 fix iframe css 2019-06-05 12:49:49 +03:00
yuri
9f178bb467 next/prev fix on email remove to trash 2019-06-05 12:45:59 +03:00
yuri
f3b2cc9bcb cleanup 2019-06-05 12:19:10 +03:00
yuri
b18a8c1b2b fix 2019-06-05 12:15:42 +03:00
yuri
30dd7e560b search buttons narrow on small screen 2019-06-05 11:51:57 +03:00
yuri
ca35a4df75 fix dashboard tpl 2019-06-05 11:46:01 +03:00
yuri
fe0a6d1a8b map none label 2019-06-05 11:43:46 +03:00
yuri
f9223087df xlsx export array 2019-06-05 11:26:46 +03:00
yuri
f3980ef9a1 fix theme manager default css 2019-06-05 11:09:50 +03:00
yuri
93553d18fc fix job text filter 2019-06-05 11:05:05 +03:00
yuri
e35561093d css fix 2019-05-20 16:37:43 +03:00
yuri
b0742aa0c6 edit dashboard fix 2019-05-20 15:46:03 +03:00
yuri
efad2b142a font fixes 2019-05-20 15:40:50 +03:00
yuri
362f895e72 client link crossorigin 2019-05-20 15:22:26 +03:00
yuri
224b73eb89 font fix 2019-05-20 15:00:48 +03:00
yuri
c4cf45e7f8 fix preload font 2019-05-20 14:46:28 +03:00
yuri
cb45e57a99 client link noTimestamp 2019-05-20 14:00:50 +03:00
yuri
e7f62f79ef font swap 2019-05-20 13:56:21 +03:00
yuri
6fb21ccd77 client links 2019-05-20 13:51:05 +03:00
yuri
f55d814cbb font display block 2019-05-20 13:50:58 +03:00
yuri
8fa22a6c7d css fix 2019-05-20 13:18:03 +03:00
yuri
b86ca3a4ec stream popover fix 2019-05-20 12:19:54 +03:00
yuri
ab51e70ff4 add dashlet style fix 2019-05-20 12:02:38 +03:00
yuri
0bde83c0e7 css fix 2019-05-20 11:49:04 +03:00
yuri
589754851c font update 2019-05-20 11:47:27 +03:00
yuri
253930ba6d btn-text fix 2019-05-20 10:40:46 +03:00
yuri
09c19bbba5 remove cancel buttons from 2 modals 2019-05-17 16:38:41 +03:00
yuri
ac96ebffa5 btn-text 2019-05-17 16:10:26 +03:00
yuri
f01715452f last viewed date short 2019-05-17 13:18:19 +03:00
yuri
875b01dcd6 lastViewed metadata scopes param 2019-05-17 13:16:07 +03:00
yuri
61def7e332 pdf currency symbol 2019-05-17 12:50:36 +03:00
yuri
86b24ff618 orm field type attribute role 2019-05-17 12:50:27 +03:00
yuri
1d29d74204 cs fix 2019-05-17 12:46:39 +03:00
yuri
2cf820e181 user activities fix and optimization 2019-05-16 16:53:09 +03:00
yuri
bf8ab4dec2 account shipping address copy button hidden 2019-05-16 16:18:41 +03:00
yuri
7b772bde3c cs fix 2019-05-16 15:24:12 +03:00
Taras Machyshyn
16bad37dc5 Installer config improvements 2019-05-16 12:41:17 +03:00
Taras Machyshyn
76b8009323 Gruntfile: ignore installation config 2019-05-16 12:31:59 +03:00
yuri
d67f806ec8 remove tabs 2019-05-16 12:21:38 +03:00
yuri
72bed367e4 webhooks 2019-05-16 12:19:42 +03:00
yuri
fe2a4f91ee Merge branch 'hotfix/5.6.3' 2019-05-16 12:19:18 +03:00
Taras Machyshyn
471dd1ab49 Added config file for installation 2019-05-16 11:56:43 +03:00
Taras Machyshyn
8efc395ca1 FileManager fix 2019-05-15 15:54:01 +03:00
yuri
dfbad82ba8 version 2019-05-15 14:34:11 +03:00
yuri
84938d88fc fix cleanup deleted records 2019-05-15 14:24:44 +03:00
yuri
ed5f66729d date time getSystemNowString 2019-05-14 16:49:47 +03:00
yuri
ee5218c46d orm entity id is changed fix 2019-05-14 15:48:27 +03:00
yuri
6758d71e2e lead capture skip email address is opted out field 2019-05-13 15:11:58 +03:00
yuri
1eec3d88cf integration test auth method 2019-05-13 12:31:49 +03:00
yuri
34f1c196f1 fix command upgrade msg 2019-05-13 10:43:59 +03:00
yuri
a87e2fb5db css fix 2019-05-13 10:25:57 +03:00
yuri
cecafa5e28 diff only one version from 2019-05-10 16:25:54 +03:00
yuri
a0e5f2d8ec cleanup 2019-05-10 16:06:58 +03:00
yuri
47080f9b78 search view changes 2019-05-10 16:06:41 +03:00
yuri
07fa53d0c2 Merge branch 'hotfix/5.6.2' 2019-05-10 15:04:36 +03:00
yuri
698f70a358 Merge branch 'hotfix/5.6.2' 2019-05-03 11:23:12 +03:00
yuri
3170571cb9 Merge branch 'hotfix/5.6.2' 2019-05-01 12:40:28 +03:00
yuri
e48c067b05 Merge branch 'hotfix/5.6.2' 2019-04-24 11:58:12 +03:00
Pomazan Bogdan
492670eb7d Правка перевода, логически не подходит (#1298) 2019-04-23 12:44:23 +03:00
1354 changed files with 26060 additions and 9544 deletions

3
.gitignore vendored
View File

@@ -15,4 +15,5 @@ npm-debug.log
/tests/integration/config.php
composer.phar
vendor/
/custom/Espo/Custom/*
/custom/Espo/Custom/*
/install/config.php

View File

@@ -9,13 +9,9 @@ DirectoryIndex index.php index.html
# PROTECTED DIRECTORIES
RewriteCond %{REQUEST_FILENAME} -d
RewriteRule ^/?(data|api|client)/ - [F]
RewriteRule ^/?(api|client)/ - [F]
RewriteRule ^/?data/config\.php$ - [F]
RewriteRule ^/?data/logs/ - [F]
RewriteRule ^/?data/cache/ - [F]
RewriteRule ^/?data/upload/ - [F]
RewriteRule ^/?data/\.backup/ - [F]
RewriteRule ^/?data/ - [F]
RewriteRule ^/?application/ - [F]
RewriteRule ^/?custom/ - [F]
RewriteRule ^/?vendor/ - [F]

View File

@@ -126,7 +126,7 @@ module.exports = function (grunt) {
start: ['build/EspoCRM-*'],
final: ['build/tmp'],
beforeFinal: {
src: ['build/tmp/custom/Espo/Custom/*', '!build/tmp/custom/Espo/Custom/.htaccess']
src: ['build/tmp/custom/Espo/Custom/*', '!build/tmp/custom/Espo/Custom/.htaccess', 'build/tmp/install/config.php']
}
},
less: lessData,
@@ -190,6 +190,7 @@ module.exports = function (grunt) {
'extension.php',
'websocket.php',
'command.php',
'oauth-callback.php',
'index.php',
'LICENSE.txt',
'.htaccess',

View File

@@ -9,7 +9,7 @@ Download the latest release from our [website](http://www.espocrm.com).
### Requirements
* PHP 7.1 or above (with pdo, json, gd, openssl, zip, imap, mbstring, curl extensions);
* MySQL 5.5.3 or above, or MariaDB.
* MySQL 5.6.0 or above, or MariaDB.
For more information about server configuration see [this article](https://www.espocrm.com/documentation/administration/server-configuration/).

View File

@@ -46,6 +46,21 @@ class Note extends \Espo\Core\Acl\Base
return false;
}
public function checkEntityCreate(EntityUser $user, Entity $entity, $data)
{
if ($entity->get('parentId') && $entity->get('parentType')) {
$parent = $this->getEntityManager()->getEntity($entity->get('parentType'), $entity->get('parentId'));
if ($parent) {
if ($this->getAclManager()->checkEntity($user, $parent, 'stream')) {
return true;
}
}
return false;
}
return true;
}
public function checkEntityEdit(EntityUser $user, Entity $entity, $data)
{
if ($user->isAdmin()) {

View File

@@ -0,0 +1,73 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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\Acl;
use \Espo\Entities\User as EntityUser;
use \Espo\ORM\Entity;
class Webhook extends \Espo\Core\Acl\Base
{
public function checkIsOwner(EntityUser $user, Entity $entity)
{
return $user->id === $entity->get('userId') && $user->isApi();
}
public function checkEntityCreate(EntityUser $user, Entity $entity, $data)
{
if ($user->isAdmin()) return true;
if (!$data) return false;
if ($user->isApi() && $user->id === $entity->get('userId')) return true;
return false;
}
public function checkEntityRead(EntityUser $user, Entity $entity, $data)
{
if ($user->isAdmin()) return true;
if (!$data) return false;
if ($user->isApi() && $user->id === $entity->get('userId')) return true;
return false;
}
public function checkEntityDelete(EntityUser $user, Entity $entity, $data)
{
if ($user->isAdmin()) return true;
if (!$data) return false;
if ($user->isApi() && $user->id === $entity->get('userId')) return true;
return false;
}
public function checkEntityEdit(EntityUser $user, Entity $entity, $data)
{
if ($user->isAdmin()) return true;
if (!$data) return false;
if ($user->isApi() && $user->id === $entity->get('userId')) return true;
return false;
}
}

View File

@@ -41,5 +41,26 @@ class Note extends \Espo\Core\AclPortal\Base
}
return false;
}
}
public function checkEntityCreate(EntityUser $user, Entity $entity, $data)
{
if ($entity->get('type') !== 'Post') return false;
if ($entity->get('type') === 'Post' && $entity->get('targetType')) {
return false;
}
if (!$entity->get('parentId') || !$entity->get('parentType')) {
return false;
}
$parent = $this->getEntityManager()->getEntity($entity->get('parentType'), $entity->get('parentId'));
if ($parent) {
if ($this->getAclManager()->checkEntity($user, $parent, 'stream')) {
return true;
}
}
return false;
}
}

View File

@@ -67,4 +67,9 @@ class ActionHistoryRecord extends \Espo\Core\Controllers\Record
{
throw new Forbidden();
}
public function beforeMassConvertCurrency()
{
throw new Forbidden();
}
}

View File

@@ -64,4 +64,9 @@ class AuthLogRecord extends \Espo\Core\Controllers\Record
{
throw new Forbidden();
}
public function beforeMassConvertCurrency()
{
throw new Forbidden();
}
}

View File

@@ -94,4 +94,9 @@ class AuthToken extends \Espo\Core\Controllers\Record
{
throw new Forbidden();
}
public function beforeMassConvertCurrency()
{
throw new Forbidden();
}
}

View File

@@ -0,0 +1,67 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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\Forbidden;
use \Espo\Core\Exceptions\BadRequest;
class DashboardTemplate extends \Espo\Core\Controllers\Record
{
protected function checkControllerAccess()
{
if (!$this->getUser()->isAdmin()) {
throw new Forbidden();
}
}
public function postActionDeployToUsers($params, $data)
{
if (empty($data->id)) throw new BadRequest();
if (empty($data->userIdList)) throw new BadRequest();
return $this->getServiceFactory()->create('DashboardTemplate')->deployToUsers(
$data->id,
$data->userIdList,
!empty($data->append)
);
}
public function postActionDeployToTeam($params, $data)
{
if (empty($data->id)) throw new BadRequest();
if (empty($data->teamId)) throw new BadRequest();
return $this->getServiceFactory()->create('DashboardTemplate')->deployToTeam(
$data->id,
$data->teamId,
!empty($data->append)
);
}
}

View File

@@ -75,7 +75,7 @@ class Email extends \Espo\Core\Controllers\Record
throw new NotFound();
}
if (!$this->getUser()->isAdmin()) {
if ($emailAccount->get('assigniedUserId') !== $this->getUser()->id) {
if ($emailAccount->get('assignedUserId') !== $this->getUser()->id) {
throw new Forbidden();
}
}

View File

@@ -92,6 +92,9 @@ class EntityManager extends \Espo\Core\Controllers\Base
if (isset($data['fullTextSearch'])) {
$params['fullTextSearch'] = $data['fullTextSearch'];
}
if (isset($data['countDisabled'])) {
$params['countDisabled'] = $data['countDisabled'];
}
$params['kanbanViewMode'] = !empty($data['kanbanViewMode']);
if (!empty($data['kanbanStatusIgnoreList'])) {

View File

@@ -47,17 +47,27 @@ class ExternalAccount extends \Espo\Core\Controllers\Record
public function actionList($params, $data, $request)
{
$integrations = $this->getEntityManager()->getRepository('Integration')->find();
$arr = array();
$list = [];
foreach ($integrations as $entity) {
if ($entity->get('enabled') && $this->getMetadata()->get('integrations.' . $entity->id .'.allowUserAccounts')) {
$arr[] = array(
$userAccountAclScope = $this->getMetadata()->get(['integrations', $entity->id, 'userAccountAclScope']);
if ($userAccountAclScope) {
if (!$this->getAcl()->checkScope($userAccountAclScope)) {
continue;
}
}
$list[] = [
'id' => $entity->id
);
];
}
}
return array(
'list' => $arr
);
return [
'list' => $list
];
}
public function actionGetOAuth2Info($params, $data, $request)

View File

@@ -121,6 +121,8 @@ class FieldManager extends \Espo\Core\Controllers\Base
$this->getContainer()->get('fieldManager')->resetToDefault($data->scope, $data->name);
$this->getContainer()->get('dataManager')->clearCache();
$this->getContainer()->get('dataManager')->rebuildMetadata();
return true;

View File

@@ -33,9 +33,13 @@ class I18n extends \Espo\Core\Controllers\Base
{
public function actionRead($params, $data, $request)
{
if ($request->get('default')) {
$default = $request->get('default') === 'true';
return $this->getServiceFactory()->create('Language')->getDataForFrontend($default);
/*if ($request->get('default')) {
return $this->getContainer()->get('defaultLanguage')->getAll();
}
return $this->getContainer()->get('language')->getAll();
return $this->getContainer()->get('language')->getAll();*/
}
}

View File

@@ -39,11 +39,7 @@ class Layout extends \Espo\Core\Controllers\Base
{
public function actionRead($params, $data)
{
$data = $this->getContainer()->get('layout')->get($params['scope'], $params['name']);
if (empty($data)) {
throw new NotFound("Layout " . $params['scope'] . ":" . $params['name'] . ' is not found.');
}
return $data;
return $this->getServiceFactory()->create('Layout')->getForFrontend($params['scope'], $params['name']);
}
public function actionUpdate($params, $data, $request)

View File

@@ -69,4 +69,11 @@ class LeadCapture extends \Espo\Core\Controllers\Record
return $this->getRecordService()->generateNewApiKeyForEntity($data->id)->getValueMap();
}
public function getActionSmtpAccountDataList()
{
if (!$this->getUser()->isAdmin()) throw new Forbidden();
return $this->getServiceFactory()->create('LeadCapture')->getSmtpAccountDataList();
}
}

View File

@@ -36,7 +36,7 @@ class Metadata extends \Espo\Core\Controllers\Base
public function actionRead($params, $data)
{
return $this->getMetadata()->getAllForFrontend();
return $this->getServiceFactory()->create('Metadata')->getDataForFrontend();
}
public function getActionGet($params, $data, $request)

View File

@@ -25,10 +25,12 @@
*
* 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\Forbidden;
class ScheduledJob extends \Espo\Core\Controllers\Record
{
protected function checkControllerAccess()

View File

@@ -25,10 +25,12 @@
*
* 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\Forbidden;
class ScheduledJobLogRecord extends \Espo\Core\Controllers\Record
{
protected function checkControllerAccess()
@@ -38,4 +40,3 @@ class ScheduledJobLogRecord extends \Espo\Core\Controllers\Record
}
}
}

View File

@@ -29,18 +29,22 @@
namespace Espo\Controllers;
use \Espo\Core\Exceptions\Error;
use \Espo\Core\Exceptions\Forbidden;
use \Espo\Core\Exceptions\BadRequest;
use Espo\Core\Exceptions\Forbidden;
use Espo\Core\Exceptions\BadRequest;
class Settings extends \Espo\Core\Controllers\Base
{
protected function getConfigData()
{
$data = $this->getServiceFactory()->create('Settings')->getConfigData();
$data->jsLibs = $this->getMetadata()->get('app.jsLibs');
$data->jsLibs = $this->getMetadata()->get(['app', 'jsLibs']);
unset($data->loginView);
$loginView = $this->getMetadata()->get(['clientDefs', 'App', 'loginView']);
if ($loginView) {
$data->loginView = $loginView;
}
return $data;
}

View File

@@ -46,6 +46,7 @@ class Stream extends \Espo\Core\Controllers\Base
$maxSize = intval($request->get('maxSize'));
$after = $request->get('after');
$filter = $request->get('filter');
$skipOwn = $request->get('skipOwn') === 'true';
$service = $this->getService('Stream');
@@ -61,7 +62,8 @@ class Stream extends \Espo\Core\Controllers\Base
'offset' => $offset,
'maxSize' => $maxSize,
'after' => $after,
'filter' => $filter
'filter' => $filter,
'skipOwn' => $skipOwn,
]);
return (object) [
@@ -96,7 +98,7 @@ class Stream extends \Espo\Core\Controllers\Base
'maxSize' => $maxSize,
'after' => $after,
'filter' => 'posts',
'where' => $where
'where' => $where,
]);
return (object) [

View File

@@ -69,24 +69,28 @@ class User extends \Espo\Core\Controllers\Record
throw new BadRequest();
}
$p = $this->getEntityManager()->getRepository('PasswordChangeRequest')->where(array(
'requestId' => $data->requestId
))->findOne();
if ($this->getConfig()->get('passwordRecoveryDisabled')) {
throw new Forbidden("Password recovery disabled");
}
if (!$p) {
$request = $this->getEntityManager()->getRepository('PasswordChangeRequest')->where([
'requestId' => $data->requestId
])->findOne();
if (!$request) {
throw new Forbidden();
}
$userId = $p->get('userId');
$userId = $request->get('userId');
if (!$userId) {
throw new Error();
}
$this->getEntityManager()->removeEntity($p);
if ($this->getService('User')->changePassword($userId, $data->password)) {
return array(
'url' => $p->get('url')
);
$this->getEntityManager()->removeEntity($request);
return [
'url' => $request->get('url')
];
}
}
@@ -113,6 +117,14 @@ class User extends \Espo\Core\Controllers\Record
return $this->getRecordService()->generateNewApiKeyForEntity($data->id)->getValueMap();
}
public function postActionGenerateNewPassword($params, $data, $request)
{
if (empty($data->id)) throw new BadRequest();
if (!$this->getUser()->isAdmin()) throw new Forbidden();
$this->getRecordService()->generateNewPasswordForUser($data->id);
return true;
}
public function beforeCreateLink()
{
if (!$this->getUser()->isAdmin()) throw new Forbidden();

View File

@@ -0,0 +1,77 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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\Error;
use \Espo\Core\Exceptions\Forbidden;
use \Espo\Core\Exceptions\NotFound;
use \Espo\Core\Exceptions\BadRequest;
class UserSecurity extends \Espo\Core\Controllers\Base
{
protected function checkControllerAccess()
{
if (!$this->getUser()->isAdmin() && !$this->getUser()->isRegular()) {
throw new Forbidden();
}
}
public function getActionRead($params, $data, $request)
{
$id = $params['id'] ?? null;
if (!$id) throw new BadRequest();
if (!$this->getUser()->isAdmin() && $id !== $this->getUser()->id) throw new Forbidden();
return $this->getService('UserSecurity')->read($id);
}
public function postActionGenerate2FAData($params, $data)
{
$data = $data ?? (object) [];
$id = $data->id;
if (!$id) throw new BadRequest();
if (!$this->getUser()->isAdmin() && $id !== $this->getUser()->id) throw new Forbidden();
return $this->getService('UserSecurity')->generate2FAData($id, $data);
}
public function putActionUpdate($params, $data)
{
$id = $params['id'] ?? null;
$data = $data ?? (object) [];
if (!$id) throw new BadRequest();
if (!$this->getUser()->isAdmin() && $id !== $this->getUser()->id) throw new Forbidden();
return $this->getService('UserSecurity')->update($id, $data);
}
}

View File

@@ -0,0 +1,50 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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\Error;
use \Espo\Core\Exceptions\Forbidden;
class Webhook extends \Espo\Core\Controllers\Record
{
protected function checkControllerAccess()
{
if (!$this->getUser()->isAdmin() && !$this->getUser()->isApi()) {
throw new Forbidden();
}
}
public function actionCreate($params, $data, $request, $response = null)
{
$result = parent::actionCreate($params, $data, $request, $response);
if ($response) $response->setStatus(201);
return $result;
}
}

View File

@@ -60,21 +60,45 @@ class GlobalRestricton
public function __construct(
\Espo\Core\Utils\Metadata $metadata,
\Espo\Core\Utils\File\Manager $fileManager,
\Espo\Core\Utils\FieldManagerUtil $fieldManagerUtil
\Espo\Core\Utils\FieldManagerUtil $fieldManagerUtil,
bool $useCache = true
)
{
$this->metadata = $metadata;
$this->fileManager = $fileManager;
$this->fieldManagerUtil = $fieldManagerUtil;
if (!file_exists($this->cacheFilePath)) {
$this->buildCacheFile();
$isFromCache = false;
if ($useCache) {
if (file_exists($this->cacheFilePath)) {
$this->data = include($this->cacheFilePath);
$isFromCache = true;
if (!($this->data instanceof \StdClass)) {
$GLOBALS['log']->error("ACL GlobalRestricton: Bad data fetched from cache.");
$this->data = null;
}
}
}
$this->data = include($this->cacheFilePath);
if (!$this->data) {
$this->buildData();
}
if ($useCache) {
if (!$isFromCache) {
$this->storeCacheFile();
}
}
}
protected function buildCacheFile()
protected function storeCacheFile()
{
$this->getFileManager()->putPhpContents($this->cacheFilePath, $this->data, true);
}
protected function buildData()
{
$scopeList = array_keys($this->getMetadata()->get(['entityDefs'], []));
@@ -127,8 +151,6 @@ class GlobalRestricton
}
$this->data = $data;
$this->getFileManager()->putPhpContents($this->cacheFilePath, $data, true);
}
protected function getMetadata()

View File

@@ -88,7 +88,7 @@ class Table
if ($this->isStrictModeForced) {
$this->isStrictMode = true;
} else {
$this->isStrictMode = $config->get('aclStrictMode', false);
$this->isStrictMode = $config->get('aclStrictMode', true);
}
$this->user = $user;

View File

@@ -59,7 +59,8 @@ class AclManager
$this->globalRestricton = new \Espo\Core\Acl\GlobalRestricton(
$container->get('metadata'),
$container->get('fileManager'),
$container->get('fieldManagerUtil')
$container->get('fieldManagerUtil'),
$container->get('config')->get('useCache')
);
}

View File

@@ -104,6 +104,13 @@ class Base extends \Espo\Core\Acl\Base
}
if ($inAccount) {
return true;
} else {
if (is_null($isOwnContact) && $entity) {
$isOwnContact = $this->checkIsOwnContact($user, $entity);
}
if ($isOwnContact) {
return true;
}
}
}
@@ -117,7 +124,6 @@ class Base extends \Espo\Core\Acl\Base
}
return false;
}
public function checkReadOnlyAccount(User $user, $data)
@@ -152,13 +158,13 @@ class Base extends \Espo\Core\Acl\Base
{
$accountIdList = $user->getLinkMultipleIdList('accounts');
if (count($accountIdList)) {
if ($entity->hasAttribute('accountId')) {
if ($entity->hasAttribute('accountId') && $entity->getRelationParam('account', 'entity') === 'Account') {
if (in_array($entity->get('accountId'), $accountIdList)) {
return true;
}
}
if ($entity->hasRelation('accounts')) {
if ($entity->hasRelation('accounts') && $entity->getRelationParam('accounts', 'entity') === 'Account') {
$repository = $this->getEntityManager()->getRepository($entity->getEntityType());
foreach ($accountIdList as $accountId) {
if ($repository->isRelated($entity, 'accounts', $accountId)) {
@@ -183,13 +189,13 @@ class Base extends \Espo\Core\Acl\Base
{
$contactId = $user->get('contactId');
if ($contactId) {
if ($entity->hasAttribute('contactId')) {
if ($entity->hasAttribute('contactId') && $entity->getRelationParam('contact', 'entity') === 'Contact') {
if ($entity->get('contactId') === $contactId) {
return true;
}
}
if ($entity->hasRelation('contacts')) {
if ($entity->hasRelation('contacts') && $entity->getRelationParam('contacts', 'entity') === 'Contact') {
$repository = $this->getEntityManager()->getRepository($entity->getEntityType());
if ($repository->isRelated($entity, 'contacts', $contactId)) {
return true;
@@ -207,6 +213,4 @@ class Base extends \Espo\Core\Acl\Base
return false;
}
}

View File

@@ -133,7 +133,9 @@ class Application
$slim->run();
} catch (\Exception $e) {
$container->get('output')->processError($e->getMessage(), $e->getCode(), true, $e);
try {
$container->get('output')->processError($e->getMessage(), $e->getCode(), true, $e);
} catch (\Slim\Exception\Stop $e) {}
}
}
@@ -179,7 +181,7 @@ class Application
}
}
$processList = array_values($processList);
if (count($runningCount) >= $maxProcessNumber) {
if ($runningCount >= $maxProcessNumber) {
$toSkip = true;
}
if (!$toSkip) {

View File

@@ -29,9 +29,137 @@
namespace Espo\Core\Console\Commands;
use Espo\Core\Exceptions\Error;
class Upgrade extends Base
{
public function run()
protected $upgradeManager;
protected $upgradeStepList = [
'copyBefore',
'rebuild',
'beforeUpgradeScript',
'rebuild',
'copy',
'rebuild',
'copyAfter',
'rebuild',
'afterUpgradeScript',
'rebuild',
];
public function run($options, $flagList, $argumentList)
{
$params = $this->normalizeParams($options, $flagList, $argumentList);
switch ($params['mode']) {
case 'local':
$this->runLocalUpgrade($params);
break;
default:
case 'remote':
$this->runRemoteUpgrade($params);
break;
}
}
/**
* Normalize params. Permitted options and flags and $arguments:
* -y - without confirmation
* -s - single process
* --file="EspoCRM-upgrade.zip"
* --step="beforeUpgradeScript"
* @param array $options
* @param array $flagList
* @param array $argumentList
* @return array
*/
protected function normalizeParams($options, $flagList, $argumentList)
{
$params = [
'mode' => 'remote',
'skipConfirmation' => false,
'singleProcess' => false,
];
if (!empty($options['file'])) {
$params['mode'] = 'local';
$params['file'] = $options['file'];
}
if (in_array('y', $flagList)) {
$params['skipConfirmation'] = true;
}
if (in_array('s', $flagList)) {
$params['singleProcess'] = true;
}
if (!empty($options['step'])) {
$params['step'] = $options['step'];
}
return $params;
}
protected function runLocalUpgrade(array $params)
{
if (empty($params['file']) || !file_exists($params['file'])) {
echo "Upgrade package is not found.\n";
return;
}
$packageFile = $params['file'];
$fromVersion = $this->getConfig()->get('version');
fwrite(\STDOUT, "Current version is {$fromVersion}.\n");
$upgradeId = $this->upload($packageFile);
$manifest = $this->getUpgradeManager()->getManifestById($upgradeId);
$nextVersion = $manifest['version'];
if (!$params['skipConfirmation']) {
fwrite(\STDOUT, "EspoCRM will be upgraded to version {$nextVersion} now. Enter [Y] to continue.\n");
if (!$this->confirm()) {
echo "Upgrade canceled.\n";
return;
}
}
fwrite(\STDOUT, "Upgrading... This may take a while...");
try {
$this->runUpgradeProcess($upgradeId, $params);
} catch (\Exception $e) {
fwrite(\STDOUT, "\n");
fwrite(\STDOUT, $e->getMessage() . "\n");
return;
}
fwrite(\STDOUT, "\n");
$app = new \Espo\Core\Application();
$currentVerison = $app->getContainer()->get('config')->get('version');
fwrite(\STDOUT, "Upgrade is complete. Current version is {$currentVerison}.\n");
$infoData = $this->getVersionInfo();
$lastVersion = $infoData->lastVersion ?? null;
if ($lastVersion && $lastVersion !== $currentVerison && $fromVersion !== $currentVerison) {
fwrite(\STDOUT, "Newer version is available.\n");
return;
}
if ($lastVersion && $lastVersion === $currentVerison) {
fwrite(\STDOUT, "You have the latest version.\n");
return;
}
}
protected function runRemoteUpgrade(array $params)
{
$infoData = $this->getVersionInfo();
if (!$infoData) return;
@@ -48,11 +176,13 @@ class Upgrade extends Base
return;
}
fwrite(\STDOUT, "EspoCRM will be upgaded to version {$nextVersion} now. Type 'Y' to continue.\n");
if (!$params['skipConfirmation']) {
fwrite(\STDOUT, "EspoCRM will be upgraded to version {$nextVersion} now. Enter [Y] to continue.\n");
if (!$this->confirm()) {
echo "Upgrade canceled.\n";
return;
if (!$this->confirm()) {
echo "Upgrade canceled.\n";
return;
}
}
fwrite(\STDOUT, "Downloading...");
@@ -64,13 +194,22 @@ class Upgrade extends Base
fwrite(\STDOUT, "Upgrading... This may take a while...");
$this->upgrade($upgradePackageFilePath);
$upgradeId = $this->upload($upgradePackageFilePath);
try {
$this->runUpgradeProcess($upgradeId, $params);
} catch (\Exception $e) {
$error = $e->getMessage();
}
$this->getFileManager()->unlink($upgradePackageFilePath);
fwrite(\STDOUT, "\n");
fwrite(\STDOUT, $resultText);
$this->getFileManager()->unlink($upgradePackageFilePath);
if (!empty($error)) {
echo $error;
return;
}
$app = new \Espo\Core\Application();
$currentVerison = $app->getContainer()->get('config')->get('version');
@@ -88,22 +227,66 @@ class Upgrade extends Base
}
}
protected function upgrade($filePath)
protected function upload($filePath)
{
$app = new \Espo\Core\Application();
$app->setupSystemUser();
$upgradeManager = new \Espo\Core\UpgradeManager($app->getContainer());
try {
$fileData = file_get_contents($filePath);
$fileData = 'data:application/zip;base64,' . base64_encode($fileData);
$upgradeId = $upgradeManager->upload($fileData);
$upgradeManager->install(['id' => $upgradeId]);
$upgradeId = $this->getUpgradeManager()->upload($fileData);
} catch (\Exception $e) {
die("Error: " . $e->getMessage() . "\n");
}
return $upgradeId;
}
protected function runUpgradeProcess($upgradeId, array $params = [])
{
$useSingleProcess = array_key_exists('singleProcess', $params) ? $params['singleProcess'] : false;
$stepList = !empty($params['step']) ? [$params['step']] : $this->upgradeStepList;
array_unshift($stepList, 'init');
array_push($stepList, 'finalize');
if (!$useSingleProcess && $this->isShellEnabled()) {
return $this->runSteps($upgradeId, $stepList);
}
return $this->runStepsInSingleProcess($upgradeId, $stepList);
}
protected function runStepsInSingleProcess($upgradeId, array $stepList)
{
$GLOBALS['log']->debug('Installation process ['.$upgradeId.']: Single process mode.');
try {
foreach ($stepList as $stepName) {
$upgradeManager = $this->getUpgradeManager(true);
$upgradeManager->runInstallStep($stepName, ['id' => $upgradeId]);
}
} catch (\Exception $e) {
$GLOBALS['log']->error('Upgrade Error: ' . $e->getMessage());
throw new Error($e->getMessage());
}
return true;
}
protected function runSteps($upgradeId, array $stepList)
{
$phpExecutablePath = $this->getPhpExecutablePath();
foreach ($stepList as $stepName) {
$command = $phpExecutablePath . " command.php upgrade-step --step=". ucfirst($stepName) ." --id=". $upgradeId;
$shellResult = shell_exec($command);
if ($shellResult !== 'true') {
$GLOBALS['log']->error('Upgrade Error: ' . $shellResult);
throw new Error($shellResult);
}
}
return true;
}
protected function confirm()
@@ -127,6 +310,29 @@ class Upgrade extends Base
return $this->getContainer()->get('fileManager');
}
protected function getUpgradeManager($reload = false)
{
if (!$this->upgradeManager || $reload) {
$app = new \Espo\Core\Application();
$app->setupSystemUser();
$this->upgradeManager = new \Espo\Core\UpgradeManager($app->getContainer());
}
return $this->upgradeManager;
}
protected function getPhpExecutablePath()
{
$phpExecutablePath = $this->getConfig()->get('phpExecutablePath');
if (!$phpExecutablePath) {
$phpExecutablePath = (new \Symfony\Component\Process\PhpExecutableFinder)->find();
}
return $phpExecutablePath;
}
protected function getVersionInfo()
{
$url = 'https://s.espocrm.com/upgrade/next/';
@@ -183,4 +389,18 @@ class Upgrade extends Base
return realpath($localFilePath);
}
protected function isShellEnabled()
{
if (!function_exists('exec') || !is_callable('shell_exec')) {
return false;
}
$result = shell_exec("echo test");
if (empty($result)) {
return false;
}
return true;
}
}

View File

@@ -0,0 +1,71 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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\Console\Commands;
class UpgradeStep extends Base
{
public function run($options, $flagList, $argumentList)
{
if (empty($options['step'])) {
echo "Step is not specified.\n";
return;
}
if (empty($options['id'])) {
echo "Upgrade ID is not specified.\n";
return;
}
$stepName = $options['step'];
$upgradeId = $options['id'];
return $this->runUpgradeStep($stepName, ['id' => $upgradeId]);
}
protected function runUpgradeStep($stepName, array $params)
{
$app = new \Espo\Core\Application();
$app->setupSystemUser();
$upgradeManager = new \Espo\Core\UpgradeManager($app->getContainer());
try {
$result = $upgradeManager->runInstallStep($stepName, $params); // throw Exception on error
} catch (\Exception $e) {
die("Error: " . $e->getMessage() . "\n");
}
if (is_bool($result)) {
$result = $result ? "true" : "false";
}
return $result;
}
}

View File

@@ -291,7 +291,7 @@ class Container
'en_US',
$this->get('fileManager'),
$this->get('metadata'),
$this->get('useCache')
$this->get('config')->get('useCache')
);
}
@@ -301,7 +301,7 @@ class Container
\Espo\Core\Utils\Language::detectLanguage($this->get('config')),
$this->get('fileManager'),
$this->get('metadata'),
$this->get('useCache')
$this->get('config')->get('useCache')
);
}

View File

@@ -536,4 +536,50 @@ class Record extends Base
return $this->getRecordService()->restoreDeleted($id);
}
public function postActionMassConvertCurrency($params, $data, $request)
{
if (!$this->getAcl()->checkScope($this->name, 'edit')) throw new Forbidden();
if ($this->getAcl()->get('massUpdatePermission') !== 'yes') throw new Forbidden();
$fieldList = $data->fieldList ?? null;
if (!empty($data->field)) {
if (!is_array($fieldList)) $fieldList = [];
$fieldList[] = $data->field;
}
$params = [];
if (property_exists($data, 'where') && !empty($data->byWhere)) {
$params['where'] = json_decode(json_encode($data->where), true);
if (property_exists($data, 'selectData')) {
$params['selectData'] = json_decode(json_encode($data->selectData), true);
}
} else if (property_exists($data, 'ids')) {
$params['ids'] = $data->ids;
}
if (empty($data->currencyRates)) throw new BadRequest();
if (empty($data->targetCurrency)) throw new BadRequest();
if (empty($data->baseCurrency)) throw new BadRequest();
return $this->getRecordService()->massConvertCurrency($params, $data->targetCurrency, $data->baseCurrency, $data->currencyRates, $fieldList);
}
public function postActionConvertCurrency($params, $data, $request)
{
if (!$this->getAcl()->checkScope($this->name, 'edit')) throw new Forbidden();
$fieldList = $data->fieldList ?? null;
if (!empty($data->field)) {
if (!is_array($fieldList)) $fieldList = [];
$fieldList[] = $data->field;
}
if (empty($data->id)) throw new BadRequest();
if (empty($data->currencyRates)) throw new BadRequest();
if (empty($data->targetCurrency)) throw new BadRequest();
if (empty($data->baseCurrency)) throw new BadRequest();
return $this->getRecordService()->convertCurrency($data->id, $data->targetCurrency, $data->baseCurrency, $data->currencyRates, $fieldList);
}
}

View File

@@ -91,8 +91,10 @@ class DataManager
*/
public function rebuildDatabase($entityList = null)
{
$schema = $this->getContainer()->get('schema');
try {
$result = $this->getContainer()->get('schema')->rebuild($entityList);
$result = $schema->rebuild($entityList);
} catch (\Exception $e) {
$result = false;
$GLOBALS['log']->error('Fault to rebuild database schema'.'. Details: '.$e->getMessage());
@@ -102,6 +104,18 @@ class DataManager
throw new Exceptions\Error("Error while rebuilding database. See log file for details.");
}
$config = $this->getContainer()->get('config');
$databaseType = strtolower($schema->getDatabaseHelper()->getDatabaseType());
if (!$config->get('actualDatabaseType') || $config->get('actualDatabaseType') != $databaseType) {
$config->set('actualDatabaseType', $databaseType);
}
$databaseVersion = $schema->getDatabaseHelper()->getDatabaseVersion();
if (!$config->get('actualDatabaseVersion') || $config->get('actualDatabaseVersion') != $databaseVersion) {
$config->set('actualDatabaseVersion', $databaseVersion);
}
$this->updateCacheTimestamp();
return $result;
@@ -180,7 +194,7 @@ class DataManager
public function updateCacheTimestamp()
{
$this->getContainer()->get('config')->updateCacheTimestamp();
$this->getContainer()->get('config')->save();
$this->getContainer()->get('config')->save(); /* correct rebuildDatabase() method when remove this line */
return true;
}

View File

@@ -0,0 +1,35 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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\Exceptions;
class ConflictSilent extends Conflict
{
public $logLevel = 'notice';
}

View File

@@ -491,7 +491,14 @@ class Xlsx extends \Espo\Core\Injectable
$sheet->setCellValue("$col$rowNumber", $value);
}
} else if ($type == 'multiEnum' || $type == 'array') {
if (!empty($row[$name])) {
$array = json_decode($row[$name]);
if (is_array($array)) {
$value = implode(', ', $array);
$sheet->setCellValue("$col$rowNumber", $value, \PhpOffice\PhpSpreadsheet\Cell\DataType::TYPE_STRING);
}
}
} else {
if (array_key_exists($name, $row)) {
$sheet->setCellValueExplicit("$col$rowNumber", $row[$name], \PhpOffice\PhpSpreadsheet\Cell\DataType::TYPE_STRING);

View File

@@ -69,7 +69,13 @@ class ClientManager
$externalAccountEntity = $this->clientMap[$hash]['externalAccountEntity'];
$externalAccountEntity->set('accessToken', $data['accessToken']);
$externalAccountEntity->set('tokenType', $data['tokenType']);
$this->getEntityManager()->saveEntity($externalAccountEntity);
$copy = $this->getEntityManager()->getEntity('ExternalAccount', $externalAccountEntity->id);
if ($copy) {
$copy->set('accessToken', $data['accessToken']);
$copy->set('tokenType', $data['tokenType']);
$this->getEntityManager()->saveEntity($copy, ['isTokenRenewal' => true]);
}
}
}
@@ -87,7 +93,12 @@ class ClientManager
$className = $this->getMetadata()->get("integrations.{$integration}.clientClassName");
$redirectUri = $this->getConfig()->get('siteUrl') . '?entryPoint=oauthCallback'; // TODO move to client class
$redirectUri = $this->getConfig()->get('siteUrl') . '?entryPoint=oauthCallback';
$redirectUriPath = $this->getMetadata()->get(['integrations', $integration, 'params', 'redirectUriPath']);
if ($redirectUriPath) {
$redirectUri = rtrim($this->getConfig()->get('siteUrl'), '/') . '/' . $redirectUriPath;
}
if (!$externalAccountEntity) {
throw new Error("External Account {$integration} not found for {$userId}");

View File

@@ -187,7 +187,11 @@ class Client
$curlOptHttpHeader = array();
foreach ($httpHeaders as $key => $value) {
$curlOptHttpHeader[] = "{$key}: {$value}";
if (is_int($key) && !is_string($key)) {
$curlOptHttpHeader[] = $value;
continue;
}
$curlOptHttpHeader[] = "{$key}: {$value}";
}
$curlOptions[CURLOPT_HTTPHEADER] = $curlOptHttpHeader;

View File

@@ -0,0 +1,35 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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\FieldValidators;
class ChecklistType extends ArrayType
{
}

View File

@@ -65,7 +65,7 @@ class AddLinkMultipleIdType extends \Espo\Core\Formula\Functions\Base
}
} else {
if (!is_string($id)) {
throw new Error();
return;
}
$this->getEntity()->addLinkMultipleId($link, $id);
}

View File

@@ -0,0 +1,58 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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\Formula\Functions\EntityGroup;
use \Espo\ORM\Entity;
use \Espo\Core\Exceptions\Error;
class GetLinkColumnType extends \Espo\Core\Formula\Functions\Base
{
protected function init()
{
$this->addDependency('entityManager');
}
public function process(\StdClass $item)
{
$args = $item->value ?? [];
if (!is_array($args)) throw new Error();
if (count($args) < 3) throw new Error("Formula: entity\\isRelated: no argument.");
$link = $this->evaluate($args[0]);
$id = $this->evaluate($args[1]);
$column = $this->evaluate($args[2]);
$entityType = $this->getEntity()->getEntityType($entityType);
$repository = $this->getInjection('entityManager')->getRepository($entityType);
return $repository->getRelationColumn($this->getEntity(), $link, $id, $column);
}
}

View File

@@ -97,14 +97,48 @@ class SumRelatedType extends \Espo\Core\Formula\Functions\Base
$selectParams['select'] = [[$foreignLink . '.id', 'foreignId'], 'SUM:' . $field];
$foreignSelectManager->addJoin($foreignLink, $selectParams);
if ($entity->getRelationType($link) === 'hasChildren') {
$foreignSelectManager->addJoin([
$entity->getEntityType(),
$foreignLink,
[
$foreignLink . '.id:' => $foreignLink . 'Id',
'deleted' => false,
$foreignLink . '.id!=' => null,
]
], $selectParams);
$selectParams['whereClause'][] = [$foreignLink . 'Type' => $entity->getEntityType()];
} else {
$foreignSelectManager->addJoin($foreignLink, $selectParams);
}
if (!empty($selectParams['distinct'])) {
$sqSelectParams = $selectParams;
$sqSelectParams['whereClause'][] = [
$foreignLink . '.id' => $entity->id
];
$sqSelectParams['select'] = ['id'];
unset($sqSelectParams['distinct']);
unset($sqSelectParams['orderBy']);
unset($sqSelectParams['order']);
$selectParams['whereClause'][] = [
'id=s' => [
'entityType' => $foreignEntityType,
'selectParams' => $sqSelectParams,
]
];
} else {
$selectParams['whereClause'][] = [
$foreignLink . '.id' => $entity->id
];
}
$selectParams['groupBy'] = [$foreignLink . '.id'];
$selectParams['whereClause'][] = [
$foreignLink . '.id' => $entity->id
];
$entityManager->getRepository($foreignEntityType)->handleSelectParams($selectParams);
$sql = $entityManager->getQuery()->createSelectQuery($foreignEntityType, $selectParams);

View File

@@ -0,0 +1,54 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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\Formula\Functions\PasswordGroup;
use \Espo\ORM\Entity;
use \Espo\Core\Exceptions\Error;
class GenerateType extends \Espo\Core\Formula\Functions\Base
{
protected function init()
{
$this->addDependency('config');
}
public function process(\StdClass $item)
{
$config = $this->getInjection('config');
$length = $config->get('passwordGenerateLength', 10);
$letterCount = $config->get('passwordGenerateLetterCount', 4);
$numberCount = $config->get('passwordGenerateNumberCount', 2);
$password = \Espo\Core\Utils\Util::generatePassword($length, $letterCount, $numberCount, true);
return $password;
}
}

View File

@@ -0,0 +1,60 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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\Formula\Functions\PasswordGroup;
use \Espo\ORM\Entity;
use \Espo\Core\Exceptions\Error;
class HashType extends \Espo\Core\Formula\Functions\Base
{
protected function init()
{
$this->addDependency('config');
}
public function process(\StdClass $item)
{
$args = $item->value ?? [];
if (!is_array($args)) throw new Error();
if (count($args) < 1)
throw new Error("Formula: password\\hash: no argument.");
$password = $this->evaluate($args[0]);
if (!is_string($password))
throw new Error("Formula: password\\hash: bad argument.");
$passwordHash = new \Espo\Core\Utils\PasswordHash($this->getInjection('config'));
$hash = $passwordHash->hash($password);
return $hash;
}
}

View File

@@ -0,0 +1,69 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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\Formula\Functions\RecordGroup;
use Espo\Core\Exceptions\Error;
class AttributeType extends \Espo\Core\Formula\Functions\AttributeType
{
protected function init()
{
$this->addDependency('entityManager');
}
public function process(\StdClass $item)
{
if (!property_exists($item, 'value')) {
throw new Error();
}
if (!is_array($item->value)) {
throw new Error();
}
if (count($item->value) < 3) {
throw new Error();
}
$entityType = $this->evaluate($item->value[0]);
$id = $this->evaluate($item->value[1]);
$attribute = $this->evaluate($item->value[2]);
if (!$entityType) throw new Error("Formula record\\attribute: Empty entityType.");
if (!$id) return null;
if (!$attribute) throw new Error("Formula record\\attribute: Empty attribute.");
$entity = $this->getInjection('entityManager')->getEntity($entityType, $id);
if (!$entity) return null;
return $this->attributeFetcher->fetch($entity, $attribute);
}
}

View File

@@ -64,8 +64,13 @@ class CountType extends \Espo\Core\Formula\Functions\Base
$selectManager = $this->getInjection('selectManagerFactory')->create($entityType);
$selectParams = $selectManager->getEmptySelectParams();
if ($filter) {
$selectManager->applyFilter($filter, $selectParams);
if (is_string($filter)) {
$selectManager->applyFilter($filter, $selectParams);
} else {
throw new Error("Formula record\\count: Bad filter.");
}
}
return $this->getInjection('entityManager')->getRepository($entityType)->count($selectParams);

View File

@@ -0,0 +1,94 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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\Formula\Functions\RecordGroup;
use Espo\Core\Exceptions\Error;
class FindOneType extends \Espo\Core\Formula\Functions\Base
{
protected function init()
{
$this->addDependency('entityManager');
$this->addDependency('selectManagerFactory');
}
public function process(\StdClass $item)
{
if (!property_exists($item, 'value')) {
throw new Error();
}
if (!is_array($item->value)) {
throw new Error();
}
if (count($item->value) < 3) {
throw new Error();
}
$entityType = $this->evaluate($item->value[0]);
$orderBy = $this->evaluate($item->value[1]);
$order = $this->evaluate($item->value[2]) ?? 'asc';
$selectManager = $this->getInjection('selectManagerFactory')->create($entityType);
$selectParams = $selectManager->getEmptySelectParams();
if (count($item->value) <= 4) {
$filter = null;
if (count($item->value) == 4) {
$filter = $this->evaluate($item->value[3]);
}
if ($filter) {
if (!is_string($filter)) throw new Error("Formula record\\findOne: Bad filter.");
$selectManager->applyFilter($filter, $selectParams);
}
} else {
$whereClause = [];
$i = 3;
while ($i < count($item->value) - 1) {
$key = $this->evaluate($item->value[$i]);
$value = $this->evaluate($item->value[$i + 1]);
$whereClause[] = [$key => $value];
$i = $i + 2;
}
$selectParams['whereClause'] = $whereClause;
}
if ($orderBy) {
$selectManager->applyOrder($orderBy, $order, $selectParams);
}
$e = $this->getInjection('entityManager')->getRepository($entityType)->select(['id'])->findOne($selectParams);
if ($e) return $e->id;
return null;
}
}

View File

@@ -0,0 +1,121 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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\Formula\Functions\RecordGroup;
use Espo\Core\Exceptions\Error;
class FindRelatedOneType extends \Espo\Core\Formula\Functions\Base
{
protected function init()
{
$this->addDependency('entityManager');
$this->addDependency('selectManagerFactory');
}
public function process(\StdClass $item)
{
if (!property_exists($item, 'value')) {
throw new Error();
}
if (!is_array($item->value)) {
throw new Error();
}
if (count($item->value) < 5) {
throw new Error();
}
$entityManager = $this->getInjection('entityManager');
$entityType = $this->evaluate($item->value[0]);
$id = $this->evaluate($item->value[1]);
$link = $this->evaluate($item->value[2]);
$orderBy = $this->evaluate($item->value[3]);
$order = $this->evaluate($item->value[4]) ?? 'asc';
if (!$entityType) throw new Error("Formula record\\findRelatedOne: Empty entityType.");
if (!$id) return null;
if (!$link) throw new Error("Formula record\\findRelatedOne: Empty link.");
$entity = $entityManager->getEntity($entityType, $id);
if (!$entity) return null;
$relationType = $entity->getRelationParam($link, 'type');
$foreignEntityType = $entity->getRelationParam($link, 'entity');
if (!$foreignEntityType) throw new Error("Formula record\\findRelatedOne: Bad or not supported link '{$link}'.");
$foreignLink = $entity->getRelationParam($link, 'foreign');
if (!$foreignLink) throw new Error("Formula record\\findRelatedOne: Not supported link '{$link}'.");
$selectManager = $this->getInjection('selectManagerFactory')->create($foreignEntityType);
$selectParams = $selectManager->getEmptySelectParams();
if ($relationType === 'hasChildren') {
$selectParams['whereClause'][] = [$foreignLink . 'Id' => $entity->id];
$selectParams['whereClause'][] = [$foreignLink . 'Type' => $entity->getEntityType()];
} else {
$selectManager->addJoin($foreignLink, $selectParams);
$selectParams['whereClause'][] = [$foreignLink . '.id' => $entity->id];
}
if (count($item->value) <= 6) {
$filter = null;
if (count($item->value) == 6) {
$filter = $this->evaluate($item->value[3]);
}
if ($filter) {
if (!is_string($filter)) throw new Error("Formula record\\findRelatedOne: Bad filter.");
$selectManager->applyFilter($filter, $selectParams);
}
} else {
$i = 5;
while ($i < count($item->value) - 1) {
$key = $this->evaluate($item->value[$i]);
$value = $this->evaluate($item->value[$i + 1]);
$selectParams['whereClause'][] = [$key => $value];
$i = $i + 2;
}
}
if ($orderBy) {
$selectManager->applyOrder($orderBy, $order, $selectParams);
}
$e = $entityManager->getRepository($foreignEntityType)->select(['id'])->findOne($selectParams);
if ($e) return $e->id;
return null;
}
}

View File

@@ -0,0 +1,51 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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\Formula\Functions\StringGroup;
use \Espo\Core\Exceptions\Error;
class PosType extends \Espo\Core\Formula\Functions\Base
{
public function process(\StdClass $item)
{
$args = $item->value ?? [];
if (count($args) < 2) throw new Error("Bad arguments passed to function string\\pos.");
$string = $this->evaluate($args[0]);
$needle = $this->evaluate($args[1]);
if (!is_string($string)) {
return false;
}
return mb_strpos($string, $needle);
}
}

View File

@@ -115,10 +115,12 @@ class Parser
$braceCounter = 0;
for ($i = 0; $i < strlen($string); $i++) {
$isStringStart = false;
if ($string[$i] === "'" && ($i === 0 || $string[$i - 1] !== "\\")) {
if (!$isString) {
$isString = true;
$isSingleQuote = true;
$isStringStart = true;
} else {
if ($isSingleQuote) {
$isString = false;
@@ -127,6 +129,7 @@ class Parser
} else if ($string[$i] === "\"" && ($i === 0 || $string[$i - 1] !== "\\")) {
if (!$isString) {
$isString = true;
$isStringStart = true;
$isSingleQuote = false;
} else {
if (!$isSingleQuote) {
@@ -137,6 +140,8 @@ class Parser
if ($isString) {
if ($string[$i] === '(' || $string[$i] === ')') {
$modifiedString[$i] = '_';
} else if (!$isStringStart) {
$modifiedString[$i] = ' ';
}
} else {
if ($string[$i] === '(') {
@@ -176,6 +181,16 @@ class Parser
$this->processStrings($expression, $modifiedExpression, $splitterIndexList, true);
$this->stripComments($expression, $modifiedExpression);
foreach ($splitterIndexList as $i => $index) {
if ($expression[$index] !== ';') {
unset($splitterIndexList[$i]);
}
}
$splitterIndexList = array_values($splitterIndexList);
$expressionOutOfBraceList = [];
for ($i = 0; $i < strlen($modifiedExpression); $i++) {
@@ -402,6 +417,43 @@ class Parser
}
}
protected function stripComments(&$expression, &$modifiedExpression)
{
$commentIndexStart = null;
for ($i = 0; $i < strlen($modifiedExpression); $i++) {
if (is_null($commentIndexStart)) {
if ($modifiedExpression[$i] === '/' && $i < strlen($modifiedExpression) - 1 && $modifiedExpression[$i + 1] === '/') {
$commentIndexStart = $i;
}
} else {
if ($modifiedExpression[$i] === "\n" || $i === strlen($modifiedExpression) - 1) {
for ($j = $commentIndexStart; $j <= $i; $j++) {
$modifiedExpression[$j] = ' ';
$expression[$j] = ' ';
}
$commentIndexStart = null;
}
}
}
for ($i = 0; $i < strlen($modifiedExpression) - 1; $i++) {
if (is_null($commentIndexStart)) {
if ($modifiedExpression[$i] === '/' && $modifiedExpression[$i + 1] === '*') {
$commentIndexStart = $i;
}
} else {
if ($modifiedExpression[$i] === '*' && $modifiedExpression[$i + 1] === '/') {
for ($j = $commentIndexStart; $j <= $i + 1; $j++) {
$modifiedExpression[$j] = ' ';
$expression[$j] = ' ';
}
$commentIndexStart = null;
}
}
}
}
protected function parseArgumentListFromFunctionContent($functionContent)
{
$functionContent = trim($functionContent);

View File

@@ -44,6 +44,7 @@ abstract class Base implements Injectable
'metadata',
'aclManager',
'user',
'serviceFactory',
];
protected $dependencies = []; // for backward compatibility
@@ -118,4 +119,9 @@ abstract class Base implements Injectable
{
return $this->getInjection('metadata');
}
protected function getServiceFactory()
{
return $this->getInjection('serviceFactory');
}
}

View File

@@ -35,26 +35,33 @@ use Espo\Core\Exceptions\Error;
use Espo\Core\Utils\File\Manager as FileManager;
use Espo\Core\Utils\DateTime;
use Espo\Core\Utils\NumberUtil;
use Espo\Core\Utils\Config;
use Espo\Core\Utils\Language;
use Espo\Core\Utils\Metadata;
use Espo\ORM\EntityManager;
require('vendor/zordius/lightncandy/src/lightncandy.php');
class Htmlizer
{
protected $fileManager;
protected $dateTime;
protected $config;
protected $acl;
protected $entityManager;
protected $metadata;
protected $language;
public function __construct(FileManager $fileManager, DateTime $dateTime, NumberUtil $number, $acl = null, $entityManager = null, $metadata = null, $language = null)
public function __construct(
FileManager $fileManager,
DateTime $dateTime,
NumberUtil $number,
$acl = null,
?EntityManager $entityManager = null,
?Metadata $metadata = null,
?Language $language = null,
?Config $config = null
)
{
$this->fileManager = $fileManager;
$this->dateTime = $dateTime;
@@ -63,6 +70,7 @@ class Htmlizer
$this->entityManager = $entityManager;
$this->metadata = $metadata;
$this->language = $language;
$this->config = $config;
}
protected function getAcl()
@@ -92,7 +100,7 @@ class Htmlizer
return $value;
}
protected function getDataFromEntity(Entity $entity, $skipLinks = false, $level = 0)
protected function getDataFromEntity(Entity $entity, $skipLinks = false, $level = 0, ?string $template = null)
{
$data = $entity->toArray();
@@ -118,15 +126,32 @@ class Htmlizer
if (!$skipLinks && $level === 0) {
foreach ($relationList as $relation) {
if (!$entity->hasLinkMultipleField($relation)) continue;
$collection = null;
$collection = $entity->getLinkMultipleCollection($relation);
$data[$relation] = $collection;
if ($entity->hasLinkMultipleField($relation)) {
$toLoad = true;
$collection = $entity->getLinkCollection($relation);
} else {
if (
$template && $entity->getRelationType($relation, ['hasMany', 'manyMany', 'hasChildren']) &&
mb_stripos($template, '{{#each '.$relation.'}}') !== false
) {
$limit = 100;
if ($this->config) {
$limit = $this->config->get('htmlizerLinkLimit') ?? $limit;
}
$collection = $entity->getLinkCollection($relation, ['limit' => $limit]);
}
}
if ($collection) {
$data[$relation] = $collection;
}
}
}
foreach ($data as $key => $value) {
if ($value instanceof \Espo\ORM\EntityCollection) {
if ($value instanceof \Espo\ORM\ICollection) {
$skipAttributeList[] = $key;
$collection = $value;
$list = [];
@@ -147,6 +172,7 @@ class Htmlizer
}
$type = $entity->getAttributeType($attribute);
$fieldType = $entity->getAttributeParam($attribute, 'fieldType');
if ($type == Entity::DATETIME) {
if (!empty($data[$attribute])) {
@@ -194,6 +220,14 @@ class Htmlizer
unset($data[$attribute]);
}
if ($fieldType === 'currency' && $this->metadata) {
if ($entity->getAttributeParam($attribute, 'attributeRole') === 'currency') {
if ($currencyValue = $data[$attribute]) {
$data[$attribute . 'Symbol'] = $this->metadata->get(['app', 'currency', 'symbolMap', $currencyValue]);
}
}
}
if (array_key_exists($attribute, $data)) {
$keyRaw = $attribute . '_RAW';
$data[$keyRaw] = $data[$attribute];
@@ -271,7 +305,7 @@ class Htmlizer
}
}
return;
}
},
],
'hbhelpers' => [
'ifEqual' => function () {
@@ -291,7 +325,19 @@ class Htmlizer
} else {
return $context['inverse'] ? $context['inverse']() : '';
}
}
},
'ifInArray' => function () {
$args = func_get_args();
$context = $args[count($args) - 1];
$array = $args[1] ?? [];
if (in_array($args[0], $array)) {
return $context['fn']();
} else {
return $context['inverse'] ? $context['inverse']() : '';
}
},
]
]);
@@ -310,7 +356,7 @@ class Htmlizer
$this->fileManager->removeFile($fileName);
}
$data = $this->getDataFromEntity($entity, $skipLinks);
$data = $this->getDataFromEntity($entity, $skipLinks, 0, $template);
if (!array_key_exists('today', $data)) {
$data['today'] = $this->dateTime->getTodayString();

View File

@@ -0,0 +1,40 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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 Hasher extends Base
{
public function load()
{
return new \Espo\Core\Utils\Hasher(
$this->getContainer()->get('config')
);
}
}

View File

@@ -0,0 +1,40 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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 Totp extends Base
{
public function load()
{
return new \Espo\Core\Utils\Authentication\TwoFA\Utils\Totp(
$this->getContainer()->get('config')
);
}
}

View File

@@ -0,0 +1,43 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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 WebhookManager extends Base
{
public function load()
{
return new \Espo\Core\Webhook\Manager(
$this->getContainer()->get('config'),
$this->getContainer()->get('fileManager'),
$this->getContainer()->get('entityManager'),
$this->getContainer()->get('fieldManagerUtil')
);
}
}

View File

@@ -212,14 +212,18 @@ class Importer
if ($parser->checkMessageAttribute($message, 'in-Reply-To') && $parser->getMessageAttribute($message, 'in-Reply-To')) {
$arr = explode(' ', $parser->getMessageAttribute($message, 'in-Reply-To'));
$inReplyTo = $arr[0];
$replied = $this->getEntityManager()->getRepository('Email')->where(array(
'messageId' => $inReplyTo
))->findOne();
if ($replied) {
$email->set('repliedId', $replied->id);
$repliedTeamIdList = $replied->getLinkMultipleIdList('teams');
foreach ($repliedTeamIdList as $repliedTeamId) {
$email->addLinkMultipleId('teams', $repliedTeamId);
if ($inReplyTo) {
if ($inReplyTo[0] !== '<') $inReplyTo = '<' . $inReplyTo . '>';
$replied = $this->getEntityManager()->getRepository('Email')->where(array(
'messageId' => $inReplyTo
))->findOne();
if ($replied) {
$email->set('repliedId', $replied->id);
$repliedTeamIdList = $replied->getLinkMultipleIdList('teams');
foreach ($repliedTeamIdList as $repliedTeamId) {
$email->addLinkMultipleId('teams', $repliedTeamId);
}
}
}
}

View File

@@ -91,7 +91,13 @@ class MailMimeParser
public function getMessageMessageId($message)
{
return $this->getMessageAttribute($message, 'Message-ID');
$messageId = $this->getMessageAttribute($message, 'Message-ID');
if ($messageId && strlen($messageId) && $messageId[0] !== '<') {
$messageId = '<' . $messageId . '>';
}
return $messageId;
}
public function getAddressNameMap($message)

View File

@@ -107,7 +107,12 @@ class Sender
if ($params['auth']) {
if (!empty($params['smtpAuthMechanism'])) {
$options['connectionClass'] = $params['smtpAuthMechanism'];
$smtpAuthMechanism = preg_replace("([\.]{2,})", '', $params['smtpAuthMechanism']);
if (in_array($smtpAuthMechanism, ['login', 'crammd5', 'plain'])) {
$options['connectionClass'] = $smtpAuthMechanism;
} else {
$options['connectionClass'] = 'login';
}
} else {
$options['connectionClass'] = 'login';
}
@@ -172,7 +177,7 @@ class Sender
return $this;
}
public function send(Email $email, $params = [], &$message = null, $attachmentList = [])
public function send(Email $email, $params = [], $message = null, $attachmentList = [])
{
if (!$message) {
$message = new Message();
@@ -284,6 +289,7 @@ class Sender
$contents = $a->get('contents');
} else {
$fileName = $this->getEntityManager()->getRepository('Attachment')->getFilePath($a);
if (!is_file($fileName)) continue;
$contents = file_get_contents($fileName);
}
$attachment = new MimePart($contents);
@@ -303,6 +309,7 @@ class Sender
$contents = $a->get('contents');
} else {
$fileName = $this->getEntityManager()->getRepository('Attachment')->getFilePath($a);
if (!is_file($fileName)) continue;
$contents = file_get_contents($fileName);
}
$attachment = new MimePart($contents);
@@ -386,6 +393,9 @@ class Sender
if (empty($messageId) || !is_string($messageId) || strlen($messageId) < 4 || strpos($messageId, 'dummy:') === 0) {
$messageId = $this->generateMessageId($email);
$email->set('messageId', '<' . $messageId . '>');
if ($email->id) {
$this->getEntityManager()->saveEntity($email, ['silent' => true]);
}
} else {
$messageId = substr($messageId, 1, strlen($messageId) - 2);
}
@@ -399,6 +409,7 @@ class Sender
$email->set('status', 'Sent');
$email->set('dateSent', date("Y-m-d H:i:s"));
} catch (\Exception $e) {
$this->useGlobal();
throw new Error($e->getMessage(), 500);
}

View File

@@ -53,6 +53,9 @@ class NotificatorFactory extends InjectableFactory
}
}
return $this->createByClassName($className);
$obj = $this->createByClassName($className);
$obj->setEntityType($entityType);
return $obj;
}
}

View File

@@ -42,6 +42,10 @@ class Base implements Injectable
protected $injections = [];
private $userIdEnabledMap = [];
protected $entityType;
public static $order = 9;
public function __construct()
@@ -65,6 +69,11 @@ class Base implements Injectable
$this->dependencyList[] = $name;
}
public function setEntityType(string $entityType)
{
$this->entityType = $entityType;
}
public function getDependencyList()
{
return $this->dependencyList;
@@ -113,6 +122,8 @@ class Base implements Injectable
protected function processForUser(Entity $entity, $assignedUserId)
{
if (!$this->isNotificationsEnabledForUser($assignedUserId)) return;
if ($entity->hasAttribute('createdById') && $entity->hasAttribute('modifiedById')) {
if ($entity->isNew()) {
$isNotSelfAssignment = $assignedUserId !== $entity->get('createdById');
@@ -125,18 +136,36 @@ class Base implements Injectable
if (!$isNotSelfAssignment) return;
$notification = $this->getEntityManager()->getEntity('Notification');
$notification->set(array(
$notification->set([
'type' => 'Assign',
'userId' => $assignedUserId,
'data' => array(
'data' => [
'entityType' => $entity->getEntityType(),
'entityId' => $entity->id,
'entityName' => $entity->get('name'),
'isNew' => $entity->isNew(),
'userId' => $this->getUser()->id,
'userName' => $this->getUser()->get('name')
)
));
'userName' => $this->getUser()->get('name'),
]
]);
$this->getEntityManager()->saveEntity($notification);
}
protected function isNotificationsEnabledForUser(string $userId)
{
if (!array_key_exists($userId, $this->userIdEnabledMap)) {
$preferences = $this->getEntityManager()->getEntity('Preferences', $userId);
$isEnabled = false;
if ($preferences) {
$isEnabled = true;
$ignoreList = $preferences->get('assignmentNotificationsIgnoreEntityTypeList') ?? [];
if (in_array($this->entityType, $ignoreList)) {
$isEnabled = false;
}
}
$this->userIdEnabledMap[$userId] = $isEnabled;
}
return $this->userIdEnabledMap[$userId];
}
}

View File

@@ -74,22 +74,33 @@ class Entity extends \Espo\ORM\Entity
}
}
public function getLinkMultipleCollection($field)
public function getLinkCollection(string $link, ?array $selectParams = null)
{
if (!$this->hasLinkMultipleField($field)) return;
if (!$selectParams) $selectParams = [];
$defs = $this->getRelationSelectParams($field);
$relSelectParams = $this->getRelationSelectParams($link);
$columnAttribute = $field . 'Columns';
$selectParams = array_merge($selectParams, $relSelectParams);
$selectParams['returnSthCollection'] = true;
$columnAttribute = $link . 'Columns';
if ($this->hasAttribute($columnAttribute) && $this->getAttributeParam($columnAttribute, 'columns')) {
$defs['additionalColumns'] = $this->getAttributeParam($columnAttribute, 'columns');
$selectParams['additionalColumns'] = $this->getAttributeParam($columnAttribute, 'columns');
}
$collection = $this->get($field, $defs);
$collection = $this->get($link, $selectParams);
return $collection;
}
public function getLinkMultipleCollection(string $link, ?array $selectParams = null)
{
if (!$this->hasLinkMultipleField($link)) return;
return $this->getLinkCollection($link, $selectParams);
}
protected function getRelationSelectParams($link)
{
$field = $link;

View File

@@ -197,10 +197,10 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
protected function afterMassRelate(Entity $entity, $relationName, array $params = [], array $options = [])
{
if (!$this->hooksDisabled && empty($options['skipHooks'])) {
$hookData = array(
$hookData = [
'relationName' => $relationName,
'relationParams' => $params
);
'relationParams' => $params,
];
$this->getEntityManager()->getHookManager()->process($this->entityType, 'afterMassRelate', $entity, $options, $hookData);
}
}
@@ -215,14 +215,24 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
{
parent::afterRelate($entity, $relationName, $foreign, $data, $options);
if ($foreign instanceof Entity) {
$foreignEntity = $foreign;
if (!$this->hooksDisabled && empty($options['skipHooks'])) {
$hookData = array(
if (!$this->hooksDisabled && empty($options['skipHooks'])) {
if (is_string($foreign)) {
$foreignId = $foreign;
$foreignEntityType = $entity->getRelationParam($relationName, 'entity');
if ($foreignEntityType) {
$foreign = $this->getEntityManager()->getEntity($foreignEntityType);
$foreign->id = $foreignId;
$foreign->setAsFetched();
}
}
if ($foreign instanceof Entity) {
$hookData = [
'relationName' => $relationName,
'relationData' => $data,
'foreignEntity' => $foreignEntity
);
'foreignEntity' => $foreign,
'foreignId' => $foreign->id,
];
$this->getEntityManager()->getHookManager()->process($this->entityType, 'afterRelate', $entity, $options, $hookData);
}
}
@@ -232,13 +242,23 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
{
parent::afterUnrelate($entity, $relationName, $foreign, $options);
if ($foreign instanceof Entity) {
$foreignEntity = $foreign;
if (!$this->hooksDisabled && empty($options['skipHooks'])) {
$hookData = array(
if (!$this->hooksDisabled && empty($options['skipHooks'])) {
if (is_string($foreign)) {
$foreignId = $foreign;
$foreignEntityType = $entity->getRelationParam($relationName, 'entity');
if ($foreignEntityType) {
$foreign = $this->getEntityManager()->getEntity($foreignEntityType);
$foreign->id = $foreignId;
$foreign->setAsFetched();
}
}
if ($foreign instanceof Entity) {
$hookData = [
'relationName' => $relationName,
'foreignEntity' => $foreignEntity
);
'foreignEntity' => $foreign,
'foreignId' => $foreign->id,
];
$this->getEntityManager()->getHookManager()->process($this->entityType, 'afterUnrelate', $entity, $options, $hookData);
}
}
@@ -301,7 +321,9 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
$entity->set('modifiedAt', $nowString);
}
if ($entity->hasAttribute('createdById')) {
if (empty($options['skipCreatedBy']) && (empty($options['import']) || !$entity->has('createdById'))) {
if (!empty($options['createdById'])) {
$entity->set('createdById', $options['createdById']);
} else if (empty($options['skipCreatedBy']) && (empty($options['import']) || !$entity->has('createdById'))) {
if ($this->getEntityManager()->getUser()) {
$entity->set('createdById', $this->getEntityManager()->getUser()->id);
}
@@ -313,7 +335,9 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
$entity->set('modifiedAt', $nowString);
}
if ($entity->hasAttribute('modifiedById')) {
if ($this->getEntityManager()->getUser()) {
if (!empty($options['modifiedById'])) {
$entity->set('modifiedById', $options['modifiedById']);
} else if ($this->getEntityManager()->getUser()) {
$entity->set('modifiedById', $this->getEntityManager()->getUser()->id);
$entity->set('modifiedByName', $this->getEntityManager()->getUser()->get('name'));
}

View File

@@ -240,4 +240,33 @@ class Tcpdf extends \TCPDF
$this->_out($out);
}
public function Output($name = 'doc.pdf', $dest = 'I')
{
if ($dest === 'I' && !$this->sign && php_sapi_name() != 'cli') {
if ($this->state < 3) {
$this->Close();
}
$name = preg_replace('/[\s]+/', '_', $name);
$name = \Espo\Core\Utils\Util::sanitizeFileName($name);
if (ob_get_contents()) {
$this->Error('Some data has already been output, can\'t send PDF file');
}
header('Content-Type: application/pdf');
if (headers_sent()) {
$this->Error('Some data has already been output to browser, can\'t send PDF file');
}
header('Cache-Control: private, must-revalidate, post-check=0, pre-check=0, max-age=1');
header('Pragma: public');
header('Expires: Sat, 26 Jul 1997 05:00:00 GMT');
header('Last-Modified: '.gmdate('D, d M Y H:i:s').' GMT');
header('Content-Disposition: inline; filename="'.$name.'"');
TCPDF_STATIC::sendOutputData($this->getBuffer(), $this->bufferlen);
return '';
}
return parent::Output($name, $dest);
}
}

View File

@@ -50,9 +50,9 @@ 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(
$portal = $this->getContainer()->get('entityManager')->getRepository('Portal')->where([
'customId' => $portalId
))->findOne();
])->findOne();
}
if (!$portal) {

View File

@@ -94,7 +94,7 @@ class Container extends \Espo\Core\Container
\Espo\Core\Utils\Language::detectLanguage($this->get('config'), $this->get('preferences')),
$this->get('fileManager'),
$this->get('metadata'),
$this->get('useCache')
$this->get('config')->get('useCache')
);
$language->setPortal($this->get('portal'));
return $language;

View File

@@ -34,8 +34,11 @@ use \Espo\Core\Utils\Json;
class Layout extends \Espo\Core\Utils\Layout
{
public function get($scope, $name)
public function get(string $scope, string $name) : ?string
{
$originalScope = $scope;
$originalName = $name;
$scope = $this->sanitizeInput($scope);
$name = $this->sanitizeInput($name);
@@ -43,87 +46,16 @@ class Layout extends \Espo\Core\Utils\Layout
return Json::encode($this->changedData[$scope][$name]);
}
$fileFullPath = Util::concatPath($this->getLayoutPath($scope, true), 'portal/' . $name . '.json');
$filePath = Util::concatPath($this->getLayoutPath($scope, true), 'portal/' . $name . '.json');
if (!file_exists($fileFullPath)) {
$fileFullPath = Util::concatPath($this->getLayoutPath($scope), 'portal/' . $name . '.json');
}
if (!file_exists($fileFullPath)) {
$fileFullPath = Util::concatPath($this->getLayoutPath($scope, true), $name . '.json');
}
if (!file_exists($fileFullPath)) {
$fileFullPath = Util::concatPath($this->getLayoutPath($scope), $name . '.json');
if (!file_exists($filePath)) {
$filePath = Util::concatPath($this->getLayoutPath($scope), 'portal/' . $name . '.json');
}
if (!file_exists($fileFullPath)) {
$defaultPath = $this->params['defaultsPath'];
$fileFullPath = Util::concatPath(Util::concatPath($defaultPath, 'layouts'), $name . '.json' );
if (!file_exists($fileFullPath)) {
return false;
}
if (file_exists($filePath)) {
return $this->getFileManager()->getContents($filePath);
}
return $this->getFileManager()->getContents($fileFullPath);
return parent::get($originalScope, $originalName);
}
public function set($data, $scope, $name)
{
$scope = $this->sanitizeInput($scope);
$name = $this->sanitizeInput($name);
if (empty($scope) || empty($name)) {
return false;
}
$this->changedData[$scope][$name] = $data;
}
public function resetToDefault($scope, $name)
{
$scope = $this->sanitizeInput($scope);
$name = $this->sanitizeInput($name);
$filePath = 'custom/Espo/Custom/Resources/layouts/' . $scope . '/' . $name . '.json';
if ($this->getFileManager()->isFile($filePath)) {
$this->getFileManager()->removeFile($filePath);
}
if (!empty($this->changedData[$scope]) && !empty($this->changedData[$scope][$name])) {
unset($this->changedData[$scope][$name]);
}
return $this->get($scope, $name);
}
/**
* Save changes
*
* @return bool
*/
public function save()
{
$result = true;
if (!empty($this->changedData)) {
foreach ($this->changedData as $scope => $rowData) {
foreach ($rowData as $layoutName => $layoutData) {
if (empty($scope) || empty($layoutName)) {
continue;
}
$layoutPath = $this->getLayoutPath($scope, true);
$data = Json::encode($layoutData, JSON_PRETTY_PRINT | JSON_UNESCAPED_UNICODE);
$result &= $this->getFileManager()->putContents(array($layoutPath, $layoutName.'.json'), $data);
}
}
}
if ($result == true) {
$this->clearChanges();
}
return (bool) $result;
}
}

View File

@@ -70,10 +70,19 @@ class Base
protected $selectAttributesDependancyMap = [];
const MIN_LENGTH_FOR_CONTENT_SEARCH = 4;
protected $fullTextOrderType = self::FT_ORDER_COMBINTED;
protected $fullTextRelevanceThreshold = null;
const FT_ORDER_COMBINTED = 0;
const FT_ORDER_RELEVANCE = 1;
const FT_ORDER_ORIGINAL = 3;
const MIN_LENGTH_FOR_CONTENT_SEARCH = 4;
const MIN_LENGTH_FOR_FULL_TEXT_SEARCH = 4;
protected $fullTextOrderRelevanceDivider = 5;
protected $fullTextSearchDataCacheHash = [];
public function __construct(EntityManager $entityManager, \Espo\Entities\User $user, Acl $acl, AclManager $aclManager, Metadata $metadata, Config $config, FieldManagerUtil $fieldManagerUtil, InjectableFactory $injectableFactory)
@@ -184,6 +193,12 @@ class Base
$result['orderBy'] = 'LIST:' . $sortBy . ':' . implode(',', $list);
return;
}
} else {
if (strpos($sortBy, '.') === false && strpos($sortBy, ':') === false) {
if (!$this->getSeed()->hasAttribute($sortBy)) {
throw new Error("Order by non-existing field '{$sortBy}'.");
}
}
}
}
if (!$desc) {
@@ -216,16 +231,15 @@ class Base
{
$this->prepareResult($result);
$boolFilterList = [];
foreach ($where as $item) {
if (!isset($item['type'])) continue;
if ($item['type'] == 'bool' && !empty($item['value']) && is_array($item['value'])) {
$boolOr = [];
foreach ($item['value'] as $filter) {
$p = $this->getBoolFilterWhere($filter);
if (!empty($p)) {
$where[] = $p;
}
$this->applyBoolFilter($filter, $result);
$boolFilterList[] = $filter;
}
} else if ($item['type'] == 'textFilter') {
if (isset($item['value']) || $item['value'] !== '') {
@@ -236,6 +250,10 @@ class Base
}
}
if (count($boolFilterList)) {
$this->applyBoolFilterList($boolFilterList, $result);
}
$whereClause = $this->convertWhere($where, false, $result);
$result['whereClause'] = array_merge($result['whereClause'], $whereClause);
@@ -351,12 +369,16 @@ class Base
if ($relationType == 'belongsTo') {
$key = $seed->getRelationParam($link, 'key');
$aliasName = 'usersTeams' . ucfirst($link);
$aliasName = 'usersTeams' . ucfirst($link) . strval(rand(10000, 99999));
$result['customJoin'] .= "
JOIN team_user AS {$aliasName}Middle ON {$aliasName}Middle.user_id = ".$query->toDb($seed->getEntityType()).".".$query->toDb($key)." AND {$aliasName}Middle.deleted = 0
JOIN team AS {$aliasName} ON {$aliasName}.deleted = 0 AND {$aliasName}Middle.team_id = {$aliasName}.id
";
$this->addLeftJoin([
'TeamUser',
$aliasName . 'Middle',
[
$aliasName . 'Middle.userId:' => $key,
$aliasName . 'Middle.deleted' => false,
]
], $result);
$result['whereClause'][] = [
$aliasName . 'Middle.teamId' => $idsValue
@@ -552,16 +574,16 @@ class Base
return;
}
$d = [
$or = [
'teamsAccess.id' => $this->getUser()->getLinkMultipleIdList('teams')
];
if ($this->hasAssignedUserField()) {
$d['assignedUserId'] = $this->getUser()->id;
$or['assignedUserId'] = $this->getUser()->id;
} else if ($this->hasCreatedByField()) {
$d['createdById'] = $this->getUser()->id;
$or['createdById'] = $this->getUser()->id;
}
$result['whereClause'][] = [
'OR' => $d
'OR' => $or
];
}
@@ -580,38 +602,42 @@ class Base
protected function accessPortalOnlyContact(&$result)
{
$d = [];
$or = [];
$contactId = $this->getUser()->get('contactId');
if ($contactId) {
if ($this->getSeed()->hasAttribute('contactId')) {
$d['contactId'] = $contactId;
if (
$this->getSeed()->hasAttribute('contactId') && $this->getSeed()->getRelationParam('contact', 'entity') === 'Contact'
) {
$or['contactId'] = $contactId;
}
if ($this->getSeed()->hasRelation('contacts')) {
if (
$this->getSeed()->hasRelation('contacts') && $this->getSeed()->getRelationParam('contacts', 'entity') === 'Contact'
) {
$this->addLeftJoin(['contacts', 'contactsAccess'], $result);
$this->setDistinct(true, $result);
$d['contactsAccess.id'] = $contactId;
$or['contactsAccess.id'] = $contactId;
}
}
if ($this->getSeed()->hasAttribute('createdById')) {
$d['createdById'] = $this->getUser()->id;
$or['createdById'] = $this->getUser()->id;
}
if ($this->getSeed()->hasAttribute('parentId') && $this->getSeed()->hasRelation('parent')) {
$contactId = $this->getUser()->get('contactId');
if ($contactId) {
$d[] = [
$or[] = [
'parentType' => 'Contact',
'parentId' => $contactId
];
}
}
if (!empty($d)) {
if (!empty($or)) {
$result['whereClause'][] = [
'OR' => $d
'OR' => $or
];
} else {
$result['whereClause'][] = [
@@ -622,27 +648,31 @@ class Base
protected function accessPortalOnlyAccount(&$result)
{
$d = [];
$or = [];
$accountIdList = $this->getUser()->getLinkMultipleIdList('accounts');
$contactId = $this->getUser()->get('contactId');
if (count($accountIdList)) {
if ($this->getSeed()->hasAttribute('accountId')) {
$d['accountId'] = $accountIdList;
if (
$this->getSeed()->hasAttribute('accountId') && $this->getSeed()->getRelationParam('account', 'entity') === 'Account'
) {
$or['accountId'] = $accountIdList;
}
if ($this->getSeed()->hasRelation('accounts')) {
if (
$this->getSeed()->hasRelation('accounts') && $this->getSeed()->getRelationParam('accounts', 'entity') === 'Account'
) {
$this->addLeftJoin(['accounts', 'accountsAccess'], $result);
$this->setDistinct(true, $result);
$d['accountsAccess.id'] = $accountIdList;
$or['accountsAccess.id'] = $accountIdList;
}
if ($this->getSeed()->hasAttribute('parentId') && $this->getSeed()->hasRelation('parent')) {
$d[] = [
$or[] = [
'parentType' => 'Account',
'parentId' => $accountIdList
];
if ($contactId) {
$d[] = [
$or[] = [
'parentType' => 'Contact',
'parentId' => $contactId
];
@@ -651,23 +681,27 @@ class Base
}
if ($contactId) {
if ($this->getSeed()->hasAttribute('contactId')) {
$d['contactId'] = $contactId;
if (
$this->getSeed()->hasAttribute('contactId') && $this->getSeed()->getRelationParam('contact', 'entity') === 'Contact'
) {
$or['contactId'] = $contactId;
}
if ($this->getSeed()->hasRelation('contacts')) {
if (
$this->getSeed()->hasRelation('contacts') && $this->getSeed()->getRelationParam('contacts', 'entity') === 'Contact'
) {
$this->addLeftJoin(['contacts', 'contactsAccess'], $result);
$this->setDistinct(true, $result);
$d['contactsAccess.id'] = $contactId;
$or['contactsAccess.id'] = $contactId;
}
}
if ($this->getSeed()->hasAttribute('createdById')) {
$d['createdById'] = $this->getUser()->id;
$or['createdById'] = $this->getUser()->id;
}
if (!empty($d)) {
if (!empty($or)) {
$result['whereClause'][] = [
'OR' => $d
'OR' => $or
];
} else {
$result['whereClause'][] = [
@@ -756,7 +790,7 @@ class Base
$this->order($orderBy, $isDesc, $result);
} else if (!empty($params['order'])) {
$orderBy = $this->getMetadata()->get(['entityDefs', $this->getEntityType(), 'collection', 'orderBy']);
$orderBy = $this->getMetadata()->get(['entityDefs', $this->entityType, 'collection', 'orderBy']);
$isDesc = $params['order'] === 'desc';
$this->order($orderBy, $isDesc, $result);
}
@@ -774,9 +808,7 @@ class Base
}
if (!empty($params['boolFilterList']) && is_array($params['boolFilterList'])) {
foreach ($params['boolFilterList'] as $filterName) {
$this->applyBoolFilter($filterName, $result);
}
$this->applyBoolFilterList($params['boolFilterList'], $result);
}
if (!empty($params['filterList']) && is_array($params['filterList'])) {
@@ -807,6 +839,22 @@ class Base
return $result;
}
public function applyDefaultOrder(array &$result)
{
$orderBy = $this->getMetadata()->get(['entityDefs', $this->entityType, 'collection', 'orderBy']);
$order = $result['order'] ?? null;
if (!$order && !is_array($orderBy)) {
$order = $this->getMetadata()->get(['entityDefs', $this->entityType, 'collection', 'order']) ?? null;
}
if ($orderBy) {
$this->applyOrder($orderBy, $order, $result);
} else {
$result['order'] = $order;
}
}
public function checkWhere(array $where, bool $checkWherePermission = true, bool $forbidComplexExpressions = false)
{
foreach ($where as $w) {
@@ -1602,11 +1650,12 @@ class Base
break;
case 'isNotLinked':
if (!$result) break;
$alias = $attribute . 'IsNotLinkedFilter' . strval(rand(10000, 99999));
$part[$alias . '.id'] = null;
$this->setDistinct(true, $result);
$this->addLeftJoin([$attribute, $alias], $result);
$part['id!=s'] = [
'selectParams' => [
'select' => ['id'],
'joins' => [$attribute],
]
];
break;
case 'isLinked':
@@ -1696,6 +1745,9 @@ class Base
case 'arrayNoneOf':
case 'arrayIsEmpty':
case 'arrayIsNotEmpty':
case 'arrayAllOf':
if (!$result) break;
$arrayValueAlias = 'arrayFilter' . strval(rand(10000, 99999));
$arrayAttribute = $attribute;
$arrayEntityType = $this->getEntityType();
@@ -1705,7 +1757,16 @@ class Base
list($arrayAttributeLink, $arrayAttribute) = explode('.', $attribute);
$seed = $this->getSeed();
$arrayEntityType = $seed->getRelationParam($arrayAttributeLink, 'entity');
$idPart = $arrayAttributeLink . '.id';
$arrayLinkAlias = $arrayAttributeLink . 'Filter' . strval(rand(10000, 99999));
$idPart = $arrayLinkAlias . '.id';
$this->addLeftJoin([$arrayAttributeLink, $arrayLinkAlias], $result);
$relationType = $seed->getRelationType($arrayAttributeLink);
if ($relationType === 'manyMany' || $relationType === 'hasMany') {
$this->setDistinct(true, $result);
}
}
if ($type === 'arrayAnyOf') {
@@ -1716,6 +1777,8 @@ class Base
$arrayValueAlias . '.attribute' => $arrayAttribute
]], $result);
$part[$arrayValueAlias . '.value'] = $value;
$this->setDistinct(true, $result);
} else if ($type === 'arrayNoneOf') {
if (is_null($value) || !$value && !is_array($value)) break;
$this->addLeftJoin(['ArrayValue', $arrayValueAlias, [
@@ -1725,6 +1788,8 @@ class Base
$arrayValueAlias . '.value=' => $value
]], $result);
$part[$arrayValueAlias . '.id'] = null;
$this->setDistinct(true, $result);
} else if ($type === 'arrayIsEmpty') {
$this->addLeftJoin(['ArrayValue', $arrayValueAlias, [
$arrayValueAlias . '.entityId:' => $idPart,
@@ -1732,6 +1797,8 @@ class Base
$arrayValueAlias . '.attribute' => $arrayAttribute
]], $result);
$part[$arrayValueAlias . '.id'] = null;
$this->setDistinct(true, $result);
} else if ($type === 'arrayIsNotEmpty') {
$this->addLeftJoin(['ArrayValue', $arrayValueAlias, [
$arrayValueAlias . '.entityId:' => $idPart,
@@ -1739,9 +1806,31 @@ class Base
$arrayValueAlias . '.attribute' => $arrayAttribute
]], $result);
$part[$arrayValueAlias . '.id!='] = null;
}
$this->setDistinct(true, $result);
$this->setDistinct(true, $result);
} else if ($type === 'arrayAllOf') {
if (is_null($value) || !$value && !is_array($value)) break;
if (!is_array($value)) {
$value = [$value];
}
foreach ($value as $arrayValue) {
$part[] = [
$idPart .'=s' => [
'entityType' => 'ArrayValue',
'selectParams' => [
'select' => ['entityId'],
'whereClause' => [
'value' => $arrayValue,
'attribute' => $arrayAttribute,
'entityType' => $arrayEntityType,
],
],
]
];
}
}
}
}
@@ -1760,42 +1849,70 @@ class Base
$this->limit($offset, $maxSize, $result);
}
public function applyPrimaryFilter(string $filterName, array &$result)
public function applyPrimaryFilter(string $filter, array &$result)
{
$this->prepareResult($result);
$method = 'filter' . ucfirst($filterName);
$method = 'filter' . ucfirst($filter);
if (method_exists($this, $method)) {
$this->$method($result);
} else {
$className = $this->getMetadata()->get(['entityDefs', $this->entityType, 'collection', 'filters', $filterName, 'className']);
$className = $this->getMetadata()->get(['entityDefs', $this->entityType, 'collection', 'filters', $filter, 'className']);
if ($className) {
if (!class_exists($className)) {
$GLOBALS['log']->error("Could find class for filter {$filterName}.");
$GLOBALS['log']->error("Could find class for filter {$filter}.");
return;
}
$impl = $this->getInjectableFactory()->createByClassName($className);
if (!$impl) {
$GLOBALS['log']->error("Could not create filter {$filterName} implementation.");
$GLOBALS['log']->error("Could not create filter {$filter} implementation.");
return;
}
$impl->applyFilter($this->entityType, $filterName, $result, $this);
$impl->applyFilter($this->entityType, $filter, $result, $this);
}
}
}
public function applyFilter(string $filterName, array &$result)
public function applyFilter(string $filter, array &$result)
{
$this->applyPrimaryFilter($filterName, $result);
$this->applyPrimaryFilter($filter, $result);
}
public function applyBoolFilter(string $filterName, array &$result)
public function applyBoolFilter(string $filter, array &$result)
{
$this->prepareResult($result);
$method = 'boolFilter' . ucfirst($filterName);
$method = 'boolFilter' . ucfirst($filter);
if (method_exists($this, $method)) {
$this->$method($result);
$wherePart = $this->$method($result);
if ($wherePart) {
$result['whereClause'][] = $wherePart;
}
}
}
public function applyBoolFilterList(array $filterList, array &$result)
{
$this->prepareResult($result);
$wherePartList = [];
foreach ($filterList as $filter) {
$method = 'boolFilter' . ucfirst($filter);
if (method_exists($this, $method)) {
$wherePart = $this->$method($result);
if ($wherePart) {
$wherePartList[] = $wherePart;
}
}
}
if (count($wherePartList)) {
if (count($wherePartList) === 1) {
$result['whereClause'][] = $wherePartList;
} else {
$result['whereClause'][] = ['OR' => $wherePartList];
}
}
}
@@ -2062,7 +2179,7 @@ class Base
$fullTextSearchColumnSanitizedList[$i] = $query->sanitize($query->toDb($field));
}
$where = $function . ':' . implode(',', $fullTextSearchColumnSanitizedList) . ':' . $textFilter;
$where = $function . ':(' . implode(',', $fullTextSearchColumnSanitizedList) . ',' . $textFilter . ')';
$result = [
'where' => $where,
@@ -2105,7 +2222,7 @@ class Base
$textFilter = str_replace('*', '%', $textFilter);
} else {
if (!$useFullTextSearch) {
$textFilterForFullTextSearch .= '*';
//$textFilterForFullTextSearch .= '*';
}
}
@@ -2129,8 +2246,37 @@ class Base
$fullTextSearchFieldList = [];
if ($fullTextSearchData) {
$fullTextGroup[] = $fullTextSearchData['where'];
if ($this->fullTextRelevanceThreshold) {
$fullTextGroup[] = [$fullTextSearchData['where'] . '>=' => $this->fullTextRelevanceThreshold];
} else {
$fullTextGroup[] = $fullTextSearchData['where'];
}
$fullTextSearchFieldList = $fullTextSearchData['fieldList'];
$relevanceExpression = $fullTextSearchData['where'];
if (!isset($result['orderBy']) || $this->fullTextOrderType === self::FT_ORDER_RELEVANCE) {
$result['orderBy'] = [[$relevanceExpression, 'desc']];
$result['order'] = null;
} else {
if ($this->fullTextOrderType === self::FT_ORDER_COMBINTED) {
$relevanceExpression =
'ROUND:(DIV:(' . $fullTextSearchData['where'] . ','.$this->fullTextOrderRelevanceDivider.'))';
if (is_string($result['orderBy'])) {
$result['orderBy'] = [
[$relevanceExpression, 'desc'],
[$result['orderBy'], $result['order'] ?? 'asc'],
];
}
}
}
$result['additionalSelect'] = $result['additionalSelect'] ?? [];
$result['additionalSelect'][] = $relevanceExpression;
$result['hasFullTextSearch'] = true;
}
foreach ($fieldList as $field) {
@@ -2186,10 +2332,10 @@ class Base
if ($fullTextSearchData) {
if (!$useFullTextSearch) {
if (in_array($field, $fullTextSearchFieldList)) {
if (!array_key_exists('OR', $fullTextGroup)) {
/*if (!array_key_exists('OR', $fullTextGroup)) {
$fullTextGroup['OR'] = [];
}
$fullTextGroup['OR'][$field . '*'] = $expression;
$fullTextGroup['OR'][$field . '*'] = $expression;*/
continue;
}
}
@@ -2246,43 +2392,78 @@ class Base
protected function boolFilterOnlyMy(&$result)
{
$wherePart = null;
if (!$this->checkIsPortal()) {
if ($this->hasAssignedUsersField()) {
$this->setDistinct(true, $result);
$this->addLeftJoin(['assignedUsers', 'assignedUsersAccess'], $result);
$result['whereClause'][] = [
'assignedUsersAccess.id' => $this->getUser()->id
$this->addLeftJoin(['assignedUsers', 'assignedUsersOnlyMyFilter'], $result);
$wherePart = [
'assignedUsersOnlyMyFilter.id' => $this->getUser()->id
];
} else if ($this->hasAssignedUserField()) {
$result['whereClause'][] = [
$wherePart = [
'assignedUserId' => $this->getUser()->id
];
} else {
$result['whereClause'][] = [
$wherePart = [
'createdById' => $this->getUser()->id
];
}
} else {
$result['whereClause'][] = [
$wherePart = [
'createdById' => $this->getUser()->id
];
}
return $wherePart;
}
protected function boolFilterOnlyMyTeam(&$result)
{
$teamIdList = $this->getUser()->getLinkMultipleIdList('teams');
if (count($teamIdList) === 0) {
return [
'id' => null
];
}
$this->addLeftJoin(['teams', 'teamsOnlyMyFilter'], $result);
$this->setDistinct(true, $result);
return [
'teamsOnlyMyFilterMiddle.teamId' => $teamIdList
];
}
protected function filterFollowed(&$result)
{
$query = $this->getEntityManager()->getQuery();
$result['customJoin'] .= "
JOIN subscription ON
subscription.entity_type = ".$query->quote($this->getEntityType())." AND
subscription.entity_id = ".$query->toDb($this->getEntityType()).".id AND
subscription.user_id = ".$query->quote($this->getUser()->id)."
";
$this->addJoin([
'Subscription',
'subscription',
[
'subscription.entityType' => $this->getEntityType(),
'subscription.entityId=:' => 'id',
'subscription.userId' => $this->getUser()->id,
]
], $result);
}
protected function boolFilterFollowed(&$result)
{
$this->filterFollowed($result);
$this->addLeftJoin([
'Subscription',
'subscription',
[
'subscription.entityType' => $this->getEntityType(),
'subscription.entityId=:' => 'id',
'subscription.userId' => $this->getUser()->id,
]
], $result);
return ['subscription.id!=' => null];
//$result['whereClause'][] = ['subscription.id!=' => null];
}
public function mergeSelectParams(array $selectParams1, ?array $selectParams2) : array

View File

@@ -72,6 +72,12 @@
"entity": "Task",
"foreign": "parent",
"layoutRelationshipsDisabled": true
},
"emails": {
"type": "hasChildren",
"entity": "Email",
"foreign": "parent",
"layoutRelationshipsDisabled": true
}
},
"collection": {

View File

@@ -1,9 +1,4 @@
{
"links": {
"meetings": "Meetings",
"calls": "Calls",
"tasks": "Tasks"
},
"labels": {
"Create {entityType}": "Créer un {entityTypeTranslated}"
}

View File

@@ -1,14 +1,4 @@
{
"fields": {
"billingAddress": "Billing Address",
"shippingAddress": "Shipping Address",
"website": "Website"
},
"links": {
"meetings": "Meetings",
"calls": "Calls",
"tasks": "Tasks"
},
"labels": {
"Create {entityType}": "Créer un {entityTypeTranslated}"
}

View File

@@ -1,36 +1,8 @@
{
"fields": {
"parent": "Parent",
"dateStart": "Date de début",
"dateEnd": "Date End",
"duration": "Duration",
"status": "Status",
"reminders": "Reminders"
},
"links": {
"parent": "Parent"
},
"options": {
"status": {
"Planned": "Planned",
"Held": "Held",
"Not Held": "Not Held"
}
"dateStart": "Date de début"
},
"labels": {
"Create {entityType}": "Créer un {entityTypeTranslated}",
"Schedule {entityType}": "Schedule {entityTypeTranslated}",
"Log {entityType}": "Log {entityTypeTranslated}",
"Set Held": "Set Held",
"Set Not Held": "Set Not Held"
},
"massActions": {
"setHeld": "Set Held",
"setNotHeld": "Set Not Held"
},
"presetFilters": {
"planned": "Planned",
"held": "Held",
"todays": "Today's"
"Create {entityType}": "Créer un {entityTypeTranslated}"
}
}

View File

@@ -1,12 +1,4 @@
{
"fields": {
"address": "Address"
},
"links": {
"meetings": "Meetings",
"calls": "Calls",
"tasks": "Tasks"
},
"labels": {
"Create {entityType}": "Créer un {entityTypeTranslated}"
}

View File

@@ -1,5 +1,5 @@
{
"labels": {
"Create {entityType}": "Buat {entityTypeTranslated}"
}
}
"labels": {
"Create {entityType}": "Buat {entityTypeTranslated}"
}
}

View File

@@ -1,5 +1,5 @@
{
"labels": {
"Create {entityType}": "Buat {entityTypeTranslated}"
}
}
"labels": {
"Create {entityType}": "Buat {entityTypeTranslated}"
}
}

View File

@@ -1,5 +1,5 @@
{
"labels": {
"Create {entityType}": "Buat {entityTypeTranslated}"
}
}
"labels": {
"Create {entityType}": "Buat {entityTypeTranslated}"
}
}

View File

@@ -1,5 +1,5 @@
{
"labels": {
"Create {entityType}": "Buat {entityTypeTranslated}"
}
}
"labels": {
"Create {entityType}": "Buat {entityTypeTranslated}"
}
}

View File

@@ -1,5 +1,5 @@
{
"labels": {
"Create {entityType}": "Buat {entityTypeTranslated}"
}
}
"labels": {
"Create {entityType}": "Buat {entityTypeTranslated}"
}
}

View File

@@ -4,7 +4,6 @@
"dateStart": "Startdato",
"dateEnd": "Sluttdato",
"duration": "Varighet",
"status": "Status",
"reminders": "Påminnelser"
},
"links": {

View File

@@ -1,8 +1,8 @@
{
"links": {
"meetings": "vergaderingen",
"calls": "gesprekken",
"tasks": "taken"
"meetings": "Vergaderingen",
"calls": "Gesprekken",
"tasks": "Taken"
},
"labels": {
"Create {entityType}": "Klant {entityTypeTranslated}"

View File

@@ -1,7 +1,7 @@
{
"fields": {
"billingAddress": "Facturatie adres",
"shippingAddress": "Verzendingsadres"
"shippingAddress": "Verzending adres"
},
"links": {
"meetings": "vergaderingen",

View File

@@ -31,6 +31,7 @@
"setNotHeld": "Set not Held"
},
"presetFilters": {
"planned": "Gepland"
"planned": "Gepland",
"todays": "Vandaag"
}
}

View File

@@ -3,9 +3,9 @@
"address": "Adres"
},
"links": {
"meetings": "vergaderingen",
"calls": "gesprekken",
"tasks": "taken"
"meetings": "Vergaderingen",
"calls": "Gesprekken",
"tasks": "Taken"
},
"labels": {
"Create {entityType}": "Klant {entityTypeTranslated}"

View File

@@ -1,5 +1,5 @@
{
"labels": {
"Create {entityType}": "Criar {entityTypeTranslated}"
}
}
"labels": {
"Create {entityType}": "Criar {entityTypeTranslated}"
}
}

View File

@@ -1,5 +1,9 @@
{
"labels": {
"Create {entityType}": "Criar {entityTypeTranslated}"
}
}
"links": {
"meetings": "Reuniões",
"tasks": "Tarefas"
},
"labels": {
"Create {entityType}": "Criar {entityTypeTranslated}"
}
}

View File

@@ -1,5 +1,13 @@
{
"labels": {
"Create {entityType}": "Criar {entityTypeTranslated}"
}
}
"fields": {
"billingAddress": "Endereço de Cobrança",
"shippingAddress": "Endereço de Entrega"
},
"links": {
"meetings": "Reuniões",
"tasks": "Tarefas"
},
"labels": {
"Create {entityType}": "Criar {entityTypeTranslated}"
}
}

View File

@@ -1,5 +1,5 @@
{
"labels": {
"Create {entityType}": "Criar {entityTypeTranslated}"
}
}
"labels": {
"Create {entityType}": "Criar {entityTypeTranslated}"
}
}

View File

@@ -1,5 +1,12 @@
{
"labels": {
"Create {entityType}": "Criar {entityTypeTranslated}"
}
}
"fields": {
"address": "Endereço"
},
"links": {
"meetings": "Reuniões",
"tasks": "Tarefas"
},
"labels": {
"Create {entityType}": "Criar {entityTypeTranslated}"
}
}

View File

@@ -0,0 +1,5 @@
{
"labels": {
"Create {entityType}": "Criar {entityTypeTranslated}"
}
}

View File

@@ -0,0 +1,9 @@
{
"links": {
"meetings": "Reuniões",
"tasks": "Tarefas"
},
"labels": {
"Create {entityType}": "Criar {entityTypeTranslated}"
}
}

View File

@@ -0,0 +1,13 @@
{
"fields": {
"billingAddress": "Endereço de Cobrança",
"shippingAddress": "Endereço de Entrega"
},
"links": {
"meetings": "Reuniões",
"tasks": "Tarefas"
},
"labels": {
"Create {entityType}": "Criar {entityTypeTranslated}"
}
}

View File

@@ -0,0 +1,5 @@
{
"labels": {
"Create {entityType}": "Criar {entityTypeTranslated}"
}
}

View File

@@ -0,0 +1,12 @@
{
"fields": {
"address": "Endereço"
},
"links": {
"meetings": "Reuniões",
"tasks": "Tarefas"
},
"labels": {
"Create {entityType}": "Criar {entityTypeTranslated}"
}
}

View File

@@ -4,7 +4,6 @@
"dateStart": "Početni datum",
"dateEnd": "Krajnji datum",
"duration": "Trajanje",
"status": "Status",
"reminders": "Podsetnici"
},
"links": {

View File

@@ -5,7 +5,10 @@
"dateEnd": "Дата завершення",
"duration": "Тривалість",
"status": "Статус",
"reminders": "Нагадування"
"reminders": "Нагадування",
"dateStartDate": "Дата початку (цілий день)",
"dateEndDate": "Дата завершення (цілий день)",
"isAllDay": "Є цілий день"
},
"links": {
"parent": "Батько"
@@ -19,7 +22,7 @@
},
"labels": {
"Create {entityType}": "Створити {entityTypeTranslated}",
"Schedule {entityType}": "Розклад {entityTypeTranslated} ",
"Schedule {entityType}": "Розклад {entityTypeTranslated}",
"Log {entityType}": "Лог {entityTypeTranslated}",
"Set Held": "Позначити виконаним",
"Set Not Held": "Позначити невиконаним"

View File

@@ -0,0 +1,62 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2019 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: https://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\Traits;
trait Injectable
{
private $injections = [];
public function inject($name, $object)
{
$this->injections[$name] = $object;
}
public function getDependencyList() : array
{
return $this->dependencyList;
}
protected function getInjection(string $name)
{
return $this->injections[$name];
}
protected function addDependency(string $name)
{
$this->dependencyList[] = $name;
}
protected function addDependencyList(array $list)
{
foreach ($list as $item) {
$this->addDependency($item);
}
}
}

View File

@@ -50,4 +50,4 @@ class UpgradeManager extends Upgrades\Base
'vendor' => 'vendorFiles',
)
);
}
}

View File

@@ -84,15 +84,23 @@ class ActionManager
return $object->run($data);
}
public function getActionClass($actionName)
{
return $this->getObject($actionName);
}
public function getManifest()
{
return $this->getObject()->getManifest();
}
protected function getObject()
protected function getObject($actionName = null)
{
$managerName = $this->getManagerName();
$actionName = $this->getAction();
if (!$actionName) {
$actionName = $this->getAction();
}
if (!isset($this->objects[$managerName][$actionName])) {
$class = '\Espo\Core\Upgrades\Actions\\' . ucfirst($managerName) . '\\' . ucfirst($actionName);
@@ -106,4 +114,4 @@ class ActionManager
return $this->objects[$managerName][$actionName];
}
}
}

View File

@@ -37,10 +37,6 @@ use Composer\Semver\Semver;
abstract class Base
{
private $config;
private $entityManager;
private $helper;
protected $data;
@@ -53,7 +49,7 @@ abstract class Base
private $zipUtil;
private $fileManager;
private $databaseHelper;
protected $processId = null;
@@ -132,31 +128,31 @@ abstract class Base
return $this->zipUtil;
}
protected function getDatabaseHelper()
{
if (!isset($this->databaseHelper)) {
$this->databaseHelper = new \Espo\Core\Utils\Database\Helper($this->getConfig());
}
return $this->databaseHelper;
}
protected function getFileManager()
{
if (!isset($this->fileManager)) {
$this->fileManager = $this->getContainer()->get('fileManager');
}
return $this->fileManager;
return $this->getContainer()->get('fileManager');
}
protected function getConfig()
{
if (!isset($this->config)) {
$this->config = $this->getContainer()->get('config');
}
return $this->config;
return $this->getContainer()->get('config');
}
public function getEntityManager()
{
if (!isset($this->entityManager)) {
$this->entityManager = $this->getContainer()->get('entityManager');
}
return $this->entityManager;
return $this->getContainer()->get('entityManager');
}
protected function throwErrorAndRemovePackage($errorMessage = '')
public function throwErrorAndRemovePackage($errorMessage = '')
{
$this->deletePackageFiles();
$this->deletePackageArchive();
@@ -186,7 +182,7 @@ abstract class Base
return $this->processId;
}
protected function setProcessId($processId)
public function setProcessId($processId)
{
$this->processId = $processId;
}
@@ -205,12 +201,26 @@ abstract class Base
//check php version
if (isset($manifest['php'])) {
$res &= $this->checkVersions($manifest['php'], System::getPhpVersion(), 'Your PHP version does not support this installation package.');
$res &= $this->checkVersions($manifest['php'], System::getPhpVersion(), 'Your PHP version ({version}) is not supported. Required version: {requiredVersion}.');
}
//check database version
if (isset($manifest['database'])) {
$databaseHelper = $this->getDatabaseHelper();
$databaseType = $databaseHelper->getDatabaseType();
$databaseTypeLc = strtolower($databaseType);
if (isset($manifest['database'][$databaseTypeLc])) {
$databaseVersion = $databaseHelper->getDatabaseVersion();
if ($databaseVersion) {
$res &= $this->checkVersions($manifest['database'][$databaseTypeLc], $databaseVersion, 'Your '. $databaseType .' version ({version}) is not supported. Required version: {requiredVersion}.');
}
}
}
//check acceptableVersions
if (isset($manifest['acceptableVersions'])) {
$res &= $this->checkVersions($manifest['acceptableVersions'], $this->getConfig()->get('version'), 'Your EspoCRM version doesn\'t match for this installation package.');
$res &= $this->checkVersions($manifest['acceptableVersions'], $this->getConfig()->get('version'), 'Your EspoCRM version ({version}) is not supported. Required version: {requiredVersion}.');
}
//check dependencies
@@ -244,6 +254,9 @@ abstract class Base
}
}
$errorMessage = preg_replace('/\{version\}/', $currentVersion, $errorMessage);
$errorMessage = preg_replace('/\{requiredVersion\}/', $version, $errorMessage);
$this->throwErrorAndRemovePackage($errorMessage);
}
@@ -266,17 +279,47 @@ abstract class Base
return true;
}
protected function getPackageType()
{
$manifest = $this->getManifest();
if (isset($manifest['type'])) {
return strtolower($manifest['type']);
}
return $this->defaultPackageType;
}
protected function checkDependencies($dependencyList)
{
return true;
}
/**
* Run scripts by type
* Run a script by a type
* @param string $type Ex. "before", "after"
* @return void
*/
protected function runScript($type)
{
$beforeInstallScript = $this->getScriptPath($type);
if ($beforeInstallScript) {
$scriptNames = $this->getParams('scriptNames');
$scriptName = $scriptNames[$type];
require_once($beforeInstallScript);
$script = new $scriptName();
try {
$script->run($this->getContainer(), $this->scriptParams);
} catch (\Exception $e) {
$this->throwErrorAndRemovePackage($e->getMessage());
}
}
}
protected function getScriptPath($type)
{
$packagePath = $this->getPackagePath();
$scriptNames = $this->getParams('scriptNames');
@@ -287,16 +330,8 @@ abstract class Base
}
$beforeInstallScript = Util::concatPath( array($packagePath, self::SCRIPTS, $scriptName) ) . '.php';
if (file_exists($beforeInstallScript)) {
require_once($beforeInstallScript);
$script = new $scriptName();
try {
$script->run($this->getContainer(), $this->scriptParams);
} catch (\Exception $e) {
$this->throwErrorAndRemovePackage($e->getMessage());
}
return $beforeInstallScript;
}
}
@@ -474,6 +509,28 @@ abstract class Base
* @return boolean
*/
protected function copyFiles($type = null, $dest = '')
{
$filesPath = $this->getCopyFilesPath($type);
if ($filesPath) {
switch ($type) {
case 'vendor':
$dest = $this->vendorDirName;
break;
}
return $this->copy($filesPath, $dest, true);
}
return true;
}
/**
* Get needed file list based on type. E.g. file list for "beforeCopy" action
* @param string $type
* @return boolean
*/
protected function getCopyFilesPath($type = null)
{
switch ($type) {
case 'before':
@@ -486,7 +543,6 @@ abstract class Base
$dirNames = $this->getParams('customDirNames');
if (isset($dirNames['vendor'])) {
$dirPath = $dirNames['vendor'];
$dest = $this->vendorDirName;
}
break;
@@ -500,11 +556,9 @@ abstract class Base
$filesPath = Util::concatPath($packagePath, $dirPath);
if (file_exists($filesPath)) {
return $this->copy($filesPath, $dest, true);
return $filesPath;
}
}
return true;
}
protected function getVendorFileList($type = 'copy')
@@ -706,12 +760,15 @@ abstract class Base
protected function checkIsWritable()
{
$fullFileList = array_merge($this->getDeleteFileList(), $this->getCopyFileList());
$backupPath = $this->getPath('backupPath');
$fullFileList = array_merge([$backupPath], $this->getDeleteFileList(), $this->getCopyFileList());
$result = $this->getFileManager()->isWritableList($fullFileList);
if (!$result) {
$permissionDeniedList = $this->getFileManager()->getLastPermissionDeniedList();
throw new Error("Permission denied for <br>". implode(", <br>", $permissionDeniedList));
$delimiter = $this->isCli() ? "\n" : "<br>";
throw new Error("Permission denied: " . $delimiter . implode($delimiter, $permissionDeniedList));
}
}
@@ -756,7 +813,8 @@ abstract class Base
'useCache' => $config->get('useCache'),
];
$this->setParam('beforeMaintenanceModeParams', $actualParams);
$configParamName = $this->getTemporaryConfigParamName();
$config->set($configParamName, $actualParams);
$save = false;
@@ -783,19 +841,40 @@ abstract class Base
protected function disableMaintenanceMode()
{
$config = $this->getConfig();
$beforeMaintenanceModeParams = $this->getParams('beforeMaintenanceModeParams', []);
$configParamName = $this->getTemporaryConfigParamName();
$temporaryUpgradeParams = $config->get($configParamName, []);
$save = false;
foreach ($beforeMaintenanceModeParams as $paramName => $paramValue) {
foreach ($temporaryUpgradeParams as $paramName => $paramValue) {
if ($config->get($paramName) != $paramValue) {
$config->set($paramName, $paramValue);
$save = true;
}
}
if ($config->has($configParamName)) {
$config->remove($configParamName);
$save = true;
}
if ($save) {
$config->save();
}
}
protected function getTemporaryConfigParamName()
{
return 'temporaryUpgradeParams' . $this->getProcessId();
}
protected function isCli()
{
if (substr(php_sapi_name(), 0, 3) == 'cli') {
return true;
}
return false;
}
}

Some files were not shown because too many files have changed in this diff Show More