Compare commits

...

210 Commits
1.5.3 ... 2.4

Author SHA1 Message Date
Dennis Smink
62084f590e wip 2022-07-21 15:24:59 +02:00
Dennis Smink
c7edf262f6 Merge branch 'develop'
# Conflicts:
#	public/css/app.css
#	public/js/1042.js
#	public/js/1304.js
#	public/js/1438.js
#	public/js/1582.js
#	public/js/1973.js
#	public/js/2232.js
#	public/js/2306.js
#	public/js/2426.js
#	public/js/2658.js
#	public/js/2668.js
#	public/js/2693.js
#	public/js/2940.js
#	public/js/2999.js
#	public/js/3038.js
#	public/js/3292.js
#	public/js/3739.js
#	public/js/4144.js
#	public/js/43.js
#	public/js/4379.js
#	public/js/4485.js
#	public/js/4596.js
#	public/js/4695.js
#	public/js/47.js
#	public/js/4710.js
#	public/js/4766.js
#	public/js/4791.js
#	public/js/4804.js
#	public/js/4946.js
#	public/js/5641.js
#	public/js/565.js
#	public/js/5766.js
#	public/js/5865.js
#	public/js/6038.js
#	public/js/6282.js
#	public/js/6289.js
#	public/js/6340.js
#	public/js/6424.js
#	public/js/6511.js
#	public/js/6545.js
#	public/js/684.js
#	public/js/6861.js
#	public/js/701.js
#	public/js/7054.js
#	public/js/7210.js
#	public/js/7399.js
#	public/js/7611.js
#	public/js/7767.js
#	public/js/8027.js
#	public/js/8309.js
#	public/js/8371.js
#	public/js/8607.js
#	public/js/8786.js
#	public/js/883.js
#	public/js/908.js
#	public/js/9195.js
#	public/js/9281.js
#	public/js/9353.js
#	public/js/app.js
2022-07-21 15:24:23 +02:00
Ralph J. Smit
1ce6e8cace Fix test 2022-07-20 10:51:43 +02:00
Dennis Smink
57f783490b ww 2022-07-20 08:26:14 +02:00
Dennis Smink
eaab262629 Merge branch '61-ploi-core-design' into develop
# Conflicts:
#	public/js/app.js
2022-07-20 08:26:10 +02:00
Dennis Smink
e2e05f9cbf wip 2022-07-20 08:25:33 +02:00
Dennis Smink
24ce8bc60d Merge branch '26-duplicate-check' into develop
# Conflicts:
#	app/Http/Controllers/SiteController.php
#	package-lock.json
#	public/js/app.js
2022-07-20 08:17:59 +02:00
Dennis Smink
75592aaeb2 Do this check server sided 2022-07-20 08:16:42 +02:00
Dennis Smink
c11ad19220 w 2022-07-20 08:08:09 +02:00
Dennis Smink
db799a7d6a w 2022-07-20 08:06:53 +02:00
Dennis Smink
7fea371857 Merge branch '25-add-ssl-for-alias-domains-automatically' into develop
# Conflicts:
#	public/js/app.js
2022-07-20 08:06:48 +02:00
Dennis Smink
d7b3899e71 Few fixes 2022-07-20 08:06:37 +02:00
Dennis Smink
e2886fb67e Merge branch '131-set-up-github-action-for-ploi-core' into develop 2022-07-20 07:56:19 +02:00
Ralph J. Smit
57c8997dd0 Fix 2022-07-19 21:32:14 +02:00
Ralph J. Smit
2df031a60f Check whether domain already exists when creating site 2022-07-19 21:17:21 +02:00
Ralph J. Smit
77384a1abe Request new certificate automatically after creating new alias 2022-07-19 19:04:01 +02:00
Ralph J. Smit
7b20082537 Implement real usernames for site system users and database name site prefix 2022-07-19 15:58:50 +02:00
Ralph J. Smit
3531e4b296 Update master.yml 2022-07-19 15:21:03 +02:00
Ralph J. Smit
4e92501985 Update master.yml 2022-07-19 15:17:20 +02:00
GitHub Actions
d7632d8289 Run Laravel Mix en build front-end assets 2022-07-19 13:12:48 +00:00
Ralph J. Smit
f404b3e9c6 Update master.yml 2022-07-19 15:11:01 +02:00
Ralph J. Smit
730f9b7451 Update run-tests.yml 2022-07-19 15:09:40 +02:00
Ralph J. Smit
2fe5fd70c9 PHP 8.0 support 2022-07-19 15:07:38 +02:00
Ralph J. Smit
6afe8738df Consistent scripts 2022-07-19 14:50:28 +02:00
Ralph J. Smit
761a940abd Add support for PHP 8.0 back 2022-07-19 14:49:08 +02:00
Ralph J. Smit
f87c1dd5ee Update run-tests.yml 2022-07-19 14:45:06 +02:00
Ralph J. Smit
de70310c90 Update run-tests.yml 2022-07-19 14:43:50 +02:00
Ralph J. Smit
11f9b1ed48 Update run-tests.yml 2022-07-19 14:42:28 +02:00
Ralph J. Smit
060a6b72a7 Update master.yml 2022-07-19 14:42:00 +02:00
Ralph J. Smit
70cc81f110 Update run-tests.yml 2022-07-19 14:41:26 +02:00
Ralph J. Smit
bcc1a9b9a8 Update master.yml 2022-07-19 14:39:10 +02:00
Ralph J. Smit
1c601a6efd Update master.yml 2022-07-19 14:36:42 +02:00
Ralph J. Smit
fee31d03a7 Update master.yml 2022-07-19 14:35:14 +02:00
Ralph J. Smit
b09dc1ba9d Update run-tests.yml 2022-07-19 14:35:11 +02:00
Ralph J. Smit
dae15c620b Update run-tests.yml 2022-07-19 14:31:36 +02:00
Ralph J. Smit
996a048a76 Update phpunit.xml 2022-07-19 14:30:46 +02:00
Ralph J. Smit
8c20f23dfd Update Pest.php 2022-07-19 14:27:27 +02:00
Ralph J. Smit
c80818df4c Update phpunit.xml 2022-07-19 14:19:10 +02:00
Ralph J. Smit
d5e77ae31f Update master.yml 2022-07-19 13:34:03 +02:00
Ralph J. Smit
a14d2c44a1 wip 2022-07-19 13:33:24 +02:00
Ralph J. Smit
3048747ed6 WIP 2022-07-18 22:45:22 +02:00
Ralph J. Smit
20bf6c4784 Compatibility with Http-facade instead of Guzzle 2022-07-18 22:42:24 +02:00
Ralph J. Smit
9b02be5be1 Update SiteController.php 2022-07-18 22:42:04 +02:00
Ralph J. Smit
d141503b6f Remove unused class 2022-07-18 22:26:31 +02:00
Ralph J. Smit
6a8e4e8edf Move Http-tests to Tests/Feature 2022-07-18 22:20:42 +02:00
Ralph J. Smit
89bbf44b3b Finish API-endpoints & tests 2022-07-18 22:19:45 +02:00
Ralph J. Smit
db1f40bf6f WIP 2022-07-16 21:44:33 +02:00
Ralph J. Smit
5933a06dd3 WIP 2022-07-05 18:50:34 +02:00
Ralph J. Smit
3c510906ee General stuff 2022-07-01 21:31:35 +02:00
Ralph J. Smit
b43f4cf292 Server testing & endpoints, general 2022-07-01 21:31:27 +02:00
Ralph J. Smit
9cc046eeed Site testing & endpoints 2022-07-01 21:30:36 +02:00
Ralph J. Smit
8291ac6714 Test and update UserController endpoint 2022-07-01 21:17:53 +02:00
Ralph J. Smit
c578ee70c0 Prepare Pest 2022-07-01 21:17:22 +02:00
Ralph J. Smit
90501e37fd Prepare data objects (hopefully remove some code after PR accepted) 2022-07-01 21:15:59 +02:00
Ralph J. Smit
ec45b0dac0 Update RouteServiceProvider.php 2022-07-01 12:01:33 +02:00
Ralph J. Smit
34b838c259 Implement & test API-authentication, simplify Api-routes 2022-07-01 12:00:16 +02:00
Dennis
28ffc8e240 Merge branch '129-add-user-friendly-404-pages' into develop 2022-07-01 07:53:44 +02:00
Dennis
c9179fbf90 proper mix 2022-07-01 07:52:51 +02:00
Dennis
fe5268971a Merge branch '114-refresh-system-version-in-system-tab' into develop
# Conflicts:
#	public/js/app.js
2022-07-01 07:52:21 +02:00
Dennis
8c246e2dba wip tests 2022-07-01 07:51:27 +02:00
Dennis
81fcfac803 Merge branch '76-the-use-of-html-tags-or-markdown-in-alert-messages' into develop
# Conflicts:
#	public/js/app.js
2022-07-01 07:50:17 +02:00
Ralph J. Smit
ff22b96a8d Fix 404 and 403 Vue error pages 2022-06-30 18:47:11 +02:00
Ralph J. Smit
4a2faf0bce Build files 2022-06-30 18:18:34 +02:00
Ralph J. Smit
5c39d07bf5 Allow version refresh in System 2022-06-30 18:14:54 +02:00
Ralph J. Smit
9870aec79f Support markdown & line breaks in Alert messages 2022-06-30 16:11:18 +02:00
Dennis
833a03e992 dev 2022-06-30 16:02:11 +02:00
Ralph J. Smit
e074ab5be4 Implement two-factor authentication 2022-06-30 15:26:44 +02:00
Dennis
b5963693e6 Prod mix 2022-06-27 10:26:40 +02:00
Dennis
1b7ea67fde Merge branch 'develop'
# Conflicts:
#	public/css/app.css
#	public/js/app.js
2022-06-27 10:25:39 +02:00
Dennis
d4f2b9839e package updates 2022-06-27 10:25:10 +02:00
Dennis
817f6a175c wip 2022-06-27 10:23:57 +02:00
Dennis
b3619e5941 wip 2022-06-20 09:14:56 +02:00
Dennis
33784410e5 wip trial 2022-06-07 13:36:49 +02:00
Dennis
6ecf7904fe wip roadmap 2022-06-07 11:31:39 +02:00
Dennis
21986f2394 add ability to run octane 2022-06-07 11:00:56 +02:00
Dennis
4d8212e56f package updates and add aws for ses emailing 2022-06-07 10:58:38 +02:00
Dennis
865f2958cf Merge branch 'develop' 2022-05-23 16:43:39 +02:00
Dennis
17890d13ad Merge branch 'develop' of https://github.com/ploi-deploy/ploi-core into develop 2022-05-23 16:43:28 +02:00
Dennis
2d33455731 Closes #1 2022-05-23 16:43:15 +02:00
Dennis
49481f9b6a Prod mix 2022-04-14 12:39:01 +02:00
Dennis
7bb800cc0a Merge branch 'develop'
# Conflicts:
#	public/css/app.css
#	public/js/app.js
2022-04-14 12:38:06 +02:00
Dennis
1b8c2c764f fx 2022-04-14 12:37:39 +02:00
Dennis
cb1a1c4c06 w 2022-04-12 10:18:47 +02:00
Dennis
ddd80a8687 site email 2022-04-11 19:13:40 +02:00
Dennis
010d4569c2 package updates 2022-04-11 18:54:44 +02:00
Dennis
62ae0f8299 wip 2022-04-09 19:06:17 +02:00
Dennis
2a3d9cabd0 wip 2022-04-09 07:54:47 +02:00
Dennis
e2a58cf2df tweaks 2022-04-07 14:48:45 +02:00
Dennis
0fd6db251b remove these 2022-04-07 14:47:31 +02:00
Dennis
94d50c11ef wip 2022-04-07 14:46:56 +02:00
Dennis
14c6faafa2 wip 2022-04-07 14:21:57 +02:00
Dennis
ea21076eda wip 2022-04-07 11:15:06 +02:00
Dennis
d378323602 wip 2022-04-07 10:52:42 +02:00
Dennis
f074dee990 wip 2022-04-05 14:59:27 +02:00
Dennis
7bd2917ec4 wip 2022-04-01 10:50:05 +02:00
Dennis
823a39ffa2 prod mix 2022-03-23 09:58:27 +01:00
Dennis
6953a8d2b2 Merge branch 'develop'
# Conflicts:
#	public/css/app.css
#	public/js/app.js
2022-03-23 09:58:01 +01:00
Dennis
fb40b450b6 BRL 2022-03-23 09:54:11 +01:00
Dennis
a28c053945 wip 2022-03-23 09:40:29 +01:00
Dennis
ea0c4ed66d wip 2022-03-23 09:06:28 +01:00
Dennis
7a0716959a Laravel 9 upgrade 2022-03-01 11:21:58 +01:00
Dennis
da322d7b1e prod mix 2022-01-21 13:57:39 +01:00
Dennis
bbff8a5403 Merge branch 'develop'
# Conflicts:
#	public/js/app.js
2022-01-21 13:57:04 +01:00
Dennis
c1ad600042 Ability to select database type, package updates, bugfixes 2022-01-21 13:56:48 +01:00
Dennis
5fc32759fd prod mix 2022-01-19 14:29:36 +01:00
Dennis
59cfd8d71b Merge branch 'develop'
# Conflicts:
#	public/css/app.css
#	public/js/app.js
2022-01-19 14:28:00 +01:00
Dennis
51d5b89df7 wip 2022-01-19 14:26:08 +01:00
Dennis
7c0dcbeb88 TW3.0 upgrade 2021-12-24 16:30:41 +01:00
Dennis
b4467f8d5b Package updates 2021-12-21 09:04:54 +01:00
Dennis
363f4ed801 default to predis here 2021-12-21 08:58:25 +01:00
Dennis
1b6b950fb5 Bugfixing 2021-12-21 08:57:55 +01:00
Dennis
90988f1538 Prod mix 2021-12-19 21:58:01 +01:00
Dennis
139ba793d0 Merge branch 'develop'
# Conflicts:
#	public/js/app.js
2021-12-19 21:57:32 +01:00
Dennis
c505dd0924 PSR 2021-12-19 21:57:23 +01:00
Dennis
ca5ee33978 Bugfix in terms page, made system available in demo 2021-12-19 21:57:10 +01:00
Dennis
c2fd5e3fa9 Prod mix 2021-12-07 15:20:10 +01:00
Dennis
b04176ce48 Merge branch 'develop'
# Conflicts:
#	public/js/app.js
2021-12-07 15:19:25 +01:00
Dennis
5136a4b9f8 Bugfix 2021-12-07 15:18:35 +01:00
Dennis
dcee703aa1 wip 2021-11-30 13:30:21 +01:00
Dennis
7c504339d9 Added favicon if logo is uploaded 2021-11-30 13:18:10 +01:00
Dennis
5ea0761fe9 prod mix 2021-11-24 11:25:14 +01:00
Dennis
c9125c3be8 Merge branch 'develop'
# Conflicts:
#	public/js/app.js
2021-11-24 11:22:00 +01:00
Dennis
84503c19db Ability to revoke card & DNS fix on site show 2021-11-24 11:20:24 +01:00
Dennis
cfd9eba5d7 Ability to remove logo and pagination to system logs 2021-11-19 14:14:20 +01:00
Dennis
10689d3d12 Ability to rotate logs in system 2021-11-19 13:52:32 +01:00
Dennis
e190fb7805 prod mix 2021-11-02 11:44:32 +01:00
Dennis
c1351f7d28 Merge branch 'develop'
# Conflicts:
#	public/js/app.js
2021-11-02 11:43:24 +01:00
Dennis
01f5469e86 Generic fixes 2021-11-02 11:43:10 +01:00
Dennis
8c5c86eb6a Preserve scroll by default 2021-11-02 11:26:53 +01:00
Dennis
995ada46aa Prod mix 2021-09-27 09:51:05 +02:00
Dennis
2e79381872 Merge branch 'develop'
# Conflicts:
#	public/js/app.js
2021-09-27 09:50:24 +02:00
Dennis
d1c7b9a418 Fixes 2021-09-27 09:49:42 +02:00
Dennis
34da2f563d Ability to configure per page number 2021-09-23 14:22:01 +02:00
Dennis
5cf77fde1c Package updates 2021-09-23 08:28:54 +02:00
Dennis
aabf6f27ac Ran PSR formatter 2021-09-23 08:27:27 +02:00
Dennis
fbcaee3bdc prod mix 2021-09-23 08:26:39 +02:00
Dennis
3082c10cdb Merge branch 'develop'
# Conflicts:
#	public/js/app.js
2021-09-23 08:26:03 +02:00
Dennis
a4d90f0017 OG image updatee readme 2021-09-23 08:25:52 +02:00
Dennis
8adfc9837e fixes 2021-09-23 08:20:22 +02:00
Dennis
aa20c8a42b prod mix 2021-09-16 09:51:17 +02:00
Dennis
0f100d6159 Merge branch 'develop'
# Conflicts:
#	public/css/app.css
#	public/js/app.js
2021-09-16 09:50:07 +02:00
Dennis
7c896243a5 wip 2021-09-16 09:49:53 +02:00
Dennis
954fef7c3e Dynamic servers tab, fix for domain uppercase 2021-09-16 09:19:41 +02:00
Dennis
105126e498 prod mix 2021-08-25 09:20:18 +02:00
Dennis
7aa5a8949d Merge branch 'develop'
# Conflicts:
#	public/js/app.js
2021-08-25 09:19:45 +02:00
Dennis
27079928a7 added demo quick-login support 2021-08-25 09:19:10 +02:00
Dennis
99968e57ec wip 2021-08-17 13:00:47 +02:00
Dennis
f75bc1a551 Production mix 2021-08-12 11:25:19 +02:00
Dennis
ef347f9381 Merge branch 'develop' 2021-08-12 11:24:30 +02:00
Dennis
f1aace3d8f Ability to select allowed regions/plans per provider 2021-08-12 11:23:01 +02:00
Dennis
b2bec62766 Merge branch 'develop'
# Conflicts:
#	public/js/app.js
2021-08-12 11:03:51 +02:00
Dennis
3a8682bbed Textual change 2021-08-12 11:03:25 +02:00
Dennis
5cfc1f97fe Logo fix 2021-08-12 10:26:29 +02:00
Dennis
cfc8220f8e Wip 2021-08-12 10:23:23 +02:00
Dennis
ac5ffefaed Horizon status to system settings tab 2021-08-10 14:04:52 +02:00
Dennis
cbd2b8e0e9 Wip on terms & privacy page configurator 2021-08-10 13:51:37 +02:00
Dennis
f0f427a7bb General improvements 2021-08-05 11:31:38 +02:00
Dennis
7b6f651015 Consistent Ploi class instantiate 2021-08-05 11:14:16 +02:00
Dennis
c793daa79a wip 2021-08-05 11:08:19 +02:00
Dennis
71b436aebe Production mix 2021-08-04 10:17:42 +02:00
Dennis
06e108da5b Merge branch 'develop'
# Conflicts:
#	public/js/app.js
2021-08-04 10:17:03 +02:00
Dennis
5677c58dd2 Package updates 2021-08-04 10:07:36 +02:00
Dennis
58ca801e30 PSR formatting 2021-08-04 09:49:47 +02:00
Dennis
e750d7caba Cleaning 2021-08-04 09:49:24 +02:00
Dennis
3c0964ef0e Bugfixes in server settings, site creation 2021-08-04 09:46:11 +02:00
Dennis
33613cdf1c Sync all sites & servers, bugfix in pagination providers 2021-08-04 09:36:28 +02:00
Dennis
abfe174825 Autofocus on login page 2021-08-04 09:19:40 +02:00
Dennis
c750f469bb Updates
- Generic demo check in base controller
- Site synchronization
- User show pagination preserve scroll
2021-08-03 11:58:38 +02:00
Dennis
c7ac56d5cc Wip custom certificates 2021-07-29 09:59:20 +02:00
Dennis
6601f44013 wip custom certificates 2021-07-29 09:07:06 +02:00
Dennis
b7baf862b6 prod mix 2021-07-27 08:33:16 +02:00
Dennis
ca79f4cf21 Merge branch 'develop'
# Conflicts:
#	public/js/app.js
2021-07-27 08:32:39 +02:00
Dennis
58ffbe8c74 URL fixes 2021-07-27 08:32:30 +02:00
Dennis
dfa7b995fc wip 2021-07-23 13:46:53 +02:00
Dennis
59c65fe6ee prod 2021-07-18 10:12:47 +02:00
Dennis
087c042c14 Merge branch 'develop'
# Conflicts:
#	public/js/app.js
2021-07-18 10:12:15 +02:00
Dennis
d1ee488ffd fixes 2021-07-18 10:11:50 +02:00
Dennis
34100bc580 prod mix 2021-07-14 09:51:24 +02:00
Dennis
22d91517fb Merge branch 'develop'
# Conflicts:
#	public/js/app.js
2021-07-14 09:50:10 +02:00
Dennis
1c088bd4e0 Bugfixes and improvements 2021-07-14 09:50:00 +02:00
Dennis
fd5bbb7f5d Merge branch 'develop' 2021-07-13 14:49:03 +02:00
Dennis
426d39bec0 wip 2021-07-13 14:48:56 +02:00
Dennis
8244c2dfb2 Merge branch 'develop' 2021-07-13 14:23:54 +02:00
Dennis
e7f9d32f68 wip 2021-07-13 14:23:49 +02:00
Dennis
71158a0030 prod 2021-07-12 12:33:07 +02:00
Dennis
573ce14b7a Merge branch 'develop'
# Conflicts:
#	public/css/app.css
#	public/js/app.js
2021-07-12 12:31:37 +02:00
Dennis
00ef1f470f add debugging 2021-07-12 12:31:19 +02:00
Dennis
a9357034c8 THB currency, annual billing added 2021-07-12 12:26:21 +02:00
Dennis
a45f5157d4 wip 2021-07-09 11:48:12 +02:00
Dennis
0d1009b8db horizon config update 2021-07-09 11:45:57 +02:00
Dennis
4f0416cd45 Added user show, pagination simplified, package updates 2021-07-09 09:08:09 +02:00
Dennis
76655d76d5 Installation command improvements 2021-07-08 14:26:28 +02:00
Dennis
3b38cbe9ac Merge branch 'develop' 2021-06-17 09:18:34 +02:00
Dennis
acf37b8850 Suppport PHP 8 2021-06-17 09:18:28 +02:00
Dennis
f47a0699d3 Merge branch 'develop' 2021-06-07 09:18:21 +02:00
Dennis
8ae429b06b Tweak 2021-06-07 09:18:14 +02:00
Dennis
626dbbcb49 Fix 2021-06-05 20:30:22 +02:00
Dennis
b623dd80fd Merge branch 'develop' 2021-05-31 15:55:32 +02:00
Dennis
3978a7c9f7 User deleting fix 2021-05-31 15:53:58 +02:00
Dennis
7d2acb7438 Merge branch 'develop' 2021-05-31 08:55:21 +02:00
Dennis
096032301c Fix when deleting user with support tickets 2021-05-31 08:55:15 +02:00
Dennis
c3e99bf2ff Merge branch 'develop' 2021-05-25 07:37:34 +02:00
Dennis
9795642bc7 Bugfix 2021-05-25 07:37:28 +02:00
Dennis
4281a432fb Merge branch 'develop' 2021-05-22 09:36:29 +02:00
Dennis
18df5589b1 Fix title 2021-05-22 09:36:25 +02:00
Dennis
79536eac2e Fresh mix 2021-05-19 09:38:13 +02:00
Dennis
6df82fca04 Clean push 2021-05-19 09:36:22 +02:00
Dennis
cc14123d27 Merge branch 'develop' 2021-05-19 09:36:04 +02:00
Dennis
bd5e7b87ff Package update && password uncomprimised 2021-05-19 09:35:54 +02:00
Dennis
42568916d6 prod mix 2021-05-18 12:17:24 +02:00
572 changed files with 18272 additions and 15511 deletions

View File

@@ -26,6 +26,7 @@ SESSION_LIFETIME=120
REDIS_HOST=127.0.0.1 REDIS_HOST=127.0.0.1
REDIS_PASSWORD=null REDIS_PASSWORD=null
REDIS_PORT=6379 REDIS_PORT=6379
REDIS_CLIENT=predis
MAIL_MAILER=smtp MAIL_MAILER=smtp
MAIL_HOST=smtp.mailtrap.io MAIL_HOST=smtp.mailtrap.io

41
.github/workflows/master.yml vendored Normal file
View File

@@ -0,0 +1,41 @@
name: Run tests & build files
on:
push:
branches:
- master
jobs:
test:
name: Run tests
uses: ./.github/workflows/run-tests.yml
deploy:
needs: test
name: Prepare build assets
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- name: Setup PHP with PECL extension
uses: shivammathur/setup-php@v2
with:
php-version: '8.1'
extensions: pcntl
- run: composer install
name: Install dependencies
- name: Set up Node
uses: actions/setup-node@v1
with:
node-version: '12.x'
- run: npm install
- run: npm run production
- name: Commit build assets
run: |
git config --local user.email "actions@github.com"
git config --local user.name "GitHub Actions"
git add .
git commit -m "Run Laravel Mix en build front-end assets"
git push origin

33
.github/workflows/run-tests.yml vendored Normal file
View File

@@ -0,0 +1,33 @@
name: "Run tests"
on:
push:
workflow_call:
jobs:
test:
name: Pest (PHP ${{ matrix.php }} ${{ matrix.os }})
strategy:
fail-fast: false
matrix:
os: [ubuntu-latest]
php: [8.0, 8.1]
runs-on: ${{ matrix.os }}
steps:
- name: Checkout code
uses: actions/checkout@v2
- name: Setup PHP
uses: shivammathur/setup-php@v2
with:
php-version: ${{ matrix.php }}
extensions: dom, curl, libxml, mbstring, zip, pcntl, pdo, sqlite, pdo_sqlite, bcmath, soap, intl, gd, exif, iconv, imagick, fileinfo, mysql
coverage: none
- name: Install dependencies
run: |
composer install
- name: Execute tests
run: vendor/bin/pest

7
.gitignore vendored
View File

@@ -1,8 +1,10 @@
/node_modules /node_modules
/node_modules.nosync
/public/hot /public/hot
/public/storage /public/storage
/storage/*.key /storage/*.key
/vendor /vendor
/vendor.nosync
.env .env
.env.backup .env.backup
.phpunit.result.cache .phpunit.result.cache
@@ -12,4 +14,9 @@ npm-debug.log
yarn-error.log yarn-error.log
.idea .idea
.php_cs.cache .php_cs.cache
.php-cs-fixer.cache
/public/js/resources*.js /public/js/resources*.js
/storage/views/header.blade.php
/storage/views/footer.blade.php
rr
.rr.yaml

View File

@@ -3,7 +3,7 @@
With Ploi Core, you'll power-launch your webhosting company. With Ploi Core, you'll power-launch your webhosting company.
Using the ploi.io system as backbone you will be able to serve your customers your custom panel & feeling. Using the ploi.io system as backbone you will be able to serve your customers your custom panel & feeling.
<p align="center"><img src="https://ploi-core.io/images/featured.png" width="100%"></p> <p align="center"><img src="https://ploi-core.io/images/og.jpg" width="100%"></p>
## Documentation ## Documentation

View File

@@ -0,0 +1,56 @@
<?php
namespace App\Actions\Server;
use App\DataTransferObjects\ServerData;
use App\Jobs\Servers\CreateServer;
use App\Mail\Admin\Server\AdminServerCreatedEmail;
use App\Models\Server;
use App\Models\User;
use Illuminate\Support\Facades\Mail;
class CreateServerAction
{
public function execute(ServerData $serverData): Server
{
[$provider, $providerRegion, $providerPlan] = $this->determineProviderRegionPlan($serverData);
$server = $serverData->getUser()->servers()->create([
'name' => $serverData->name,
'database_type' => $serverData->database_type,
]);
$server->provider()->associate($provider);
$server->providerRegion()->associate($providerRegion);
$server->providerPlan()->associate($providerPlan);
$server->save();
dispatch(new CreateServer($server));
$this->sendAdminServerCreatedEmails($server);
return $server;
}
protected function determineProviderRegionPlan(ServerData $serverData): array
{
$provider = $serverData->getUser()->package->providers()->findOrFail($serverData->provider_id);
$region = $provider->regions()->findOrFail($serverData->provider_region_id);
$plan = $provider->plans()->findOrFail($serverData->provider_plan_id);
return [$provider, $region, $plan];
}
protected function sendAdminServerCreatedEmails(Server $server): void
{
if (! setting('receive_email_on_server_creation')) {
return;
}
$admins = User::query()->where('role', User::ADMIN)->get();
foreach ($admins as $admin) {
Mail::to($admin)->send(new AdminServerCreatedEmail(auth()->user(), $server));
}
}
}

View File

@@ -0,0 +1,76 @@
<?php
namespace App\Actions\Site;
use App\DataTransferObjects\SiteData;
use App\Jobs\Sites\CreateSite;
use App\Mail\Admin\Site\AdminSiteCreatedEmail;
use App\Models\Server;
use App\Models\Site;
use App\Models\User;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Support\Facades\Mail;
class CreateSiteAction
{
public function execute(SiteData $siteData): ?Site
{
$server = $this->determineServer($siteData);
if ( ! $server ) {
return null;
}
$site = $server->sites()->create($siteData->toArray());
$siteData->getUser()->sites()->save($site);
dispatch(new CreateSite($site));
$siteData->getUser()->systemLogs()->create([
'title' => 'New site :site created',
'description' => 'A new site has been created',
])->model()->associate($site)->save();
$this->sendAdminSiteCreatedEmails($server, $site, $siteData->getUser());
return $site;
}
protected function determineServer(SiteData $siteData): ?Server
{
if ( $siteData->server_id ) {
return $siteData->getUser()->servers()->findOrFail($siteData->server_id);
}
$server = Server::query()
->where('maximum_sites', '>', 0)
->where(function ($query) use ($siteData) {
return $query
->where(fn ($query) => $query->whereHas('users', fn ($query) => $query->where('user_id', $siteData->getUser()->id)))
->orWhere(function ($query) {
return $query->doesntHave('users');
});
})
->withCount('sites')
->inRandomOrder()
->first();
return $server && $server->sites_count < $server->maximum_sites
? $server
: null;
}
protected function sendAdminSiteCreatedEmails(Server $server, Model|Site $site, User $user): void
{
if ( ! setting('receive_email_on_site_creation') ) {
return;
}
$admins = User::where('role', User::ADMIN)->get();
foreach ($admins as $admin) {
Mail::to($admin)->send(new AdminSiteCreatedEmail(user: $user, server: $server, site: $site));
}
}
}

26
app/Casts/SiteAlias.php Normal file
View File

@@ -0,0 +1,26 @@
<?php
namespace App\Casts;
use Illuminate\Contracts\Database\Eloquent\CastsAttributes;
class SiteAlias implements CastsAttributes
{
public function get($model, string $key, $value, array $attributes)
{
if (!$value) {
return [];
}
$data = json_decode($value, true);
sort($data);
return $data;
}
public function set($model, string $key, $value, array $attributes)
{
return json_encode($value);
}
}

View File

@@ -0,0 +1,60 @@
<?php
namespace App\Console\Commands\Core;
use App\Models\SystemLog;
use Illuminate\Console\Command;
class Cleanup extends Command
{
protected $signature = 'core:cleanup';
protected $description = 'Clean up any old logs';
public function handle()
{
if (!setting('rotate_logs_after')) {
return Command::SUCCESS;
}
$rotationDate = $this->getRotationDate();
$rotated = SystemLog::query()
->where('created_at', '<', $rotationDate)
->delete();
$this->info('Rotated ' . $rotated . ' system logs!');
return Command::SUCCESS;
}
protected function getRotationDate()
{
switch (setting('rotate_logs_after')) {
case 'weeks-1':
return now()->subWeek();
break;
case 'months-1':
return now()->subMonth();
break;
case 'months-3':
return now()->subMonths(3);
break;
case 'months-6':
return now()->subMonths(6);
break;
case 'years-1':
return now()->subYear();
break;
case 'years-2':
return now()->subYears(2);
break;
case 'years-3':
return now()->subYears(3);
break;
case 'years-4':
return now()->subYears(4);
break;
}
}
}

View File

@@ -7,6 +7,7 @@ use App\Models\User;
use RuntimeException; use RuntimeException;
use App\Models\Package; use App\Models\Package;
use App\Services\Ploi\Ploi; use App\Services\Ploi\Ploi;
use Illuminate\Support\Arr;
use Illuminate\Support\Str; use Illuminate\Support\Str;
use Illuminate\Console\Command; use Illuminate\Console\Command;
use App\Services\VersionChecker; use App\Services\VersionChecker;
@@ -17,7 +18,7 @@ use Illuminate\Support\Facades\Artisan;
class Install extends Command class Install extends Command
{ {
protected $company; protected $company;
protected $signature = 'core:install'; protected $signature = 'core:install {--force}';
protected $description = 'Installation command for Ploi Core'; protected $description = 'Installation command for Ploi Core';
protected $versionChecker; protected $versionChecker;
protected $installationFile = 'app/installation'; protected $installationFile = 'app/installation';
@@ -37,7 +38,15 @@ class Install extends Command
$this->createInstallationFile(); $this->createInstallationFile();
$this->linkStorage(); $this->linkStorage();
$this->info('Succes! Installation has finished.'); $this->info('Success! Installation has finished.');
$this->line(' ');
$this->writeSeparationLine();
$this->info('Make sure to also setup emailing, the cronjob and the queue worker.');
$this->line(' ');
$this->info('Setting up emailing: https://docs.ploi-core.io/getting-started/setting-up-email');
$this->info('Setting up cronjob & queue worker: https://docs.ploi-core.io/getting-started/installation');
$this->writeSeparationLine();
$this->line(' ');
$this->info('Visit your platform at ' . env('APP_URL')); $this->info('Visit your platform at ' . env('APP_URL'));
} }
@@ -48,28 +57,32 @@ class Install extends Command
protected function askAboutAdministrationAccount() protected function askAboutAdministrationAccount()
{ {
$this->info('Let\'s start by setting up your administration account.'); if (!User::query()->where('role', User::ADMIN)->count()) {
$this->info('Let\'s start by setting up your administration account.');
$name = $this->ask('What is your name', $this->company['user_name']); $name = $this->ask('What is your name', $this->company['user_name']);
$email = $this->ask('What is your e-mail address', $this->company['email']); $email = $this->ask('What is your e-mail address', $this->company['email']);
$password = $this->secret('What password do you desire'); $password = $this->secret('What password do you desire');
$check = User::where('email', $email)->count(); $check = User::where('email', $email)->count();
if ($check) { if ($check) {
$this->line(''); $this->line('');
$this->comment('This user is already present in your system, please refresh your database or use different credentials.'); $this->comment('This user is already present in your system, please refresh your database or use different credentials.');
$this->comment('Aborting installation..'); $this->comment('Aborting installation..');
exit(); exit();
}
User::forceCreate([
'name' => $name,
'email' => $email,
'password' => $password,
'role' => User::ADMIN
]);
} else {
$this->line('Already found a administrator user in your system. Use that user to login.');
} }
User::forceCreate([
'name' => $name,
'email' => $email,
'password' => $password,
'role' => User::ADMIN
]);
} }
protected function askAboutDefaultPackages() protected function askAboutDefaultPackages()
@@ -140,7 +153,9 @@ class Install extends Command
->get((new Ploi)->url . 'ping'); ->get((new Ploi)->url . 'ping');
if (!$response->ok() || !$response->json()) { if (!$response->ok() || !$response->json()) {
return false; return [
'error' => Arr::get($response->json(), 'message', 'An unknown error has occurred.')
];
} }
return $response->json(); return $response->json();
@@ -155,7 +170,7 @@ class Install extends Command
protected function intro() protected function intro()
{ {
$this->info('*---------------------------------------------------------------------------*'); $this->writeSeparationLine();
$this->line('Ploi Core Installation'); $this->line('Ploi Core Installation');
$this->line('Ploi Core version: ' . $this->versionChecker->currentVersion); $this->line('Ploi Core version: ' . $this->versionChecker->currentVersion);
$this->line('Ploi Core remote: ' . $this->versionChecker->remoteVersion); $this->line('Ploi Core remote: ' . $this->versionChecker->remoteVersion);
@@ -165,13 +180,13 @@ class Install extends Command
$this->line('Website: https://ploi-core.io'); $this->line('Website: https://ploi-core.io');
$this->line('E-mail: core@ploi.io'); $this->line('E-mail: core@ploi.io');
$this->line('Terms of service: https://ploi-core.io/terms'); $this->line('Terms of service: https://ploi-core.io/terms');
$this->info('*---------------------------------------------------------------------------*'); $this->writeSeparationLine();
$this->line(''); $this->line('');
} }
protected function isInstalled() protected function isInstalled()
{ {
if (file_exists(storage_path($this->installationFile))) { if (file_exists(storage_path($this->installationFile)) && !$this->option('force')) {
$this->line(''); $this->line('');
$this->comment('Ploi Core has already been installed before.'); $this->comment('Ploi Core has already been installed before.');
$this->comment('If you still want to start installation, remove this file to continue: ./storage/' . $this->installationFile); $this->comment('If you still want to start installation, remove this file to continue: ./storage/' . $this->installationFile);
@@ -187,9 +202,9 @@ class Install extends Command
{ {
if (!config('app.key')) { if (!config('app.key')) {
$this->call('key:generate'); $this->call('key:generate');
}
$this->info('Application key has been set'); $this->info('Application key has been set');
}
} }
protected function checkApplicationUrl() protected function checkApplicationUrl()
@@ -276,6 +291,18 @@ class Install extends Command
exit(); exit();
} }
if (isset($this->company['error'])) {
$this->error($this->company['error']);
exit();
}
if ($this->company['user']['subscription'] !== 'unlimited') {
$this->error('Your subscription does not cover the usage of Ploi Core. Please upgrade your subscription to Unlimited.');
exit();
}
$this->writeToEnvironmentFile('PLOI_TOKEN', $ploiApiToken); $this->writeToEnvironmentFile('PLOI_TOKEN', $ploiApiToken);
$this->writeToEnvironmentFile('PLOI_CORE_TOKEN', $ploiCoreKey); $this->writeToEnvironmentFile('PLOI_CORE_TOKEN', $ploiCoreKey);
@@ -288,7 +315,7 @@ class Install extends Command
protected function runDatabaseMigrations() protected function runDatabaseMigrations()
{ {
$this->info('Running database migrations..'); $this->info('Running database migrations..');
$this->call('migrate'); $this->call('migrate', ['--force' => true]);
$this->info('Database migrations successful'); $this->info('Database migrations successful');
} }
@@ -379,4 +406,9 @@ class Install extends Command
{ {
$this->laravel['config'][$key] = $value; $this->laravel['config'][$key] = $value;
} }
protected function writeSeparationLine()
{
$this->info('*---------------------------------------------------------------------------*');
}
} }

View File

@@ -9,7 +9,7 @@ class Synchronize extends Command
{ {
protected $signature = 'core:synchronize'; protected $signature = 'core:synchronize';
protected $description = 'Synchronze data'; protected $description = 'Synchronize data';
public function handle() public function handle()
{ {

View File

@@ -0,0 +1,24 @@
<?php
namespace App\Console\Commands\Core;
use App\Models\User;
use Illuminate\Console\Command;
class Trial extends Command
{
protected $signature = 'core:trial';
protected $description = 'Check for expired trials';
public function handle()
{
User::query()
->where('trial_ends_at', '<', now())
->each(function (User $user) {
$user->trial_ends_at = null;
$user->package_id = setting('default_package');
$user->save();
});
}
}

View File

@@ -2,8 +2,10 @@
namespace App\Console; namespace App\Console;
use App\Console\Commands\Core\Trial;
use App\Jobs\Core\Ping; use App\Jobs\Core\Ping;
use App\Console\Commands\Core\Css; use App\Console\Commands\Core\Css;
use App\Console\Commands\Core\Cleanup;
use App\Console\Commands\Core\Install; use App\Console\Commands\Core\Install;
use App\Console\Commands\Core\CssBackup; use App\Console\Commands\Core\CssBackup;
use App\Console\Commands\Core\Synchronize; use App\Console\Commands\Core\Synchronize;
@@ -17,6 +19,8 @@ class Kernel extends ConsoleKernel
CssBackup::class, CssBackup::class,
Install::class, Install::class,
Synchronize::class, Synchronize::class,
Cleanup::class,
Trial::class,
]; ];
protected function schedule(Schedule $schedule) protected function schedule(Schedule $schedule)
@@ -24,5 +28,8 @@ class Kernel extends ConsoleKernel
$schedule->call(function () { $schedule->call(function () {
dispatch(new Ping())->delay(now()->addMinutes(rand(1, 30))); dispatch(new Ping())->delay(now()->addMinutes(rand(1, 30)));
})->dailyAt('02:00'); })->dailyAt('02:00');
$schedule->command('core:cleanup')->daily();
$schedule->command('core:trial')->dailyAt('10:00');
} }
} }

View File

@@ -0,0 +1,37 @@
<?php
namespace App\DataTransferObjects;
use App\DataTransferObjects\Support\Concerns\BelongsToUser;
use App\DataTransferObjects\Support\Data;
use App\Models\Provider;
use App\Models\ProviderPlan;
use App\Models\ProviderRegion;
use App\Models\User;
use Spatie\LaravelData\Attributes\Validation\AlphaDash;
use Spatie\LaravelData\Attributes\Validation\Exists;
use Spatie\LaravelData\Attributes\Validation\In;
use Spatie\LaravelData\Attributes\Validation\IntegerType;
use Spatie\LaravelData\Attributes\Validation\Max;
use Spatie\LaravelData\Attributes\Validation\NotIn;
use Spatie\LaravelData\Attributes\Validation\StringType;
class ServerData extends Data
{
use BelongsToUser;
public function __construct(
#[StringType, AlphaDash, Max( 40 )]
public string $name,
#[NotIn( 0 ), Exists( Provider::class, 'id' )]
public int $provider_id,
#[NotIn( 0 ), Exists( ProviderRegion::class, 'id' )]
public int $provider_region_id,
#[NotIn( 0 ), Exists( ProviderPlan::class, 'id' )]
public int $provider_plan_id,
#[StringType, In( ['mysql', 'mariadb', 'postgresql', 'postgresql13'] )]
public string $database_type,
#[Exists( User::class, 'id' ), IntegerType]
public ?int $user_id = null,
) {}
}

View File

@@ -0,0 +1,52 @@
<?php
namespace App\DataTransferObjects;
use App\DataTransferObjects\Support\Concerns\BelongsToUser;
use App\DataTransferObjects\Support\Data;
use App\DataTransferObjects\Support\Rules\CustomRule;
use App\DataTransferObjects\Support\WithUser;
use App\Models\Server;
use App\Models\Site;
use App\Models\User;
use App\Rules\Hostname;
use App\Rules\ValidateMaximumSites;
use Illuminate\Support\Arr;
use Spatie\LaravelData\Attributes\Validation\Exists;
use Spatie\LaravelData\Attributes\Validation\IntegerType;
use Spatie\LaravelData\Attributes\Validation\StringType;
class SiteData extends Data
{
use BelongsToUser;
public function __construct(
public ?int $id = null,
public ?string $status = null,
#[Exists( Server::class, 'id' ), IntegerType]
public ?int $server_id = null,
#[StringType, CustomRule(Hostname::class, ValidateMaximumSites::class)]
public ?string $domain = null,
#[Exists(User::class, 'id'), IntegerType]
public ?int $user_id = null,
) {}
public static function authorize(): bool
{
if ( auth()->guest() ) {
return true;
}
return auth()->user()->can('create', Site::class);
}
public function toArray(): array
{
return Arr::only(parent::toArray(), [
'id',
'status',
'server_id',
'domain',
]);
}
}

View File

@@ -0,0 +1,15 @@
<?php
namespace App\DataTransferObjects\Support\Casts;
use Illuminate\Support\Carbon;
use Spatie\LaravelData\Casts\Cast;
use Spatie\LaravelData\Support\DataProperty;
class CarbonCast implements Cast
{
public function cast(DataProperty $property, mixed $value): mixed
{
return Carbon::parse($value);
}
}

View File

@@ -0,0 +1,13 @@
<?php
namespace App\DataTransferObjects\Support\Concerns;
use App\Models\User;
trait BelongsToUser
{
public function getUser(): ?User
{
return User::find($this->user_id);
}
}

View File

@@ -0,0 +1,17 @@
<?php
namespace App\DataTransferObjects\Support;
use Illuminate\Contracts\Pagination\Paginator;
use Illuminate\Pagination\AbstractCursorPaginator;
use Illuminate\Pagination\AbstractPaginator;
use Illuminate\Support\Enumerable;
use Spatie\LaravelData\DataCollection;
class Data extends \Spatie\LaravelData\Data
{
public static function collection(Paginator|Enumerable|array|AbstractCursorPaginator|DataCollection|AbstractPaginator $items): \App\DataTransferObjects\Support\DataCollection
{
return new \App\DataTransferObjects\Support\DataCollection(static::class, $items);
}
}

View File

@@ -0,0 +1,23 @@
<?php
namespace App\DataTransferObjects\Support;
use Spatie\LaravelData\Support\TransformationType;
class DataCollection extends \Spatie\LaravelData\DataCollection
{
public function transform(TransformationType $type): array
{
$transformer = new DataCollectionTransformer(
$this->dataClass,
$type,
$this->getInclusionTree(),
$this->getExclusionTree(),
$this->items,
$this->through,
$this->filter
);
return $transformer->transform();
}
}

View File

@@ -0,0 +1,40 @@
<?php
namespace App\DataTransferObjects\Support;
use Illuminate\Support\Arr;
class DataCollectionTransformer extends \Spatie\LaravelData\Transformers\DataCollectionTransformer
{
protected function wrapPaginatedArray(array $paginated): array
{
return [
'data' => $paginated['data'],
'links' => [
'first' => $paginated['first_page_url'],
'last' => $paginated['last_page_url'],
'prev' => $paginated['prev_page_url'],
'next' => $paginated['next_page_url'],
],
'meta' => [
'current_page' => $paginated['current_page'],
'from' => $paginated['from'],
'last_page' => $paginated['last_page'],
'links' => $paginated['links'],
'path' => $paginated['path'],
'per_page' => $paginated['per_page'],
'to' => $paginated['to'],
'total' => $paginated['total'],
],
];
return [
'data' => $paginated['data'],
'links' => $paginated['links'] ?? [],
'meta' => Arr::except($paginated, [
'data',
'links',
]),
];
}
}

View File

@@ -0,0 +1,24 @@
<?php
namespace App\DataTransferObjects\Support\Rules;
use Attribute;
use Spatie\LaravelData\Attributes\Validation\ValidationAttribute;
#[Attribute( Attribute::TARGET_PROPERTY )]
class CustomRule extends ValidationAttribute
{
protected array $rules = [];
public function __construct(...$rules)
{
$this->rules = $rules;
}
public function getRules(): array
{
return collect($this->rules)
->map(fn (string $rule) => new $rule())
->all();
}
}

View File

@@ -0,0 +1,14 @@
<?php
namespace App\DataTransferObjects\Support\Transformers;
use Spatie\LaravelData\Support\DataProperty;
use Spatie\LaravelData\Transformers\Transformer;
class CarbonTransformer implements Transformer
{
public function transform(DataProperty $property, mixed $value): mixed
{
return $value->toISOString();
}
}

View File

@@ -0,0 +1,31 @@
<?php
namespace App\DataTransferObjects;
use App\DataTransferObjects\Support\Data;
use App\Models\Package;
use App\Models\User;
use Illuminate\Support\Carbon;
use Spatie\LaravelData\Attributes\Validation\Email;
use Spatie\LaravelData\Attributes\Validation\Exists;
use Spatie\LaravelData\Attributes\Validation\IntegerType;
use Spatie\LaravelData\Attributes\Validation\Max;
use Spatie\LaravelData\Attributes\Validation\StringType;
use Spatie\LaravelData\Attributes\Validation\Unique;
class UserData extends Data
{
public function __construct(
public ?int $id = null,
public ?string $avatar = null,
#[StringType, Max(255)]
public ?string $name = null,
#[StringType, Email, Max(255), Unique(User::class)]
public ?string $email = null,
#[Exists( Package::class, 'id'), IntegerType]
public ?int $package_id = null,
#[StringType]
public ?string $blocked = null,
public ?Carbon $created_at = null,
) {}
}

View File

@@ -2,7 +2,11 @@
namespace App\Exceptions; namespace App\Exceptions;
use Exception;
use Throwable; use Throwable;
use Illuminate\Http\Request;
use App\Http\Middleware\HandleInertiaRequests;
use Symfony\Component\HttpFoundation\Response;
use Illuminate\Foundation\Exceptions\Handler as ExceptionHandler; use Illuminate\Foundation\Exceptions\Handler as ExceptionHandler;
class Handler extends ExceptionHandler class Handler extends ExceptionHandler
@@ -29,10 +33,10 @@ class Handler extends ExceptionHandler
/** /**
* Report or log an exception. * Report or log an exception.
* *
* @param \Throwable $exception * @param Throwable $exception
* @return void * @return void
* *
* @throws \Exception * @throws Exception
*/ */
public function report(Throwable $exception) public function report(Throwable $exception)
{ {
@@ -42,24 +46,21 @@ class Handler extends ExceptionHandler
/** /**
* Render an exception into an HTTP response. * Render an exception into an HTTP response.
* *
* @param \Illuminate\Http\Request $request * @param Request $request
* @param \Throwable $exception * @param Throwable $exception
* @return \Symfony\Component\HttpFoundation\Response * @return Response
* *
* @throws \Throwable * @throws Throwable
*/ */
public function render($request, Throwable $exception) public function render($request, Throwable $exception)
{ {
$response = parent::render($request, $exception); $response = parent::render($request, $exception);
if (in_array($response->status(), [404, 403])) { // Only return an Inertia-response when there are special Vue-templates (403 and 404) and when it isn't an API request.
\Route::any($request->path(), function () use ($exception, $request) { if (in_array($response->status(), [403, 404]) && ! $request->routeIs('api.*')) {
return parent::render($request, $exception); return app(HandleInertiaRequests::class)
})->middleware('web'); ->handle($request, fn () => inertia()->render('Errors/' . $response->status(), ['status' => $response->status()])
->toResponse($request));
return inertia()->render('Errors/' . $response->status(), ['status' => $response->status()])
->toResponse($request)
->setStatusCode($response->status());
} }
return $response; return $response;

View File

@@ -1,17 +1,19 @@
<?php <?php
use App\Models\Setting;
use Illuminate\Contracts\Foundation\Application;
use Illuminate\Support\Arr; use Illuminate\Support\Arr;
if (!function_exists('setting')) { if (!function_exists('setting')) {
/** /**
* @param null $key * @param null $key
* @param null $default * @param null $default
* @return array|ArrayAccess|bool|\Illuminate\Contracts\Foundation\Application|mixed * @return array|ArrayAccess|bool|Application|mixed
*/ */
function setting($key = null, $default = null) function setting($key = null, $default = null)
{ {
if (is_array($key)) { if (is_array($key)) {
\App\Models\Setting::updateOrCreate([ Setting::updateOrCreate([
'key' => key($key) 'key' => key($key)
], [ ], [
'value' => Arr::first($key) 'value' => Arr::first($key)
@@ -20,6 +22,7 @@ if (!function_exists('setting')) {
try { try {
cache()->forget('core.settings'); cache()->forget('core.settings');
} catch (Exception $e) { } catch (Exception $e) {
//
} }
return true; return true;
@@ -28,7 +31,7 @@ if (!function_exists('setting')) {
$value = Arr::get(app('settings'), $key, $default); $value = Arr::get(app('settings'), $key, $default);
// Boolean casting // Boolean casting
if ($value === "0" || $value === "1") { if ($value === "0" || $value === "1" && $key !== 'trial_package') {
return (bool) $value; return (bool) $value;
} }

View File

@@ -10,14 +10,17 @@ use App\Http\Controllers\Controller;
class DashboardController extends Controller class DashboardController extends Controller
{ {
public function index() public function __invoke()
{ {
return inertia('Admin/Dashboard', [ return inertia('Admin/Dashboard', [
'servers' => Server::count(), 'servers' => Server::count(),
'sites' => Site::count(), 'sites' => Site::count(),
'users' => User::count(), 'users' => User::count(),
'logs' => SystemLog::latest()->limit(10)->with('model')->get() 'logs' => SystemLog::query()
->map(function (SystemLog $systemLog) { ->latest()
->with('model')
->paginate(5)
->through(function (SystemLog $systemLog) {
return [ return [
'title' => __($systemLog->title, [ 'title' => __($systemLog->title, [
'site' => $systemLog->model->domain ?? '-Unknown-' 'site' => $systemLog->model->domain ?? '-Unknown-'

View File

@@ -14,6 +14,8 @@ class ProviderController extends Controller
return inertia('Admin/Services/Provider/Edit', [ return inertia('Admin/Services/Provider/Edit', [
'provider' => $provider, 'provider' => $provider,
'availablePlans' => $provider->plans()->pluck('label', 'id'),
'availableRegions' => $provider->regions()->pluck('label', 'id'),
]); ]);
} }

View File

@@ -4,12 +4,39 @@ namespace App\Http\Controllers\Admin;
use App\Models\User; use App\Models\User;
use App\Models\Server; use App\Models\Server;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Http\Request; use Illuminate\Http\Request;
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
use App\Http\Resources\Admin\ServerResource;
use App\Http\Requests\Admin\ServerAttachRequest; use App\Http\Requests\Admin\ServerAttachRequest;
class ServerController extends Controller class ServerController extends Controller
{ {
public function index()
{
return inertia('Admin/Servers/Index', [
'filters' => request()->all('search'),
'servers' => ServerResource::collection(
Server::query()
->when(request()->input('search'), function (Builder $query, $value) {
return $query
->where('name', 'like', '%' . $value . '%')
->orWhere('ip', 'like', '%' . $value . '%')
->orWhereHas('users', function (Builder $query) use ($value) {
return $query
->where('name', 'LIKE', '%' . $value . '%')
->orWhere('email', 'LIKE', '%' . $value . '%');
});
})
->with('users:id,name')
->withCount('sites')
->latest()
->paginate(config('core.pagination.per_page'))
->withQueryString()
)
]);
}
public function edit($id) public function edit($id)
{ {
$server = Server::findOrFail($id); $server = Server::findOrFail($id);

View File

@@ -12,12 +12,12 @@ class ServiceController extends Controller
public function index() public function index()
{ {
return inertia('Admin/Services/Index', [ return inertia('Admin/Services/Index', [
'servers' => Server::withCount('sites')->latest()->paginate(5, ['*'], 'servers_per_page'), 'servers' => Server::query()->withCount('sites', 'users')->latest()->paginate(config('core.pagination.per_page'), ['*'], 'servers_per_page'),
'sites' => Site::with('server:id,name')->latest()->paginate(5, ['*'], 'sites_per_page'), 'sites' => Site::with('server:id,name')->withCount('users')->latest()->paginate(config('core.pagination.per_page'), ['*'], 'sites_per_page'),
'providers' => Provider::query() 'providers' => Provider::query()
->withCount('regions', 'plans', 'servers') ->withCount('regions', 'plans', 'servers')
->latest() ->latest()
->paginate(5, ['*'], 'providers_per_page'), ->paginate(config('core.pagination.per_page'), ['*'], 'providers_per_page'),
]); ]);
} }
} }

View File

@@ -4,6 +4,7 @@ namespace App\Http\Controllers\Admin;
use App\Models\Package; use App\Models\Package;
use Illuminate\Support\Str; use Illuminate\Support\Str;
use Illuminate\Http\Request;
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
use Illuminate\Support\Facades\Storage; use Illuminate\Support\Facades\Storage;
use App\Http\Requests\Admin\SettingRequest; use App\Http\Requests\Admin\SettingRequest;
@@ -21,10 +22,15 @@ class SettingController extends Controller
'allow_registration' => setting('allow_registration'), 'allow_registration' => setting('allow_registration'),
'default_package' => setting('default_package'), 'default_package' => setting('default_package'),
'receive_email_on_server_creation' => setting('receive_email_on_server_creation'), 'receive_email_on_server_creation' => setting('receive_email_on_server_creation'),
'receive_email_on_site_creation' => setting('receive_email_on_site_creation'),
'isolate_per_site_per_user' => setting('isolate_per_site_per_user'), 'isolate_per_site_per_user' => setting('isolate_per_site_per_user'),
'enable_api' => setting('enable_api'), 'enable_api' => setting('enable_api'),
'api_token' => setting('api_token') ? decrypt(setting('api_token')) : null, 'api_token' => setting('api_token') ? decrypt(setting('api_token')) : null,
'default_language' => setting('default_language', 'en') 'rotate_logs_after' => setting('rotate_logs_after') ? setting('rotate_logs_after') : null,
'default_language' => setting('default_language', 'en'),
'has_logo' => (bool)setting('logo'),
'trial' => setting('trial'),
'trial_package' => setting('trial_package'),
]; ];
$packages = Package::pluck('name', 'id'); $packages = Package::pluck('name', 'id');
@@ -47,10 +53,14 @@ class SettingController extends Controller
'documentation', 'documentation',
'default_package', 'default_package',
'receive_email_on_server_creation', 'receive_email_on_server_creation',
'receive_email_on_site_creation',
'isolate_per_site_per_user', 'isolate_per_site_per_user',
'enable_api', 'enable_api',
'api_token', 'api_token',
'default_language' 'default_language',
'rotate_logs_after',
'trial',
'trial_package'
]) as $key => $value) { ]) as $key => $value) {
if ($key === 'api_token') { if ($key === 'api_token') {
$value = encrypt($value); $value = encrypt($value);
@@ -87,4 +97,51 @@ class SettingController extends Controller
return redirect()->route('admin.settings')->with('success', __('Settings have been updated')); return redirect()->route('admin.settings')->with('success', __('Settings have been updated'));
} }
public function terms()
{
return inertia('Admin/Terms', [
'terms_settings' => [
'logo' => setting('logo'),
'name' => setting('name'),
'terms_required' => setting('accept_terms_required'),
'terms' => setting('terms'),
'privacy' => setting('privacy')
]
]);
}
public function updateTerms(Request $request)
{
setting(['accept_terms_required' => $request->input('terms_required'),]);
setting(['terms' => $request->input('terms'),]);
setting(['privacy' => $request->input('privacy'),]);
return redirect()->route('admin.settings.terms')->with('success', __('Terms have been updated'));
}
public function termsTemplate(Request $request)
{
$template = file_get_contents(storage_path('templates/terms-of-service.md'));
$template = str_replace([
'{NAME}',
'{WEBSITE}',
'{DATE}'
], [
setting('name'),
config('app.url'),
date('Y-m-d')
], $template);
return ['content' => $template];
}
public function removeLogo(Request $request)
{
Storage::delete(setting('logo'));
setting(['logo' => null]);
return redirect()->back()->with('success', 'Logo has ben removed');
}
} }

View File

@@ -2,6 +2,7 @@
namespace App\Http\Controllers\Admin; namespace App\Http\Controllers\Admin;
use App\Http\Resources\Admin\SiteResource;
use App\Models\Site; use App\Models\Site;
use App\Models\User; use App\Models\User;
use Illuminate\Http\Request; use Illuminate\Http\Request;
@@ -10,6 +11,23 @@ use App\Http\Requests\Admin\ServerAttachRequest;
class SiteController extends Controller class SiteController extends Controller
{ {
public function index()
{
return inertia('Admin/Sites/Index', [
'filters' => request()->all('search'),
'sites' => SiteResource::collection(
Site::query()
->when(request()->input('search'), function ($query, $value) {
return $query->where('domain', 'like', '%' . $value . '%');
})
->with('server:id,name', 'users:id,name')
->latest()
->paginate(config('core.pagination.per_page'))
->withQueryString()
)
]);
}
public function edit($id) public function edit($id)
{ {
$site = Site::findOrFail($id); $site = Site::findOrFail($id);

View File

@@ -3,7 +3,6 @@
namespace App\Http\Controllers\Admin; namespace App\Http\Controllers\Admin;
use App\Models\Provider; use App\Models\Provider;
use App\Services\Ploi\Ploi;
use Illuminate\Http\Request; use Illuminate\Http\Request;
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
@@ -11,15 +10,15 @@ class SynchronizeProviderController extends Controller
{ {
public function index() public function index()
{ {
if (config('app.demo')) { if ($this->isDemo()) {
return redirect('/')->with('info', __('This feature is not available in demo mode.')); return redirect('/')->with('info', __('This feature is not available in demo mode.'));
} }
$ploi = new Ploi(config('services.ploi.token')); $availableProviders = $this->getPloi()->user()->serverProviders()->getData();
$availableProviders = $ploi->user()->serverProviders()->getData(); $currentProviders = Provider::query()
->whereNotIn('id', array_keys((array)$availableProviders))
$currentProviders = Provider::whereNotIn('id', array_keys((array)$availableProviders))->get(); ->get();
return inertia('Admin/Services/Providers', [ return inertia('Admin/Services/Providers', [
'availableProviders' => $availableProviders, 'availableProviders' => $availableProviders,
@@ -29,7 +28,7 @@ class SynchronizeProviderController extends Controller
public function synchronize(Request $request, $providerId) public function synchronize(Request $request, $providerId)
{ {
$ploiProvider = (new Ploi)->user()->serverProviders($providerId)->getData(); $ploiProvider = $this->getPloi()->user()->serverProviders($providerId)->getData();
$provider = Provider::updateOrCreate([ $provider = Provider::updateOrCreate([
'ploi_id' => $ploiProvider->id, 'ploi_id' => $ploiProvider->id,

View File

@@ -3,7 +3,6 @@
namespace App\Http\Controllers\Admin; namespace App\Http\Controllers\Admin;
use App\Models\Server; use App\Models\Server;
use App\Services\Ploi\Ploi;
use Illuminate\Http\Request; use Illuminate\Http\Request;
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
@@ -11,15 +10,15 @@ class SynchronizeServerController extends Controller
{ {
public function index() public function index()
{ {
if (config('app.demo')) { if ($this->isDemo()) {
return redirect('/')->with('info', __('This feature is not available in demo mode.')); return redirect('/')->with('info', __('This feature is not available in demo mode.'));
} }
$ploi = new Ploi(config('services.ploi.token')); $availableServers = $this->getPloi()->synchronize()->servers()->getData();
$availableServers = $ploi->synchronize()->servers()->getData(); $currentServers = Server::query()
->whereNotIn('id', array_keys((array)$availableServers))
$currentServers = Server::whereNotIn('id', array_keys((array)$availableServers))->get(); ->get();
return inertia('Admin/Services/Servers', [ return inertia('Admin/Services/Servers', [
'availableServers' => $availableServers, 'availableServers' => $availableServers,
@@ -29,15 +28,37 @@ class SynchronizeServerController extends Controller
public function synchronizeServer(Request $request) public function synchronizeServer(Request $request)
{ {
Server::updateOrCreate([ Server::query()
'ploi_id' => $request->input('id') ->updateOrCreate([
], [ 'ploi_id' => $request->input('id')
'status' => $request->input('status'), ], [
'name' => $request->input('name'), 'status' => $request->input('status'),
'ip' => $request->input('ip_address'), 'name' => $request->input('name'),
'ssh_port' => $request->input('ssh_port', 22), 'ip' => $request->input('ip_address'),
'internal_ip' => $request->input('internal_ip'), 'ssh_port' => $request->input('ssh_port', 22),
'available_php_versions' => $request->input('installed_php_versions') 'internal_ip' => $request->input('internal_ip'),
]); 'available_php_versions' => $request->input('installed_php_versions')
]);
}
public function synchronizeAll(Request $request)
{
$availableServers = $this->getPloi()->synchronize()->servers()->getData();
foreach ($availableServers as $availableServer) {
Server::query()
->updateOrCreate([
'ploi_id' => $availableServer->id
], [
'status' => $availableServer->status,
'name' => $availableServer->name,
'ip' => $availableServer->ip_address,
'ssh_port' => $availableServer->ssh_port,
'internal_ip' => $availableServer->internal_ip,
'available_php_versions' => $availableServer->installed_php_versions
]);
}
return response('ok');
} }
} }

View File

@@ -2,12 +2,85 @@
namespace App\Http\Controllers\Admin; namespace App\Http\Controllers\Admin;
use App\Models\Site;
use App\Models\Server;
use Illuminate\Http\Request;
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
class SynchronizeSiteController extends Controller class SynchronizeSiteController extends Controller
{ {
public function index() public function index()
{ {
return inertia('Admin/Services/Sites'); if ($this->isDemo()) {
return redirect('/')->with('info', __('This feature is not available in demo mode.'));
}
$availableSites = $this->getPloi()->synchronize()->sites()->getData();
$currentSites = Site::query()
->whereNotIn('id', array_keys((array)$availableSites))
->get();
return inertia('Admin/Services/Sites', [
'availableSites' => $availableSites,
'currentSites' => $currentSites
]);
}
public function synchronizeSite(Request $request)
{
$server = Server::query()->where('ploi_id', $request->input('server_id'))->firstOrFail();
/* @var $site \App\Models\Site */
$site = Site::query()
->updateOrCreate([
'ploi_id' => $request->input('id')
], [
'domain' => $request->input('domain'),
'php_version' => $request->input('php_version'),
'project' => $request->input('project_type')
]);
$site->status = $request->input('status');
$site->server_id = $server->id;
$site->save();
$certificates = $this->getPloi()->server($request->input('server_id'))->sites($request->input('id'))->certificates()->get()->getData();
if ($certificates) {
foreach ($certificates as $certificate) {
$site->certificates()->updateOrCreate([
'ploi_id' => $certificate->id,
], [
'status' => $certificate->status,
'ploi_id' => $certificate->id,
'domain' => $certificate->domain,
'type' => $certificate->type,
]);
}
}
return response('ok');
}
public function synchronizeAll(Request $request)
{
$availableSites = $this->getPloi()->synchronize()->sites()->getData();
foreach ($availableSites as $availableSite) {
$server = Server::query()->where('ploi_id', $availableSite->server_id)->firstOrFail();
$site = Site::query()
->updateOrCreate([
'ploi_id' => $availableSite->id
], [
'domain' => $availableSite->domain,
'php_version' => $availableSite->php_version,
]);
$site->status = $availableSite->status;
$site->server_id = $server->id;
$site->save();
}
} }
} }

View File

@@ -2,31 +2,37 @@
namespace App\Http\Controllers\Admin; namespace App\Http\Controllers\Admin;
use Inertia\Response;
use Illuminate\Http\Request; use Illuminate\Http\Request;
use App\Jobs\Core\UpdateSystem; use App\Jobs\Core\UpdateSystem;
use App\Services\VersionChecker; use App\Services\VersionChecker;
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
use Illuminate\Http\RedirectResponse;
use Laravel\Horizon\Contracts\MasterSupervisorRepository;
class SystemController extends Controller class SystemController extends Controller
{ {
public function index() public function index(Request $request, MasterSupervisorRepository $masterSupervisorRepository): Response|RedirectResponse
{ {
if (config('app.demo')) { if ($request->input('flush', false)) {
return redirect('/')->with('info', __('This feature is not available in demo mode.')); app(VersionChecker::class)->flushVersionData();
return redirect()->route('admin.system')->with('success', __('Refreshed versions'));
} }
$version = (new VersionChecker)->getVersions(); $version = app(VersionChecker::class)->getVersions();
return inertia('Admin/System', [ return inertia('Admin/System', [
'version' => [ 'version' => [
'out_of_date' => $version->isOutOfDate(), 'out_of_date' => $version->isOutOfDate(),
'current' => $version->currentVersion, 'current' => $version->currentVersion,
'remote' => $version->remoteVersion 'remote' => $version->remoteVersion
] ],
'horizonRunning' => (bool) $masterSupervisorRepository->all(),
]); ]);
} }
public function update(Request $request) public function update(Request $request): RedirectResponse
{ {
dispatch(new UpdateSystem); dispatch(new UpdateSystem);

View File

@@ -11,12 +11,14 @@ class UserController extends Controller
{ {
public function index() public function index()
{ {
$users = User::with('package:id,name') $users = User::query()
->withCount('sites', 'servers')
->with('package:id,name')
->when(request()->input('search'), function ($query, $value) { ->when(request()->input('search'), function ($query, $value) {
return $query->where('name', 'like', '%' . $value . '%')->orWhere('email', 'like', '%' . $value . '%'); return $query->where('name', 'like', '%' . $value . '%')->orWhere('email', 'like', '%' . $value . '%');
}) })
->latest() ->latest()
->paginate(5); ->paginate(config('core.pagination.per_page'));
return inertia('Admin/Users/Index', [ return inertia('Admin/Users/Index', [
'filters' => request()->all('search'), 'filters' => request()->all('search'),
@@ -45,8 +47,8 @@ class UserController extends Controller
$user->save(); $user->save();
} }
if ($package = $request->input('package')) { if ($request->input('package') && Package::find($request->input('package'))) {
$user->package_id = $package; $user->package_id = $request->input('package');
$user->save(); $user->save();
} }
@@ -55,7 +57,17 @@ class UserController extends Controller
public function show($id) public function show($id)
{ {
// TODO: Implement show feature for a user $user = User::query()->findOrFail($id);
$servers = $user->servers()->withCount('sites')->latest()->paginate(config('core.pagination.per_page'), ['*'], 'page_servers');
$sites = $user->sites()->with('server:id,name')->latest()->paginate(config('core.pagination.per_page'), ['*'], 'page_sites');
return inertia('Admin/Users/Show', [
'user' => $user,
'sites' => $sites,
'servers' => $servers,
]);
} }
public function edit($id) public function edit($id)

View File

@@ -0,0 +1,30 @@
<?php
namespace App\Http\Controllers\Api;
use App\Actions\Server\CreateServerAction;
use App\DataTransferObjects\ServerData;
use App\Http\Controllers\Controller;
use Illuminate\Http\Request;
use Illuminate\Http\Response;
class ServerController extends Controller
{
public function store(Request $request): Response
{
$data = $request->validate([
'name' => ['required'],
'provider_id' => ['required'],
'provider_region_id' => ['required'],
'provider_plan_id' => ['required'],
'database_type' => ['required'],
'user_id' => ['required'],
]);
$server = app(CreateServerAction::class)->execute(
ServerData::validate($data)
);
return response(content: ['data' => ServerData::from($server)->toArray()], status: 201);
}
}

View File

@@ -0,0 +1,48 @@
<?php
namespace App\Http\Controllers\Api;
use App\Actions\Site\CreateSiteAction;
use App\DataTransferObjects\SiteData;
use App\Http\Controllers\Controller;
use App\Models\Site;
use Illuminate\Http\JsonResponse;
use Illuminate\Http\Request;
use Illuminate\Http\Response;
class SiteController extends Controller
{
public function index(): mixed
{
return SiteData::collection(Site::paginate());
}
public function store(Request $request): Response|JsonResponse
{
// Required parameters are validated at the controller level. For example in the API you need to manually pass a user_id,
// whilst in the "authenticated part" the user id is takes as Auth::id(). Validation of universal rules is done at the
// data-object level (e.g. exists:server_id).
$data = $request->validate([
'server_id' => ['required'],
'domain' => ['required'],
'user_id' => ['required'],
]);
$site = app(CreateSiteAction::class)->execute(
SiteData::validate($data)
);
$site->refresh();
return $site
? response(content: ['data' => SiteData::from($site)->toArray()], status: 201)
: response()->json([
'message' => __('It seems there is no free server room for this site to take place. Please get in touch with support to resolve this.'),
], 422);
}
public function show(Site $site): Response
{
return response(content: ['data' => SiteData::from($site)]);
}
}

View File

@@ -2,27 +2,62 @@
namespace App\Http\Controllers\Api; namespace App\Http\Controllers\Api;
use App\Models\User; use App\DataTransferObjects\UserData;
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
use App\Http\Requests\Api\UserRequest; use App\Models\User;
use App\Http\Resources\Api\UserResource; use Illuminate\Http\Request;
use Illuminate\Http\Response;
use Illuminate\Support\Arr;
class UserController extends Controller class UserController extends Controller
{ {
public function index() public function index(): mixed
{ {
return UserResource::collection(User::latest()->paginate()); return UserData::collection(User::latest()->paginate());
} }
public function store(UserRequest $request) public function show(User $user): Response
{ {
$user = User::create($request->validated()); return response(content: ['data' => UserData::from($user)], status: 200);
return new UserResource($user);
} }
public function show($id) public function store(Request $request): Response
{ {
return new UserResource(User::findOrFail($id)); $data = $request->validate([
'name' => ['required'],
'email' => ['required'],
'package_id' => ['nullable'],
]);
$userData = UserData::validate($data);
$user = User::create($userData->toArray());
return response(content: ['data' => UserData::from($user)], status: 201);
}
public function update(User $user, Request $request): Response
{
$data = $request->validate([
'name' => [],
'email' => [],
'package_id' => [],
'blocked' => [],
]);
$userData = UserData::validate($data);
$user->update(
Arr::only($userData->toArray(), array_keys($data))
);
return response(content: ['data' => UserData::from($user)], status: 200);
}
public function destroy(User $user): Response
{
$user->delete();
return response(status: 200);
} }
} }

View File

@@ -0,0 +1,27 @@
<?php
namespace App\Http\Controllers\Auth;
use App\Http\Controllers\Controller;
use App\Providers\RouteServiceProvider;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
class AuthenticateTwoFactorController extends Controller
{
public function index()
{
return inertia('Auth/ConfirmTwoFactorAuthentication');
}
public function confirm(Request $request): RedirectResponse
{
$request->validate([
'code' => 'totp'
]);
session()->put('auth.two_factor_authenticated_at', now());
return redirect()->away(RouteServiceProvider::HOME);
}
}

View File

@@ -7,6 +7,7 @@ use Illuminate\Http\Request;
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
use App\Providers\RouteServiceProvider; use App\Providers\RouteServiceProvider;
use Illuminate\Support\Facades\Validator; use Illuminate\Support\Facades\Validator;
use Illuminate\Validation\Rules\Password;
use Illuminate\Foundation\Auth\RegistersUsers; use Illuminate\Foundation\Auth\RegistersUsers;
class RegisterController extends Controller class RegisterController extends Controller
@@ -27,27 +28,55 @@ class RegisterController extends Controller
protected function validator(array $data) protected function validator(array $data)
{ {
return Validator::make($data, [ $rules = [
'name' => ['required', 'string', 'max:255'], 'name' => ['required', 'string', 'max:255'],
'email' => ['required', 'string', 'email', 'max:255', 'unique:users'], 'email' => ['required', 'string', 'email', 'max:255', 'unique:users'],
'password' => ['required', 'string', 'min:8', 'confirmed'], 'password' => [
]); 'required',
'string',
'confirmed',
Password::defaults()
],
];
if (setting('accept_terms_required')) {
$rules['terms'] = [
'accepted'
];
}
return Validator::make($data, $rules);
} }
protected function create(array $data) protected function create(array $data)
{ {
return User::create([ $fields = [
'name' => $data['name'], 'name' => $data['name'],
'email' => $data['email'], 'email' => $data['email'],
'password' => $data['password'], 'password' => $data['password'],
]); ];
if ($days = setting('trial')) {
$fields['trial_ends_at'] = now()->addDays($days);
}
return User::create($fields);
} }
protected function registered(Request $request, $user) protected function registered(Request $request, $user)
{ {
if (setting('default_package') && setting('default_package') != 'false') { if (
setting('default_package') &&
setting('default_package') != 'false' &&
!setting('trial')
) {
$user->package_id = setting('default_package'); $user->package_id = setting('default_package');
$user->save(); $user->save();
} }
if (setting('trial') && setting('trial_package')) {
$user->package_id = setting('trial_package');
$user->save();
}
} }
} }

View File

@@ -2,11 +2,17 @@
namespace App\Http\Controllers; namespace App\Http\Controllers;
use App\Traits\HasPloi;
use Illuminate\Routing\Controller as BaseController; use Illuminate\Routing\Controller as BaseController;
use Illuminate\Foundation\Validation\ValidatesRequests; use Illuminate\Foundation\Validation\ValidatesRequests;
use Illuminate\Foundation\Auth\Access\AuthorizesRequests; use Illuminate\Foundation\Auth\Access\AuthorizesRequests;
class Controller extends BaseController class Controller extends BaseController
{ {
use ValidatesRequests, AuthorizesRequests; use ValidatesRequests, AuthorizesRequests, HasPloi;
protected function isDemo()
{
return config('app.demo');
}
} }

View File

@@ -2,10 +2,29 @@
namespace App\Http\Controllers; namespace App\Http\Controllers;
use Illuminate\Support\Str;
class PageController extends Controller class PageController extends Controller
{ {
public function installationIncomplete() public function installationIncomplete()
{ {
return inertia('Core/InstallationIncomplete'); return inertia('Core/InstallationIncomplete');
} }
public function show($slug)
{
if ($slug === 'terms-of-service' && setting('terms')) {
return inertia('Pages/Terms', [
'content' => Str::markdown(setting('terms'))
]);
}
if ($slug === 'privacy-policy' && setting('privacy')) {
return inertia('Pages/Privacy', [
'content' => Str::markdown(setting('privacy'))
]);
}
abort(404);
}
} }

View File

@@ -22,7 +22,11 @@ class ProfileBillingController extends Controller
$packages = Package::query() $packages = Package::query()
->where(function ($query) { ->where(function ($query) {
return $query return $query
->where('price_monthly', '>', 0) ->where(function ($query) {
return $query
->where('price_monthly', '>', 0)
->orWhere('price_yearly', '>', 0);
})
->whereNotNull('plan_id'); ->whereNotNull('plan_id');
}) })
->when($request->input('sortBy.' . $sortByType), function ($query, $value) use ($sortByType) { ->when($request->input('sortBy.' . $sortByType), function ($query, $value) use ($sortByType) {
@@ -55,7 +59,14 @@ class ProfileBillingController extends Controller
->transform(function (Package $package) { ->transform(function (Package $package) {
$currency = $this->transformCurrency($package->currency); $currency = $this->transformCurrency($package->currency);
$package->period = 'monthly';
if ($package->price_yearly > 0) {
$package->period = 'yearly';
}
$package->price_monthly = ($currency ?? '[Unknown currency]') . number_format($package->price_monthly, 2, ',', '.'); $package->price_monthly = ($currency ?? '[Unknown currency]') . number_format($package->price_monthly, 2, ',', '.');
$package->price_yearly = ($currency ?? '[Unknown currency]') . number_format($package->price_yearly, 2, ',', '.');
return $package; return $package;
}); });
@@ -66,11 +77,14 @@ class ProfileBillingController extends Controller
return inertia('Profile/BillingError'); return inertia('Profile/BillingError');
} }
$subscription = $user->subscription();
return inertia('Profile/Billing', [ return inertia('Profile/Billing', [
'packages' => $packages, 'packages' => $packages,
'countries' => countries(), 'countries' => countries(),
'subscription' => $user->subscription('default'), 'subscription' => $subscription,
'public_key' => config('cashier.key'), 'public_key' => config('cashier.key'),
'ends' => $subscription ? Carbon::createFromTimeStamp($subscription->asStripeSubscription()->current_period_end)->format('F jS, Y') ?? null : null,
'data_client_secret' => $clientSecret, 'data_client_secret' => $clientSecret,
'card' => [ 'card' => [
'last_four' => $user->card_last_four, 'last_four' => $user->card_last_four,
@@ -111,6 +125,16 @@ class ProfileBillingController extends Controller
return redirect()->route('profile.billing.index')->with('success', 'Your card has been added, you can now update your plan'); return redirect()->route('profile.billing.index')->with('success', 'Your card has been added, you can now update your plan');
} }
public function deleteCard(Request $request)
{
/** @var User $user */
$user = $request->user();
$user->deletePaymentMethods();
return redirect()->route('profile.billing.index')->with('success', 'Your credit card has been removed from your account');
}
public function updatePlan(Request $request) public function updatePlan(Request $request)
{ {
/** @var User $user */ /** @var User $user */
@@ -198,8 +222,10 @@ class ProfileBillingController extends Controller
Package::CURRENCY_AUD => 'AUD $', Package::CURRENCY_AUD => 'AUD $',
Package::CURRENCY_GBP => 'GBP £', Package::CURRENCY_GBP => 'GBP £',
Package::CURRENCY_INR => 'INR ₹', Package::CURRENCY_INR => 'INR ₹',
Package::CURRENCY_THB => 'THB ',
Package::CURRENCY_BRL=> 'BRL R$ ',
]; ];
return $currencies[strtolower($key)]; return $currencies[strtolower($key)] ?? '$';
} }
} }

View File

@@ -2,10 +2,11 @@
namespace App\Http\Controllers\Profile; namespace App\Http\Controllers\Profile;
use Illuminate\Http\Request;
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
use App\Http\Requests\UserProfileRequest; use App\Http\Requests\UserProfileRequest;
use App\Http\Resources\UserProfileResource; use App\Http\Resources\UserProfileResource;
use App\Models\User;
use Illuminate\Http\Request;
class ProfileController extends Controller class ProfileController extends Controller
{ {
@@ -37,4 +38,21 @@ class ProfileController extends Controller
return $mode; return $mode;
} }
public function destroy(Request $request)
{
/* @var $user User */
$user = $request->user();
$user->sites()->detach();
$user->servers()->detach();
$user->supportTicketReplies()->delete();
$user->supportTickets()->delete();
$user->delete();
auth()->logout();
return redirect()->route('login');
}
} }

View File

@@ -1,27 +0,0 @@
<?php
namespace App\Http\Controllers\Profile;
use Illuminate\Http\Request;
use App\Rules\MatchOldPassword;
use App\Http\Controllers\Controller;
class ProfilePasswordController extends Controller
{
public function index()
{
return inertia('Profile/Security');
}
public function update(Request $request)
{
$request->validate([
'current_password' => ['required', new MatchOldPassword],
'password' => ['required', 'string', 'min:8', 'confirmed'],
]);
$request->user()->update(['password' => $request->input('password')]);
return redirect()->route('profile.security.index')->with('success', __('Your password has been updated'));
}
}

View File

@@ -0,0 +1,45 @@
<?php
namespace App\Http\Controllers\Profile;
use App\Http\Controllers\Controller;
use App\Rules\MatchOldPassword;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
class ProfileSecurityController extends Controller
{
public function index(Request $request)
{
$twoFactorAuth = $request->user()->twoFactorAuth()->first();
// Only show recovery codes once.
$recoveryCodes = $twoFactorAuth?->recovery_codes_generated_at?->gt(now()->subSecond(1))
? $request->user()->getRecoveryCodes()
: [];
return inertia('Profile/Security', [
'twoFactor' => [
'secret' => [
'qr_code' => $twoFactorAuth?->toQr(),
'uri' => $twoFactorAuth?->toUri(),
'string' => $twoFactorAuth?->toString(),
],
'recoveryCodes' => $recoveryCodes,
'enabled' => $request->user()->hasTwoFactorEnabled(),
],
]);
}
public function updatePassword(Request $request): RedirectResponse
{
$request->validate([
'current_password' => ['required', new MatchOldPassword],
'password' => ['required', 'string', 'min:8', 'confirmed'],
]);
$request->user()->update(['password' => $request->input('password')]);
return redirect()->route('profile.security.index')->with('success', __('Your password has been updated'));
}
}

View File

@@ -0,0 +1,31 @@
<?php
namespace App\Http\Controllers\Profile\TwoFactorAuthentication;
use App\Http\Controllers\Controller;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
class ConfirmTwoFactorAuthenticationController extends Controller
{
public function __invoke(Request $request): RedirectResponse
{
$request->validate([
'code' => 'required|numeric',
]);
$activated = $request->user()->confirmTwoFactorAuth($request->input('code'));
if ($activated) {
session()->put('auth.two_factor_authenticated_at', now());
return redirect()
->route('profile.security.index')
->with('success', __('Your two factor authentication has been activated'));
}
return redirect()
->route('profile.security.index')
->with('error', __('Please check your confirmation code'));
}
}

View File

@@ -0,0 +1,19 @@
<?php
namespace App\Http\Controllers\Profile\TwoFactorAuthentication;
use App\Http\Controllers\Controller;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
class RegenerateRecoveryCodesController extends Controller
{
public function __invoke(Request $request): RedirectResponse
{
$request->user()->generateRecoveryCodes();
return redirect()
->route('profile.security.index')
->with('success', __('Your recovery codes have been regenerated'));
}
}

View File

@@ -0,0 +1,30 @@
<?php
namespace App\Http\Controllers\Profile\TwoFactorAuthentication;
use Illuminate\Http\Request;
use App\Http\Controllers\Controller;
use Illuminate\Http\RedirectResponse;
class TwoFactorAuthenticationController extends Controller
{
public function create(Request $request): RedirectResponse
{
if ($request->user()->hasTwoFactorEnabled()) {
return redirect()->route('profile.security.index')->with('error', __('Your two factor authentication is already enabled'));
}
$request->user()->createTwoFactorAuth();
return redirect()->route('profile.security.index')->with('success', __('Your two factor authentication has been enabled'));
}
public function destroy(Request $request): RedirectResponse
{
$request->user()->disableTwoFactorAuth();
return redirect()
->route('profile.security.index')
->with('success', __('Two factor authentication has been disabled'));
}
}

View File

@@ -2,16 +2,15 @@
namespace App\Http\Controllers; namespace App\Http\Controllers;
use App\Models\User; use App\Actions\Server\CreateServerAction;
use Illuminate\Http\Request; use App\DataTransferObjects\ServerData;
use App\Jobs\Servers\CreateServer;
use App\Jobs\Servers\DeleteServer;
use App\Http\Requests\ServerRequest;
use Illuminate\Support\Facades\Mail;
use App\Http\Resources\ServerResource;
use App\Mail\Server\ServerCreatedEmail;
use App\Http\Requests\ServerUpdateRequest; use App\Http\Requests\ServerUpdateRequest;
use App\Mail\Admin\Server\AdminServerCreatedEmail; use App\Http\Resources\ServerResource;
use App\Jobs\Servers\DeleteServer;
use App\Models\Server;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Auth;
class ServerController extends Controller class ServerController extends Controller
{ {
@@ -30,33 +29,23 @@ class ServerController extends Controller
]); ]);
} }
public function store(ServerRequest $request) public function store(Request $request): RedirectResponse
{ {
$provider = $request->user()->package->providers()->findOrFail($request->input('provider')); $this->authorize('create', Server::class);
$region = $provider->regions()->findOrFail($request->input('region'));
$plan = $provider->plans()->findOrFail($request->input('plan'));
/* @var $server \App\Models\Server */ $data = $request->validate([
$server = $request->user()->servers()->create([ 'name' => ['required'],
'name' => $request->input('name') 'provider_id' => ['required'],
'provider_region_id' => ['required'],
'provider_plan_id' => ['required'],
'database_type' => ['required'],
]); ]);
$server->provider()->associate($provider); $data['user_id'] = Auth::id();
$server->providerRegion()->associate($region);
$server->providerPlan()->associate($plan);
$server->save();
dispatch(new CreateServer($server)); app(CreateServerAction::class)->execute(
ServerData::validate($data)
Mail::to($request->user())->send(new ServerCreatedEmail($request->user(), $server)); );
if (setting('receive_email_on_server_creation')) {
$admins = User::query()->where('role', User::ADMIN)->get();
foreach ($admins as $admin) {
Mail::to($admin)->send(new AdminServerCreatedEmail($request->user(), $server));
}
}
return redirect()->route('servers.index'); return redirect()->route('servers.index');
} }
@@ -65,6 +54,10 @@ class ServerController extends Controller
{ {
$server = auth()->user()->servers()->findOrFail($id); $server = auth()->user()->servers()->findOrFail($id);
if ($server->status === Server::STATUS_BUSY) {
return redirect()->back()->with('info', 'This server is currently busy, please check back later.');
}
return inertia('Servers/Show', [ return inertia('Servers/Show', [
'server' => $server, 'server' => $server,
'sites' => $server->sites()->latest()->paginate(5, ['*'], 'sites_per_page'), 'sites' => $server->sites()->latest()->paginate(5, ['*'], 'sites_per_page'),
@@ -104,9 +97,21 @@ class ServerController extends Controller
{ {
$provider = $request->user()->package->providers()->findOrFail($providerId); $provider = $request->user()->package->providers()->findOrFail($providerId);
$regions = $provider->regions()
->when($provider->allowed_regions, function ($query) use ($provider) {
return $query->whereIn('id', $provider->allowed_regions);
})
->pluck('label', 'id');
$plans = $provider->plans()
->when($provider->allowed_plans, function ($query) use ($provider) {
return $query->whereIn('id', $provider->allowed_plans);
})
->pluck('label', 'id');
return [ return [
'regions' => $provider->regions()->pluck('label', 'id'), 'regions' => $regions,
'plans' => $provider->plans()->pluck('label', 'id'), 'plans' => $plans,
]; ];
} }
} }

View File

@@ -0,0 +1,43 @@
<?php
namespace App\Http\Controllers;
use App\Http\Requests\SiteAliasRequest;
use App\Http\Resources\SiteAliasResource;
use App\Jobs\Aliases\CreateAlias;
use App\Jobs\Aliases\DeleteAlias;
class SiteAliasController extends Controller
{
public function index($id)
{
$site = auth()->user()->sites()->findOrFail($id);
return inertia('Sites/Aliases', [
'site' => $site,
'aliases' => $site->aliases
]);
}
public function store(SiteAliasRequest $request, $id)
{
$site = $request->user()->sites()->findOrFail($id);
$site->addAlias($request->input('domain'));
dispatch(new CreateAlias($site, $request->input('domain'), $request->boolean('request_new_certificate')));
return redirect()->route('sites.aliases.index', $id)->with('success', __('Alias has been created'));
}
public function destroy($id, $alias)
{
$site = auth()->user()->sites()->findOrFail($id);
dispatch(new DeleteAlias($site, $alias));
$site->removeAlias($alias);
return redirect()->route('sites.aliases.index', $id)->with('success', __('Alias has been deleted'));
}
}

View File

@@ -25,6 +25,9 @@ class SiteCertificateController extends Controller
$certificate = $site->certificates()->create([ $certificate = $site->certificates()->create([
'domain' => $request->input('domain'), 'domain' => $request->input('domain'),
'type' => $request->input('type', 'letsencrypt'),
'certificate' => $request->input('certificate'),
'private' => $request->input('private')
]); ]);
$certificate->server_id = $site->server_id; $certificate->server_id = $site->server_id;

View File

@@ -2,14 +2,22 @@
namespace App\Http\Controllers; namespace App\Http\Controllers;
use App\Models\Server;
use Illuminate\Support\Arr; use Illuminate\Support\Arr;
use Illuminate\Http\Request; use Illuminate\Http\Request;
use App\Jobs\Sites\CreateSite;
use App\Jobs\Sites\DeleteSite; use App\Jobs\Sites\DeleteSite;
use App\Http\Requests\SiteRequest;
use App\Http\Resources\SiteResource; use App\Http\Resources\SiteResource;
use Illuminate\Support\Facades\Auth;
use App\Http\Requests\SiteRequest;
use App\Jobs\Sites\CreateSite;
use App\Mail\Admin\Site\AdminSiteCreatedEmail;
use App\Models\Server;
use App\Models\Site;
use App\Models\User;
use Illuminate\Http\JsonResponse;
use Illuminate\Support\Facades\Hash; use Illuminate\Support\Facades\Hash;
use App\DataTransferObjects\SiteData;
use Illuminate\Http\RedirectResponse;
use App\Actions\Site\CreateSiteAction;
class SiteController extends Controller class SiteController extends Controller
{ {
@@ -17,27 +25,45 @@ class SiteController extends Controller
{ {
$sites = auth()->user() $sites = auth()->user()
->sites() ->sites()
->with('server:id,name')
->when(request('server'), function ($query, $value) { ->when(request('server'), function ($query, $value) {
return $query->where('server_id', $value); return $query->where('server_id', $value);
}) })
->latest() ->latest()
->paginate(10); ->paginate(config('core.pagination.per_page'));
$availableServers = auth()->user()->servers()->pluck('name', 'id'); $availableServers = auth()->user()->servers()->pluck('name', 'id');
return inertia('Sites/Index', [ return inertia('Sites/Index', [
'sites' => SiteResource::collection($sites), 'sites' => SiteResource::collection($sites),
'availableServers' => $availableServers 'availableServers' => $availableServers,
]); ]);
} }
public function store(SiteRequest $request) public function store(SiteRequest $request): RedirectResponse
{ {
if (Site::query()->where('domain', $request->input('domain'))->exists()) {
return redirect()->back()->withErrors([
'domain' => 'This domain is not available.'
]);
}
if ($serverId = $request->input('server_id')) { if ($serverId = $request->input('server_id')) {
$server = $request->user()->servers()->findOrFail($serverId); $server = $request->user()->servers()->findOrFail($serverId);
} else { } else {
$server = Server::query() $server = Server::query()
->doesntHave('users') ->where('maximum_sites', '>', 0)
->where(function ($query) {
return $query
->where(function ($query) {
return $query->whereHas('users', function ($query) {
return $query->where('user_id', auth()->id());
});
})
->orWhere(function ($query) {
return $query->doesntHave('users');
});
})
->withCount('sites') ->withCount('sites')
->inRandomOrder() ->inRandomOrder()
->first(); ->first();
@@ -53,32 +79,31 @@ class SiteController extends Controller
]); ]);
} }
$site = $server->sites()->create($request->all()); $request->merge(['user_id' => auth()->id()]);
$request->user()->sites()->save($site); $site = app(CreateSiteAction::class)->execute(
SiteData::validate($request)
);
dispatch(new CreateSite($site)); return $site
? redirect()->route('sites.index')->with('success', __('Your website is being created'))
$request->user()->systemLogs()->create([ : redirect()->back()->withErrors([
'title' => 'New site :site created', 'domain' => __('It seems there is no free server room for this site to take place. Please get in touch with support to resolve this.'),
'description' => 'A new site has been created' ]);
])->model()->associate($site)->save();
return redirect()->route('sites.index')->with('success', __('Your website is being created'));
} }
public function show($id) public function show($id)
{ {
$site = auth()->user()->sites()->findOrFail($id); $site = auth()->user()->sites()->findOrFail($id);
if (!$site->isActive()) { if (! $site->isActive()) {
return redirect()->route('sites.index')->with('info', __('This site does not seem to be active, please wait for the process to finish')); return redirect()->route('sites.index')->with('info', __('This site does not seem to be active, please wait for the process to finish'));
} }
return inertia('Sites/Show', [ return inertia('Sites/Show', [
'site' => $site, 'site' => $site,
'system_user' => $site->getSystemUser(false), 'system_user' => $site->getSystemUser(false),
'ip_address' => $site->server->ip 'ip_address' => $site->server->ip,
]); ]);
} }
@@ -100,14 +125,14 @@ class SiteController extends Controller
if ($request->user()->requires_password_for_ftp) { if ($request->user()->requires_password_for_ftp) {
$this->validate($request, ['password' => 'required|string']); $this->validate($request, ['password' => 'required|string']);
if (!Hash::check($request->input('password'), $request->user()->password)) { if (! Hash::check($request->input('password'), $request->user()->password)) {
return response([ return response([
'message' => 'The given data was invalid', 'message' => 'The given data was invalid',
'errors' => [ 'errors' => [
'password' => [ 'password' => [
trans('auth.failed') trans('auth.failed'),
] ],
] ],
], 422); ], 422);
} }
} }

View File

@@ -2,10 +2,11 @@
namespace App\Http\Controllers; namespace App\Http\Controllers;
use App\Jobs\Databases\CreateDatabase;
use App\Jobs\Databases\DeleteDatabase;
use App\Http\Requests\SiteDatabaseRequest; use App\Http\Requests\SiteDatabaseRequest;
use App\Http\Resources\SiteDatabaseResource; use App\Http\Resources\SiteDatabaseResource;
use App\Jobs\Databases\CreateDatabase;
use App\Jobs\Databases\DeleteDatabase;
use Illuminate\Support\Str;
class SiteDatabaseController extends Controller class SiteDatabaseController extends Controller
{ {
@@ -24,11 +25,11 @@ class SiteDatabaseController extends Controller
$site = auth()->user()->sites()->findOrFail($id); $site = auth()->user()->sites()->findOrFail($id);
$database = $site->databases()->create([ $database = $site->databases()->create([
'name' => $request->input('name') 'name' => Str::of($site->domain)->limit(8)->remove(['.', '-'])->lower()->append('_')->append($request->input('name'))
]); ]);
$database->users()->create([ $database->users()->create([
'name' => $request->input('user_name'), 'name' => $request->input('user_name', ),
]); ]);
$database->server_id = $site->server_id; $database->server_id = $site->server_id;

View File

@@ -16,7 +16,7 @@ class Kernel extends HttpKernel
protected $middleware = [ protected $middleware = [
// \App\Http\Middleware\TrustHosts::class, // \App\Http\Middleware\TrustHosts::class,
\App\Http\Middleware\TrustProxies::class, \App\Http\Middleware\TrustProxies::class,
\Fruitcake\Cors\HandleCors::class, \Illuminate\Http\Middleware\HandleCors::class,
\App\Http\Middleware\PreventRequestsDuringMaintenance::class, \App\Http\Middleware\PreventRequestsDuringMaintenance::class,
\Illuminate\Foundation\Http\Middleware\ValidatePostSize::class, \Illuminate\Foundation\Http\Middleware\ValidatePostSize::class,
\App\Http\Middleware\TrimStrings::class, \App\Http\Middleware\TrimStrings::class,
@@ -65,9 +65,11 @@ class Kernel extends HttpKernel
'auth' => \App\Http\Middleware\Authenticate::class, 'auth' => \App\Http\Middleware\Authenticate::class,
'auth.basic' => \Illuminate\Auth\Middleware\AuthenticateWithBasicAuth::class, 'auth.basic' => \Illuminate\Auth\Middleware\AuthenticateWithBasicAuth::class,
'auth.blocked' => \App\Http\Middleware\UserBlocked::class, 'auth.blocked' => \App\Http\Middleware\UserBlocked::class,
'auth.2fa' => \App\Http\Middleware\EnforceTwoFactorAuthenticationIfEnabled::class,
'bindings' => \Illuminate\Routing\Middleware\SubstituteBindings::class, 'bindings' => \Illuminate\Routing\Middleware\SubstituteBindings::class,
'cache.headers' => \Illuminate\Http\Middleware\SetCacheHeaders::class, 'cache.headers' => \Illuminate\Http\Middleware\SetCacheHeaders::class,
'can' => \Illuminate\Auth\Middleware\Authorize::class, 'can' => \Illuminate\Auth\Middleware\Authorize::class,
'has.access' => \App\Http\Middleware\HasAccessToThisGroup::class,
'guest' => \App\Http\Middleware\RedirectIfAuthenticated::class, 'guest' => \App\Http\Middleware\RedirectIfAuthenticated::class,
'global.api.authenticated' => \App\Http\Middleware\GlobalApiAuthenticated::class, 'global.api.authenticated' => \App\Http\Middleware\GlobalApiAuthenticated::class,
'password.confirm' => \Illuminate\Auth\Middleware\RequirePassword::class, 'password.confirm' => \Illuminate\Auth\Middleware\RequirePassword::class,

View File

@@ -10,7 +10,7 @@ class Demo
protected $safeRoutes = [ protected $safeRoutes = [
'login', 'login',
'logout', 'logout',
//'profile/toggle-theme' 'profile/toggle-theme'
]; ];
protected $allowedIps = [ protected $allowedIps = [

View File

@@ -0,0 +1,30 @@
<?php
namespace App\Http\Middleware;
use Closure;
use Illuminate\Http\Request;
use Illuminate\Support\Carbon;
use Laragear\TwoFactor\Facades\Auth2FA;
class EnforceTwoFactorAuthenticationIfEnabled
{
public function handle(Request $request, Closure $next): mixed
{
if (auth()->guest()) {
return $next($request);
}
if (! $request->user()->hasTwoFactorEnabled()) {
return $next($request);
}
$twoFactorAuthenticatedAt = session()->get('auth.two_factor_authenticated_at');
if ($twoFactorAuthenticatedAt && Carbon::parse($twoFactorAuthenticatedAt)->gt(now()->subHours(3))) {
return $next($request);
}
return redirect()->route('auth.confirm-2fa.index');
}
}

View File

@@ -8,28 +8,27 @@ use App\Services\Ploi\Exceptions\Http\Unauthenticated;
class GlobalApiAuthenticated class GlobalApiAuthenticated
{ {
/** public function handle(Request $request, Closure $next): mixed
* Handle an incoming request.
*
* @param \Illuminate\Http\Request $request
* @param \Closure $next
* @return mixed
*/
public function handle(Request $request, Closure $next)
{ {
if (!$this->isAuthenticated($request)) { abort_unless($this->hasApiEnabled(), 404);
abort_unless($this->isAuthenticated($request), 403);
if (! $this->isAuthenticated($request)) {
throw new Unauthenticated('Unauthenticated for global access.'); throw new Unauthenticated('Unauthenticated for global access.');
} }
return $next($request); return $next($request);
} }
protected function hasApiEnabled(): bool
{
return setting('enable_api') && (bool) setting('api_token');
}
protected function isAuthenticated(Request $request) protected function isAuthenticated(Request $request)
{ {
return return $request->bearerToken()
setting('enable_api') && && $request->bearerToken() === decrypt(setting('api_token'));
setting('api_token') &&
$request->bearerToken() &&
$request->bearerToken() === decrypt(setting('api_token'));
} }
} }

View File

@@ -3,12 +3,12 @@
namespace App\Http\Middleware; namespace App\Http\Middleware;
use App\Models\Alert; use App\Models\Alert;
use Inertia\Middleware;
use Illuminate\Support\Arr;
use App\Models\UserProvider; use App\Models\UserProvider;
use Illuminate\Http\Request; use Illuminate\Http\Request;
use Illuminate\Support\Arr;
use Illuminate\Support\Facades\Auth; use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\Session; use Illuminate\Support\Facades\Session;
use Inertia\Middleware;
class HandleInertiaRequests extends Middleware class HandleInertiaRequests extends Middleware
{ {
@@ -16,7 +16,7 @@ class HandleInertiaRequests extends Middleware
* Determines the current asset version. * Determines the current asset version.
* *
* @see https://inertiajs.com/asset-versioning * @see https://inertiajs.com/asset-versioning
* @param \Illuminate\Http\Request $request * @param Request $request
* @return string|null * @return string|null
*/ */
public function version(Request $request) public function version(Request $request)
@@ -28,7 +28,7 @@ class HandleInertiaRequests extends Middleware
* Defines the props that are shared by default. * Defines the props that are shared by default.
* *
* @see https://inertiajs.com/shared-data * @see https://inertiajs.com/shared-data
* @param \Illuminate\Http\Request $request * @param Request $request
* @return array * @return array
*/ */
public function share(Request $request) public function share(Request $request)
@@ -68,13 +68,14 @@ class HandleInertiaRequests extends Middleware
] : null, ] : null,
'package' => auth()->user() && auth()->user()->package ? [ 'package' => auth()->user() && auth()->user()->package ? [
'name' => auth()->user()->package->name, 'name' => auth()->user()->package->name,
'maximum_sites' => auth()->user()->package->maximum_sites 'maximum_sites' => auth()->user()->package->maximum_sites,
'trial' => auth()->user()->onTrial()
] : [ ] : [
'name' => __('None') 'name' => __('None')
], ],
'can' => $can, 'can' => $can,
'integrations' => [ 'integrations' => [
'cloudflare' => (bool)auth()->user() ? auth()->user()->providers()->where('type', UserProvider::TYPE_CLOUDFLARE)->count() : false, 'cloudflare' => (bool) auth()->user() ? auth()->user()->providers()->where('type', UserProvider::TYPE_CLOUDFLARE)->count() : false,
] ]
]; ];
}, },
@@ -87,7 +88,10 @@ class HandleInertiaRequests extends Middleware
'documentation' => setting('documentation', false), 'documentation' => setting('documentation', false),
'logo' => setting('logo'), 'logo' => setting('logo'),
'allow_registration' => setting('allow_registration'), 'allow_registration' => setting('allow_registration'),
'billing' => config('cashier.key') && config('cashier.secret') 'billing' => config('cashier.key') && config('cashier.secret'),
'has_terms' => (bool)setting('terms'),
'has_privacy' => (bool)setting('privacy'),
'accept_terms_required' => (bool)setting('accept_terms_required')
]; ];
}, },
'flash' => function () { 'flash' => function () {
@@ -121,7 +125,7 @@ class HandleInertiaRequests extends Middleware
} }
return [ return [
'message' => $alert->message, 'message_html' => $alert->messageHtml,
'type' => $alert->type 'type' => $alert->type
]; ];
} }

View File

@@ -0,0 +1,27 @@
<?php
namespace App\Http\Middleware;
use Closure;
use Illuminate\Support\Arr;
use Illuminate\Http\Request;
class HasAccessToThisGroup
{
public function handle(Request $request, Closure $next, $group)
{
if ($group === 'servers') {
$package = $request->user()->package ?? [];
if (
!Arr::get($package->server_permissions, 'create', false) &&
!Arr::get($package->server_permissions, 'update', false) &&
!Arr::get($package->server_permissions, 'delete', false)
) {
abort(404);
}
}
return $next($request);
}
}

View File

@@ -2,15 +2,15 @@
namespace App\Http\Middleware; namespace App\Http\Middleware;
use Illuminate\Http\Middleware\TrustProxies as Middleware;
use Illuminate\Http\Request; use Illuminate\Http\Request;
use Fideloper\Proxy\TrustProxies as Middleware;
class TrustProxies extends Middleware class TrustProxies extends Middleware
{ {
/** /**
* The trusted proxies for this application. * The trusted proxies for this application.
* *
* @var array|string|null * @var array<int, string>|string|null
*/ */
protected $proxies; protected $proxies;
@@ -19,5 +19,10 @@ class TrustProxies extends Middleware
* *
* @var int * @var int
*/ */
protected $headers = Request::HEADER_X_FORWARDED_ALL; protected $headers =
Request::HEADER_X_FORWARDED_FOR |
Request::HEADER_X_FORWARDED_HOST |
Request::HEADER_X_FORWARDED_PORT |
Request::HEADER_X_FORWARDED_PROTO |
Request::HEADER_X_FORWARDED_AWS_ELB;
} }

View File

@@ -41,6 +41,8 @@ class PackageRequest extends FormRequest
Package::CURRENCY_CAD, Package::CURRENCY_CAD,
Package::CURRENCY_GBP, Package::CURRENCY_GBP,
Package::CURRENCY_INR, Package::CURRENCY_INR,
Package::CURRENCY_THB,
Package::CURRENCY_BRL,
]) ])
], ],
'maximum_sites' => [ 'maximum_sites' => [
@@ -58,7 +60,11 @@ class PackageRequest extends FormRequest
], ],
'price_monthly' => [ 'price_monthly' => [
'nullable', 'nullable',
'numeric' 'numeric',
],
'price_yearly' => [
'nullable',
'numeric',
], ],
'server_permissions' => [ 'server_permissions' => [
'array' 'array'
@@ -78,8 +84,12 @@ class PackageRequest extends FormRequest
$merge['price_monthly'] = 0.000; $merge['price_monthly'] = 0.000;
} }
if (!$this->price_yearly) {
$merge['price_yearly'] = 0.000;
}
// If we don't have the currency filled in, merge a default // If we don't have the currency filled in, merge a default
if (!$this->price_monthly) { if (!$this->price_monthly || !$this->price_yearly) {
$merge['currency'] = Package::CURRENCY_USD; $merge['currency'] = Package::CURRENCY_USD;
} }

View File

@@ -28,6 +28,14 @@ class ProviderRequest extends FormRequest
'required', 'required',
'string', 'string',
'max:255' 'max:255'
],
'allowed_plans' => [
'nullable',
'array'
],
'allowed_regions' => [
'nullable',
'array'
] ]
]; ];
} }

View File

@@ -39,6 +39,10 @@ class SettingRequest extends FormRequest
'nullable', 'nullable',
'image', 'image',
'max:2000' 'max:2000'
],
'trial_package' => [
'required_with:trial'
] ]
]; ];
} }

View File

@@ -0,0 +1,15 @@
<?php
namespace App\Http\Requests\Api;
use Illuminate\Foundation\Http\FormRequest;
class SiteRequest extends FormRequest
{
public function rules(): array
{
return [
'server_id'
];
}
}

View File

@@ -1,41 +0,0 @@
<?php
namespace App\Http\Requests\Api;
use Illuminate\Foundation\Http\FormRequest;
class UserRequest extends FormRequest
{
/**
* Determine if the user is authorized to make this request.
*
* @return bool
*/
public function authorize()
{
return $this->bearerToken() && $this->bearerToken() === setting('api_token');
}
/**
* Get the validation rules that apply to the request.
*
* @return array
*/
public function rules()
{
return [
'name' => [
'required',
'string',
'max:255'
],
'email' => [
'required',
'string',
'email',
'max:255',
'unique:users'
]
];
}
}

View File

@@ -4,6 +4,7 @@ namespace App\Http\Requests;
use App\Models\Server; use App\Models\Server;
use Illuminate\Foundation\Http\FormRequest; use Illuminate\Foundation\Http\FormRequest;
use Illuminate\Validation\Rule;
class ServerRequest extends FormRequest class ServerRequest extends FormRequest
{ {
@@ -24,6 +25,9 @@ class ServerRequest extends FormRequest
*/ */
public function rules() public function rules()
{ {
return [
];
return [ return [
'name' => [ 'name' => [
'required', 'required',
@@ -45,6 +49,11 @@ class ServerRequest extends FormRequest
'required', 'required',
'not_in:0', 'not_in:0',
'exists:provider_plans,id' 'exists:provider_plans,id'
],
'database_type' => [
'required',
'string',
Rule::in(['mysql', 'mariadb', 'postgresql', 'postgresql13'])
] ]
]; ];
} }

View File

@@ -0,0 +1,28 @@
<?php
namespace App\Http\Requests;
use App\Rules\Hostname;
use Illuminate\Foundation\Http\FormRequest;
class SiteAliasRequest extends FormRequest
{
public function authorize()
{
return auth()->check();
}
public function rules()
{
return [
'domain' => [
'required',
'string',
new Hostname,
],
'request_new_certificate' => [
'required',
]
];
}
}

View File

@@ -25,15 +25,36 @@ class SiteCertificateRequest extends FormRequest
*/ */
public function rules() public function rules()
{ {
return [ $rules = [];
'domain' => [
'required', if ($this->input('type') === 'letsencrypt') {
'string', $rules = [
Rule::unique('certificates', 'domain')->where(function ($query) { 'domain' => [
return $query->where('site_id', $this->route('site')); 'required',
}), 'string',
new LetsEncryptMatchHostWithIp($this->route('site')) Rule::unique('certificates', 'domain')->where(function ($query) {
] return $query->where('site_id', $this->route('site'));
]; }),
new LetsEncryptMatchHostWithIp($this->route('site'))
]
];
}
if ($this->input('type') === 'custom') {
$rules = [
'certificate' => [
'required',
'string',
'min:5'
],
'private' => [
'required',
'string',
'min:5'
]
];
}
return $rules;
} }
} }

View File

@@ -0,0 +1,21 @@
<?php
namespace App\Http\Resources\Admin;
use Illuminate\Http\Resources\Json\JsonResource;
class ServerResource extends JsonResource
{
public function toArray($request)
{
return [
'id' => $this->id,
'name' => $this->name,
'ip' => $this->ip,
'users' => $this->users,
'sites_count' => $this->sites_count,
'maximum_sites' => $this->maximum_sites,
'created_at' => $this->created_at->format('Y-m-d H:i:s')
];
}
}

View File

@@ -0,0 +1,13 @@
<?php
namespace App\Http\Resources\Admin;
use Illuminate\Http\Resources\Json\JsonResource;
class SiteResource extends JsonResource
{
public function toArray($request)
{
return parent::toArray($request);
}
}

View File

@@ -6,13 +6,7 @@ use Illuminate\Http\Resources\Json\JsonResource;
class UserResource extends JsonResource class UserResource extends JsonResource
{ {
/** public function toArray($request): array
* Transform the resource into an array.
*
* @param \Illuminate\Http\Request $request
* @return array
*/
public function toArray($request)
{ {
return [ return [
'id' => $this->id, 'id' => $this->id,

View File

@@ -7,12 +7,6 @@ use Illuminate\Http\Resources\Json\JsonResource;
class SiteResource extends JsonResource class SiteResource extends JsonResource
{ {
/**
* Transform the resource into an array.
*
* @param \Illuminate\Http\Request $request
* @return array
*/
public function toArray($request) public function toArray($request)
{ {
/* @var $this \App\Models\Site */ /* @var $this \App\Models\Site */
@@ -20,7 +14,11 @@ class SiteResource extends JsonResource
'id' => $this->id, 'id' => $this->id,
'status' => $this->parseStatus($this->status), 'status' => $this->parseStatus($this->status),
'domain' => $this->domain, 'domain' => $this->domain,
'php_version' => $this->php_version,
'project' => $this->project, 'project' => $this->project,
'server' => $this->server ? [
'name' => $this->server->name
] : null,
'created_at' => $this->created_at 'created_at' => $this->created_at
]; ];
} }

View File

@@ -0,0 +1,69 @@
<?php
namespace App\Jobs\Aliases;
use App\Jobs\Certificates\CreateCertificate;
use App\Jobs\Certificates\DeleteCertificate;
use App\Models\Certificate;
use App\Models\Site;
use App\Traits\HasPloi;
use Illuminate\Bus\Queueable;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Bus\Dispatchable;
use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Queue\SerializesModels;
class CreateAlias implements ShouldQueue
{
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public function __construct(
public Site $site,
public string $alias,
public bool $requestNewCertificate = false,
) {}
public function handle()
{
$this->getPloi()
->server($this->site->server->ploi_id)
->sites($this->site->ploi_id)
->aliases()
->create([$this->alias]);
if ($this->requestNewCertificate) {
$currentCertificate = $this
->site
->certificates()
->whereIn('status', [Certificate::STATUS_ACTIVE, Certificate::STATUS_BUSY])
->latest()
->first();
if (! $currentCertificate) {
return;
}
dispatch(new DeleteCertificate($this->site->server->ploi_id, $this->site->ploi_id, $currentCertificate->ploi_id));
$newCertificate = $this->site->certificates()->create([
'domain' => $currentCertificate->domain . ',' . $this->alias,
'type' => $currentCertificate->type,
'certificate' => $currentCertificate->certificate,
'private' => $currentCertificate->private
]);
$currentCertificate->delete();
$newCertificate->server_id = $this->site->server_id;
$newCertificate->save();
dispatch(new CreateCertificate($newCertificate))->delay(now()->addSeconds(5));
}
}
public function failed()
{
$this->site->aliases = array_diff($this->site->aliases, [$this->alias]);
$this->site->save();
}
}

View File

@@ -0,0 +1,35 @@
<?php
namespace App\Jobs\Aliases;
use App\Models\Site;
use App\Traits\HasPloi;
use Illuminate\Bus\Queueable;
use Illuminate\Contracts\Queue\ShouldBeUnique;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Bus\Dispatchable;
use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Queue\SerializesModels;
class DeleteAlias implements ShouldQueue
{
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public Site $site;
public string $alias;
public function __construct(Site $site, $alias)
{
$this->site = $site;
$this->alias = $alias;
}
public function handle()
{
$this->getPloi()
->server($this->site->server->ploi_id)
->sites($this->site->ploi_id)
->aliases()
->delete($this->alias);
}
}

View File

@@ -2,45 +2,46 @@
namespace App\Jobs\Apps; namespace App\Jobs\Apps;
use App\Models\Database;
use App\Models\Site; use App\Models\Site;
use App\Services\Ploi\Ploi; use App\Traits\HasPloi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue; use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Contracts\Queue\ShouldQueue; use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Bus\Dispatchable; use Illuminate\Foundation\Bus\Dispatchable;
use Illuminate\Support\Arr;
class InstallApp implements ShouldQueue class InstallApp implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $site; public Site $site;
public $type; public $type;
public $options; public $options;
/** public function __construct(Site $site, string $type = Site::PROJECT_WORDPRESS, array $options = [])
* Create a new job instance.
*
* @param Site $site
* @param string $type
* @param array $options
*/
public function __construct(Site $site, $type = Site::PROJECT_WORDPRESS, array $options = [])
{ {
$this->site = $site; $this->site = $site;
$this->type = $type; $this->type = $type;
$this->options = $options; $this->options = $options;
} }
/**
* Execute the job.
*
* @return void
*/
public function handle() public function handle()
{ {
$ploi = new Ploi(config('services.ploi.token')); $response = $this->getPloi()
->server($this->site->server->ploi_id)
->sites($this->site->ploi_id)
->app()
->install($this->type, $this->options);
$ploi->server($this->site->server->ploi_id)->sites($this->site->ploi_id)->app()->install($this->type, $this->options); if (Arr::get($this->options, 'create_database', false)) {
$database = $this->site->databases()->create([
'name' => $response->database_name,
]);
$database->status = Database::STATUS_ACTIVE;
$database->save();
}
} }
} }

View File

@@ -3,7 +3,7 @@
namespace App\Jobs\Apps; namespace App\Jobs\Apps;
use App\Models\Site; use App\Models\Site;
use App\Services\Ploi\Ploi; use App\Traits\HasPloi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue; use Illuminate\Queue\InteractsWithQueue;
@@ -12,9 +12,9 @@ use Illuminate\Foundation\Bus\Dispatchable;
class UninstallApp implements ShouldQueue class UninstallApp implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $site; public Site $site;
/** /**
* Create a new job instance. * Create a new job instance.
@@ -37,9 +37,11 @@ class UninstallApp implements ShouldQueue
return; return;
} }
$ploi = new Ploi(config('services.ploi.token')); $this->getPloi()
->server($this->site->server->ploi_id)
$ploi->server($this->site->server->ploi_id)->sites($this->site->ploi_id)->app()->uninstall($this->site->project); ->sites($this->site->ploi_id)
->app()
->uninstall($this->site->project);
$this->site->project = null; $this->site->project = null;
$this->site->save(); $this->site->save();

View File

@@ -2,8 +2,8 @@
namespace App\Jobs\Certificates; namespace App\Jobs\Certificates;
use App\Traits\HasPloi;
use App\Models\Certificate; use App\Models\Certificate;
use App\Services\Ploi\Ploi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue; use Illuminate\Queue\InteractsWithQueue;
@@ -12,38 +12,44 @@ use Illuminate\Foundation\Bus\Dispatchable;
class CreateCertificate implements ShouldQueue class CreateCertificate implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $certificate; public Certificate $certificate;
/**
* Create a new job instance.
*
* @param Certificate $certificate
*/
public function __construct(Certificate $certificate) public function __construct(Certificate $certificate)
{ {
$this->certificate = $certificate; $this->certificate = $certificate;
} }
/**
* Execute the job.
*
* @return void
*/
public function handle() public function handle()
{ {
$ploi = new Ploi(config('services.ploi.token')); if ($this->certificate->type === 'letsencrypt') {
$ploiCertificate = $this->getPloi()
->server($this->certificate->server->ploi_id)
->sites($this->certificate->site->ploi_id)
->certificates()
->create(
$this->certificate->domain
);
$ploiCertificate = $ploi->server($this->certificate->server->ploi_id) $this->certificate->ploi_id = $ploiCertificate->id;
->sites($this->certificate->site->ploi_id) $this->certificate->save();
->certificates() }
->create(
$this->certificate->domain
);
$this->certificate->ploi_id = $ploiCertificate->id; if ($this->certificate->type === 'custom') {
$this->certificate->save(); $ploiCertificate = $this->getPloi()
->server($this->certificate->server->ploi_id)
->sites($this->certificate->site->ploi_id)
->certificates()
->create(
$this->certificate->certificate,
$this->certificate->type,
$this->certificate->private
);
$this->certificate->ploi_id = $ploiCertificate->id;
$this->certificate->save();
}
// Lets fetch the status after 5 seconds // Lets fetch the status after 5 seconds
dispatch(new FetchCertificateStatus($this->certificate))->delay(now()->addSeconds(5)); dispatch(new FetchCertificateStatus($this->certificate))->delay(now()->addSeconds(5));

View File

@@ -2,7 +2,7 @@
namespace App\Jobs\Certificates; namespace App\Jobs\Certificates;
use App\Services\Ploi\Ploi; use App\Traits\HasPloi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue; use Illuminate\Queue\InteractsWithQueue;
@@ -11,7 +11,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class DeleteCertificate implements ShouldQueue class DeleteCertificate implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $serverPloiId; public $serverPloiId;
public $sitePloiId; public $sitePloiId;
@@ -42,9 +42,8 @@ class DeleteCertificate implements ShouldQueue
return; return;
} }
$ploi = new Ploi(config('services.ploi.token')); $this->getPloi()
->server($this->serverPloiId)
$ploi->server($this->serverPloiId)
->sites($this->sitePloiId) ->sites($this->sitePloiId)
->certificates() ->certificates()
->delete($this->certificatePloiId); ->delete($this->certificatePloiId);

View File

@@ -2,8 +2,8 @@
namespace App\Jobs\Certificates; namespace App\Jobs\Certificates;
use App\Traits\HasPloi;
use App\Models\Certificate; use App\Models\Certificate;
use App\Services\Ploi\Ploi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use App\Traits\JobHasThresholds; use App\Traits\JobHasThresholds;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
@@ -13,7 +13,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class FetchCertificateStatus implements ShouldQueue class FetchCertificateStatus implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, JobHasThresholds; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, JobHasThresholds, HasPloi;
public $certificate; public $certificate;
@@ -23,7 +23,7 @@ class FetchCertificateStatus implements ShouldQueue
* @param Certificate $certificate * @param Certificate $certificate
* @param int $threshold * @param int $threshold
*/ */
public function __construct(Certificate $certificate, $threshold = 0) public function __construct(Certificate $certificate, int $threshold = 0)
{ {
$this->certificate = $certificate; $this->certificate = $certificate;
$this->setThreshold($threshold); $this->setThreshold($threshold);
@@ -43,9 +43,8 @@ class FetchCertificateStatus implements ShouldQueue
return; return;
} }
$ploi = new Ploi(config('services.ploi.token')); $ploiCronjob = $this->getPloi()
->server($this->certificate->server->ploi_id)
$ploiCronjob = $ploi->server($this->certificate->server->ploi_id)
->sites($this->certificate->site->ploi_id) ->sites($this->certificate->site->ploi_id)
->certificates() ->certificates()
->get($this->certificate->ploi_id) ->get($this->certificate->ploi_id)

View File

@@ -3,7 +3,7 @@
namespace App\Jobs\Cronjobs; namespace App\Jobs\Cronjobs;
use App\Models\Cronjob; use App\Models\Cronjob;
use App\Services\Ploi\Ploi; use App\Traits\HasPloi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue; use Illuminate\Queue\InteractsWithQueue;
@@ -12,7 +12,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class CreateCronjob implements ShouldQueue class CreateCronjob implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $cronjob; public $cronjob;
@@ -35,13 +35,14 @@ class CreateCronjob implements ShouldQueue
{ {
$owner = $this->cronjob->site->users()->first(); $owner = $this->cronjob->site->users()->first();
$ploi = new Ploi(config('services.ploi.token')); $ploiCronjob = $this->getPloi()
->server($this->cronjob->server->ploi_id)
$ploiCronjob = $ploi->server($this->cronjob->server->ploi_id)->cronjobs()->create( ->cronjobs()
$this->cronjob->command, ->create(
$this->cronjob->frequency, $this->cronjob->command,
$owner->user_name $this->cronjob->frequency,
); $owner->user_name
);
$this->cronjob->ploi_id = $ploiCronjob->id; $this->cronjob->ploi_id = $ploiCronjob->id;
$this->cronjob->save(); $this->cronjob->save();

View File

@@ -2,7 +2,7 @@
namespace App\Jobs\Cronjobs; namespace App\Jobs\Cronjobs;
use App\Services\Ploi\Ploi; use App\Traits\HasPloi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue; use Illuminate\Queue\InteractsWithQueue;
@@ -11,7 +11,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class DeleteCronjob implements ShouldQueue class DeleteCronjob implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $serverPloiId; public $serverPloiId;
public $cronjobPloiId; public $cronjobPloiId;
@@ -39,8 +39,9 @@ class DeleteCronjob implements ShouldQueue
return; return;
} }
$ploi = new Ploi(config('services.ploi.token')); $this->getPloi()
->server($this->serverPloiId)
$ploi->server($this->serverPloiId)->cronjobs()->delete($this->cronjobPloiId); ->cronjobs()
->delete($this->cronjobPloiId);
} }
} }

View File

@@ -3,7 +3,7 @@
namespace App\Jobs\Cronjobs; namespace App\Jobs\Cronjobs;
use App\Models\Cronjob; use App\Models\Cronjob;
use App\Services\Ploi\Ploi; use App\Traits\HasPloi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use App\Traits\JobHasThresholds; use App\Traits\JobHasThresholds;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
@@ -13,7 +13,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class FetchCronjobStatus implements ShouldQueue class FetchCronjobStatus implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, JobHasThresholds; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, JobHasThresholds, HasPloi;
public $cronjob; public $cronjob;
@@ -23,7 +23,7 @@ class FetchCronjobStatus implements ShouldQueue
* @param Cronjob $cronjob * @param Cronjob $cronjob
* @param int $threshold * @param int $threshold
*/ */
public function __construct(Cronjob $cronjob, $threshold = 0) public function __construct(Cronjob $cronjob, int $threshold = 0)
{ {
$this->cronjob = $cronjob; $this->cronjob = $cronjob;
$this->setThreshold($threshold); $this->setThreshold($threshold);
@@ -43,9 +43,11 @@ class FetchCronjobStatus implements ShouldQueue
return; return;
} }
$ploi = new Ploi(config('services.ploi.token')); $ploiCronjob = $this->getPloi()
->server($this->cronjob->server->ploi_id)
$ploiCronjob = $ploi->server($this->cronjob->server->ploi_id)->cronjobs()->get($this->cronjob->ploi_id)->getData(); ->cronjobs()
->get($this->cronjob->ploi_id)
->getData();
if ($ploiCronjob->status !== Cronjob::STATUS_ACTIVE) { if ($ploiCronjob->status !== Cronjob::STATUS_ACTIVE) {
$this->incrementThreshold(); $this->incrementThreshold();

View File

@@ -2,8 +2,8 @@
namespace App\Jobs\Databases; namespace App\Jobs\Databases;
use App\Traits\HasPloi;
use App\Models\Database; use App\Models\Database;
use App\Services\Ploi\Ploi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue; use Illuminate\Queue\InteractsWithQueue;
@@ -12,7 +12,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class CreateDatabase implements ShouldQueue class CreateDatabase implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $database; public $database;
public $password; public $password;
@@ -38,9 +38,10 @@ class CreateDatabase implements ShouldQueue
{ {
$databaseUser = $this->database->users()->first(); $databaseUser = $this->database->users()->first();
$ploi = new Ploi(config('services.ploi.token')); $ploiDatabase = $this->getPloi()
->server($this->database->server->ploi_id)
$ploiDatabase = $ploi->server($this->database->server->ploi_id)->databases()->create($this->database->name, $databaseUser->name, $this->password); ->databases()
->create($this->database->name, $databaseUser->name, $this->password);
$this->database->ploi_id = $ploiDatabase->id; $this->database->ploi_id = $ploiDatabase->id;
$this->database->save(); $this->database->save();

View File

@@ -2,7 +2,7 @@
namespace App\Jobs\Databases; namespace App\Jobs\Databases;
use App\Services\Ploi\Ploi; use App\Traits\HasPloi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue; use Illuminate\Queue\InteractsWithQueue;
@@ -11,7 +11,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class DeleteDatabase implements ShouldQueue class DeleteDatabase implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $serverPloiId; public $serverPloiId;
public $databasePloiId; public $databasePloiId;
@@ -40,8 +40,6 @@ class DeleteDatabase implements ShouldQueue
return; return;
} }
$ploi = new Ploi(config('services.ploi.token')); $this->getPloi()->server($this->serverPloiId)->databases()->delete($this->databasePloiId);
$ploi->server($this->serverPloiId)->databases()->delete($this->databasePloiId);
} }
} }

View File

@@ -2,8 +2,8 @@
namespace App\Jobs\Databases; namespace App\Jobs\Databases;
use App\Traits\HasPloi;
use App\Models\Database; use App\Models\Database;
use App\Services\Ploi\Ploi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use App\Traits\JobHasThresholds; use App\Traits\JobHasThresholds;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
@@ -13,7 +13,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class FetchDatabaseStatus implements ShouldQueue class FetchDatabaseStatus implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, JobHasThresholds; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, JobHasThresholds, HasPloi;
public $database; public $database;
@@ -23,7 +23,7 @@ class FetchDatabaseStatus implements ShouldQueue
* @param Database $database * @param Database $database
* @param int $threshold * @param int $threshold
*/ */
public function __construct(Database $database, $threshold = 0) public function __construct(Database $database, int $threshold = 0)
{ {
$this->database = $database; $this->database = $database;
$this->setThreshold($threshold); $this->setThreshold($threshold);
@@ -43,9 +43,11 @@ class FetchDatabaseStatus implements ShouldQueue
return; return;
} }
$ploi = new Ploi(config('services.ploi.token')); $ploiDatabase = $this->getPloi()
->server($this->database->server->ploi_id)
$ploiDatabase = $ploi->server($this->database->server->ploi_id)->databases()->get($this->database->ploi_id)->getData(); ->databases()
->get($this->database->ploi_id)
->getData();
if ($ploiDatabase->status !== Database::STATUS_ACTIVE) { if ($ploiDatabase->status !== Database::STATUS_ACTIVE) {
$this->incrementThreshold(); $this->incrementThreshold();

View File

@@ -2,8 +2,8 @@
namespace App\Jobs\Redirects; namespace App\Jobs\Redirects;
use App\Traits\HasPloi;
use App\Models\Redirect; use App\Models\Redirect;
use App\Services\Ploi\Ploi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue; use Illuminate\Queue\InteractsWithQueue;
@@ -12,7 +12,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class CreateRedirect implements ShouldQueue class CreateRedirect implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $redirect; public $redirect;
@@ -33,9 +33,7 @@ class CreateRedirect implements ShouldQueue
*/ */
public function handle() public function handle()
{ {
$ploi = new Ploi(config('services.ploi.token')); $ploiRedirect = $this->getPloi()
$ploiRedirect = $ploi
->server($this->redirect->server->ploi_id) ->server($this->redirect->server->ploi_id)
->sites($this->redirect->site->ploi_id) ->sites($this->redirect->site->ploi_id)
->redirects() ->redirects()

View File

@@ -2,7 +2,7 @@
namespace App\Jobs\Redirects; namespace App\Jobs\Redirects;
use App\Services\Ploi\Ploi; use App\Traits\HasPloi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue; use Illuminate\Queue\InteractsWithQueue;
@@ -11,7 +11,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class DeleteRedirect implements ShouldQueue class DeleteRedirect implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $serverPloiId; public $serverPloiId;
public $sitePloiId; public $sitePloiId;
@@ -42,9 +42,8 @@ class DeleteRedirect implements ShouldQueue
return; return;
} }
$ploi = new Ploi(config('services.ploi.token')); $this->getPloi()
->server($this->serverPloiId)
$ploi->server($this->serverPloiId)
->sites($this->sitePloiId) ->sites($this->sitePloiId)
->redirects() ->redirects()
->delete($this->redirectPloiId); ->delete($this->redirectPloiId);

View File

@@ -2,8 +2,8 @@
namespace App\Jobs\Redirects; namespace App\Jobs\Redirects;
use App\Traits\HasPloi;
use App\Models\Redirect; use App\Models\Redirect;
use App\Services\Ploi\Ploi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use App\Traits\JobHasThresholds; use App\Traits\JobHasThresholds;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
@@ -13,7 +13,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class FetchRedirectStatus implements ShouldQueue class FetchRedirectStatus implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, JobHasThresholds; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, JobHasThresholds, HasPloi;
public $redirect; public $redirect;
@@ -23,7 +23,7 @@ class FetchRedirectStatus implements ShouldQueue
* @param Redirect $redirect * @param Redirect $redirect
* @param int $threshold * @param int $threshold
*/ */
public function __construct(Redirect $redirect, $threshold = 0) public function __construct(Redirect $redirect, int $threshold = 0)
{ {
$this->redirect = $redirect; $this->redirect = $redirect;
$this->setThreshold($threshold); $this->setThreshold($threshold);
@@ -43,9 +43,8 @@ class FetchRedirectStatus implements ShouldQueue
return; return;
} }
$ploi = new Ploi(config('services.ploi.token')); $ploiRedirect = $this->getPloi()
->server($this->redirect->server->ploi_id)
$ploiRedirect = $ploi->server($this->redirect->server->ploi_id)
->sites($this->redirect->site->ploi_id) ->sites($this->redirect->site->ploi_id)
->redirects() ->redirects()
->get($this->redirect->ploi_id)->getData(); ->get($this->redirect->ploi_id)->getData();

View File

@@ -3,54 +3,45 @@
namespace App\Jobs\Servers; namespace App\Jobs\Servers;
use App\Models\Server; use App\Models\Server;
use App\Services\Ploi\Ploi; use App\Traits\HasPloi;
use Exception;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Contracts\Queue\ShouldQueue; use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Bus\Dispatchable; use Illuminate\Foundation\Bus\Dispatchable;
use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Queue\SerializesModels;
class CreateServer implements ShouldQueue class CreateServer implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $server; public $server;
public $tries = 1; public $tries = 1;
/**
* Create a new job instance.
*
* @param Server $server
*/
public function __construct(Server $server) public function __construct(Server $server)
{ {
$this->server = $server; $this->server = $server;
} }
/**
* Execute the job.
*
* @return void
*/
public function handle() public function handle()
{ {
$ploi = new Ploi(config('services.ploi.token')); $ploiServer = $this->getPloi()->server()->create(
$ploiServer = $ploi->server()->create(
$this->server->name, $this->server->name,
$this->server->provider->ploi_id, $this->server->provider->ploi_id,
$this->server->providerRegion->region_id, $this->server->providerRegion->region_id,
$this->server->providerPlan->plan_id, $this->server->providerPlan->plan_id,
'server',
$this->server->database_type
); );
$this->server->ploi_id = $ploiServer->id; $this->server->ploi_id = $ploiServer->id;
$this->server->save(); $this->server->save();
// Lets fetch the status after 5 minutes // Let's fetch the status after 5 minutes
dispatch(new FetchServerStatus($this->server))->delay(now()->addMinutes(5)); dispatch(new FetchServerStatus($this->server))->delay(now()->addMinutes(5));
} }
public function failed(\Exception $exception) public function failed(Exception $exception)
{ {
$this->server->delete(); $this->server->delete();
} }

View File

@@ -2,7 +2,7 @@
namespace App\Jobs\Servers; namespace App\Jobs\Servers;
use App\Services\Ploi\Ploi; use App\Traits\HasPloi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue; use Illuminate\Queue\InteractsWithQueue;
@@ -11,7 +11,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class DeleteServer implements ShouldQueue class DeleteServer implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $serverPloiId; public $serverPloiId;
@@ -32,8 +32,6 @@ class DeleteServer implements ShouldQueue
*/ */
public function handle() public function handle()
{ {
$ploi = new Ploi(config('services.ploi.token')); $this->getPloi()->server($this->serverPloiId)->delete();
$ploi->server($this->serverPloiId)->delete();
} }
} }

View File

@@ -3,7 +3,7 @@
namespace App\Jobs\Servers; namespace App\Jobs\Servers;
use App\Models\Server; use App\Models\Server;
use App\Services\Ploi\Ploi; use App\Traits\HasPloi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use App\Traits\JobHasThresholds; use App\Traits\JobHasThresholds;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
@@ -13,7 +13,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class FetchServerStatus implements ShouldQueue class FetchServerStatus implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, JobHasThresholds; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, JobHasThresholds, HasPloi;
public $server; public $server;
@@ -43,9 +43,7 @@ class FetchServerStatus implements ShouldQueue
return; return;
} }
$ploi = new Ploi; $ploiServer = $this->getPloi()->server($this->server->ploi_id)->get()->getData();
$ploiServer = $ploi->server($this->server->ploi_id)->get()->getData();
if ($ploiServer->status !== Server::STATUS_ACTIVE) { if ($ploiServer->status !== Server::STATUS_ACTIVE) {
$this->incrementThreshold(); $this->incrementThreshold();

View File

@@ -1,35 +0,0 @@
<?php
namespace App\Jobs\Servers;
use App\Models\Server;
use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Bus\Dispatchable;
class SynchronizeServers implements ShouldQueue
{
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
/**
* Execute the job.
*
* @return void
*/
public function handle()
{
$ploi = new \App\Services\Ploi\Ploi(config('services.ploi.token'));
$servers = $ploi->server()->get()->getData();
foreach ($servers as $server) {
Server::firstOrCreate([
'ip' => $server->ip_address,
], [
'name' => $server->name
]);
}
}
}

View File

@@ -3,7 +3,7 @@
namespace App\Jobs\Sites; namespace App\Jobs\Sites;
use App\Models\Site; use App\Models\Site;
use App\Services\Ploi\Ploi; use App\Traits\HasPloi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue; use Illuminate\Queue\InteractsWithQueue;
@@ -12,7 +12,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class ChangePhpVersion implements ShouldQueue class ChangePhpVersion implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $site; public $site;
public $version; public $version;
@@ -36,9 +36,7 @@ class ChangePhpVersion implements ShouldQueue
*/ */
public function handle() public function handle()
{ {
$ploi = new Ploi(config('services.ploi.token')); $this->getPloi()->server($this->site->server->ploi_id)->sites($this->site->ploi_id)->phpVersion($this->version);
$ploi->server($this->site->server->ploi_id)->sites($this->site->ploi_id)->phpVersion($this->version);
$this->site->php_version = $this->version; $this->site->php_version = $this->version;
$this->site->save(); $this->site->save();

View File

@@ -2,8 +2,9 @@
namespace App\Jobs\Sites; namespace App\Jobs\Sites;
use Throwable;
use App\Models\Site; use App\Models\Site;
use App\Services\Ploi\Ploi; use App\Traits\HasPloi;
use Illuminate\Support\Arr; use Illuminate\Support\Arr;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
@@ -13,32 +14,18 @@ use Illuminate\Foundation\Bus\Dispatchable;
class CreateSite implements ShouldQueue class CreateSite implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $site; public function __construct(
public Site $site,
/** ) {
* Create a new job instance.
*
* @param Site $site
*/
public function __construct(Site $site)
{
$this->site = $site;
} }
/** public function handle(): void
* Execute the job.
*
* @return void
*/
public function handle()
{ {
$ploi = new Ploi(config('services.ploi.token'));
$systemUser = $this->site->getSystemUser(); $systemUser = $this->site->getSystemUser();
$ploiSite = $ploi->server($this->site->server->ploi_id)->sites()->create( $ploiSite = $this->getPloi()->server($this->site->server->ploi_id)->sites()->create(
$this->site->domain, $this->site->domain,
'/public', '/public',
'/', '/',
@@ -53,8 +40,12 @@ class CreateSite implements ShouldQueue
dispatch(new FetchSiteStatus($this->site))->delay(now()->addSeconds(3)); dispatch(new FetchSiteStatus($this->site))->delay(now()->addSeconds(3));
} }
public function failed(\Exception $exception) public function failed(Throwable $exception): void
{ {
$this->site->delete(); $this->site->delete();
if (app()->isLocal()) {
throw $exception;
}
} }
} }

View File

@@ -2,7 +2,7 @@
namespace App\Jobs\Sites; namespace App\Jobs\Sites;
use App\Services\Ploi\Ploi; use App\Traits\HasPloi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
use Illuminate\Queue\InteractsWithQueue; use Illuminate\Queue\InteractsWithQueue;
@@ -11,7 +11,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class DeleteSite implements ShouldQueue class DeleteSite implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, HasPloi;
public $serverPloiId; public $serverPloiId;
public $sitePloiId; public $sitePloiId;
@@ -34,8 +34,6 @@ class DeleteSite implements ShouldQueue
*/ */
public function handle() public function handle()
{ {
$ploi = new Ploi(config('services.ploi.token')); $this->getPloi()->server($this->serverPloiId)->sites()->delete($this->sitePloiId);
$ploi->server($this->serverPloiId)->sites()->delete($this->sitePloiId);
} }
} }

View File

@@ -3,7 +3,7 @@
namespace App\Jobs\Sites; namespace App\Jobs\Sites;
use App\Models\Site; use App\Models\Site;
use App\Services\Ploi\Ploi; use App\Traits\HasPloi;
use Illuminate\Bus\Queueable; use Illuminate\Bus\Queueable;
use App\Traits\JobHasThresholds; use App\Traits\JobHasThresholds;
use Illuminate\Queue\SerializesModels; use Illuminate\Queue\SerializesModels;
@@ -13,7 +13,7 @@ use Illuminate\Foundation\Bus\Dispatchable;
class FetchSiteStatus implements ShouldQueue class FetchSiteStatus implements ShouldQueue
{ {
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, JobHasThresholds; use Dispatchable, InteractsWithQueue, Queueable, SerializesModels, JobHasThresholds, HasPloi;
public $site; public $site;
@@ -43,9 +43,7 @@ class FetchSiteStatus implements ShouldQueue
return; return;
} }
$ploi = new Ploi; $ploiSite = $this->getPloi()->server($this->site->server->ploi_id)->sites()->get($this->site->ploi_id)->getData();
$ploiSite = $ploi->server($this->site->server->ploi_id)->sites()->get($this->site->ploi_id)->getData();
if ($ploiSite->status !== Site::STATUS_ACTIVE) { if ($ploiSite->status !== Site::STATUS_ACTIVE) {
$this->incrementThreshold(); $this->incrementThreshold();

View File

@@ -0,0 +1,13 @@
<?php
namespace App\Listeners;
use Illuminate\Auth\Events\Logout;
class ResetTwoFactorAuthenticationSession
{
public function handle(Logout $event): void
{
session()->put('auth.two_factor_authenticated_at', null);
}
}

View File

@@ -13,26 +13,15 @@ class AdminServerCreatedEmail extends Mailable implements ShouldQueue
{ {
use Queueable, SerializesModels; use Queueable, SerializesModels;
public $user; public User $user;
public $server; public Server $server;
/**
* Create a new message instance.
*
* @param User $user
* @param Server $server
*/
public function __construct(User $user, Server $server) public function __construct(User $user, Server $server)
{ {
$this->user = $user; $this->user = $user;
$this->server = $server; $this->server = $server;
} }
/**
* Build the message.
*
* @return $this
*/
public function build() public function build()
{ {
return $this return $this

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