mirror of
https://github.com/pelican-dev/panel.git
synced 2025-12-08 16:40:14 +01:00
Compare commits
415 Commits
v1.0.0-bet
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
b6aeb954c4 | ||
|
|
7c0d53c796 | ||
|
|
71bd267166 | ||
|
|
25d8adbcc6 | ||
|
|
27b896c6d2 | ||
|
|
bda2f9a699 | ||
|
|
04375439d7 | ||
|
|
0fe8917668 | ||
|
|
c312ef493f | ||
|
|
6c02f9a663 | ||
|
|
2dd6e3d4fc | ||
|
|
575e5bdb0d | ||
|
|
efa8eef57c | ||
|
|
d16e7dd876 | ||
|
|
897b95ec13 | ||
|
|
97f5a0f20b | ||
|
|
d0af45a0c7 | ||
|
|
78ab098d02 | ||
|
|
cdccca8fa2 | ||
|
|
bb33bcca4f | ||
|
|
611b8649e0 | ||
|
|
b1b723485f | ||
|
|
25c8ff3f1f | ||
|
|
07763d912b | ||
|
|
65bb99e2b0 | ||
|
|
a195b56f93 | ||
|
|
d78c977d75 | ||
|
|
5e25ea4a43 | ||
|
|
886836c60a | ||
|
|
f575e3edfa | ||
|
|
1a66b3fab4 | ||
|
|
0f1efcfd15 | ||
|
|
3f89c6ddd8 | ||
|
|
20cb7850ef | ||
|
|
108dad09fb | ||
|
|
445c9364bc | ||
|
|
acec117b1e | ||
|
|
89199dfbe5 | ||
|
|
216a3484f1 | ||
|
|
5c3b0919aa | ||
|
|
f4ee33fa4f | ||
|
|
d8368c4cec | ||
|
|
aa35d7d001 | ||
|
|
3c25b43b46 | ||
|
|
0891db5342 | ||
|
|
172436e012 | ||
|
|
2b5403a4da | ||
|
|
a30c45fbbe | ||
|
|
b06df23823 | ||
|
|
1ff965611e | ||
|
|
cec141889a | ||
|
|
6ed84b5584 | ||
|
|
49f24e37b6 | ||
|
|
e0c4e47a6c | ||
|
|
4bda7cba75 | ||
|
|
852f7beb39 | ||
|
|
d61583cd7b | ||
|
|
21f9f259d0 | ||
|
|
b2aff5445b | ||
|
|
1f26750a2a | ||
|
|
6d83c6d908 | ||
|
|
574a391e73 | ||
|
|
605fcbe61a | ||
|
|
0214b127e4 | ||
|
|
e6aa76ef2c | ||
|
|
d38075e3cb | ||
|
|
0fec6adc3e | ||
|
|
5e3c22ea5e | ||
|
|
d1a808a746 | ||
|
|
3bcdeea800 | ||
|
|
e6bd6e416f | ||
|
|
8e006ac32d | ||
|
|
430f28a847 | ||
|
|
1a4fa5e67a | ||
|
|
a65469b33b | ||
|
|
d587cf3ee5 | ||
|
|
2cd9fa2cde | ||
|
|
d735e858a2 | ||
|
|
317fa46894 | ||
|
|
e589f972fb | ||
|
|
266e3779d5 | ||
|
|
4652680a7b | ||
|
|
e99f7179c6 | ||
|
|
1f56b8e114 | ||
|
|
574e03a986 | ||
|
|
05f3422dda | ||
|
|
dbe4bdd62d | ||
|
|
f6710dbbe4 | ||
|
|
e4f807b297 | ||
|
|
cd965678b7 | ||
|
|
a58ae874f3 | ||
|
|
432fb8a514 | ||
|
|
bb02ec4c6c | ||
|
|
69b669e345 | ||
|
|
80993f38a9 | ||
|
|
19103b16b8 | ||
|
|
246997754e | ||
|
|
df75dbe2ad | ||
|
|
f02b58c320 | ||
|
|
8aa0fc7fc2 | ||
|
|
2fc30e14fd | ||
|
|
ec5fd3262a | ||
|
|
81178f81b4 | ||
|
|
5373f1e30a | ||
|
|
9f35f1c3ee | ||
|
|
a5858a6d9b | ||
|
|
e3b3c92dcb | ||
|
|
42c84c2df5 | ||
|
|
4792542f20 | ||
|
|
bb40a5273f | ||
|
|
e5c24fe8b6 | ||
|
|
c10280af4b | ||
|
|
6db1d82738 | ||
|
|
68f8244298 | ||
|
|
ce393af7a6 | ||
|
|
932809fec5 | ||
|
|
3d2390dbcc | ||
|
|
d5d50d4150 | ||
|
|
cba8717188 | ||
|
|
df4543a079 | ||
|
|
8dc99e6390 | ||
|
|
8f1ec20e96 | ||
|
|
61dcb9a3ba | ||
|
|
0e34886d7e | ||
|
|
806820592f | ||
|
|
1900c04b71 | ||
|
|
32eb1abd4a | ||
|
|
47557021fd | ||
|
|
2ef81eae1a | ||
|
|
420730ba1f | ||
|
|
925ab26fb4 | ||
|
|
2952e22619 | ||
|
|
079eaed010 | ||
|
|
6671d45651 | ||
|
|
3543b4773a | ||
|
|
02f788a659 | ||
|
|
7ace3978d8 | ||
|
|
8f277aaca0 | ||
|
|
76451fa0ad | ||
|
|
0104a08ba4 | ||
|
|
5eff006843 | ||
|
|
a8241bf9f3 | ||
|
|
4aae2562ea | ||
|
|
42db5b328a | ||
|
|
bc4dfb3e92 | ||
|
|
3b9c81534f | ||
|
|
f31aa78f6f | ||
|
|
b5ebd544f4 | ||
|
|
c77a37ec89 | ||
|
|
4d78e5dcd1 | ||
|
|
15075b6ab8 | ||
|
|
a8f233e204 | ||
|
|
795cad43b9 | ||
|
|
46934d7a85 | ||
|
|
06067f375c | ||
|
|
d1df53c683 | ||
|
|
b03d2cf919 | ||
|
|
27a8423f55 | ||
|
|
ad70934430 | ||
|
|
900f8d0fe1 | ||
|
|
6a4ac515a7 | ||
|
|
7c315ac995 | ||
|
|
49e9440e0f | ||
|
|
02e3e43f1e | ||
|
|
8eddef6f04 | ||
|
|
d2f1936bbf | ||
|
|
36863f94c0 | ||
|
|
75863c50d1 | ||
|
|
ec0727b406 | ||
|
|
5b2e9d94ca | ||
|
|
8840d109ef | ||
|
|
71225bd2dc | ||
|
|
bab8ec6e18 | ||
|
|
d307a2095b | ||
|
|
a777f4e0ff | ||
|
|
86a71afc6c | ||
|
|
88943563c7 | ||
|
|
20071a64fa | ||
|
|
d0d3418e03 | ||
|
|
083e3dc62a | ||
|
|
d7e60f2456 | ||
|
|
38e746240d | ||
|
|
986063dce4 | ||
|
|
71d0326cb2 | ||
|
|
62ca53eeaf | ||
|
|
9f2305f351 | ||
|
|
340d1b543c | ||
|
|
61098b11f2 | ||
|
|
4d03d6b948 | ||
|
|
1f67054777 | ||
|
|
4a9814f16c | ||
|
|
e0697d3288 | ||
|
|
d165da20ec | ||
|
|
ae27b179fe | ||
|
|
1113ffe0f7 | ||
|
|
5531bc0ba1 | ||
|
|
a3819122db | ||
|
|
c5528a61f3 | ||
|
|
5a7c6ac6e5 | ||
|
|
5e8cccef19 | ||
|
|
0ccb248d91 | ||
|
|
514d961c24 | ||
|
|
f8e802afcd | ||
|
|
556551b4f3 | ||
|
|
23ddded61e | ||
|
|
c5aa8a3980 | ||
|
|
21ac75efae | ||
|
|
9655700cde | ||
|
|
c9b7e979c0 | ||
|
|
77a3b0640d | ||
|
|
de4cb38766 | ||
|
|
74bd7f9991 | ||
|
|
ba7f814300 | ||
|
|
cdcd1c521e | ||
|
|
4d0aabe91e | ||
|
|
68f72b9b4d | ||
|
|
dca37ccc95 | ||
|
|
6a088d0c4f | ||
|
|
7731f16b0f | ||
|
|
9a1e7de4ae | ||
|
|
c61b6920b9 | ||
|
|
6107524522 | ||
|
|
57a13a2701 | ||
|
|
4dd414ad87 | ||
|
|
0156ac1509 | ||
|
|
387471716b | ||
|
|
1dc5ec027e | ||
|
|
b05eabfdb0 | ||
|
|
3039c1c698 | ||
|
|
de166bca03 | ||
|
|
af609994b6 | ||
|
|
bd2a00760d | ||
|
|
65deffc6e6 | ||
|
|
34865d4288 | ||
|
|
2961c3e88b | ||
|
|
e7a950ffcb | ||
|
|
ece732d9e5 | ||
|
|
456c4f46bc | ||
|
|
0ba497a2eb | ||
|
|
3b744f37dd | ||
|
|
b34778f736 | ||
|
|
84c351d0ae | ||
|
|
520cea7f09 | ||
|
|
35ce1d34ab | ||
|
|
17555a1d09 | ||
|
|
837121b1fb | ||
|
|
af9f2c653e | ||
|
|
c22e7456b5 | ||
|
|
97fb66f5d6 | ||
|
|
51037c5c20 | ||
|
|
23d13d9e83 | ||
|
|
6c20426757 | ||
|
|
1224210668 | ||
|
|
258c97bf14 | ||
|
|
7034c4d013 | ||
|
|
e5cba893e4 | ||
|
|
fd49f472c3 | ||
|
|
c8556a4c56 | ||
|
|
6de6306a19 | ||
|
|
1f8a5cdd1d | ||
|
|
30ae860d69 | ||
|
|
f400e2db76 | ||
|
|
1f7562563a | ||
|
|
2296e41a8b | ||
|
|
7971dc13fc | ||
|
|
8406f4686c | ||
|
|
67705b14b4 | ||
|
|
bc115af5fd | ||
|
|
da35703f75 | ||
|
|
c54bfd714b | ||
|
|
b83e3657d6 | ||
|
|
e2c87a8206 | ||
|
|
e38a736b61 | ||
|
|
26e20453bf | ||
|
|
292523d153 | ||
|
|
85d625d118 | ||
|
|
c8230771ec | ||
|
|
79691ba663 | ||
|
|
a6326f64fb | ||
|
|
03745eb4be | ||
|
|
c0fda71e20 | ||
|
|
f2f1026a97 | ||
|
|
e1eaf805ea | ||
|
|
03ec20e3a0 | ||
|
|
a5ffff8c8c | ||
|
|
82ef6c1408 | ||
|
|
2d581c7cbd | ||
|
|
7f0266be5e | ||
|
|
1ae9490b8f | ||
|
|
a53b3fda10 | ||
|
|
e9ddf80d10 | ||
|
|
3f1e99f1df | ||
|
|
435c615ff1 | ||
|
|
3effd98013 | ||
|
|
e354bc9be7 | ||
|
|
14d351103c | ||
|
|
92c23451af | ||
|
|
2046fa453a | ||
|
|
b39a8186ae | ||
|
|
8ae3c88c91 | ||
|
|
329a29f7da | ||
|
|
98a2cab5ca | ||
|
|
8407547574 | ||
|
|
fccd7e5e75 | ||
|
|
c0225b9e10 | ||
|
|
544aaab960 | ||
|
|
914e215bc0 | ||
|
|
90fd73f6a4 | ||
|
|
0037b4a1d4 | ||
|
|
3deada57c6 | ||
|
|
6427903f9f | ||
|
|
b16e19b4fb | ||
|
|
7e99d5cd8e | ||
|
|
05b1a44a34 | ||
|
|
058b613c98 | ||
|
|
0e2ab4b711 | ||
|
|
ee838316e6 | ||
|
|
ffd94b8892 | ||
|
|
a186900262 | ||
|
|
bf14755287 | ||
|
|
038504fbec | ||
|
|
22a0a52f7b | ||
|
|
862afaa0e9 | ||
|
|
a4dd8cca4c | ||
|
|
e67e0830eb | ||
|
|
b444112085 | ||
|
|
f23d4d6971 | ||
|
|
2a3781f5a8 | ||
|
|
cb245dc722 | ||
|
|
3ffbf9e46a | ||
|
|
8221c80ec2 | ||
|
|
702a6bb750 | ||
|
|
02d7ad04ad | ||
|
|
7409f020ba | ||
|
|
98d8510f11 | ||
|
|
6c6d458445 | ||
|
|
51fda2eaf4 | ||
|
|
92fbd75772 | ||
|
|
fa8ae0aea5 | ||
|
|
377b3f170d | ||
|
|
566e7c1b24 | ||
|
|
b9d4773bd7 | ||
|
|
49638e75e5 | ||
|
|
80c404a48c | ||
|
|
befe6be80b | ||
|
|
3639d7ccec | ||
|
|
20f271041a | ||
|
|
c3b8b71f9c | ||
|
|
c73d0544d9 | ||
|
|
484a3b445a | ||
|
|
c0fa8c1cd8 | ||
|
|
e562a35057 | ||
|
|
636279c6eb | ||
|
|
ed88ce9ae3 | ||
|
|
0cce716e2c | ||
|
|
3639f0cb50 | ||
|
|
9c3f47590c | ||
|
|
630031e1c2 | ||
|
|
2c00f90ba6 | ||
|
|
875dca54f5 | ||
|
|
a03b604f2d | ||
|
|
8261184b57 | ||
|
|
bca02ced86 | ||
|
|
a768fadaea | ||
|
|
7471347b55 | ||
|
|
1457c4bd06 | ||
|
|
8b943fa160 | ||
|
|
5c5c9654b4 | ||
|
|
dd20cb0f11 | ||
|
|
88deb35dc8 | ||
|
|
0f92632c06 | ||
|
|
a85fc5c88e | ||
|
|
8d7eff13fb | ||
|
|
c39c29e50b | ||
|
|
db3b16e609 | ||
|
|
72b9c309d3 | ||
|
|
68a6dc45cb | ||
|
|
9a258efe53 | ||
|
|
3310746107 | ||
|
|
42706dba14 | ||
|
|
ec6529ac4c | ||
|
|
bced93c5be | ||
|
|
cb1c953540 | ||
|
|
c689f6860b | ||
|
|
a73404c1b4 | ||
|
|
61cbe5465f | ||
|
|
5bea1ea80a | ||
|
|
b69136d7a4 | ||
|
|
a8c3082b79 | ||
|
|
a47ad071c9 | ||
|
|
ab953b2f4d | ||
|
|
03d6c88f65 | ||
|
|
b4eab02254 | ||
|
|
23f39acd4e | ||
|
|
82b0aff105 | ||
|
|
adca50a372 | ||
|
|
c5230efad6 | ||
|
|
e5d9d53aa3 | ||
|
|
29f3defc73 | ||
|
|
2dbb9a5f9b | ||
|
|
a05e330b19 | ||
|
|
4a7951995e | ||
|
|
3d29243cf0 | ||
|
|
c52439132d | ||
|
|
517f17cbcc | ||
|
|
f8d119b458 | ||
|
|
fbeb747fc3 | ||
|
|
f563128237 | ||
|
|
f2f3ee548f | ||
|
|
0b3dce132f | ||
|
|
5bf23b972d | ||
|
|
22d02c0df5 | ||
|
|
253abf65b1 | ||
|
|
d452e3d2f2 | ||
|
|
0051370f24 |
@ -3,5 +3,4 @@ APP_DEBUG=false
|
|||||||
APP_KEY=
|
APP_KEY=
|
||||||
APP_URL=http://panel.test
|
APP_URL=http://panel.test
|
||||||
APP_INSTALLED=false
|
APP_INSTALLED=false
|
||||||
APP_TIMEZONE=UTC
|
|
||||||
APP_LOCALE=en
|
APP_LOCALE=en
|
||||||
|
|||||||
15
.github/CODEOWNERS
vendored
15
.github/CODEOWNERS
vendored
@ -1,15 +0,0 @@
|
|||||||
# Lines starting with '#' are comments.
|
|
||||||
# Each line is a file pattern followed by one or more owners.
|
|
||||||
|
|
||||||
# More details are here: https://help.github.com/articles/about-codeowners/
|
|
||||||
|
|
||||||
# The '*' pattern is global owners.
|
|
||||||
|
|
||||||
# Order is important. The last matching pattern has the most precedence.
|
|
||||||
# The folders are ordered as follows:
|
|
||||||
|
|
||||||
# In each subsection folders are ordered first by depth, then alphabetically.
|
|
||||||
# This should make it easy to add new rules without breaking existing ones.
|
|
||||||
|
|
||||||
# Global
|
|
||||||
* @pelican-dev/panel
|
|
||||||
7
.github/ISSUE_TEMPLATE/bug-report.yml
vendored
7
.github/ISSUE_TEMPLATE/bug-report.yml
vendored
@ -64,10 +64,9 @@ body:
|
|||||||
label: Error Logs
|
label: Error Logs
|
||||||
description: |
|
description: |
|
||||||
Run the following command to collect logs on your system.
|
Run the following command to collect logs on your system.
|
||||||
|
Wings: `sudo wings diagnostics --hastebin-url=https://logs.pelican.dev`
|
||||||
Wings: `sudo wings diagnostics`
|
Panel: `tail -n 300 /var/www/pelican/storage/logs/laravel-$(date +%F).log | curl --data-binary @- https://logs.pelican.dev`
|
||||||
Panel: `tail -n 150 /var/www/pelican/storage/logs/laravel-$(date +%F).log | curl -X POST -F 'c=@-' paste.pelistuff.com`
|
placeholder: "https://logs.pelican.dev/c17f750e"
|
||||||
placeholder: "https://pelipaste.com/a1h6z"
|
|
||||||
render: bash
|
render: bash
|
||||||
validations:
|
validations:
|
||||||
required: false
|
required: false
|
||||||
|
|||||||
4
.github/workflows/build.yaml
vendored
4
.github/workflows/build.yaml
vendored
@ -11,9 +11,9 @@ jobs:
|
|||||||
name: UI
|
name: UI
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: true
|
||||||
matrix:
|
matrix:
|
||||||
node-version: [18, 20]
|
node-version: [20, 22]
|
||||||
steps:
|
steps:
|
||||||
- name: Code Checkout
|
- name: Code Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
|||||||
133
.github/workflows/ci.yaml
vendored
133
.github/workflows/ci.yaml
vendored
@ -6,12 +6,75 @@ on:
|
|||||||
- main
|
- main
|
||||||
pull_request:
|
pull_request:
|
||||||
|
|
||||||
|
env:
|
||||||
|
APP_ENV: testing
|
||||||
|
APP_DEBUG: "false"
|
||||||
|
APP_KEY: ThisIsARandomStringForTests12345
|
||||||
|
APP_TIMEZONE: UTC
|
||||||
|
APP_URL: http://localhost/
|
||||||
|
CACHE_DRIVER: array
|
||||||
|
MAIL_MAILER: array
|
||||||
|
SESSION_DRIVER: array
|
||||||
|
QUEUE_CONNECTION: sync
|
||||||
|
GUZZLE_TIMEOUT: 60
|
||||||
|
GUZZLE_CONNECT_TIMEOUT: 60
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
sqlite:
|
||||||
|
name: SQLite
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
fail-fast: true
|
||||||
|
matrix:
|
||||||
|
php: [8.2, 8.3, 8.4]
|
||||||
|
env:
|
||||||
|
DB_CONNECTION: sqlite
|
||||||
|
DB_DATABASE: testing.sqlite
|
||||||
|
steps:
|
||||||
|
- name: Code Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Get cache directory
|
||||||
|
id: composer-cache
|
||||||
|
run: |
|
||||||
|
echo "dir=$(composer config cache-files-dir)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
- name: Cache
|
||||||
|
uses: actions/cache@v4
|
||||||
|
with:
|
||||||
|
path: ${{ steps.composer-cache.outputs.dir }}
|
||||||
|
key: ${{ runner.os }}-composer-${{ matrix.php }}-${{ hashFiles('**/composer.lock') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-composer-${{ matrix.php }}-
|
||||||
|
|
||||||
|
- name: Setup PHP
|
||||||
|
uses: shivammathur/setup-php@v2
|
||||||
|
with:
|
||||||
|
php-version: ${{ matrix.php }}
|
||||||
|
extensions: bcmath, curl, gd, mbstring, mysql, openssl, pdo, tokenizer, xml, zip
|
||||||
|
tools: composer:v2
|
||||||
|
coverage: none
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: composer install --no-interaction --no-suggest --no-progress --no-scripts
|
||||||
|
|
||||||
|
- name: Create SQLite file
|
||||||
|
run: touch database/testing.sqlite
|
||||||
|
|
||||||
|
- name: Unit tests
|
||||||
|
run: vendor/bin/pest tests/Unit
|
||||||
|
env:
|
||||||
|
DB_HOST: UNIT_NO_DB
|
||||||
|
SKIP_MIGRATIONS: true
|
||||||
|
|
||||||
|
- name: Integration tests
|
||||||
|
run: vendor/bin/pest tests/Integration
|
||||||
|
|
||||||
mysql:
|
mysql:
|
||||||
name: MySQL
|
name: MySQL
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: true
|
||||||
matrix:
|
matrix:
|
||||||
php: [8.2, 8.3, 8.4]
|
php: [8.2, 8.3, 8.4]
|
||||||
database: ["mysql:8"]
|
database: ["mysql:8"]
|
||||||
@ -25,21 +88,10 @@ jobs:
|
|||||||
- 3306
|
- 3306
|
||||||
options: --health-cmd="mysqladmin ping" --health-interval=10s --health-timeout=5s --health-retries=3
|
options: --health-cmd="mysqladmin ping" --health-interval=10s --health-timeout=5s --health-retries=3
|
||||||
env:
|
env:
|
||||||
APP_ENV: testing
|
|
||||||
APP_DEBUG: "false"
|
|
||||||
APP_KEY: ThisIsARandomStringForTests12345
|
|
||||||
APP_TIMEZONE: UTC
|
|
||||||
APP_URL: http://localhost/
|
|
||||||
CACHE_DRIVER: array
|
|
||||||
MAIL_MAILER: array
|
|
||||||
SESSION_DRIVER: array
|
|
||||||
QUEUE_CONNECTION: sync
|
|
||||||
DB_CONNECTION: mysql
|
DB_CONNECTION: mysql
|
||||||
DB_HOST: 127.0.0.1
|
DB_HOST: 127.0.0.1
|
||||||
DB_DATABASE: testing
|
DB_DATABASE: testing
|
||||||
DB_USERNAME: root
|
DB_USERNAME: root
|
||||||
GUZZLE_TIMEOUT: 60
|
|
||||||
GUZZLE_CONNECT_TIMEOUT: 60
|
|
||||||
steps:
|
steps:
|
||||||
- name: Code Checkout
|
- name: Code Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
@ -84,7 +136,7 @@ jobs:
|
|||||||
name: MariaDB
|
name: MariaDB
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: true
|
||||||
matrix:
|
matrix:
|
||||||
php: [8.2, 8.3, 8.4]
|
php: [8.2, 8.3, 8.4]
|
||||||
database: ["mariadb:10.6", "mariadb:10.11", "mariadb:11.4"]
|
database: ["mariadb:10.6", "mariadb:10.11", "mariadb:11.4"]
|
||||||
@ -98,21 +150,10 @@ jobs:
|
|||||||
- 3306
|
- 3306
|
||||||
options: --health-cmd="mariadb-admin ping || mysqladmin ping" --health-interval=10s --health-timeout=5s --health-retries=3
|
options: --health-cmd="mariadb-admin ping || mysqladmin ping" --health-interval=10s --health-timeout=5s --health-retries=3
|
||||||
env:
|
env:
|
||||||
APP_ENV: testing
|
|
||||||
APP_DEBUG: "false"
|
|
||||||
APP_KEY: ThisIsARandomStringForTests12345
|
|
||||||
APP_TIMEZONE: UTC
|
|
||||||
APP_URL: http://localhost/
|
|
||||||
CACHE_DRIVER: array
|
|
||||||
MAIL_MAILER: array
|
|
||||||
SESSION_DRIVER: array
|
|
||||||
QUEUE_CONNECTION: sync
|
|
||||||
DB_CONNECTION: mariadb
|
DB_CONNECTION: mariadb
|
||||||
DB_HOST: 127.0.0.1
|
DB_HOST: 127.0.0.1
|
||||||
DB_DATABASE: testing
|
DB_DATABASE: testing
|
||||||
DB_USERNAME: root
|
DB_USERNAME: root
|
||||||
GUZZLE_TIMEOUT: 60
|
|
||||||
GUZZLE_CONNECT_TIMEOUT: 60
|
|
||||||
steps:
|
steps:
|
||||||
- name: Code Checkout
|
- name: Code Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
@ -153,27 +194,35 @@ jobs:
|
|||||||
DB_PORT: ${{ job.services.database.ports[3306] }}
|
DB_PORT: ${{ job.services.database.ports[3306] }}
|
||||||
DB_USERNAME: root
|
DB_USERNAME: root
|
||||||
|
|
||||||
sqlite:
|
postgresql:
|
||||||
name: SQLite
|
name: PostgreSQL
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: true
|
||||||
matrix:
|
matrix:
|
||||||
php: [8.2, 8.3, 8.4]
|
php: [8.2, 8.3, 8.4]
|
||||||
|
database: ["postgres:14"]
|
||||||
|
services:
|
||||||
|
database:
|
||||||
|
image: ${{ matrix.database }}
|
||||||
env:
|
env:
|
||||||
APP_ENV: testing
|
POSTGRES_DB: testing
|
||||||
APP_DEBUG: "false"
|
POSTGRES_USER: postgres
|
||||||
APP_KEY: ThisIsARandomStringForTests12345
|
POSTGRES_PASSWORD: postgres
|
||||||
APP_TIMEZONE: UTC
|
POSTGRES_HOST_AUTH_METHOD: trust
|
||||||
APP_URL: http://localhost/
|
ports:
|
||||||
CACHE_DRIVER: array
|
- 5432:5432
|
||||||
MAIL_MAILER: array
|
options: >-
|
||||||
SESSION_DRIVER: array
|
--health-cmd pg_isready
|
||||||
QUEUE_CONNECTION: sync
|
--health-interval 10s
|
||||||
DB_CONNECTION: sqlite
|
--health-timeout 5s
|
||||||
DB_DATABASE: testing.sqlite
|
--health-retries 5
|
||||||
GUZZLE_TIMEOUT: 60
|
env:
|
||||||
GUZZLE_CONNECT_TIMEOUT: 60
|
DB_CONNECTION: pgsql
|
||||||
|
DB_HOST: 127.0.0.1
|
||||||
|
DB_DATABASE: testing
|
||||||
|
DB_USERNAME: postgres
|
||||||
|
DB_PASSWORD: postgres
|
||||||
steps:
|
steps:
|
||||||
- name: Code Checkout
|
- name: Code Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
@ -189,7 +238,6 @@ jobs:
|
|||||||
path: ${{ steps.composer-cache.outputs.dir }}
|
path: ${{ steps.composer-cache.outputs.dir }}
|
||||||
key: ${{ runner.os }}-composer-${{ matrix.php }}-${{ hashFiles('**/composer.lock') }}
|
key: ${{ runner.os }}-composer-${{ matrix.php }}-${{ hashFiles('**/composer.lock') }}
|
||||||
restore-keys: |
|
restore-keys: |
|
||||||
${{ runner.os }}-composer-${{ matrix.php }}-
|
|
||||||
|
|
||||||
- name: Setup PHP
|
- name: Setup PHP
|
||||||
uses: shivammathur/setup-php@v2
|
uses: shivammathur/setup-php@v2
|
||||||
@ -202,9 +250,6 @@ jobs:
|
|||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: composer install --no-interaction --no-suggest --no-progress --no-scripts
|
run: composer install --no-interaction --no-suggest --no-progress --no-scripts
|
||||||
|
|
||||||
- name: Create SQLite file
|
|
||||||
run: touch database/testing.sqlite
|
|
||||||
|
|
||||||
- name: Unit tests
|
- name: Unit tests
|
||||||
run: vendor/bin/pest tests/Unit
|
run: vendor/bin/pest tests/Unit
|
||||||
env:
|
env:
|
||||||
|
|||||||
7
.github/workflows/docker-publish.yml
vendored
7
.github/workflows/docker-publish.yml
vendored
@ -66,8 +66,6 @@ jobs:
|
|||||||
permissions:
|
permissions:
|
||||||
contents: read
|
contents: read
|
||||||
packages: write
|
packages: write
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
# Start a temp local registry because workflow can not pull from localy loaded images
|
# Start a temp local registry because workflow can not pull from localy loaded images
|
||||||
services:
|
services:
|
||||||
registry:
|
registry:
|
||||||
@ -134,6 +132,11 @@ jobs:
|
|||||||
docker push localhost:5000/base-php:arm64
|
docker push localhost:5000/base-php:arm64
|
||||||
rm base-php-arm64.tar base-php-amd64.tar
|
rm base-php-arm64.tar base-php-amd64.tar
|
||||||
|
|
||||||
|
- name: Update version in config/app.php (tag)
|
||||||
|
if: "github.event_name == 'release' && github.event.action == 'published'"
|
||||||
|
run: |
|
||||||
|
sed -i "s/'version' => 'canary',/'version' => '${{ steps.build_info.outputs.version_tag }}',/" config/app.php
|
||||||
|
|
||||||
- name: Build and Push (tag)
|
- name: Build and Push (tag)
|
||||||
uses: docker/build-push-action@v6
|
uses: docker/build-push-action@v6
|
||||||
if: "github.event_name == 'release' && github.event.action == 'published'"
|
if: "github.event_name == 'release' && github.event.action == 'published'"
|
||||||
|
|||||||
4
.github/workflows/lint.yaml
vendored
4
.github/workflows/lint.yaml
vendored
@ -33,7 +33,7 @@ jobs:
|
|||||||
name: PHPStan
|
name: PHPStan
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: true
|
||||||
matrix:
|
matrix:
|
||||||
php: [ 8.2, 8.3, 8.4 ]
|
php: [ 8.2, 8.3, 8.4 ]
|
||||||
steps:
|
steps:
|
||||||
@ -68,4 +68,4 @@ jobs:
|
|||||||
run: composer install --no-interaction --no-suggest --no-progress --no-scripts
|
run: composer install --no-interaction --no-suggest --no-progress --no-scripts
|
||||||
|
|
||||||
- name: PHPStan
|
- name: PHPStan
|
||||||
run: vendor/bin/phpstan --memory-limit=-1
|
run: vendor/bin/phpstan --memory-limit=-1 --error-format=github
|
||||||
|
|||||||
6
.gitignore
vendored
6
.gitignore
vendored
@ -1,7 +1,6 @@
|
|||||||
/.phpunit.cache
|
/.phpunit.cache
|
||||||
/node_modules
|
/node_modules
|
||||||
/public/build
|
/public/build
|
||||||
/public/hot
|
|
||||||
/public/storage
|
/public/storage
|
||||||
/storage/*.key
|
/storage/*.key
|
||||||
/storage/pail
|
/storage/pail
|
||||||
@ -22,10 +21,9 @@ yarn-error.log
|
|||||||
/.idea
|
/.idea
|
||||||
/.nova
|
/.nova
|
||||||
/.vscode
|
/.vscode
|
||||||
|
/.ddev
|
||||||
|
|
||||||
public/assets/manifest.json
|
public/assets/manifest.json
|
||||||
/database/*.sqlite
|
/database/*.sqlite*
|
||||||
/database/*.sqlite-journal
|
|
||||||
filament-monaco-editor/
|
|
||||||
_ide_helper*
|
_ide_helper*
|
||||||
/.phpstorm.meta.php
|
/.phpstorm.meta.php
|
||||||
|
|||||||
40
Dockerfile
40
Dockerfile
@ -1,16 +1,9 @@
|
|||||||
# syntax=docker.io/docker/dockerfile:1.13-labs
|
# syntax=docker.io/docker/dockerfile:1.13-labs
|
||||||
# Pelican Production Dockerfile
|
# Pelican Production Dockerfile
|
||||||
|
|
||||||
|
##
|
||||||
# For those who want to build this Dockerfile themselves, uncomment lines 6-12 and replace "localhost:5000/base-php:$TARGETARCH" on lines 17 and 67 with "base".
|
# If you want to build this locally you want to run `docker build -f Dockerfile.dev`
|
||||||
|
##
|
||||||
# FROM --platform=$TARGETOS/$TARGETARCH php:8.3-fpm-alpine as base
|
|
||||||
|
|
||||||
# ADD --chmod=0755 https://github.com/mlocati/docker-php-extension-installer/releases/latest/download/install-php-extensions /usr/local/bin/
|
|
||||||
|
|
||||||
# RUN install-php-extensions bcmath gd intl zip opcache pcntl posix pdo_mysql
|
|
||||||
|
|
||||||
# RUN rm /usr/local/bin/install-php-extensions
|
|
||||||
|
|
||||||
# ================================
|
# ================================
|
||||||
# Stage 1-1: Composer Install
|
# Stage 1-1: Composer Install
|
||||||
@ -70,8 +63,8 @@ FROM --platform=$TARGETOS/$TARGETARCH localhost:5000/base-php:$TARGETARCH AS fin
|
|||||||
WORKDIR /var/www/html
|
WORKDIR /var/www/html
|
||||||
|
|
||||||
# Install additional required libraries
|
# Install additional required libraries
|
||||||
RUN apk update && apk add --no-cache \
|
RUN apk add --no-cache \
|
||||||
caddy ca-certificates supervisor supercronic
|
caddy ca-certificates supervisor supercronic fcgi
|
||||||
|
|
||||||
COPY --chown=root:www-data --chmod=640 --from=composerbuild /build .
|
COPY --chown=root:www-data --chmod=640 --from=composerbuild /build .
|
||||||
COPY --chown=root:www-data --chmod=640 --from=yarnbuild /build/public ./public
|
COPY --chown=root:www-data --chmod=640 --from=yarnbuild /build/public ./public
|
||||||
@ -82,14 +75,18 @@ RUN chown root:www-data ./ \
|
|||||||
&& chmod 750 ./ \
|
&& chmod 750 ./ \
|
||||||
# Files should not have execute set, but directories need it
|
# Files should not have execute set, but directories need it
|
||||||
&& find ./ -type d -exec chmod 750 {} \; \
|
&& find ./ -type d -exec chmod 750 {} \; \
|
||||||
# Symlink to env/database path, as www-data won't be able to write to webroot
|
# Create necessary directories
|
||||||
|
&& mkdir -p /pelican-data/storage /var/www/html/storage/app/public /var/run/supervisord /etc/supercronic \
|
||||||
|
# Symlinks for env, database, and avatars
|
||||||
&& ln -s /pelican-data/.env ./.env \
|
&& ln -s /pelican-data/.env ./.env \
|
||||||
&& ln -s /pelican-data/database/database.sqlite ./database/database.sqlite \
|
&& ln -s /pelican-data/database/database.sqlite ./database/database.sqlite \
|
||||||
# Create necessary directories
|
&& ln -sf /var/www/html/storage/app/public /var/www/html/public/storage \
|
||||||
&& mkdir -p /pelican-data /var/run/supervisord /etc/supercronic \
|
&& ln -s /pelican-data/storage/avatars /var/www/html/storage/app/public/avatars \
|
||||||
# Finally allow www-data write permissions where necessary
|
&& ln -s /pelican-data/storage/fonts /var/www/html/storage/app/public/fonts \
|
||||||
&& chown -R www-data:www-data /pelican-data ./storage ./bootstrap/cache /var/run/supervisord \
|
# Allow www-data write permissions where necessary
|
||||||
&& chmod -R u+rwX,g+rwX,o-rwx /pelican-data ./storage ./bootstrap/cache /var/run/supervisord
|
&& chown -R www-data:www-data /pelican-data ./storage ./bootstrap/cache /var/run/supervisord /var/www/html/public/storage \
|
||||||
|
&& chmod -R u+rwX,g+rwX,o-rwx /pelican-data ./storage ./bootstrap/cache /var/run/supervisord \
|
||||||
|
&& chown -R www-data: /usr/local/etc/php/
|
||||||
|
|
||||||
# Configure Supervisor
|
# Configure Supervisor
|
||||||
COPY docker/supervisord.conf /etc/supervisord.conf
|
COPY docker/supervisord.conf /etc/supervisord.conf
|
||||||
@ -97,10 +94,11 @@ COPY docker/Caddyfile /etc/caddy/Caddyfile
|
|||||||
# Add Laravel scheduler to crontab
|
# Add Laravel scheduler to crontab
|
||||||
COPY docker/crontab /etc/supercronic/crontab
|
COPY docker/crontab /etc/supercronic/crontab
|
||||||
|
|
||||||
COPY docker/entrypoint.sh ./docker/entrypoint.sh
|
COPY docker/entrypoint.sh /entrypoint.sh
|
||||||
|
COPY docker/healthcheck.sh /healthcheck.sh
|
||||||
|
|
||||||
HEALTHCHECK --interval=5m --timeout=10s --start-period=5s --retries=3 \
|
HEALTHCHECK --interval=5m --timeout=10s --start-period=5s --retries=3 \
|
||||||
CMD curl -f http://localhost/up || exit 1
|
CMD /bin/ash /healthcheck.sh
|
||||||
|
|
||||||
EXPOSE 80 443
|
EXPOSE 80 443
|
||||||
|
|
||||||
@ -108,5 +106,5 @@ VOLUME /pelican-data
|
|||||||
|
|
||||||
USER www-data
|
USER www-data
|
||||||
|
|
||||||
ENTRYPOINT [ "/bin/ash", "docker/entrypoint.sh" ]
|
ENTRYPOINT [ "/bin/ash", "/entrypoint.sh" ]
|
||||||
CMD [ "supervisord", "-n", "-c", "/etc/supervisord.conf" ]
|
CMD [ "supervisord", "-n", "-c", "/etc/supervisord.conf" ]
|
||||||
|
|||||||
@ -1,10 +1,10 @@
|
|||||||
# ================================
|
# ================================
|
||||||
# Stage 0: Build PHP Base Image
|
# Stage 0: Build PHP Base Image
|
||||||
# ================================
|
# ================================
|
||||||
FROM --platform=$TARGETOS/$TARGETARCH php:8.3-fpm-alpine
|
FROM --platform=$TARGETOS/$TARGETARCH php:8.4-fpm-alpine
|
||||||
|
|
||||||
ADD --chmod=0755 https://github.com/mlocati/docker-php-extension-installer/releases/latest/download/install-php-extensions /usr/local/bin/
|
ADD --chmod=0755 https://github.com/mlocati/docker-php-extension-installer/releases/latest/download/install-php-extensions /usr/local/bin/
|
||||||
|
|
||||||
RUN install-php-extensions bcmath gd intl zip opcache pcntl posix pdo_mysql
|
RUN install-php-extensions bcmath gd intl zip opcache pcntl posix pdo_mysql pdo_pgsql
|
||||||
|
|
||||||
RUN rm /usr/local/bin/install-php-extensions
|
RUN rm /usr/local/bin/install-php-extensions
|
||||||
114
Dockerfile.dev
Normal file
114
Dockerfile.dev
Normal file
@ -0,0 +1,114 @@
|
|||||||
|
# syntax=docker.io/docker/dockerfile:1.13-labs
|
||||||
|
# Pelican Development Dockerfile
|
||||||
|
|
||||||
|
FROM --platform=$TARGETOS/$TARGETARCH php:8.4-fpm-alpine AS base
|
||||||
|
|
||||||
|
ADD --chmod=0755 https://github.com/mlocati/docker-php-extension-installer/releases/latest/download/install-php-extensions /usr/local/bin/
|
||||||
|
|
||||||
|
RUN install-php-extensions bcmath gd intl zip opcache pcntl posix pdo_mysql pdo_pgsql
|
||||||
|
|
||||||
|
RUN rm /usr/local/bin/install-php-extensions
|
||||||
|
|
||||||
|
# ================================
|
||||||
|
# Stage 1-1: Composer Install
|
||||||
|
# ================================
|
||||||
|
FROM --platform=$TARGETOS/$TARGETARCH base AS composer
|
||||||
|
|
||||||
|
WORKDIR /build
|
||||||
|
|
||||||
|
COPY --from=composer:latest /usr/bin/composer /usr/local/bin/composer
|
||||||
|
|
||||||
|
# Copy bare minimum to install Composer dependencies
|
||||||
|
COPY composer.json composer.lock ./
|
||||||
|
|
||||||
|
RUN composer install --no-dev --no-interaction --no-autoloader --no-scripts
|
||||||
|
|
||||||
|
# ================================
|
||||||
|
# Stage 1-2: Yarn Install
|
||||||
|
# ================================
|
||||||
|
FROM --platform=$TARGETOS/$TARGETARCH node:20-alpine AS yarn
|
||||||
|
|
||||||
|
WORKDIR /build
|
||||||
|
|
||||||
|
# Copy bare minimum to install Yarn dependencies
|
||||||
|
COPY package.json yarn.lock ./
|
||||||
|
|
||||||
|
RUN yarn config set network-timeout 300000 \
|
||||||
|
&& yarn install --frozen-lockfile
|
||||||
|
|
||||||
|
# ================================
|
||||||
|
# Stage 2-1: Composer Optimize
|
||||||
|
# ================================
|
||||||
|
FROM --platform=$TARGETOS/$TARGETARCH composer AS composerbuild
|
||||||
|
|
||||||
|
# Copy full code to optimize autoload
|
||||||
|
COPY --exclude=Caddyfile --exclude=docker/ . ./
|
||||||
|
|
||||||
|
RUN composer dump-autoload --optimize
|
||||||
|
|
||||||
|
# ================================
|
||||||
|
# Stage 2-2: Build Frontend Assets
|
||||||
|
# ================================
|
||||||
|
FROM --platform=$TARGETOS/$TARGETARCH yarn AS yarnbuild
|
||||||
|
|
||||||
|
WORKDIR /build
|
||||||
|
|
||||||
|
# Copy full code
|
||||||
|
COPY --exclude=Caddyfile --exclude=docker/ . ./
|
||||||
|
COPY --from=composer /build .
|
||||||
|
|
||||||
|
RUN yarn run build
|
||||||
|
|
||||||
|
# ================================
|
||||||
|
# Stage 5: Build Final Application Image
|
||||||
|
# ================================
|
||||||
|
FROM --platform=$TARGETOS/$TARGETARCH base AS final
|
||||||
|
|
||||||
|
WORKDIR /var/www/html
|
||||||
|
|
||||||
|
# Install additional required libraries
|
||||||
|
RUN apk add --no-cache \
|
||||||
|
caddy ca-certificates supervisor supercronic fcgi coreutils
|
||||||
|
|
||||||
|
COPY --chown=root:www-data --chmod=640 --from=composerbuild /build .
|
||||||
|
COPY --chown=root:www-data --chmod=640 --from=yarnbuild /build/public ./public
|
||||||
|
|
||||||
|
# Set permissions
|
||||||
|
# First ensure all files are owned by root and restrict www-data to read access
|
||||||
|
RUN chown root:www-data ./ \
|
||||||
|
&& chmod 750 ./ \
|
||||||
|
# Files should not have execute set, but directories need it
|
||||||
|
&& find ./ -type d -exec chmod 750 {} \; \
|
||||||
|
# Create necessary directories
|
||||||
|
&& mkdir -p /pelican-data/storage /var/www/html/storage/app/public /var/run/supervisord /etc/supercronic \
|
||||||
|
# Symlinks for env, database, and avatars
|
||||||
|
&& ln -s /pelican-data/.env ./.env \
|
||||||
|
&& ln -s /pelican-data/database/database.sqlite ./database/database.sqlite \
|
||||||
|
&& ln -sf /var/www/html/storage/app/public /var/www/html/public/storage \
|
||||||
|
&& ln -s /pelican-data/storage/avatars /var/www/html/storage/app/public/avatars \
|
||||||
|
&& ln -s /pelican-data/storage/fonts /var/www/html/storage/app/public/fonts \
|
||||||
|
# Allow www-data write permissions where necessary
|
||||||
|
&& chown -R www-data:www-data /pelican-data ./storage ./bootstrap/cache /var/run/supervisord /var/www/html/public/storage \
|
||||||
|
&& chmod -R u+rwX,g+rwX,o-rwx /pelican-data ./storage ./bootstrap/cache /var/run/supervisord \
|
||||||
|
&& chown -R www-data: /usr/local/etc/php/
|
||||||
|
|
||||||
|
# Configure Supervisor
|
||||||
|
COPY docker/supervisord.conf /etc/supervisord.conf
|
||||||
|
COPY docker/Caddyfile /etc/caddy/Caddyfile
|
||||||
|
# Add Laravel scheduler to crontab
|
||||||
|
COPY docker/crontab /etc/supercronic/crontab
|
||||||
|
|
||||||
|
COPY docker/entrypoint.sh /entrypoint.sh
|
||||||
|
COPY docker/healthcheck.sh /healthcheck.sh
|
||||||
|
|
||||||
|
HEALTHCHECK --interval=5m --timeout=10s --start-period=5s --retries=3 \
|
||||||
|
CMD /bin/ash /healthcheck.sh
|
||||||
|
|
||||||
|
EXPOSE 80 443
|
||||||
|
|
||||||
|
VOLUME /pelican-data
|
||||||
|
|
||||||
|
USER www-data
|
||||||
|
|
||||||
|
ENTRYPOINT [ "/bin/ash", "/entrypoint.sh" ]
|
||||||
|
CMD [ "supervisord", "-n", "-c", "/etc/supervisord.conf" ]
|
||||||
@ -14,9 +14,9 @@ class NodeVersionsCheck extends Check
|
|||||||
|
|
||||||
public function run(): Result
|
public function run(): Result
|
||||||
{
|
{
|
||||||
$all = Node::query()->count();
|
$all = Node::all();
|
||||||
|
|
||||||
if ($all === 0) {
|
if ($all->isEmpty()) {
|
||||||
$result = Result::make()
|
$result = Result::make()
|
||||||
->notificationMessage(trans('admin/health.results.nodeversions.no_nodes_created'))
|
->notificationMessage(trans('admin/health.results.nodeversions.no_nodes_created'))
|
||||||
->shortSummary(trans('admin/health.results.nodeversions.no_nodes'));
|
->shortSummary(trans('admin/health.results.nodeversions.no_nodes'));
|
||||||
@ -25,16 +25,18 @@ class NodeVersionsCheck extends Check
|
|||||||
return $result;
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
$latestVersion = $this->versionService->latestWingsVersion();
|
$outdated = $all
|
||||||
|
->filter(fn (Node $node) => !isset($node->systemInformation()['exception']) && !$this->versionService->isLatestWings($node->systemInformation()['version']))
|
||||||
$outdated = Node::query()->get()
|
|
||||||
->filter(fn (Node $node) => !isset($node->systemInformation()['exception']) && $node->systemInformation()['version'] !== $latestVersion)
|
|
||||||
->count();
|
->count();
|
||||||
|
|
||||||
|
$all = $all->count();
|
||||||
|
$latestVersion = $this->versionService->latestWingsVersion();
|
||||||
|
|
||||||
$result = Result::make()
|
$result = Result::make()
|
||||||
->meta([
|
->meta([
|
||||||
'all' => $all,
|
'all' => $all,
|
||||||
'outdated' => $outdated,
|
'outdated' => $outdated,
|
||||||
|
'latestVersion' => $latestVersion,
|
||||||
])
|
])
|
||||||
->shortSummary($outdated === 0 ? trans('admin/health.results.nodeversions.all_up_to_date') : trans('admin/health.results.nodeversions.outdated', ['outdated' => $outdated, 'all' => $all]));
|
->shortSummary($outdated === 0 ? trans('admin/health.results.nodeversions.all_up_to_date') : trans('admin/health.results.nodeversions.outdated', ['outdated' => $outdated, 'all' => $all]));
|
||||||
|
|
||||||
|
|||||||
@ -2,10 +2,14 @@
|
|||||||
|
|
||||||
namespace App\Console\Commands\Egg;
|
namespace App\Console\Commands\Egg;
|
||||||
|
|
||||||
|
use App\Enums\EggFormat;
|
||||||
use App\Models\Egg;
|
use App\Models\Egg;
|
||||||
use App\Services\Eggs\Sharing\EggExporterService;
|
use App\Services\Eggs\Sharing\EggExporterService;
|
||||||
use Exception;
|
use Exception;
|
||||||
use Illuminate\Console\Command;
|
use Illuminate\Console\Command;
|
||||||
|
use Illuminate\Support\Facades\Http;
|
||||||
|
use JsonException;
|
||||||
|
use Symfony\Component\Yaml\Yaml;
|
||||||
|
|
||||||
class CheckEggUpdatesCommand extends Command
|
class CheckEggUpdatesCommand extends Command
|
||||||
{
|
{
|
||||||
@ -23,6 +27,9 @@ class CheckEggUpdatesCommand extends Command
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws JsonException
|
||||||
|
*/
|
||||||
private function check(Egg $egg, EggExporterService $exporterService): void
|
private function check(Egg $egg, EggExporterService $exporterService): void
|
||||||
{
|
{
|
||||||
if (is_null($egg->update_url)) {
|
if (is_null($egg->update_url)) {
|
||||||
@ -31,22 +38,24 @@ class CheckEggUpdatesCommand extends Command
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
$currentJson = json_decode($exporterService->handle($egg->id));
|
$ext = strtolower(pathinfo(parse_url($egg->update_url, PHP_URL_PATH), PATHINFO_EXTENSION));
|
||||||
unset($currentJson->exported_at);
|
$isYaml = in_array($ext, ['yaml', 'yml']);
|
||||||
|
|
||||||
$updatedEgg = file_get_contents($egg->update_url);
|
$local = $isYaml
|
||||||
assert($updatedEgg !== false);
|
? Yaml::parse($exporterService->handle($egg->id, EggFormat::YAML))
|
||||||
$updatedJson = json_decode($updatedEgg);
|
: json_decode($exporterService->handle($egg->id, EggFormat::JSON), true);
|
||||||
unset($updatedJson->exported_at);
|
|
||||||
|
|
||||||
if (md5(json_encode($currentJson, JSON_THROW_ON_ERROR)) === md5(json_encode($updatedJson, JSON_THROW_ON_ERROR))) {
|
$remote = Http::timeout(5)->connectTimeout(1)->get($egg->update_url)->throw()->body();
|
||||||
$this->info("$egg->name: Up-to-date");
|
$remote = $isYaml ? Yaml::parse($remote) : json_decode($remote, true);
|
||||||
cache()->put("eggs.$egg->uuid.update", false, now()->addHour());
|
|
||||||
|
|
||||||
return;
|
unset($local['exported_at'], $remote['exported_at']);
|
||||||
}
|
|
||||||
|
|
||||||
$this->warn("$egg->name: Found update");
|
$localHash = md5(json_encode($local, JSON_THROW_ON_ERROR));
|
||||||
cache()->put("eggs.$egg->uuid.update", true, now()->addHour());
|
$remoteHash = md5(json_encode($remote, JSON_THROW_ON_ERROR));
|
||||||
|
|
||||||
|
$status = $localHash === $remoteHash ? 'Up-to-date' : 'Found update';
|
||||||
|
$this->{($localHash === $remoteHash) ? 'info' : 'warn'}("$egg->name: $status");
|
||||||
|
|
||||||
|
cache()->put("eggs.$egg->uuid.update", $localHash !== $remoteHash, now()->addHour());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
46
app/Console/Commands/Egg/UpdateEggIndexCommand.php
Normal file
46
app/Console/Commands/Egg/UpdateEggIndexCommand.php
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Console\Commands\Egg;
|
||||||
|
|
||||||
|
use Exception;
|
||||||
|
use Illuminate\Console\Command;
|
||||||
|
use Illuminate\Support\Facades\Http;
|
||||||
|
|
||||||
|
class UpdateEggIndexCommand extends Command
|
||||||
|
{
|
||||||
|
protected $signature = 'p:egg:update-index';
|
||||||
|
|
||||||
|
public function handle(): int
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
$data = Http::timeout(5)->connectTimeout(1)->get('https://raw.githubusercontent.com/pelican-eggs/pelican-eggs.github.io/refs/heads/main/content/pelican.json')->throw()->json();
|
||||||
|
} catch (Exception $exception) {
|
||||||
|
$this->error($exception->getMessage());
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
$index = [];
|
||||||
|
foreach ($data['nests'] as $nest) {
|
||||||
|
$nestName = $nest['nest_type'];
|
||||||
|
|
||||||
|
$this->info("Nest: $nestName");
|
||||||
|
|
||||||
|
$nestEggs = [];
|
||||||
|
foreach ($nest['Eggs'] as $egg) {
|
||||||
|
$eggName = $egg['egg']['name'];
|
||||||
|
|
||||||
|
$this->comment("Egg: $eggName");
|
||||||
|
|
||||||
|
$nestEggs[$egg['download_url']] = $eggName;
|
||||||
|
}
|
||||||
|
$index[$nestName] = $nestEggs;
|
||||||
|
|
||||||
|
$this->info('');
|
||||||
|
}
|
||||||
|
|
||||||
|
cache()->forever('eggs.index', $index);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -3,7 +3,6 @@
|
|||||||
namespace App\Console\Commands\Environment;
|
namespace App\Console\Commands\Environment;
|
||||||
|
|
||||||
use Illuminate\Console\Command;
|
use Illuminate\Console\Command;
|
||||||
use Illuminate\Support\Facades\Artisan;
|
|
||||||
|
|
||||||
class AppSettingsCommand extends Command
|
class AppSettingsCommand extends Command
|
||||||
{
|
{
|
||||||
@ -21,9 +20,13 @@ class AppSettingsCommand extends Command
|
|||||||
|
|
||||||
if (!config('app.key')) {
|
if (!config('app.key')) {
|
||||||
$this->comment('Generating app key');
|
$this->comment('Generating app key');
|
||||||
Artisan::call('key:generate');
|
$this->call('key:generate');
|
||||||
}
|
}
|
||||||
|
|
||||||
Artisan::call('filament:optimize');
|
$this->comment('Creating storage link');
|
||||||
|
$this->call('storage:link');
|
||||||
|
|
||||||
|
$this->comment('Caching components & icons');
|
||||||
|
$this->call('filament:optimize');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -6,6 +6,7 @@ use App\Traits\EnvironmentWriterTrait;
|
|||||||
use Illuminate\Console\Command;
|
use Illuminate\Console\Command;
|
||||||
use Illuminate\Contracts\Console\Kernel;
|
use Illuminate\Contracts\Console\Kernel;
|
||||||
use Illuminate\Database\DatabaseManager;
|
use Illuminate\Database\DatabaseManager;
|
||||||
|
use PDOException;
|
||||||
|
|
||||||
class DatabaseSettingsCommand extends Command
|
class DatabaseSettingsCommand extends Command
|
||||||
{
|
{
|
||||||
@ -105,7 +106,7 @@ class DatabaseSettingsCommand extends Command
|
|||||||
]);
|
]);
|
||||||
|
|
||||||
$this->database->connection('_panel_command_test')->getPdo();
|
$this->database->connection('_panel_command_test')->getPdo();
|
||||||
} catch (\PDOException $exception) {
|
} catch (PDOException $exception) {
|
||||||
$this->output->error(sprintf('Unable to connect to the MySQL server using the provided credentials. The error returned was "%s".', $exception->getMessage()));
|
$this->output->error(sprintf('Unable to connect to the MySQL server using the provided credentials. The error returned was "%s".', $exception->getMessage()));
|
||||||
$this->output->error(trans('commands.database_settings.DB_error_2'));
|
$this->output->error(trans('commands.database_settings.DB_error_2'));
|
||||||
|
|
||||||
@ -165,7 +166,7 @@ class DatabaseSettingsCommand extends Command
|
|||||||
]);
|
]);
|
||||||
|
|
||||||
$this->database->connection('_panel_command_test')->getPdo();
|
$this->database->connection('_panel_command_test')->getPdo();
|
||||||
} catch (\PDOException $exception) {
|
} catch (PDOException $exception) {
|
||||||
$this->output->error(sprintf('Unable to connect to the MariaDB server using the provided credentials. The error returned was "%s".', $exception->getMessage()));
|
$this->output->error(sprintf('Unable to connect to the MariaDB server using the provided credentials. The error returned was "%s".', $exception->getMessage()));
|
||||||
$this->output->error(trans('commands.database_settings.DB_error_2'));
|
$this->output->error(trans('commands.database_settings.DB_error_2'));
|
||||||
|
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Console\Commands\Environment;
|
namespace App\Console\Commands\Environment;
|
||||||
|
|
||||||
|
use App\Exceptions\PanelException;
|
||||||
use App\Traits\EnvironmentWriterTrait;
|
use App\Traits\EnvironmentWriterTrait;
|
||||||
use Illuminate\Console\Command;
|
use Illuminate\Console\Command;
|
||||||
|
|
||||||
@ -28,7 +29,7 @@ class EmailSettingsCommand extends Command
|
|||||||
/**
|
/**
|
||||||
* Handle command execution.
|
* Handle command execution.
|
||||||
*
|
*
|
||||||
* @throws \App\Exceptions\PanelException
|
* @throws PanelException
|
||||||
*/
|
*/
|
||||||
public function handle(): void
|
public function handle(): void
|
||||||
{
|
{
|
||||||
|
|||||||
@ -18,6 +18,17 @@ class QueueWorkerServiceCommand extends Command
|
|||||||
|
|
||||||
public function handle(): void
|
public function handle(): void
|
||||||
{
|
{
|
||||||
|
if (@file_exists('/.dockerenv')) {
|
||||||
|
$result = Process::run('supervisorctl restart queue-worker');
|
||||||
|
if ($result->failed()) {
|
||||||
|
$this->error('Error restarting service: ' . $result->errorOutput());
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
$this->line('Queue worker service file updated successfully.');
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
$serviceName = $this->option('service-name') ?? $this->ask('Queue worker service name', 'pelican-queue');
|
$serviceName = $this->option('service-name') ?? $this->ask('Queue worker service name', 'pelican-queue');
|
||||||
$path = '/etc/systemd/system/' . $serviceName . '.service';
|
$path = '/etc/systemd/system/' . $serviceName . '.service';
|
||||||
|
|
||||||
|
|||||||
@ -35,7 +35,7 @@ class RedisSetupCommand extends Command
|
|||||||
{
|
{
|
||||||
$this->variables['CACHE_STORE'] = 'redis';
|
$this->variables['CACHE_STORE'] = 'redis';
|
||||||
$this->variables['QUEUE_CONNECTION'] = 'redis';
|
$this->variables['QUEUE_CONNECTION'] = 'redis';
|
||||||
$this->variables['SESSION_DRIVERS'] = 'redis';
|
$this->variables['SESSION_DRIVER'] = 'redis';
|
||||||
|
|
||||||
$this->requestRedisSettings();
|
$this->requestRedisSettings();
|
||||||
|
|
||||||
|
|||||||
@ -4,8 +4,9 @@ namespace App\Console\Commands\Maintenance;
|
|||||||
|
|
||||||
use Carbon\Carbon;
|
use Carbon\Carbon;
|
||||||
use Illuminate\Console\Command;
|
use Illuminate\Console\Command;
|
||||||
use Illuminate\Contracts\Filesystem\Filesystem;
|
|
||||||
use Illuminate\Contracts\Filesystem\Factory as FilesystemFactory;
|
use Illuminate\Contracts\Filesystem\Factory as FilesystemFactory;
|
||||||
|
use Illuminate\Contracts\Filesystem\Filesystem;
|
||||||
|
use SplFileInfo;
|
||||||
|
|
||||||
class CleanServiceBackupFilesCommand extends Command
|
class CleanServiceBackupFilesCommand extends Command
|
||||||
{
|
{
|
||||||
@ -32,9 +33,10 @@ class CleanServiceBackupFilesCommand extends Command
|
|||||||
*/
|
*/
|
||||||
public function handle(): void
|
public function handle(): void
|
||||||
{
|
{
|
||||||
|
/** @var SplFileInfo[] */
|
||||||
$files = $this->disk->files('services/.bak');
|
$files = $this->disk->files('services/.bak');
|
||||||
|
|
||||||
collect($files)->each(function (\SplFileInfo $file) {
|
collect($files)->each(function ($file) {
|
||||||
$lastModified = Carbon::createFromTimestamp($this->disk->lastModified($file->getPath()));
|
$lastModified = Carbon::createFromTimestamp($this->disk->lastModified($file->getPath()));
|
||||||
if ($lastModified->diffInMinutes(Carbon::now()) > self::BACKUP_THRESHOLD_MINUTES) {
|
if ($lastModified->diffInMinutes(Carbon::now()) > self::BACKUP_THRESHOLD_MINUTES) {
|
||||||
$this->disk->delete($file->getPath());
|
$this->disk->delete($file->getPath());
|
||||||
|
|||||||
@ -5,6 +5,7 @@ namespace App\Console\Commands\Maintenance;
|
|||||||
use App\Models\Backup;
|
use App\Models\Backup;
|
||||||
use Carbon\CarbonImmutable;
|
use Carbon\CarbonImmutable;
|
||||||
use Illuminate\Console\Command;
|
use Illuminate\Console\Command;
|
||||||
|
use InvalidArgumentException;
|
||||||
|
|
||||||
class PruneOrphanedBackupsCommand extends Command
|
class PruneOrphanedBackupsCommand extends Command
|
||||||
{
|
{
|
||||||
@ -16,7 +17,7 @@ class PruneOrphanedBackupsCommand extends Command
|
|||||||
{
|
{
|
||||||
$since = $this->option('prune-age') ?? config('backups.prune_age', 360);
|
$since = $this->option('prune-age') ?? config('backups.prune_age', 360);
|
||||||
if (!$since || !is_digit($since)) {
|
if (!$since || !is_digit($since)) {
|
||||||
throw new \InvalidArgumentException('The "--prune-age" argument must be a value greater than 0.');
|
throw new InvalidArgumentException('The "--prune-age" argument must be a value greater than 0.');
|
||||||
}
|
}
|
||||||
|
|
||||||
$query = Backup::query()
|
$query = Backup::query()
|
||||||
|
|||||||
@ -2,8 +2,9 @@
|
|||||||
|
|
||||||
namespace App\Console\Commands\Node;
|
namespace App\Console\Commands\Node;
|
||||||
|
|
||||||
|
use App\Exceptions\Model\DataValidationException;
|
||||||
|
use App\Models\Node;
|
||||||
use Illuminate\Console\Command;
|
use Illuminate\Console\Command;
|
||||||
use App\Services\Nodes\NodeCreationService;
|
|
||||||
|
|
||||||
class MakeNodeCommand extends Command
|
class MakeNodeCommand extends Command
|
||||||
{
|
{
|
||||||
@ -24,24 +25,17 @@ class MakeNodeCommand extends Command
|
|||||||
{--overallocateCpu= : Enter the amount of cpu to overallocate (% or -1 to overallocate the maximum).}
|
{--overallocateCpu= : Enter the amount of cpu to overallocate (% or -1 to overallocate the maximum).}
|
||||||
{--uploadSize= : Enter the maximum upload filesize.}
|
{--uploadSize= : Enter the maximum upload filesize.}
|
||||||
{--daemonListeningPort= : Enter the daemon listening port.}
|
{--daemonListeningPort= : Enter the daemon listening port.}
|
||||||
|
{--daemonConnectingPort= : Enter the daemon connecting port.}
|
||||||
{--daemonSFTPPort= : Enter the daemon SFTP listening port.}
|
{--daemonSFTPPort= : Enter the daemon SFTP listening port.}
|
||||||
{--daemonSFTPAlias= : Enter the daemon SFTP alias.}
|
{--daemonSFTPAlias= : Enter the daemon SFTP alias.}
|
||||||
{--daemonBase= : Enter the base folder.}';
|
{--daemonBase= : Enter the base folder.}';
|
||||||
|
|
||||||
protected $description = 'Creates a new node on the system via the CLI.';
|
protected $description = 'Creates a new node on the system via the CLI.';
|
||||||
|
|
||||||
/**
|
|
||||||
* MakeNodeCommand constructor.
|
|
||||||
*/
|
|
||||||
public function __construct(private NodeCreationService $creationService)
|
|
||||||
{
|
|
||||||
parent::__construct();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handle the command execution process.
|
* Handle the command execution process.
|
||||||
*
|
*
|
||||||
* @throws \App\Exceptions\Model\DataValidationException
|
* @throws DataValidationException
|
||||||
*/
|
*/
|
||||||
public function handle(): void
|
public function handle(): void
|
||||||
{
|
{
|
||||||
@ -65,11 +59,12 @@ class MakeNodeCommand extends Command
|
|||||||
$data['cpu_overallocate'] = $this->option('overallocateCpu') ?? $this->ask(trans('commands.make_node.cpu_overallocate'), '-1');
|
$data['cpu_overallocate'] = $this->option('overallocateCpu') ?? $this->ask(trans('commands.make_node.cpu_overallocate'), '-1');
|
||||||
$data['upload_size'] = $this->option('uploadSize') ?? $this->ask(trans('commands.make_node.upload_size'), '256');
|
$data['upload_size'] = $this->option('uploadSize') ?? $this->ask(trans('commands.make_node.upload_size'), '256');
|
||||||
$data['daemon_listen'] = $this->option('daemonListeningPort') ?? $this->ask(trans('commands.make_node.daemonListen'), '8080');
|
$data['daemon_listen'] = $this->option('daemonListeningPort') ?? $this->ask(trans('commands.make_node.daemonListen'), '8080');
|
||||||
|
$data['daemon_connect'] = $this->option('daemonConnectingPort') ?? $this->ask(trans('commands.make_node.daemonConnect'), '8080');
|
||||||
$data['daemon_sftp'] = $this->option('daemonSFTPPort') ?? $this->ask(trans('commands.make_node.daemonSFTP'), '2022');
|
$data['daemon_sftp'] = $this->option('daemonSFTPPort') ?? $this->ask(trans('commands.make_node.daemonSFTP'), '2022');
|
||||||
$data['daemon_sftp_alias'] = $this->option('daemonSFTPAlias') ?? $this->ask(trans('commands.make_node.daemonSFTPAlias'), '');
|
$data['daemon_sftp_alias'] = $this->option('daemonSFTPAlias') ?? $this->ask(trans('commands.make_node.daemonSFTPAlias'), '');
|
||||||
$data['daemon_base'] = $this->option('daemonBase') ?? $this->ask(trans('commands.make_node.daemonBase'), '/var/lib/pelican/volumes');
|
$data['daemon_base'] = $this->option('daemonBase') ?? $this->ask(trans('commands.make_node.daemonBase'), '/var/lib/pelican/volumes');
|
||||||
|
|
||||||
$node = $this->creationService->handle($data);
|
$node = Node::create($data);
|
||||||
$this->line(trans('commands.make_node.success', ['name' => $data['name'], 'id' => $node->id]));
|
$this->line(trans('commands.make_node.success', ['name' => $data['name'], 'id' => $node->id]));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -17,7 +17,7 @@ class NodeConfigurationCommand extends Command
|
|||||||
{
|
{
|
||||||
$column = ctype_digit((string) $this->argument('node')) ? 'id' : 'uuid';
|
$column = ctype_digit((string) $this->argument('node')) ? 'id' : 'uuid';
|
||||||
|
|
||||||
/** @var \App\Models\Node $node */
|
/** @var Node $node */
|
||||||
$node = Node::query()->where($column, $this->argument('node'))->firstOr(function () {
|
$node = Node::query()->where($column, $this->argument('node'))->firstOr(function () {
|
||||||
$this->error(trans('commands.node_config.error_not_exist'));
|
$this->error(trans('commands.node_config.error_not_exist'));
|
||||||
|
|
||||||
|
|||||||
@ -2,10 +2,10 @@
|
|||||||
|
|
||||||
namespace App\Console\Commands\Schedule;
|
namespace App\Console\Commands\Schedule;
|
||||||
|
|
||||||
use Illuminate\Console\Command;
|
|
||||||
use App\Models\Schedule;
|
use App\Models\Schedule;
|
||||||
use Illuminate\Database\Eloquent\Builder;
|
|
||||||
use App\Services\Schedules\ProcessScheduleService;
|
use App\Services\Schedules\ProcessScheduleService;
|
||||||
|
use Illuminate\Console\Command;
|
||||||
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
use Throwable;
|
use Throwable;
|
||||||
|
|
||||||
class ProcessRunnableCommand extends Command
|
class ProcessRunnableCommand extends Command
|
||||||
@ -64,7 +64,7 @@ class ProcessRunnableCommand extends Command
|
|||||||
} catch (Throwable $exception) {
|
} catch (Throwable $exception) {
|
||||||
logger()->error($exception, ['schedule_id' => $schedule->id]);
|
logger()->error($exception, ['schedule_id' => $schedule->id]);
|
||||||
|
|
||||||
$this->error(trans('commands.schedule.process.no_tasks') . " #$schedule->id: " . $exception->getMessage());
|
$this->error(trans('commands.schedule.process.error_message', ['schedules' => " #$schedule->id: " . $exception->getMessage()]));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -3,12 +3,12 @@
|
|||||||
namespace App\Console\Commands\Server;
|
namespace App\Console\Commands\Server;
|
||||||
|
|
||||||
use App\Models\Server;
|
use App\Models\Server;
|
||||||
|
use App\Repositories\Daemon\DaemonServerRepository;
|
||||||
|
use Exception;
|
||||||
use Illuminate\Console\Command;
|
use Illuminate\Console\Command;
|
||||||
use Illuminate\Database\Eloquent\Builder;
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
use Illuminate\Validation\ValidationException;
|
|
||||||
use Illuminate\Validation\Factory as ValidatorFactory;
|
use Illuminate\Validation\Factory as ValidatorFactory;
|
||||||
use App\Repositories\Daemon\DaemonPowerRepository;
|
use Illuminate\Validation\ValidationException;
|
||||||
use Exception;
|
|
||||||
|
|
||||||
class BulkPowerActionCommand extends Command
|
class BulkPowerActionCommand extends Command
|
||||||
{
|
{
|
||||||
@ -19,7 +19,7 @@ class BulkPowerActionCommand extends Command
|
|||||||
|
|
||||||
protected $description = 'Perform bulk power management on large groupings of servers or nodes at once.';
|
protected $description = 'Perform bulk power management on large groupings of servers or nodes at once.';
|
||||||
|
|
||||||
public function handle(DaemonPowerRepository $powerRepository, ValidatorFactory $validator): void
|
public function handle(DaemonServerRepository $serverRepository, ValidatorFactory $validator): void
|
||||||
{
|
{
|
||||||
$action = $this->argument('action');
|
$action = $this->argument('action');
|
||||||
$nodes = empty($this->option('nodes')) ? [] : explode(',', $this->option('nodes'));
|
$nodes = empty($this->option('nodes')) ? [] : explode(',', $this->option('nodes'));
|
||||||
@ -52,7 +52,7 @@ class BulkPowerActionCommand extends Command
|
|||||||
|
|
||||||
$bar = $this->output->createProgressBar($count);
|
$bar = $this->output->createProgressBar($count);
|
||||||
|
|
||||||
$this->getQueryBuilder($servers, $nodes)->get()->each(function ($server, int $index) use ($action, $powerRepository, &$bar): mixed {
|
$this->getQueryBuilder($servers, $nodes)->get()->each(function ($server, int $index) use ($action, $serverRepository, &$bar): mixed {
|
||||||
$bar->clear();
|
$bar->clear();
|
||||||
|
|
||||||
if (!$server instanceof Server) {
|
if (!$server instanceof Server) {
|
||||||
@ -60,7 +60,7 @@ class BulkPowerActionCommand extends Command
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$powerRepository->setServer($server)->send($action);
|
$serverRepository->setServer($server)->power($action);
|
||||||
} catch (Exception $exception) {
|
} catch (Exception $exception) {
|
||||||
$this->output->error(trans('command/messages.server.power.action_failed', [
|
$this->output->error(trans('command/messages.server.power.action_failed', [
|
||||||
'name' => $server->name,
|
'name' => $server->name,
|
||||||
|
|||||||
@ -1,193 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Console\Commands;
|
|
||||||
|
|
||||||
use Illuminate\Console\Command;
|
|
||||||
use App\Console\Kernel;
|
|
||||||
use Symfony\Component\Process\Process;
|
|
||||||
use Symfony\Component\Console\Helper\ProgressBar;
|
|
||||||
|
|
||||||
class UpgradeCommand extends Command
|
|
||||||
{
|
|
||||||
protected const DEFAULT_URL = 'https://github.com/pelican-dev/panel/releases/%s/panel.tar.gz';
|
|
||||||
|
|
||||||
protected $signature = 'p:upgrade
|
|
||||||
{--user= : The user that PHP runs under. All files will be owned by this user.}
|
|
||||||
{--group= : The group that PHP runs under. All files will be owned by this group.}
|
|
||||||
{--url= : The specific archive to download.}
|
|
||||||
{--release= : A specific version to download from GitHub. Leave blank to use latest.}
|
|
||||||
{--skip-download : If set no archive will be downloaded.}';
|
|
||||||
|
|
||||||
protected $description = 'Downloads a new archive from GitHub and then executes the normal upgrade commands.';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Executes an upgrade command which will run through all of our standard
|
|
||||||
* Panel commands and enable users to basically just download
|
|
||||||
* the archive and execute this and be done.
|
|
||||||
*
|
|
||||||
* This places the application in maintenance mode as well while the commands
|
|
||||||
* are being executed.
|
|
||||||
*
|
|
||||||
* @throws \Exception
|
|
||||||
*/
|
|
||||||
public function handle(): void
|
|
||||||
{
|
|
||||||
$skipDownload = $this->option('skip-download');
|
|
||||||
if (!$skipDownload) {
|
|
||||||
$this->output->warning(trans('commands.upgrade.integrity'));
|
|
||||||
$this->output->comment(trans('commands.upgrade.source_url'));
|
|
||||||
$this->line($this->getUrl());
|
|
||||||
}
|
|
||||||
|
|
||||||
$user = 'www-data';
|
|
||||||
$group = 'www-data';
|
|
||||||
if ($this->input->isInteractive()) {
|
|
||||||
if (!$skipDownload) {
|
|
||||||
$skipDownload = !$this->confirm(trans('commands.upgrade.skipDownload'), true);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (is_null($this->option('user'))) {
|
|
||||||
$userDetails = function_exists('posix_getpwuid') ? posix_getpwuid(fileowner('public')) : [];
|
|
||||||
$user = $userDetails['name'] ?? 'www-data';
|
|
||||||
|
|
||||||
$message = trans('commands.upgrade.webserver_user', ['user' => $user]);
|
|
||||||
if (!$this->confirm($message, true)) {
|
|
||||||
$user = $this->anticipate(
|
|
||||||
trans('commands.upgrade.name_webserver'),
|
|
||||||
[
|
|
||||||
'www-data',
|
|
||||||
'nginx',
|
|
||||||
'apache',
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (is_null($this->option('group'))) {
|
|
||||||
$groupDetails = function_exists('posix_getgrgid') ? posix_getgrgid(filegroup('public')) : [];
|
|
||||||
$group = $groupDetails['name'] ?? 'www-data';
|
|
||||||
|
|
||||||
$message = trans('commands.upgrade.group_webserver', ['group' => $user]);
|
|
||||||
if (!$this->confirm($message, true)) {
|
|
||||||
$group = $this->anticipate(
|
|
||||||
trans('commands.upgrade.group_webserver_question'),
|
|
||||||
[
|
|
||||||
'www-data',
|
|
||||||
'nginx',
|
|
||||||
'apache',
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!$this->confirm(trans('commands.upgrade.are_your_sure'))) {
|
|
||||||
$this->warn(trans('commands.upgrade.terminated'));
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ini_set('output_buffering', '0');
|
|
||||||
$bar = $this->output->createProgressBar($skipDownload ? 9 : 10);
|
|
||||||
$bar->start();
|
|
||||||
|
|
||||||
if (!$skipDownload) {
|
|
||||||
$this->withProgress($bar, function () {
|
|
||||||
$this->line("\$upgrader> curl -L \"{$this->getUrl()}\" | tar -xzv");
|
|
||||||
$process = Process::fromShellCommandline("curl -L \"{$this->getUrl()}\" | tar -xzv");
|
|
||||||
$process->run(function ($type, $buffer) {
|
|
||||||
$this->{$type === Process::ERR ? 'error' : 'line'}($buffer);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->withProgress($bar, function () {
|
|
||||||
$this->line('$upgrader> php artisan down');
|
|
||||||
$this->call('down');
|
|
||||||
});
|
|
||||||
|
|
||||||
$this->withProgress($bar, function () {
|
|
||||||
$this->line('$upgrader> chmod -R 755 storage bootstrap/cache');
|
|
||||||
$process = new Process(['chmod', '-R', '755', 'storage', 'bootstrap/cache']);
|
|
||||||
$process->run(function ($type, $buffer) {
|
|
||||||
$this->{$type === Process::ERR ? 'error' : 'line'}($buffer);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
$this->withProgress($bar, function () {
|
|
||||||
$command = ['composer', 'install', '--no-ansi'];
|
|
||||||
if (config('app.env') === 'production' && !config('app.debug')) {
|
|
||||||
$command[] = '--optimize-autoloader';
|
|
||||||
$command[] = '--no-dev';
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->line('$upgrader> ' . implode(' ', $command));
|
|
||||||
$process = new Process($command);
|
|
||||||
$process->setTimeout(10 * 60);
|
|
||||||
$process->run(function ($type, $buffer) {
|
|
||||||
$this->line($buffer);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
/** @var \Illuminate\Foundation\Application $app */
|
|
||||||
$app = require __DIR__ . '/../../../bootstrap/app.php';
|
|
||||||
/** @var \App\Console\Kernel $kernel */
|
|
||||||
$kernel = $app->make(Kernel::class);
|
|
||||||
$kernel->bootstrap();
|
|
||||||
$this->setLaravel($app);
|
|
||||||
|
|
||||||
$this->withProgress($bar, function () {
|
|
||||||
$this->line('$upgrader> php artisan view:clear');
|
|
||||||
$this->call('view:clear');
|
|
||||||
});
|
|
||||||
|
|
||||||
$this->withProgress($bar, function () {
|
|
||||||
$this->line('$upgrader> php artisan config:clear');
|
|
||||||
$this->call('config:clear');
|
|
||||||
});
|
|
||||||
|
|
||||||
$this->withProgress($bar, function () {
|
|
||||||
$this->line('$upgrader> php artisan migrate --force --seed');
|
|
||||||
$this->call('migrate', ['--force' => true, '--seed' => true]);
|
|
||||||
});
|
|
||||||
|
|
||||||
$this->withProgress($bar, function () use ($user, $group) {
|
|
||||||
$this->line("\$upgrader> chown -R {$user}:{$group} *");
|
|
||||||
$process = Process::fromShellCommandline("chown -R {$user}:{$group} *", $this->getLaravel()->basePath());
|
|
||||||
$process->setTimeout(10 * 60);
|
|
||||||
$process->run(function ($type, $buffer) {
|
|
||||||
$this->{$type === Process::ERR ? 'error' : 'line'}($buffer);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
$this->withProgress($bar, function () {
|
|
||||||
$this->line('$upgrader> php artisan queue:restart');
|
|
||||||
$this->call('queue:restart');
|
|
||||||
});
|
|
||||||
|
|
||||||
$this->withProgress($bar, function () {
|
|
||||||
$this->line('$upgrader> php artisan up');
|
|
||||||
$this->call('up');
|
|
||||||
});
|
|
||||||
|
|
||||||
$this->newLine(2);
|
|
||||||
$this->info(trans('commands.upgrade.success'));
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function withProgress(ProgressBar $bar, \Closure $callback): void
|
|
||||||
{
|
|
||||||
$bar->clear();
|
|
||||||
$callback();
|
|
||||||
$bar->advance();
|
|
||||||
$bar->display();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function getUrl(): string
|
|
||||||
{
|
|
||||||
if ($this->option('url')) {
|
|
||||||
return $this->option('url');
|
|
||||||
}
|
|
||||||
|
|
||||||
return sprintf(self::DEFAULT_URL, $this->option('release') ? 'download/v' . $this->option('release') : 'latest/download');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -3,8 +3,8 @@
|
|||||||
namespace App\Console\Commands\User;
|
namespace App\Console\Commands\User;
|
||||||
|
|
||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
use Webmozart\Assert\Assert;
|
|
||||||
use Illuminate\Console\Command;
|
use Illuminate\Console\Command;
|
||||||
|
use Webmozart\Assert\Assert;
|
||||||
|
|
||||||
class DeleteUserCommand extends Command
|
class DeleteUserCommand extends Command
|
||||||
{
|
{
|
||||||
@ -35,7 +35,7 @@ class DeleteUserCommand extends Command
|
|||||||
if ($this->input->isInteractive()) {
|
if ($this->input->isInteractive()) {
|
||||||
$tableValues = [];
|
$tableValues = [];
|
||||||
foreach ($results as $user) {
|
foreach ($results as $user) {
|
||||||
$tableValues[] = [$user->id, $user->email, $user->name];
|
$tableValues[] = [$user->id, $user->email, $user->username];
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->table(['User ID', 'Email', 'Name'], $tableValues);
|
$this->table(['User ID', 'Email', 'Name'], $tableValues);
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Console\Commands\User;
|
namespace App\Console\Commands\User;
|
||||||
|
|
||||||
|
use App\Exceptions\Model\DataValidationException;
|
||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
use Illuminate\Console\Command;
|
use Illuminate\Console\Command;
|
||||||
|
|
||||||
@ -14,20 +15,22 @@ class DisableTwoFactorCommand extends Command
|
|||||||
/**
|
/**
|
||||||
* Handle command execution process.
|
* Handle command execution process.
|
||||||
*
|
*
|
||||||
* @throws \App\Exceptions\Model\DataValidationException
|
* @throws DataValidationException
|
||||||
*/
|
*/
|
||||||
public function handle(): void
|
public function handle(): void
|
||||||
{
|
{
|
||||||
if ($this->input->isInteractive()) {
|
if ($this->input->isInteractive()) {
|
||||||
$this->output->warning(trans('command/messages.user.2fa_help_text.0') . trans('command/messages.user.2fa_help_text.1'));
|
$this->output->warning(trans('command/messages.user.2fa_help_text'));
|
||||||
}
|
}
|
||||||
|
|
||||||
$email = $this->option('email') ?? $this->ask(trans('command/messages.user.ask_email'));
|
$email = $this->option('email') ?? $this->ask(trans('command/messages.user.ask_email'));
|
||||||
|
|
||||||
$user = User::query()->where('email', $email)->firstOrFail();
|
$user = User::where('email', $email)->firstOrFail();
|
||||||
$user->use_totp = false;
|
$user->update([
|
||||||
$user->totp_secret = null;
|
'mfa_app_secret' => null,
|
||||||
$user->save();
|
'mfa_app_recovery_codes' => null,
|
||||||
|
'mfa_email_enabled' => false,
|
||||||
|
]);
|
||||||
|
|
||||||
$this->info(trans('command/messages.user.2fa_disabled', ['email' => $user->email]));
|
$this->info(trans('command/messages.user.2fa_disabled', ['email' => $user->email]));
|
||||||
}
|
}
|
||||||
|
|||||||
@ -2,9 +2,10 @@
|
|||||||
|
|
||||||
namespace App\Console\Commands\User;
|
namespace App\Console\Commands\User;
|
||||||
|
|
||||||
|
use App\Exceptions\Model\DataValidationException;
|
||||||
|
use App\Services\Users\UserCreationService;
|
||||||
use Exception;
|
use Exception;
|
||||||
use Illuminate\Console\Command;
|
use Illuminate\Console\Command;
|
||||||
use App\Services\Users\UserCreationService;
|
|
||||||
use Illuminate\Support\Facades\DB;
|
use Illuminate\Support\Facades\DB;
|
||||||
|
|
||||||
class MakeUserCommand extends Command
|
class MakeUserCommand extends Command
|
||||||
@ -25,7 +26,7 @@ class MakeUserCommand extends Command
|
|||||||
* Handle command request to create a new user.
|
* Handle command request to create a new user.
|
||||||
*
|
*
|
||||||
* @throws Exception
|
* @throws Exception
|
||||||
* @throws \App\Exceptions\Model\DataValidationException
|
* @throws DataValidationException
|
||||||
*/
|
*/
|
||||||
public function handle(): int
|
public function handle(): int
|
||||||
{
|
{
|
||||||
|
|||||||
@ -3,11 +3,11 @@
|
|||||||
namespace App\Console;
|
namespace App\Console;
|
||||||
|
|
||||||
use App\Console\Commands\Egg\CheckEggUpdatesCommand;
|
use App\Console\Commands\Egg\CheckEggUpdatesCommand;
|
||||||
|
use App\Console\Commands\Egg\UpdateEggIndexCommand;
|
||||||
use App\Console\Commands\Maintenance\CleanServiceBackupFilesCommand;
|
use App\Console\Commands\Maintenance\CleanServiceBackupFilesCommand;
|
||||||
use App\Console\Commands\Maintenance\PruneImagesCommand;
|
use App\Console\Commands\Maintenance\PruneImagesCommand;
|
||||||
use App\Console\Commands\Maintenance\PruneOrphanedBackupsCommand;
|
use App\Console\Commands\Maintenance\PruneOrphanedBackupsCommand;
|
||||||
use App\Console\Commands\Schedule\ProcessRunnableCommand;
|
use App\Console\Commands\Schedule\ProcessRunnableCommand;
|
||||||
use App\Jobs\NodeStatistics;
|
|
||||||
use App\Models\ActivityLog;
|
use App\Models\ActivityLog;
|
||||||
use App\Models\Webhook;
|
use App\Models\Webhook;
|
||||||
use Illuminate\Console\Scheduling\Schedule;
|
use Illuminate\Console\Scheduling\Schedule;
|
||||||
@ -31,17 +31,20 @@ class Kernel extends ConsoleKernel
|
|||||||
*/
|
*/
|
||||||
protected function schedule(Schedule $schedule): void
|
protected function schedule(Schedule $schedule): void
|
||||||
{
|
{
|
||||||
|
if (config('cache.default') === 'redis') {
|
||||||
// https://laravel.com/docs/10.x/upgrade#redis-cache-tags
|
// https://laravel.com/docs/10.x/upgrade#redis-cache-tags
|
||||||
|
// This only needs to run when using redis. anything else throws an error.
|
||||||
$schedule->command('cache:prune-stale-tags')->hourly();
|
$schedule->command('cache:prune-stale-tags')->hourly();
|
||||||
|
}
|
||||||
|
|
||||||
// Execute scheduled commands for servers every minute, as if there was a normal cron running.
|
// Execute scheduled commands for servers every minute, as if there was a normal cron running.
|
||||||
$schedule->command(ProcessRunnableCommand::class)->everyMinute()->withoutOverlapping();
|
$schedule->command(ProcessRunnableCommand::class)->everyMinute()->withoutOverlapping();
|
||||||
|
|
||||||
$schedule->command(CleanServiceBackupFilesCommand::class)->daily();
|
$schedule->command(CleanServiceBackupFilesCommand::class)->daily();
|
||||||
$schedule->command(PruneImagesCommand::class)->daily();
|
$schedule->command(PruneImagesCommand::class)->daily();
|
||||||
$schedule->command(CheckEggUpdatesCommand::class)->hourly();
|
|
||||||
|
|
||||||
$schedule->job(new NodeStatistics())->everyFiveSeconds()->withoutOverlapping();
|
$schedule->command(CheckEggUpdatesCommand::class)->daily();
|
||||||
|
$schedule->command(UpdateEggIndexCommand::class)->daily();
|
||||||
|
|
||||||
if (config('backups.prune_age')) {
|
if (config('backups.prune_age')) {
|
||||||
// Every 30 minutes, run the backup pruning command so that any abandoned backups can be deleted.
|
// Every 30 minutes, run the backup pruning command so that any abandoned backups can be deleted.
|
||||||
|
|||||||
37
app/Enums/BackupStatus.php
Normal file
37
app/Enums/BackupStatus.php
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Enums;
|
||||||
|
|
||||||
|
use Filament\Support\Contracts\HasColor;
|
||||||
|
use Filament\Support\Contracts\HasIcon;
|
||||||
|
use Filament\Support\Contracts\HasLabel;
|
||||||
|
|
||||||
|
enum BackupStatus: string implements HasColor, HasIcon, HasLabel
|
||||||
|
{
|
||||||
|
case InProgress = 'in_progress';
|
||||||
|
case Successful = 'successful';
|
||||||
|
case Failed = 'failed';
|
||||||
|
|
||||||
|
public function getIcon(): string
|
||||||
|
{
|
||||||
|
return match ($this) {
|
||||||
|
self::InProgress => 'tabler-circle-dashed',
|
||||||
|
self::Successful => 'tabler-circle-check',
|
||||||
|
self::Failed => 'tabler-circle-x',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getColor(): string
|
||||||
|
{
|
||||||
|
return match ($this) {
|
||||||
|
self::InProgress => 'primary',
|
||||||
|
self::Successful => 'success',
|
||||||
|
self::Failed => 'danger',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getLabel(): string
|
||||||
|
{
|
||||||
|
return trans('server/backup.backup_status.' . $this->value);
|
||||||
|
}
|
||||||
|
}
|
||||||
11
app/Enums/ConsoleWidgetPosition.php
Normal file
11
app/Enums/ConsoleWidgetPosition.php
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Enums;
|
||||||
|
|
||||||
|
enum ConsoleWidgetPosition: string
|
||||||
|
{
|
||||||
|
case Top = 'top';
|
||||||
|
case AboveConsole = 'above_console';
|
||||||
|
case BelowConsole = 'below_console';
|
||||||
|
case Bottom = 'bottom';
|
||||||
|
}
|
||||||
@ -62,13 +62,13 @@ enum ContainerStatus: string implements HasColor, HasIcon, HasLabel
|
|||||||
self::Removing => 'warning',
|
self::Removing => 'warning',
|
||||||
self::Missing => 'danger',
|
self::Missing => 'danger',
|
||||||
self::Stopping => 'warning',
|
self::Stopping => 'warning',
|
||||||
self::Offline => 'gray',
|
self::Offline => 'danger',
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getLabel(): string
|
public function getLabel(): string
|
||||||
{
|
{
|
||||||
return str($this->value)->title();
|
return trans('server/console.status.' . $this->value);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function isOffline(): bool
|
public function isOffline(): bool
|
||||||
@ -88,7 +88,7 @@ enum ContainerStatus: string implements HasColor, HasIcon, HasLabel
|
|||||||
|
|
||||||
public function isStartable(): bool
|
public function isStartable(): bool
|
||||||
{
|
{
|
||||||
return !in_array($this, [ContainerStatus::Running, ContainerStatus::Starting, ContainerStatus::Stopping, ContainerStatus::Restarting]);
|
return !in_array($this, [ContainerStatus::Running, ContainerStatus::Starting, ContainerStatus::Stopping, ContainerStatus::Restarting, ContainerStatus::Missing]);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function isRestartable(): bool
|
public function isRestartable(): bool
|
||||||
@ -97,18 +97,16 @@ enum ContainerStatus: string implements HasColor, HasIcon, HasLabel
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
return !in_array($this, [ContainerStatus::Offline]);
|
return !in_array($this, [ContainerStatus::Offline, ContainerStatus::Missing]);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function isStoppable(): bool
|
public function isStoppable(): bool
|
||||||
{
|
{
|
||||||
return !in_array($this, [ContainerStatus::Starting, ContainerStatus::Stopping, ContainerStatus::Restarting, ContainerStatus::Exited, ContainerStatus::Offline]);
|
return !in_array($this, [ContainerStatus::Starting, ContainerStatus::Stopping, ContainerStatus::Restarting, ContainerStatus::Exited, ContainerStatus::Offline, ContainerStatus::Missing]);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function isKillable(): bool
|
public function isKillable(): bool
|
||||||
{
|
{
|
||||||
// [ContainerStatus::Restarting, ContainerStatus::Removing, ContainerStatus::Dead, ContainerStatus::Created]
|
return !in_array($this, [ContainerStatus::Offline, ContainerStatus::Running, ContainerStatus::Exited, ContainerStatus::Missing]);
|
||||||
|
|
||||||
return !in_array($this, [ContainerStatus::Offline, ContainerStatus::Running, ContainerStatus::Exited]);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
9
app/Enums/CustomRenderHooks.php
Normal file
9
app/Enums/CustomRenderHooks.php
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Enums;
|
||||||
|
|
||||||
|
enum CustomRenderHooks: string
|
||||||
|
{
|
||||||
|
case FooterStart = 'pelican::footer.start';
|
||||||
|
case FooterEnd = 'pelican::footer.end';
|
||||||
|
}
|
||||||
37
app/Enums/CustomizationKey.php
Normal file
37
app/Enums/CustomizationKey.php
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Enums;
|
||||||
|
|
||||||
|
enum CustomizationKey: string
|
||||||
|
{
|
||||||
|
case ConsoleRows = 'console_rows';
|
||||||
|
case ConsoleFont = 'console_font';
|
||||||
|
case ConsoleFontSize = 'console_font_size';
|
||||||
|
case ConsoleGraphPeriod = 'console_graph_period';
|
||||||
|
case TopNavigation = 'top_navigation';
|
||||||
|
case DashboardLayout = 'dashboard_layout';
|
||||||
|
|
||||||
|
public function getDefaultValue(): string|int|bool
|
||||||
|
{
|
||||||
|
return match ($this) {
|
||||||
|
self::ConsoleRows => 30,
|
||||||
|
self::ConsoleFont => 'monospace',
|
||||||
|
self::ConsoleFontSize => 14,
|
||||||
|
self::ConsoleGraphPeriod => 30,
|
||||||
|
self::TopNavigation => config('panel.filament.default-navigation', 'sidebar'),
|
||||||
|
self::DashboardLayout => 'grid',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return array<string, string|int|bool> */
|
||||||
|
public static function getDefaultCustomization(): array
|
||||||
|
{
|
||||||
|
$default = [];
|
||||||
|
|
||||||
|
foreach (self::cases() as $key) {
|
||||||
|
$default[$key->value] = $key->getDefaultValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
return $default;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,141 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Enums;
|
|
||||||
|
|
||||||
use Filament\Support\Contracts\HasLabel;
|
|
||||||
|
|
||||||
enum EditorLanguages: string implements HasLabel
|
|
||||||
{
|
|
||||||
case plaintext = 'plaintext';
|
|
||||||
case abap = 'abap';
|
|
||||||
case apex = 'apex';
|
|
||||||
case azcali = 'azcali';
|
|
||||||
case bat = 'bat';
|
|
||||||
case bicep = 'bicep';
|
|
||||||
case cameligo = 'cameligo';
|
|
||||||
case coljure = 'coljure';
|
|
||||||
case coffeescript = 'coffeescript';
|
|
||||||
case c = 'c';
|
|
||||||
case cpp = 'cpp';
|
|
||||||
case csharp = 'csharp';
|
|
||||||
case csp = 'csp';
|
|
||||||
case css = 'css';
|
|
||||||
case cypher = 'cypher';
|
|
||||||
case dart = 'dart';
|
|
||||||
case dockerfile = 'dockerfile';
|
|
||||||
case ecl = 'ecl';
|
|
||||||
case elixir = 'elixir';
|
|
||||||
case flow9 = 'flow9';
|
|
||||||
case fsharp = 'fsharp';
|
|
||||||
case go = 'go';
|
|
||||||
case graphql = 'graphql';
|
|
||||||
case handlebars = 'handlebars';
|
|
||||||
case hcl = 'hcl';
|
|
||||||
case html = 'html';
|
|
||||||
case ini = 'ini';
|
|
||||||
case java = 'java';
|
|
||||||
case javascript = 'javascript';
|
|
||||||
case julia = 'julia';
|
|
||||||
case json = 'json';
|
|
||||||
case kotlin = 'kotlin';
|
|
||||||
case less = 'less';
|
|
||||||
case lexon = 'lexon';
|
|
||||||
case lua = 'lua';
|
|
||||||
case liquid = 'liquid';
|
|
||||||
case m3 = 'm3';
|
|
||||||
case markdown = 'markdown';
|
|
||||||
case mdx = 'mdx';
|
|
||||||
case mips = 'mips';
|
|
||||||
case msdax = 'msdax';
|
|
||||||
case mysql = 'mysql';
|
|
||||||
case objectivec = 'objective-c';
|
|
||||||
case pascal = 'pascal';
|
|
||||||
case pascaligo = 'pascaligo';
|
|
||||||
case perl = 'perl';
|
|
||||||
case pgsql = 'pgsql';
|
|
||||||
case php = 'php';
|
|
||||||
case pla = 'pla';
|
|
||||||
case postiats = 'postiats';
|
|
||||||
case powerquery = 'powerquery';
|
|
||||||
case powershell = 'powershell';
|
|
||||||
case proto = 'proto';
|
|
||||||
case pug = 'pug';
|
|
||||||
case python = 'python';
|
|
||||||
case qsharp = 'qsharp';
|
|
||||||
case r = 'r';
|
|
||||||
case razor = 'razor';
|
|
||||||
case redis = 'redis';
|
|
||||||
case redshift = 'redshift';
|
|
||||||
case restructuredtext = 'restructuredtext';
|
|
||||||
case ruby = 'ruby';
|
|
||||||
case rust = 'rust';
|
|
||||||
case sb = 'sb';
|
|
||||||
case scala = 'scala';
|
|
||||||
case scheme = 'scheme';
|
|
||||||
case scss = 'scss';
|
|
||||||
case shell = 'shell';
|
|
||||||
case sol = 'sol';
|
|
||||||
case aes = 'aes';
|
|
||||||
case sparql = 'sparql';
|
|
||||||
case sql = 'sql';
|
|
||||||
case st = 'st';
|
|
||||||
case swift = 'swift';
|
|
||||||
case systemverilog = 'systemverilog';
|
|
||||||
case verilog = 'verilog';
|
|
||||||
case tcl = 'tcl';
|
|
||||||
case twig = 'twig';
|
|
||||||
case typescript = 'typescript';
|
|
||||||
case typespec = 'typespec';
|
|
||||||
case vb = 'vb';
|
|
||||||
case wgsl = 'wgsl';
|
|
||||||
case xml = 'xml';
|
|
||||||
case yaml = 'yaml';
|
|
||||||
|
|
||||||
public static function fromWithAlias(string $match): self
|
|
||||||
{
|
|
||||||
return match ($match) {
|
|
||||||
'h' => self::c,
|
|
||||||
|
|
||||||
'cc', 'hpp' => self::cpp,
|
|
||||||
|
|
||||||
'cs' => self::csharp,
|
|
||||||
|
|
||||||
'class' => self::java,
|
|
||||||
|
|
||||||
'htm' => self::html,
|
|
||||||
|
|
||||||
'js', 'mjs', 'cjs' => self::javascript,
|
|
||||||
|
|
||||||
'kt', 'kts' => self::kotlin,
|
|
||||||
|
|
||||||
'md' => self::markdown,
|
|
||||||
|
|
||||||
'm' => self::objectivec,
|
|
||||||
|
|
||||||
'pl', 'pm' => self::perl,
|
|
||||||
|
|
||||||
'php3', 'php4', 'php5', 'phtml' => self::php,
|
|
||||||
|
|
||||||
'py', 'pyc', 'pyo', 'pyi' => self::python,
|
|
||||||
|
|
||||||
'rdata', 'rds' => self::r,
|
|
||||||
|
|
||||||
'rb', 'erb' => self::ruby,
|
|
||||||
|
|
||||||
'sc' => self::scala,
|
|
||||||
|
|
||||||
'sh', 'zsh' => self::shell,
|
|
||||||
|
|
||||||
'ts', 'tsx' => self::typescript,
|
|
||||||
|
|
||||||
'yml' => self::yaml,
|
|
||||||
|
|
||||||
default => self::tryFrom($match) ?? self::plaintext,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getLabel(): string
|
|
||||||
{
|
|
||||||
return $this->name;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
9
app/Enums/EggFormat.php
Normal file
9
app/Enums/EggFormat.php
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Enums;
|
||||||
|
|
||||||
|
enum EggFormat: string
|
||||||
|
{
|
||||||
|
case YAML = 'yaml';
|
||||||
|
case JSON = 'json';
|
||||||
|
}
|
||||||
9
app/Enums/HeaderActionPosition.php
Normal file
9
app/Enums/HeaderActionPosition.php
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Enums;
|
||||||
|
|
||||||
|
enum HeaderActionPosition: string
|
||||||
|
{
|
||||||
|
case Before = 'before';
|
||||||
|
case After = 'after';
|
||||||
|
}
|
||||||
9
app/Enums/HeaderWidgetPosition.php
Normal file
9
app/Enums/HeaderWidgetPosition.php
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Enums;
|
||||||
|
|
||||||
|
enum HeaderWidgetPosition: string
|
||||||
|
{
|
||||||
|
case Before = 'before';
|
||||||
|
case After = 'after';
|
||||||
|
}
|
||||||
@ -14,4 +14,24 @@ enum RolePermissionModels: string
|
|||||||
case Server = 'server';
|
case Server = 'server';
|
||||||
case User = 'user';
|
case User = 'user';
|
||||||
case Webhook = 'webhook';
|
case Webhook = 'webhook';
|
||||||
|
|
||||||
|
public function viewAny(): string
|
||||||
|
{
|
||||||
|
return RolePermissionPrefixes::ViewAny->value . ' ' . $this->value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function view(): string
|
||||||
|
{
|
||||||
|
return RolePermissionPrefixes::View->value . ' ' . $this->value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function create(): string
|
||||||
|
{
|
||||||
|
return RolePermissionPrefixes::Create->value . ' ' . $this->value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function update(): string
|
||||||
|
{
|
||||||
|
return RolePermissionPrefixes::Update->value . ' ' . $this->value;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
27
app/Enums/ScheduleStatus.php
Normal file
27
app/Enums/ScheduleStatus.php
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Enums;
|
||||||
|
|
||||||
|
use Filament\Support\Contracts\HasColor;
|
||||||
|
use Filament\Support\Contracts\HasLabel;
|
||||||
|
|
||||||
|
enum ScheduleStatus: string implements HasColor, HasLabel
|
||||||
|
{
|
||||||
|
case Inactive = 'inactive';
|
||||||
|
case Processing = 'processing';
|
||||||
|
case Active = 'active';
|
||||||
|
|
||||||
|
public function getColor(): string
|
||||||
|
{
|
||||||
|
return match ($this) {
|
||||||
|
self::Inactive => 'danger',
|
||||||
|
self::Processing => 'warning',
|
||||||
|
self::Active => 'success',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getLabel(): string
|
||||||
|
{
|
||||||
|
return trans('server/schedule.schedule_status.' . $this->value);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -2,9 +2,50 @@
|
|||||||
|
|
||||||
namespace App\Enums;
|
namespace App\Enums;
|
||||||
|
|
||||||
enum ServerResourceType
|
use App\Models\Server;
|
||||||
|
|
||||||
|
enum ServerResourceType: string
|
||||||
{
|
{
|
||||||
case Unit;
|
case Uptime = 'uptime';
|
||||||
case Percentage;
|
case CPU = 'cpu_absolute';
|
||||||
case Time;
|
case Memory = 'memory_bytes';
|
||||||
|
case Disk = 'disk_bytes';
|
||||||
|
|
||||||
|
case CPULimit = 'cpu';
|
||||||
|
case MemoryLimit = 'memory';
|
||||||
|
case DiskLimit = 'disk';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return int resource amount in bytes
|
||||||
|
*/
|
||||||
|
public function getResourceAmount(Server $server): int
|
||||||
|
{
|
||||||
|
if ($this->isLimit()) {
|
||||||
|
$resourceAmount = $server->{$this->value} ?? 0;
|
||||||
|
|
||||||
|
if (!$this->isPercentage()) {
|
||||||
|
// Our limits are entered as MiB/ MB so we need to convert them to bytes
|
||||||
|
$resourceAmount *= config('panel.use_binary_prefix') ? 1024 * 1024 : 1000 * 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $resourceAmount;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $server->retrieveResources()[$this->value] ?? 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isLimit(): bool
|
||||||
|
{
|
||||||
|
return $this === ServerResourceType::CPULimit || $this === ServerResourceType::MemoryLimit || $this === ServerResourceType::DiskLimit;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isTime(): bool
|
||||||
|
{
|
||||||
|
return $this === ServerResourceType::Uptime;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isPercentage(): bool
|
||||||
|
{
|
||||||
|
return $this === ServerResourceType::CPU || $this === ServerResourceType::CPULimit;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -8,7 +8,6 @@ use Filament\Support\Contracts\HasLabel;
|
|||||||
|
|
||||||
enum ServerState: string implements HasColor, HasIcon, HasLabel
|
enum ServerState: string implements HasColor, HasIcon, HasLabel
|
||||||
{
|
{
|
||||||
case Normal = 'normal';
|
|
||||||
case Installing = 'installing';
|
case Installing = 'installing';
|
||||||
case InstallFailed = 'install_failed';
|
case InstallFailed = 'install_failed';
|
||||||
case ReinstallFailed = 'reinstall_failed';
|
case ReinstallFailed = 'reinstall_failed';
|
||||||
@ -18,7 +17,6 @@ enum ServerState: string implements HasColor, HasIcon, HasLabel
|
|||||||
public function getIcon(): string
|
public function getIcon(): string
|
||||||
{
|
{
|
||||||
return match ($this) {
|
return match ($this) {
|
||||||
self::Normal => 'tabler-heart',
|
|
||||||
self::Installing => 'tabler-heart-bolt',
|
self::Installing => 'tabler-heart-bolt',
|
||||||
self::InstallFailed => 'tabler-heart-x',
|
self::InstallFailed => 'tabler-heart-x',
|
||||||
self::ReinstallFailed => 'tabler-heart-x',
|
self::ReinstallFailed => 'tabler-heart-x',
|
||||||
@ -27,10 +25,17 @@ enum ServerState: string implements HasColor, HasIcon, HasLabel
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getColor(): string
|
public function getColor(bool $hex = false): string
|
||||||
{
|
{
|
||||||
|
if ($hex) {
|
||||||
|
return match ($this) {
|
||||||
|
self::Installing, self::RestoringBackup => '#2563EB',
|
||||||
|
self::Suspended => '#D97706',
|
||||||
|
self::InstallFailed, self::ReinstallFailed => '#EF4444',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
return match ($this) {
|
return match ($this) {
|
||||||
self::Normal => 'primary',
|
|
||||||
self::Installing => 'primary',
|
self::Installing => 'primary',
|
||||||
self::InstallFailed => 'danger',
|
self::InstallFailed => 'danger',
|
||||||
self::ReinstallFailed => 'danger',
|
self::ReinstallFailed => 'danger',
|
||||||
|
|||||||
11
app/Enums/StartupVariableType.php
Normal file
11
app/Enums/StartupVariableType.php
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Enums;
|
||||||
|
|
||||||
|
enum StartupVariableType: string
|
||||||
|
{
|
||||||
|
case Text = 'text';
|
||||||
|
case Number = 'number';
|
||||||
|
case Select = 'select';
|
||||||
|
case Toggle = 'toggle';
|
||||||
|
}
|
||||||
34
app/Enums/WebhookType.php
Normal file
34
app/Enums/WebhookType.php
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Enums;
|
||||||
|
|
||||||
|
use Filament\Support\Contracts\HasColor;
|
||||||
|
use Filament\Support\Contracts\HasIcon;
|
||||||
|
use Filament\Support\Contracts\HasLabel;
|
||||||
|
|
||||||
|
enum WebhookType: string implements HasColor, HasIcon, HasLabel
|
||||||
|
{
|
||||||
|
case Regular = 'regular';
|
||||||
|
case Discord = 'discord';
|
||||||
|
|
||||||
|
public function getLabel(): string
|
||||||
|
{
|
||||||
|
return trans('admin/webhook.' . $this->value);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getColor(): ?string
|
||||||
|
{
|
||||||
|
return match ($this) {
|
||||||
|
self::Regular => null,
|
||||||
|
self::Discord => 'blurple',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getIcon(): string
|
||||||
|
{
|
||||||
|
return match ($this) {
|
||||||
|
self::Regular => 'tabler-world-www',
|
||||||
|
self::Discord => 'tabler-brand-discord',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -2,9 +2,9 @@
|
|||||||
|
|
||||||
namespace App\Events;
|
namespace App\Events;
|
||||||
|
|
||||||
use Illuminate\Support\Str;
|
|
||||||
use App\Models\ActivityLog;
|
use App\Models\ActivityLog;
|
||||||
use Illuminate\Database\Eloquent\Model;
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
use Illuminate\Support\Str;
|
||||||
|
|
||||||
class ActivityLogged extends Event
|
class ActivityLogged extends Event
|
||||||
{
|
{
|
||||||
|
|||||||
@ -1,11 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Events\Auth;
|
|
||||||
|
|
||||||
use App\Models\User;
|
|
||||||
use App\Events\Event;
|
|
||||||
|
|
||||||
class DirectLogin extends Event
|
|
||||||
{
|
|
||||||
public function __construct(public User $user, public bool $remember) {}
|
|
||||||
}
|
|
||||||
@ -1,16 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Events\Auth;
|
|
||||||
|
|
||||||
use App\Events\Event;
|
|
||||||
use Illuminate\Queue\SerializesModels;
|
|
||||||
|
|
||||||
class FailedPasswordReset extends Event
|
|
||||||
{
|
|
||||||
use SerializesModels;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new event instance.
|
|
||||||
*/
|
|
||||||
public function __construct(public string $ip, public string $email) {}
|
|
||||||
}
|
|
||||||
@ -2,8 +2,8 @@
|
|||||||
|
|
||||||
namespace App\Events\Auth;
|
namespace App\Events\Auth;
|
||||||
|
|
||||||
use App\Models\User;
|
|
||||||
use App\Events\Event;
|
use App\Events\Event;
|
||||||
|
use App\Models\User;
|
||||||
|
|
||||||
class ProvidedAuthenticationToken extends Event
|
class ProvidedAuthenticationToken extends Event
|
||||||
{
|
{
|
||||||
|
|||||||
@ -4,13 +4,14 @@ namespace App\Exceptions;
|
|||||||
|
|
||||||
use Exception;
|
use Exception;
|
||||||
use Filament\Notifications\Notification;
|
use Filament\Notifications\Notification;
|
||||||
|
use Illuminate\Container\Container;
|
||||||
use Illuminate\Http\JsonResponse;
|
use Illuminate\Http\JsonResponse;
|
||||||
use Illuminate\Http\RedirectResponse;
|
use Illuminate\Http\RedirectResponse;
|
||||||
use Illuminate\Http\Request;
|
use Illuminate\Http\Request;
|
||||||
use Psr\Log\LoggerInterface;
|
|
||||||
use Illuminate\Http\Response;
|
use Illuminate\Http\Response;
|
||||||
use Illuminate\Container\Container;
|
use Psr\Log\LoggerInterface;
|
||||||
use Symfony\Component\HttpKernel\Exception\HttpExceptionInterface;
|
use Symfony\Component\HttpKernel\Exception\HttpExceptionInterface;
|
||||||
|
use Throwable;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @deprecated
|
* @deprecated
|
||||||
@ -28,7 +29,7 @@ class DisplayException extends PanelException implements HttpExceptionInterface
|
|||||||
/**
|
/**
|
||||||
* DisplayException constructor.
|
* DisplayException constructor.
|
||||||
*/
|
*/
|
||||||
public function __construct(string $message, ?\Throwable $previous = null, protected string $level = self::LEVEL_ERROR, int $code = 0)
|
public function __construct(string $message, ?Throwable $previous = null, protected string $level = self::LEVEL_ERROR, int $code = 0)
|
||||||
{
|
{
|
||||||
parent::__construct($message, $code, $previous);
|
parent::__construct($message, $code, $previous);
|
||||||
}
|
}
|
||||||
@ -79,11 +80,11 @@ class DisplayException extends PanelException implements HttpExceptionInterface
|
|||||||
* Log the exception to the logs using the defined error level only if the previous
|
* Log the exception to the logs using the defined error level only if the previous
|
||||||
* exception is set.
|
* exception is set.
|
||||||
*
|
*
|
||||||
* @throws \Throwable
|
* @throws Throwable
|
||||||
*/
|
*/
|
||||||
public function report(): void
|
public function report(): void
|
||||||
{
|
{
|
||||||
if (!$this->getPrevious() instanceof \Exception || !Handler::isReportable($this->getPrevious())) {
|
if (!$this->getPrevious() instanceof Exception || !Handler::isReportable($this->getPrevious())) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -2,24 +2,27 @@
|
|||||||
|
|
||||||
namespace App\Exceptions;
|
namespace App\Exceptions;
|
||||||
|
|
||||||
use Illuminate\Support\Arr;
|
use Exception;
|
||||||
use Illuminate\Support\Str;
|
use Illuminate\Auth\Access\AuthorizationException;
|
||||||
use Illuminate\Http\JsonResponse;
|
use Illuminate\Auth\AuthenticationException;
|
||||||
use Illuminate\Support\Collection;
|
|
||||||
use Illuminate\Container\Container;
|
use Illuminate\Container\Container;
|
||||||
use Illuminate\Database\Connection;
|
use Illuminate\Database\Connection;
|
||||||
use Illuminate\Http\RedirectResponse;
|
|
||||||
use Illuminate\Foundation\Application;
|
|
||||||
use Illuminate\Auth\AuthenticationException;
|
|
||||||
use Illuminate\Session\TokenMismatchException;
|
|
||||||
use Illuminate\Validation\ValidationException;
|
|
||||||
use Symfony\Component\HttpFoundation\Response;
|
|
||||||
use Illuminate\Auth\Access\AuthorizationException;
|
|
||||||
use Illuminate\Database\Eloquent\ModelNotFoundException;
|
use Illuminate\Database\Eloquent\ModelNotFoundException;
|
||||||
use Symfony\Component\HttpKernel\Exception\HttpException;
|
use Illuminate\Foundation\Application;
|
||||||
use Symfony\Component\Mailer\Exception\TransportException;
|
|
||||||
use Illuminate\Foundation\Exceptions\Handler as ExceptionHandler;
|
use Illuminate\Foundation\Exceptions\Handler as ExceptionHandler;
|
||||||
|
use Illuminate\Http\JsonResponse;
|
||||||
|
use Illuminate\Http\RedirectResponse;
|
||||||
|
use Illuminate\Http\Request;
|
||||||
|
use Illuminate\Session\TokenMismatchException;
|
||||||
|
use Illuminate\Support\Arr;
|
||||||
|
use Illuminate\Support\Collection;
|
||||||
|
use Illuminate\Support\Str;
|
||||||
|
use Illuminate\Validation\ValidationException;
|
||||||
|
use PDOException;
|
||||||
|
use Symfony\Component\HttpFoundation\Response;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\HttpException;
|
||||||
use Symfony\Component\HttpKernel\Exception\HttpExceptionInterface;
|
use Symfony\Component\HttpKernel\Exception\HttpExceptionInterface;
|
||||||
|
use Symfony\Component\Mailer\Exception\TransportException;
|
||||||
use Throwable;
|
use Throwable;
|
||||||
|
|
||||||
class Handler extends ExceptionHandler
|
class Handler extends ExceptionHandler
|
||||||
@ -79,7 +82,7 @@ class Handler extends ExceptionHandler
|
|||||||
$this->dontReport = [];
|
$this->dontReport = [];
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->reportable(function (\PDOException $ex) {
|
$this->reportable(function (PDOException $ex) {
|
||||||
$ex = $this->generateCleanedExceptionStack($ex);
|
$ex = $this->generateCleanedExceptionStack($ex);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -88,7 +91,7 @@ class Handler extends ExceptionHandler
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private function generateCleanedExceptionStack(\Throwable $exception): string
|
private function generateCleanedExceptionStack(Throwable $exception): string
|
||||||
{
|
{
|
||||||
$cleanedStack = '';
|
$cleanedStack = '';
|
||||||
foreach ($exception->getTrace() as $index => $item) {
|
foreach ($exception->getTrace() as $index => $item) {
|
||||||
@ -117,11 +120,11 @@ 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
|
||||||
*
|
*
|
||||||
* @throws \Throwable
|
* @throws Throwable
|
||||||
*/
|
*/
|
||||||
public function render($request, \Throwable $e): Response
|
public function render($request, Throwable $e): Response
|
||||||
{
|
{
|
||||||
$connections = $this->container->make(Connection::class);
|
$connections = $this->container->make(Connection::class);
|
||||||
|
|
||||||
@ -143,7 +146,7 @@ class Handler extends ExceptionHandler
|
|||||||
* Transform a validation exception into a consistent format to be returned for
|
* Transform a validation exception into a consistent format to be returned for
|
||||||
* calls to the API.
|
* calls to the API.
|
||||||
*
|
*
|
||||||
* @param \Illuminate\Http\Request $request
|
* @param Request $request
|
||||||
*/
|
*/
|
||||||
public function invalidJson($request, ValidationException $exception): JsonResponse
|
public function invalidJson($request, ValidationException $exception): JsonResponse
|
||||||
{
|
{
|
||||||
@ -249,7 +252,7 @@ class Handler extends ExceptionHandler
|
|||||||
/**
|
/**
|
||||||
* Return an array of exceptions that should not be reported.
|
* Return an array of exceptions that should not be reported.
|
||||||
*/
|
*/
|
||||||
public static function isReportable(\Exception $exception): bool
|
public static function isReportable(Exception $exception): bool
|
||||||
{
|
{
|
||||||
return (new self(Container::getInstance()))->shouldReport($exception);
|
return (new self(Container::getInstance()))->shouldReport($exception);
|
||||||
}
|
}
|
||||||
@ -257,7 +260,7 @@ class Handler extends ExceptionHandler
|
|||||||
/**
|
/**
|
||||||
* Convert an authentication exception into an unauthenticated response.
|
* Convert an authentication exception into an unauthenticated response.
|
||||||
*
|
*
|
||||||
* @param \Illuminate\Http\Request $request
|
* @param Request $request
|
||||||
*/
|
*/
|
||||||
protected function unauthenticated($request, AuthenticationException $exception): JsonResponse|RedirectResponse
|
protected function unauthenticated($request, AuthenticationException $exception): JsonResponse|RedirectResponse
|
||||||
{
|
{
|
||||||
@ -291,7 +294,7 @@ class Handler extends ExceptionHandler
|
|||||||
*
|
*
|
||||||
* @return array<mixed>
|
* @return array<mixed>
|
||||||
*/
|
*/
|
||||||
public static function toArray(\Throwable $e): array
|
public static function toArray(Throwable $e): array
|
||||||
{
|
{
|
||||||
return self::exceptionToArray($e);
|
return self::exceptionToArray($e);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -4,13 +4,14 @@ namespace App\Exceptions\Http;
|
|||||||
|
|
||||||
use Illuminate\Http\Response;
|
use Illuminate\Http\Response;
|
||||||
use Symfony\Component\HttpKernel\Exception\HttpException;
|
use Symfony\Component\HttpKernel\Exception\HttpException;
|
||||||
|
use Throwable;
|
||||||
|
|
||||||
class HttpForbiddenException extends HttpException
|
class HttpForbiddenException extends HttpException
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* HttpForbiddenException constructor.
|
* HttpForbiddenException constructor.
|
||||||
*/
|
*/
|
||||||
public function __construct(?string $message = null, ?\Throwable $previous = null)
|
public function __construct(?string $message = null, ?Throwable $previous = null)
|
||||||
{
|
{
|
||||||
parent::__construct(Response::HTTP_FORBIDDEN, $message, $previous);
|
parent::__construct(Response::HTTP_FORBIDDEN, $message, $previous);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -5,6 +5,7 @@ namespace App\Exceptions\Http\Server;
|
|||||||
use App\Enums\ServerState;
|
use App\Enums\ServerState;
|
||||||
use App\Models\Server;
|
use App\Models\Server;
|
||||||
use Symfony\Component\HttpKernel\Exception\ConflictHttpException;
|
use Symfony\Component\HttpKernel\Exception\ConflictHttpException;
|
||||||
|
use Throwable;
|
||||||
|
|
||||||
class ServerStateConflictException extends ConflictHttpException
|
class ServerStateConflictException extends ConflictHttpException
|
||||||
{
|
{
|
||||||
@ -12,7 +13,7 @@ class ServerStateConflictException extends ConflictHttpException
|
|||||||
* Exception thrown when the server is in an unsupported state for API access or
|
* Exception thrown when the server is in an unsupported state for API access or
|
||||||
* certain operations within the codebase.
|
* certain operations within the codebase.
|
||||||
*/
|
*/
|
||||||
public function __construct(Server $server, ?\Throwable $previous = null)
|
public function __construct(Server $server, ?Throwable $previous = null)
|
||||||
{
|
{
|
||||||
$message = 'This server is currently in an unsupported state, please try again later.';
|
$message = 'This server is currently in an unsupported state, please try again later.';
|
||||||
if ($server->isSuspended()) {
|
if ($server->isSuspended()) {
|
||||||
|
|||||||
@ -2,13 +2,15 @@
|
|||||||
|
|
||||||
namespace App\Exceptions;
|
namespace App\Exceptions;
|
||||||
|
|
||||||
use Spatie\Ignition\Contracts\Solution;
|
use App\Exceptions\Solutions\ManifestDoesNotExistSolution;
|
||||||
|
use Exception;
|
||||||
use Spatie\Ignition\Contracts\ProvidesSolution;
|
use Spatie\Ignition\Contracts\ProvidesSolution;
|
||||||
|
use Spatie\Ignition\Contracts\Solution;
|
||||||
|
|
||||||
class ManifestDoesNotExistException extends \Exception implements ProvidesSolution
|
class ManifestDoesNotExistException extends Exception implements ProvidesSolution
|
||||||
{
|
{
|
||||||
public function getSolution(): Solution
|
public function getSolution(): Solution
|
||||||
{
|
{
|
||||||
return new Solutions\ManifestDoesNotExistSolution();
|
return new ManifestDoesNotExistSolution();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -2,11 +2,11 @@
|
|||||||
|
|
||||||
namespace App\Exceptions\Model;
|
namespace App\Exceptions\Model;
|
||||||
|
|
||||||
use Illuminate\Support\MessageBag;
|
|
||||||
use Illuminate\Database\Eloquent\Model;
|
|
||||||
use Illuminate\Contracts\Validation\Validator;
|
|
||||||
use App\Exceptions\PanelException;
|
use App\Exceptions\PanelException;
|
||||||
use Illuminate\Contracts\Support\MessageProvider;
|
use Illuminate\Contracts\Support\MessageProvider;
|
||||||
|
use Illuminate\Contracts\Validation\Validator;
|
||||||
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
use Illuminate\Support\MessageBag;
|
||||||
use Symfony\Component\HttpKernel\Exception\HttpExceptionInterface;
|
use Symfony\Component\HttpKernel\Exception\HttpExceptionInterface;
|
||||||
|
|
||||||
class DataValidationException extends PanelException implements HttpExceptionInterface, MessageProvider
|
class DataValidationException extends PanelException implements HttpExceptionInterface, MessageProvider
|
||||||
|
|||||||
@ -2,4 +2,6 @@
|
|||||||
|
|
||||||
namespace App\Exceptions;
|
namespace App\Exceptions;
|
||||||
|
|
||||||
class PanelException extends \Exception {}
|
use Exception;
|
||||||
|
|
||||||
|
class PanelException extends Exception {}
|
||||||
|
|||||||
7
app/Exceptions/Repository/FileExistsException.php
Normal file
7
app/Exceptions/Repository/FileExistsException.php
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Exceptions\Repository;
|
||||||
|
|
||||||
|
use Exception;
|
||||||
|
|
||||||
|
class FileExistsException extends Exception {}
|
||||||
7
app/Exceptions/Repository/FileNotEditableException.php
Normal file
7
app/Exceptions/Repository/FileNotEditableException.php
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Exceptions\Repository;
|
||||||
|
|
||||||
|
use Exception;
|
||||||
|
|
||||||
|
class FileNotEditableException extends Exception {}
|
||||||
@ -0,0 +1,7 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Exceptions\Service\Deployment;
|
||||||
|
|
||||||
|
use App\Exceptions\DisplayException;
|
||||||
|
|
||||||
|
class NoViableNodeException extends DisplayException {}
|
||||||
@ -2,8 +2,8 @@
|
|||||||
|
|
||||||
namespace App\Exceptions\Service;
|
namespace App\Exceptions\Service;
|
||||||
|
|
||||||
use Illuminate\Http\Response;
|
|
||||||
use App\Exceptions\DisplayException;
|
use App\Exceptions\DisplayException;
|
||||||
|
use Illuminate\Http\Response;
|
||||||
|
|
||||||
class HasActiveServersException extends DisplayException
|
class HasActiveServersException extends DisplayException
|
||||||
{
|
{
|
||||||
|
|||||||
@ -3,6 +3,7 @@
|
|||||||
namespace App\Exceptions\Service;
|
namespace App\Exceptions\Service;
|
||||||
|
|
||||||
use App\Exceptions\DisplayException;
|
use App\Exceptions\DisplayException;
|
||||||
|
use Throwable;
|
||||||
|
|
||||||
class ServiceLimitExceededException extends DisplayException
|
class ServiceLimitExceededException extends DisplayException
|
||||||
{
|
{
|
||||||
@ -10,7 +11,7 @@ class ServiceLimitExceededException extends DisplayException
|
|||||||
* Exception thrown when something goes over a defined limit, such as allocated
|
* Exception thrown when something goes over a defined limit, such as allocated
|
||||||
* ports, tasks, databases, etc.
|
* ports, tasks, databases, etc.
|
||||||
*/
|
*/
|
||||||
public function __construct(string $message, ?\Throwable $previous = null)
|
public function __construct(string $message, ?Throwable $previous = null)
|
||||||
{
|
{
|
||||||
parent::__construct($message, $previous, self::LEVEL_WARNING);
|
parent::__construct($message, $previous, self::LEVEL_WARNING);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,17 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Exceptions\Service\User;
|
|
||||||
|
|
||||||
use App\Exceptions\DisplayException;
|
|
||||||
|
|
||||||
class TwoFactorAuthenticationTokenInvalid extends DisplayException
|
|
||||||
{
|
|
||||||
public string $title = 'Invalid 2FA Code';
|
|
||||||
|
|
||||||
public string $icon = 'tabler-2fa';
|
|
||||||
|
|
||||||
public function __construct()
|
|
||||||
{
|
|
||||||
parent::__construct('The provided two-factor authentication token was not valid.');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
14
app/Extensions/Avatar/AvatarSchemaInterface.php
Normal file
14
app/Extensions/Avatar/AvatarSchemaInterface.php
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Avatar;
|
||||||
|
|
||||||
|
use App\Models\User;
|
||||||
|
|
||||||
|
interface AvatarSchemaInterface
|
||||||
|
{
|
||||||
|
public function getId(): string;
|
||||||
|
|
||||||
|
public function getName(): string;
|
||||||
|
|
||||||
|
public function get(User $user): ?string;
|
||||||
|
}
|
||||||
55
app/Extensions/Avatar/AvatarService.php
Normal file
55
app/Extensions/Avatar/AvatarService.php
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Avatar;
|
||||||
|
|
||||||
|
use App\Models\User;
|
||||||
|
use Illuminate\Support\Facades\Storage;
|
||||||
|
|
||||||
|
class AvatarService
|
||||||
|
{
|
||||||
|
/** @var AvatarSchemaInterface[] */
|
||||||
|
private array $schemas = [];
|
||||||
|
|
||||||
|
public function __construct(
|
||||||
|
private readonly bool $allowUploadedAvatars,
|
||||||
|
private readonly string $activeSchema,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public function get(string $id): ?AvatarSchemaInterface
|
||||||
|
{
|
||||||
|
return array_get($this->schemas, $id);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getActiveSchema(): ?AvatarSchemaInterface
|
||||||
|
{
|
||||||
|
return $this->get($this->activeSchema);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getAvatarUrl(User $user): ?string
|
||||||
|
{
|
||||||
|
if ($this->allowUploadedAvatars) {
|
||||||
|
$path = "avatars/$user->id.png";
|
||||||
|
|
||||||
|
if (Storage::disk('public')->exists($path)) {
|
||||||
|
return Storage::url($path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->getActiveSchema()?->get($user);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function register(AvatarSchemaInterface $schema): void
|
||||||
|
{
|
||||||
|
if (array_key_exists($schema->getId(), $this->schemas)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->schemas[$schema->getId()] = $schema;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return array<string, string> */
|
||||||
|
public function getMappings(): array
|
||||||
|
{
|
||||||
|
return collect($this->schemas)->mapWithKeys(fn ($schema) => [$schema->getId() => $schema->getName()])->all();
|
||||||
|
}
|
||||||
|
}
|
||||||
24
app/Extensions/Avatar/Schemas/GravatarSchema.php
Normal file
24
app/Extensions/Avatar/Schemas/GravatarSchema.php
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Avatar\Schemas;
|
||||||
|
|
||||||
|
use App\Extensions\Avatar\AvatarSchemaInterface;
|
||||||
|
use App\Models\User;
|
||||||
|
|
||||||
|
class GravatarSchema implements AvatarSchemaInterface
|
||||||
|
{
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return 'gravatar';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getName(): string
|
||||||
|
{
|
||||||
|
return 'Gravatar';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function get(User $user): string
|
||||||
|
{
|
||||||
|
return 'https://gravatar.com/avatar/' . md5($user->email);
|
||||||
|
}
|
||||||
|
}
|
||||||
25
app/Extensions/Avatar/Schemas/UiAvatarsSchema.php
Normal file
25
app/Extensions/Avatar/Schemas/UiAvatarsSchema.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Avatar\Schemas;
|
||||||
|
|
||||||
|
use App\Extensions\Avatar\AvatarSchemaInterface;
|
||||||
|
use App\Models\User;
|
||||||
|
|
||||||
|
class UiAvatarsSchema implements AvatarSchemaInterface
|
||||||
|
{
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return 'uiavatars';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getName(): string
|
||||||
|
{
|
||||||
|
return 'UI Avatars';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function get(User $user): ?string
|
||||||
|
{
|
||||||
|
// UI Avatars is the default of filament so just return null here
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -2,15 +2,16 @@
|
|||||||
|
|
||||||
namespace App\Extensions\Backups;
|
namespace App\Extensions\Backups;
|
||||||
|
|
||||||
use Closure;
|
use App\Extensions\Filesystem\S3Filesystem;
|
||||||
use Aws\S3\S3Client;
|
use Aws\S3\S3Client;
|
||||||
|
use Closure;
|
||||||
|
use Illuminate\Foundation\Application;
|
||||||
use Illuminate\Support\Arr;
|
use Illuminate\Support\Arr;
|
||||||
use Illuminate\Support\Str;
|
use Illuminate\Support\Str;
|
||||||
use Webmozart\Assert\Assert;
|
use InvalidArgumentException;
|
||||||
use Illuminate\Foundation\Application;
|
|
||||||
use League\Flysystem\FilesystemAdapter;
|
use League\Flysystem\FilesystemAdapter;
|
||||||
use App\Extensions\Filesystem\S3Filesystem;
|
|
||||||
use League\Flysystem\InMemory\InMemoryFilesystemAdapter;
|
use League\Flysystem\InMemory\InMemoryFilesystemAdapter;
|
||||||
|
use Webmozart\Assert\Assert;
|
||||||
|
|
||||||
class BackupManager
|
class BackupManager
|
||||||
{
|
{
|
||||||
@ -64,7 +65,7 @@ class BackupManager
|
|||||||
$config = $this->getConfig($name);
|
$config = $this->getConfig($name);
|
||||||
|
|
||||||
if (empty($config['adapter'])) {
|
if (empty($config['adapter'])) {
|
||||||
throw new \InvalidArgumentException("Backup disk [$name] does not have a configured adapter.");
|
throw new InvalidArgumentException("Backup disk [$name] does not have a configured adapter.");
|
||||||
}
|
}
|
||||||
|
|
||||||
$adapter = $config['adapter'];
|
$adapter = $config['adapter'];
|
||||||
@ -82,7 +83,7 @@ class BackupManager
|
|||||||
return $instance;
|
return $instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
throw new \InvalidArgumentException("Adapter [$adapter] is not supported.");
|
throw new InvalidArgumentException("Adapter [$adapter] is not supported.");
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
48
app/Extensions/Captcha/CaptchaService.php
Normal file
48
app/Extensions/Captcha/CaptchaService.php
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Captcha;
|
||||||
|
|
||||||
|
use App\Extensions\Captcha\Schemas\CaptchaSchemaInterface;
|
||||||
|
use Illuminate\Support\Collection;
|
||||||
|
use Illuminate\Support\Str;
|
||||||
|
|
||||||
|
class CaptchaService
|
||||||
|
{
|
||||||
|
/** @var array<string, CaptchaSchemaInterface> */
|
||||||
|
private array $schemas = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return CaptchaSchemaInterface[]
|
||||||
|
*/
|
||||||
|
public function getAll(): array
|
||||||
|
{
|
||||||
|
return $this->schemas;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function get(string $id): ?CaptchaSchemaInterface
|
||||||
|
{
|
||||||
|
return array_get($this->schemas, $id);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function register(CaptchaSchemaInterface $schema): void
|
||||||
|
{
|
||||||
|
if (array_key_exists($schema->getId(), $this->schemas)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
config()->set('captcha.' . Str::lower($schema->getId()), $schema->getConfig());
|
||||||
|
$this->schemas[$schema->getId()] = $schema;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return Collection<CaptchaSchemaInterface> */
|
||||||
|
public function getActiveSchemas(): Collection
|
||||||
|
{
|
||||||
|
return collect($this->schemas)
|
||||||
|
->filter(fn (CaptchaSchemaInterface $schema) => $schema->isEnabled());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getActiveSchema(): ?CaptchaSchemaInterface
|
||||||
|
{
|
||||||
|
return $this->getActiveSchemas()->first();
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,118 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Extensions\Captcha\Providers;
|
|
||||||
|
|
||||||
use Filament\Forms\Components\Component;
|
|
||||||
use Filament\Forms\Components\TextInput;
|
|
||||||
use Illuminate\Foundation\Application;
|
|
||||||
use Illuminate\Support\Str;
|
|
||||||
|
|
||||||
abstract class CaptchaProvider
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* @var array<string, static>
|
|
||||||
*/
|
|
||||||
protected static array $providers = [];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return self|static[]
|
|
||||||
*/
|
|
||||||
public static function get(?string $id = null): array|self
|
|
||||||
{
|
|
||||||
return $id ? static::$providers[$id] : static::$providers;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function __construct(protected Application $app)
|
|
||||||
{
|
|
||||||
if (array_key_exists($this->getId(), static::$providers)) {
|
|
||||||
if (!$this->app->runningUnitTests()) {
|
|
||||||
logger()->warning("Tried to create duplicate Captcha provider with id '{$this->getId()}'");
|
|
||||||
}
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
config()->set('captcha.' . Str::lower($this->getId()), $this->getConfig());
|
|
||||||
|
|
||||||
static::$providers[$this->getId()] = $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
abstract public function getId(): string;
|
|
||||||
|
|
||||||
abstract public function getComponent(): Component;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return array<string, string|string[]|bool|null>
|
|
||||||
*/
|
|
||||||
public function getConfig(): array
|
|
||||||
{
|
|
||||||
$id = Str::upper($this->getId());
|
|
||||||
|
|
||||||
return [
|
|
||||||
'site_key' => env("CAPTCHA_{$id}_SITE_KEY"),
|
|
||||||
'secret_key' => env("CAPTCHA_{$id}_SECRET_KEY"),
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return Component[]
|
|
||||||
*/
|
|
||||||
public function getSettingsForm(): array
|
|
||||||
{
|
|
||||||
$id = Str::upper($this->getId());
|
|
||||||
|
|
||||||
return [
|
|
||||||
TextInput::make("CAPTCHA_{$id}_SITE_KEY")
|
|
||||||
->label('Site Key')
|
|
||||||
->placeholder('Site Key')
|
|
||||||
->columnSpan(2)
|
|
||||||
->required()
|
|
||||||
->password()
|
|
||||||
->revealable()
|
|
||||||
->autocomplete(false)
|
|
||||||
->default(env("CAPTCHA_{$id}_SITE_KEY")),
|
|
||||||
TextInput::make("CAPTCHA_{$id}_SECRET_KEY")
|
|
||||||
->label('Secret Key')
|
|
||||||
->placeholder('Secret Key')
|
|
||||||
->columnSpan(2)
|
|
||||||
->required()
|
|
||||||
->password()
|
|
||||||
->revealable()
|
|
||||||
->autocomplete(false)
|
|
||||||
->default(env("CAPTCHA_{$id}_SECRET_KEY")),
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getName(): string
|
|
||||||
{
|
|
||||||
return Str::title($this->getId());
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getIcon(): ?string
|
|
||||||
{
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function isEnabled(): bool
|
|
||||||
{
|
|
||||||
$id = Str::upper($this->getId());
|
|
||||||
|
|
||||||
return env("CAPTCHA_{$id}_ENABLED", false);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return array<string, string|bool>
|
|
||||||
*/
|
|
||||||
public function validateResponse(?string $captchaResponse = null): array
|
|
||||||
{
|
|
||||||
return [
|
|
||||||
'success' => false,
|
|
||||||
'message' => 'validateResponse not defined',
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
public function verifyDomain(string $hostname, ?string $requestUrl = null): bool
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,106 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Extensions\Captcha\Providers;
|
|
||||||
|
|
||||||
use App\Filament\Components\Forms\Fields\TurnstileCaptcha;
|
|
||||||
use Exception;
|
|
||||||
use Filament\Forms\Components\Component;
|
|
||||||
use Filament\Forms\Components\Placeholder;
|
|
||||||
use Filament\Forms\Components\Toggle;
|
|
||||||
use Illuminate\Foundation\Application;
|
|
||||||
use Illuminate\Support\Facades\Http;
|
|
||||||
use Illuminate\Support\HtmlString;
|
|
||||||
|
|
||||||
class TurnstileProvider extends CaptchaProvider
|
|
||||||
{
|
|
||||||
public function getId(): string
|
|
||||||
{
|
|
||||||
return 'turnstile';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getComponent(): Component
|
|
||||||
{
|
|
||||||
return TurnstileCaptcha::make('turnstile');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return array<string, string|string[]|bool|null>
|
|
||||||
*/
|
|
||||||
public function getConfig(): array
|
|
||||||
{
|
|
||||||
return array_merge(parent::getConfig(), [
|
|
||||||
'verify_domain' => env('CAPTCHA_TURNSTILE_VERIFY_DOMAIN'),
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return Component[]
|
|
||||||
*/
|
|
||||||
public function getSettingsForm(): array
|
|
||||||
{
|
|
||||||
return array_merge(parent::getSettingsForm(), [
|
|
||||||
Toggle::make('CAPTCHA_TURNSTILE_VERIFY_DOMAIN')
|
|
||||||
->label(trans('admin/setting.captcha.verify'))
|
|
||||||
->columnSpan(2)
|
|
||||||
->inline(false)
|
|
||||||
->onIcon('tabler-check')
|
|
||||||
->offIcon('tabler-x')
|
|
||||||
->onColor('success')
|
|
||||||
->offColor('danger')
|
|
||||||
->default(env('CAPTCHA_TURNSTILE_VERIFY_DOMAIN', true)),
|
|
||||||
Placeholder::make('info')
|
|
||||||
->label(trans('admin/setting.captcha.info_label'))
|
|
||||||
->columnSpan(2)
|
|
||||||
->content(new HtmlString(trans('admin/setting.captcha.info'))),
|
|
||||||
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getIcon(): string
|
|
||||||
{
|
|
||||||
return 'tabler-brand-cloudflare';
|
|
||||||
}
|
|
||||||
|
|
||||||
public static function register(Application $app): self
|
|
||||||
{
|
|
||||||
return new self($app);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return array<string, string|bool>
|
|
||||||
*/
|
|
||||||
public function validateResponse(?string $captchaResponse = null): array
|
|
||||||
{
|
|
||||||
$captchaResponse ??= request()->get('cf-turnstile-response');
|
|
||||||
|
|
||||||
if (!$secret = env('CAPTCHA_TURNSTILE_SECRET_KEY')) {
|
|
||||||
throw new Exception('Turnstile secret key is not defined.');
|
|
||||||
}
|
|
||||||
|
|
||||||
$response = Http::asJson()
|
|
||||||
->timeout(15)
|
|
||||||
->connectTimeout(5)
|
|
||||||
->throw()
|
|
||||||
->post('https://challenges.cloudflare.com/turnstile/v0/siteverify', [
|
|
||||||
'secret' => $secret,
|
|
||||||
'response' => $captchaResponse,
|
|
||||||
]);
|
|
||||||
|
|
||||||
return count($response->json()) ? $response->json() : [
|
|
||||||
'success' => false,
|
|
||||||
'message' => 'Unknown error occurred, please try again',
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
public function verifyDomain(string $hostname, ?string $requestUrl = null): bool
|
|
||||||
{
|
|
||||||
if (!env('CAPTCHA_TURNSTILE_VERIFY_DOMAIN', true)) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
$requestUrl ??= request()->url;
|
|
||||||
$requestUrl = parse_url($requestUrl);
|
|
||||||
|
|
||||||
return $hostname === array_get($requestUrl, 'host');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
59
app/Extensions/Captcha/Schemas/BaseSchema.php
Normal file
59
app/Extensions/Captcha/Schemas/BaseSchema.php
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Captcha\Schemas;
|
||||||
|
|
||||||
|
use Filament\Forms\Components\TextInput;
|
||||||
|
use Filament\Schemas\Components\Component;
|
||||||
|
use Illuminate\Support\Str;
|
||||||
|
|
||||||
|
abstract class BaseSchema
|
||||||
|
{
|
||||||
|
abstract public function getId(): string;
|
||||||
|
|
||||||
|
public function getName(): string
|
||||||
|
{
|
||||||
|
return Str::upper($this->getId());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, string|string[]|bool|null>
|
||||||
|
*/
|
||||||
|
public function getConfig(): array
|
||||||
|
{
|
||||||
|
$id = Str::upper($this->getId());
|
||||||
|
|
||||||
|
return [
|
||||||
|
'site_key' => env("CAPTCHA_{$id}_SITE_KEY"),
|
||||||
|
'secret_key' => env("CAPTCHA_{$id}_SECRET_KEY"),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return Component[]
|
||||||
|
*/
|
||||||
|
public function getSettingsForm(): array
|
||||||
|
{
|
||||||
|
$id = Str::upper($this->getId());
|
||||||
|
|
||||||
|
return [
|
||||||
|
TextInput::make("CAPTCHA_{$id}_SITE_KEY")
|
||||||
|
->label('Site Key')
|
||||||
|
->placeholder('Site Key')
|
||||||
|
->columnSpan(2)
|
||||||
|
->required()
|
||||||
|
->password()
|
||||||
|
->revealable()
|
||||||
|
->autocomplete(false)
|
||||||
|
->default(env("CAPTCHA_{$id}_SITE_KEY")),
|
||||||
|
TextInput::make("CAPTCHA_{$id}_SECRET_KEY")
|
||||||
|
->label('Secret Key')
|
||||||
|
->placeholder('Secret Key')
|
||||||
|
->columnSpan(2)
|
||||||
|
->required()
|
||||||
|
->password()
|
||||||
|
->revealable()
|
||||||
|
->autocomplete(false)
|
||||||
|
->default(env("CAPTCHA_{$id}_SECRET_KEY")),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
30
app/Extensions/Captcha/Schemas/CaptchaSchemaInterface.php
Normal file
30
app/Extensions/Captcha/Schemas/CaptchaSchemaInterface.php
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Captcha\Schemas;
|
||||||
|
|
||||||
|
use Filament\Schemas\Components\Component;
|
||||||
|
|
||||||
|
interface CaptchaSchemaInterface
|
||||||
|
{
|
||||||
|
public function getId(): string;
|
||||||
|
|
||||||
|
public function getName(): string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, string|string[]|bool|null>
|
||||||
|
*/
|
||||||
|
public function getConfig(): array;
|
||||||
|
|
||||||
|
public function isEnabled(): bool;
|
||||||
|
|
||||||
|
public function getFormComponent(): Component;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return Component[]
|
||||||
|
*/
|
||||||
|
public function getSettingsForm(): array;
|
||||||
|
|
||||||
|
public function getIcon(): ?string;
|
||||||
|
|
||||||
|
public function validateResponse(?string $captchaResponse = null): void;
|
||||||
|
}
|
||||||
@ -1,11 +1,10 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
namespace App\Filament\Components\Forms\Fields;
|
namespace App\Extensions\Captcha\Schemas\Turnstile;
|
||||||
|
|
||||||
use App\Rules\ValidTurnstileCaptcha;
|
|
||||||
use Filament\Forms\Components\Field;
|
use Filament\Forms\Components\Field;
|
||||||
|
|
||||||
class TurnstileCaptcha extends Field
|
class Component extends Field
|
||||||
{
|
{
|
||||||
protected string $viewIdentifier = 'turnstile';
|
protected string $viewIdentifier = 'turnstile';
|
||||||
|
|
||||||
@ -19,8 +18,6 @@ class TurnstileCaptcha extends Field
|
|||||||
|
|
||||||
$this->required();
|
$this->required();
|
||||||
|
|
||||||
$this->after(function (TurnstileCaptcha $component) {
|
$this->rule(new Rule());
|
||||||
$component->rule(new ValidTurnstileCaptcha());
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
23
app/Extensions/Captcha/Schemas/Turnstile/Rule.php
Normal file
23
app/Extensions/Captcha/Schemas/Turnstile/Rule.php
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Captcha\Schemas\Turnstile;
|
||||||
|
|
||||||
|
use App\Extensions\Captcha\CaptchaService;
|
||||||
|
use Closure;
|
||||||
|
use Exception;
|
||||||
|
use Illuminate\Contracts\Validation\ValidationRule;
|
||||||
|
use Illuminate\Support\Facades\App;
|
||||||
|
|
||||||
|
class Rule implements ValidationRule
|
||||||
|
{
|
||||||
|
public function validate(string $attribute, mixed $value, Closure $fail): void
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
App::call(fn (CaptchaService $service) => $service->get('turnstile')->validateResponse($value));
|
||||||
|
} catch (Exception $exception) {
|
||||||
|
report($exception);
|
||||||
|
|
||||||
|
$fail('Captcha validation failed: ' . $exception->getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
117
app/Extensions/Captcha/Schemas/Turnstile/TurnstileSchema.php
Normal file
117
app/Extensions/Captcha/Schemas/Turnstile/TurnstileSchema.php
Normal file
@ -0,0 +1,117 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Captcha\Schemas\Turnstile;
|
||||||
|
|
||||||
|
use App\Extensions\Captcha\Schemas\BaseSchema;
|
||||||
|
use App\Extensions\Captcha\Schemas\CaptchaSchemaInterface;
|
||||||
|
use Exception;
|
||||||
|
use Filament\Forms\Components\Toggle;
|
||||||
|
use Filament\Infolists\Components\TextEntry;
|
||||||
|
use Illuminate\Support\Facades\Http;
|
||||||
|
use Illuminate\Support\HtmlString;
|
||||||
|
|
||||||
|
class TurnstileSchema extends BaseSchema implements CaptchaSchemaInterface
|
||||||
|
{
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return 'turnstile';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isEnabled(): bool
|
||||||
|
{
|
||||||
|
return env('CAPTCHA_TURNSTILE_ENABLED', false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getFormComponent(): Component
|
||||||
|
{
|
||||||
|
return Component::make('turnstile');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, string|string[]|bool|null>
|
||||||
|
*/
|
||||||
|
public function getConfig(): array
|
||||||
|
{
|
||||||
|
return array_merge(parent::getConfig(), [
|
||||||
|
'verify_domain' => env('CAPTCHA_TURNSTILE_VERIFY_DOMAIN'),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return \Filament\Support\Components\Component[]
|
||||||
|
*
|
||||||
|
* @throws Exception
|
||||||
|
*/
|
||||||
|
public function getSettingsForm(): array
|
||||||
|
{
|
||||||
|
return array_merge(parent::getSettingsForm(), [
|
||||||
|
Toggle::make('CAPTCHA_TURNSTILE_VERIFY_DOMAIN')
|
||||||
|
->label(trans('admin/setting.captcha.verify'))
|
||||||
|
->columnSpan(2)
|
||||||
|
->inline(false)
|
||||||
|
->onIcon('tabler-check')
|
||||||
|
->offIcon('tabler-x')
|
||||||
|
->onColor('success')
|
||||||
|
->offColor('danger')
|
||||||
|
->default(env('CAPTCHA_TURNSTILE_VERIFY_DOMAIN', true)),
|
||||||
|
TextEntry::make('info')
|
||||||
|
->label(trans('admin/setting.captcha.info_label'))
|
||||||
|
->columnSpan(2)
|
||||||
|
->state(new HtmlString(trans('admin/setting.captcha.info'))),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getIcon(): ?string
|
||||||
|
{
|
||||||
|
return 'tabler-brand-cloudflare';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws Exception
|
||||||
|
*/
|
||||||
|
public function validateResponse(?string $captchaResponse = null): void
|
||||||
|
{
|
||||||
|
$captchaResponse ??= request()->get('cf-turnstile-response');
|
||||||
|
|
||||||
|
if (!$secret = env('CAPTCHA_TURNSTILE_SECRET_KEY')) {
|
||||||
|
throw new Exception('Turnstile secret key is not defined.');
|
||||||
|
}
|
||||||
|
|
||||||
|
$response = Http::asJson()
|
||||||
|
->timeout(15)
|
||||||
|
->connectTimeout(5)
|
||||||
|
->throw()
|
||||||
|
->post('https://challenges.cloudflare.com/turnstile/v0/siteverify', [
|
||||||
|
'secret' => $secret,
|
||||||
|
'response' => $captchaResponse,
|
||||||
|
])
|
||||||
|
->json();
|
||||||
|
|
||||||
|
if (!$response['success']) {
|
||||||
|
match ($response['error-codes'][0] ?? null) {
|
||||||
|
'missing-input-secret' => throw new Exception('The secret parameter was not passed.'),
|
||||||
|
'invalid-input-secret' => throw new Exception('The secret parameter was invalid, did not exist, or is a testing secret key with a non-testing response.'),
|
||||||
|
'missing-input-response' => throw new Exception('The response parameter (token) was not passed.'),
|
||||||
|
'invalid-input-response' => throw new Exception('The response parameter (token) is invalid or has expired.'),
|
||||||
|
'bad-request' => throw new Exception('The request was rejected because it was malformed.'),
|
||||||
|
'timeout-or-duplicate' => throw new Exception('The response parameter (token) has already been validated before.'),
|
||||||
|
default => throw new Exception('An internal error happened while validating the response.'),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$this->verifyDomain($response['hostname'] ?? '')) {
|
||||||
|
throw new Exception('Domain verification failed.');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function verifyDomain(string $hostname): bool
|
||||||
|
{
|
||||||
|
if (!env('CAPTCHA_TURNSTILE_VERIFY_DOMAIN', true)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
$requestUrl = parse_url(request()->url());
|
||||||
|
|
||||||
|
return $hostname === array_get($requestUrl, 'host');
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,35 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Extensions;
|
|
||||||
|
|
||||||
use App\Models\DatabaseHost;
|
|
||||||
|
|
||||||
class DynamicDatabaseConnection
|
|
||||||
{
|
|
||||||
public const DB_CHARSET = 'utf8';
|
|
||||||
|
|
||||||
public const DB_COLLATION = 'utf8_unicode_ci';
|
|
||||||
|
|
||||||
public const DB_DRIVER = 'mysql';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Adds a dynamic database connection entry to the runtime config.
|
|
||||||
*/
|
|
||||||
public function set(string $connection, DatabaseHost|int $host, string $database = 'mysql'): void
|
|
||||||
{
|
|
||||||
if (!$host instanceof DatabaseHost) {
|
|
||||||
$host = DatabaseHost::query()->findOrFail($host);
|
|
||||||
}
|
|
||||||
|
|
||||||
config()->set('database.connections.' . $connection, [
|
|
||||||
'driver' => self::DB_DRIVER,
|
|
||||||
'host' => $host->host,
|
|
||||||
'port' => $host->port,
|
|
||||||
'database' => $database,
|
|
||||||
'username' => $host->username,
|
|
||||||
'password' => $host->password,
|
|
||||||
'charset' => self::DB_CHARSET,
|
|
||||||
'collation' => self::DB_COLLATION,
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
15
app/Extensions/Features/FeatureSchemaInterface.php
Normal file
15
app/Extensions/Features/FeatureSchemaInterface.php
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Features;
|
||||||
|
|
||||||
|
use Filament\Actions\Action;
|
||||||
|
|
||||||
|
interface FeatureSchemaInterface
|
||||||
|
{
|
||||||
|
/** @return string[] */
|
||||||
|
public function getListeners(): array;
|
||||||
|
|
||||||
|
public function getId(): string;
|
||||||
|
|
||||||
|
public function getAction(): Action;
|
||||||
|
}
|
||||||
52
app/Extensions/Features/FeatureService.php
Normal file
52
app/Extensions/Features/FeatureService.php
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Features;
|
||||||
|
|
||||||
|
class FeatureService
|
||||||
|
{
|
||||||
|
/** @var FeatureSchemaInterface[] */
|
||||||
|
private array $schemas = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return FeatureSchemaInterface[]
|
||||||
|
*/
|
||||||
|
public function getAll(): array
|
||||||
|
{
|
||||||
|
return $this->schemas;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function get(string $id): ?FeatureSchemaInterface
|
||||||
|
{
|
||||||
|
return array_get($this->schemas, $id);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param ?string[] $features
|
||||||
|
* @return FeatureSchemaInterface[]
|
||||||
|
*/
|
||||||
|
public function getActiveSchemas(?array $features = []): array
|
||||||
|
{
|
||||||
|
return collect($this->schemas)->only($features)->all();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function register(FeatureSchemaInterface $schema): void
|
||||||
|
{
|
||||||
|
if (array_key_exists($schema->getId(), $this->schemas)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->schemas[$schema->getId()] = $schema;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param ?string[] $features
|
||||||
|
* @return array<string, array<string>>
|
||||||
|
*/
|
||||||
|
public function getMappings(?array $features = []): array
|
||||||
|
{
|
||||||
|
return collect($this->getActiveSchemas($features))
|
||||||
|
->mapWithKeys(fn (FeatureSchemaInterface $schema) => [
|
||||||
|
$schema->getId() => $schema->getListeners(),
|
||||||
|
])->all();
|
||||||
|
}
|
||||||
|
}
|
||||||
119
app/Extensions/Features/Schemas/GSLTokenSchema.php
Normal file
119
app/Extensions/Features/Schemas/GSLTokenSchema.php
Normal file
@ -0,0 +1,119 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Features\Schemas;
|
||||||
|
|
||||||
|
use App\Extensions\Features\FeatureSchemaInterface;
|
||||||
|
use App\Facades\Activity;
|
||||||
|
use App\Models\Permission;
|
||||||
|
use App\Models\Server;
|
||||||
|
use App\Models\ServerVariable;
|
||||||
|
use App\Repositories\Daemon\DaemonServerRepository;
|
||||||
|
use Closure;
|
||||||
|
use Exception;
|
||||||
|
use Filament\Actions\Action;
|
||||||
|
use Filament\Facades\Filament;
|
||||||
|
use Filament\Forms\Components\TextInput;
|
||||||
|
use Filament\Infolists\Components\TextEntry;
|
||||||
|
use Filament\Notifications\Notification;
|
||||||
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
|
use Illuminate\Support\Facades\Blade;
|
||||||
|
use Illuminate\Support\Facades\Validator;
|
||||||
|
use Illuminate\Support\HtmlString;
|
||||||
|
|
||||||
|
class GSLTokenSchema implements FeatureSchemaInterface
|
||||||
|
{
|
||||||
|
/** @return array<string> */
|
||||||
|
public function getListeners(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'(gsl token expired)',
|
||||||
|
'(account not found)',
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return 'gsl_token';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws Exception
|
||||||
|
*/
|
||||||
|
public function getAction(): Action
|
||||||
|
{
|
||||||
|
/** @var Server $server */
|
||||||
|
$server = Filament::getTenant();
|
||||||
|
|
||||||
|
/** @var ServerVariable $serverVariable */
|
||||||
|
$serverVariable = $server->serverVariables()->whereHas('variable', function (Builder $query) {
|
||||||
|
$query->where('env_variable', 'STEAM_ACC');
|
||||||
|
})->first();
|
||||||
|
|
||||||
|
return Action::make($this->getId())
|
||||||
|
->requiresConfirmation()
|
||||||
|
->modalHeading('Invalid GSL token')
|
||||||
|
->modalDescription('It seems like your Gameserver Login Token (GSL token) is invalid or has expired.')
|
||||||
|
->modalSubmitActionLabel('Update GSL Token')
|
||||||
|
->disabledSchema(fn () => !user()?->can(Permission::ACTION_STARTUP_UPDATE, $server))
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('info')
|
||||||
|
->label(new HtmlString(Blade::render('You can either <x-filament::link href="https://steamcommunity.com/dev/managegameservers" target="_blank">generate a new one</x-filament::link> and enter it below or leave the field blank to remove it completely.'))),
|
||||||
|
TextInput::make('gsltoken')
|
||||||
|
->label('GSL Token')
|
||||||
|
->rules([
|
||||||
|
fn (): Closure => function (string $attribute, $value, Closure $fail) use ($serverVariable) {
|
||||||
|
$validator = Validator::make(['validatorkey' => $value], [
|
||||||
|
'validatorkey' => $serverVariable->variable->rules,
|
||||||
|
]);
|
||||||
|
|
||||||
|
if ($validator->fails()) {
|
||||||
|
$message = str($validator->errors()->first())->replace('validatorkey', $serverVariable->variable->name);
|
||||||
|
|
||||||
|
$fail($message);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
])
|
||||||
|
->hintIcon('tabler-code', fn () => implode('|', $serverVariable->variable->rules))
|
||||||
|
->label(fn () => $serverVariable->variable->name)
|
||||||
|
->prefix(fn () => '{{' . $serverVariable->variable->env_variable . '}}')
|
||||||
|
->helperText(fn () => empty($serverVariable->variable->description) ? '—' : $serverVariable->variable->description),
|
||||||
|
])
|
||||||
|
->action(function (array $data, DaemonServerRepository $serverRepository) use ($server, $serverVariable) {
|
||||||
|
/** @var Server $server */
|
||||||
|
$server = Filament::getTenant();
|
||||||
|
try {
|
||||||
|
$new = $data['gsltoken'] ?? '';
|
||||||
|
$original = $serverVariable->variable_value;
|
||||||
|
|
||||||
|
$serverVariable->update([
|
||||||
|
'variable_value' => $new,
|
||||||
|
]);
|
||||||
|
|
||||||
|
if ($original !== $new) {
|
||||||
|
|
||||||
|
Activity::event('server:startup.edit')
|
||||||
|
->property([
|
||||||
|
'variable' => $serverVariable->variable->env_variable,
|
||||||
|
'old' => $original,
|
||||||
|
'new' => $new,
|
||||||
|
])
|
||||||
|
->log();
|
||||||
|
}
|
||||||
|
|
||||||
|
$serverRepository->setServer($server)->power('restart');
|
||||||
|
|
||||||
|
Notification::make()
|
||||||
|
->title('GSL Token updated')
|
||||||
|
->body('Server will restart now.')
|
||||||
|
->success()
|
||||||
|
->send();
|
||||||
|
} catch (Exception $exception) {
|
||||||
|
Notification::make()
|
||||||
|
->title('Could not update GSL Token')
|
||||||
|
->body($exception->getMessage())
|
||||||
|
->danger()
|
||||||
|
->send();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
89
app/Extensions/Features/Schemas/JavaVersionSchema.php
Normal file
89
app/Extensions/Features/Schemas/JavaVersionSchema.php
Normal file
@ -0,0 +1,89 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Features\Schemas;
|
||||||
|
|
||||||
|
use App\Extensions\Features\FeatureSchemaInterface;
|
||||||
|
use App\Facades\Activity;
|
||||||
|
use App\Models\Permission;
|
||||||
|
use App\Models\Server;
|
||||||
|
use App\Repositories\Daemon\DaemonServerRepository;
|
||||||
|
use Exception;
|
||||||
|
use Filament\Actions\Action;
|
||||||
|
use Filament\Facades\Filament;
|
||||||
|
use Filament\Forms\Components\Select;
|
||||||
|
use Filament\Infolists\Components\TextEntry;
|
||||||
|
use Filament\Notifications\Notification;
|
||||||
|
|
||||||
|
class JavaVersionSchema implements FeatureSchemaInterface
|
||||||
|
{
|
||||||
|
/** @return array<string> */
|
||||||
|
public function getListeners(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'java.lang.UnsupportedClassVersionError',
|
||||||
|
'unsupported major.minor version',
|
||||||
|
'has been compiled by a more recent version of the java runtime',
|
||||||
|
'minecraft 1.17 requires running the server with java 16 or above',
|
||||||
|
'minecraft 1.18 requires running the server with java 17 or above',
|
||||||
|
'minecraft 1.19 requires running the server with java 17 or above',
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return 'java_version';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getAction(): Action
|
||||||
|
{
|
||||||
|
/** @var Server $server */
|
||||||
|
$server = Filament::getTenant();
|
||||||
|
|
||||||
|
return Action::make($this->getId())
|
||||||
|
->requiresConfirmation()
|
||||||
|
->modalHeading('Unsupported Java Version')
|
||||||
|
->modalDescription('This server is currently running an unsupported version of Java and cannot be started.')
|
||||||
|
->modalSubmitActionLabel('Update Docker Image')
|
||||||
|
->disabledSchema(fn () => !user()?->can(Permission::ACTION_STARTUP_DOCKER_IMAGE, $server))
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('java')
|
||||||
|
->label('Please select a supported version from the list below to continue starting the server.'),
|
||||||
|
Select::make('image')
|
||||||
|
->label('Docker Image')
|
||||||
|
->disabled(fn () => !in_array($server->image, $server->egg->docker_images))
|
||||||
|
->options(fn () => collect($server->egg->docker_images)->mapWithKeys(fn ($key, $value) => [$key => $value]))
|
||||||
|
->selectablePlaceholder(false)
|
||||||
|
->default(fn () => $server->image)
|
||||||
|
->notIn(fn () => $server->image)
|
||||||
|
->required()
|
||||||
|
->preload(),
|
||||||
|
])
|
||||||
|
->action(function (array $data, DaemonServerRepository $serverRepository) use ($server) {
|
||||||
|
try {
|
||||||
|
$new = $data['image'];
|
||||||
|
$original = $server->image;
|
||||||
|
$server->forceFill(['image' => $new])->saveOrFail();
|
||||||
|
|
||||||
|
if ($original !== $server->image) {
|
||||||
|
Activity::event('server:startup.image')
|
||||||
|
->property(['old' => $original, 'new' => $new])
|
||||||
|
->log();
|
||||||
|
}
|
||||||
|
|
||||||
|
$serverRepository->setServer($server)->power('restart');
|
||||||
|
|
||||||
|
Notification::make()
|
||||||
|
->title('Docker image updated')
|
||||||
|
->body('Server will restart now.')
|
||||||
|
->success()
|
||||||
|
->send();
|
||||||
|
} catch (Exception $exception) {
|
||||||
|
Notification::make()
|
||||||
|
->title('Could not update docker image')
|
||||||
|
->body($exception->getMessage())
|
||||||
|
->danger()
|
||||||
|
->send();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
61
app/Extensions/Features/Schemas/MinecraftEulaSchema.php
Normal file
61
app/Extensions/Features/Schemas/MinecraftEulaSchema.php
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Features\Schemas;
|
||||||
|
|
||||||
|
use App\Extensions\Features\FeatureSchemaInterface;
|
||||||
|
use App\Models\Server;
|
||||||
|
use App\Repositories\Daemon\DaemonFileRepository;
|
||||||
|
use App\Repositories\Daemon\DaemonServerRepository;
|
||||||
|
use Exception;
|
||||||
|
use Filament\Actions\Action;
|
||||||
|
use Filament\Facades\Filament;
|
||||||
|
use Filament\Notifications\Notification;
|
||||||
|
use Illuminate\Support\Facades\Blade;
|
||||||
|
use Illuminate\Support\HtmlString;
|
||||||
|
|
||||||
|
class MinecraftEulaSchema implements FeatureSchemaInterface
|
||||||
|
{
|
||||||
|
/** @return array<string> */
|
||||||
|
public function getListeners(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'you need to agree to the eula in order to run the server',
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return 'eula';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getAction(): Action
|
||||||
|
{
|
||||||
|
return Action::make($this->getId())
|
||||||
|
->requiresConfirmation()
|
||||||
|
->modalHeading('Minecraft EULA')
|
||||||
|
->modalDescription(new HtmlString(Blade::render('By pressing "I Accept" below you are indicating your agreement to the <x-filament::link href="https://minecraft.net/eula" target="_blank">Minecraft EULA </x-filament::link>.')))
|
||||||
|
->modalSubmitActionLabel('I Accept')
|
||||||
|
->action(function (DaemonFileRepository $fileRepository, DaemonServerRepository $serverRepository) {
|
||||||
|
try {
|
||||||
|
/** @var Server $server */
|
||||||
|
$server = Filament::getTenant();
|
||||||
|
|
||||||
|
$fileRepository->setServer($server)->putContent('eula.txt', 'eula=true');
|
||||||
|
|
||||||
|
$serverRepository->setServer($server)->power('restart');
|
||||||
|
|
||||||
|
Notification::make()
|
||||||
|
->title('Minecraft EULA accepted')
|
||||||
|
->body('Server will restart now.')
|
||||||
|
->success()
|
||||||
|
->send();
|
||||||
|
} catch (Exception $exception) {
|
||||||
|
Notification::make()
|
||||||
|
->title('Could not accept Minecraft EULA')
|
||||||
|
->body($exception->getMessage())
|
||||||
|
->danger()
|
||||||
|
->send();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
66
app/Extensions/Features/Schemas/PIDLimitSchema.php
Normal file
66
app/Extensions/Features/Schemas/PIDLimitSchema.php
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Features\Schemas;
|
||||||
|
|
||||||
|
use App\Extensions\Features\FeatureSchemaInterface;
|
||||||
|
use Filament\Actions\Action;
|
||||||
|
use Illuminate\Support\Facades\Blade;
|
||||||
|
use Illuminate\Support\HtmlString;
|
||||||
|
|
||||||
|
class PIDLimitSchema implements FeatureSchemaInterface
|
||||||
|
{
|
||||||
|
/** @return array<string> */
|
||||||
|
public function getListeners(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'pthread_create failed',
|
||||||
|
'failed to create thread',
|
||||||
|
'unable to create thread',
|
||||||
|
'unable to create native thread',
|
||||||
|
'unable to create new native thread',
|
||||||
|
'exception in thread "craft async scheduler management thread"',
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return 'pid_limit';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getAction(): Action
|
||||||
|
{
|
||||||
|
return Action::make($this->getId())
|
||||||
|
->requiresConfirmation()
|
||||||
|
->icon('tabler-alert-triangle')
|
||||||
|
->modalHeading(fn () => user()?->isAdmin() ? 'Memory or process limit reached...' : 'Possible resource limit reached...')
|
||||||
|
->modalDescription(new HtmlString(Blade::render(
|
||||||
|
user()?->isAdmin() ? <<<'HTML'
|
||||||
|
<p>
|
||||||
|
This server has reached the maximum process or memory limit.
|
||||||
|
</p>
|
||||||
|
<p class="mt-4">
|
||||||
|
Increasing <code>container_pid_limit</code> in the wings
|
||||||
|
configuration, <code>config.yml</code>, might help resolve
|
||||||
|
this issue.
|
||||||
|
</p>
|
||||||
|
<p class="mt-4">
|
||||||
|
<b>Note: Wings must be restarted for the configuration file changes to take effect</b>
|
||||||
|
</p>
|
||||||
|
HTML
|
||||||
|
:
|
||||||
|
<<<'HTML'
|
||||||
|
<p>
|
||||||
|
This server is attempting to use more resources than allocated. Please contact the administrator
|
||||||
|
and give them the error below.
|
||||||
|
</p>
|
||||||
|
<p class="mt-4">
|
||||||
|
<code>
|
||||||
|
pthread_create failed, Possibly out of memory or process/resource limits reached
|
||||||
|
</code>
|
||||||
|
</p>
|
||||||
|
HTML
|
||||||
|
)))
|
||||||
|
->modalCancelActionLabel('Close')
|
||||||
|
->action(fn () => null);
|
||||||
|
}
|
||||||
|
}
|
||||||
54
app/Extensions/Features/Schemas/SteamDiskSpaceSchema.php
Normal file
54
app/Extensions/Features/Schemas/SteamDiskSpaceSchema.php
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\Features\Schemas;
|
||||||
|
|
||||||
|
use App\Extensions\Features\FeatureSchemaInterface;
|
||||||
|
use Filament\Actions\Action;
|
||||||
|
use Illuminate\Support\Facades\Blade;
|
||||||
|
use Illuminate\Support\HtmlString;
|
||||||
|
|
||||||
|
class SteamDiskSpaceSchema implements FeatureSchemaInterface
|
||||||
|
{
|
||||||
|
/** @return array<string> */
|
||||||
|
public function getListeners(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'steamcmd needs 250mb of free disk space to update',
|
||||||
|
'0x202 after update job',
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return 'steam_disk_space';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getAction(): Action
|
||||||
|
{
|
||||||
|
return Action::make($this->getId())
|
||||||
|
->requiresConfirmation()
|
||||||
|
->modalHeading('Out of available disk space...')
|
||||||
|
->modalDescription(new HtmlString(Blade::render(
|
||||||
|
user()?->isAdmin() ? <<<'HTML'
|
||||||
|
<p>
|
||||||
|
This server has run out of available disk space and cannot complete the install or update
|
||||||
|
process.
|
||||||
|
</p>
|
||||||
|
<p class="mt-4">
|
||||||
|
Ensure the machine has enough disk space by typing{' '}
|
||||||
|
<code class="rounded py-1 px-2">df -h</code> on the machine hosting
|
||||||
|
this server. Delete files or increase the available disk space to resolve the issue.
|
||||||
|
</p>
|
||||||
|
HTML
|
||||||
|
:
|
||||||
|
<<<'HTML'
|
||||||
|
<p>
|
||||||
|
This server has run out of available disk space and cannot complete the install or update
|
||||||
|
process. Please get in touch with the administrator(s) and inform them of disk space issues.
|
||||||
|
</p>
|
||||||
|
HTML
|
||||||
|
)))
|
||||||
|
->modalCancelActionLabel('Close')
|
||||||
|
->action(fn () => null);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -6,7 +6,7 @@ use App\Models\ApiKey;
|
|||||||
use Laravel\Sanctum\NewAccessToken as SanctumAccessToken;
|
use Laravel\Sanctum\NewAccessToken as SanctumAccessToken;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @property \App\Models\ApiKey $accessToken
|
* @property ApiKey $accessToken
|
||||||
*/
|
*/
|
||||||
class NewAccessToken extends SanctumAccessToken
|
class NewAccessToken extends SanctumAccessToken
|
||||||
{
|
{
|
||||||
|
|||||||
@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Extensions\Lcobucci\JWT\Encoding;
|
namespace App\Extensions\Lcobucci\JWT\Encoding;
|
||||||
|
|
||||||
|
use DateTimeImmutable;
|
||||||
use Lcobucci\JWT\ClaimsFormatter;
|
use Lcobucci\JWT\ClaimsFormatter;
|
||||||
use Lcobucci\JWT\Token\RegisteredClaims;
|
use Lcobucci\JWT\Token\RegisteredClaims;
|
||||||
|
|
||||||
@ -20,7 +21,7 @@ final class TimestampDates implements ClaimsFormatter
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
assert($claims[$claim] instanceof \DateTimeImmutable);
|
assert($claims[$claim] instanceof DateTimeImmutable);
|
||||||
$claims[$claim] = $claims[$claim]->getTimestamp();
|
$claims[$claim] = $claims[$claim]->getTimestamp();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
39
app/Extensions/OAuth/OAuthSchemaInterface.php
Normal file
39
app/Extensions/OAuth/OAuthSchemaInterface.php
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\OAuth;
|
||||||
|
|
||||||
|
use Filament\Schemas\Components\Component;
|
||||||
|
use Filament\Schemas\Components\Wizard\Step;
|
||||||
|
|
||||||
|
interface OAuthSchemaInterface
|
||||||
|
{
|
||||||
|
public function getId(): string;
|
||||||
|
|
||||||
|
public function getName(): string;
|
||||||
|
|
||||||
|
public function getConfigKey(): string;
|
||||||
|
|
||||||
|
/** @return ?class-string */
|
||||||
|
public function getSocialiteProvider(): ?string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<string, string|string[]|bool|null>
|
||||||
|
*/
|
||||||
|
public function getServiceConfig(): array;
|
||||||
|
|
||||||
|
/** @return Component[] */
|
||||||
|
public function getSettingsForm(): array;
|
||||||
|
|
||||||
|
/** @return Step[] */
|
||||||
|
public function getSetupSteps(): array;
|
||||||
|
|
||||||
|
public function getIcon(): ?string;
|
||||||
|
|
||||||
|
public function getHexColor(): ?string;
|
||||||
|
|
||||||
|
public function isEnabled(): bool;
|
||||||
|
|
||||||
|
public function shouldCreateMissingUsers(): bool;
|
||||||
|
|
||||||
|
public function shouldLinkMissingUsers(): bool;
|
||||||
|
}
|
||||||
71
app/Extensions/OAuth/OAuthService.php
Normal file
71
app/Extensions/OAuth/OAuthService.php
Normal file
@ -0,0 +1,71 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\OAuth;
|
||||||
|
|
||||||
|
use App\Models\User;
|
||||||
|
use Illuminate\Support\Facades\Event;
|
||||||
|
use Laravel\Socialite\Contracts\User as OAuthUser;
|
||||||
|
use SocialiteProviders\Manager\SocialiteWasCalled;
|
||||||
|
|
||||||
|
class OAuthService
|
||||||
|
{
|
||||||
|
/** @var OAuthSchemaInterface[] */
|
||||||
|
private array $schemas = [];
|
||||||
|
|
||||||
|
/** @return OAuthSchemaInterface[] */
|
||||||
|
public function getAll(): array
|
||||||
|
{
|
||||||
|
return $this->schemas;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function get(string $id): ?OAuthSchemaInterface
|
||||||
|
{
|
||||||
|
return array_get($this->schemas, $id);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @return OAuthSchemaInterface[] */
|
||||||
|
public function getEnabled(): array
|
||||||
|
{
|
||||||
|
return collect($this->schemas)
|
||||||
|
->filter(fn (OAuthSchemaInterface $schema) => $schema->isEnabled())
|
||||||
|
->all();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function register(OAuthSchemaInterface $schema): void
|
||||||
|
{
|
||||||
|
if (array_key_exists($schema->getId(), $this->schemas)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
config()->set('services.' . $schema->getId(), array_merge($schema->getServiceConfig(), ['redirect' => '/auth/oauth/callback/' . $schema->getId()]));
|
||||||
|
|
||||||
|
if ($schema->getSocialiteProvider()) {
|
||||||
|
Event::listen(fn (SocialiteWasCalled $event) => $event->extendSocialite($schema->getId(), $schema->getSocialiteProvider()));
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->schemas[$schema->getId()] = $schema;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function linkUser(User $user, OAuthSchemaInterface $schema, OAuthUser $oauthUser): User
|
||||||
|
{
|
||||||
|
$oauth = $user->oauth ?? [];
|
||||||
|
$oauth[$schema->getId()] = $oauthUser->getId();
|
||||||
|
|
||||||
|
$user->update(['oauth' => $oauth]);
|
||||||
|
|
||||||
|
return $user->refresh();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function unlinkUser(User $user, OAuthSchemaInterface $schema): User
|
||||||
|
{
|
||||||
|
$oauth = $user->oauth ?? [];
|
||||||
|
if (!isset($oauth[$schema->getId()])) {
|
||||||
|
return $user;
|
||||||
|
}
|
||||||
|
|
||||||
|
unset($oauth[$schema->getId()]);
|
||||||
|
$user->update(['oauth' => $oauth]);
|
||||||
|
|
||||||
|
return $user->refresh();
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,38 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Extensions\OAuth\Providers;
|
|
||||||
|
|
||||||
use Illuminate\Foundation\Application;
|
|
||||||
|
|
||||||
final class CommonProvider extends OAuthProvider
|
|
||||||
{
|
|
||||||
protected function __construct(protected Application $app, private string $id, private ?string $providerClass, private ?string $icon, private ?string $hexColor)
|
|
||||||
{
|
|
||||||
parent::__construct($app);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getId(): string
|
|
||||||
{
|
|
||||||
return $this->id;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getProviderClass(): ?string
|
|
||||||
{
|
|
||||||
return $this->providerClass;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getIcon(): ?string
|
|
||||||
{
|
|
||||||
return $this->icon;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getHexColor(): ?string
|
|
||||||
{
|
|
||||||
return $this->hexColor;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static function register(Application $app, string $id, ?string $providerClass = null, ?string $icon = null, ?string $hexColor = null): static
|
|
||||||
{
|
|
||||||
return new self($app, $id, $providerClass, $icon, $hexColor);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,64 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Extensions\OAuth\Providers;
|
|
||||||
|
|
||||||
use Filament\Forms\Components\Placeholder;
|
|
||||||
use Filament\Forms\Components\TextInput;
|
|
||||||
use Filament\Forms\Components\Wizard\Step;
|
|
||||||
use Illuminate\Foundation\Application;
|
|
||||||
use Illuminate\Support\HtmlString;
|
|
||||||
use Illuminate\Support\Str;
|
|
||||||
use SocialiteProviders\Discord\Provider;
|
|
||||||
use Webbingbrasil\FilamentCopyActions\Forms\Actions\CopyAction;
|
|
||||||
|
|
||||||
final class DiscordProvider extends OAuthProvider
|
|
||||||
{
|
|
||||||
public function __construct(protected Application $app)
|
|
||||||
{
|
|
||||||
parent::__construct($app);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getId(): string
|
|
||||||
{
|
|
||||||
return 'discord';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getProviderClass(): string
|
|
||||||
{
|
|
||||||
return Provider::class;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getSetupSteps(): array
|
|
||||||
{
|
|
||||||
return array_merge([
|
|
||||||
Step::make('Register new Discord OAuth App')
|
|
||||||
->schema([
|
|
||||||
Placeholder::make('')
|
|
||||||
->content(new HtmlString('<p>Visit the <u><a href="https://discord.com/developers/applications" target="_blank">Discord Developer Portal</a></u> and click on <b>New Application</b>. Enter a <b>Name</b> (e.g. your panel name) and click on <b>Create</b>.</p><p>Copy the <b>Client ID</b> and the <b>Client Secret</b>, you will need them in the final step.</p>')),
|
|
||||||
Placeholder::make('')
|
|
||||||
->content(new HtmlString('<p>Under <b>Redirects</b> add the below URL.</p>')),
|
|
||||||
TextInput::make('_noenv_callback')
|
|
||||||
->label('Redirect URL')
|
|
||||||
->dehydrated()
|
|
||||||
->disabled()
|
|
||||||
->hintAction(fn () => request()->isSecure() ? CopyAction::make() : null)
|
|
||||||
->formatStateUsing(fn () => config('app.url') . (Str::endsWith(config('app.url'), '/') ? '' : '/') . 'auth/oauth/callback/discord'),
|
|
||||||
]),
|
|
||||||
], parent::getSetupSteps());
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getIcon(): string
|
|
||||||
{
|
|
||||||
return 'tabler-brand-discord-f';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getHexColor(): string
|
|
||||||
{
|
|
||||||
return '#5865F2';
|
|
||||||
}
|
|
||||||
|
|
||||||
public static function register(Application $app): self
|
|
||||||
{
|
|
||||||
return new self($app);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,63 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Extensions\OAuth\Providers;
|
|
||||||
|
|
||||||
use Filament\Forms\Components\Placeholder;
|
|
||||||
use Filament\Forms\Components\TextInput;
|
|
||||||
use Filament\Forms\Components\Wizard\Step;
|
|
||||||
use Illuminate\Foundation\Application;
|
|
||||||
use Illuminate\Support\HtmlString;
|
|
||||||
use Illuminate\Support\Str;
|
|
||||||
use Webbingbrasil\FilamentCopyActions\Forms\Actions\CopyAction;
|
|
||||||
|
|
||||||
final class GithubProvider extends OAuthProvider
|
|
||||||
{
|
|
||||||
public function __construct(protected Application $app)
|
|
||||||
{
|
|
||||||
parent::__construct($app);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getId(): string
|
|
||||||
{
|
|
||||||
return 'github';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getSetupSteps(): array
|
|
||||||
{
|
|
||||||
return array_merge([
|
|
||||||
Step::make('Register new Github OAuth App')
|
|
||||||
->schema([
|
|
||||||
Placeholder::make('')
|
|
||||||
->content(new HtmlString('<p>Visit the <u><a href="https://github.com/settings/developers" target="_blank">Github Developer Dashboard</a></u>, go to <b>OAuth Apps</b> and click on <b>New OAuth App</b>.</p><p>Enter an <b>Application name</b> (e.g. your panel name), set <b>Homepage URL</b> to your panel url and enter the below url as <b>Authorization callback URL</b>.</p>')),
|
|
||||||
TextInput::make('_noenv_callback')
|
|
||||||
->label('Authorization callback URL')
|
|
||||||
->dehydrated()
|
|
||||||
->disabled()
|
|
||||||
->hintAction(fn () => request()->isSecure() ? CopyAction::make() : null)
|
|
||||||
->default(fn () => config('app.url') . (Str::endsWith(config('app.url'), '/') ? '' : '/') . 'auth/oauth/callback/github'),
|
|
||||||
Placeholder::make('')
|
|
||||||
->content(new HtmlString('<p>When you filled all fields click on <b>Register application</b>.</p>')),
|
|
||||||
]),
|
|
||||||
Step::make('Create Client Secret')
|
|
||||||
->schema([
|
|
||||||
Placeholder::make('')
|
|
||||||
->content(new HtmlString('<p>Once you registered your app, generate a new <b>Client Secret</b>.</p><p>You will also need the <b>Client ID</b>.</p>')),
|
|
||||||
]),
|
|
||||||
], parent::getSetupSteps());
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getIcon(): string
|
|
||||||
{
|
|
||||||
return 'tabler-brand-github-f';
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getHexColor(): string
|
|
||||||
{
|
|
||||||
return '#4078c0';
|
|
||||||
}
|
|
||||||
|
|
||||||
public static function register(Application $app): self
|
|
||||||
{
|
|
||||||
return new self($app);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,131 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Extensions\OAuth\Providers;
|
|
||||||
|
|
||||||
use Filament\Forms\Components\Component;
|
|
||||||
use Filament\Forms\Components\TextInput;
|
|
||||||
use Filament\Forms\Components\Wizard\Step;
|
|
||||||
use Illuminate\Foundation\Application;
|
|
||||||
use Illuminate\Support\Facades\Event;
|
|
||||||
use Illuminate\Support\Str;
|
|
||||||
use SocialiteProviders\Manager\SocialiteWasCalled;
|
|
||||||
|
|
||||||
abstract class OAuthProvider
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* @var array<string, static>
|
|
||||||
*/
|
|
||||||
protected static array $providers = [];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return self|static[]
|
|
||||||
*/
|
|
||||||
public static function get(?string $id = null): array|self
|
|
||||||
{
|
|
||||||
return $id ? static::$providers[$id] : static::$providers;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function __construct(protected Application $app)
|
|
||||||
{
|
|
||||||
if (array_key_exists($this->getId(), static::$providers)) {
|
|
||||||
if (!$this->app->runningUnitTests()) {
|
|
||||||
logger()->warning("Tried to create duplicate OAuth provider with id '{$this->getId()}'");
|
|
||||||
}
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
config()->set('services.' . $this->getId(), array_merge($this->getServiceConfig(), ['redirect' => '/auth/oauth/callback/' . $this->getId()]));
|
|
||||||
|
|
||||||
if ($this->getProviderClass()) {
|
|
||||||
Event::listen(function (SocialiteWasCalled $event) {
|
|
||||||
$event->extendSocialite($this->getId(), $this->getProviderClass());
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
static::$providers[$this->getId()] = $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
abstract public function getId(): string;
|
|
||||||
|
|
||||||
public function getProviderClass(): ?string
|
|
||||||
{
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return array<string, string|string[]|bool|null>
|
|
||||||
*/
|
|
||||||
public function getServiceConfig(): array
|
|
||||||
{
|
|
||||||
$id = Str::upper($this->getId());
|
|
||||||
|
|
||||||
return [
|
|
||||||
'client_id' => env("OAUTH_{$id}_CLIENT_ID"),
|
|
||||||
'client_secret' => env("OAUTH_{$id}_CLIENT_SECRET"),
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return Component[]
|
|
||||||
*/
|
|
||||||
public function getSettingsForm(): array
|
|
||||||
{
|
|
||||||
$id = Str::upper($this->getId());
|
|
||||||
|
|
||||||
return [
|
|
||||||
TextInput::make("OAUTH_{$id}_CLIENT_ID")
|
|
||||||
->label('Client ID')
|
|
||||||
->placeholder('Client ID')
|
|
||||||
->columnSpan(2)
|
|
||||||
->required()
|
|
||||||
->password()
|
|
||||||
->revealable()
|
|
||||||
->autocomplete(false)
|
|
||||||
->default(env("OAUTH_{$id}_CLIENT_ID")),
|
|
||||||
TextInput::make("OAUTH_{$id}_CLIENT_SECRET")
|
|
||||||
->label('Client Secret')
|
|
||||||
->placeholder('Client Secret')
|
|
||||||
->columnSpan(2)
|
|
||||||
->required()
|
|
||||||
->password()
|
|
||||||
->revealable()
|
|
||||||
->autocomplete(false)
|
|
||||||
->default(env("OAUTH_{$id}_CLIENT_SECRET")),
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return Step[]
|
|
||||||
*/
|
|
||||||
public function getSetupSteps(): array
|
|
||||||
{
|
|
||||||
return [
|
|
||||||
Step::make('OAuth Config')
|
|
||||||
->columns(4)
|
|
||||||
->schema($this->getSettingsForm()),
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getName(): string
|
|
||||||
{
|
|
||||||
return Str::title($this->getId());
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getIcon(): ?string
|
|
||||||
{
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getHexColor(): ?string
|
|
||||||
{
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function isEnabled(): bool
|
|
||||||
{
|
|
||||||
$id = Str::upper($this->getId());
|
|
||||||
|
|
||||||
return env("OAUTH_{$id}_ENABLED", false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,36 +1,50 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
namespace App\Extensions\OAuth\Providers;
|
namespace App\Extensions\OAuth\Schemas;
|
||||||
|
|
||||||
use Filament\Forms\Components\ColorPicker;
|
use Filament\Forms\Components\ColorPicker;
|
||||||
use Filament\Forms\Components\TextInput;
|
use Filament\Forms\Components\TextInput;
|
||||||
use Illuminate\Foundation\Application;
|
use Filament\Infolists\Components\TextEntry;
|
||||||
|
use Filament\Schemas\Components\Wizard\Step;
|
||||||
|
use Illuminate\Support\Facades\Blade;
|
||||||
|
use Illuminate\Support\HtmlString;
|
||||||
use SocialiteProviders\Authentik\Provider;
|
use SocialiteProviders\Authentik\Provider;
|
||||||
|
|
||||||
final class AuthentikProvider extends OAuthProvider
|
final class AuthentikSchema extends OAuthSchema
|
||||||
{
|
{
|
||||||
public function __construct(protected Application $app)
|
|
||||||
{
|
|
||||||
parent::__construct($app);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getId(): string
|
public function getId(): string
|
||||||
{
|
{
|
||||||
return 'authentik';
|
return 'authentik';
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getProviderClass(): string
|
public function getSocialiteProvider(): string
|
||||||
{
|
{
|
||||||
return Provider::class;
|
return Provider::class;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getServiceConfig(): array
|
public function getServiceConfig(): array
|
||||||
{
|
{
|
||||||
return [
|
return array_merge(parent::getServiceConfig(), [
|
||||||
'base_url' => env('OAUTH_AUTHENTIK_BASE_URL'),
|
'base_url' => env('OAUTH_AUTHENTIK_BASE_URL'),
|
||||||
'client_id' => env('OAUTH_AUTHENTIK_CLIENT_ID'),
|
]);
|
||||||
'client_secret' => env('OAUTH_AUTHENTIK_CLIENT_SECRET'),
|
}
|
||||||
];
|
|
||||||
|
public function getSetupSteps(): array
|
||||||
|
{
|
||||||
|
return array_merge([
|
||||||
|
Step::make('Create Authentik Application')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('create_application')
|
||||||
|
->hiddenLabel()
|
||||||
|
->state(new HtmlString(Blade::render('<p>On your Authentik dashboard select <b>Applications</b>, then select <b>Create with Provider</b>.</p><p>On the creation step select <b>OAuth2/OpenID Provider</b> and on the configure step set <b>Redirect URIs/Origins</b> to the value below.</p>'))),
|
||||||
|
TextInput::make('_noenv_callback')
|
||||||
|
->label('Callback URL')
|
||||||
|
->dehydrated()
|
||||||
|
->disabled()
|
||||||
|
->hintCopy()
|
||||||
|
->default(fn () => url('/auth/oauth/callback/authentik')),
|
||||||
|
]),
|
||||||
|
], parent::getSetupSteps());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getSettingsForm(): array
|
public function getSettingsForm(): array
|
||||||
@ -66,9 +80,4 @@ final class AuthentikProvider extends OAuthProvider
|
|||||||
{
|
{
|
||||||
return env('OAUTH_AUTHENTIK_DISPLAY_COLOR', '#fd4b2d');
|
return env('OAUTH_AUTHENTIK_DISPLAY_COLOR', '#fd4b2d');
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function register(Application $app): self
|
|
||||||
{
|
|
||||||
return new self($app);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
45
app/Extensions/OAuth/Schemas/BitbucketSchema.php
Normal file
45
app/Extensions/OAuth/Schemas/BitbucketSchema.php
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\OAuth\Schemas;
|
||||||
|
|
||||||
|
use Filament\Forms\Components\TextInput;
|
||||||
|
use Filament\Infolists\Components\TextEntry;
|
||||||
|
use Filament\Schemas\Components\Wizard\Step;
|
||||||
|
use Illuminate\Support\Facades\Blade;
|
||||||
|
use Illuminate\Support\HtmlString;
|
||||||
|
|
||||||
|
final class BitbucketSchema extends OAuthSchema
|
||||||
|
{
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return 'bitbucket';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getSetupSteps(): array
|
||||||
|
{
|
||||||
|
return array_merge([
|
||||||
|
Step::make('Register new Bitbucket Consumer')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('create_application')
|
||||||
|
->hiddenLabel()
|
||||||
|
->state(new HtmlString(Blade::render('<p>Visit the <x-filament::link href="https://support.atlassian.com/bitbucket-cloud/docs/use-oauth-on-bitbucket-cloud" target="_blank">Bitbucket OAuth Documentation</x-filament::link> and follow the steps in <b>Create a consumer</b>.</p><p>For the <b>Callback URL</b> use the value below.</p>'))),
|
||||||
|
TextInput::make('_noenv_callback')
|
||||||
|
->label('Callback URL')
|
||||||
|
->dehydrated()
|
||||||
|
->disabled()
|
||||||
|
->hintCopy()
|
||||||
|
->default(fn () => url('/auth/oauth/callback/bitbucket')),
|
||||||
|
]),
|
||||||
|
], parent::getSetupSteps());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getIcon(): string
|
||||||
|
{
|
||||||
|
return 'tabler-brand-bitbucket-f';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getHexColor(): string
|
||||||
|
{
|
||||||
|
return '#205081';
|
||||||
|
}
|
||||||
|
}
|
||||||
39
app/Extensions/OAuth/Schemas/CommonSchema.php
Normal file
39
app/Extensions/OAuth/Schemas/CommonSchema.php
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\OAuth\Schemas;
|
||||||
|
|
||||||
|
final class CommonSchema extends OAuthSchema
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
private readonly string $id,
|
||||||
|
private readonly ?string $name = null,
|
||||||
|
private readonly ?string $configName = null,
|
||||||
|
private readonly ?string $icon = null,
|
||||||
|
private readonly ?string $hexColor = null,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return $this->id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getName(): string
|
||||||
|
{
|
||||||
|
return $this->name ?? parent::getName();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getConfigKey(): string
|
||||||
|
{
|
||||||
|
return $this->configName ?? parent::getConfigKey();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getIcon(): ?string
|
||||||
|
{
|
||||||
|
return $this->icon;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getHexColor(): ?string
|
||||||
|
{
|
||||||
|
return $this->hexColor;
|
||||||
|
}
|
||||||
|
}
|
||||||
54
app/Extensions/OAuth/Schemas/DiscordSchema.php
Normal file
54
app/Extensions/OAuth/Schemas/DiscordSchema.php
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\OAuth\Schemas;
|
||||||
|
|
||||||
|
use Filament\Forms\Components\TextInput;
|
||||||
|
use Filament\Infolists\Components\TextEntry;
|
||||||
|
use Filament\Schemas\Components\Wizard\Step;
|
||||||
|
use Illuminate\Support\Facades\Blade;
|
||||||
|
use Illuminate\Support\HtmlString;
|
||||||
|
use SocialiteProviders\Discord\Provider;
|
||||||
|
|
||||||
|
final class DiscordSchema extends OAuthSchema
|
||||||
|
{
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return 'discord';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getSocialiteProvider(): string
|
||||||
|
{
|
||||||
|
return Provider::class;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getSetupSteps(): array
|
||||||
|
{
|
||||||
|
return array_merge([
|
||||||
|
Step::make('Register new Discord OAuth App')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('create_application')
|
||||||
|
->hiddenLabel()
|
||||||
|
->state(new HtmlString(Blade::render('<p>Visit the <x-filament::link href="https://discord.com/developers/applications" target="_blank">Discord Developer Portal</x-filament::link> and click on <b>New Application</b>. Enter a <b>Name</b> (e.g. your panel name) and click on <b>Create</b>.</p><p>Copy the <b>Client ID</b> and the <b>Client Secret</b> from the OAuth2 tab, you will need them in the final step.</p>'))),
|
||||||
|
TextEntry::make('set_redirect')
|
||||||
|
->hiddenLabel()
|
||||||
|
->state(new HtmlString('<p>Under <b>Redirects</b> add the below URL.</p>')),
|
||||||
|
TextInput::make('_noenv_callback')
|
||||||
|
->label('Redirect URL')
|
||||||
|
->dehydrated()
|
||||||
|
->disabled()
|
||||||
|
->hintCopy()
|
||||||
|
->formatStateUsing(fn () => url('/auth/oauth/callback/discord')),
|
||||||
|
]),
|
||||||
|
], parent::getSetupSteps());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getIcon(): string
|
||||||
|
{
|
||||||
|
return 'tabler-brand-discord-f';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getHexColor(): string
|
||||||
|
{
|
||||||
|
return '#5865F2';
|
||||||
|
}
|
||||||
|
}
|
||||||
48
app/Extensions/OAuth/Schemas/FacebookSchema.php
Normal file
48
app/Extensions/OAuth/Schemas/FacebookSchema.php
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\OAuth\Schemas;
|
||||||
|
|
||||||
|
use Filament\Forms\Components\TextInput;
|
||||||
|
use Filament\Infolists\Components\TextEntry;
|
||||||
|
use Filament\Schemas\Components\Wizard\Step;
|
||||||
|
use Illuminate\Support\Facades\Blade;
|
||||||
|
use Illuminate\Support\HtmlString;
|
||||||
|
|
||||||
|
final class FacebookSchema extends OAuthSchema
|
||||||
|
{
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return 'facebook';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getSetupSteps(): array
|
||||||
|
{
|
||||||
|
return array_merge([
|
||||||
|
Step::make('Register new Facebook Application')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('create_application')
|
||||||
|
->hiddenLabel()
|
||||||
|
->state(new HtmlString(Blade::render('<p>Visit the <x-filament::link href="https://developers.facebook.com/apps" target="_blank">Facebook Developer Dashboard</x-filament::link> and select or create a new app you will use for authentication. Make sure to have "Authenticate and request data from users with Facebook Login" as one of the Use Cases.</p><p>Once selected go to <b>Use Cases</b> and customize "Authenticate and request data from users with Facebook Login", from there go to <b>Settings</b> and add <b>Valid OAuth Redirect URIs</b> using the value below.</p>'))),
|
||||||
|
TextInput::make('_noenv_callback')
|
||||||
|
->label('Valid OAuth Redirect URIs')
|
||||||
|
->dehydrated()
|
||||||
|
->disabled()
|
||||||
|
->hintCopy()
|
||||||
|
->default(fn () => url('/auth/oauth/callback/facebook')),
|
||||||
|
TextEntry::make('get_app_info')
|
||||||
|
->hiddenLabel()
|
||||||
|
->state(new HtmlString(Blade::render('<p>To obtain the OAuth values go to <b>App Settings > Basic</b>.</p>'))),
|
||||||
|
]),
|
||||||
|
], parent::getSetupSteps());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getIcon(): string
|
||||||
|
{
|
||||||
|
return 'tabler-brand-facebook-f';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getHexColor(): string
|
||||||
|
{
|
||||||
|
return '#1877f2';
|
||||||
|
}
|
||||||
|
}
|
||||||
54
app/Extensions/OAuth/Schemas/GithubSchema.php
Normal file
54
app/Extensions/OAuth/Schemas/GithubSchema.php
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\OAuth\Schemas;
|
||||||
|
|
||||||
|
use Filament\Forms\Components\TextInput;
|
||||||
|
use Filament\Infolists\Components\TextEntry;
|
||||||
|
use Filament\Schemas\Components\Wizard\Step;
|
||||||
|
use Illuminate\Support\Facades\Blade;
|
||||||
|
use Illuminate\Support\HtmlString;
|
||||||
|
|
||||||
|
final class GithubSchema extends OAuthSchema
|
||||||
|
{
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return 'github';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getSetupSteps(): array
|
||||||
|
{
|
||||||
|
return array_merge([
|
||||||
|
Step::make('Register new Github OAuth App')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('create_application')
|
||||||
|
->hiddenLabel()
|
||||||
|
->state(new HtmlString(Blade::render('<p>Visit the <x-filament::link href="https://github.com/settings/developers" target="_blank">Github Developer Dashboard</x-filament::link>, go to <b>OAuth Apps</b> and click on <b>New OAuth App</b>.</p><p>Enter an <b>Application name</b> (e.g. your panel name), set <b>Homepage URL</b> to your panel url and enter the below url as <b>Authorization callback URL</b>.</p>'))),
|
||||||
|
TextInput::make('_noenv_callback')
|
||||||
|
->label('Authorization callback URL')
|
||||||
|
->dehydrated()
|
||||||
|
->disabled()
|
||||||
|
->hintCopy()
|
||||||
|
->default(fn () => url('/auth/oauth/callback/github')),
|
||||||
|
TextEntry::make('register_application')
|
||||||
|
->hiddenLabel()
|
||||||
|
->state(new HtmlString('<p>When you filled all fields click on <b>Register application</b>.</p>')),
|
||||||
|
]),
|
||||||
|
Step::make('Create Client Secret')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('create_client_secret')
|
||||||
|
->hiddenLabel()
|
||||||
|
->state(new HtmlString('<p>Once you registered your app, generate a new <b>Client Secret</b>.</p><p>You will also need the <b>Client ID</b>.</p>')),
|
||||||
|
]),
|
||||||
|
], parent::getSetupSteps());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getIcon(): string
|
||||||
|
{
|
||||||
|
return 'tabler-brand-github-f';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getHexColor(): string
|
||||||
|
{
|
||||||
|
return '#4078c0';
|
||||||
|
}
|
||||||
|
}
|
||||||
65
app/Extensions/OAuth/Schemas/GitlabSchema.php
Normal file
65
app/Extensions/OAuth/Schemas/GitlabSchema.php
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\OAuth\Schemas;
|
||||||
|
|
||||||
|
use Filament\Forms\Components\TextInput;
|
||||||
|
use Filament\Infolists\Components\TextEntry;
|
||||||
|
use Filament\Schemas\Components\Wizard\Step;
|
||||||
|
use Illuminate\Support\Facades\Blade;
|
||||||
|
use Illuminate\Support\HtmlString;
|
||||||
|
|
||||||
|
final class GitlabSchema extends OAuthSchema
|
||||||
|
{
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return 'gitlab';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getServiceConfig(): array
|
||||||
|
{
|
||||||
|
return array_merge(parent::getServiceConfig(), [
|
||||||
|
'host' => env('OAUTH_GITLAB_HOST'),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getSettingsForm(): array
|
||||||
|
{
|
||||||
|
return array_merge(parent::getSettingsForm(), [
|
||||||
|
TextInput::make('OAUTH_GITLAB_HOST')
|
||||||
|
->label('Custom Host')
|
||||||
|
->placeholder('Only set a custom host if you are self hosting gitlab')
|
||||||
|
->columnSpan(2)
|
||||||
|
->url()
|
||||||
|
->autocomplete(false)
|
||||||
|
->default(env('OAUTH_GITLAB_HOST')),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getSetupSteps(): array
|
||||||
|
{
|
||||||
|
return array_merge([
|
||||||
|
Step::make('Register new Gitlab OAuth App')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('register_application')
|
||||||
|
->hiddenLabel()
|
||||||
|
->state(new HtmlString(Blade::render('Check out the <x-filament::link href="https://docs.gitlab.com/integration/oauth_provider/" target="_blank">Gitlab docs</x-filament::link> on how to create the oauth app.'))),
|
||||||
|
TextInput::make('_noenv_callback')
|
||||||
|
->label('Redirect URI')
|
||||||
|
->dehydrated()
|
||||||
|
->disabled()
|
||||||
|
->hintCopy()
|
||||||
|
->default(fn () => url('/auth/oauth/callback/gitlab')),
|
||||||
|
]),
|
||||||
|
], parent::getSetupSteps());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getIcon(): string
|
||||||
|
{
|
||||||
|
return 'tabler-brand-gitlab';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getHexColor(): string
|
||||||
|
{
|
||||||
|
return '#fca326';
|
||||||
|
}
|
||||||
|
}
|
||||||
54
app/Extensions/OAuth/Schemas/GoogleSchema.php
Normal file
54
app/Extensions/OAuth/Schemas/GoogleSchema.php
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\OAuth\Schemas;
|
||||||
|
|
||||||
|
use Filament\Forms\Components\TextInput;
|
||||||
|
use Filament\Infolists\Components\TextEntry;
|
||||||
|
use Filament\Schemas\Components\Wizard\Step;
|
||||||
|
use Illuminate\Support\Facades\Blade;
|
||||||
|
use Illuminate\Support\HtmlString;
|
||||||
|
|
||||||
|
final class GoogleSchema extends OAuthSchema
|
||||||
|
{
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return 'google';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getSetupSteps(): array
|
||||||
|
{
|
||||||
|
return array_merge([
|
||||||
|
Step::make('Register new OAuth client')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('create_application')
|
||||||
|
->hiddenLabel()
|
||||||
|
->state(new HtmlString(Blade::render('<p>Visit the <x-filament::link href="https://console.developers.google.com/" target="_blank">Google API Console</x-filament::link> and create or select the project you want to use.</p><p>Navigate or search <b>Credentials</b>, click on the <b>Create Credentials</b> button and select <b>OAuth client ID</b>. On the Application type select <b>Web Application</b>.</p><p>On <b>Authorized JavaScript origins</b> and <b>Authorized redirect URIs</b> add and use the values below.</p>'))),
|
||||||
|
TextInput::make('_noenv_origin')
|
||||||
|
->label('Authorized JavaScript origins')
|
||||||
|
->dehydrated()
|
||||||
|
->disabled()
|
||||||
|
->hintCopy()
|
||||||
|
->default(fn () => url('')),
|
||||||
|
TextInput::make('_noenv_callback')
|
||||||
|
->label('Authorized redirect URIs')
|
||||||
|
->dehydrated()
|
||||||
|
->disabled()
|
||||||
|
->hintCopy()
|
||||||
|
->default(fn () => url('/auth/oauth/callback/google')),
|
||||||
|
TextEntry::make('register_application')
|
||||||
|
->hiddenLabel()
|
||||||
|
->state(new HtmlString('<p>When you filled all fields click on <b>Create</b>.</p>')),
|
||||||
|
]),
|
||||||
|
], parent::getSetupSteps());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getIcon(): string
|
||||||
|
{
|
||||||
|
return 'tabler-brand-google-f';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getHexColor(): string
|
||||||
|
{
|
||||||
|
return '#4285f4';
|
||||||
|
}
|
||||||
|
}
|
||||||
45
app/Extensions/OAuth/Schemas/LinkedinSchema.php
Normal file
45
app/Extensions/OAuth/Schemas/LinkedinSchema.php
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Extensions\OAuth\Schemas;
|
||||||
|
|
||||||
|
use Filament\Forms\Components\TextInput;
|
||||||
|
use Filament\Infolists\Components\TextEntry;
|
||||||
|
use Filament\Schemas\Components\Wizard\Step;
|
||||||
|
use Illuminate\Support\Facades\Blade;
|
||||||
|
use Illuminate\Support\HtmlString;
|
||||||
|
|
||||||
|
final class LinkedinSchema extends OAuthSchema
|
||||||
|
{
|
||||||
|
public function getId(): string
|
||||||
|
{
|
||||||
|
return 'linkedin';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getSetupSteps(): array
|
||||||
|
{
|
||||||
|
return array_merge([
|
||||||
|
Step::make('Obtain Linkedin App OAuth Config')
|
||||||
|
->schema([
|
||||||
|
TextEntry::make('create_application')
|
||||||
|
->hiddenLabel()
|
||||||
|
->state(new HtmlString(Blade::render('<p><x-filament::link href="https://www.linkedin.com/developers/apps/new" target="_blank">Create</x-filament::link> or <x-filament::link href="https://www.linkedin.com/developers/apps" target="_blank">select</x-filament::link> the one you will be using for authentication.</p><p>Select the <b>Auth</b> tab and set <b>Authorized redirect URLs for your app</b> to the value below.</p>'))),
|
||||||
|
TextInput::make('_noenv_callback')
|
||||||
|
->label('Authorized redirect URL')
|
||||||
|
->dehydrated()
|
||||||
|
->disabled()
|
||||||
|
->hintCopy()
|
||||||
|
->default(fn () => url('/auth/oauth/callback/linkedin')),
|
||||||
|
]),
|
||||||
|
], parent::getSetupSteps());
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getIcon(): string
|
||||||
|
{
|
||||||
|
return 'tabler-brand-linkedin-f';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getHexColor(): string
|
||||||
|
{
|
||||||
|
return '#0a66c2';
|
||||||
|
}
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user