Compare commits

...

596 Commits
2.9.0 ... 3.2.0

Author SHA1 Message Date
yuri
66fb7b4e81 fix import 2015-04-10 11:49:42 +03:00
yuri
d83a19e316 fix lang 2015-04-10 10:24:58 +03:00
yuri
452e39903e ua lang 2015-04-09 15:28:07 +03:00
yuri
797fcc172c email to lead: use reply to 2015-04-09 14:06:11 +03:00
yuri
cdb7c919b3 sort layout field list 2015-04-09 12:27:38 +03:00
yuri
0f7f438935 layoutSearch fix 2015-04-09 12:12:10 +03:00
yuri
7c01eedda8 followCreatedEntities param 2015-04-09 12:05:40 +03:00
yuri
bab7f3e9fc email notifications 2015-04-09 11:56:47 +03:00
yuri
761d85a112 fix email import and send 2015-04-09 11:02:55 +03:00
yuri
52e323123a notifications 2 2015-04-08 18:23:14 +03:00
yuri
01af1677cf notifications 2015-04-08 17:38:32 +03:00
yuri
d4f4fcb10e email optimization 2015-04-07 16:54:54 +03:00
yuri
74fd228bfb fix email small layout 2015-04-07 15:46:31 +03:00
yuri
f452d2e5d2 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-04-07 15:25:29 +03:00
yuri
3a01348e02 improve draft email 2015-04-07 15:25:15 +03:00
Taras Machyshyn
b1743ee4a2 Cleared passwords when loggin input data 2015-04-07 12:32:13 +03:00
Taras Machyshyn
ab14769387 Changed current year for installer 2015-04-07 12:04:13 +03:00
yuri
5c800601e4 cleanup addition 2015-04-07 11:23:25 +03:00
yuri
8aea81340b default layout changes 2015-04-07 10:28:26 +03:00
yuri
8513911f21 Merge branch 'master' of https://github.com/espocrm/espocrm 2015-04-06 17:38:33 +03:00
yuri
6a46c453cd add client custom dir 2015-04-06 17:38:10 +03:00
yuri
2029dc5e10 fix tasks panel 2015-04-06 17:34:04 +03:00
yuri
70664d1a7a fix template entityDefs 2015-04-06 16:13:08 +03:00
yuri
7d212ec62f fix default detail layout 2015-04-06 16:10:20 +03:00
yuri
fa7b145e07 fix link manager 2015-04-06 16:02:10 +03:00
yuri
830645cb73 parent type length 100 2015-04-06 15:43:47 +03:00
yuri
94e03b3c18 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-04-06 15:36:46 +03:00
Taras Machyshyn
3ace85eebc Improved Orm convertor 2015-04-06 15:36:14 +03:00
yuri
c6cfdd9e90 version 2015-04-06 12:37:57 +03:00
yuri
d1b2dfae3e default task status 2015-04-06 12:29:48 +03:00
yuri
98fb345a5a dont create auth token if logged browser basic auth 2015-04-06 11:26:18 +03:00
Yuri Kuznetsov
481314886f Update README.md 2015-04-03 18:59:02 +03:00
yuri
ad90cd171a leadSource field 2015-04-03 12:50:14 +03:00
yuri
8272e0e652 default duration 2015-04-03 12:07:11 +03:00
yuri
f218de2e04 fix messages 2015-04-03 11:58:58 +03:00
yuri
53529fc506 duration options 2015-04-03 11:47:55 +03:00
yuri
aef99fb4fd lang 2015-04-03 11:08:42 +03:00
yuri
86f690c79f midAlias ORM 2015-04-03 10:29:28 +03:00
yuri
71f9e90436 join many alias 2015-04-02 17:48:46 +03:00
yuri
75608f41b1 Merge branch 'hotfix/3.1.2' 2015-04-02 15:30:19 +03:00
yuri
0730ae5b3f fix forgot password 2015-04-02 15:27:29 +03:00
yuri
a7f537879e change dev index.php 2015-04-02 15:20:05 +03:00
yuri
6066cf1d65 change version 2015-04-02 14:24:20 +03:00
yuri
ccda236b6e Merge branch 'hotfix/3.1.2' 2015-04-02 11:46:22 +03:00
yuri
32ec348369 pt_BR fixes 2015-04-02 11:46:04 +03:00
yuri
d85f66171d Merge branch 'hotfix/3.1.2' 2015-04-02 11:13:26 +03:00
yuri
8c6aa46ec8 fix email plain 2015-04-02 11:12:56 +03:00
yuri
d2f7bc475a fix role and team list layout 2015-04-02 10:57:25 +03:00
yuri
410cf02518 filters layout changes 2015-04-01 16:49:44 +03:00
yuri
958414017e Merge branch 'hotfix/3.1.2' 2015-04-01 16:22:30 +03:00
yuri
8fbfad9637 async false for upgrade 2015-04-01 16:22:17 +03:00
yuri
07e8f94748 fix form leave out 2015-04-01 15:52:21 +03:00
yuri
4db162f9f7 fix form leave out 2015-04-01 15:33:03 +03:00
yuri
459dfb7937 Merge branch 'master' of https://github.com/espocrm/espocrm 2015-04-01 15:07:55 +03:00
yuri
56b0536152 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-04-01 15:07:42 +03:00
yuri
5bb7dd3ccd list buttons 2015-04-01 15:07:27 +03:00
yuri
6e50e4fd8e hide cc field if empty 2015-04-01 12:23:00 +03:00
yuri
878e1616da change email layout and email from name fix 2015-04-01 12:16:35 +03:00
yuri
64a410099e form leave out 2015-04-01 11:32:37 +03:00
yuri
9cb8664ddf fix token remove 2015-04-01 09:49:41 +03:00
Taras Machyshyn
6d5c3c8ad5 Code optimization for Orm convertation 2015-03-31 17:23:53 +03:00
Yuri Kuznetsov
f07c8fca64 Update README.md 2015-03-31 16:41:56 +03:00
yuri
944c76ad93 imprort improvements 2015-03-31 15:50:33 +03:00
yuri
ae8f9df02b fix metadata to orm 2015-03-31 10:57:10 +03:00
yuri
6da3659b7d Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-03-31 10:48:13 +03:00
yuri
1f1d9f3abd Merge branch 'hotfix/3.1.2' 2015-03-31 10:48:08 +03:00
yuri
530d7f2d81 import email: findParent lead 2015-03-31 10:47:53 +03:00
Taras Machyshyn
e67ecded2a Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-03-31 10:43:31 +03:00
Taras Machyshyn
b9343d5e64 Improved Orm convertation 2015-03-31 10:43:14 +03:00
yuri
898393c7ac Merge branch 'hotfix/3.1.2' 2015-03-30 17:44:16 +03:00
yuri
9b2ea5298d fix iframe links 2015-03-30 17:14:37 +03:00
yuri
0d2b56c58f signature fix 2015-03-30 17:07:28 +03:00
yuri
076c920dc7 upgrade summernote 2015-03-30 16:58:45 +03:00
yuri
638df642ec import dev 2015-03-30 16:47:27 +03:00
yuri
d2f4190612 field changes 2015-03-30 16:18:29 +03:00
yuri
7201f517e8 row actions change 2015-03-30 15:32:18 +03:00
yuri
c91365f627 change calendar query 2015-03-30 15:03:41 +03:00
yuri
f3bc4dbccf import dev 2015-03-30 13:39:14 +03:00
yuri
d4daef6012 autoincrement creatable 2015-03-30 12:29:53 +03:00
yuri
dbf4f68a44 user filters 2015-03-30 12:28:54 +03:00
yuri
8a7325963b import revert/remove duplicates 2015-03-30 12:26:41 +03:00
yuri
fc0d8dffcd fix prev 2015-03-30 11:34:54 +03:00
yuri
f8498e3adc save and remove options 2015-03-30 11:22:14 +03:00
yuri
e4b51ba675 fix isOverdue 2015-03-27 16:49:32 +02:00
yuri
83f9ead607 record getEntity 2015-03-27 16:44:34 +02:00
yuri
bf6778770e fix label 2015-03-27 16:23:40 +02:00
yuri
093b80293a import dev 2015-03-27 16:20:44 +02:00
yuri
34dee314e7 import dev 2015-03-27 15:46:59 +02:00
yuri
313b94168e search.tpl fix 2015-03-27 13:10:56 +02:00
yuri
d6085bbdcf createdAt to filters 2015-03-27 12:43:09 +02:00
yuri
160dc5d61c Merge branch 'hotfix/3.1.2' 2015-03-27 12:10:41 +02:00
yuri
987b383d7c fix plainText signature 2015-03-27 12:10:12 +02:00
yuri
8b4cb1568e import dev 2015-03-27 12:00:10 +02:00
yuri
aef4028180 replace tabs 2015-03-27 11:52:07 +02:00
yuri
9b779ee8cd change streamRelated for account 2015-03-26 18:08:35 +02:00
yuri
ccc339cf48 import 1 2015-03-26 17:23:26 +02:00
yuri
6aabbfd944 replace tabs 2015-03-26 13:58:32 +02:00
yuri
0e87626e34 cleanup 2015-03-26 13:22:20 +02:00
yuri
8641181511 fix task panel 2015-03-26 13:20:21 +02:00
yuri
123c3ef8ab grand filters change 2015-03-26 13:14:13 +02:00
yuri
07bd5a9d60 opp by stage w/o Closed Won 2015-03-25 16:07:05 +02:00
yuri
deca89039a calendar dashlet changes 2015-03-25 15:58:08 +02:00
yuri
1542fa86d8 Merge branch 'hotfix/3.1.1' 2015-03-25 15:28:11 +02:00
yuri
93ca814fe3 fix save filters margin 2015-03-25 15:27:54 +02:00
yuri
4d9e5ee302 accountId improvements 2015-03-25 15:25:39 +02:00
yuri
0a697cdc0a note superParent 2 2015-03-25 14:42:19 +02:00
yuri
1b01e476ac clearnup 2015-03-25 14:09:10 +02:00
yuri
ee689fb351 remove dashletOptions when dashlet removed 2015-03-25 14:08:34 +02:00
yuri
92a76d80f2 Merge branch 'hotfix/3.1.1' 2015-03-25 10:26:56 +02:00
yuri
a662c1a5fe improve uniqid 2015-03-25 10:26:02 +02:00
yuri
3ffa949276 fix create in select 2015-03-24 17:44:16 +02:00
yuri
371966e2cf add super parent for note 2015-03-24 17:24:38 +02:00
yuri
8f1a4d2a02 link, linkParent and belondToParent fix 2015-03-24 16:44:15 +02:00
yuri
3128b6f25a add index to email 2015-03-24 15:51:36 +02:00
yuri
0920821f1e change lang message 2015-03-24 12:12:24 +02:00
yuri
0f8f9c01ff change css 2015-03-24 12:06:29 +02:00
yuri
1d9621bc91 change version 2015-03-24 11:25:09 +02:00
yuri
e138daec5f fix table head hight 2015-03-24 11:16:26 +02:00
yuri
4a51e754d7 fix in inbound email 2015-03-24 11:01:55 +02:00
yuri
e6e0bc1703 fix inbound email double notification 2015-03-24 10:52:33 +02:00
yuri
49fbc6e082 email fromName changes 2015-03-24 10:39:08 +02:00
yuri
a0ed610f60 search by target lists 2015-03-23 18:07:39 +02:00
yuri
9f8c0eb4a2 cleanup 2015-03-23 16:50:32 +02:00
yuri
be29fac010 rename role to title 2015-03-23 16:02:07 +02:00
yuri
b3cefd9bcb change case layout 2015-03-23 15:54:34 +02:00
yuri
501d6f2692 fix markdown parsing 2015-03-23 15:45:48 +02:00
yuri
3d7560024f fixes email reply 2015-03-23 14:39:01 +02:00
yuri
a201b45d04 change stream messages 2015-03-23 11:45:30 +02:00
yuri
41f88a4015 modal.detail scroll to top hack 2015-03-23 11:18:45 +02:00
yuri
52a131ef93 fix dashlet actions 2015-03-23 10:34:54 +02:00
yuri
7ab28601ca dashlet options clone 2015-03-23 10:04:34 +02:00
yuri
5de509a0de fix warnings 2015-03-23 10:00:08 +02:00
yuri
2534b14dce fix frontend acl 2015-03-23 09:50:40 +02:00
yuri
27f98b95eb fix currency field 2015-03-20 17:09:02 +02:00
yuri
b08bee78b9 iframe fix 2015-03-20 16:57:46 +02:00
yuri
a39327a95b 2015 2015-03-20 16:23:18 +02:00
yuri
f979a8dd46 fix options 2015-03-20 15:23:22 +02:00
yuri
ecf1645e05 cleanup 2015-03-20 14:31:24 +02:00
yuri
d015bec89c navbar changes 2015-03-20 14:31:07 +02:00
yuri
92e1d4b8e6 clearnup 2015-03-19 17:13:02 +02:00
yuri
a75df456e2 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-03-19 17:11:43 +02:00
yuri
5393c2f314 header changes 2015-03-19 17:08:04 +02:00
yuri
c49bcc29db logo changes 2015-03-19 16:21:45 +02:00
yuri
3e74fb49ac contact list layout 2015-03-19 15:43:51 +02:00
Taras Machyshyn
cf885cafb4 Upgrade improvements 2015-03-19 15:38:20 +02:00
yuri
7a7c605687 fix frontend/index.php 2015-03-19 15:35:38 +02:00
yuri
5676d42801 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-03-19 15:02:45 +02:00
Taras Machyshyn
c893a4441d Added a file to Gruntfile.js 2015-03-19 15:01:36 +02:00
yuri
c5e1d0847f Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-03-19 14:13:19 +02:00
Taras Machyshyn
f0b4c2dc79 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-03-19 13:49:41 +02:00
Taras Machyshyn
1584f25af5 Fixed a bug with the upgrade 2015-03-19 13:49:25 +02:00
yuri
761e7b2344 detaul buttons changes 2015-03-19 12:58:36 +02:00
yuri
7099c92c39 fixes with el 2015-03-19 12:44:23 +02:00
yuri
be26ec2651 inline edit disanled option 2015-03-19 11:49:08 +02:00
yuri
7a2f49ad43 list fixes 2015-03-19 10:54:25 +02:00
yuri
22477385f7 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-03-18 18:18:20 +02:00
yuri
e7ae0862ba menu title 2015-03-18 18:18:10 +02:00
yuri
f501a7377d navbar changes 2015-03-18 18:16:19 +02:00
yuri
9ce9ccdb5c update bootstrap 2015-03-18 17:37:46 +02:00
yuri
f8a425b7d1 fix select all 2015-03-18 17:37:23 +02:00
yuri
8ab92f8668 navbar changes 2015-03-18 17:30:56 +02:00
Taras Machyshyn
e598665242 Added upgrade.php script to upgrade EspoCRM via CLI 2015-03-18 17:22:07 +02:00
yuri
1bc984f8fa fix select all icon 2015-03-18 17:18:24 +02:00
Taras Machyshyn
3a0b1a22f0 Improved logging when permission denied for a log file 2015-03-18 17:09:47 +02:00
Taras Machyshyn
8fd41d5ea7 Bug fixes 2015-03-18 17:00:58 +02:00
yuri
787adc72a5 changes in calls/mettings dashlets 2015-03-18 16:52:43 +02:00
yuri
a99b043d95 add bool filter to cases 2015-03-18 16:45:00 +02:00
yuri
2079c19dc0 add deferred status to tasks 2015-03-18 16:35:17 +02:00
yuri
9f9ba2c915 cases dashlet change 2015-03-18 16:35:06 +02:00
yuri
31db584658 change task dashlet 2015-03-18 16:13:54 +02:00
yuri
a09c7db717 fix array field json 2015-03-18 15:44:48 +02:00
yuri
19bd38bb85 array field cursor 2015-03-18 12:22:41 +02:00
yuri
7e678a8378 opportunity account not required 2015-03-18 12:14:02 +02:00
yuri
93c5d7eb91 fix options view helper 2015-03-18 11:51:12 +02:00
yuri
a9fcf5d510 Merge branch 'hotfix/3.0.2' 2015-03-18 11:40:36 +02:00
yuri
6f1266dcea Merge branch 'hotfix/3.0.2' of ssh://172.20.0.1/var/git/espo/backend into hotfix/3.0.2 2015-03-18 11:40:25 +02:00
yuri
2395fb6986 change version 2015-03-18 11:40:05 +02:00
yuri
48a0fa5e96 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-03-18 10:52:02 +02:00
yuri
95a0f78fa7 array field param noEmprtString 2015-03-18 10:51:51 +02:00
yuri
85f18fe338 change default dashboard tab 2015-03-18 10:38:02 +02:00
yuri
cbfad8eb79 dashboard changes 2015-03-18 10:37:22 +02:00
Taras Machyshyn
5cd4a92049 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-03-17 18:04:53 +02:00
Taras Machyshyn
e5b4de1683 Jobs: added attempts for jobs 2015-03-17 18:04:40 +02:00
yuri
cdd07a9f44 dashboard tabs 2015-03-17 17:54:42 +02:00
yuri
121eaabe40 fix quick view modal 2015-03-17 14:53:57 +02:00
yuri
f559c512fd change info color 2015-03-17 11:54:21 +02:00
yuri
9aa5ed63d0 bool filters labels 2015-03-17 11:53:12 +02:00
yuri
fa9c0e5b3b fix query in empty array 2015-03-17 11:39:47 +02:00
yuri
469c6f3b45 email-to-lead fix 2015-03-17 11:05:21 +02:00
yuri
e23e013432 Lead Create for campaign 2015-03-17 10:45:51 +02:00
yuri
e051ef4935 email fixes 2015-03-16 19:04:58 +02:00
yuri
2a4e816448 fix email from 2015-03-16 18:09:11 +02:00
yuri
4095530491 clearnup 2015-03-16 17:26:55 +02:00
yuri
5a40507b54 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-03-16 17:26:06 +02:00
yuri
f438b0898b email to lead and fucking fixes 2015-03-16 17:25:53 +02:00
Taras Machyshyn
58cfeeb1be Removed unnecessary debug message 2015-03-16 15:51:43 +02:00
yuri
6dbd521501 fix email and phone fields 2015-03-16 14:40:57 +02:00
yuri
541cb13ec3 view helper fix 2 2015-03-16 13:42:33 +02:00
yuri
e7fbcbbac6 view helper fix 2015-03-16 13:36:52 +02:00
yuri
7bc3d8a826 show from name in email 2015-03-16 13:13:07 +02:00
yuri
a814a95c9a list row dropdown css change 2015-03-16 13:04:26 +02:00
yuri
e95a40f24d quick view 2015-03-16 12:43:57 +02:00
yuri
b5972da08e autocomplete fix for assignmentPermission 2015-03-16 11:47:03 +02:00
yuri
9f488c5539 fix text field 2015-03-16 11:15:19 +02:00
yuri
aeea86155c fix email record list 2015-03-16 10:25:10 +02:00
yuri
843ca2a7c5 fix email address primary pdo quote 2015-03-13 15:32:04 +02:00
yuri
bbb67856c0 Merge branch 'master' of https://github.com/espocrm/espocrm 2015-03-13 15:27:40 +02:00
Yuri Kuznetsov
1100e82364 Merge pull request #38 from Threxxy/patch-1
INSERTing number fix for PhoneNumber.php
2015-03-13 15:27:28 +02:00
Pete
7dfafaee22 INSERTing number fix for PhoneNumber.php
Adding a new phone number to a contact was giving the error: SQLSTATE[HY000]: General error: 1366 Incorrect integer value: '' for column 'primary' at row 1

Cast the boolean to an int and it appears to be fixed.
2015-03-13 13:08:41 +00:00
yuri
cd1aab0b1e cleanup 2015-03-13 13:52:00 +02:00
yuri
fa9c2806ca mass actions 2015-03-13 13:49:22 +02:00
yuri
736a53c8d0 calendar changes 2015-03-12 17:11:56 +02:00
yuri
0b713be508 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-03-12 16:24:21 +02:00
yuri
715b290174 task date 2015-03-12 16:24:07 +02:00
yuri
54d241d247 fix notice 2015-03-12 10:28:52 +02:00
yuri
bf8f9023e2 fix task panel filter 2015-03-12 10:15:44 +02:00
Taras Machyshyn
f57c81efe4 correction ru_RU translation 2015-03-11 16:44:57 +02:00
Taras Machyshyn
6d12cf44b4 fixed problem with espocrm news for https connections 2015-03-11 16:35:23 +02:00
yuri
4e76855a12 cleanup 2015-03-11 16:16:35 +02:00
yuri
724b2adbbd fix calendar date-end 2015-03-11 16:09:08 +02:00
yuri
c95031912a calendar-page 2015-03-11 12:59:31 +02:00
yuri
4f7c6dacee assignment permission 3 2015-03-11 12:34:12 +02:00
yuri
716e434099 assignmentPermission 2 2015-03-11 12:15:51 +02:00
yuri
f1f566d56f readOnly fix 2015-03-11 11:52:39 +02:00
yuri
4cf1b8dbda assignmentPermission 1 2015-03-10 16:49:43 +02:00
yuri
87ce26f3cd fix default jobs scheduling value 2015-03-10 16:24:53 +02:00
yuri
94c157df7a fix user side 2015-03-10 15:55:23 +02:00
yuri
d13ccb8dc7 reset model if status = 400 and not new 2015-03-10 15:53:10 +02:00
yuri
b73d82ba00 cancel prev 2015-03-10 15:49:19 +02:00
yuri
afdeeeffa7 reset model if bad request 2015-03-10 15:45:21 +02:00
yuri
16cd135cb7 fixes in oauth2 client 2015-03-10 10:14:52 +02:00
yuri
0c6df91604 email draft 2015-03-09 17:06:26 +02:00
yuri
58a6ced886 fix unread 2015-03-09 16:23:11 +02:00
yuri
d0d3f39f5d email signature 2015-03-09 15:48:13 +02:00
yuri
82a96c743c email/phone paste fix 2015-03-09 12:19:26 +02:00
yuri
a18453c3af play notification sound only onnce 2015-03-09 11:47:25 +02:00
yuri
a75dc61748 mettings held action 2015-03-09 11:23:05 +02:00
yuri
0a60663548 documents for opp 2015-03-09 11:16:30 +02:00
yuri
53cac1c299 convert lead: emails 2015-03-09 11:14:12 +02:00
yuri
4d624aa6d1 campaign log error Espo 2015-03-09 11:06:50 +02:00
yuri
ca807c7ec8 Merge branch 'feature/campaign' 2015-03-06 18:11:02 +02:00
yuri
f245b30d90 fix presets 2015-03-06 18:10:40 +02:00
yuri
0cfb29ad6d cleanup 2015-03-06 17:19:49 +02:00
yuri
27ed23cb51 email improvements 2015-03-06 17:18:45 +02:00
Taras Machyshyn
1794042cea fix bug into CronManager 2015-03-06 16:55:23 +02:00
yuri
6657078a89 email personStringData field 2015-03-06 12:16:02 +02:00
yuri
a156e869fc if email duplicat then add user/team 2015-03-06 11:13:42 +02:00
yuri
06ffe9d373 fix ORM Query 2015-03-05 18:38:42 +02:00
yuri
817fc2fd40 fix ORM Query 2015-03-05 18:36:07 +02:00
yuri
2ec4464a52 dev 2015-03-05 15:12:55 +02:00
yuri
e79874c039 Merge branch 'master' into feature/campaign 2015-03-05 13:07:07 +02:00
yuri
72b9d77e44 changes in side panels 2015-03-05 13:06:08 +02:00
yuri
c622b976de side panel action refresh 2015-03-05 12:27:17 +02:00
yuri
4e0d570f90 dev 2015-03-05 12:26:11 +02:00
yuri
ffce8cb0f7 Merge branch 'master' into feature/campaign 2015-03-05 11:34:19 +02:00
yuri
6dd2e365ad fix currency notStorable 2015-03-05 11:34:11 +02:00
yuri
c324729cdf Merge branch 'master' into feature/campaign 2015-03-05 11:16:23 +02:00
yuri
5b3eee299c fix currency joins 2015-03-05 11:16:11 +02:00
yuri
2ff6946b91 dev 2015-03-05 11:14:02 +02:00
yuri
c3a44a413a add types for inline download 2015-03-04 16:43:04 +02:00
yuri
a4050b1476 Merge branch 'master' into feature/campaign 2015-03-04 14:52:50 +02:00
yuri
2da6d68fe3 fix email preview 2015-03-04 14:50:08 +02:00
yuri
ed19b9d328 check isPost in User change own password 2015-03-04 12:26:09 +02:00
yuri
f41e4f2df4 Merge branch 'master' into feature/campaign 2015-03-04 12:15:32 +02:00
yuri
410bd7f177 fix panel tpls 2015-03-04 12:15:17 +02:00
yuri
caa659020e dev 2015-03-04 12:13:47 +02:00
yuri
877820f5f5 merge with master 2015-03-04 11:47:55 +02:00
yuri
18c6847481 email preview in stream 2015-03-04 11:27:04 +02:00
yuri
274c043f44 improve action events 2015-03-04 11:15:34 +02:00
yuri
529d8c1100 changes in relationship panel 2015-03-04 11:03:32 +02:00
yuri
c05c169948 disable ability to remove ow user 2015-03-04 10:31:44 +02:00
yuri
99db3f5416 fix email attachmentsField handle 2015-03-04 10:29:58 +02:00
yuri
1fa976a43c fix email acl real 2015-03-04 10:24:47 +02:00
yuri
32198144cc add document to tablist 2015-03-04 10:08:26 +02:00
yuri
29420a5306 attendeess panels 2015-03-04 10:06:57 +02:00
yuri
67f6f9b250 Merge branch 'master' into feature/campaign 2015-03-04 09:52:46 +02:00
yuri
c2bbc2e61e showModal action 2015-03-04 09:52:31 +02:00
yuri
2d74cddf86 Merge branch 'master' into feature/campaign 2015-03-03 18:01:45 +02:00
yuri
53049cf1f4 merge layout improve 2015-03-03 17:36:43 +02:00
yuri
6aa012891d dev 2015-03-03 17:26:00 +02:00
yuri
5a7f3ed101 dev 2015-03-03 15:58:17 +02:00
yuri
131743e702 Merge branch 'master' into feature/campaign 2015-03-03 15:51:51 +02:00
yuri
b5cde9a157 fix view-helper 2015-03-03 15:51:40 +02:00
yuri
b150aad977 dev 2015-03-03 15:45:20 +02:00
yuri
75d220e74d dev 2015-03-03 13:38:15 +02:00
Taras Machyshyn
39a3d215d6 Improved set owner, group permissions 2015-03-03 12:42:02 +02:00
yuri
8fe87ccd04 Merge branch 'master' into feature/campaign 2015-03-03 12:32:55 +02:00
yuri
621a2ffbbf add Views.Record.RowActions.Empty 2015-03-03 12:32:19 +02:00
yuri
0f582c59ab dev 2015-03-03 12:31:10 +02:00
yuri
ce18f41400 Merge branch 'master' into feature/campaign 2015-03-03 12:25:31 +02:00
yuri
d780847232 relationship-remove-only 2015-03-03 12:25:15 +02:00
yuri
c074423f0b campaign dev 2015-03-03 12:24:18 +02:00
yuri
31cb2fed88 Merge branch 'master' into feature/campaign 2015-03-03 11:24:26 +02:00
yuri
225f407545 cleanup 2015-03-03 11:23:19 +02:00
yuri
f10dd19446 campaign 2015-03-03 11:20:56 +02:00
yuri
78d5547e08 Merge branch 'hotfix/3.0.2' 2015-03-03 11:00:32 +02:00
yuri
af08f23ca1 de_DE changes 2015-03-03 11:00:19 +02:00
yuri
99d2f14008 campaign dev 2015-03-02 17:52:23 +02:00
yuri
a90a2b859d Merge branch 'master' into feature/campaign 2015-03-02 13:26:26 +02:00
yuri
b0edae1eae Merge branch 'hotfix/3.0.2' 2015-03-02 13:25:56 +02:00
yuri
a5d923c4e3 add ms. to salutation 2015-03-02 13:25:17 +02:00
yuri
86e118341e Merge branch 'master' into feature/campaign 2015-03-02 12:56:06 +02:00
yuri
ed152c9d4c Merge branch 'hotfix/3.0.2' 2015-03-02 12:55:53 +02:00
yuri
8fa805f5eb fix person name salutation 2015-03-02 12:55:35 +02:00
yuri
16ce5385f4 campaign dev 2015-03-02 12:24:57 +02:00
yuri
d1fe581706 Merge branch 'master' into feature/campaign 2015-03-02 12:02:55 +02:00
yuri
dc651b5d77 refactor detail-bottom 2015-03-02 12:02:34 +02:00
yuri
be3dabe5a7 campaign dev 2015-03-02 11:55:43 +02:00
yuri
21326ec3b5 Merge branch 'master' into feature/campaign 2015-03-02 10:25:38 +02:00
yuri
8b851463bc change container padding 2015-02-27 17:32:42 +02:00
yuri
d4a3eb1026 change target list layout 2015-02-27 17:18:24 +02:00
yuri
0980120184 fix taget list service 2015-02-27 15:49:48 +02:00
yuri
db275028d1 Merge branch 'master' into feature/campaign 2015-02-27 15:49:33 +02:00
yuri
f78004b9d3 fix loadAddition 2015-02-27 15:47:30 +02:00
yuri
bac0227675 Merge branch 'master' into feature/campaign 2015-02-27 15:44:59 +02:00
yuri
9f84faa776 fix loadAdditionalFieldsForList 2015-02-27 15:44:28 +02:00
yuri
639c52d1d6 fix relation 2015-02-27 15:36:46 +02:00
yuri
811db09a1d calculate entryCount 2015-02-27 15:24:27 +02:00
yuri
9c61e03542 Merge branch 'master' into feature/campaign 2015-02-27 15:24:11 +02:00
yuri
3e4e86bd5a refactor Record 2015-02-27 15:15:03 +02:00
yuri
5acf253a98 campaign 2015-02-27 15:10:06 +02:00
yuri
84cfcd4352 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-27 12:28:02 +02:00
Taras Machyshyn
73f3673a11 Improved unique indexes for relationship tables 2015-02-27 12:27:37 +02:00
yuri
2c18f75152 Merge branch 'master' into feature/campaign 2015-02-27 12:25:39 +02:00
yuri
3601ee35bd change layouts 2015-02-27 12:25:28 +02:00
yuri
76379f49be Merge branch 'master' into feature/campaign 2015-02-27 12:21:47 +02:00
yuri
1f1fc9111f change lead list layout 2015-02-27 12:21:33 +02:00
yuri
334bd55ad6 Merge branch 'master' into feature/campaign 2015-02-27 12:20:44 +02:00
yuri
2f9f0964ed change contact list layouts 2015-02-27 12:20:28 +02:00
yuri
b5b04f85d0 Merge branch 'master' into feature/campaign 2015-02-27 12:14:50 +02:00
yuri
554f5de0af fix createLink 2015-02-27 12:14:36 +02:00
yuri
5f528c9f6f Merge branch 'master' into feature/campaign 2015-02-27 12:10:28 +02:00
yuri
64b98a7a3c mass relate 3 2015-02-27 12:10:16 +02:00
yuri
74c9d5a8e6 fix email address and phone number repository 2015-02-27 12:02:28 +02:00
yuri
ff2ca7f18c mass relate dev 2 2015-02-26 17:14:44 +02:00
yuri
a0203d3310 fix search manager 2015-02-26 16:05:57 +02:00
yuri
bddd55213f merge with hotfix 2015-02-26 15:55:00 +02:00
yuri
fa97c0ede4 fix selectRElated filters copied 2015-02-26 15:50:41 +02:00
yuri
1dbfe36d1e select all dev 2015-02-26 15:48:11 +02:00
yuri
ec898f297a Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-26 14:49:00 +02:00
Taras Machyshyn
67049d2f08 DBAL: added an unique index for midKeys of manyMany relationship table 2015-02-26 14:42:14 +02:00
yuri
a2c7f84907 ORM: massRelate 2015-02-26 13:24:32 +02:00
yuri
338ba010a2 fix relationship 2015-02-26 11:52:27 +02:00
yuri
a9737e8352 Merge branch 'master' into feature/campaign 2015-02-26 11:23:01 +02:00
yuri
fb8a4b90ca maxEmailAccountCount 2015-02-26 11:16:57 +02:00
yuri
6876bd9a6d cleanup 2015-02-25 15:16:25 +02:00
yuri
392f107de1 emails: dont show attachments if there are no 2015-02-25 14:57:58 +02:00
yuri
17ab2cc717 fix quickDetail 2015-02-25 13:42:01 +02:00
yuri
abddb6c686 change email history layout 2015-02-25 13:21:08 +02:00
yuri
0a5ec4b81c cleanup and backdrop for quickDetail 2015-02-25 13:14:50 +02:00
yuri
ddba654f00 quickDetail and fixes in list expanded 2015-02-25 13:08:32 +02:00
yuri
27a5833b36 change text field detailMaxNewLineCount 2015-02-25 11:11:09 +02:00
yuri
c4cc29b452 remove contract php files 2015-02-25 10:22:20 +02:00
yuri
ac56846e1b trim person name, phone and email 2015-02-25 10:21:04 +02:00
yuri
efb7592782 fix typo 2015-02-25 10:15:00 +02:00
yuri
6c40c5a20c improve emails notes 2015-02-25 10:13:34 +02:00
yuri
724c79d089 fix sentEmail note in case 2015-02-24 18:36:47 +02:00
yuri
7148cb232b improve wysiwyg field 2015-02-24 17:52:02 +02:00
yuri
9943d1794b fix iframe resize 2015-02-24 17:06:57 +02:00
yuri
972615c8e6 Added Campaigns and Target Lists 2015-02-24 15:57:34 +02:00
yuri
213546b83f enable multi-enum filter 2015-02-23 17:29:09 +02:00
yuri
e23345edc6 use getEntityType 2015-02-23 17:09:15 +02:00
yuri
7af38e8d21 fix leads detail-side 2015-02-23 17:05:15 +02:00
yuri
b1e62ecb20 add getEntityType into entity 2015-02-23 17:05:00 +02:00
yuri
f05e3924ff fix mass-update layouy and leads side panel 2015-02-23 16:54:06 +02:00
yuri
2ccb9c3a17 noJoin links 2015-02-23 16:53:48 +02:00
yuri
c64d27f413 fix record list dashlet 2015-02-23 16:12:58 +02:00
yuri
2a3711d0b4 improve search manager 2015-02-23 16:05:15 +02:00
yuri
2baace4ef9 fix Zend issues 2015-02-23 15:30:45 +02:00
yuri
aa501a4429 fix my opportunities dashlet 2015-02-20 17:35:49 +02:00
yuri
3095f61011 change opportunity dashlet layout 2015-02-20 17:31:33 +02:00
yuri
001de70fe8 show only future calls and meetings in dashlet 2015-02-20 17:21:06 +02:00
yuri
58733450d1 steam status styles 2015-02-20 17:06:45 +02:00
yuri
0605672c32 cleanup 2015-02-20 16:33:49 +02:00
yuri
0c570b0d83 cleanup 2015-02-20 16:31:17 +02:00
yuri
0833428b71 no subject 2015-02-20 16:10:17 +02:00
yuri
4c4714a846 disabledCountQuery 2015-02-20 14:59:40 +02:00
yuri
155cd428f6 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-20 11:53:04 +02:00
yuri
06bde4f1ce move autoFollow to job 2015-02-20 11:51:31 +02:00
yuri
6112a9d02d job: pending jobs fix 2015-02-20 11:51:11 +02:00
yuri
a3f4beac61 autofill executeTime if notset 2015-02-20 11:50:25 +02:00
yuri
b116821731 cron: don't fail jobs if pending and expired 2015-02-20 10:46:46 +02:00
Taras Machyshyn
3439267926 merge changes 2015-02-19 17:12:31 +02:00
yuri
1ac55737b4 optimization 2015-02-19 16:07:18 +02:00
yuri
c66cfeb297 auto-follow 2015-02-19 15:28:49 +02:00
yuri
85f9fe6f3b fix link autocomplete 2015-02-19 13:44:10 +02:00
yuri
fe9154fde7 autocomplete off 2015-02-19 13:21:03 +02:00
yuri
bdc7bbada4 fix autocomplete off 2015-02-19 13:19:03 +02:00
yuri
9f4a203abe Merge branch 'hotfix/3.0.1' 2015-02-18 18:07:19 +02:00
yuri
a89872b785 3.0.1 2015-02-18 17:13:41 +02:00
yuri
b6bc0471b8 Merge branch 'hotfix/3.0.1' of ssh://172.20.0.1/var/git/espo/backend into hotfix/3.0.1 2015-02-18 17:05:02 +02:00
yuri
c38544e6ad refresh dashlet on header 2015-02-18 15:56:54 +02:00
yuri
6122edb091 cleanup 2015-02-18 15:13:36 +02:00
yuri
684441040c limitQuery 2015-02-18 15:10:30 +02:00
yuri
55c733b13a ability to to execute count query to know about out of range 2015-02-18 15:06:53 +02:00
yuri
dc7571d30c fix findMyQuery 2015-02-18 15:05:53 +02:00
yuri
699c1d1f23 ORM: findByQuery 2015-02-18 12:36:11 +02:00
yuri
f67fbbd621 calendar distinguish by status 2015-02-18 12:09:42 +02:00
yuri
dbcc7e61c1 calendar resize fix 2015-02-18 11:30:50 +02:00
Taras Machyshyn
afd0f5a5ef Extensions: don't run afterUninstall script for updating an extension 2015-02-18 09:53:53 +02:00
yuri
859f1f0cd9 ability to refresh calendar 2015-02-17 17:41:32 +02:00
yuri
3f8ccb69fc fix calendar shade color 2015-02-17 17:28:58 +02:00
yuri
ba65ccffe4 Merge branch 'hotfix/3.0.1' 2015-02-17 16:49:40 +02:00
yuri
533fff4d30 Merge branch 'hotfix/3.0.1' of ssh://172.20.0.1/var/git/espo/backend into hotfix/3.0.1 2015-02-17 16:49:03 +02:00
yuri
7b0b3d05ad fix zend mail 2015-02-17 16:46:13 +02:00
Taras Machyshyn
e09c101f82 Upgrades: fixed an error 2015-02-17 16:45:21 +02:00
yuri
c87c657b02 calendar shade past activities 2015-02-17 16:45:03 +02:00
Taras Machyshyn
4e6b300477 Upgrades: improved restore functionality 2015-02-17 16:39:51 +02:00
Taras Machyshyn
7453eb583a impoved Uprades code 2015-02-17 15:53:35 +02:00
Taras Machyshyn
b5d2eb93aa Upgrades: impoved beforeRunAction(), afterRunAction() 2015-02-17 15:53:27 +02:00
Taras Machyshyn
b5b8ab4a8e added initialize(), finalize() methods to upgrade scripts 2015-02-17 15:53:18 +02:00
Taras Machyshyn
b6010a7ad7 added a new fileManager test, fixes 2015-02-17 15:53:07 +02:00
Taras Machyshyn
ebdb649bd7 Improved upgrade/install extenions feature 2015-02-17 15:52:53 +02:00
Taras Machyshyn
ec17306992 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-17 15:50:36 +02:00
Taras Machyshyn
9b6bf4171b impoved Uprades code 2015-02-17 15:43:51 +02:00
yuri
252b5ef729 Merge branch 'hotfix/3.0.1' 2015-02-17 15:39:24 +02:00
yuri
220d4d1b77 improve importer 2015-02-17 15:21:47 +02:00
yuri
2f8e1aeaf2 fix field iframe wysiwyg height 2015-02-17 14:57:17 +02:00
Taras Machyshyn
ced41021f0 Upgrades: impoved beforeRunAction(), afterRunAction() 2015-02-17 13:46:37 +02:00
Taras Machyshyn
36e3cc4688 added initialize(), finalize() methods to upgrade scripts 2015-02-17 13:02:02 +02:00
yuri
20db3b0bd2 improve email quick-edit 2015-02-17 12:25:39 +02:00
Taras Machyshyn
0ab702726d added a new fileManager test, fixes 2015-02-17 12:03:55 +02:00
yuri
6d796c0b20 fix stream note header 2015-02-17 11:50:04 +02:00
yuri
100fec9409 increase notifications panel width 2015-02-17 11:15:54 +02:00
yuri
0b5beeba85 play sound when notification comes 2015-02-17 11:10:07 +02:00
Taras Machyshyn
dec5ab6a16 Improved upgrade/install extenions feature 2015-02-17 11:02:01 +02:00
yuri
909d31b9fe frontend: improve notification 2015-02-17 10:59:28 +02:00
yuri
d9522cf555 fix navbar resize 2015-02-17 10:39:01 +02:00
yuri
e42a877bf6 try catch for importing emails 2015-02-17 10:36:44 +02:00
yuri
68cbadda4c improve header menu 2015-02-16 16:00:51 +02:00
yuri
8beeb64cbf log checkemail errors 2015-02-16 14:56:28 +02:00
yuri
da3a18e766 Duplicate email check by messageId instead of messageIdInternal 2015-02-16 12:43:57 +02:00
yuri
ffb94b960d fix entity manager 2015-02-13 16:24:31 +02:00
yuri
f73ac0779b restoreDate and afterSave 2015-02-13 12:30:36 +02:00
yuri
8003af682e email account create fix 2015-02-12 18:32:37 +02:00
yuri
cbf7bf18f2 fix preferences saving 2015-02-12 17:54:22 +02:00
yuri
ceaa0a8322 refresh email list view after send 2015-02-12 16:49:53 +02:00
yuri
8ca410d994 fix calendar 2015-02-12 16:44:12 +02:00
yuri
5fc7f2509d fix app.js 2015-02-12 16:08:15 +02:00
yuri
f2e1ba3780 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-11 15:53:49 +02:00
yuri
93ac871640 change year 2015-02-11 15:53:39 +02:00
Taras Machyshyn
187a8a02a1 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-11 12:44:18 +02:00
Taras Machyshyn
bcf5686eec fixed getModuleList() 2015-02-11 12:44:05 +02:00
yuri
027507b61e display task in top bar in calendar 2015-02-11 11:59:15 +02:00
yuri
49b3d17952 change default filters 2015-02-11 11:27:00 +02:00
yuri
e0958cfeec improve notifications 2015-02-11 11:00:43 +02:00
Taras Machyshyn
00e12b50b7 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-10 15:02:17 +02:00
Taras Machyshyn
44b8b00106 fixed calendar issue in IE 2015-02-10 15:02:02 +02:00
yuri
e0855e3092 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-10 13:24:10 +02:00
yuri
cb0d70430a ability to remove stream records for admin 2015-02-10 13:22:58 +02:00
Taras Machyshyn
042575ce6b improved 'fieldManager' 2015-02-10 12:46:19 +02:00
yuri
5cd18b57d2 industry list update 2015-02-10 12:25:14 +02:00
yuri
d1e46e3d9a fix field manager 2015-02-10 11:56:13 +02:00
yuri
1a3348f2c2 change user entityDefs links 2015-02-10 11:41:58 +02:00
yuri
75edc5c165 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-10 11:30:17 +02:00
yuri
60184ebbbe ability to disable fields and link for layout manager 2015-02-10 11:30:05 +02:00
yuri
ddb6cb7483 Clear link-multiple field input if not selected 2015-02-10 10:32:45 +02:00
Taras Machyshyn
2ddd44a2fe Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-09 17:11:08 +02:00
Taras Machyshyn
ce55866445 added check 'php' version for extension/upgrade packages 2015-02-09 17:10:46 +02:00
Taras Machyshyn
2bc7f85a58 added getPhpVersion() 2015-02-09 17:07:06 +02:00
yuri
a27df2c41f Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-09 13:01:36 +02:00
yuri
170581dd28 fetched values for linkMultiple fields 2015-02-09 13:01:18 +02:00
Taras Machyshyn
00b0c904ae fixed warnings 2015-02-09 12:14:27 +02:00
yuri
fdab17265f change version 2015-02-09 12:12:27 +02:00
Taras Machyshyn
96d0a7db00 bug fixes 2015-02-09 11:32:39 +02:00
Taras Machyshyn
2835928ac0 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-06 17:37:32 +02:00
Taras Machyshyn
4c22b42b99 fixed cache bug for Extensions 2015-02-06 17:37:11 +02:00
yuri
64ddaa40a8 add reminder time 2015-02-06 16:17:04 +02:00
yuri
bc7006e193 open url field in new pahe 2015-02-06 15:51:45 +02:00
yuri
1ae0a9df28 inline download: add types 2015-02-06 15:05:56 +02:00
yuri
a4f9280eba cleanup 2015-02-06 13:23:36 +02:00
yuri
2528fc34c8 Enable User tab 2015-02-06 13:16:41 +02:00
yuri
979f07bf9b Merge improvements 2 2015-02-06 12:19:13 +02:00
yuri
52bff4de1c fix service factory 2015-02-06 12:16:31 +02:00
Taras Machyshyn
4c017361f8 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-06 11:38:41 +02:00
Taras Machyshyn
84aa1339d9 fixed a bug with php reserved names for classes 2015-02-06 11:38:26 +02:00
yuri
987cd4a121 Import Email: fix issue with text and html attachments 2015-02-06 11:38:20 +02:00
yuri
e50ad5106f Load user data each time app starts 2015-02-06 10:47:34 +02:00
yuri
071cbcb0fb cleanup 2015-02-06 10:47:18 +02:00
yuri
43f1cb9af9 Field Manager:Reload metadata when field is saved 2015-02-06 10:46:50 +02:00
yuri
220c55e9b4 fix entity manager 2015-02-06 10:02:14 +02:00
yuri
927610efb0 clearnup 2015-02-06 09:55:30 +02:00
yuri
78fcaf1fa3 update handlebars 2015-02-06 09:39:55 +02:00
yuri
3bd9af031d cleanup 2015-02-05 19:05:06 +02:00
yuri
16ee3d68e8 merge changes 1 2015-02-05 18:13:08 +02:00
yuri
d49ee3c187 fix select and create 2015-02-05 16:31:39 +02:00
Taras Machyshyn
2e544f1ccf Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-05 15:05:08 +02:00
Taras Machyshyn
1c51125e66 changed 'acceptedVersions' identifier to semantic versioner 2015-02-05 15:04:51 +02:00
yuri
1ed47f5d0d fix navbar update width 2015-02-05 13:35:47 +02:00
yuri
133fa0cb36 note order 2015-02-05 12:53:44 +02:00
yuri
295904bf4c change admin tpl 2015-02-05 11:59:32 +02:00
yuri
951230f7e1 change admin tpl 2015-02-05 11:57:24 +02:00
yuri
ee58886206 fix imap encoding issue 2015-02-05 11:56:52 +02:00
yuri
41575f6f13 add industry 2015-02-05 11:22:36 +02:00
yuri
18ae33d417 send password only to active users 2015-02-05 11:16:22 +02:00
yuri
36daf5a762 iframe 2015-02-05 11:01:55 +02:00
yuri
8a41a2cd05 change calls/meetings dashlet order 2015-02-05 10:24:01 +02:00
yuri
61497d2a01 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-04 17:07:00 +02:00
yuri
a3045a88f3 Inbound Email improve 2015-02-04 17:06:50 +02:00
Taras Machyshyn
fd072520a9 fixed tests 2015-02-04 17:01:01 +02:00
yuri
6ef9d8428f improve js actions 2015-02-03 16:15:39 +02:00
yuri
33a7176165 blockquote 2015-02-03 15:53:52 +02:00
yuri
044de7d744 stream info 2015-02-03 13:15:14 +02:00
yuri
801ba05cd8 afterDelete beforeDelete 2015-02-03 12:45:54 +02:00
yuri
0dc5a4e4f3 after/before Create/Update in Record service 2015-02-03 12:43:56 +02:00
yuri
230ed63e67 getEntityBeforeUpdate 2015-02-03 12:38:10 +02:00
yuri
7316866a1a Merge branch 'hotfix/2.9.3' 2015-02-03 12:36:44 +02:00
yuri
c7be54d9c9 fix textcomplete 2 2015-02-03 12:36:26 +02:00
yuri
dcc118ec5d Merge branch 'hotfix/2.9.3' 2015-02-03 12:18:14 +02:00
yuri
84e98054b7 fix textcomplete 2015-02-03 12:17:39 +02:00
yuri
f37308f5a5 lead source changes 2015-02-03 11:58:54 +02:00
yuri
9688b3be3c clearnup 2015-02-02 18:09:41 +02:00
yuri
6490a1ca97 cleanup 2015-02-02 17:19:55 +02:00
yuri
c15fc89f52 refresh panel 2015-02-02 15:35:35 +02:00
yuri
9207030ccb row actions change to avoid user removing 2015-02-02 15:00:27 +02:00
yuri
018eb44de6 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-02 12:49:46 +02:00
yuri
bc92b96a6b user isActive 2 2015-02-02 12:49:37 +02:00
yuri
ee57c9223c User:isActive 2015-02-02 12:22:37 +02:00
Taras Machyshyn
f799ffc61f Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-02-02 12:10:21 +02:00
Taras Machyshyn
bdb1f2f3b6 improved Util::merge() functionality 2015-02-02 12:10:04 +02:00
yuri
fd755df67b Merge branch 'hotfix/2.9.3' 2015-02-02 11:21:14 +02:00
yuri
e9edd01d08 modified date and id for remove 2015-02-02 11:21:00 +02:00
yuri
dea7e0b33e modified date and id for remove 2015-02-02 11:20:28 +02:00
yuri
25b50baa1a showPanel/hidePanel 2015-01-30 16:09:43 +02:00
yuri
8a53657b9c text search by email address 2015-01-30 15:51:47 +02:00
yuri
c6ecbf4942 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-01-30 14:55:51 +02:00
yuri
56d11807a1 access check in select managers change 2015-01-30 14:52:34 +02:00
yuri
cf3b4b284a record.base 2015-01-30 14:47:14 +02:00
yuri
953a59cb57 no create for users and teams in subpanels 2015-01-30 13:13:38 +02:00
yuri
f60061a387 Merge branch 'hotfix/2.9.3' 2015-01-30 12:54:00 +02:00
yuri
c627d84273 cleanup 2015-01-30 12:53:52 +02:00
Taras Machyshyn
5a16ee0493 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-01-29 17:11:28 +02:00
Taras Machyshyn
14252d6e9e FileManager: improved 'remove()' functionality 2015-01-29 17:11:16 +02:00
yuri
c0de3e1c4e Merge branch 'stable' 2015-01-29 13:00:11 +02:00
yuri
e0c3530ae5 change version 2015-01-29 12:34:13 +02:00
yuri
8b4070f9ae Merge branch 'hotfix/2.9.2' 2015-01-29 12:14:13 +02:00
yuri
0b6c5c2862 Merge branch 'hotfix/2.9.2' of ssh://172.20.0.1/var/git/espo/backend into hotfix/2.9.2 2015-01-29 12:13:54 +02:00
yuri
ce9ff10cb0 fix imap 2015-01-29 12:03:43 +02:00
Taras Machyshyn
de688bfa12 CronManager minor changes 2015-01-29 11:44:47 +02:00
yuri
f4ef5fc36f add tentative event status 2015-01-28 17:04:45 +02:00
yuri
b1e184c6d1 link manager dev 2015-01-28 16:38:27 +02:00
yuri
fbf665fd76 Merge branch 'hotfix/2.9.2' 2015-01-28 10:39:14 +02:00
yuri
95fd66a7a6 Entity::isSaved 2015-01-28 10:38:57 +02:00
yuri
4b98ea79e0 change link manager edit modal 2015-01-28 10:19:17 +02:00
yuri
2e916a2ba8 Merge branch 'master' of ssh://172.20.0.1/var/git/espo/backend 2015-01-27 17:31:13 +02:00
yuri
26a4d1c6ff link manager dev 2015-01-27 17:31:00 +02:00
Taras Machyshyn
33fef4c90e Merge branch 'hotfix/2.9.2' 2015-01-27 17:19:18 +02:00
yuri
2dc15294fc Merge branch 'hotfix/2.9.2' 2015-01-27 16:30:26 +02:00
yuri
171df33736 move logic to Repository::afterSave 2015-01-27 16:27:05 +02:00
yuri
9b846b45bc link manager dev 2015-01-27 16:21:22 +02:00
yuri
bc65975f74 fix in default config 2015-01-27 11:53:55 +02:00
yuri
a5308831f2 Merge branch 'hotfix/2.9.2' 2015-01-27 11:42:17 +02:00
yuri
ef05e4e9f4 add cryptKey to systemItems 2015-01-27 11:40:51 +02:00
yuri
83f0a81eb7 add global Search Entity List param 2015-01-27 11:21:06 +02:00
yuri
9e28d4a261 change meetings and calls dashlets layouts 2015-01-27 10:30:48 +02:00
yuri
a9b3320302 Merge branch 'hotfix/2.9.2' 2015-01-26 17:36:21 +02:00
yuri
b67580ee1d Merge branch 'hotfix/2.9.2' of ssh://172.20.0.1/var/git/espo/backend into hotfix/2.9.2 2015-01-26 17:36:06 +02:00
yuri
884ccb5265 Select Managers changes and Email changes 2015-01-26 17:35:43 +02:00
Taras Machyshyn
ac56c3f79e added 'Cleanup' job 2015-01-26 15:24:10 +02:00
Taras Machyshyn
e218056683 Cron job improvements 2015-01-26 12:28:00 +02:00
yuri
838288a463 Merge branch 'hotfix/2.9.2' 2015-01-26 12:11:30 +02:00
yuri
e74b90d048 fix email notFound 2015-01-26 11:12:25 +02:00
yuri
b2a653a3dc fix App controller 2015-01-26 11:09:54 +02:00
yuri
48ee7d4cc7 Merge branch 'hotfix/2.9.2' of ssh://172.20.0.1/var/git/espo/backend into hotfix/2.9.2 2015-01-23 18:19:25 +02:00
yuri
1d38f80748 improve email accounts 2015-01-23 18:19:02 +02:00
Taras Machyshyn
c712366737 cron jobs improvements and fixes 2015-01-23 17:44:59 +02:00
yuri
71f40fd440 checlemailaccounts job 2015-01-23 17:12:12 +02:00
yuri
9049ebfa8c fix email importer 2015-01-23 17:05:33 +02:00
yuri
f4b9356173 em dev 2015-01-22 15:55:11 +02:00
yuri
beaa4a29ba Merge branch 'hotfix/2.9.1' 2015-01-22 15:32:36 +02:00
yuri
632ee66155 Merge branch 'hotfix/2.9.1' of ssh://172.20.0.1/var/git/espo/backend into hotfix/2.9.1 2015-01-22 15:14:25 +02:00
yuri
4446bc3167 fix oauth client 2015-01-22 15:14:16 +02:00
Taras Machyshyn
785934c7cc Language: chnaged set(), delete() methods 2015-01-22 14:49:40 +02:00
Taras Machyshyn
ac36096d55 Merge branch 'hotfix/2.9.1' of ssh://172.20.0.1/var/git/espo/backend into hotfix/2.9.1 2015-01-22 13:15:15 +02:00
Taras Machyshyn
8ec8cb3c56 Metadata: fixed description 2015-01-22 13:14:55 +02:00
Taras Machyshyn
1f7cb2402d changed Metadata methods set(), get(), added save() 2015-01-22 13:14:06 +02:00
yuri
987ba2faa5 EM dev 2015-01-22 12:10:53 +02:00
yuri
a24f697dca Merge branch 'hotfix/2.9.1' 2015-01-22 11:41:42 +02:00
yuri
3825893ec1 integrations changes 2015-01-22 11:28:17 +02:00
yuri
a4ed78b953 change version 2015-01-21 17:22:08 +02:00
yuri
c7947d0c46 Merge branch 'hotfix/2.9.1' 2015-01-21 17:13:48 +02:00
yuri
eed9059913 integration changes 2015-01-21 17:11:12 +02:00
yuri
20a1053413 fix select manager 2015-01-21 15:50:41 +02:00
yuri
16aca57f17 em deb 2015-01-21 15:30:02 +02:00
yuri
a225748840 make documents customizable 2015-01-21 15:19:06 +02:00
yuri
fff1be9d07 entity manager dev 2015-01-21 15:13:37 +02:00
yuri
b73ca3f1bc change accountInfo messages 2015-01-20 18:14:48 +02:00
yuri
93a4f999b6 entity manager 2 2015-01-20 18:01:05 +02:00
yuri
3030643ce4 entity manager 1 2015-01-20 16:40:04 +02:00
Taras Machyshyn
e814ea9ec6 added 'save()' method to Layout 2015-01-20 15:34:29 +02:00
Taras Machyshyn
de16e9a9ce added '2015' to 'About' page 2015-01-20 12:36:52 +02:00
Taras Machyshyn
62fba991e2 fixed pt_BR translation syntax errors 2015-01-20 12:31:46 +02:00
Taras Machyshyn
832c63e014 added 'getPhpContents()' to FileManager 2015-01-20 12:22:04 +02:00
yuri
46913c3574 cleanup 2015-01-20 10:55:04 +02:00
yuri
0194d364b8 change opportunity list small layout 2015-01-19 15:09:30 +02:00
yuri
1c7785f6ba fix salutation list 2015-01-19 10:41:45 +02:00
761 changed files with 25866 additions and 9228 deletions

View File

@@ -17,7 +17,7 @@
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
************************************************************************/
module.exports = function (grunt) {
var jsFilesToMinify = [
@@ -65,10 +65,10 @@ module.exports = function (grunt) {
'client/src/controllers/base.js',
'client/src/view.js',
];
grunt.initConfig({
pkg: grunt.file.readJSON('package.json'),
mkdir: {
tmp: {
options: {
@@ -126,7 +126,8 @@ module.exports = function (grunt) {
'modules/**',
'img/**',
'css/**',
'sounds/**'
'sounds/**',
'custom/**'
],
dest: 'build/tmp/client',
},
@@ -154,6 +155,7 @@ module.exports = function (grunt) {
'bootstrap.php',
'cron.php',
'rebuild.php',
'upgrade.php',
'index.php',
'LICENSE.txt',
'.htaccess',

View File

@@ -2,7 +2,7 @@
<a href='http://www.espocrm.com'>EspoCRM is an Open Source CRM</a> (Customer Relationship Management) software that allows you to see, enter and evaluate all your company relationships regardless of the type. People, companies or opportunities - all in an easy and intuitive interface.
It's a web application with a frontend designed as a single page application based on backbone.js and a RESTful backend written in PHP.
It's a web application with a frontend designed as a single page application based on backbone.js and a REST API backend written in PHP.
Download the latest release from our [website](http://www.espocrm.com).
@@ -33,6 +33,21 @@ You need to have nodejs and Grunt CLI installed.
The build will be created in the `build` directory.
### How to make translation
Build po file with command:
`node po.js en_EN`
(specify needed language instead of en_EN)
After that tranlate the generated po file.
Build json files from the translated po file:
1. Put your po file espocrm-en_EN.po to the `build` directory
2. Run `node lang.js en_EN`
The files will be created in build directory.
### License
EspoCRM is published under the GNU GPLv3 [license](https://raw.githubusercontent.com/espocrm/espocrm/master/LICENSE.txt).

View File

@@ -72,7 +72,7 @@ class Admin extends \Espo\Core\Controllers\Base
{
$upgradeManager = new \Espo\Core\UpgradeManager($this->getContainer());
$upgradeManager->install($data['id']);
$upgradeManager->install($data);
return true;
}

View File

@@ -18,13 +18,13 @@
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
************************************************************************/
namespace Espo\Controllers;
use \Espo\Core\Exceptions\BadRequest;
class App extends \Espo\Core\Controllers\Record
class App extends \Espo\Core\Controllers\Base
{
public function actionUser()
{

View File

@@ -18,7 +18,7 @@
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
************************************************************************/
namespace Espo\Controllers;
@@ -29,18 +29,18 @@ use \Espo\Core\Exceptions\Error;
class Email extends \Espo\Core\Controllers\Record
{
public function actionGetCopiedAttachments($params, $data, $request)
{
{
$id = $request->get('id');
return $this->getRecordService()->getCopiedAttachments($id);
}
public function actionSendTestEmail($params, $data, $request)
{
if (!$request->isPost()) {
throw new BadRequest();
}
if (empty($data['password'])) {
if ($data['type'] == 'preferences') {
if (!$this->getUser()->isAdmin() && $data['id'] != $this->getUser()->id) {
@@ -50,7 +50,7 @@ class Email extends \Espo\Core\Controllers\Record
if (!$preferences) {
throw new Error();
}
$data['password'] = $this->getContainer()->get('crypt')->decrypt($preferences->get('smtpPassword'));
} else {
if (!$this->getUser()->isAdmin()) {
@@ -59,8 +59,29 @@ class Email extends \Espo\Core\Controllers\Record
$data['password'] = $this->getConfig()->get('smtpPassword');
}
}
return $this->getRecordService()->sendTestEmail($data);
}
public function actionMarkAsRead($params, $data, $request)
{
if (!$request->isPost()) {
throw new BadRequest();
}
if (empty($data['ids']) || !is_array($data['ids'])) {
throw new BadRequest();
}
$ids = $data['ids'];
return $this->getRecordService()->markAsReadByIds($ids);
}
public function actionMarkAllAsRead($params, $data, $request)
{
if (!$request->isPost()) {
throw new BadRequest();
}
return $this->getRecordService()->markAllAsRead();
}
}

View File

@@ -18,16 +18,16 @@
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
************************************************************************/
namespace Espo\Controllers;
use \Espo\Core\Exceptions\Forbidden;
class EmailAccount extends \Espo\Core\Controllers\Record
{
{
public function actionGetFolders($params, $data, $request)
{
{
return $this->getRecordService()->getFolders(array(
'host' => $request->get('host'),
'port' => $request->get('port'),

View File

@@ -0,0 +1,227 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Controllers;
use \Espo\Core\Exceptions\BadRequest;
use \Espo\Core\Exceptions\Forbidden;
use \Espo\Core\Exceptions\Error;
class EntityManager extends \Espo\Core\Controllers\Base
{
protected function checkControllerAccess()
{
if (!$this->getUser()->isAdmin()) {
throw new Forbidden();
}
}
public function actionCreateEntity($params, $data, $request)
{
if (!$request->isPost()) {
throw new BadRequest();
}
if (empty($data['name']) || empty($data['type'])) {
throw new BadRequest();
}
$name = $data['name'];
$type = $data['type'];
$name = filter_var($name, \FILTER_SANITIZE_STRING);
$type = filter_var($type, \FILTER_SANITIZE_STRING);
$params = array();
if (!empty($data['labelSingular'])) {
$params['labelSingular'] = $data['labelSingular'];
}
if (!empty($data['labelPlural'])) {
$params['labelPlural'] = $data['labelPlural'];
}
if (!empty($data['stream'])) {
$params['stream'] = $data['stream'];
}
$result = $this->getContainer()->get('entityManagerUtil')->create($name, $type, $params);
if ($result) {
$tabList = $this->getConfig()->get('tabList', []);
$tabList[] = $name;
$this->getConfig()->set('tabList', $tabList);
$this->getConfig()->save();
$this->getContainer()->get('dataManager')->rebuild();
} else {
throw new Error();
}
return true;
}
public function actionUpdateEntity($params, $data, $request)
{
if (!$request->isPost()) {
throw new BadRequest();
}
if (empty($data['name'])) {
throw new BadRequest();
}
$name = $data['name'];
$name = filter_var($name, \FILTER_SANITIZE_STRING);
$result = $this->getContainer()->get('entityManagerUtil')->update($name, $data);
if ($result) {
$this->getContainer()->get('dataManager')->clearCache();
} else {
throw new Error();
}
return true;
}
public function actionRemoveEntity($params, $data, $request)
{
if (!$request->isPost()) {
throw new BadRequest();
}
if (empty($data['name'])) {
throw new BadRequest();
}
$name = $data['name'];
$name = filter_var($name, \FILTER_SANITIZE_STRING);
$result = $this->getContainer()->get('entityManagerUtil')->delete($name);
if ($result) {
$tabList = $this->getConfig()->get('tabList', []);
if (($key = array_search($name, $tabList)) !== false) {
unset($tabList[$key]);
$tabList = array_values($tabList);
}
$this->getConfig()->set('tabList', $tabList);
$this->getConfig()->save();
$this->getContainer()->get('dataManager')->clearCache();
} else {
throw new Error();
}
return true;
}
public function actionCreateLink($params, $data, $request)
{
if (!$request->isPost()) {
throw new BadRequest();
}
$paramList = [
'entity',
'entityForeign',
'link',
'linkForeign',
'label',
'labelForeign',
'linkType'
];
$d = array();
foreach ($paramList as $item) {
if (empty($data[$item])) {
throw new BadRequest();
}
$d[$item] = filter_var($data[$item], \FILTER_SANITIZE_STRING);
}
$result = $this->getContainer()->get('entityManagerUtil')->createLink($d);
if ($result) {
$this->getContainer()->get('dataManager')->rebuild();
} else {
throw new Error();
}
return true;
}
public function actionUpdateLink($params, $data, $request)
{
if (!$request->isPost()) {
throw new BadRequest();
}
$paramList = [
'entity',
'entityForeign',
'link',
'linkForeign',
'label',
'labelForeign'
];
$d = array();
foreach ($paramList as $item) {
$d[$item] = filter_var($data[$item], \FILTER_SANITIZE_STRING);
}
$result = $this->getContainer()->get('entityManagerUtil')->updateLink($d);
if ($result) {
$this->getContainer()->get('dataManager')->clearCache();
} else {
throw new Error();
}
return true;
}
public function actionRemoveLink($params, $data, $request)
{
if (!$request->isPost()) {
throw new BadRequest();
}
$paramList = [
'entity',
'link',
];
$d = array();
foreach ($paramList as $item) {
$d[$item] = filter_var($data[$item], \FILTER_SANITIZE_STRING);
}
$result = $this->getContainer()->get('entityManagerUtil')->deleteLink($d);
if ($result) {
$this->getContainer()->get('dataManager')->clearCache();
} else {
throw new Error();
}
return true;
}
}

View File

@@ -61,7 +61,7 @@ class Extension extends \Espo\Core\Controllers\Record
$manager = new \Espo\Core\ExtensionManager($this->getContainer());
$manager->install($data['id']);
$manager->install($data);
return true;
}
@@ -74,7 +74,7 @@ class Extension extends \Espo\Core\Controllers\Record
$manager = new \Espo\Core\ExtensionManager($this->getContainer());
$manager->uninstall($data['id']);
$manager->uninstall($data);
return true;
}
@@ -103,7 +103,7 @@ class Extension extends \Espo\Core\Controllers\Record
{
$manager = new \Espo\Core\ExtensionManager($this->getContainer());
$manager->delete($params['id']);
$manager->delete($params);
return true;
}

View File

@@ -28,10 +28,10 @@ use \Espo\Core\Exceptions\Forbidden;
class ExternalAccount extends \Espo\Core\Controllers\Record
{
public static $defaultAction = 'list';
public function actionList($params, $data, $request)
{
$integrations = $this->getEntityManager()->getRepository('Integration')->find();
$integrations = $this->getEntityManager()->getRepository('Integration')->find();
$arr = array();
foreach ($integrations as $entity) {
if ($entity->get('enabled') && $this->getMetadata()->get('integrations.' . $entity->id .'.allowUserAccounts')) {
@@ -44,22 +44,22 @@ class ExternalAccount extends \Espo\Core\Controllers\Record
'list' => $arr
);
}
public function actionGetOAuth2Info($params, $data, $request)
{
$id = $request->get('id');
list($integration, $userId) = explode('__', $id);
if ($this->getUser()->id != $userId) {
throw new Forbidden();
}
}
$entity = $this->getEntityManager()->getEntity('Integration', $integration);
if ($entity) {
return array(
'clientId' => $entity->get('clientId'),
'redirectUri' => $this->getConfig()->get('siteUrl') . '/oauthcallback',
'redirectUri' => $this->getConfig()->get('siteUrl') . '?entryPoint=oauthCallback',
'isConnected' => $this->getRecordService()->ping($integration, $userId)
);
}
@@ -67,57 +67,57 @@ class ExternalAccount extends \Espo\Core\Controllers\Record
public function actionRead($params, $data, $request)
{
list($integration, $userId) = explode('__', $params['id']);
list($integration, $userId) = explode('__', $params['id']);
if ($this->getUser()->id != $userId) {
throw new Forbidden();
}
$entity = $this->getEntityManager()->getEntity('ExternalAccount', $params['id']);
}
$entity = $this->getEntityManager()->getEntity('ExternalAccount', $params['id']);
return $entity->toArray();
}
public function actionUpdate($params, $data)
{
return $this->actionPatch($params, $data);
}
public function actionPatch($params, $data)
{
list($integration, $userId) = explode('__', $params['id']);
if ($this->getUser()->id != $userId) {
throw new Forbidden();
}
if (isset($data['enabled']) && !$data['enabled']) {
$data['data'] = null;
}
}
$entity = $this->getEntityManager()->getEntity('ExternalAccount', $params['id']);
$entity->set($data);
$this->getEntityManager()->saveEntity($entity);
return $entity->toArray();
return $entity->toArray();
}
public function actionAuthorizationCode($params, $data, $request)
{
if (!$request->isPost()) {
throw new Error('Bad HTTP method type.');
}
}
$id = $data['id'];
$code = $data['code'];
list($integration, $userId) = explode('__', $id);
list($integration, $userId) = explode('__', $id);
if ($this->getUser()->id != $userId) {
throw new Forbidden();
}
$service = $this->getRecordService();
$service = $this->getRecordService();
return $service->authorizationCode($integration, $userId, $code);
}
}

View File

@@ -18,50 +18,94 @@
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
************************************************************************/
namespace Espo\Controllers;
use Espo\Core\Utils as Utils;
use \Espo\Core\Exceptions\Error;
use \Espo\Core\Exceptions\Forbidden;
use \Espo\Core\Exceptions\BadRequest;
class Import extends \Espo\Core\Controllers\Record
{
protected function checkControllerAccess()
{
if (!$this->getUser()->isAdmin()) {
throw new Forbidden();
}
}
public function actionPatch($params, $data)
{
throw new BadRequest();
}
public function actionUpdate($params, $data)
{
throw new BadRequest();
}
public function actionMassUpdate($params, $data, $request)
{
throw new BadRequest();
}
public function actionCreateLink($params, $data)
{
throw new BadRequest();
}
public function actionRemoveLink($params, $data)
{
throw new BadRequest();
}
class Import extends \Espo\Core\Controllers\Base
{
protected function getFileManager()
{
return $this->getContainer()->get('fileManager');
}
protected function getEntityManager()
{
return $this->getContainer()->get('entityManager');
}
public function actionUploadFile($params, $data)
{
{
$contents = $data;
$attachment = $this->getEntityManager()->getEntity('Attachment');
$attachment->set('type', 'text/csv');
$attachment->set('role', 'Import File');
$attachment->set('role', 'Import File');
$attachment->set('name', 'import-file.csv');
$this->getEntityManager()->saveEntity($attachment);
$this->getFileManager()->putContents('data/upload/' . $attachment->id, $contents);
return array(
'attachmentId' => $attachment->id
);
}
public function actionRevert($params, $data)
{
return $this->getService('Import')->revert($data['entityType'], $data['idsToRemove']);
{
if (empty($data['id'])) {
throw new BadRequest();
}
return $this->getService('Import')->revert($data['id']);
}
public function actionRemoveDuplicates($params, $data)
{
if (empty($data['id'])) {
throw new BadRequest();
}
return $this->getService('Import')->removeDuplicates($data['id']);
}
public function actionCreate($params, $data)
{
{
$importParams = array(
'headerRow' => $data['headerRow'],
'fieldDelimiter' => $data['fieldDelimiter'],
@@ -74,13 +118,13 @@ class Import extends \Espo\Core\Controllers\Base
'defaultValues' => $data['defaultValues'],
'action' => $data['action'],
);
$attachmentId = $data['attachmentId'];
if (!$this->getAcl()->check($data['entityType'], 'edit')) {
throw new Forbidden();
}
return $this->getService('Import')->import($data['entityType'], $data['fields'], $attachmentId, $importParams);
}
}

View File

@@ -33,7 +33,7 @@ class Layout extends \Espo\Core\Controllers\Base
{
$data = $this->getContainer()->get('layout')->get($params['scope'], $params['name']);
if (empty($data)) {
throw new NotFound("Layout " . $params['scope'] . ":" . $params['name'] . ' is not found');
throw new NotFound("Layout " . $params['scope'] . ":" . $params['name'] . ' is not found.');
}
return $data;
}
@@ -43,16 +43,18 @@ class Layout extends \Espo\Core\Controllers\Base
if (!$this->getUser()->isAdmin()) {
throw new Forbidden();
}
$result = $this->getContainer()->get('layout')->set($data, $params['scope'], $params['name']);
$layoutManager = $this->getContainer()->get('layout');
$layoutManager->set($data, $params['scope'], $params['name']);
$result = $layoutManager->save();
if ($result === false) {
throw new Error("Error while saving layout");
throw new Error("Error while saving layout.");
}
$this->getContainer()->get('dataManager')->updateCacheTimestamp();
return $this->getContainer()->get('layout')->get($params['scope'], $params['name']);
return $layoutManager->get($params['scope'], $params['name']);
}
public function actionPatch($params, $data)

View File

@@ -18,7 +18,7 @@
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
************************************************************************/
namespace Espo\Controllers;
@@ -28,22 +28,22 @@ use \Espo\Core\Exceptions\BadRequest;
use \Espo\Core\Exceptions\NotFound;
class Preferences extends \Espo\Core\Controllers\Base
{
{
protected function getPreferences()
{
return $this->getContainer()->get('preferences');
}
protected function getEntityManager()
{
return $this->getContainer()->get('entityManager');
}
protected function getCrypt()
{
return $this->getContainer()->get('crypt');
}
protected function handleUserAccess($userId)
{
if (!$this->getUser()->isAdmin()) {
@@ -52,7 +52,7 @@ class Preferences extends \Espo\Core\Controllers\Base
}
}
}
public function actionDelete($params, $data)
{
$userId = $params['id'];
@@ -60,38 +60,38 @@ class Preferences extends \Espo\Core\Controllers\Base
throw new BadRequest();
}
$this->handleUserAccess($userId);
return $this->getEntityManager()->getRepository('Preferences')->resetToDefaults($userId);
return $this->getEntityManager()->getRepository('Preferences')->resetToDefaults($userId);
}
public function actionPatch($params, $data)
{
return $this->actionUpdate($params, $data);
}
}
public function actionUpdate($params, $data)
{
$userId = $params['id'];
$this->handleUserAccess($userId);
if (array_key_exists('smtpPassword', $data)) {
$data['smtpPassword'] = $this->getCrypt()->encrypt($data['smtpPassword']);
}
$user = $this->getEntityManager()->getEntity('User', $userId);
$user = $this->getEntityManager()->getEntity('User', $userId);
$entity = $this->getEntityManager()->getEntity('Preferences', $userId);
if ($entity) {
if ($entity && $user) {
$entity->set($data);
$this->getEntityManager()->saveEntity($entity);
$entity->set('smtpEmailAddress', $user->get('emailAddress'));
$entity->set('smtpEmailAddress', $user->get('emailAddress'));
$entity->set('name', $user->get('name'));
$entity->clear('smtpPassword');
return $entity->toArray();
return $entity->toArray();
}
throw new Error();
}
@@ -101,18 +101,19 @@ class Preferences extends \Espo\Core\Controllers\Base
$userId = $params['id'];
$this->handleUserAccess($userId);
$entity = $this->getEntityManager()->getEntity('Preferences', $userId);
$entity = $this->getEntityManager()->getEntity('Preferences', $userId);
$user = $this->getEntityManager()->getEntity('User', $userId);
if (!$entity || !$user) {
throw new NotFound();
}
$entity->set('smtpEmailAddress', $user->get('emailAddress'));
$entity->set('name', $user->get('name'));
$entity->clear('smtpPassword');
if ($entity) {
return $entity->toArray();
}
throw new NotFound();
return $entity->toArray();
}
}

View File

@@ -18,7 +18,7 @@
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
************************************************************************/
namespace Espo\Controllers;
@@ -35,30 +35,33 @@ class User extends \Espo\Core\Controllers\Record
if (empty($userId)) {
throw new Error();
}
if (!$this->getUser()->isAdmin() && $this->getUser()->id != $userId) {
throw new Forbidden();
}
$user = $this->getEntityManager()->getEntity('User', $userId);
if (empty($user)) {
throw new NotFound();
}
$acl = new \Espo\Core\Acl($user, $this->getConfig(), $this->getContainer()->get('fileManager'), $this->getMetadata());
return $acl->toArray();
}
public function actionChangeOwnPassword($params, $data)
public function actionChangeOwnPassword($params, $data, $request)
{
if (!$request->isPost()) {
throw new BadRequest();
}
return $this->getService('User')->changePassword($this->getUser()->id, $data['password']);
}
public function actionChangePasswordByRequest($params, $data, $request)
{
if (!$request->isPost()) {
throw new Forbidden();
throw new BadRequest();
}
if (empty($data['requestId']) || empty($data['password'])) {
throw new BadRequest();

View File

@@ -28,7 +28,9 @@ use \Espo\ORM\Entity;
class Acl
{
private $data = array();
private $data = array(
'table' => array()
);
private $cacheFile;
@@ -74,16 +76,16 @@ class Acl
public function checkScope($scope, $action = null, $isOwner = null, $inTeam = null, $entity = null)
{
if (array_key_exists($scope, $this->data)) {
if ($this->data[$scope] === false) {
if (array_key_exists($scope, $this->data['table'])) {
if ($this->data['table'][$scope] === false) {
return false;
}
if ($this->data[$scope] === true) {
if ($this->data['table'][$scope] === true) {
return true;
}
if (!is_null($action)) {
if (array_key_exists($action, $this->data[$scope])) {
$value = $this->data[$scope][$action];
if (array_key_exists($action, $this->data['table'][$scope])) {
$value = $this->data['table'][$scope][$action];
if ($value === 'all' || $value === true) {
return true;
@@ -125,14 +127,29 @@ class Acl
return $this->data;
}
public function get($permission)
{
if ($this->user->isAdmin()) {
return true;
}
if ($permission == 'table') {
return null;
}
if (array_key_exists($permission, $this->data)) {
return $this->data[$permission];
}
return null;
}
public function getLevel($scope, $action)
{
if ($this->user->isAdmin()) {
return 'all';
}
if (array_key_exists($scope, $this->data)) {
if (array_key_exists($action, $this->data[$scope])) {
return $this->data[$scope][$action];
if (array_key_exists($scope, $this->data['table'])) {
if (array_key_exists($action, $this->data['table'][$scope])) {
return $this->data['table'][$scope][$action];
}
}
return false;
@@ -156,8 +173,8 @@ class Acl
public function checkReadOnlyTeam($scope)
{
if (isset($this->data[$scope]) && isset($this->data[$scope]['read'])) {
return $this->data[$scope]['read'] === 'team';
if (isset($this->data['table'][$scope]) && isset($this->data['table'][$scope]['read'])) {
return $this->data['table'][$scope]['read'] === 'team';
}
return false;
}
@@ -167,8 +184,8 @@ class Acl
if ($this->user->isAdmin()) {
return false;
}
if (isset($this->data[$scope]) && isset($this->data[$scope]['read'])) {
return $this->data[$scope]['read'] === 'own';
if (isset($this->data['table'][$scope]) && isset($this->data['table'][$scope]['read'])) {
return $this->data['table'][$scope]['read'] === 'own';
}
return false;
}
@@ -213,12 +230,14 @@ class Acl
private function load()
{
$aclTables = array();
$aclTables = [];
$assignmentPermissionList = [];
$userRoles = $this->user->get('roles');
foreach ($userRoles as $role) {
$aclTables[] = $role->get('data');
$assignmentPermissionList[] = $role->get('assignmentPermission');
}
$teams = $this->user->get('teams');
@@ -226,21 +245,47 @@ class Acl
$teamRoles = $team->get('roles');
foreach ($teamRoles as $role) {
$aclTables[] = $role->get('data');
$assignmentPermissionList[] = $role->get('assignmentPermission');
}
}
$this->data = $this->merge($aclTables);
$this->data['table'] = $this->merge($aclTables);
$this->data['assignmentPermission'] = $this->mergeValues($assignmentPermissionList, 'all');
}
private function initSolid()
{
if (!$this->metadata) {
return;
}
$data = $this->metadata->get('app.acl.solid', array());
foreach ($data as $entityName => $item) {
$this->data[$entityName] = $item;
$this->data['table'][$entityName] = $item;
}
}
private function mergeValues(array $list, $defaultValue)
{
$result = null;
foreach ($list as $level) {
if ($level != 'not-set') {
if (is_null($result)) {
$result = $level;
continue;
}
if (array_search($result, $this->levelList) > array_search($level, $this->levelList)) {
$result = $level;
}
}
}
if (is_null($result)) {
$result = $defaultValue;
}
return $result;
}
private function merge($tables)
{
$data = array();

View File

@@ -79,6 +79,28 @@ class Container
return $className;
}
protected function loadLog()
{
$logConfig = $this->get('config')->get('logger');
$log = new \Espo\Core\Utils\Log('Espo');
$levelCode = $log->getLevelCode($logConfig['level']);
if ($logConfig['isRotate']) {
$handler = new \Espo\Core\Utils\Log\Monolog\Handler\RotatingFileHandler($logConfig['path'], $logConfig['maxRotateFiles'], $levelCode);
} else {
$handler = new \Espo\Core\Utils\Log\Monolog\Handler\StreamHandler($logConfig['path'], $levelCode);
}
$log->pushHandler($handler);
$errorHandler = new \Monolog\ErrorHandler($log);
$errorHandler->registerExceptionHandler(null, false);
$errorHandler->registerErrorHandler(array(), false);
return $log;
}
protected function loadContainer()
{
return $this;
@@ -221,7 +243,7 @@ class Container
private function loadScheduledJob()
{
return new \Espo\Core\Cron\ScheduledJob(
return new \Espo\Core\Utils\ScheduledJob(
$this
);
}

View File

@@ -25,6 +25,7 @@ namespace Espo\Core\Controllers;
use \Espo\Core\Exceptions\Error;
use \Espo\Core\Exceptions\Forbidden;
use \Espo\Core\Exceptions\NotFound;
use \Espo\Core\Exceptions\BadRequest;
use \Espo\Core\Utils\Util;
class Record extends Base
@@ -193,9 +194,17 @@ class Record extends Base
$ids = $request->get('ids');
$where = $request->get('where');
$byWhere = $request->get('byWhere');
$params = array();
if ($byWhere) {
$params['where'] = $where;
} else {
$params['ids'] = $ids;
}
return array(
'id' => $this->getRecordService()->export($ids, $where)
'id' => $this->getRecordService()->export($params)
);
}
@@ -204,12 +213,20 @@ class Record extends Base
if (!$this->getAcl()->check($this->name, 'edit')) {
throw new Forbidden();
}
if (empty($data['attributes'])) {
throw new BadRequest();
}
$params = array();
if (array_key_exists('where', $data) && !empty($data['byWhere'])) {
$params['where'] = json_decode(json_encode($data['where']), true);
} else if (array_key_exists('ids', $data)) {
$params['ids'] = $data['ids'];
}
$ids = $data['ids'];
$where = $data['where'];
$attributes = $data['attributes'];
$idsUpdated = $this->getRecordService()->massUpdate($attributes, $ids, $where);
$idsUpdated = $this->getRecordService()->massUpdate($attributes, $params);
return $idsUpdated;
}
@@ -220,37 +237,56 @@ class Record extends Base
throw new Forbidden();
}
$ids = $data['ids'];
$where = $data['where'];
$params = array();
if (array_key_exists('where', $data) && !empty($data['byWhere'])) {
$where = json_decode(json_encode($data['where']), true);
$params['where'] = $where;
}
if (array_key_exists('ids', $data)) {
$where = json_decode(json_encode($data['where']), true);
$params['ids'] = $data['ids'];
}
$idsRemoved = $this->getRecordService()->massRemove($ids, $where);
$idsRemoved = $this->getRecordService()->massRemove($params);
return $idsRemoved;
}
public function actionCreateLink($params, $data)
{
if (empty($params['id']) || empty($params['link'])) {
throw BadRequest();
}
$id = $params['id'];
$link = $params['link'];
$foreignIds = array();
if (isset($data['id'])) {
$foreignIds[] = $data['id'];
}
if (isset($data['ids']) && is_array($data['ids'])) {
foreach ($data['ids'] as $foreignId) {
$foreignIds[] = $foreignId;
if (!empty($data['massRelate'])) {
if (!is_array($data['where'])) {
throw new BadRequest();
}
$where = json_decode(json_encode($data['where']), true);
return $this->getRecordService()->linkEntityMass($id, $link, $where);
} else {
$foreignIds = array();
if (isset($data['id'])) {
$foreignIds[] = $data['id'];
}
if (isset($data['ids']) && is_array($data['ids'])) {
foreach ($data['ids'] as $foreignId) {
$foreignIds[] = $foreignId;
}
}
}
$result = false;
foreach ($foreignIds as $foreignId) {
if ($this->getRecordService()->linkEntity($id, $link, $foreignId)) {
$result = $result || true;
$result = false;
foreach ($foreignIds as $foreignId) {
if ($this->getRecordService()->linkEntity($id, $link, $foreignId)) {
$result = true;
}
}
if ($result) {
return true;
}
}
if ($result) {
return true;
}
throw new Error();
@@ -261,6 +297,10 @@ class Record extends Base
$id = $params['id'];
$link = $params['link'];
if (empty($params['id']) || empty($params['link'])) {
throw BadRequest();
}
$foreignIds = array();
if (isset($data['id'])) {
$foreignIds[] = $data['id'];
@@ -301,5 +341,24 @@ class Record extends Base
$id = $params['id'];
return $this->getRecordService()->unfollow($id);
}
public function actionMerge($params, $data, $request)
{
if (!$request->isPost()) {
throw new BadRequest();
}
if (empty($data['targetId']) || empty($data['sourceIds']) || !is_array($data['sourceIds'])) {
throw new BadRequest();
}
$targetId = $data['targetId'];
$sourceIds = $data['sourceIds'];
if (!$this->getAcl()->check($this->name, 'edit')) {
throw new Forbidden();
}
return $this->getRecordService()->merge($targetId, $sourceIds);
}
}

View File

@@ -22,17 +22,18 @@
namespace Espo\Core;
use \PDO;
use Espo\Core\Utils\Json;
use Espo\Core\Exceptions\NotFound;
class CronManager
{
private $container;
private $config;
private $fileManager;
private $entityManager;
private $scheduledJobCron;
private $serviceCron;
private $jobService;
private $scheduledJobService;
private $scheduledJobUtil;
const PENDING = 'Pending';
const RUNNING = 'Running';
@@ -41,19 +42,18 @@ class CronManager
protected $lastRunTime = 'data/cache/application/cronLastRunTime.php';
public function __construct(\Espo\Core\Container $container)
{
$this->container = $container;
$this->config = $this->container->get('config');
$this->fileManager = $this->container->get('fileManager');
$this->entityManager = $this->container->get('entityManager');
$this->serviceFactory = $this->container->get('serviceFactory');
$this->scheduledJobCron = $this->container->get('scheduledJob');
$this->serviceCron = new \Espo\Core\Cron\Service( $this->container->get('serviceFactory'));
$this->jobService = $this->container->get('serviceFactory')->create('job');
$this->scheduledJobService = $this->container->get('serviceFactory')->create('scheduledJob');
$this->scheduledJobUtil = $this->container->get('scheduledJob');
$this->cronJob = new \Espo\Core\Utils\Cron\Job($this->config, $this->entityManager);
$this->cronScheduledJob = new \Espo\Core\Utils\Cron\ScheduledJob($this->config, $this->entityManager);
}
protected function getContainer()
@@ -71,32 +71,38 @@ class CronManager
return $this->fileManager;
}
protected function getJobService()
protected function getEntityManager()
{
return $this->jobService;
return $this->entityManager;
}
protected function getScheduledJobService()
protected function getServiceFactory()
{
return $this->scheduledJobService;
return $this->serviceFactory;
}
protected function getScheduledJobCron()
protected function getScheduledJobUtil()
{
return $this->scheduledJobCron;
return $this->scheduledJobUtil;
}
protected function getServiceCron()
protected function getCronJob()
{
return $this->serviceCron;
return $this->cronJob;
}
protected function getCronScheduledJob()
{
return $this->cronScheduledJob;
}
protected function getLastRunTime()
{
$lastRunTime = $this->getFileManager()->getContents($this->lastRunTime);
if (!is_int($lastRunTime)) {
$lastRunTime = time() - (intval($this->getConfig()->get('cron.minExecutionTime')) + 60);
$lastRunData = $this->getFileManager()->getPhpContents($this->lastRunTime);
$lastRunTime = time() - intval($this->getConfig()->get('cron.minExecutionTime')) - 1;
if (is_array($lastRunData) && !empty($lastRunData['time'])) {
$lastRunTime = $lastRunData['time'];
}
return $lastRunTime;
@@ -104,7 +110,10 @@ class CronManager
protected function setLastRunTime($time)
{
return $this->getFileManager()->putContentsPHP($this->lastRunTime, $time);
$data = array(
'time' => $time,
);
return $this->getFileManager()->putPhpContents($this->lastRunTime, $data);
}
protected function checkLastRunTime()
@@ -120,65 +129,143 @@ class CronManager
return false;
}
/**
* Run Cron
*
* @return void
*/
public function run()
{
if (!$this->checkLastRunTime()) {
$GLOBALS['log']->info('Cron Manager: Stop cron running, too frequency execution');
return; //stop cron running, too frequency execution
$GLOBALS['log']->info('CronManager: Stop cron running, too frequent execution.');
return;
}
$this->setLastRunTime(time());
$entityManager = $this->getEntityManager();
$cronJob = $this->getCronJob();
$cronScheduledJob = $this->getCronScheduledJob();
//Check scheduled jobs and create related jobs
$this->createJobsFromScheduledJobs();
$pendingJobs = $this->getJobService()->getPendingJobs();
$pendingJobs = $cronJob->getPendingJobs();
foreach ($pendingJobs as $job) {
$this->getJobService()->updateEntity($job['id'], array(
'status' => self::RUNNING,
));
$jobEntity = $entityManager->getEntity('Job', $job['id']);
if (!isset($jobEntity)) {
$GLOBALS['log']->error('CronManager: empty Job entity ['.$job['id'].'].');
continue;
}
$jobEntity->set('status', self::RUNNING);
$entityManager->saveEntity($jobEntity);
$isSuccess = true;
try {
if (!empty($job['scheduled_job_id'])) {
$this->getScheduledJobCron()->run($job);
$this->runScheduledJob($job);
} else {
$this->getServiceCron()->run($job);
$this->runService($job);
}
} catch (\Exception $e) {
$isSuccess = false;
$GLOBALS['log']->error('Failed job running, job ['.$job['id'].']. Error Details: '.$e->getMessage());
$GLOBALS['log']->error('CronManager: Failed job running, job ['.$job['id'].']. Error Details: '.$e->getMessage());
}
$status = $isSuccess ? self::SUCCESS : self::FAILED;
$this->getJobService()->updateEntity($job['id'], array(
'status' => $status,
));
$jobEntity->set('status', $status);
$entityManager->saveEntity($jobEntity);
//set status in the schedulerJobLog
if (!empty($job['scheduled_job_id'])) {
$this->getScheduledJobService()->addLogRecord($job['scheduled_job_id'], $status);
$cronScheduledJob->addLogRecord($job['scheduled_job_id'], $status);
}
}
}
/**
* Run Scheduled Job
*
* @param array $job
*
* @return void
*/
protected function runScheduledJob(array $job)
{
$jobName = $job['method'];
$className = $this->getScheduledJobUtil()->get($jobName);
if ($className === false) {
throw new NotFound();
}
$jobClass = new $className($this->container);
$method = $this->getScheduledJobUtil()->getMethodName();
if (!method_exists($jobClass, $method)) {
throw new NotFound();
}
$jobClass->$method();
}
/**
* Run Service
*
* @param array $job
*
* @return void
*/
protected function runService(array $job)
{
$serviceName = $job['service_name'];
if (!$this->getServiceFactory()->checkExists($serviceName)) {
throw new NotFound();
}
$service = $this->getServiceFactory()->create($serviceName);
$serviceMethod = $job['method'];
if (!method_exists($service, $serviceMethod)) {
throw new NotFound();
}
$data = $job['data'];
if (Json::isJSON($data)) {
$data = Json::decode($data, true);
}
$service->$serviceMethod($data);
}
/**
* Check scheduled jobs and create related jobs
* @return array List of created Jobs
*
* @return array List of created Jobs
*/
protected function createJobsFromScheduledJobs()
{
$activeScheduledJobs = $this->getScheduledJobService()->getActiveJobs();
$entityManager = $this->getEntityManager();
$activeScheduledJobs = $this->getCronScheduledJob()->getActiveJobs();
$cronJob = $this->getCronJob();
$runningScheduledJobs = $cronJob->getActiveJobs('scheduled_job_id', self::RUNNING, PDO::FETCH_COLUMN);
$createdJobs = array();
foreach ($activeScheduledJobs as $scheduledJob) {
if (in_array($scheduledJob['id'], $runningScheduledJobs)) {
continue;
}
$scheduling = $scheduledJob['scheduling'];
$cronExpression = \Cron\CronExpression::factory($scheduling);
@@ -186,26 +273,30 @@ class CronManager
try {
$prevDate = $cronExpression->getPreviousRunDate()->format('Y-m-d H:i:s');
} catch (\Exception $e) {
$GLOBALS['log']->error('ScheduledJob ['.$scheduledJob['id'].']: CronExpression - Impossible CRON expression ['.$scheduling.']');
$GLOBALS['log']->error('CronManager: ScheduledJob ['.$scheduledJob['id'].']: CronExpression - Impossible CRON expression ['.$scheduling.']');
continue;
}
if ($cronExpression->isDue()) {
$prevDate = date('Y-m-d H:i:00');
$prevDate = date('Y-m-d H:i:s');
}
$existsJob = $this->getJobService()->getJobByScheduledJob($scheduledJob['id'], $prevDate);
$existsJob = $cronJob->getJobByScheduledJob($scheduledJob['id'], $prevDate);
if (!isset($existsJob) || empty($existsJob)) {
//create a job
$data = array(
//create a new job
$jobEntity = $entityManager->getEntity('Job');
$jobEntity->set(array(
'name' => $scheduledJob['name'],
'status' => self::PENDING,
'scheduledJobId' => $scheduledJob['id'],
'executeTime' => $prevDate,
'method' => $scheduledJob['job'],
);
$createdJobs[] = $this->getJobService()->createEntity($data);
));
$jobEntityId = $entityManager->saveEntity($jobEntity);
if (!empty($jobEntityId)) {
$createdJobs[] = $jobEntityId;
}
}
}

View File

@@ -28,7 +28,6 @@ class ExtensionManager extends Upgrades\Base
protected $params = array(
'packagePath' => 'data/upload/extensions',
'backupPath' => 'data/.backup/extensions',
'scriptNames' => array(

View File

@@ -29,33 +29,33 @@ use \Espo\Core\Exceptions\NotFound;
class ClientManager
{
protected $entityManager;
protected $metadata;
protected $clientMap = array();
public function __construct($entityManager, $metadata, $config)
{
$this->entityManager = $entityManager;
$this->metadata = $metadata;
$this->config = $config;
}
protected function getMetadata()
{
return $this->metadata;
}
protected function getEntityManager()
{
return $this->entityManager;
}
protected function getConfig()
{
return $this->config;
}
public function storeAccessToken($hash, $data)
{
if (!empty($this->clientMap[$hash]) && !empty($this->clientMap[$hash]['externalAccountEntity'])) {
@@ -64,37 +64,37 @@ class ClientManager
$externalAccountEntity->set('tokenType', $data['tokenType']);
$this->getEntityManager()->saveEntity($externalAccountEntity);
}
}
}
public function create($integration, $userId)
{
$authMethod = $this->getMetadata()->get("integrations.{$integration}.authMethod");
$methodName = 'create' . ucfirst($authMethod);
return $this->$methodName($integration, $userId);
$authMethod = $this->getMetadata()->get("integrations.{$integration}.authMethod");
$methodName = 'create' . ucfirst($authMethod);
return $this->$methodName($integration, $userId);
}
protected function createOAuth2($integration, $userId)
{
{
$integrationEntity = $this->getEntityManager()->getEntity('Integration', $integration);
$externalAccountEntity = $this->getEntityManager()->getEntity('ExternalAccount', $integration . '__' . $userId);
$className = $this->getMetadata()->get("integrations.{$integration}.clientClassName");
$redirectUri = $this->getConfig()->get('siteUrl') . '/oauthcallback'; // TODO move to client class
$redirectUri = $this->getConfig()->get('siteUrl') . '?entryPoint=oauthCallback'; // TODO move to client class
if (!$externalAccountEntity) {
throw new Error("External Account {$integration} not found for {$userId}");
}
if (!$integrationEntity->get('enabled')) {
return null;
}
}
if (!$externalAccountEntity->get('enabled')) {
return null;
}
$oauth2Client = new \Espo\Core\ExternalAccount\OAuth2\Client();
}
$oauth2Client = new \Espo\Core\ExternalAccount\OAuth2\Client();
$client = new $className($oauth2Client, array(
'endpoint' => $this->getMetadata()->get("integrations.{$integration}.params.endpoint"),
'tokenEndpoint' => $this->getMetadata()->get("integrations.{$integration}.params.tokenEndpoint"),
@@ -105,12 +105,12 @@ class ClientManager
'refreshToken' => $externalAccountEntity->get('refreshToken'),
'tokenType' => $externalAccountEntity->get('tokenType'),
), $this);
$this->addToClientMap($client, $integrationEntity, $externalAccountEntity, $userId);
return $client;
return $client;
}
protected function addToClientMap($client, $integrationEntity, $externalAccountEntity, $userId)
{
$this->clientMap[spl_object_hash($client)] = array(

View File

@@ -29,9 +29,9 @@ use \Espo\Core\ExternalAccount\OAuth2\Client;
abstract class OAuth2Abstract implements IClient
{
protected $client = null;
protected $manager = null;
protected $paramList = array(
'endpoint',
'tokenEndpoint',
@@ -42,33 +42,33 @@ abstract class OAuth2Abstract implements IClient
'refreshToken',
'redirectUri',
);
protected $clientId = null;
protected $clientSecret = null;
protected $accessToken = null;
protected $refreshToken = null;
protected $redirectUri = null;
public function __construct($client, array $params = array(), $manager = null)
{
$this->client = $client;
$this->setParams($params);
$this->manager = $manager;
}
public function getParam($name)
{
if (in_array($name, $this->paramList)) {
return $this->$name;
}
}
public function setParam($name, $value)
{
if (in_array($name, $this->paramList)) {
@@ -79,7 +79,7 @@ abstract class OAuth2Abstract implements IClient
$this->$name = $value;
}
}
public function setParams(array $params)
{
foreach ($this->paramList as $name) {
@@ -88,42 +88,41 @@ abstract class OAuth2Abstract implements IClient
}
}
}
protected function afterTokenRefreshed($data)
{
if ($this->manager) {
$this->manager->storeAccessToken(spl_object_hash($this), $data);
}
}
public function getAccessTokenFromAuthorizationCode($code)
{
$r = $this->client->getAccessToken($this->getParam('tokenEndpoint'), Client::GRANT_TYPE_AUTHORIZATION_CODE, array(
'code' => $code,
'redirect_uri' => $this->getParam('redirectUri')
));
if ($r['code'] == 200) {
$data = array();
if (!empty($r['result'])) {
$data['accessToken'] = $r['result']['access_token'];
$data['tokenType'] = $r['result']['token_type'];
$data['refreshToken'] = $r['result']['refresh_token'];
$data['refreshToken'] = $r['result']['refresh_token'];
}
return $data;
}
return null;
}
abstract protected function getPingUrl();
public function ping()
{
if (empty($this->accessToken) || empty($this->clientId) || empty($this->clientSecret)) {
return false;
}
$url = $this->getPingUrl();
try {
@@ -133,35 +132,48 @@ abstract class OAuth2Abstract implements IClient
return false;
}
}
public function request($url, $params = array(), $httpMethod = Client::HTTP_METHOD_GET, $allowRenew = true)
public function request($url, $params = null, $httpMethod = Client::HTTP_METHOD_GET, $contentType = null, $allowRenew = true)
{
$r = $this->client->request($url, $params, $httpMethod);
$httpHeaders = array();
if (!empty($contentType)) {
$httpHeaders['Content-Type'] = $contentType;
switch ($contentType) {
case Client::CONTENT_TYPE_MULTIPART_FORM_DATA:
$httpHeaders['Content-Length'] = strlen($params);
break;
case Client::CONTENT_TYPE_APPLICATION_JSON:
$httpHeaders['Content-Length'] = strlen($params);
break;
}
}
$r = $this->client->request($url, $params, $httpMethod, $httpHeaders);
$code = null;
if (!empty($r['code'])) {
$code = $r['code'];
}
if ($code == 200) {
}
if ($code >= 200 && $code < 300) {
return $r['result'];
} else {
$handledData = $this->handleErrorResponse($r);
if ($allowRenew && is_array($handledData)) {
if ($handledData['action'] == 'refreshToken') {
if ($this->refreshToken()) {
return $this->request($url, $params, $httpMethod, false);
if ($this->refreshToken()) {
return $this->request($url, $params, $httpMethod, $contentType, false);
}
} else if ($handledData['action'] == 'renew') {
return $this->request($url, $params, $httpMethod, false);
return $this->request($url, $params, $httpMethod, $contentType, false);
}
}
}
throw new Error("Error after requesting {$httpMethod} {$url}.", $code);
}
protected function refreshToken()
{
if (!empty($this->refreshToken)) {
@@ -183,11 +195,11 @@ abstract class OAuth2Abstract implements IClient
}
}
}
protected function handleErrorResponse($r)
{
if ($r['code'] == 401 && !empty($r['result'])) {
$result = $r['result'];
$result = $r['result'];
if (strpos($r['header'], 'error=invalid_token') !== false) {
return array(
'action' => 'refreshToken'
@@ -198,7 +210,7 @@ abstract class OAuth2Abstract implements IClient
);
}
} else if ($r['code'] == 400 && !empty($r['result'])) {
if ($r['result']['error'] == 'invalid_token') {
if ($r['result']['error'] == 'invalid_token') {
return array(
'action' => 'refreshToken'
);

View File

@@ -32,8 +32,9 @@ class Client
const TOKEN_TYPE_BEARER = 'Bearer';
const TOKEN_TYPE_OAUTH = 'OAuth';
const CONTENT_TYPE_APPLICATION = 0;
const CONTENT_TYPE_MULTIPART = 1;
const CONTENT_TYPE_APPLICATION_X_WWW_FORM_URLENENCODED = 'application/x-www-form-urlencoded';
const CONTENT_TYPE_MULTIPART_FORM_DATA = 'multipart/form-data';
const CONTENT_TYPE_APPLICATION_JSON = 'application/json';
const HTTP_METHOD_GET = 'GET';
const HTTP_METHOD_POST = 'POST';
@@ -118,7 +119,7 @@ class Client
$this->accessTokenSecret = $accessTokenSecret;
}
public function request($url, $params = array(), $httpMethod = self::HTTP_METHOD_GET, array $httpHeaders = array(), $contentType = self::CONTENT_TYPE_MULTIPART)
public function request($url, $params = null, $httpMethod = self::HTTP_METHOD_GET, array $httpHeaders = array())
{
if ($this->accessToken) {
switch ($this->tokenType) {
@@ -137,10 +138,10 @@ class Client
}
}
return $this->execute($url, $params, $httpMethod, $httpHeaders, $contentType);
return $this->execute($url, $params, $httpMethod, $httpHeaders);
}
private function execute($url, $params = array(), $httpMethod, array $httpHeaders = array(), $contentType = self::CONTENT_TYPE_MULTIPART)
private function execute($url, $params = null, $httpMethod, array $httpHeaders = array())
{
$curlOptions = array(
CURLOPT_RETURNTRANSFER => true,
@@ -153,8 +154,10 @@ class Client
$curlOptions[CURLOPT_POST] = true;
case self::HTTP_METHOD_PUT:
case self::HTTP_METHOD_PATCH:
if (self::CONTENT_TYPE_APPLICATION === $contentType) {
if (is_array($params)) {
$postFields = http_build_query($params, null, '&');
} else {
$postFields = $params;
}
$curlOptions[CURLOPT_POSTFIELDS] = $postFields;
break;
@@ -165,7 +168,9 @@ class Client
if (strpos($url, '?') === false) {
$url .= '?';
}
$url .= http_build_query($params, null, '&');
if (is_array($params)) {
$url .= http_build_query($params, null, '&');
}
break;
default:
break;
@@ -228,7 +233,7 @@ class Client
$params['grant_type'] = $grantType;
$httpHeaders = array();
switch ($this->clientAuth) {
switch ($this->tokenType) {
case self::AUTH_TYPE_URI:
case self::AUTH_TYPE_FORM:
$params['client_id'] = $this->clientId;
@@ -242,7 +247,7 @@ class Client
throw new \Exception();
}
return $this->execute($url, $params, self::HTTP_METHOD_POST, $httpHeaders, self::CONTENT_TYPE_APPLICATION);
return $this->execute($url, $params, self::HTTP_METHOD_POST, $httpHeaders);
}
}

View File

@@ -73,7 +73,7 @@ class HookManager
protected function loadHooks()
{
if ($this->getConfig()->get('useCache') && file_exists($this->cacheFile)) {
$this->data = $this->getFileManager()->getContents($this->cacheFile);
$this->data = $this->getFileManager()->getPhpContents($this->cacheFile);
return;
}
@@ -89,14 +89,14 @@ class HookManager
$this->data = array_merge_recursive($this->data, $this->getHookData($this->paths['customPath']));
if ($this->getConfig()->get('useCache')) {
$this->getFileManager()->putContentsPHP($this->cacheFile, $this->data);
$this->getFileManager()->putPhpContents($this->cacheFile, $this->data);
}
}
public function process($scope, $hookName, $injection = null)
public function process($scope, $hookName, $injection = null, array $options = array())
{
if ($scope != 'Common') {
$this->process('Common', $hookName, $injection);
$this->process('Common', $hookName, $injection, $options);
}
if (!empty($this->data[$scope])) {
@@ -106,7 +106,7 @@ class HookManager
$this->hooks[$className] = $this->createHookByClassName($className);
}
$hook = $this->hooks[$className];
$hook->$hookName($injection);
$hook->$hookName($injection, $options);
}
}
}
@@ -170,7 +170,9 @@ class HookManager
$sortedHookList = array_merge($sortedHookList, $hookDetails);
}
$hooks[$scopeName][$hookName] = isset($hooks[$scopeName][$hookName]) ? array_merge($hooks[$scopeName][$hookName], $sortedHookList) : $sortedHookList;
$normalizedScopeName = Util::normilizeScopeName($scopeName);
$hooks[$normalizedScopeName][$hookName] = isset($hooks[$normalizedScopeName][$hookName]) ? array_merge($hooks[$normalizedScopeName][$hookName], $sortedHookList) : $sortedHookList;
}
}
}

View File

@@ -24,7 +24,7 @@ namespace Espo\Core\Hooks;
use \Espo\Core\Interfaces\Injectable;
class Base implements Injectable
abstract class Base implements Injectable
{
protected $dependencies = array(
'entityManager',
@@ -52,6 +52,11 @@ class Base implements Injectable
return $this->dependencies;
}
protected function addDependency($name)
{
$this->dependencies[] = $name;
}
protected function getInjection($name)
{
return $this->injections[$name];

View File

@@ -24,7 +24,6 @@ namespace Espo\Core\Jobs;
use \Espo\Core\Container;
abstract class Base
{
private $container;

View File

@@ -22,31 +22,17 @@
namespace Espo\Core\Loaders;
use Espo\Core\Utils,
Espo\Core\Utils\Log\Monolog\Handler;
class Log extends Base
class EntityManagerUtil extends Base
{
public function load()
{
$logConfig = $this->getContainer()->get('config')->get('logger');
$entityManager = new \Espo\Core\Utils\EntityManager(
$this->getContainer()->get('metadata'),
$this->getContainer()->get('language'),
$this->getContainer()->get('fileManager')
);
$log = new Utils\Log('Espo');
$levelCode = $log->getLevelCode($logConfig['level']);
if ($logConfig['isRotate']) {
$handler = new Handler\RotatingFileHandler($logConfig['path'], $logConfig['maxRotateFiles'], $levelCode);
} else {
$handler = new Handler\StreamHandler($logConfig['path'], $levelCode);
}
$log->pushHandler($handler);
$errorHandler = new \Monolog\ErrorHandler($log);
$errorHandler->registerExceptionHandler(null, false);
$errorHandler->registerErrorHandler(array(), false);
return $log;
return $entityManager;
}
}

View File

@@ -1,4 +1,4 @@
<?php
<?php
/************************************************************************
* This file is part of EspoCRM.
*
@@ -24,21 +24,23 @@ namespace Espo\Core\Mail;
use \Zend\Mime\Mime as Mime;
use \Espo\ORM\Entity;
class Importer
{
private $entityManager;
private $fileManager;
private $config;
public function __construct($entityManager, $fileManager, $config)
{
$this->entityManager = $entityManager;
$this->fileManager = $fileManager;
$this->config = $config;
}
protected function getEntityManager()
{
return $this->entityManager;
@@ -47,45 +49,58 @@ class Importer
{
return $this->config;
}
protected function getFileManager()
{
return $this->fileManager;
}
public function importMessage($message, $userId, $teamsIds = array())
{
try {
$email = $this->getEntityManager()->getEntity('Email');
$subject = $message->subject;
if ($subject !== '0' && empty($subject)) {
$subject = '--empty--';
$subject = '(No Subject)';
}
$email->set('isHtml', false);
$email->set('name', $subject);
$email->set('status', 'Archived');
$email->set('attachmentsIds', array());
$email->set('assignedUserId', $userId);
$email->set('teamsIds', $teamsIds);
$fromArr = $this->getAddressListFromMessage($message, 'from');
if (isset($message->from)) {
$email->set('fromName', $message->from);
$email->set('fromString', $message->from);
}
if (isset($message->replyTo)) {
$email->set('replyToString', $message->replyTo);
}
$toArr = $this->getAddressListFromMessage($message, 'to');
$ccArr = $this->getAddressListFromMessage($message, 'cc');
$email->set('from', $fromArr[0]);
$email->set('to', implode(';', $this->getAddressListFromMessage($message, 'to')));
$email->set('cc', implode(';', $this->getAddressListFromMessage($message, 'cc')));
$email->set('to', implode(';', $toArr));
$email->set('cc', implode(';', $ccArr));
if (isset($message->messageId) && !empty($message->messageId)) {
$email->set('messageId', $message->messageId);
if (isset($message->deliveredTo)) {
$email->set('messageIdInternal', $message->messageId . '-' . $message->deliveredTo);
}
}
if ($this->checkIsDuplicate($email)) {
if ($duplicate = $this->findDuplicate($email)) {
$this->getEntityManager()->getRepository('Email')->relate($duplicate, 'users', $userId);
if (!empty($teamsIds)) {
foreach ($teamsIds as $teamId) {
$this->getEntityManager()->getRepository('Email')->relate($duplicate, 'teams', $teamId);
}
}
return false;
}
@@ -103,17 +118,17 @@ class Importer
$email->set('deliveryDate', $deliveryDate);
}
}
$inlineIds = array();
if ($message->isMultipart()) {
foreach (new \RecursiveIteratorIterator($message) as $part) {
$this->importPartDataToEmail($email, $part, $inlineIds);
}
} else {
$this->importPartDataToEmail($email, $message, $inlineIds);
$this->importPartDataToEmail($email, $message, $inlineIds, 'text/plain');
}
$body = $email->get('body');
if (!empty($body)) {
foreach ($inlineIds as $cid => $attachmentId) {
@@ -122,6 +137,8 @@ class Importer
$email->set('body', $body);
}
$parentFound = false;
if (isset($message->references) && !empty($message->references)) {
$reference = str_replace(array('/', '@'), " ", trim($message->references, '<>'));
$parentType = $parentId = null;
@@ -131,54 +148,85 @@ class Importer
if (!empty($parentType) && !empty($parentId)) {
$email->set('parentType', $parentType);
$email->set('parentId', $parentId);
$parentFound = true;
}
}
}
if (!$email->has('parentId')) {
if (!$parentFound) {
$from = $email->get('from');
if ($from) {
$contact = $this->getEntityManager()->getRepository('Contact')->where(array(
'emailAddress' => $from
))->findOne();
if ($contact) {
if (!$this->getConfig()->get('b2cMode')) {
if ($contact->get('accountId')) {
$email->set('parentType', 'Account');
$email->set('parentId', $contact->get('accountId'));
}
} else {
$email->set('parentType', 'Contact');
$email->set('parentId', $contact->id);
}
$parentFound = $this->findParent($email, $from);
}
if (!$parentFound) {
if (!empty($toArr)) {
$parentFound = $this->findParent($email, $toArr[0]);
}
}
}
$this->getEntityManager()->saveEntity($email);
return $email;
} catch (\Exception $e) {}
}
protected function checkIsDuplicate($email)
protected function findParent(Entity $email, $emailAddress)
{
if ($email->get('messageIdInternal')) {
$contact = $this->getEntityManager()->getRepository('Contact')->where(array(
'emailAddress' => $emailAddress
))->findOne();
if ($contact) {
if (!$this->getConfig()->get('b2cMode')) {
if ($contact->get('accountId')) {
$email->set('parentType', 'Account');
$email->set('parentId', $contact->get('accountId'));
return true;
}
} else {
$email->set('parentType', 'Contact');
$email->set('parentId', $contact->id);
return true;
}
} else {
$account = $this->getEntityManager()->getRepository('Account')->where(array(
'emailAddress' => $emailAddress
))->findOne();
if ($account) {
$email->set('parentType', 'Account');
$email->set('parentId', $account->id);
return true;
} else {
$lead = $this->getEntityManager()->getRepository('Lead')->where(array(
'emailAddress' => $emailAddress
))->findOne();
if ($lead) {
$email->set('parentType', 'Lead');
$email->set('parentId', $lead->id);
return true;
}
}
}
}
protected function findDuplicate(Entity $email)
{
if ($email->get('messageId')) {
$duplicate = $this->getEntityManager()->getRepository('Email')->where(array(
'messageIdInternal' => $email->get('messageIdInternal')
'messageId' => $email->get('messageId')
))->findOne();
if ($duplicate) {
return true;
return $duplicate;
}
}
}
protected function getAddressListFromMessage($message, $type)
{
$addressList = array();
if (isset($message->$type)) {
$list = $message->getHeader($type)->getAddressList();
foreach ($list as $address) {
$addressList[] = $address->getEmail();
@@ -186,82 +234,107 @@ class Importer
}
return $addressList;
}
protected function importPartDataToEmail(\Espo\Entities\Email $email, $part, &$inlineIds = array())
protected function importPartDataToEmail(\Espo\Entities\Email $email, $part, &$inlineIds = array(), $defaultContentType = null)
{
try {
$type = strtok($part->contentType, ';');
$type = null;
if ($part->getHeaders() && isset($part->contentType)) {
$type = strtok($part->contentType, ';');
}
if (empty($type)) {
if (!empty($defaultContentType)) {
$type = $defaultContentType;
} else {
return;
}
}
$encoding = null;
switch ($type) {
case 'text/plain':
$content = $this->getContentFromPart($part);
if (!$email->get('body')) {
$email->set('body', $content);
}
$email->set('bodyPlain', $content);
break;
case 'text/html':
$content = $this->getContentFromPart($part);
$email->set('body', $content);
$email->set('isHtml', true);
break;
default:
$content = $part->getContent();
$disposition = null;
$fileName = null;
$contentId = null;
if (isset($part->ContentDisposition)) {
if (strpos($part->ContentDisposition, 'attachment') === 0) {
if (preg_match('/filename="?([^"]+)"?/i', $part->ContentDisposition, $m)) {
$fileName = $m[1];
$disposition = 'attachment';
}
} else if (strpos($part->ContentDisposition, 'inline') === 0) {
$contentId = trim($part->contentID, '<>');
$fileName = $contentId;
$disposition = 'inline';
$isAttachment = true;
if ($type == 'text/plain' || $type == 'text/html') {
$isAttachment = false;
$content = $this->getContentFromPart($part);
if ($type == 'text/plain') {
if ($email->get('bodyPlain')) {
$isAttachment = true;
} else {
$email->set('bodyPlain', $content);
if (!$email->get('body')) {
$email->set('body', $content);
}
}
if (isset($part->contentTransferEncoding)) {
$encoding = strtolower($part->getHeader('Content-Transfer-Encoding')->getTransferEncoding());
}
$attachment = $this->getEntityManager()->getEntity('Attachment');
$attachment->set('name', $fileName);
$attachment->set('type', $type);
if ($disposition == 'inline') {
$attachment->set('role', 'Inline Attachment');
} else if ($type == 'text/html') {
if ($email->get('isHtml')) {
$isAttachment = true;
} else {
$attachment->set('role', 'Attachment');
$email->set('body', $content);
$email->set('isHtml', true);
}
if ($encoding == 'base64') {
$content = base64_decode($content);
}
$attachment->set('size', strlen($content));
$this->getEntityManager()->saveEntity($attachment);
$path = 'data/upload/' . $attachment->id;
$this->getFileManager()->putContents($path, $content);
if ($disposition == 'attachment') {
$attachmentsIds = $email->get('attachmentsIds');
$attachmentsIds[] = $attachment->id;
$email->set('attachmentsIds', $attachmentsIds);
} else if ($disposition == 'inline') {
$inlineIds[$contentId] = $attachment->id;
}
}
if ($isAttachment) {
$content = $part->getContent();
$disposition = null;
$fileName = null;
$contentId = null;
if (isset($part->ContentDisposition)) {
if (strpos($part->ContentDisposition, 'attachment') === 0) {
if (preg_match('/filename="?([^"]+)"?/i', $part->ContentDisposition, $m)) {
$fileName = $m[1];
$disposition = 'attachment';
}
} else if (strpos($part->ContentDisposition, 'inline') === 0) {
$contentId = trim($part->contentID, '<>');
$fileName = $contentId;
$disposition = 'inline';
}
}
if (isset($part->contentTransferEncoding)) {
$encoding = strtolower($part->getHeader('Content-Transfer-Encoding')->getTransferEncoding());
}
$attachment = $this->getEntityManager()->getEntity('Attachment');
$attachment->set('name', $fileName);
$attachment->set('type', $type);
if ($disposition == 'inline') {
$attachment->set('role', 'Inline Attachment');
} else {
$attachment->set('role', 'Attachment');
}
if ($encoding == 'base64') {
$content = base64_decode($content);
}
$attachment->set('size', strlen($content));
$this->getEntityManager()->saveEntity($attachment);
$path = 'data/upload/' . $attachment->id;
$this->getFileManager()->putContents($path, $content);
if ($disposition == 'attachment') {
$attachmentsIds = $email->get('attachmentsIds');
$attachmentsIds[] = $attachment->id;
$email->set('attachmentsIds', $attachmentsIds);
} else if ($disposition == 'inline') {
$inlineIds[$contentId] = $attachment->id;
}
}
} catch (\Exception $e) {}
}
protected function getContentFromPart($part)
{
if ($part instanceof \Zend\Mime\Part) {
@@ -271,20 +344,20 @@ class Importer
}
} else {
$content = $part->getContent();
$encoding = null;
if (isset($part->contentTransferEncoding)) {
$cteHeader = $part->getHeader('Content-Transfer-Encoding');
$encoding = strtolower($cteHeader->getTransferEncoding());
}
if ($encoding == 'base64') {
$content = base64_decode($content);
}
$charset = 'UTF-8';
if (isset($part->contentType)) {
$ctHeader = $part->getHeader('Content-Type');
$charsetParamValue = $ctHeader->getParameter('charset');
@@ -292,17 +365,17 @@ class Importer
$charset = strtoupper($charsetParamValue);
}
}
if ($charset !== 'UTF-8') {
$content = mb_convert_encoding($content, 'UTF-8', $charset);
}
if (isset($part->contentTransferEncoding)) {
$cteHeader = $part->getHeader('Content-Transfer-Encoding');
if ($cteHeader->getTransferEncoding() == 'quoted-printable') {
$content = quoted_printable_decode($content);
}
}
if ($charset !== 'UTF-8') {
$content = mb_convert_encoding($content, 'UTF-8', $charset);
}
}
return $content;
}

View File

@@ -0,0 +1,46 @@
<?php
namespace Espo\Core\Mail\Mail;
use ArrayIterator;
use Countable;
use Iterator;
use Traversable;
use Zend\Loader\PluginClassLocator;
class Headers extends \Zend\Mail\Headers
{
public static function fromString($string, $EOL = self::EOL)
{
$headers = new static();
$currentLine = '';
// iterate the header lines, some might be continuations
foreach (explode($EOL, $string) as $line) {
// check if a header name is present
if (preg_match('/^(?P<name>[\x21-\x39\x3B-\x7E]+):.*$/', $line, $matches)) {
if ($currentLine) {
// a header name was present, then store the current complete line
$headers->addHeaderLine($currentLine);
}
$currentLine = trim($line);
} elseif (preg_match('/^\s+.*$/', $line, $matches)) {
// continuation: append to current line
$currentLine .= $line;
} elseif (preg_match('/^\s*$/', $line)) {
// empty line indicates end of headers
break;
} else {
// Line does not match header format!
throw new Exception\RuntimeException(sprintf(
'Line "%s"does not match header format!',
$line
));
}
}
if ($currentLine) {
$headers->addHeaderLine($currentLine);
}
return $headers;
}
}

View File

@@ -1,4 +1,4 @@
<?php
<?php
/************************************************************************
* This file is part of EspoCRM.
*
@@ -20,20 +20,22 @@
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Mail\Storage;
namespace Espo\Core\Mail\Mail\Storage;
class Imap extends \Zend\Mail\Storage\Imap
{
{
protected $messageClass = '\\Espo\\Core\\Mail\\Mail\\Storage\\Message';
public function getIdsFromUID($uid)
{
$uid = intval($uid) + 1;
return $this->protocol->search(array('UID ' . $uid . ':*'));
}
public function getIdsFromDate($date)
{
{
return $this->protocol->search(array('SINCE "' . $date . '"'));
}
}

View File

@@ -0,0 +1,155 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Mail\Mail\Storage;
use Espo\Core\Mail\Mail\Headers;
use Zend\Mail\Header\HeaderInterface;
use Zend\Mime;
use Zend\Mail\Storage\Exception;
use Zend\Mail\Storage\AbstractStorage;
class Message extends \Zend\Mail\Storage\Message
{
public function __construct(array $params)
{
if (isset($params['file'])) {
if (!is_resource($params['file'])) {
ErrorHandler::start();
$params['raw'] = file_get_contents($params['file']);
$error = ErrorHandler::stop();
if ($params['raw'] === false) {
throw new Exception\RuntimeException('could not open file', 0, $error);
}
} else {
$params['raw'] = stream_get_contents($params['file']);
}
}
if (!empty($params['flags'])) {
// set key and value to the same value for easy lookup
$this->flags = array_combine($params['flags'], $params['flags']);
}
if (isset($params['handler'])) {
if (!$params['handler'] instanceof AbstractStorage) {
throw new Exception\InvalidArgumentException('handler is not a valid mail handler');
}
if (!isset($params['id'])) {
throw new Exception\InvalidArgumentException('need a message id with a handler');
}
$this->mail = $params['handler'];
$this->messageNum = $params['id'];
}
$params['strict'] = isset($params['strict']) ? $params['strict'] : false;
if (isset($params['raw'])) {
self::splitMessage($params['raw'], $this->headers, $this->content, Mime\Mime::LINEEND, $params['strict']);
} elseif (isset($params['headers'])) {
if (is_array($params['headers'])) {
$this->headers = new Headers();
$this->headers->addHeaders($params['headers']);
} else {
if ($params['headers'] instanceof \Zend\Mail\Headers) {
$this->headers = $params['headers'];
} else {
if (empty($params['noToplines'])) {
self::splitMessage($params['headers'], $this->headers, $this->topLines);
} else {
$this->headers = Headers::fromString($params['headers']);
}
}
}
if (isset($params['content'])) {
$this->content = $params['content'];
}
}
}
public function __isset($name)
{
$headers = $this->getHeaders();
if (empty($headers) || !is_object($headers)) {
return false;
}
return $this->getHeaders()->has($name);
}
public function isMultipart()
{
if (!isset($this->contentType)) {
return false;
}
try {
return stripos($this->contentType, 'multipart/') === 0;
} catch (Exception\ExceptionInterface $e) {
return false;
}
}
public static function splitMessage($message, &$headers, &$body, $EOL = Mime\Mime::LINEEND, $strict = false)
{
if ($message instanceof Headers) {
$message = $message->toString();
}
// check for valid header at first line
$firstline = strtok($message, "\n");
if (!preg_match('%^[^\s]+[^:]*:%', $firstline)) {
$headers = array();
// TODO: we're ignoring \r for now - is this function fast enough and is it safe to assume noone needs \r?
$body = str_replace(array("\r", "\n"), array('', $EOL), $message);
return;
}
// see @ZF2-372, pops the first line off a message if it doesn't contain a header
if (!$strict) {
$parts = explode(':', $firstline, 2);
if (count($parts) != 2) {
$message = substr($message, strpos($message, $EOL)+1);
}
}
// find an empty line between headers and body
// default is set new line
if (strpos($message, $EOL . $EOL)) {
list($headers, $body) = explode($EOL . $EOL, $message, 2);
// next is the standard new line
} elseif ($EOL != "\r\n" && strpos($message, "\r\n\r\n")) {
list($headers, $body) = explode("\r\n\r\n", $message, 2);
// next is the other "standard" new line
} elseif ($EOL != "\n" && strpos($message, "\n\n")) {
list($headers, $body) = explode("\n\n", $message, 2);
// at last resort find anything that looks like a new line
} else {
ErrorHandler::start(E_NOTICE|E_WARNING);
list($headers, $body) = preg_split("%([\r\n]+)\\1%U", $message, 2);
ErrorHandler::stop();
}
$headers = Headers::fromString($headers, $EOL);
}
}

View File

@@ -68,7 +68,7 @@ class Sender
$this->params = $params;
$this->transport = new SmtpTransport();
$opts = array(
'name' => 'admin',
'host' => $params['server'],
@@ -83,14 +83,14 @@ class Sender
if ($params['security']) {
$opts['connection_config']['ssl'] = strtolower($params['security']);
}
if (in_array('fromName', $params)) {
$this->params['fromName'] = $params['fromName'];
}
if (in_array('fromAddress', $params)) {
$this->params['fromAddress'] = $params['fromAddress'];
}
$options = new SmtpOptions($opts);
$this->transport->setOptions($options);
@@ -160,10 +160,10 @@ class Sender
} else {
$fromName = $config->get('outboundEmailFromName');
}
$message->addFrom($fromAddress, $fromName);
}
if (!empty($params['replyToAddress'])) {
$replyToName = null;
if (!empty($params['replyToName'])) {
@@ -203,7 +203,6 @@ class Sender
}
$attachmentPartList = array();
$attachmentCollection = $email->get('attachments');
$attachmentInlineCollection = $email->getInlineAttachments();
@@ -220,7 +219,7 @@ class Sender
$attachmentPartList[] = $attachment;
}
}
if (!empty($attachmentInlineCollection)) {
foreach ($attachmentInlineCollection as $a) {
$fileName = 'data/upload/' . $a->id;
@@ -239,12 +238,12 @@ class Sender
$message->setSubject($email->get('name'));
$body = new MimeMessage();
$textPart = new MimePart($email->getBodyPlainForSending());
$textPart->type = 'text/plain';
$textPart->encoding = Mime::ENCODING_QUOTEDPRINTABLE;
$textPart->charset = 'utf-8';
if ($email->get('isHtml')) {
$htmlPart = new MimePart($email->getBodyForSending());
$htmlPart->encoding = Mime::ENCODING_QUOTEDPRINTABLE;
@@ -285,9 +284,12 @@ class Sender
$message->setBody($body);
if ($message->getHeaders()->has('content-type')) {
$message->getHeaders()->get('content-type')->setType($messageType);
if (!$message->getHeaders()->has('content-type')) {
$contentTypeHeader = new \Zend\Mail\Header\ContentType();
$message->getHeaders()->addHeader($contentTypeHeader);
}
$message->getHeaders()->get('content-type')->setType($messageType);
$message->setEncoding('UTF-8');
try {

View File

@@ -0,0 +1,103 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Notificators;
use \Espo\Core\Interfaces\Injectable;
use \Espo\ORM\Entity;
class Base implements Injectable
{
protected $dependencies = array(
'user',
'entityManager',
);
protected $injections = array();
public static $order = 9;
public function __construct()
{
$this->init();
}
protected function init()
{
}
protected function addDependency($name)
{
$this->dependencies[] = $name;
}
public function getDependencyList()
{
return $this->dependencies;
}
protected function getInjection($name)
{
return $this->injections[$name];
}
public function inject($name, $object)
{
$this->injections[$name] = $object;
}
protected function getEntityManager()
{
return $this->injections['entityManager'];
}
protected function getUser()
{
return $this->injections['user'];
}
public function process(Entity $entity)
{
if ($entity->has('assignedUserId') && $entity->get('assignedUserId')) {
$assignedUserId = $entity->get('assignedUserId');
if ($assignedUserId != $this->getUser()->id && $entity->isFieldChanged('assignedUserId')) {
$notification = $this->getEntityManager()->getEntity('Notification');
$notification->set(array(
'type' => 'Assign',
'userId' => $assignedUserId,
'data' => array(
'entityType' => $entity->getEntityType(),
'entityId' => $entity->id,
'entityName' => $entity->get('name'),
'isNew' => $entity->isNew(),
'userId' => $this->getUser()->id,
'userName' => $this->getUser()->get('name')
)
));
$this->getEntityManager()->saveEntity($notification);
}
}
}
}

View File

@@ -41,17 +41,20 @@ class Entity extends \Espo\ORM\Entity
$columnsData = new \stdClass();
}
foreach ($collection as $e) {
$id = $e->id;
$ids[] = $id;
$names->$id = $e->get('name');
if (!empty($columns)) {
$columnsData->$id = new \stdClass();
foreach ($columns as $column => $f) {
$columnsData->$id->$column = $e->get($f);
if ($collection) {
foreach ($collection as $e) {
$id = $e->id;
$ids[] = $id;
$names->$id = $e->get('name');
if (!empty($columns)) {
$columnsData->$id = new \stdClass();
foreach ($columns as $column => $f) {
$columnsData->$id->$column = $e->get($f);
}
}
}
}
$this->set($field . 'Ids', $ids);
$this->set($field . 'Names', $names);
if (!empty($columns)) {

View File

@@ -38,6 +38,8 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
protected $injections = array();
private $restoreData = null;
public function inject($name, $object)
{
$this->injections[$name] = $object;
@@ -79,6 +81,9 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
foreach ($defs['fields'] as $field => $d) {
if (isset($d['type']) && $d['type'] == 'currency') {
if (!empty($d['notStorable'])) {
continue;
}
if (empty($params['customJoin'])) {
$params['customJoin'] = '';
}
@@ -135,49 +140,67 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
}
}
protected function beforeRemove(Entity $entity)
protected function beforeRemove(Entity $entity, array $options = array())
{
parent::beforeRemove($entity);
$this->getEntityManager()->getHookManager()->process($this->entityName, 'beforeRemove', $entity);
parent::beforeRemove($entity, $options);
$this->getEntityManager()->getHookManager()->process($this->entityName, 'beforeRemove', $entity, $options);
$nowString = date('Y-m-d H:i:s', time());
if ($entity->hasField('modifiedAt')) {
$entity->set('modifiedAt', $nowString);
}
if ($entity->hasField('modifiedById')) {
$entity->set('modifiedById', $this->getEntityManager()->getUser()->id);
}
}
protected function afterRemove(Entity $entity)
protected function afterRemove(Entity $entity, array $options = array())
{
parent::afterRemove($entity);
$this->getEntityManager()->getHookManager()->process($this->entityName, 'afterRemove', $entity);
parent::afterRemove($entity, $options);
$this->getEntityManager()->getHookManager()->process($this->entityName, 'afterRemove', $entity, $options);
}
public function remove(Entity $entity)
public function remove(Entity $entity, array $options = array())
{
$this->getEntityManager()->getHookManager()->process($this->entityName, 'beforeRemove', $entity);
$this->getEntityManager()->getHookManager()->process($this->entityName, 'beforeRemove', $entity, $options);
$result = parent::remove($entity);
$result = parent::remove($entity, $options);
if ($result) {
$this->getEntityManager()->getHookManager()->process($this->entityName, 'afterRemove', $entity);
$this->getEntityManager()->getHookManager()->process($this->entityName, 'afterRemove', $entity, $options);
}
return $result;
}
protected function beforeSave(Entity $entity)
protected function beforeSave(Entity $entity, array $options = array())
{
parent::beforeSave($entity);
$this->getEntityManager()->getHookManager()->process($this->entityName, 'beforeSave', $entity);
parent::beforeSave($entity, $options);
$this->getEntityManager()->getHookManager()->process($this->entityName, 'beforeSave', $entity, $options);
}
protected function afterSave(Entity $entity)
protected function afterSave(Entity $entity, array $options = array())
{
parent::afterSave($entity);
$this->getEntityManager()->getHookManager()->process($this->entityName, 'afterSave', $entity);
if (!empty($this->restoreData)) {
$entity->set($this->restoreData);
$this->restoreData = null;
}
parent::afterSave($entity, $options);
$this->handleEmailAddressSave($entity);
$this->handlePhoneNumberSave($entity);
$this->handleSpecifiedRelations($entity);
$this->getEntityManager()->getHookManager()->process($this->entityName, 'afterSave', $entity, $options);
}
public function save(Entity $entity)
public function save(Entity $entity, array $options = array())
{
$nowString = date('Y-m-d H:i:s', time());
$restoreData = array();
if ($entity->isNew()) {
if (!$entity->has('id')) {
$entity->set('id', uniqid());
$entity->set('id', Util::generateId());
}
if ($entity->hasField('createdAt')) {
@@ -198,11 +221,13 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
}
$entity->clear('modifiedById');
} else {
if ($entity->hasField('modifiedAt')) {
$entity->set('modifiedAt', $nowString);
}
if ($entity->hasField('modifiedById')) {
$entity->set('modifiedById', $this->entityManager->getUser()->id);
if (empty($options['silent'])) {
if ($entity->hasField('modifiedAt')) {
$entity->set('modifiedAt', $nowString);
}
if ($entity->hasField('modifiedById')) {
$entity->set('modifiedById', $this->entityManager->getUser()->id);
}
}
if ($entity->has('createdById')) {
@@ -214,13 +239,9 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
$entity->clear('createdById');
$entity->clear('createdAt');
}
$result = parent::save($entity);
$this->restoreData = $restoreData;
$entity->set($restoreData);
$this->handleEmailAddressSave($entity);
$this->handlePhoneNumberSave($entity);
$this->handleSpecifiedRelations($entity);
$result = parent::save($entity, $options);
return $result;
}
@@ -249,6 +270,10 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
if ($entity->has($fieldName) || $entity->has($columnsFieldsName)) {
if ($this->getMetadata()->get("entityDefs." . $entity->getEntityType() . ".fields.{$name}.noSave")) {
continue;
}
if ($entity->has($fieldName)) {
$specifiedIds = $entity->get($fieldName);
} else {
@@ -264,24 +289,36 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
$existingColumnsData = new \stdClass();
$defs = array();
$columns = $this->getMetadata()->get("entityDefs." . $entity->getEntityName() . ".fields.{$name}.columns");
$columns = $this->getMetadata()->get("entityDefs." . $entity->getEntityType() . ".fields.{$name}.columns");
if (!empty($columns)) {
$columnData = $entity->get($columnsFieldsName);
$defs['additionalColumns'] = $columns;
}
foreach ($entity->get($name, $defs) as $foreignEntity) {
$existingIds[] = $foreignEntity->id;
if (!empty($columns)) {
$data = new \stdClass();
foreach ($columns as $columnName => $columnField) {
$foreignId = $foreignEntity->id;
$data->$columnName = $foreignEntity->get($columnField);
$foreignCollection = $entity->get($name, $defs);
if ($foreignCollection) {
foreach ($foreignCollection as $foreignEntity) {
$existingIds[] = $foreignEntity->id;
if (!empty($columns)) {
$data = new \stdClass();
foreach ($columns as $columnName => $columnField) {
$foreignId = $foreignEntity->id;
$data->$columnName = $foreignEntity->get($columnField);
}
$existingColumnsData->$foreignId = $data;
$entity->setFetched($columnsFieldsName, $existingColumnsData);
}
$existingColumnsData->$foreignId = $data;
}
}
}
if ($entity->has($fieldName)) {
$entity->setFetched($fieldName, $existingIds);
}
if ($entity->has($columnsFieldsName) && !empty($columns)) {
$entity->setFetched($columnsFieldsName, $existingColumnsData);
}
foreach ($existingIds as $id) {
if (!in_array($id, $specifiedIds)) {
$toRemoveIds[] = $id;
@@ -295,10 +332,12 @@ class RDB extends \Espo\ORM\Repositories\RDB implements Injectable
}
}
}
foreach ($specifiedIds as $id) {
if (!in_array($id, $existingIds)) {
$data = null;
if (!empty($columns)) {
if (!empty($columns) && isset($columnData->$id)) {
$data = $columnData->$id;
}
$this->relate($entity, $name, $id, $data);

View File

@@ -18,7 +18,7 @@
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
************************************************************************/
namespace Espo\Core;
@@ -29,11 +29,11 @@ use \Espo\Core\Utils\Util;
class SelectManagerFactory
{
private $entityManager;
private $user;
private $acl;
private $metadata;
public function __construct($entityManager, \Espo\Entities\User $user, Acl $acl, $metadata)
@@ -43,26 +43,28 @@ class SelectManagerFactory
$this->acl = $acl;
$this->metadata = $metadata;
}
public function create($entityName)
{
$className = '\\Espo\\Custom\\SelectManagers\\' . Util::normilizeClassName($entityName);
$normalizedName = Util::normilizeClassName($entityName);
$className = '\\Espo\\Custom\\SelectManagers\\' . $normalizedName;
if (!class_exists($className)) {
$moduleName = $this->metadata->getScopeModuleName($entityName);
if ($moduleName) {
$className = '\\Espo\\Modules\\' . $moduleName . '\\SelectManagers\\' . Util::normilizeClassName($entityName);
$className = '\\Espo\\Modules\\' . $moduleName . '\\SelectManagers\\' . $normalizedName;
} else {
$className = '\\Espo\\SelectManagers\\' . Util::normilizeClassName($entityName);
}
$className = '\\Espo\\SelectManagers\\' . $normalizedName;
}
if (!class_exists($className)) {
$className = '\\Espo\\Core\\SelectManagers\\Base';
}
}
$selectManager = new $className($this->entityManager, $this->user, $this->acl, $this->metadata);
$selectManager->setEntityName($entityName);
return $selectManager;
}
}
}

View File

@@ -42,6 +42,8 @@ class Base
private $seed = null;
private $userTimeZone = null;
const MIN_LENGTH_FOR_CONTENT_SEARCH = 4;
public function __construct($entityManager, \Espo\Entities\User $user, Acl $acl, $metadata)
@@ -49,9 +51,20 @@ class Base
$this->entityManager = $entityManager;
$this->user = $user;
$this->acl = $acl;
$this->metadata = $metadata;
}
protected function getEntityManager()
{
return $this->entityManager;
}
protected function getUser()
{
return $this->user;
}
public function setEntityName($entityName)
{
$this->entityName = $entityName;
@@ -100,46 +113,54 @@ class Base
return $this->seed;
}
protected function textFilter($value, &$result)
{
$fieldDefs = $this->getSeed()->getFields();
$fieldList = $this->getTextFilterFields();
$d = array();
foreach ($fieldList as $field) {
if (
strlen($item['value']) >= self::MIN_LENGTH_FOR_CONTENT_SEARCH
&&
!empty($fieldDefs[$field]['type']) && $fieldDefs[$field]['type'] == 'text'
) {
$d[$field . '*'] = '%' . $value . '%';
} else {
$d[$field . '*'] = $value . '%';
}
}
$result['whereClause'][] = array(
'OR' => $d
);
}
protected function where($params, &$result)
{
if (!empty($params['where']) && is_array($params['where'])) {
$where = array();
foreach ($params['where'] as $item) {
if ($item['type'] == 'boolFilters' && !empty($item['value']) && is_array($item['value'])) {
if ($item['type'] == 'bool' && !empty($item['value']) && is_array($item['value'])) {
foreach ($item['value'] as $filter) {
$p = $this->getBoolFilterWhere($filter);
if (!empty($p)) {
$params['where'][] = $p;
}
$this->boolFilter($filter, $result);
}
} else if ($item['type'] == 'textFilter' && !empty($item['value'])) {
if (!empty($item['value'])) {
if (empty($result['whereClause'])) {
$result['whereClause'] = array();
}
$fieldDefs = $this->getSeed()->getFields();
$fieldList = $this->getTextFilterFields();
$d = array();
foreach ($fieldList as $field) {
if (
strlen($item['value']) >= self::MIN_LENGTH_FOR_CONTENT_SEARCH
&&
!empty($fieldDefs[$field]['type']) && $fieldDefs[$field]['type'] == 'text'
) {
$d[$field . '*'] = '%' . $item['value'] . '%';
} else {
$d[$field . '*'] = $item['value'] . '%';
}
}
$where['OR'] = $d;
$this->textFilter($item['value'], $result);
}
} else if ($item['type'] == 'primary' && !empty($item['value'])) {
$this->primaryFilter($item['value'], $result);
}
}
$linkedWith = array();
$ignoreList = array('linkedWith', 'boolFilters');
foreach ($params['where'] as $item) {
$ignoreList = array('linkedWith', 'bool', 'primary');
foreach ($params['where'] as $item) {
if (!in_array($item['type'], $ignoreList)) {
$part = $this->getWherePart($item);
if (!empty($part)) {
@@ -160,7 +181,6 @@ class Base
if (is_array($idsValue) && count($idsValue) == 1) {
$idsValue = $idsValue[0];
}
$relDefs = $this->getSeed()->getRelations();
@@ -170,7 +190,7 @@ class Base
$joins[] = $link;
if (!empty($defs['relationName']) && !empty($defs['midKeys'])) {
$key = $defs['midKeys'][1];
$relationName = $defs['relationName'];
$relationName = lcfirst($defs['relationName']);
$part[$relationName . '.' . $key] = $idsValue;
}
} else if ($defs['type'] == 'belongsTo') {
@@ -180,8 +200,6 @@ class Base
}
}
}
}
if (!empty($part)) {
@@ -192,17 +210,13 @@ class Base
}
$result['whereClause'] = $where;
$result['whereClause'] = array_merge($result['whereClause'], $where);
}
}
protected function q($params, &$result)
{
if (!empty($params['q'])) {
if (empty($result['whereClause'])) {
$result['whereClause'] = array();
}
$fieldDefs = $this->getSeed()->getFields();
$value = $params['q'];
@@ -221,38 +235,49 @@ class Base
}
}
$result['whereClause']['OR'] = $d;
$result['whereClause'][] = array(
'OR' => $d
);
}
}
protected function access(&$result)
{
if ($this->acl->checkReadOnlyOwn($this->entityName)) {
if (!array_key_exists('whereClause', $result)) {
$result['whereClause'] = array();
}
$result['whereClause']['assignedUserId'] = $this->user->id;
$this->accessOnlyOwn($result);
}
if (!$this->user->isAdmin() && $this->acl->checkReadOnlyTeam($this->entityName)) {
if (!array_key_exists('whereClause', $result)) {
$result['whereClause'] = array();
}
$result['distinct'] = true;
if (!array_key_exists('joins', $result)) {
$result['joins'] = array();
}
if (!in_array('teams', $result['joins'])) {
$result['leftJoins'][] = 'teams';
}
$result['whereClause']['OR'] = array(
'Team.id' => $this->user->get('teamsIds'),
'assignedUserId' => $this->user->id
);
$this->accessOnlyTeam($result);
}
}
protected function accessOnlyOwn(&$result)
{
if (!$this->getSeed()->hasField('assignedUserId')) {
return;
}
$result['whereClause'][] = array(
'assignedUserId' => $this->getUser()->id
);
}
protected function accessOnlyTeam(&$result)
{
if (!$this->getSeed()->hasField('teamsIds')) {
return;
}
$result['distinct'] = true;
if (!in_array('teams', $result['joins'])) {
$result['leftJoins'][] = 'teams';
}
$result['whereClause'][] = array(
'OR' => array(
'teams.id' => $this->user->get('teamsIds'),
'assignedUserId' => $this->getUser()->id
)
);
}
public function getAclParams()
{
$result = array();
@@ -262,7 +287,11 @@ class Base
public function getSelectParams(array $params, $withAcl = false)
{
$result = array();
$result = array(
'joins' => array(),
'leftJoins' => array(),
'whereClause' => array()
);
$this->order($params, $result);
$this->limit($params, $result);
@@ -276,10 +305,121 @@ class Base
return $result;
}
protected function getUserTimeZone()
{
if (empty($this->userTimeZone)) {
$preferences = $this->getEntityManager()->getEntity('Preferences', $this->getUser()->id);
$timeZone = $preferences->get('timeZone');
$this->userTimeZone = $timeZone;
}
return $this->userTimeZone;
}
protected function convertDateTimeWhere($item)
{
$format = 'Y-m-d H:i:s';
$value = null;
$timeZone = 'UTC';
if (empty($item['field'])) {
return null;
}
if (empty($item['type'])) {
return null;
}
if (!empty($item['value'])) {
$value = $item['value'];
}
if (!empty($item['timeZone'])) {
$timeZone = $item['timeZone'];
}
$type = $item['type'];
$field = $item['field'];
if (empty($value) && in_array($type, array('on', 'before', 'after'))) {
return null;
}
$where = array();
$where['field'] = $field;
$dt = new \DateTime('now', new \DateTimeZone($timeZone));
switch ($type) {
case 'today':
$where['type'] = 'between';
$dt->setTime(0, 0, 0);
$dt->setTimezone(new \DateTimeZone('UTC'));
$from = $dt->format($format);
$dt->modify('+1 day');
$to = $dt->format($format);
$where['value'] = [$from, $to];
break;
case 'past':
$where['type'] = 'before';
$dt->setTimezone(new \DateTimeZone('UTC'));
$where['value'] = $dt->format($format);
break;
case 'future':
$where['type'] = 'after';
$dt->setTimezone(new \DateTimeZone('UTC'));
$where['value'] = $dt->format($format);
break;
case 'on':
$where['type'] = 'between';
$dt = new \DateTime($value, new \DateTimeZone($timeZone));
$dt->setTimezone(new \DateTimeZone('UTC'));
$from = $dt->format($format);
$dt->modify('+1 day');
$to = $dt->format($format);
$where['value'] = [$from, $to];
break;
case 'before':
$where['type'] = 'before';
$dt = new \DateTime($value, new \DateTimeZone($timeZone));
$dt->setTimezone(new \DateTimeZone('UTC'));
$where['value'] = $dt->format($format);
break;
case 'after':
$where['type'] = 'after';
$dt = new \DateTime($value, new \DateTimeZone($timeZone));
$dt->setTimezone(new \DateTimeZone('UTC'));
$where['value'] = $dt->format($format);
break;
case 'between':
$where['type'] = 'between';
if (is_array($value)) {
$dt = new \DateTime($value[0], new \DateTimeZone($timeZone));
$dt->setTimezone(new \DateTimeZone('UTC'));
$from = $dt->format($format);
$dt = new \DateTime($value[1], new \DateTimeZone($timeZone));
$dt->setTimezone(new \DateTimeZone('UTC'));
$to = $dt->format($format);
$where['value'] = [$from, $to];
}
break;
default:
$where['type'] = $type;
}
$result = $this->getWherePart($where);
return $result;
}
protected function getWherePart($item)
{
$part = array();
if (!empty($item['dateTime'])) {
return $this->convertDateTimeWhere($item);
}
if (!empty($item['type'])) {
switch ($item['type']) {
case 'or':
@@ -341,7 +481,7 @@ class Base
$part[$item['field'] . '<'] = date('Y-m-d');
break;
case 'future':
$part[$item['field'] . '>'] = date('Y-m-d');
$part[$item['field'] . '>='] = date('Y-m-d');
break;
case 'currentMonth':
$dt = new \DateTime();
@@ -408,6 +548,22 @@ class Base
return $part;
}
protected function boolFilter($filterName, &$result)
{
$method = 'boolFilter' . ucfirst($filterName);
if (method_exists($this, $method)) {
$this->$method($result);
}
}
protected function primaryFilter($filterName, &$result)
{
$method = 'filter' . ucfirst($filterName);
if (method_exists($this, $method)) {
$this->$method($result);
}
}
protected function getBoolFilterWhere($filterName)
{
$method = 'getBoolFilterWhere' . ucfirst($filterName);
@@ -416,12 +572,10 @@ class Base
}
}
protected function getBoolFilterWhereOnlyMy()
protected function boolFilterOnlyMy(&$result)
{
return array(
'type' => 'equals',
'field' => 'assignedUserId',
'value' => $this->user->id,
$result['whereClause'][] = array(
'assignedUserId' => $this->getUser()->id
);
}
}

View File

@@ -48,31 +48,6 @@ class ServiceFactory
$this->container = $container;
}
protected function init()
{
$config = $this->getContainer()->get('config');
if (file_exists($this->cacheFile) && $config->get('useCache')) {
$this->data = $this->getFileManager()->getContents($this->cacheFile);
} else {
$this->data = $this->getClassNameHash($this->paths['corePath']);
foreach ($this->getContainer()->get('metadata')->getModuleList() as $moduleName) {
$path = str_replace('{*}', $moduleName, $this->paths['modulePath']);
$this->data = array_merge($this->data, $this->getClassNameHash($path));
}
$this->data = array_merge($this->data, $this->getClassNameHash($this->paths['customPath']));
if ($config->get('useCache')) {
$result = $this->getFileManager()->putContentsPHP($this->cacheFile, $this->data);
if ($result == false) {
throw new \Espo\Core\Exceptions\Error();
}
}
}
}
protected function getFileManager()
{
return $this->container->get('fileManager');
@@ -83,15 +58,19 @@ class ServiceFactory
return $this->container;
}
protected function init()
{
$classParser = $this->getContainer()->get('classParser');
$classParser->setAllowedMethods(null);
$this->data = $classParser->getData($this->paths, $this->cacheFile);
}
protected function getClassName($name)
{
$name = Util::normilizeClassName($name);
if (!isset($this->data)) {
$this->init();
}
$name = ucfirst($name);
if (isset($this->data[$name])) {
return $this->data[$name];
}
@@ -127,28 +106,5 @@ class ServiceFactory
}
throw new Error("Class '$className' does not exist");
}
// TODO delegate to another class
protected function getClassNameHash($dirs)
{
if (is_string($dirs)) {
$dirs = (array) $dirs;
}
$data = array();
foreach ($dirs as $dir) {
if (file_exists($dir)) {
$fileList = $this->getFileManager()->getFileList($dir, false, '\.php$', true);
foreach ($fileList as $file) {
$filePath = Util::concatPath($dir, $file);
$className = Util::getClassName($filePath);
$fileName = $this->getFileManager()->getFileName($filePath);
$data[$fileName] = $className;
}
}
}
return $data;
}
}

View File

@@ -53,6 +53,11 @@ abstract class Base implements Injectable
return $this->injections[$name];
}
protected function addDependency($name)
{
$this->dependencies[] = $name;
}
public function getDependencyList()
{
return $this->dependencies;

View File

@@ -0,0 +1,30 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Templates\Controllers;
class Base extends \Espo\Core\Controllers\Record
{
}

View File

@@ -0,0 +1,30 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Templates\Controllers;
class Person extends \Espo\Core\Controllers\Record
{
}

View File

@@ -0,0 +1,29 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Templates\Entities;
class Base extends \Espo\Core\ORM\Entity
{
}

View File

@@ -0,0 +1,29 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Templates\Entities;
class Person extends \Espo\Core\Entities\Person
{
}

View File

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

View File

@@ -0,0 +1,66 @@
{
"fields": {
"name": {
"type": "varchar",
"required": true
},
"description": {
"type": "text"
},
"createdAt": {
"type": "datetime",
"readOnly": true
},
"modifiedAt": {
"type": "datetime",
"readOnly": true
},
"createdBy": {
"type": "link",
"readOnly": true
},
"modifiedBy": {
"type": "link",
"readOnly": true
},
"assignedUser": {
"type": "link",
"required": true
},
"teams": {
"type": "linkMultiple"
}
},
"links": {
"createdBy": {
"type": "belongsTo",
"entity": "User"
},
"modifiedBy": {
"type": "belongsTo",
"entity": "User"
},
"assignedUser": {
"type": "belongsTo",
"entity": "User"
},
"teams": {
"type": "hasMany",
"entity": "Team",
"relationName": "EntityTeam",
"layoutRelationshipsDisabled": true
}
},
"collection": {
"sortBy": "createdAt",
"asc": false
},
"indexes": {
"name": {
"columns": ["name", "deleted"]
},
"assignedUser": {
"columns": ["assignedUserId", "deleted"]
}
}
}

View File

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

View File

@@ -0,0 +1,111 @@
{
"fields": {
"name": {
"type": "personName"
},
"salutationName": {
"type": "enum",
"options": ["", "Mr.", "Mrs.", "Ms.", "Dr."]
},
"firstName": {
"type": "varchar",
"maxLength": 100,
"default": ""
},
"lastName": {
"type": "varchar",
"maxLength": 100,
"required": true,
"default": ""
},
"description": {
"type": "text"
},
"createdAt": {
"type": "datetime",
"readOnly": true
},
"emailAddress": {
"type": "email"
},
"phoneNumber": {
"type": "phone",
"typeList": ["Mobile", "Office", "Home", "Fax", "Other"],
"defaultType": "Mobile"
},
"address": {
"type": "address"
},
"addressStreet": {
"type": "text",
"maxLength": 255,
"dbType": "varchar"
},
"addressCity": {
"type": "varchar"
},
"addressState": {
"type": "varchar"
},
"addressCountry": {
"type": "varchar"
},
"addressPostalCode": {
"type": "varchar"
},
"modifiedAt": {
"type": "datetime",
"readOnly": true
},
"createdBy": {
"type": "link",
"readOnly": true
},
"modifiedBy": {
"type": "link",
"readOnly": true
},
"assignedUser": {
"type": "link",
"required": true
},
"teams": {
"type": "linkMultiple"
}
},
"links": {
"createdBy": {
"type": "belongsTo",
"entity": "User"
},
"modifiedBy": {
"type": "belongsTo",
"entity": "User"
},
"assignedUser": {
"type": "belongsTo",
"entity": "User"
},
"teams": {
"type": "hasMany",
"entity": "Team",
"relationName": "EntityTeam",
"layoutRelationshipsDisabled": true
}
},
"collection": {
"sortBy": "createdAt",
"asc": false
},
"indexes": {
"firstName": {
"columns": ["firstName", "deleted"]
},
"name": {
"columns": ["firstName", "lastName"]
},
"assignedUser": {
"columns": ["assignedUserId", "deleted"]
}
}
}

View File

@@ -0,0 +1,30 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Templates\Repositories;
class Base extends \Espo\Core\ORM\Repositories\RDB
{
}

View File

@@ -0,0 +1,30 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Templates\Repositories;
class Person extends \Espo\Core\ORM\Repositories\RDB
{
}

View File

@@ -0,0 +1,30 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Templates\Services;
class Base extends \Espo\Services\Record
{
}

View File

@@ -0,0 +1,30 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Templates\Services;
class Person extends \Espo\Services\Record
{
}

View File

@@ -30,6 +30,7 @@ class UpgradeManager extends Upgrades\Base
protected $params = array(
'packagePath' => 'data/upload/upgrades',
'backupPath' => 'data/.backup/upgrades',
'scriptNames' => array(
'before' => 'BeforeUpgrade',

View File

@@ -22,9 +22,11 @@
namespace Espo\Core\Upgrades\Actions;
use Espo\Core\Utils\Util,
Espo\Core\Utils\Json,
Espo\Core\Exceptions\Error;
use Espo\Core\Utils\Util;
use Espo\Core\Utils\System;
use Espo\Core\Utils\Json;
use Espo\Core\Exceptions\Error;
use vierbergenlars\SemVer;
abstract class Base
{
@@ -32,6 +34,8 @@ abstract class Base
private $entityManager;
private $helper;
protected $data;
protected $params = null;
@@ -129,7 +133,7 @@ abstract class Base
return $this->config;
}
protected function getEntityManager()
public function getEntityManager()
{
if (!isset($this->entityManager)) {
$this->entityManager = $this->getContainer()->get('entityManager');
@@ -179,46 +183,60 @@ abstract class Base
*/
protected function isAcceptable()
{
$manifest = $this->getManifest();
$res = $this->checkPackageType();
$res &= $this->checkVersions();
//check php version
if (isset($manifest['php'])) {
$res &= $this->checkVersions($manifest['php'], System::getPhpVersion(), 'Your PHP version does not support this installation package.');
}
//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.');
}
//check dependencies
if (!empty($manifest['dependencies'])) {
$res &= $this->checkDependencies($manifest['dependencies']);
}
return (bool) $res;
}
protected function checkVersions()
public function checkVersions($versionList, $currentVersion, $errorMessage = '')
{
$manifest = $this->getManifest();
/** check acceptable versions */
$version = $manifest['acceptableVersions'];
if (empty($version)) {
if (empty($versionList)) {
return true;
}
$currentVersion = $this->getConfig()->get('version');
if (is_string($version)) {
$version = (array) $version;
if (is_string($versionList)) {
$versionList = (array) $versionList;
}
foreach ($version as $strVersion) {
try {
$semver = new SemVer\version($currentVersion);
} catch (\Exception $e) {
$GLOBALS['log']->error('Cannot recognize currentVersion ['.$currentVersion.'], error: '.$e->getMessage().'.');
return false;
}
$strVersion = trim($strVersion);
foreach ($versionList as $version) {
if ($strVersion == $currentVersion) {
return true;
$isInRange = false;
try {
$isInRange = $semver->satisfies(new SemVer\expression($version));
} catch (\Exception $e) {
$GLOBALS['log']->error('Version identification error: '.$e->getMessage().'.');
}
$strVersion = str_replace('\\', '', $strVersion);
$strVersion = preg_quote($strVersion);
$strVersion = str_replace('\\*', '+', $strVersion);
if (preg_match('/^'.$strVersion.'/', $currentVersion)) {
if ($isInRange) {
return true;
}
}
$this->throwErrorAndRemovePackage('Your EspoCRM version doesn\'t match for this installation package.');
$this->throwErrorAndRemovePackage($errorMessage);
}
protected function checkPackageType()
@@ -240,6 +258,11 @@ abstract class Base
return true;
}
protected function checkDependencies($dependencyList)
{
return true;
}
/**
* Run scripts by type
* @param string $type Ex. "before", "after"
@@ -293,7 +316,7 @@ abstract class Base
/**
* Get a list of files defined in manifest.json
*
* @return [type] [description]
* @return array
*/
protected function getDeleteFileList()
{
@@ -341,6 +364,18 @@ abstract class Base
return $this->data['fileList'];
}
protected function getRestoreFileList()
{
if (!isset($this->data['restoreFileList'])) {
$backupPath = $this->getPath('backupPath');
$backupFilePath = Util::concatPath($backupPath, self::FILES);
$this->data['restoreFileList'] = $this->getFileManager()->getFileList($backupFilePath, true, '', true, true);
}
return $this->data['restoreFileList'];
}
protected function copy($sourcePath, $destPath, $recursively = false, array $fileList = null, $copyOnlyFiles = false)
{
try {
@@ -467,18 +502,31 @@ abstract class Base
/**
* Execute an action. For ex., execute uninstall action in install
*
* @param [type] $actionName [description]
* @param [type] $data [description]
* @return [type] [description]
* @param string $actionName
* @param string $data
*
* @return void
*/
protected function executeAction($actionName, $data)
{
$currentAction = $this->getActionManager()->getAction();
$actionManager = $this->getActionManager();
$this->getActionManager()->setAction($actionName);
$this->getActionManager()->run($data);
$currentAction = $actionManager->getAction();
$actionManager->setAction($actionName);
$actionManager->run($data);
$actionManager->setAction($currentAction);
}
protected function initialize()
{
}
protected function finalize()
{
$this->getActionManager()->setAction($currentAction);
}
protected function beforeRunAction()
@@ -491,5 +539,38 @@ abstract class Base
}
protected function clearCache()
{
return $this->getContainer()->get('dataManager')->clearCache();
}
}
protected function checkIsWritable()
{
$fullFileList = array_merge($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));
}
}
protected function backupExistingFiles()
{
$fullFileList = array_merge($this->getDeleteFileList(), $this->getCopyFileList());
$backupPath = $this->getPath('backupPath');
return $this->copy('', array($backupPath, self::FILES), false, $fullFileList);
}
protected function getHelper()
{
if (!isset($this->helper)) {
$this->helper = new Helper();
}
$this->helper->setActionObject($this);
return $this->helper;
}
}

View File

@@ -24,14 +24,18 @@ namespace Espo\Core\Upgrades\Actions\Base;
class Delete extends \Espo\Core\Upgrades\Actions\Base
{
public function run($processId)
public function run($data)
{
$processId = $data['id'];
$GLOBALS['log']->debug('Delete package process ['.$processId.']: start run.');
if (empty($processId)) {
throw new Error('Delete package package ID was not specified.');
}
$this->initialize();
$this->setProcessId($processId);
$this->beforeRunAction();
@@ -41,6 +45,8 @@ class Delete extends \Espo\Core\Upgrades\Actions\Base
$this->afterRunAction();
$this->finalize();
$GLOBALS['log']->debug('Delete package process ['.$processId.']: end run.');
}

View File

@@ -23,6 +23,7 @@
namespace Espo\Core\Upgrades\Actions\Base;
use Espo\Core\Exceptions\Error;
use Espo\Core\Utils\Util;
class Install extends \Espo\Core\Upgrades\Actions\Base
{
@@ -39,8 +40,10 @@ class Install extends \Espo\Core\Upgrades\Actions\Base
* @param string $processId Upgrade/Extension ID, gotten in upload stage
* @return bool
*/
public function run($processId)
public function run($data)
{
$processId = $data['id'];
$GLOBALS['log']->debug('Installation process ['.$processId.']: start run.');
if (empty($processId)) {
@@ -49,6 +52,8 @@ class Install extends \Espo\Core\Upgrades\Actions\Base
$this->setProcessId($processId);
$this->initialize();
$this->isCopied = false;
/** check if an archive is unzipped, if no then unzip */
@@ -58,22 +63,25 @@ class Install extends \Espo\Core\Upgrades\Actions\Base
$this->isAcceptable();
}
//check permissions copied and deleted files
$this->checkIsWritable();
$this->backupExistingFiles();
$this->beforeRunAction();
/* run before install script */
$this->runScript('before');
/* remove files defined in a manifest */
if (!$this->deleteFiles()) {
$this->throwErrorAndRemovePackage('Permission denied to delete files.');
}
/* copy files from directory "Files" to EspoCRM files */
if (!$this->copyFiles()) {
$this->throwErrorAndRemovePackage('Cannot copy files.');
}
$this->isCopied = true;
/* remove files defined in a manifest */
$this->deleteFiles(true);
if (!$this->systemRebuild()) {
$this->throwErrorAndRemovePackage('Error occurred while EspoCRM rebuild.');
}
@@ -83,23 +91,37 @@ class Install extends \Espo\Core\Upgrades\Actions\Base
$this->afterRunAction();
$this->clearCache();
/* delete unziped files */
$this->deletePackageFiles();
$this->finalize();
$GLOBALS['log']->debug('Installation process ['.$processId.']: end run.');
}
protected function restoreFiles()
{
$backupPath = $this->getPath('backupPath');
$res = true;
if ($this->isCopied) {
$res &= $this->copy(array($backupPath, self::FILES), '', true);
$GLOBALS['log']->info('Restore: copy back');
if (!$this->isCopied) {
return;
}
$res &= $this->getFileManager()->removeInDir($backupPath, true);
$GLOBALS['log']->info('Installer: Restore previous files.');
$backupPath = $this->getPath('backupPath');
$backupFilePath = Util::concatPath($backupPath, self::FILES);
$backupFileList = $this->getRestoreFileList();
$copyFileList = $this->getCopyFileList();
$deleteFileList = array_diff($copyFileList, $backupFileList);
$res = $this->copy($backupFilePath, '', true);
$res &= $this->getFileManager()->remove($deleteFileList, null, true);
if ($res) {
$this->getFileManager()->removeInDir($backupPath, true);
}
return $res;
}

View File

@@ -22,13 +22,16 @@
namespace Espo\Core\Upgrades\Actions\Base;
use Espo\Core\Exceptions\Error,
Espo\Core\Utils\Util;
use Espo\Core\Exceptions\Error;
use Espo\Core\Utils\Util;
use Espo\Core\Utils\Json;
class Uninstall extends \Espo\Core\Upgrades\Actions\Base
{
public function run($processId)
public function run($data)
{
$processId = $data['id'];
$GLOBALS['log']->debug('Uninstallation process ['.$processId.']: start run.');
if (empty($processId)) {
@@ -37,44 +40,50 @@ class Uninstall extends \Espo\Core\Upgrades\Actions\Base
$this->setProcessId($processId);
$this->initialize();
$this->checkIsWritable();
$this->beforeRunAction();
/* run before install script */
$this->runScript('beforeUninstall');
if (!isset($data['isNotRunScriptBefore']) || !$data['isNotRunScriptBefore']) {
$this->runScript('beforeUninstall');
}
$backupPath = $this->getPath('backupPath');
if (file_exists($backupPath)) {
/* remove extension files, saved in fileList */
if (!$this->deleteFiles(true)) {
throw new Error('Permission denied to delete files.');
}
/* copy core files */
if (!$this->copyFiles()) {
throw new Error('Cannot copy files.');
throw new $this->throwErrorAndRemovePackage('Cannot copy files.');
}
/* remove extension files, saved in fileList */
if (!$this->deleteFiles(true)) {
throw new $this->throwErrorAndRemovePackage('Permission denied to delete files.');
}
}
if (!$this->systemRebuild()) {
throw new Error('Error occurred while EspoCRM rebuild.');
throw new $this->throwErrorAndRemovePackage('Error occurred while EspoCRM rebuild.');
}
/* run before install script */
$this->runScript('afterUninstall');
/* run after uninstall script */
if (!isset($data['isNotRunScriptAfter']) || !$data['isNotRunScriptAfter']) {
$this->runScript('afterUninstall');
}
$this->afterRunAction();
$this->clearCache();
/* delete backup files */
$this->deletePackageFiles();
$GLOBALS['log']->debug('Uninstallation process ['.$processId.']: end run.');
}
$this->finalize();
protected function getDeleteFileList()
{
$extensionEntity = $this->getExtensionEntity();
return $extensionEntity->get('fileList');
$GLOBALS['log']->debug('Uninstallation process ['.$processId.']: end run.');
}
protected function restoreFiles()
@@ -87,6 +96,13 @@ class Uninstall extends \Espo\Core\Upgrades\Actions\Base
}
$res = $this->copy($filesPath, '', true);
$manifestJson = $this->getFileManager()->getContents(array($packagePath, $this->manifestName));
$manifest = Json::decode($manifestJson, true);
if (!empty($manifest['delete'])) {
$res &= $this->getFileManager()->remove($manifest['delete'], null, true);
}
$res &= $this->getFileManager()->removeInDir($packagePath, true);
return $res;
@@ -129,4 +145,41 @@ class Uninstall extends \Espo\Core\Upgrades\Actions\Base
throw new Error($errorMessage);
}
protected function getCopyFileList()
{
if (!isset($this->data['fileList'])) {
$backupPath = $this->getPath('backupPath');
$filesPath = Util::concatPath($backupPath, self::FILES);
$this->data['fileList'] = $this->getFileManager()->getFileList($filesPath, true, '', true, true);
}
return $this->data['fileList'];
}
protected function getRestoreFileList()
{
if (!isset($this->data['restoreFileList'])) {
$packagePath = $this->getPackagePath();
$filesPath = Util::concatPath($packagePath, self::FILES);
if (!file_exists($filesPath)) {
$this->unzipArchive($packagePath);
}
$this->data['restoreFileList'] = $this->getFileManager()->getFileList($filesPath, true, '', true, true);
}
return $this->data['restoreFileList'];
}
protected function getDeleteFileList()
{
$packageFileList = $this->getRestoreFileList();
$backupFileList = $this->getCopyFileList();
$deleteFileList = array_diff($packageFileList, $backupFileList);
return $deleteFileList;
}
}

View File

@@ -38,6 +38,10 @@ class Upload extends \Espo\Core\Upgrades\Actions\Base
$GLOBALS['log']->debug('Installation process ['.$processId.']: start upload the package.');
$this->initialize();
$this->beforeRunAction();
$packagePath = $this->getPackagePath();
$packageArchivePath = $this->getPackagePath(true);
@@ -55,6 +59,10 @@ class Upload extends \Espo\Core\Upgrades\Actions\Base
$this->isAcceptable();
$this->afterRunAction();
$this->finalize();
$GLOBALS['log']->debug('Installation process ['.$processId.']: end upload the package.');
return $processId;

View File

@@ -35,29 +35,15 @@ class Delete extends \Espo\Core\Upgrades\Actions\Base\Delete
*/
protected function getExtensionEntity()
{
return $this->extensionEntity;
}
/**
* Set Extension Entity
*
* @param \Espo\Entities\Extension $extensionEntity
*/
protected function setExtensionEntity(\Espo\Entities\Extension $extensionEntity)
{
$this->extensionEntity = $extensionEntity;
}
protected function beforeRunAction()
{
$processId = $this->getProcessId();
/** get extension entity */
$extensionEntity = $this->getEntityManager()->getEntity('Extension', $processId);
if (!isset($extensionEntity)) {
throw new Error('Extension Entity not found.');
if (!isset($this->extensionEntity)) {
$processId = $this->getProcessId();
$this->extensionEntity = $this->getEntityManager()->getEntity('Extension', $processId);
if (!isset($this->extensionEntity)) {
throw new Error('Extension Entity not found.');
}
}
$this->setExtensionEntity($extensionEntity);
return $this->extensionEntity;
}
protected function afterRunAction()

View File

@@ -37,8 +37,6 @@ class Install extends \Espo\Core\Upgrades\Actions\Base\Install
$this->uninstallExtension();
$this->deleteExtension();
}
$this->copyExistingFiles();
}
protected function afterRunAction()
@@ -51,12 +49,11 @@ class Install extends \Espo\Core\Upgrades\Actions\Base\Install
*
* @return bool
*/
protected function copyExistingFiles()
protected function backupExistingFiles()
{
$fileList = $this->getCopyFileList();
$backupPath = $this->getPath('backupPath');
parent::backupExistingFiles();
$res = $this->copy('', array($backupPath, self::FILES), false, $fileList);
$backupPath = $this->getPath('backupPath');
/** copy scripts files */
$packagePath = $this->getPackagePath();
@@ -65,19 +62,6 @@ class Install extends \Espo\Core\Upgrades\Actions\Base\Install
return $res;
}
protected function restoreFiles()
{
$res = true;
if ($this->isCopied) {
$extensionFileList = $this->getCopyFileList();
$res &= $this->getFileManager()->remove($extensionFileList);
}
$res &= parent::restoreFiles();
return $res;
}
protected function isNew()
{
$extensionEntity = $this->getExtensionEntity();
@@ -154,7 +138,7 @@ class Install extends \Espo\Core\Upgrades\Actions\Base\Install
$data = array(
'id' => $this->getProcessId(),
'name' => $manifest['name'],
'name' => trim($manifest['name']),
'isInstalled' => true,
'version' => $manifest['version'],
'fileList' => $fileList,
@@ -187,8 +171,8 @@ class Install extends \Espo\Core\Upgrades\Actions\Base\Install
* Throw an exception and remove package files.
* Redeclared to prevent of deleting a package of installed extension.
*
* @param string $errorMessage [description]
* @return [type] [description]
* @param string $errorMessage
* @return void
*/
protected function throwErrorAndRemovePackage($errorMessage = '')
{
@@ -208,7 +192,11 @@ class Install extends \Espo\Core\Upgrades\Actions\Base\Install
{
$extensionEntity = $this->getExtensionEntity();
$this->executeAction(ExtensionManager::UNINSTALL, $extensionEntity->get('id'));
$this->executeAction(ExtensionManager::UNINSTALL, array(
'id' => $extensionEntity->get('id'),
'isNotRunScriptAfter' => true,
)
);
}
/**
@@ -220,10 +208,11 @@ class Install extends \Espo\Core\Upgrades\Actions\Base\Install
{
$extensionEntity = $this->getExtensionEntity();
$this->executeAction(ExtensionManager::DELETE, $extensionEntity->get('id'));
$this->executeAction(ExtensionManager::DELETE, array('id' => $extensionEntity->get('id')));
}
protected function checkDependencies($dependencyList)
{
return $this->getHelper()->checkDependencies($dependencyList);
}
}

View File

@@ -35,29 +35,15 @@ class Uninstall extends \Espo\Core\Upgrades\Actions\Base\Uninstall
*/
protected function getExtensionEntity()
{
return $this->extensionEntity;
}
/**
* Set Extension Entity
*
* @param \Espo\Entities\Extension $extensionEntity [description]
*/
protected function setExtensionEntity(\Espo\Entities\Extension $extensionEntity)
{
$this->extensionEntity = $extensionEntity;
}
protected function beforeRunAction()
{
$processId = $this->getProcessId();
/** get extension entity */
$extensionEntity = $this->getEntityManager()->getEntity('Extension', $processId);
if (!isset($extensionEntity)) {
throw new Error('Extension Entity not found.');
if (!isset($this->extensionEntity)) {
$processId = $this->getProcessId();
$this->extensionEntity = $this->getEntityManager()->getEntity('Extension', $processId);
if (!isset($this->extensionEntity)) {
throw new Error('Extension Entity not found.');
}
}
$this->setExtensionEntity($extensionEntity);
return $this->extensionEntity;
}
protected function afterRunAction()
@@ -68,4 +54,14 @@ class Uninstall extends \Espo\Core\Upgrades\Actions\Base\Uninstall
$extensionEntity->set('isInstalled', false);
$this->getEntityManager()->saveEntity($extensionEntity);
}
protected function getRestoreFileList()
{
if (!isset($this->data['restoreFileList'])) {
$extensionEntity = $this->getExtensionEntity();
$this->data['restoreFileList'] = $extensionEntity->get('fileList');
}
return $this->data['restoreFileList'];
}
}

View File

@@ -24,5 +24,8 @@ namespace Espo\Core\Upgrades\Actions\Extension;
class Upload extends \Espo\Core\Upgrades\Actions\Base\Upload
{
protected function checkDependencies($dependencyList)
{
return $this->getHelper()->checkDependencies($dependencyList);
}
}

View File

@@ -0,0 +1,77 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Upgrades\Actions;
use Espo\Core\Exceptions\Error;
class Helper
{
private $actionObject;
public function __construct($actionObject = null)
{
if (isset($actionObject)) {
$this->setActionObject($actionObject);
}
}
public function setActionObject(\Espo\Core\Upgrades\Actions\Base $actionObject)
{
$this->actionObject = $actionObject;
}
protected function getActionObject()
{
return $this->actionObject;
}
/**
* Check dependencies
*
* @param array | string $dependencyList
*
* @return bool
*/
public function checkDependencies($dependencyList)
{
if (!is_array($dependencyList)) {
$dependencyList = (array) $dependencyList;
}
$actionObject = $this->getActionObject();
foreach ($dependencyList as $extensionName => $extensionVersion) {
$dependencyExtensionEntity = $actionObject->getEntityManager()->getRepository('Extension')->where(array(
'name' => trim($extensionName),
'isInstalled' => true,
))->findOne();
$errorMessage = 'Dependency Error: The extension "'.$extensionName.'" with version "'.$extensionVersion.'" is missing.';
if (!isset($dependencyExtensionEntity) || !$actionObject->checkVersions($extensionVersion, $dependencyExtensionEntity->get('version'), $errorMessage)) {
throw new Error($errorMessage);
}
}
return true;
}
}

View File

@@ -24,17 +24,12 @@ namespace Espo\Core\Upgrades\Actions\Upgrade;
class Install extends \Espo\Core\Upgrades\Actions\Base\Install
{
protected function systemRebuild()
protected function finalize()
{
$manifest = $this->getManifest();
$res = $this->getConfig()->set('version', $manifest['version']);
if (method_exists($this->getConfig(), 'save')) {
$res = $this->getConfig()->save();
}
$res &= parent::systemRebuild();
return $res;
$this->getConfig()->set('version', $manifest['version']);
$this->getConfig()->save();
}
/**

View File

@@ -67,7 +67,9 @@ class Output
$currentRoute = $this->getSlim()->router()->getCurrentRoute();
if (isset($currentRoute)) {
$GLOBALS['log']->error('API ['.$this->getSlim()->request()->getMethod().']:'.$currentRoute->getPattern().', Params:'.print_r($currentRoute->getParams(), true).', InputData: '.$this->getSlim()->request()->getBody().' - '.$message);
$inputData = $this->getSlim()->request()->getBody();
$inputData = $this->clearPasswords($inputData);
$GLOBALS['log']->error('API ['.$this->getSlim()->request()->getMethod().']:'.$currentRoute->getPattern().', Params:'.print_r($currentRoute->getParams(), true).', InputData: '.$inputData.' - '.$message);
}
$this->displayError($message, $code, $isPrint);
@@ -118,5 +120,17 @@ class Output
return null;
}
/**
* Clear passwords for inputData
*
* @param string $inputData
*
* @return string
*/
protected function clearPasswords($inputData)
{
return preg_replace('/"(.*?password.*?)":".*?"/i', '"$1":"*****"', $inputData);
}
}

View File

@@ -23,6 +23,7 @@
namespace Espo\Core\Utils;
use \Espo\Core\Exceptions\Error;
use \Espo\Core\Exceptions\Forbidden;
class Auth
{
@@ -44,6 +45,8 @@ class Auth
$authenticationMethod = $this->config->get('authenticationMethod', 'Espo');
$authenticationClassName = "\\Espo\\Core\\Utils\\Authentication\\" . $authenticationMethod;
$this->authentication = new $authenticationClassName($this->config, $this->entityManager, $this);
$this->request = $this->container->get('slim')->request();
}
public function useNoAuth($isAdmin = false)
@@ -63,8 +66,6 @@ class Auth
public function login($username, $password)
{
$GLOBALS['log']->debug('AUTH: Try to authenticate');
$entityManager = $this->entityManager;
$authToken = $entityManager->getRepository('AuthToken')->where(array('token' => $password))->findOne();
@@ -72,24 +73,28 @@ class Auth
$user = $this->authentication->login($username, $password, $authToken);
if ($user) {
if (!$user->isActive()) {
$GLOBALS['log']->debug("AUTH: Trying to login as user '".$user->get('userName')."' which is not active.");
return false;
}
$entityManager->setUser($user);
$this->container->setUser($user);
$GLOBALS['log']->debug('AUTH: Result of authenticate is [true]');
if (!$authToken) {
$authToken = $entityManager->getEntity('AuthToken');
$token = $this->createToken($user);
$authToken->set('token', $token);
$authToken->set('hash', $user->get('password'));
$authToken->set('ipAddress', $_SERVER['REMOTE_ADDR']);
$authToken->set('userId', $user->id);
if ($this->request->headers->get('HTTP_ESPO_AUTHORIZATION')) {
if (!$authToken) {
$authToken = $entityManager->getEntity('AuthToken');
$token = $this->createToken($user);
$authToken->set('token', $token);
$authToken->set('hash', $user->get('password'));
$authToken->set('ipAddress', $_SERVER['REMOTE_ADDR']);
$authToken->set('userId', $user->id);
}
$authToken->set('lastAccess', date('Y-m-d H:i:s'));
$entityManager->saveEntity($authToken);
$user->set('token', $authToken->get('token'));
}
$authToken->set('lastAccess', date('Y-m-d H:i:s'));
$entityManager->saveEntity($authToken);
$user->set('token', $authToken->get('token'));
return true;
}
}

View File

@@ -82,14 +82,14 @@ class Autoload
protected function init()
{
if (file_exists($this->cacheFile) && $this->getConfig()->get('useCache')) {
$this->data = $this->getFileManager()->getContents($this->cacheFile);
$this->data = $this->getFileManager()->getPhpContents($this->cacheFile);
return;
}
$this->data = $this->unify();
if ($this->getConfig()->get('useCache')) {
$result = $this->getFileManager()->putContentsPHP($this->cacheFile, $this->data);
$result = $this->getFileManager()->putPhpContents($this->cacheFile, $this->data);
if ($result == false) {
throw new \Espo\Core\Exceptions\Error('Autoload: Cannot save unified autoload.');
}

View File

@@ -150,7 +150,8 @@ class Config
$removeData = empty($this->removeData) ? null : $this->removeData;
$result = $this->getFileManager()->mergeContentsPHP($this->configPath, $values, $removeData);
$result = $this->getFileManager()->mergePhpContents($this->configPath, $values, $removeData);
if ($result) {
$this->changedData = array();
$this->removeData = array();
@@ -162,7 +163,7 @@ class Config
public function getDefaults()
{
return $this->getFileManager()->getContents($this->defaultConfigPath);
return $this->getFileManager()->getPhpContents($this->defaultConfigPath);
}
/**
@@ -178,9 +179,9 @@ class Config
$configPath = file_exists($this->configPath) ? $this->configPath : $this->defaultConfigPath;
$this->data = $this->getFileManager()->getContents($configPath);
$this->data = $this->getFileManager()->getPhpContents($configPath);
$systemConfig = $this->getFileManager()->getContents($this->systemConfigPath);
$systemConfig = $this->getFileManager()->getPhpContents($this->systemConfigPath);
$this->data = Util::merge($systemConfig, $this->data);
return $this->data;
@@ -261,7 +262,7 @@ class Config
}
if (empty($this->adminItems)) {
$this->adminItems= Util::merge($data['systemItems'], $data['adminItems']);
$this->adminItems = array_merge($data['systemItems'], $data['adminItems']);
}
return $this->adminItems;

View File

@@ -20,17 +20,45 @@
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Services;
namespace Espo\Core\Utils\Cron;
use \PDO;
use \Espo\Core\CronManager;
use \Espo\Core\Utils\Config;
use \Espo\Core\ORM\EntityManager;
class Job extends Record
class Job
{
private $config;
private $entityManager;
public function __construct(Config $config, EntityManager $entityManager)
{
$this->config = $config;
$this->entityManager = $entityManager;
}
protected function getConfig()
{
return $this->config;
}
protected function getEntityManager()
{
return $this->entityManager;
}
/**
* Get Pending Jobs
*
* @return array
*/
public function getPendingJobs()
{
/** Mark Failed old jobs and remove pending duplicates */
$this->markFailedJobs();
$this->markJobAttempts();
$this->removePendingJobDuplicates();
$jobList = $this->getActiveJobs();
@@ -52,21 +80,21 @@ class Job extends Record
*
* @param string $displayColumns
* @param string $status
*
* @return array
*/
protected function getActiveJobs($displayColumns = '*', $status = CronManager::PENDING, $fetchMode = PDO::FETCH_ASSOC)
public function getActiveJobs($displayColumns = '*', $status = CronManager::PENDING, $fetchMode = PDO::FETCH_ASSOC)
{
$jobConfigs = $this->getConfig()->get('cron');
$currentTime = time();
$periodTime = $currentTime - intval($jobConfigs['jobPeriod']);
$limit = empty($jobConfigs['maxJobNumber']) ? '' : 'LIMIT '.$jobConfigs['maxJobNumber'];
$query = "SELECT " . $displayColumns . " FROM job WHERE
`status` = '" . $status . "'
AND execute_time BETWEEN '".date('Y-m-d H:i:s', $periodTime)."' AND '".date('Y-m-d H:i:s', $currentTime)."'
AND execute_time <= '".date('Y-m-d H:i:s', $currentTime)."'
AND deleted = 0
ORDER BY execute_time DESC ".$limit;
ORDER BY execute_time ASC ".$limit;
$pdo = $this->getEntityManager()->getPDO();
$sth = $pdo->prepare($query);
@@ -77,11 +105,22 @@ class Job extends Record
return $rows;
}
public function getJobByScheduledJob($scheduledJobId, $date)
/**
* Get Jobs by ScheduledJobId and date
*
* @param string $scheduledJobId
* @param string $time
*
* @return array
*/
public function getJobByScheduledJob($scheduledJobId, $time)
{
$dateObj = new \DateTime($time);
$timeWithoutSeconds = $dateObj->format('Y-m-d H:i:');
$query = "SELECT * FROM job WHERE
scheduled_job_id = '".$scheduledJobId."'
AND execute_time = '".$date."'
AND execute_time LIKE '".$timeWithoutSeconds."%'
AND deleted = 0
LIMIT 1";
@@ -107,7 +146,7 @@ class Job extends Record
$periodTime = $currentTime - intval($jobConfigs['jobPeriod']);
$update = "UPDATE job SET `status` = '" . CronManager::FAILED ."' WHERE
(`status` = '" . CronManager::PENDING ."' OR `status` = '" . CronManager::RUNNING ."')
(`status` = '" . CronManager::RUNNING ."')
AND execute_time < '".date('Y-m-d H:i:s', $periodTime)."' ";
$pdo = $this->getEntityManager()->getPDO();
@@ -122,10 +161,22 @@ class Job extends Record
*/
protected function removePendingJobDuplicates()
{
$duplicateJobs = $this->getActiveJobs('DISTINCT scheduled_job_id');
$pdo = $this->getEntityManager()->getPDO();
$query = "SELECT scheduled_job_id FROM job
WHERE
scheduled_job_id IS NOT NULL
AND `status` = '".CronManager::PENDING."'
AND execute_time <= '".date('Y-m-d H:i:s')."'
AND deleted = 0
GROUP BY scheduled_job_id
HAVING count( * ) > 1
ORDER BY execute_time ASC";
$sth = $pdo->prepare($query);
$sth->execute();
$duplicateJobs = $sth->fetchAll(PDO::FETCH_ASSOC);
foreach ($duplicateJobs as $row) {
if (!empty($row['scheduled_job_id'])) {
@@ -147,6 +198,40 @@ class Job extends Record
}
}
/**
* Mark job attempts
*
* @return void
*/
protected function markJobAttempts()
{
$query = "SELECT * FROM job WHERE
`status` = '" . CronManager::FAILED . "'
AND deleted = 0
AND execute_time <= '".date('Y-m-d H:i:s')."'
AND attempts > 0";
}
$pdo = $this->getEntityManager()->getPDO();
$sth = $pdo->prepare($query);
$sth->execute();
$rows = $sth->fetchAll(PDO::FETCH_ASSOC);
if ($rows) {
foreach ($rows as $row) {
$row['failed_attempts'] = isset($row['failed_attempts']) ? $row['failed_attempts'] : 0;
$attempts = $row['attempts'] - 1;
$failedAttempts = $row['failed_attempts'] + 1;
$update = "UPDATE job SET
`status` = '" . CronManager::PENDING ."',
attempts = '".$attempts."',
failed_attempts = '".$failedAttempts."'
WHERE
id = '".$row['id']."'
";
$pdo->prepare($update)->execute();
}
}
}
}

View File

@@ -18,41 +18,68 @@
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
************************************************************************/
namespace Espo\Services;
namespace Espo\Core\Utils\Cron;
use \PDO;
use \Espo\Core\Utils\Config;
use \Espo\Core\ORM\EntityManager;
class ScheduledJob extends \Espo\Services\Record
{
class ScheduledJob
{
private $config;
private $entityManager;
public function __construct(Config $config, EntityManager $entityManager)
{
$this->config = $config;
$this->entityManager = $entityManager;
}
protected function getConfig()
{
return $this->config;
}
protected function getEntityManager()
{
return $this->entityManager;
}
/**
* Get active Scheduler Jobs
*
* @return array
*/
public function getActiveJobs()
{
$query = "SELECT * FROM scheduled_job WHERE
`status` = 'Active'
AND deleted = 0";
AND deleted = 0";
$pdo = $this->getEntityManager()->getPDO();
$sth = $pdo->prepare($query);
$sth = $pdo->prepare($query);
$sth->execute();
$rows = $sth->fetchAll(PDO::FETCH_ASSOC);
$list = array();
foreach ($rows as $row) {
$list[] = $row;
}
return $list;
}
return $list;
}
/**
* Add record to ScheduledJobLogRecord about executed job
* @param string $scheduledJobId
* @param string $status
*
* @return string Id of created ScheduledJobLogRecord
* @param string $scheduledJobId
* @param string $status
*
* @return string ID of created ScheduledJobLogRecord
*/
public function addLogRecord($scheduledJobId, $status)
{
@@ -72,12 +99,7 @@ class ScheduledJob extends \Espo\Services\Record
'executionTime' => $lastRun,
));
$scheduledJobLogId = $entityManager->saveEntity($scheduledJobLog);
//$entityManager->getRepository('ScheduledJobLogRecord')->relate($scheduledJobLog, 'scheduledJob', $scheduledJob);
return $scheduledJobLogId;
}
}
return $scheduledJobLogId;
}
}

View File

@@ -158,6 +158,7 @@ class Converter
}
$ormMeta[$entityName]['fields'] = $this->convertFields($entityName, $entityMeta);
$ormMeta = $this->correctFields($entityName, $ormMeta);
$convertedLinks = $this->convertLinks($entityName, $entityMeta, $ormMeta);
@@ -168,47 +169,6 @@ class Converter
public function afterProcess(array $ormMeta)
{
$entityDefs = $this->getEntityDefs();
$currentOrmMeta = $ormMeta;
//load custom field definitions and customCodes
foreach($currentOrmMeta as $entityName => $entityParams) {
foreach($entityParams['fields'] as $fieldName => $fieldParams) {
//load custom field definitions
$fieldType = ucfirst($fieldParams['type']);
$className = '\Espo\Custom\Core\Utils\Database\Orm\Fields\\'.$fieldType;
if (!class_exists($className)) {
$className = '\Espo\Core\Utils\Database\Orm\Fields\\'.$fieldType;
}
if (class_exists($className) && method_exists($className, 'load')) {
$helperClass = new $className($this->metadata, $ormMeta, $entityDefs);
$fieldResult = $helperClass->process( $fieldName, $entityName );
if (isset($fieldResult['unset'])) {
$ormMeta = Util::unsetInArray($ormMeta, $fieldResult['unset']);
unset($fieldResult['unset']);
}
$ormMeta = Util::merge($ormMeta, $fieldResult);
} //END: load custom field definitions
//todo move to separate file
//add a field 'isFollowed' for scopes with 'stream => true'
$scopeDefs = $this->getMetadata()->get('scopes.'.$entityName);
if (isset($scopeDefs['stream']) && $scopeDefs['stream']) {
if (!isset($entityParams['fields']['isFollowed'])) {
$ormMeta[$entityName]['fields']['isFollowed'] = array(
'type' => 'varchar',
'notStorable' => true,
);
}
} //END: add a field 'isFollowed' for stream => true
}
}
foreach($ormMeta as $entityName => &$entityParams) {
foreach($entityParams['fields'] as $fieldName => &$fieldParams) {
@@ -226,7 +186,9 @@ class Converter
case 'foreignType':
$fieldParams['dbType'] = Entity::VARCHAR;
$fieldParams['len'] = $this->defaultLength['varchar'];
if (empty($fieldParams['len'])) {
$fieldParams['len'] = $this->defaultLength['varchar'];
}
break;
case 'bool':
@@ -234,7 +196,6 @@ class Converter
break;
}
}
}
return $ormMeta;
@@ -308,6 +269,56 @@ class Converter
return $outputMeta;
}
/**
* Correct fields defenitions based on \Espo\Custom\Core\Utils\Database\Orm\Fields
*
* @param array $ormMeta
*
* @return array
*/
protected function correctFields($entityName, array $ormMeta)
{
$entityDefs = $this->getEntityDefs();
$entityMeta = $ormMeta[$entityName];
//load custom field definitions and customCodes
foreach($entityMeta['fields'] as $fieldName => $fieldParams) {
//load custom field definitions
$fieldType = ucfirst($fieldParams['type']);
$className = '\Espo\Custom\Core\Utils\Database\Orm\Fields\\' . $fieldType;
if (!class_exists($className)) {
$className = '\Espo\Core\Utils\Database\Orm\Fields\\' . $fieldType;
}
if (class_exists($className) && method_exists($className, 'load')) {
$helperClass = new $className($this->metadata, $ormMeta, $entityDefs);
$fieldResult = $helperClass->process( $fieldName, $entityName );
if (isset($fieldResult['unset'])) {
$ormMeta = Util::unsetInArray($ormMeta, $fieldResult['unset']);
unset($fieldResult['unset']);
}
$ormMeta = Util::merge($ormMeta, $fieldResult);
} //END: load custom field definitions
}
//todo move to separate file
//add a field 'isFollowed' for scopes with 'stream => true'
$scopeDefs = $this->getMetadata()->get('scopes.'.$entityName);
if (isset($scopeDefs['stream']) && $scopeDefs['stream']) {
if (!isset($entityMeta['fields']['isFollowed'])) {
$ormMeta[$entityName]['fields']['isFollowed'] = array(
'type' => 'varchar',
'notStorable' => true,
);
}
} //END: add a field 'isFollowed' for stream => true
return $ormMeta;
}
protected function convertField($entityName, $fieldName, array $fieldParams, $fieldTypeMeta = null)
{
/** set default type if exists */

View File

@@ -34,30 +34,38 @@ class Currency extends Base
$alias = Util::toUnderScore($fieldName) . "_currency_alias";
return array(
$d = array(
$entityName => array(
'fields' => array(
$fieldName => array(
"type" => "float",
"orderBy" => $converedFieldName . " {direction}"
),
$fieldName . 'Converted' => array(
'type' => 'float',
'select' => Util::toUnderScore($entityName) . "." . $currencyColumnName . " * {$alias}.rate" ,
'where' =>
array (
"=" => Util::toUnderScore($entityName) . "." . $currencyColumnName . " * {$alias}.rate = {value}",
">" => Util::toUnderScore($entityName) . "." . $currencyColumnName . " * {$alias}.rate > {value}",
"<" => Util::toUnderScore($entityName) . "." . $currencyColumnName . " * {$alias}.rate < {value}",
">=" => Util::toUnderScore($entityName) . "." . $currencyColumnName . " * {$alias}.rate >= {value}",
"<=" => Util::toUnderScore($entityName) . "." . $currencyColumnName . " * {$alias}.rate <= {value}",
"<>" => Util::toUnderScore($entityName) . "." . $currencyColumnName . " * {$alias}.rate <> {value}"
),
'notStorable' => true,
'orderBy' => $converedFieldName . " {direction}"
),
)
),
),
);
$params = $this->getFieldParams($fieldName);
if (!empty($params['notStorable'])) {
$d[$entityName]['fields'][$fieldName]['notStorable'] = true;
} else {
$d[$entityName]['fields'][$fieldName . 'Converted'] = array(
'type' => 'float',
'select' => Util::toUnderScore($entityName) . "." . $currencyColumnName . " * {$alias}.rate" ,
'where' =>
array (
"=" => Util::toUnderScore($entityName) . "." . $currencyColumnName . " * {$alias}.rate = {value}",
">" => Util::toUnderScore($entityName) . "." . $currencyColumnName . " * {$alias}.rate > {value}",
"<" => Util::toUnderScore($entityName) . "." . $currencyColumnName . " * {$alias}.rate < {value}",
">=" => Util::toUnderScore($entityName) . "." . $currencyColumnName . " * {$alias}.rate >= {value}",
"<=" => Util::toUnderScore($entityName) . "." . $currencyColumnName . " * {$alias}.rate <= {value}",
"<>" => Util::toUnderScore($entityName) . "." . $currencyColumnName . " * {$alias}.rate <> {value}"
),
'notStorable' => true,
'orderBy' => $converedFieldName . " {direction}"
);
}
return $d;
}
}

View File

@@ -30,7 +30,7 @@ class Email extends Base
$entityName => array(
'fields' => array(
$fieldName => array(
'select' => 'email_address.name',
'select' => 'emailAddresses.name',
'where' =>
array (
'LIKE' => \Espo\Core\Utils\Util::toUnderScore($entityName) . ".id IN (
@@ -58,7 +58,7 @@ class Email extends Base
email_address.deleted = 0 AND email_address.name <> {value}
)"
),
'orderBy' => 'email_address.name {direction}',
'orderBy' => 'emailAddresses.name {direction}',
),
$fieldName .'Data' => array(
'type' => 'text',

View File

@@ -0,0 +1,57 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Utils\Database\Orm\Fields;
class Link extends Base
{
protected function load($fieldName, $entityName)
{
$fieldParams = $this->getFieldParams();
$data = array(
$entityName => array (
'fields' => array(
$fieldName.'Id' => array(
'type' => 'foreignId',
'index' => $fieldName,
),
$fieldName.'Name' => array(
'type' => 'varchar',
'notStorable' => true,
),
),
),
'unset' => array(
$entityName => array(
'fields.'.$fieldName,
),
),
);
if (!empty($fieldParams['notStorable'])) {
$data[$entityName]['fields'][$fieldName.'Id']['notStorable'] = true;
}
return $data;
}
}

View File

@@ -44,6 +44,11 @@ class LinkParent extends Base
),
),
),
'unset' => array(
$entityName => array(
'fields.'.$fieldName,
),
),
);
}
}

View File

@@ -30,7 +30,7 @@ class Phone extends Base
$entityName => array(
'fields' => array(
$fieldName => array(
'select' => 'phone_number.name',
'select' => 'phoneNumbers.name',
'where' =>
array (
'LIKE' => \Espo\Core\Utils\Util::toUnderScore($entityName) . ".id IN (
@@ -58,7 +58,7 @@ class Phone extends Base
phone_number.deleted = 0 AND phone_number.name <> {value}
)"
),
'orderBy' => 'phone_number.name {direction}',
'orderBy' => 'phoneNumbers.name {direction}',
),
$fieldName .'Data' => array(
'type' => 'text',

View File

@@ -36,6 +36,7 @@ class Base extends \Espo\Core\Utils\Database\Orm\Base
'conditions',
'additionalColumns',
'midKeys',
'noJoin'
);
protected function getParams()

View File

@@ -26,16 +26,30 @@ class BelongsTo extends Base
{
protected function load($linkName, $entityName)
{
$linkParams = $this->getLinkParams();
$foreignEntityName = $this->getForeignEntityName();
if (!empty($linkParams['noJoin'])) {
$fieldNameDefs = array(
'type' => 'varchar',
'notStorable' => true,
'relation' => $linkName,
'foreign' => $this->getForeignField('name', $foreignEntityName),
);
} else {
$fieldNameDefs = array(
'type' => 'foreign',
'relation' => $linkName,
'foreign' => $this->getForeignField('name', $foreignEntityName),
'notStorable' => false,
);
}
return array (
$entityName => array (
'fields' => array(
$linkName.'Name' => array(
'type' => 'foreign',
'relation' => $linkName,
'foreign' => $this->getForeignField('name', $foreignEntityName),
),
$linkName.'Name' => $fieldNameDefs,
$linkName.'Id' => array(
'type' => 'foreignId',
'index' => true,
@@ -46,7 +60,7 @@ class BelongsTo extends Base
'type' => 'belongsTo',
'entity' => $foreignEntityName,
'key' => $linkName.'Id',
'foreignKey' => 'id', //????
'foreignKey' => 'id',
),
),
),

View File

@@ -0,0 +1,60 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Utils\Database\Orm\Relations;
class BelongsToParent extends Base
{
protected function load($linkName, $entityName)
{
$linkParams = $this->getLinkParams();
return array(
$entityName => array (
'fields' => array(
$linkName.'Id' => array(
'type' => 'foreignId',
'index' => $linkName,
),
$linkName.'Type' => array(
'type' => 'foreignType',
'notNull' => false,
'index' => $linkName,
'len' => 100
),
$linkName.'Name' => array(
'type' => 'varchar',
'notStorable' => true,
),
),
'relations' => array(
$linkName => array(
'type' => 'belongsToParent',
'key' => $linkName.'Id',
),
),
),
);
}
}

View File

@@ -241,16 +241,19 @@ class Converter
$table->addColumn('id', 'int', array('length'=>$this->defaultLength['int'], 'autoincrement' => true, 'notnull' => true,)); //'unique' => true,
//add midKeys to a schema
$uniqueIndex = array();
foreach($relationParams['midKeys'] as $index => $midKey) {
$usMidKey = Util::toUnderScore($midKey);
$table->addColumn($usMidKey, $this->idParams['dbType'], array('length'=>$this->idParams['len']));
$table->addIndex(array($usMidKey));
$columnName = Util::toUnderScore($midKey);
$table->addColumn($columnName, $this->idParams['dbType'], array('length'=>$this->idParams['len']));
$table->addIndex(array($columnName));
} //END: add midKeys to a schema
$uniqueIndex[] = $columnName;
}
//END: add midKeys to a schema
//add additionalColumns
if (isset($relationParams['additionalColumns'])) {
if (!empty($relationParams['additionalColumns'])) {
foreach($relationParams['additionalColumns'] as $fieldName => $fieldParams) {
if (!isset($fieldParams['type'])) {
@@ -264,6 +267,17 @@ class Converter
}
} //END: add additionalColumns
//add unique indexes
if (!empty($relationParams['conditions'])) {
foreach ($relationParams['conditions'] as $fieldName => $fieldParams) {
$uniqueIndex[] = Util::toUnderScore($fieldName);
}
}
if (!empty($uniqueIndex)) {
$table->addUniqueIndex($uniqueIndex);
}
//END: add unique indexes
$table->addColumn('deleted', 'bool', array('default' => 0));
$table->setPrimaryKey(array("id"));
@@ -271,7 +285,6 @@ class Converter
return $table;
}
protected function getDbFieldParams($fieldParams)
{
$dbFieldParams = array();
@@ -333,7 +346,7 @@ class Converter
$fileList = $this->getFileManager()->getFileList($this->customTablePath, false, '\.php$', true);
foreach($fileList as $fileName) {
$fileData = $this->getFileManager()->getContents( array($this->customTablePath, $fileName) );
$fileData = $this->getFileManager()->getPhpContents( array($this->customTablePath, $fileName) );
if (is_array($fileData)) {
$customTables = Util::merge($customTables, $fileData);
}

View File

@@ -0,0 +1,48 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
return array(
'Autofollow' => array(
'fields' => array(
'id' => array(
'type' => 'id',
'dbType' => 'int',
'len' => '11',
'autoincrement' => true,
'unique' => true,
),
'entityType' => array(
'type' => 'varchar',
'len' => '100',
'index' => 'entityType',
),
'userId' => array(
'type' => 'varchar',
'len' => '24',
'index' => true,
)
)
)
);

View File

@@ -0,0 +1,67 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
return array(
'ImportEntity' => array(
'fields' => array(
'id' => array(
'type' => 'id',
'dbType' => 'int',
'len' => '11',
'autoincrement' => true,
'unique' => true
),
'entityId' => array(
'type' => 'varchar',
'len' => '24',
'index' => 'entity'
),
'entityType' => array(
'type' => 'varchar',
'len' => '100',
'index' => 'entity'
),
'importId' => array(
'type' => 'varchar',
'len' => '24',
'index' => true
),
'isImported' => array(
'type' => 'bool'
),
'isUpdated' => array(
'type' => 'bool'
),
'isDuplicate' => array(
'type' => 'bool'
),
),
"indexes" => array(
"entityImport" => array(
"columns" => ["importId", "entityType"]
)
)
),
);

View File

@@ -25,42 +25,56 @@ namespace Espo\Core\Utils;
class DateTime
{
protected $dataFormat;
protected $timeFormat;
protected $timezone;
protected $internalDateTimeFormat = 'Y-m-d H:i:s';
protected $internalDateFormat = 'Y-m-d';
protected $dateFormats = array(
'MM/DD/YYYY' => 'm/d/Y',
'YYYY-MM-DD' => 'Y-m-d',
'DD.MM.YYYY' => 'd.m.Y',
);
protected $timeFormats = array(
'HH:mm' => 'H:i',
'hh:mm A' => 'h:i A',
'hh:mm a' => 'h:ia',
'hh:mmA' => 'h:iA',
);
public function __construct($dateFormat = 'YYYY-MM-DD', $timeFormat = 'HH:mm', $timeZone = 'UTC')
{
$this->dateFormat = $dateFormat;
$this->timeFormat = $timeFormat;
$this->timezone = new \DateTimeZone($timeZone);
}
public function getInternalDateTimeFormat()
{
return $this->internalDateTimeFormat;
}
public function getInternalDateFormat()
{
return $this->internalDateFormat;
}
protected function getPhpDateFormat()
{
return $this->dateFormats[$this->dateFormat];
}
protected function getPhpDateTimeFormat()
{
return $this->dateFormats[$this->dateFormat] . ' ' . $this->timeFormats[$this->timeFormat];
}
public function convertSystemDateToGlobal($string)
{
$dateTime = \DateTime::createFromFormat('Y-m-d', $string);
@@ -69,7 +83,7 @@ class DateTime
}
return null;
}
public function convertSystemDateTimeToGlobal($string)
{
$dateTime = \DateTime::createFromFormat('Y-m-d H:i:s', $string);

View File

@@ -0,0 +1,421 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Utils;
use \Espo\Core\Exceptions\Error;
use \Espo\Core\Exceptions\Forbidden;
use \Espo\Core\Exceptions\Conflict;
use \Espo\Core\Utils\Json;
class EntityManager
{
private $metadata;
private $language;
private $fileManager;
private $metadataUtils;
public function __construct(Metadata $metadata, Language $language, File\Manager $fileManager)
{
$this->metadata = $metadata;
$this->language = $language;
$this->fileManager = $fileManager;
$this->metadataUtils = new \Espo\Core\Utils\Metadata\Utils($this->metadata);
}
protected function getMetadata()
{
return $this->metadata;
}
protected function getLanguage()
{
return $this->language;
}
protected function getFileManager()
{
return $this->fileManager;
}
protected function getMetadataUtils()
{
return $this->metadataUtils;
}
public function create($name, $type, $params = array())
{
if ($this->getMetadata()->get('scopes.' . $name)) {
throw new Conflict('Entity ['.$name.'] already exists.');
}
if (empty($name) || empty($type)) {
throw new Error();
}
$normalizedName = Util::normilizeClassName($name);
$contents = "<" . "?" . "php\n".
"namespace Espo\Custom\Entities;\n".
"class {$normalizedName} extends \Espo\Core\Templates\Entities\\{$type}\n".
"{\n".
" protected \$entityType = \"$name\";\n".
"}\n";
$filePath = "custom/Espo/Custom/Entities/{$normalizedName}.php";
$this->getFileManager()->putContents($filePath, $contents);
$contents = "<" . "?" . "php\n".
"namespace Espo\Custom\Controllers;\n".
"class {$normalizedName} extends \Espo\Core\Templates\Controllers\\{$type}\n".
"{\n".
"}\n";
$filePath = "custom/Espo/Custom/Controllers/{$normalizedName}.php";
$this->getFileManager()->putContents($filePath, $contents);
$contents = "<" . "?" . "php\n".
"namespace Espo\Custom\Services;\n".
"class {$normalizedName} extends \Espo\Core\Templates\Services\\{$type}\n".
"{\n".
"}\n";
$filePath = "custom/Espo/Custom/Services/{$normalizedName}.php";
$this->getFileManager()->putContents($filePath, $contents);
$contents = "<" . "?" . "php\n".
"namespace Espo\Custom\Repositories;\n".
"class {$normalizedName} extends \Espo\Core\Templates\Repositories\\{$type}\n".
"{\n".
"}\n";
$filePath = "custom/Espo/Custom/Repositories/{$normalizedName}.php";
$this->getFileManager()->putContents($filePath, $contents);
$stream = false;
if (!empty($params['stream'])) {
$stream = $params['stream'];
}
$labelSingular = $name;
if (!empty($params['labelSingular'])) {
$labelSingular = $params['labelSingular'];
}
$labelPlural = $name;
if (!empty($params['labelPlural'])) {
$labelPlural = $params['labelPlural'];
}
$labelCreate = $this->getLanguage()->translate('Create') . ' ' . $labelSingular;
$scopeData = array(
'entity' => true,
'layouts' => true,
'tab' => true,
'acl' => true,
'module' => 'Custom',
'isCustom' => true,
'customizable' => true,
'importable' => true,
'type' => $type,
'stream' => $stream,
'notifications' => true
);
$this->getMetadata()->set('scopes', $name, $scopeData);
$filePath = "application/Espo/Core/Templates/Metadata/{$type}/entityDefs.json";
$entityDefsData = Json::decode($this->getFileManager()->getContents($filePath), true);
$this->getMetadata()->set('entityDefs', $name, $entityDefsData);
$filePath = "application/Espo/Core/Templates/Metadata/{$type}/clientDefs.json";
$clientDefsData = Json::decode($this->getFileManager()->getContents($filePath), true);
$this->getMetadata()->set('clientDefs', $name, $clientDefsData);
$this->getLanguage()->set('Global', 'scopeNames', $name, $labelSingular);
$this->getLanguage()->set('Global', 'scopeNamesPlural', $name, $labelPlural);
$this->getLanguage()->set($name, 'labels', 'Create ' . $name, $labelCreate);
$this->getMetadata()->save();
$this->getLanguage()->save();
return true;
}
public function update($name, $data)
{
if (!$this->getMetadata()->get('scopes.' . $name)) {
throw new Error('Entity ['.$name.'] does not exist.');
}
if (isset($data['stream'])) {
$scopeData = array(
'stream' => (true == $data['stream'])
);
$this->getMetadata()->set('scopes', $name, $scopeData);
}
if (!empty($data['labelSingular'])) {
$labelSingular = $data['labelSingular'];
$this->getLanguage()->set('Global', 'scopeNames', $name, $labelSingular);
$labelCreate = $this->getLanguage()->translate('Create') . ' ' . $labelSingular;
$this->getLanguage()->set($name, 'labels', 'Create ' . $name, $labelCreate);
}
if (!empty($data['labelPlural'])) {
$labelPlural = $data['labelPlural'];
$this->getLanguage()->set('Global', 'scopeNamesPlural', $name, $labelPlural);
}
$this->getMetadata()->save();
$this->getLanguage()->save();
return true;
}
public function delete($name)
{
if (!$this->isCustom($name)) {
throw new Forbidden;
}
$normalizedName = Util::normilizeClassName($name);
$unsets = array(
'entityDefs',
'clientDefs',
'scopes'
);
$res = $this->getMetadata()->delete('entityDefs', $name);
$res = $this->getMetadata()->delete('clientDefs', $name);
$res = $this->getMetadata()->delete('scopes', $name);
$this->getFileManager()->removeFile("custom/Espo/Custom/Resources/metadata/entityDefs/{$name}.json");
$this->getFileManager()->removeFile("custom/Espo/Custom/Resources/metadata/clientDefs/{$name}.json");
$this->getFileManager()->removeFile("custom/Espo/Custom/Resources/metadata/scopes/{$name}.json");
$this->getFileManager()->removeFile("custom/Espo/Custom/Entities/{$normalizedName}.php");
$this->getFileManager()->removeFile("custom/Espo/Custom/Services/{$normalizedName}.php");
$this->getFileManager()->removeFile("custom/Espo/Custom/Controllers/{$normalizedName}.php");
$this->getFileManager()->removeFile("custom/Espo/Custom/Repositories/{$normalizedName}.php");
try {
$this->getLanguage()->delete('Global', 'scopeNames', $name);
$this->getLanguage()->delete('Global', 'scopeNamesPlural', $name);
} catch (\Exception $e) {}
$this->getMetadata()->save();
$this->getLanguage()->save();
return true;
}
protected function isCustom($name)
{
return $this->getMetadata()->get('scopes.' . $name . '.isCustom');
}
public function createLink(array $params)
{
$linkType = $params['linkType'];
$entity = $params['entity'];
$link = $params['link'];
$entityForeign = $params['entityForeign'];
$linkForeign = $params['linkForeign'];
$label = $params['label'];
$labelForeign = $params['labelForeign'];
if (empty($linkType)) {
throw new Error();
}
if (empty($entity) || empty($entityForeign)) {
throw new Error();
}
if (empty($entityForeign) || empty($linkForeign)) {
throw new Error();
}
if ($this->getMetadata()->get('entityDefs.' . $entity . '.links.' . $link)) {
throw new Conflict('Link ['.$entity.'::'.$link.'] already exists.');
}
if ($this->getMetadata()->get('entityDefs.' . $entityForeign . '.links.' . $linkForeign)) {
throw new Conflict('Link ['.$entityForeign.'::'.$linkForeign.'] already exists.');
}
switch ($linkType) {
case 'oneToMany':
if ($this->getMetadata()->get('entityDefs.' . $entityForeign . '.field.' . $linkForeign)) {
throw new Conflict('Field ['.$entityForeign.'::'.$linkForeign.'] already exists.');
}
$dataLeft = array(
'links' => array(
$link => array(
'type' => 'hasMany',
'foreign' => $linkForeign,
'entity' => $entityForeign,
'isCustom' => true
)
)
);
$dataRight = array(
'fields' => array(
$linkForeign => array(
'type' => 'link'
)
),
'links' => array(
$linkForeign => array(
'type' => 'belongsTo',
'foreign' => $link,
'entity' => $entity,
'isCustom' => true
)
)
);
break;
case 'manyToOne':
if ($this->getMetadata()->get('entityDefs.' . $entity . '.field.' . $link)) {
throw new Conflict('Field ['.$entity.'::'.$link.'] already exists.');
}
$dataLeft = array(
'fields' => array(
$link => array(
'type' => 'link'
)
),
'links' => array(
$link => array(
'type' => 'belongsTo',
'foreign' => $linkForeign,
'entity' => $entityForeign,
'isCustom' => true
)
)
);
$dataRight = array(
'links' => array(
$linkForeign => array(
'type' => 'hasMany',
'foreign' => $link,
'entity' => $entity,
'isCustom' => true
)
)
);
break;
case 'manyToMany':
$dataLeft = array(
'links' => array(
$link => array(
'type' => 'hasMany',
'foreign' => $linkForeign,
'entity' => $entityForeign,
'isCustom' => true
)
)
);
$dataRight = array(
'links' => array(
$linkForeign => array(
'type' => 'hasMany',
'foreign' => $link,
'entity' => $entity,
'isCustom' => true
)
)
);
break;
}
$this->getMetadata()->set('entityDefs', $entity, $dataLeft);
$this->getMetadata()->set('entityDefs', $entityForeign, $dataRight);
$this->getMetadata()->save();
$this->getLanguage()->set($entity, 'fields', $link, $label);
$this->getLanguage()->set($entity, 'links', $link, $label);
$this->getLanguage()->set($entityForeign, 'fields', $linkForeign, $labelForeign);
$this->getLanguage()->set($entityForeign, 'links', $linkForeign, $labelForeign);
$this->getLanguage()->save();
return true;
}
public function updateLink(array $params)
{
$entity = $params['entity'];
$link = $params['link'];
$entityForeign = $params['entityForeign'];
$linkForeign = $params['linkForeign'];
$label = $params['label'];
$labelForeign = $params['labelForeign'];
if (empty($entity) || empty($entityForeign)) {
throw new Error();
}
if (empty($entityForeign) || empty($linkForeign)) {
throw new Error();
}
$this->getLanguage()->set($entity, 'fields', $link, $label);
$this->getLanguage()->set($entity, 'links', $link, $label);
$this->getLanguage()->set($entityForeign, 'fields', $linkForeign, $labelForeign);
$this->getLanguage()->set($entityForeign, 'links', $linkForeign, $labelForeign);
$this->getLanguage()->save();
return true;
}
public function deleteLink(array $params)
{
$entity = $params['entity'];
$link = $params['link'];
if (!$this->getMetadata()->get("entityDefs.{$entity}.links.{$link}.isCustom")) {
throw new Error();
}
$entityForeign = $this->getMetadata()->get("entityDefs.{$entity}.links.{$link}.entity");
$linkForeign = $this->getMetadata()->get("entityDefs.{$entity}.links.{$link}.foreign");
if (empty($entity) || empty($entityForeign)) {
throw new Error();
}
if (empty($entityForeign) || empty($linkForeign)) {
throw new Error();
}
$this->getMetadata()->delete('entityDefs', $entity, array(
'fields.' . $link,
'links.' . $link
));
$this->getMetadata()->delete('entityDefs', $entityForeign, array(
'fields.' . $linkForeign,
'links.' . $linkForeign
));
$this->getMetadata()->save();
return true;
}
}

View File

@@ -39,7 +39,6 @@ class FieldManager
protected $customOptionName = 'isCustom';
public function __construct(Metadata $metadata, Language $language)
{
$this->metadata = $metadata;
@@ -122,7 +121,8 @@ class FieldManager
'links.'.$name,
);
$res = $this->getMetadata()->delete($unsets, $this->metadataType, $scope);
$this->getMetadata()->delete($this->metadataType, $scope, $unsets);
$res = $this->getMetadata()->save();
$res &= $this->deleteLabel($name, $scope);
return (bool) $res;
@@ -132,25 +132,26 @@ class FieldManager
{
$fieldDef = $this->normalizeDefs($name, $fieldDef, $scope);
$data = Json::encode($fieldDef);
$res = $this->getMetadata()->set($data, $this->metadataType, $scope);
$this->getMetadata()->set($this->metadataType, $scope, $fieldDef);
$res = $this->getMetadata()->save();
return $res;
}
protected function setTranslatedOptions($name, $value, $scope)
{
return $this->getLanguage()->set($name, $value, 'options', $scope);
return $this->getLanguage()->set($scope, 'options', $name, $value);
}
protected function setLabel($name, $value, $scope)
{
return $this->getLanguage()->set($name, $value, 'fields', $scope);
return $this->getLanguage()->set($scope, 'fields', $name, $value);
}
protected function deleteLabel($name, $scope)
{
$this->getLanguage()->delete($name, 'fields', $scope);
$this->getLanguage()->delete($scope, 'fields', $name);
$this->getLanguage()->delete($scope, 'options', $name);
return $this->getLanguage()->save();
}
@@ -177,6 +178,7 @@ class FieldManager
$unnecessaryFields = array(
'name',
'label',
'translatedOptions',
);
foreach ($unnecessaryFields as $fieldName) {

View File

@@ -60,7 +60,7 @@ class ClassParser
return $this->metadata;
}
public function setAllowedMethods(array $methods)
public function setAllowedMethods($methods)
{
$this->allowedMethods = $methods;
}
@@ -87,7 +87,7 @@ class ClassParser
}
if ($cacheFile && file_exists($cacheFile) && $this->getConfig()->get('useCache')) {
$data = $this->getFileManager()->getContents($cacheFile);
$data = $this->getFileManager()->getPhpContents($cacheFile);
} else {
$data = $this->getClassNameHash($paths['corePath']);
@@ -104,7 +104,7 @@ class ClassParser
}
if ($cacheFile && $this->getConfig()->get('useCache')) {
$result = $this->getFileManager()->putContentsPHP($cacheFile, $data);
$result = $this->getFileManager()->putPhpContents($cacheFile, $data);
if ($result == false) {
throw new \Espo\Core\Exceptions\Error();
}
@@ -129,11 +129,18 @@ class ClassParser
$filePath = Util::concatPath($dir, $file);
$className = Util::getClassName($filePath);
$fileName = $this->getFileManager()->getFileName($filePath);
$fileName = ucfirst($fileName);
$scopeName = ucfirst($fileName);
$normalizedScopeName = Util::normilizeScopeName($scopeName);
if (empty($this->allowedMethods)) {
$data[$normalizedScopeName] = $className;
continue;
}
foreach ($this->allowedMethods as $methodName) {
if (method_exists($className, $methodName)) {
$data[$fileName] = $className;
$data[$normalizedScopeName] = $className;
}
}

View File

@@ -29,6 +29,8 @@ class Manager
{
private $permission;
private $permissionDeniedList = array();
public function __construct(\Espo\Core\Utils\Config $config = null)
{
$params = null;
@@ -60,12 +62,14 @@ class Manager
*/
public function getFileList($path, $recursively = false, $filter = '', $onlyFileType = null, $isReturnSingleArray = false)
{
if (!file_exists($path)) {
return false;
}
$path = $this->concatPaths($path);
$result = array();
if (!file_exists($path) || !is_dir($path)) {
return $result;
}
$cdir = scandir($path);
foreach ($cdir as $key => $value)
{
@@ -156,17 +160,31 @@ class Manager
$fullPath = $this->concatPaths($path);
if (file_exists($fullPath)) {
if (strtolower(substr($fullPath, -4))=='.php') {
return include($fullPath);
if (isset($maxlen)) {
return file_get_contents($fullPath, $useIncludePath, $context, $offset, $maxlen);
} else {
if (isset($maxlen)) {
return file_get_contents($fullPath, $useIncludePath, $context, $offset, $maxlen);
} else {
return file_get_contents($fullPath, $useIncludePath, $context, $offset);
}
return file_get_contents($fullPath, $useIncludePath, $context, $offset);
}
}
return false;
}
/**
* Get PHP array from PHP file
*
* @param string | array $path
* @return array | bool
*/
public function getPhpContents($path)
{
$fullPath = $this->concatPaths($path);
if (file_exists($fullPath) && strtolower(substr($fullPath, -4)) == '.php') {
$phpContents = include($fullPath);
if (is_array($phpContents)) {
return $phpContents;
}
}
return false;
@@ -187,7 +205,7 @@ class Manager
$fullPath = $this->concatPaths($path); //todo remove after changing the params
if ($this->checkCreateFile($fullPath) === false) {
throw new Error('Permission denied in '. $fullPath);
throw new Error('Permission denied for '. $fullPath);
}
$res = (file_put_contents($fullPath, $data, $flags, $context) !== FALSE);
@@ -206,7 +224,7 @@ class Manager
*
* @return bool
*/
public function putContentsPHP($path, $data)
public function putPhpContents($path, $data)
{
return $this->putContents($path, $this->getPHPFormat($data), LOCK_EX);
}
@@ -235,19 +253,23 @@ class Manager
*
* @param string | array $path
* @param string $content JSON string
* @param bool $isJSON
* @param bool $isReturnJson
* @param string | array $removeOptions - List of unset keys from content
* @param bool $isReturn - Is result to be returned or stored
* @param bool $isPhp - Is merge php files
*
* @return bool | array
*/
public function mergeContents($path, $content, $isJSON = false, $removeOptions = null, $isReturn = false)
public function mergeContents($path, $content, $isReturnJson = false, $removeOptions = null, $isPhp = false)
{
$fileContent = $this->getContents($path);
if ($isPhp) {
$fileContent = $this->getPhpContents($path);
} else {
$fileContent = $this->getContents($path);
}
$fullPath = $this->concatPaths($path);
if (file_exists($fullPath) && ($fileContent === false || empty($fileContent))) {
throw new Error('Failed to read file [' . $fullPath .'].');
throw new Error('FileManager: Failed to read file [' . $fullPath .'].');
}
$savedDataArray = Utils\Json::getArrayData($fileContent);
@@ -259,12 +281,13 @@ class Manager
}
$data = Utils\Util::merge($savedDataArray, $newDataArray);
if ($isJSON) {
if ($isReturnJson) {
$data = Utils\Json::encode($data, JSON_PRETTY_PRINT);
}
if ($isReturn) {
return $data;
if ($isPhp) {
return $this->putPhpContents($path, $data);
}
return $this->putContents($path, $data);
@@ -278,11 +301,9 @@ class Manager
* @param string | array $removeOptions - List of unset keys from content
* @return bool
*/
public function mergeContentsPHP($path, $content, $removeOptions = null)
public function mergePhpContents($path, $content, $removeOptions = null)
{
$data = $this->mergeContents($path, $content, false, $removeOptions, true);
return $this->putContentsPHP($path, $data);
return $this->mergeContents($path, $content, false, $removeOptions, true);
}
/**
@@ -302,7 +323,7 @@ class Manager
* Unset some element of content data
*
* @param string | array $path
* @param array | string $unsets [description]
* @param array | string $unsets
* @return bool
*/
public function unsetContents($path, $unsets, $isJSON = true)
@@ -315,7 +336,12 @@ class Manager
$currentDataArray = Utils\Json::getArrayData($currentData);
$unsettedData = Utils\Util::unsetInArray($currentDataArray, $unsets);
$unsettedData = Utils\Util::unsetInArray($currentDataArray, $unsets, true);
if (is_null($unsettedData) || (is_array($unsettedData) && empty($unsettedData))) {
$fullPath = $this->concatPaths($path);
return $this->unlink($fullPath);
}
if ($isJSON) {
return $this->putContentsJson($path, $unsettedData);
@@ -349,9 +375,11 @@ class Manager
*
* @param string | array $path
* @param int $permission - ex. 0755
* @param bool $recursive
*
* @return bool
*/
public function mkdir($path, $permission = null)
public function mkdir($path, $permission = null, $recursive = false)
{
$fullPath = $this->concatPaths($path);
@@ -359,14 +387,23 @@ class Manager
return true;
}
$defaultPermissions = $this->getPermissionUtils()->getDefaultPermissions();
if (!isset($permission)) {
$defaultPermissions = $this->getPermissionUtils()->getDefaultPermissions();
$permission = (string) $defaultPermissions['dir'];
$permission = base_convert($permission, 8, 10);
}
try {
$result = mkdir($fullPath, $permission, true);
if (!empty($defaultPermissions['user'])) {
$this->getPermissionUtils()->chown($fullPath);
}
if (!empty($defaultPermissions['group'])) {
$this->getPermissionUtils()->chgrp($fullPath);
}
} catch (\Exception $e) {
$GLOBALS['log']->critical('Permission denied: unable to create the folder on the server - '.$fullPath);
}
@@ -422,7 +459,7 @@ class Manager
if (!empty($permissionDeniedList)) {
$betterPermissionList = $this->getPermissionUtils()->arrangePermissionList($permissionDeniedList);
throw new Error("Permission denied in <br>". implode(", <br>", $betterPermissionList));
throw new Error("Permission denied for <br>". implode(", <br>", $betterPermissionList));
}
$res = true;
@@ -449,7 +486,7 @@ class Manager
* @param string $filePath
* @return string
*/
protected function checkCreateFile($filePath)
public function checkCreateFile($filePath)
{
$defaultPermissions = $this->getPermissionUtils()->getDefaultPermissions();
@@ -466,8 +503,8 @@ class Manager
$dirPermission = $defaultPermissions['dir'];
$dirPermission = is_string($dirPermission) ? base_convert($dirPermission,8,10) : $dirPermission;
if (!mkdir($pathParts['dirname'], $dirPermission, true)) {
throw new Error('Permission denied: unable to create a folder on the server - '.$pathParts['dirname']);
if (!$this->mkdir($pathParts['dirname'], $dirPermission, true)) {
throw new Error('Permission denied: unable to create a folder on the server - ' . $pathParts['dirname']);
}
}
@@ -576,6 +613,24 @@ class Manager
$items = (array) $items;
}
$permissionDeniedList = array();
foreach ($items as $item) {
if (isset($dirPath)) {
$item = Utils\Util::concatPath($dirPath, $item);
}
if (!is_writable($item)) {
$permissionDeniedList[] = $item;
} else if (!is_writable(dirname($item))) {
$permissionDeniedList[] = dirname($item);
}
}
if (!empty($permissionDeniedList)) {
$betterPermissionList = $this->getPermissionUtils()->arrangePermissionList($permissionDeniedList);
throw new Error("Permission denied for <br>". implode(", <br>", $betterPermissionList));
}
$result = true;
foreach ($items as $item) {
if (isset($dirPath)) {
@@ -734,5 +789,73 @@ return '.var_export($content, true).';
?>';
}
/**
* Check if $paths are writable. Permission denied list are defined in getLastPermissionDeniedList()
*
* @param array $paths
*
* @return boolean
*/
public function isWritableList(array $paths)
{
$permissionDeniedList = array();
$result = true;
foreach ($paths as $path) {
$rowResult = $this->isWritable($path);
if (!$rowResult) {
$permissionDeniedList[] = $path;
}
$result &= $rowResult;
}
if (!empty($permissionDeniedList)) {
$this->permissionDeniedList = $this->getPermissionUtils()->arrangePermissionList($permissionDeniedList);
}
return (bool) $result;
}
/**
* Get last permission denied list
*
* @return array
*/
public function getLastPermissionDeniedList()
{
return $this->permissionDeniedList;
}
/**
* Check if $path is writable
*
* @param string | array $path
*
* @return boolean
*/
public function isWritable($path)
{
$existFile = $this->getExistsPath($path);
return is_writable($existFile);
}
/**
* Get exists path. Ex. if check /var/www/espocrm/custom/someFile.php and this file doesn't extist, result will be /var/www/espocrm/custom
*
* @param string | array $path
*
* @return string
*/
protected function getExistsPath($path)
{
$fullPath = $this->concatPaths($path);
if (!file_exists($fullPath)) {
$fullPath = $this->getExistsPath(pathinfo($fullPath, PATHINFO_DIRNAME));
}
return $fullPath;
}
}

View File

@@ -92,7 +92,6 @@ class Permission
return $this->params;
}
/**
* Get default settings
*
@@ -137,7 +136,6 @@ class Permission
return $result;
}
/**
* Get current permissions
*
@@ -216,7 +214,6 @@ class Permission
return $this->chmodRecurse($path, $permission['file'], $permission['dir']);
}
/**
* Change permissions recursive
*
@@ -232,27 +229,20 @@ class Permission
return false;
}
if (is_file($path)) {
if (!is_dir($path)) {
return $this->chmodReal($path, $fileOctal);
}
if (is_dir($path)) {
$allFiles = $this->getFileManager()->getFileList($path);
$result = $this->chmodReal($path, $dirOctal);
foreach ($items as $item) {
$this->chmodRecurse($path. Utils\Util::getSeparator() .$item, $fileOctal, $dirOctal);
}
return $this->chmodReal($path, $dirOctal);
$allFiles = $this->getFileManager()->getFileList($path);
foreach ($allFiles as $item) {
$result &= $this->chmodRecurse($path . Utils\Util::getSeparator() . $item, $fileOctal, $dirOctal);
}
return false;
return (bool) $result;
}
/**
* Change owner permission
*
@@ -262,7 +252,7 @@ class Permission
*
* @return bool
*/
public function chown($path, $user='', $recurse=false)
public function chown($path, $user = '', $recurse = false)
{
if (!file_exists($path)) {
return false;
@@ -295,13 +285,18 @@ class Permission
return false;
}
$allFiles = $this->getFileManager()->getFileList($path);
foreach ($items as $item) {
$this->chownRecurse($path. Utils\Util::getSeparator() .$item, $user);
if (!is_dir($path)) {
return $this->chownReal($path, $user);
}
return $this->chownReal($path, $user);
$result = $this->chownReal($path, $user);
$allFiles = $this->getFileManager()->getFileList($path);
foreach ($allFiles as $item) {
$result &= $this->chownRecurse($path . Utils\Util::getSeparator() . $item, $user);
}
return (bool) $result;
}
/**
@@ -347,15 +342,19 @@ class Permission
return false;
}
$allFiles = $this->getFileManager()->getFileList($path);
foreach ($items as $item) {
$this->chgrpRecurse($path. Utils\Util::getSeparator() .$item, $group);
if (!is_dir($path)) {
return $this->chgrpReal($path, $group);
}
return $this->chgrpReal($path, $group);
}
$result = $this->chgrpReal($path, $group);
$allFiles = $this->getFileManager()->getFileList($path);
foreach ($allFiles as $item) {
$result &= $this->chgrpRecurse($path . Utils\Util::getSeparator() . $item, $group);
}
return (bool) $result;
}
/**
* Change permissions recursive

View File

@@ -220,7 +220,6 @@ class Language
}
$this->clearChanges();
$this->init(true);
return (bool) $result;
}
@@ -234,6 +233,7 @@ class Language
{
$this->changedData = array();
$this->deletedData = array();
$this->init(true);
}
/**
@@ -253,53 +253,76 @@ class Language
/**
* Set/change a label
* @param string | array $label
* @param mixed $value
* @param string $category
*
* @param string $scope
* @param string $category
* @param string | array $name
* @param mixed $value
*
* @return void
*/
public function set($label, $value, $category = 'labels', $scope = 'Global')
public function set($scope, $category, $name, $value)
{
if (is_array($label)) {
foreach ($label as $rowLabel => $rowValue) {
$this->set($rowLabel, $rowValue, $category, $scope);
if (is_array($name)) {
foreach ($name as $rowLabel => $rowValue) {
$this->set($scope, $category, $rowLabel, $rowValue);
}
return;
}
$this->changedData[$scope][$category][$label] = $value;
$this->changedData[$scope][$category][$name] = $value;
$currentLanguage = $this->getLanguage();
$this->data[$currentLanguage][$scope][$category][$label] = $value;
if (!isset($this->data[$currentLanguage])) {
$this->init();
}
$this->data[$currentLanguage][$scope][$category][$name] = $value;
$this->undelete($scope, $category, $name);
}
/**
* Remove a label
*
* @param string $label
* @param string $name
* @param string $category
* @param string $scope
*
* @return void
*/
public function delete($label, $category = 'labels', $scope = 'Global')
public function delete($scope, $category, $name)
{
if (is_array($label)) {
foreach ($label as $rowLabel) {
$this->delete($rowLabel, $category, $scope);
if (is_array($name)) {
foreach ($name as $rowLabel) {
$this->delete($scope, $category, $rowLabel);
}
return;
}
$this->deletedData[$scope][$category][] = $label;
$this->deletedData[$scope][$category][] = $name;
$currentLanguage = $this->getLanguage();
if (isset($this->data[$currentLanguage][$scope][$category][$label])) {
unset($this->data[$currentLanguage][$scope][$category][$label]);
if (!isset($this->data[$currentLanguage])) {
$this->init();
}
if (isset($this->changedData[$scope][$category][$label])) {
unset($this->changedData[$scope][$category][$label]);
if (isset($this->data[$currentLanguage][$scope][$category][$name])) {
unset($this->data[$currentLanguage][$scope][$category][$name]);
}
if (isset($this->changedData[$scope][$category][$name])) {
unset($this->changedData[$scope][$category][$name]);
}
}
protected function undelete($scope, $category, $name)
{
if (isset($this->deletedData[$scope][$category])) {
foreach ($this->deletedData[$scope][$category] as $key => $labelName) {
if ($name === $labelName) {
unset($this->deletedData[$scope][$category][$key]);
}
}
}
}
@@ -319,7 +342,7 @@ class Language
if ($this->getConfig()->get('useCache')) {
$i18nCacheFile = str_replace('{*}', $i18nName, $this->cacheFile);
$result &= $this->getFileManager()->putContentsPHP($i18nCacheFile, $i18nData);
$result &= $this->getFileManager()->putPhpContents($i18nCacheFile, $i18nData);
}
}
@@ -330,7 +353,7 @@ class Language
$currentLanguage = $this->getLanguage();
if (empty($this->data[$currentLanguage])) {
$this->data[$currentLanguage] = $this->getFileManager()->getContents($this->getLangCacheFile());
$this->data[$currentLanguage] = $this->getFileManager()->getPhpContents($this->getLangCacheFile());
}
}
}

View File

@@ -18,23 +18,26 @@
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
************************************************************************/
namespace Espo\Core\Utils;
class Layout
{
private $fileManager;
private $metadata;
private $metadata;
private $changedData = array();
/**
* @var string - uses for loading default values
*/
private $name = 'layout';
private $name = 'layout';
protected $params = array(
protected $params = array(
'defaultsPath' => 'application/Espo/Core/defaults',
);
);
/**
@@ -42,10 +45,10 @@ class Layout
*/
private $paths = array(
'corePath' => 'application/Espo/Resources/layouts',
'modulePath' => 'application/Espo/Modules/{*}/Resources/layouts',
'customPath' => 'custom/Espo/Custom/Resources/layouts',
);
'modulePath' => 'application/Espo/Modules/{*}/Resources/layouts',
'customPath' => 'custom/Espo/Custom/Resources/layouts',
);
public function __construct(\Espo\Core\Utils\File\Manager $fileManager, \Espo\Core\Utils\Metadata $metadata)
{
@@ -63,7 +66,6 @@ class Layout
return $this->metadata;
}
/**
* Get Layout context
*
@@ -73,13 +75,17 @@ class Layout
* @return json
*/
public function get($controller, $name)
{
$fileFullPath = Util::concatPath($this->getLayoutPath($controller, true), $name.'.json');
if (!file_exists($fileFullPath)) {
$fileFullPath = Util::concatPath($this->getLayoutPath($controller), $name.'.json');
}
{
if (isset($this->changedData[$controller][$name])) {
return Json::encode($this->changedData[$controller][$name]);
}
if (!file_exists($fileFullPath)) {
$fileFullPath = Util::concatPath($this->getLayoutPath($controller, true), $name.'.json');
if (!file_exists($fileFullPath)) {
$fileFullPath = Util::concatPath($this->getLayoutPath($controller), $name.'.json');
}
if (!file_exists($fileFullPath)) {
//load defaults
$defaultPath = $this->params['defaultsPath'];
$fileFullPath = Util::concatPath( Util::concatPath($defaultPath, $this->name), $name.'.json' );
@@ -91,34 +97,68 @@ class Layout
}
return $this->getFileManager()->getContents($fileFullPath);
}
}
/**
* Set Layout data
* Ex. $controller= Account, $name= detail then will be created a file layoutFolder/Account/detail.json
* Ex. $controller = Account, $name = detail then will be created a file layoutFolder/Account/detail.json
*
* @param JSON string $data
* @param array $data
* @param string $controller - ex. Account
* @param string $name - detail
*
* @return bool
*
* @return void
*/
public function set($data, $controller, $name)
{
if (empty($controller) || empty($name)) {
return false;
}
$layoutPath = $this->getLayoutPath($controller, true);
if (!Json::isJSON($data)) {
$data = Json::encode($data);
}
return $this->getFileManager()->putContents(array($layoutPath, $name.'.json'), $data);
$this->changedData[$controller][$name] = $data;
}
/**
* Save changes
*
* @return bool
*/
public function save()
{
$result = true;
if (!empty($this->changedData)) {
foreach ($this->changedData as $controllerName => $rowData) {
foreach ($rowData as $layoutName => $layoutData) {
if (empty($controllerName) || empty($layoutName)) {
continue;
}
$layoutPath = $this->getLayoutPath($controllerName, true);
$data = Json::encode($layoutData);
$result &= $this->getFileManager()->putContents(array($layoutPath, $layoutName.'.json'), $data);
}
}
}
if ($result == true) {
$this->clearChanges();
}
return (bool) $result;
}
/**
* Clear unsaved changes
*
* @return void
*/
public function clearChanges()
{
$this->changedData = array();
}
/**
* Merge layout data
@@ -133,15 +173,15 @@ class Layout
public function merge($data, $controller, $name)
{
$prevData = $this->get($controller, $name);
$prevDataArray= Json::getArrayData($prevData);
$dataArray= Json::getArrayData($data);
$data= Util::merge($prevDataArray, $dataArray);
$data= Json::encode($data);
$prevDataArray = Json::getArrayData($prevData);
$dataArray = Json::getArrayData($data);
$data = Util::merge($prevDataArray, $dataArray);
$data = Json::encode($data);
return $this->set($data, $controller, $name);
}
}
/**
* Get Layout path, ex. application/Modules/Crm/Layouts/Account
@@ -152,23 +192,23 @@ class Layout
* @return string
*/
protected function getLayoutPath($entityName, $isCustom = false)
{
$path = $this->paths['customPath'];
{
$path = $this->paths['customPath'];
if (!$isCustom) {
$moduleName = $this->getMetadata()->getScopeModuleName($entityName);
$path = $this->paths['corePath'];
if ($moduleName !== false) {
$path = str_replace('{*}', $moduleName, $this->paths['modulePath']);
}
}
}
}
$path = Util::concatPath($path, $entityName);
return $path;
}
}
}

View File

@@ -51,9 +51,15 @@ class StreamHandler extends \Monolog\Handler\StreamHandler
$this->errorMessage = null;
set_error_handler(array($this, 'customErrorHandler'));
$this->getFileManager()->appendContents($this->url, $this->pruneMessage($record));
restore_error_handler();
if (!is_writable($this->url)) {
$this->getFileManager()->checkCreateFile($this->url);
}
if (is_writable($this->url)) {
set_error_handler(array($this, 'customErrorHandler'));
$this->getFileManager()->appendContents($this->url, $this->pruneMessage($record));
restore_error_handler();
}
if (isset($this->errorMessage)) {
throw new \UnexpectedValueException(sprintf('The stream or file "%s" could not be opened: '.$this->errorMessage, $this->url));

View File

@@ -41,6 +41,13 @@ class Metadata
*/
private $name = 'metadata';
/**
* Path to modules
*
* @var string
*/
private $pathToModules = 'application/Espo/Modules';
private $cacheFile = 'data/cache/application/metadata.php';
private $paths = array(
@@ -61,6 +68,10 @@ class Metadata
*/
protected $defaultModuleOrder = 10;
private $deletedData = array();
private $changedData = array();
public function __construct(\Espo\Core\Utils\Config $config, \Espo\Core\Utils\File\Manager $fileManager)
{
$this->config = $config;
@@ -124,13 +135,13 @@ class Metadata
}
if (file_exists($this->cacheFile) && !$reload) {
$this->meta = $this->getFileManager()->getContents($this->cacheFile);
$this->meta = $this->getFileManager()->getPhpContents($this->cacheFile);
} else {
$this->meta = $this->getUnifier()->unify($this->name, $this->paths, true);
$this->meta = $this->setLanguageFromConfig($this->meta);
if ($this->getConfig()->get('useCache')) {
$isSaved = $this->getFileManager()->putContentsPHP($this->cacheFile, $this->meta);
$isSaved = $this->getFileManager()->putPhpContents($this->cacheFile, $this->meta);
if ($isSaved === false) {
$GLOBALS['log']->emergency('Metadata:init() - metadata has not been saved to a cache file');
}
@@ -213,52 +224,139 @@ class Metadata
/**
* Set Metadata data
* Ex. $type= menu, $scope= Account then will be created a file metadataFolder/menu/Account.json
* Ex. $key1 = menu, $key2 = Account then will be created a file metadataFolder/menu/Account.json
*
* @param string $key1
* @param string $key2
* @param JSON string $data
* @param string $type - ex. menu
* @param string $scope - Account
*
* @return bool
*/
public function set($data, $type, $scope)
public function set($key1, $key2, $data)
{
$path = $this->paths['customPath'];
$newData = array(
$key1 => array(
$key2 => $data,
),
);
$result = $this->getFileManager()->mergeContents(array($path, $type, $scope.'.json'), $data, true);
if ($result === false) {
throw new Error("Error saving metadata. See log file for details.");
}
$this->changedData = Util::merge($this->changedData, $newData);
$this->meta = Util::merge($this->getData(), $newData);
$this->init(true);
return $result;
$this->undelete($key1, $key2, $data);
}
/**
* Unset some fields and other stuff in metadat
*
* @param string $key1
* @param string $key2
* @param array | string $unsets Ex. 'fields.name'
* @param string $type Ex. 'entityDefs'
* @param string $scope
*
* @return bool
*/
public function delete($unsets, $type, $scope)
public function delete($key1, $key2, $unsets)
{
if (!is_array($unsets)) {
$unsets = (array) $unsets;
}
$normalizedData = array(
'__APPEND__',
);
$metaUnsetData = array();
foreach ($unsets as $unsetItem) {
$normalizedData[] = $unsetItem;
$metaUnsetData[] = implode('.', array($key1, $key2, $unsetItem));
}
$unsetData = array(
$key1 => array(
$key2 => $normalizedData,
),
);
$this->deletedData = Util::merge($this->deletedData, $unsetData);
$this->deletedData = Util::unsetInArrayByValue('__APPEND__', $this->deletedData);
$this->meta = Util::unsetInArray($this->getData(), $metaUnsetData);
}
/**
* Undelete the deleted items
*
* @param string $key1
* @param string $key2
* @param array $data
* @return void
*/
protected function undelete($key1, $key2, $data)
{
if (isset($this->deletedData[$key1][$key2])) {
foreach ($this->deletedData[$key1][$key2] as $unsetIndex => $unsetItem) {
$value = Util::getValueByKey($data, $unsetItem);
if (isset($value)) {
unset($this->deletedData[$key1][$key2][$unsetIndex]);
}
}
}
}
/**
* Clear unsaved changes
*
* @return void
*/
public function clearChanges()
{
$this->changedData = array();
$this->deletedData = array();
$this->init(true);
}
/**
* Save changes
*
* @return bool
*/
public function save()
{
$path = $this->paths['customPath'];
$result = $this->getFileManager()->unsetContents(array($path, $type, $scope.'.json'), $unsets, true);
if ($result == false) {
$GLOBALS['log']->warning('Delete metadata items available only for custom code.');
$result = true;
if (!empty($this->changedData)) {
foreach ($this->changedData as $key1 => $keyData) {
foreach ($keyData as $key2 => $data) {
if (!empty($data)) {
$result &= $this->getFileManager()->mergeContents(array($path, $key1, $key2.'.json'), $data, true);
}
}
}
}
$this->init(true);
if (!empty($this->deletedData)) {
foreach ($this->deletedData as $key1 => $keyData) {
foreach ($keyData as $key2 => $unsetData) {
if (!empty($unsetData)) {
$rowResult = $this->getFileManager()->unsetContents(array($path, $key1, $key2.'.json'), $unsetData, true);
if ($rowResult == false) {
$GLOBALS['log']->warning('Metadata items ['.$key1.'.'.$key2.'] can be deleted for custom code only.');
}
$result &= $rowResult;
}
}
}
}
return $result;
if ($result == false) {
throw new Error("Error saving metadata. See log file for details.");
}
$this->clearChanges();
return (bool) $result;
}
public function getOrmMetadata($reload = false)
{
if (!empty($this->ormMeta) && is_array($this->ormMeta) && !$reload) {
@@ -270,7 +368,7 @@ class Metadata
}
if (empty($this->ormMeta)) {
$this->ormMeta = $this->getFileManager()->getContents($this->ormCacheFile);
$this->ormMeta = $this->getFileManager()->getPhpContents($this->ormCacheFile);
}
return $this->ormMeta;
@@ -278,8 +376,10 @@ class Metadata
public function setOrmMetadata(array $ormMeta)
{
$result = true;
if ($this->getConfig()->get('useCache')) {
$result = $this->getFileManager()->putContentsPHP($this->ormCacheFile, $ormMeta);
$result = $this->getFileManager()->putPhpContents($this->ormCacheFile, $ormMeta);
if ($result == false) {
throw new \Espo\Core\Exceptions\Error('Metadata::setOrmMetadata() - Cannot save ormMetadata to a file');
}
@@ -313,30 +413,27 @@ class Metadata
}
/**
* Get Scopes
* Load modules
*
* @return array
* @return void
*/
public function getScopes()
protected function loadModuleList()
{
if (!empty($this->scopes)) {
return $this->scopes;
}
$modules = $this->getFileManager()->getFileList($this->pathToModules, false, '', false);
$scopeList = $this->get('scopes');
if (!is_array($scopeList)) {
$this->init(true);
$scopeList = $this->get('scopes');
}
$scopes = array();
if (is_array($scopeList)) {
foreach ($scopeList as $name => $details) {
$scopes[$name] = isset($details['module']) ? $details['module'] : false;
$modulesToSort = array();
if (is_array($modules)) {
foreach ($modules as $moduleName) {
if (!empty($moduleName) && !isset($modulesToSort[$moduleName])) {
$modulesToSort[$moduleName] = $this->getModuleConfig()->get($moduleName . '.order', $this->defaultModuleOrder);
}
}
}
return $this->scopes = $scopes;
krsort($modulesToSort);
asort($modulesToSort);
$this->moduleList = array_keys($modulesToSort);
}
/**
@@ -346,24 +443,10 @@ class Metadata
*/
public function getModuleList()
{
if (!empty($this->moduleList)) {
return $this->moduleList;
if (!isset($this->moduleList)) {
$this->loadModuleList();
}
$scopes = $this->getScopes();
$modulesToSort = array();
foreach ($scopes as $moduleName) {
if (!empty($moduleName) && !isset($modulesToSort[$moduleName])) {
$modulesToSort[$moduleName] = $this->getModuleConfig()->get($moduleName . '.order', $this->defaultModuleOrder);
}
}
krsort($modulesToSort);
asort($modulesToSort);
$this->moduleList = array_keys($modulesToSort);
return $this->moduleList;
}
@@ -399,26 +482,4 @@ class Metadata
return $path;
}
/**
* Check if scope exists
*
* @param string $scopeName
*
* @return bool
*/
public function isScopeExists($scopeName)
{
$scopeModuleMap = $this->getScopes();
$lowerEntityName = strtolower($scopeName);
foreach($scopeModuleMap as $rowEntityName => $rowModuleName) {
if ($lowerEntityName == strtolower($rowEntityName)) {
return true;
}
}
return false;
}
}

View File

@@ -84,12 +84,12 @@ class Module
protected function init()
{
if (file_exists($this->cacheFile) && $this->getConfig()->get('useCache')) {
$this->data = $this->getFileManager()->getContents($this->cacheFile);
$this->data = $this->getFileManager()->getPhpContents($this->cacheFile);
} else {
$this->data = $this->getUnifier()->unify($this->paths, true);
if ($this->getConfig()->get('useCache')) {
$result = $this->getFileManager()->putContentsPHP($this->cacheFile, $this->data);
$result = $this->getFileManager()->putPhpContents($this->cacheFile, $this->data);
if ($result == false) {
throw new \Espo\Core\Exceptions\Error('Module - Cannot save unified modules.');
}

View File

@@ -95,12 +95,12 @@ class Route
protected function init()
{
if (file_exists($this->cacheFile) && $this->getConfig()->get('useCache')) {
$this->data = $this->getFileManager()->getContents($this->cacheFile);
$this->data = $this->getFileManager()->getPhpContents($this->cacheFile);
} else {
$this->data = $this->unify();
if ($this->getConfig()->get('useCache')) {
$result = $this->getFileManager()->putContentsPHP($this->cacheFile, $this->data);
$result = $this->getFileManager()->putPhpContents($this->cacheFile, $this->data);
if ($result == false) {
throw new \Espo\Core\Exceptions\Error('Route - Cannot save unified routes');
}
@@ -127,7 +127,7 @@ class Route
protected function getAddData($currData, $routeFile)
{
if (file_exists($routeFile)) {
$content= $this->getFileManager()->getContents($routeFile);
$content = $this->getFileManager()->getContents($routeFile);
$arrayContent = Json::getArrayData($content);
if (empty($arrayContent)) {
$GLOBALS['log']->error('Route::unify() - Empty file or syntax error - ['.$routeFile.']');

View File

@@ -20,14 +20,14 @@
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Core\Cron;
namespace Espo\Core\Utils;
use Espo\Core\Exceptions\NotFound,
Espo\Core\Utils\Util;
use Espo\Core\Exceptions\NotFound;
class ScheduledJob
{
private $container;
private $systemUtil;
protected $data = null;
@@ -54,7 +54,6 @@ class ScheduledJob
'default' => '* * * * * {PHP-BIN-DIR} -f {CRON-FILE}',
);
public function __construct(\Espo\Core\Container $container)
{
$this->container = $container;
@@ -76,19 +75,9 @@ class ScheduledJob
return $this->systemUtil;
}
public function run(array $job)
public function getMethodName()
{
$jobName = $job['method'];
$className = $this->getClassName($jobName);
if ($className === false) {
throw new NotFound();
}
$jobClass = new $className($this->container);
$method = $this->allowedMethod;
$jobClass->$method();
return $this->allowedMethod;
}
/**
@@ -161,7 +150,6 @@ class ScheduledJob
$this->data = $classParser->getData($this->paths, $this->cacheFile);
}
public function getSetupMessage()
{
$language = $this->getContainer()->get('language');

View File

@@ -101,4 +101,20 @@ class System
{
return (defined("PHP_BINDIR"))? PHP_BINDIR.DIRECTORY_SEPARATOR.'php' : 'php';
}
/**
* Get php version (only digits and dots)
*
* @return string
*/
public static function getPhpVersion()
{
$version = phpversion();
if (preg_match('/^[0-9\.]+[0-9]/', $version, $matches)) {
return $matches[0];
}
return $version;
}
}

View File

@@ -159,7 +159,7 @@ class Util
if ($appendKey !== false) {
unset($newValue[$appendKey]);
$newValue = array_merge($currentArray[$newName], $newValue);
} else if (!static::isSingleArray($newValue)) {
} else if (!static::isSingleArray($newValue) || !static::isSingleArray($currentArray[$newName])) {
$newValue = static::merge($currentArray[$newName], $newValue);
}
@@ -288,6 +288,23 @@ class Util
return $name;
}
/**
* Remove 'Obj' if name is reserved PHP word.
*
* @param string $name
* @return string
*/
public static function normilizeScopeName($name)
{
foreach (self::$reservedWords as $reservedWord) {
if ($reservedWord.'Obj' == $name) {
return $reservedWord;
}
}
return $name;
}
/**
* Get Naming according to prefix or postfix type
*
@@ -351,10 +368,11 @@ class Util
* array('EntityName1.unset1', 'EntityName1.unset2', .....)
* OR
* 'EntityName1.unset1'
* @param bool $unsetParentEmptyArray - If unset empty parent array after unsets
*
* @return array
*/
public static function unsetInArray(array $content, $unsets)
public static function unsetInArray(array $content, $unsets, $unsetParentEmptyArray = false)
{
if (empty($unsets)) {
return $content;
@@ -379,11 +397,19 @@ class Util
$unsetElem = $currVal . "['{$lastKey}']";
$currVal = "
$evalString = "
if (isset({$unsetElem}) || ( is_array({$currVal}) && array_key_exists('{$lastKey}', {$currVal}) )) {
unset({$unsetElem});
} ";
eval($currVal);
eval($evalString);
if ($unsetParentEmptyArray) {
$evalString = "
if (is_array({$currVal}) && empty({$currVal})) {
unset({$currVal});
} ";
eval($evalString);
}
}
}
}
@@ -484,6 +510,11 @@ class Util
return true;
}
public static function generateId()
{
return uniqid() . substr(md5(rand()), 0, 4);
}
}

View File

@@ -69,6 +69,7 @@ return array (
'ru_RU',
'pl_PL',
'pt_BR',
'uk_UA',
'vi_VN'
),
'language' => 'en_US',
@@ -82,18 +83,23 @@ return array (
'authenticationMethod' => 'Espo',
'globalSearchEntityList' =>
array (
0 => 'Account',
1 => 'Contact',
2 => 'Lead',
3 => 'Opportunity',
'Account',
'Contact',
'Lead',
'Opportunity',
),
"tabList" => array("Account", "Contact", "Lead", "Opportunity", "Calendar", "Meeting", "Call", "Task", "Case", "Email"),
"tabList" => array("Account", "Contact", "Lead", "Opportunity", "Calendar", "Meeting", "Call", "Task", "Case", "Email", "Document", "Campaign"),
"quickCreateList" => array("Account", "Contact", "Lead", "Opportunity", "Meeting", "Call", "Task", "Case"),
'calendarDefaultEntity' => 'Meeting',
'disableExport' => false,
'assignmentEmailNotifications' => false,
'assignmentEmailNotificationsEntityList' => array('Lead', 'Opportunity', 'Task', 'Case'),
'assignmentNotificationsEntityList' => array('Meeting', 'Call', 'Task', 'Email'),
'emailMessageMaxSize' => 10,
'notificationsCheckInterval' => 10,
'disabledCountQueryEntityList' => array('Email'),
'maxEmailAccountCount' => 2,
'followCreatedEntities' => false,
'isInstalled' => false,
);

View File

@@ -1,6 +1,6 @@
[{
"label":"Overview",
"rows": [
[{"name":"name"}]
[{"name":"name"}, false]
]
}]

View File

@@ -54,6 +54,7 @@ return array (
'maxJobNumber' => 15, /** Max number of jobs per one execution */
'jobPeriod' => 7800, /** Period for jobs, ex. if cron executed at 15:35, it will execute all pending jobs for times from 14:05 to 15:35 */
'minExecutionTime' => 50, /** to avoid too frequency execution **/
'attempts' => 3, /** attempts to run jobs **/
),
'crud' => array(
'get' => 'read',
@@ -83,6 +84,7 @@ return array (
'permissionMap',
'permissionRules',
'passwordSalt',
'cryptKey'
),
'adminItems' =>
array (
@@ -114,6 +116,7 @@ return array (
'ldapTryUsernameSplit',
'ldapOptReferrals',
'ldapCreateEspoUser',
'maxEmailAccountCount'
),
'isInstalled' => false,
);

View File

@@ -18,23 +18,22 @@
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
************************************************************************/
namespace Espo\Entities;
class Email extends \Espo\Core\ORM\Entity
{
protected function _getSubject()
{
return $this->get('name');
}
protected function _setSubject($value)
{
return $this->set('name', $value);
}
public function addAttachment(\Espo\Entities\Attachment $attachment)
{
if (!empty($this->id)) {
@@ -45,16 +44,16 @@ class Email extends \Espo\Core\ORM\Entity
}
}
}
public function getBodyPlain()
{
$bodyPlain = $this->get('bodyPlain');
$bodyPlain = $this->get('bodyPlain');
if (!empty($bodyPlain)) {
return $bodyPlain;
}
$body = $this->get('body');
$breaks = array("<br />","<br>","<br/>","<br />","&lt;br /&gt;","&lt;br/&gt;","&lt;br&gt;");
$body = str_ireplace($breaks, "\r\n", $body);
$body = strip_tags($body);
@@ -65,7 +64,7 @@ class Email extends \Espo\Core\ORM\Entity
{
return $this->getBodyPlain();
}
public function getBodyForSending()
{
$body = $this->get('body');
@@ -75,19 +74,19 @@ class Email extends \Espo\Core\ORM\Entity
$body = str_replace("?entryPoint=attachment&amp;id={$attachment->id}", "cid:{$attachment->id}", $body);
}
}
$body = str_replace("<table class=\"table table-bordered\">", "<table class=\"table table-bordered\" width=\"100%\">", $body);
return $body;
}
public function getInlineAttachments()
{
$attachmentList = array();
$body = $this->get('body');
if (!empty($body)) {
if (preg_match_all("/\?entryPoint=attachment&amp;id=([^&=\"']+)/", $body, $matches)) {
if (!empty($matches[1]) && is_array($matches[1])) {
if (preg_match_all("/\?entryPoint=attachment&amp;id=([^&=\"']+)/", $body, $matches)) {
if (!empty($matches[1]) && is_array($matches[1])) {
foreach($matches[1] as $id) {
$attachment = $this->entityManager->getEntity('Attachment', $id);
if ($attachment) {
@@ -96,7 +95,7 @@ class Email extends \Espo\Core\ORM\Entity
}
}
}
}
return $attachmentList;
}

View File

@@ -0,0 +1,28 @@
<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2015 Yuri Kuznetsov, Taras Machyshyn, Oleksiy Avramenko
* Website: http://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
namespace Espo\Entities;
class Import extends \Espo\Core\ORM\Entity
{
}

View File

@@ -18,14 +18,19 @@
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
************************************************************************/
************************************************************************/
namespace Espo\Entities;
class User extends \Espo\Core\Entities\Person
{
{
public function isAdmin()
{
return $this->get('isAdmin');
}
public function isActive()
{
return $this->get('isActive');
}
}

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