prepare public symlink
This commit is contained in:
@@ -33,6 +33,21 @@
|
|||||||
state: directory
|
state: directory
|
||||||
mode: "0775"
|
mode: "0775"
|
||||||
|
|
||||||
|
- name: Migrate existing public/storage directory content before symlink
|
||||||
|
shell: |
|
||||||
|
set -e
|
||||||
|
cd "{{ prod_base_dir }}"
|
||||||
|
if [ -d public/storage ] && [ ! -L public/storage ]; then
|
||||||
|
if command -v rsync >/dev/null 2>&1; then
|
||||||
|
rsync -a public/storage/ storage/app/public/
|
||||||
|
else
|
||||||
|
cp -a public/storage/. storage/app/public/
|
||||||
|
fi
|
||||||
|
rm -rf public/storage
|
||||||
|
fi
|
||||||
|
args:
|
||||||
|
executable: /bin/bash
|
||||||
|
|
||||||
- name: Ensure public storage symlink exists
|
- name: Ensure public storage symlink exists
|
||||||
file:
|
file:
|
||||||
src: "{{ prod_base_dir }}/storage/app/public"
|
src: "{{ prod_base_dir }}/storage/app/public"
|
||||||
|
|||||||
@@ -63,10 +63,31 @@ class SystemStatusController extends Controller
|
|||||||
'rsync_version' => $this->resolveBinaryVersion($rsyncPath, ['--version']),
|
'rsync_version' => $this->resolveBinaryVersion($rsyncPath, ['--version']),
|
||||||
'proc_functions' => $procFunctionStatus,
|
'proc_functions' => $procFunctionStatus,
|
||||||
'storage_writable' => is_writable(storage_path()),
|
'storage_writable' => is_writable(storage_path()),
|
||||||
|
'storage_public_linked' => $this->isPublicStorageLinked(),
|
||||||
'updates_writable' => is_writable(storage_path('app/updates')) || @mkdir(storage_path('app/updates'), 0755, true),
|
'updates_writable' => is_writable(storage_path('app/updates')) || @mkdir(storage_path('app/updates'), 0755, true),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function isPublicStorageLinked(): bool
|
||||||
|
{
|
||||||
|
$publicStorage = public_path('storage');
|
||||||
|
$storagePublic = storage_path('app/public');
|
||||||
|
|
||||||
|
if (!is_link($publicStorage)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$target = readlink($publicStorage);
|
||||||
|
if ($target === false) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$resolvedTarget = realpath(dirname($publicStorage) . DIRECTORY_SEPARATOR . $target);
|
||||||
|
$expectedTarget = realpath($storagePublic);
|
||||||
|
|
||||||
|
return $resolvedTarget !== false && $expectedTarget !== false && $resolvedTarget === $expectedTarget;
|
||||||
|
}
|
||||||
|
|
||||||
private function resolveBinary(string $name): ?string
|
private function resolveBinary(string $name): ?string
|
||||||
{
|
{
|
||||||
$process = new Process(['sh', '-lc', "command -v {$name}"]);
|
$process = new Process(['sh', '-lc', "command -v {$name}"]);
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ use Illuminate\Http\Request;
|
|||||||
use Illuminate\Support\Facades\File;
|
use Illuminate\Support\Facades\File;
|
||||||
use Illuminate\Support\Facades\Http;
|
use Illuminate\Support\Facades\Http;
|
||||||
use Illuminate\Support\Str;
|
use Illuminate\Support\Str;
|
||||||
|
use RuntimeException;
|
||||||
use Symfony\Component\Process\Process;
|
use Symfony\Component\Process\Process;
|
||||||
|
|
||||||
class SystemUpdateController extends Controller
|
class SystemUpdateController extends Controller
|
||||||
@@ -113,7 +114,7 @@ class SystemUpdateController extends Controller
|
|||||||
$append('Syncing files...');
|
$append('Syncing files...');
|
||||||
$usedRsync = false;
|
$usedRsync = false;
|
||||||
$rsyncPath = trim((string) shell_exec('command -v rsync'));
|
$rsyncPath = trim((string) shell_exec('command -v rsync'));
|
||||||
$protectedPaths = ['custom', 'public/custom'];
|
$protectedPaths = ['storage', 'public/storage', 'custom', 'public/custom'];
|
||||||
if ($rsyncPath !== '') {
|
if ($rsyncPath !== '') {
|
||||||
$usedRsync = true;
|
$usedRsync = true;
|
||||||
$rsync = new Process([
|
$rsync = new Process([
|
||||||
@@ -149,6 +150,8 @@ class SystemUpdateController extends Controller
|
|||||||
File::copyDirectory($sourceDir, base_path());
|
File::copyDirectory($sourceDir, base_path());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$this->ensurePublicStorageLink();
|
||||||
|
|
||||||
$append('Installing composer dependencies...');
|
$append('Installing composer dependencies...');
|
||||||
$composer = new Process(['composer', 'install', '--no-dev', '--optimize-autoloader'], base_path());
|
$composer = new Process(['composer', 'install', '--no-dev', '--optimize-autoloader'], base_path());
|
||||||
$composer->setTimeout(600);
|
$composer->setTimeout(600);
|
||||||
@@ -212,4 +215,39 @@ class SystemUpdateController extends Controller
|
|||||||
], 500);
|
], 500);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function ensurePublicStorageLink(): void
|
||||||
|
{
|
||||||
|
$storagePublic = storage_path('app/public');
|
||||||
|
$publicStorage = public_path('storage');
|
||||||
|
|
||||||
|
if (file_exists($storagePublic) && !is_dir($storagePublic)) {
|
||||||
|
@rename($storagePublic, $storagePublic.'.bak.'.date('Ymd_His'));
|
||||||
|
}
|
||||||
|
if (!is_dir($storagePublic) && !@mkdir($storagePublic, 0775, true) && !is_dir($storagePublic)) {
|
||||||
|
throw new RuntimeException('Failed to prepare storage/app/public directory.');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (is_link($publicStorage)) {
|
||||||
|
$target = readlink($publicStorage);
|
||||||
|
$resolved = $target !== false ? realpath(dirname($publicStorage).DIRECTORY_SEPARATOR.$target) : false;
|
||||||
|
$expected = realpath($storagePublic);
|
||||||
|
if ($resolved !== $expected) {
|
||||||
|
@unlink($publicStorage);
|
||||||
|
}
|
||||||
|
} elseif (is_dir($publicStorage)) {
|
||||||
|
File::copyDirectory($publicStorage, $storagePublic);
|
||||||
|
File::deleteDirectory($publicStorage);
|
||||||
|
} elseif (file_exists($publicStorage)) {
|
||||||
|
@rename($publicStorage, $publicStorage.'.bak.'.date('Ymd_His'));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!is_link($publicStorage) && !@symlink($storagePublic, $publicStorage)) {
|
||||||
|
throw new RuntimeException('Failed to recreate public/storage symlink.');
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (['avatars', 'logos', 'favicons', 'rank-badges'] as $dir) {
|
||||||
|
File::ensureDirectoryExists($storagePublic.DIRECTORY_SEPARATOR.$dir);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,7 +4,9 @@ namespace App\Http\Controllers;
|
|||||||
|
|
||||||
use Illuminate\Http\JsonResponse;
|
use Illuminate\Http\JsonResponse;
|
||||||
use Illuminate\Http\Request;
|
use Illuminate\Http\Request;
|
||||||
|
use Illuminate\Support\Facades\File;
|
||||||
use Illuminate\Support\Facades\Storage;
|
use Illuminate\Support\Facades\Storage;
|
||||||
|
use RuntimeException;
|
||||||
|
|
||||||
class UploadController extends Controller
|
class UploadController extends Controller
|
||||||
{
|
{
|
||||||
@@ -14,6 +16,7 @@ class UploadController extends Controller
|
|||||||
if (!$user) {
|
if (!$user) {
|
||||||
return response()->json(['message' => 'Unauthorized'], 401);
|
return response()->json(['message' => 'Unauthorized'], 401);
|
||||||
}
|
}
|
||||||
|
$this->ensurePublicStorageReady();
|
||||||
|
|
||||||
$data = $request->validate([
|
$data = $request->validate([
|
||||||
'file' => [
|
'file' => [
|
||||||
@@ -45,6 +48,7 @@ class UploadController extends Controller
|
|||||||
if (!$user || !$user->roles()->where('name', 'ROLE_ADMIN')->exists()) {
|
if (!$user || !$user->roles()->where('name', 'ROLE_ADMIN')->exists()) {
|
||||||
return response()->json(['message' => 'Forbidden'], 403);
|
return response()->json(['message' => 'Forbidden'], 403);
|
||||||
}
|
}
|
||||||
|
$this->ensurePublicStorageReady();
|
||||||
|
|
||||||
$data = $request->validate([
|
$data = $request->validate([
|
||||||
'file' => ['required', 'file', 'mimes:jpg,jpeg,png,gif,webp,svg,ico', 'max:5120'],
|
'file' => ['required', 'file', 'mimes:jpg,jpeg,png,gif,webp,svg,ico', 'max:5120'],
|
||||||
@@ -64,6 +68,7 @@ class UploadController extends Controller
|
|||||||
if (!$user || !$user->roles()->where('name', 'ROLE_ADMIN')->exists()) {
|
if (!$user || !$user->roles()->where('name', 'ROLE_ADMIN')->exists()) {
|
||||||
return response()->json(['message' => 'Forbidden'], 403);
|
return response()->json(['message' => 'Forbidden'], 403);
|
||||||
}
|
}
|
||||||
|
$this->ensurePublicStorageReady();
|
||||||
|
|
||||||
$data = $request->validate([
|
$data = $request->validate([
|
||||||
'file' => ['required', 'file', 'mimes:png,ico', 'max:2048'],
|
'file' => ['required', 'file', 'mimes:png,ico', 'max:2048'],
|
||||||
@@ -76,4 +81,49 @@ class UploadController extends Controller
|
|||||||
'url' => Storage::url($path),
|
'url' => Storage::url($path),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function ensurePublicStorageReady(): void
|
||||||
|
{
|
||||||
|
$storagePublic = storage_path('app/public');
|
||||||
|
$publicStorage = public_path('storage');
|
||||||
|
|
||||||
|
if (file_exists($storagePublic) && !is_dir($storagePublic)) {
|
||||||
|
@rename($storagePublic, $storagePublic.'.bak.'.date('Ymd_His'));
|
||||||
|
}
|
||||||
|
if (!is_dir($storagePublic) && !@mkdir($storagePublic, 0775, true) && !is_dir($storagePublic)) {
|
||||||
|
throw new RuntimeException('Failed to create storage/app/public directory.');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (is_link($publicStorage)) {
|
||||||
|
$target = readlink($publicStorage);
|
||||||
|
$resolved = $target !== false ? realpath(dirname($publicStorage).DIRECTORY_SEPARATOR.$target) : false;
|
||||||
|
$expected = realpath($storagePublic);
|
||||||
|
if ($resolved === $expected) {
|
||||||
|
$this->ensureUploadSubdirs($storagePublic);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
@unlink($publicStorage);
|
||||||
|
} elseif (is_dir($publicStorage)) {
|
||||||
|
File::copyDirectory($publicStorage, $storagePublic);
|
||||||
|
File::deleteDirectory($publicStorage);
|
||||||
|
} elseif (file_exists($publicStorage)) {
|
||||||
|
@rename($publicStorage, $publicStorage.'.bak.'.date('Ymd_His'));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!@symlink($storagePublic, $publicStorage) && !is_link($publicStorage)) {
|
||||||
|
throw new RuntimeException('Failed to create public/storage symlink.');
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->ensureUploadSubdirs($storagePublic);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function ensureUploadSubdirs(string $storagePublic): void
|
||||||
|
{
|
||||||
|
foreach (['avatars', 'favicons', 'logos', 'rank-badges'] as $dir) {
|
||||||
|
$path = $storagePublic.DIRECTORY_SEPARATOR.$dir;
|
||||||
|
if (!is_dir($path)) {
|
||||||
|
@mkdir($path, 0775, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -98,5 +98,5 @@
|
|||||||
"minimum-stability": "stable",
|
"minimum-stability": "stable",
|
||||||
"prefer-stable": true,
|
"prefer-stable": true,
|
||||||
"version": "26.0.3",
|
"version": "26.0.3",
|
||||||
"build": "103"
|
"build": "104"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,6 +20,39 @@ resolve_php_bin() {
|
|||||||
echo "php"
|
echo "php"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ensure_storage_link() {
|
||||||
|
local storage_public="storage/app/public"
|
||||||
|
local public_storage="public/storage"
|
||||||
|
|
||||||
|
echo "Ensuring public storage link..."
|
||||||
|
|
||||||
|
if [[ -e "$storage_public" && ! -d "$storage_public" ]]; then
|
||||||
|
local backup_path="${storage_public}.bak.$(date +%Y%m%d_%H%M%S)"
|
||||||
|
echo "Found invalid $storage_public (not a directory). Backing up to $backup_path"
|
||||||
|
mv "$storage_public" "$backup_path"
|
||||||
|
fi
|
||||||
|
|
||||||
|
mkdir -p "$storage_public"
|
||||||
|
|
||||||
|
# If public/storage is a real directory, migrate files before converting to symlink.
|
||||||
|
if [[ -d "$public_storage" && ! -L "$public_storage" ]]; then
|
||||||
|
echo "Migrating existing files from $public_storage to $storage_public"
|
||||||
|
if command -v rsync >/dev/null 2>&1; then
|
||||||
|
rsync -a "$public_storage"/ "$storage_public"/
|
||||||
|
else
|
||||||
|
cp -a "$public_storage"/. "$storage_public"/
|
||||||
|
fi
|
||||||
|
rm -rf "$public_storage"
|
||||||
|
elif [[ -e "$public_storage" && ! -L "$public_storage" ]]; then
|
||||||
|
local public_backup="${public_storage}.bak.$(date +%Y%m%d_%H%M%S)"
|
||||||
|
echo "Found invalid $public_storage (not a directory/symlink). Backing up to $public_backup"
|
||||||
|
mv "$public_storage" "$public_backup"
|
||||||
|
fi
|
||||||
|
|
||||||
|
ln -sfn ../storage/app/public "$public_storage"
|
||||||
|
mkdir -p "$storage_public/logos" "$storage_public/favicons" "$storage_public/rank-badges"
|
||||||
|
}
|
||||||
|
|
||||||
resolve_configured_php_bin() {
|
resolve_configured_php_bin() {
|
||||||
local configured="${1:-}"
|
local configured="${1:-}"
|
||||||
local current="${2:-php}"
|
local current="${2:-php}"
|
||||||
@@ -248,6 +281,8 @@ main() {
|
|||||||
echo "Running with PHP binary: $PHP_BIN artisan migrate --force"
|
echo "Running with PHP binary: $PHP_BIN artisan migrate --force"
|
||||||
"$PHP_BIN" artisan migrate --force
|
"$PHP_BIN" artisan migrate --force
|
||||||
|
|
||||||
|
ensure_storage_link
|
||||||
|
|
||||||
echo "Syncing version/build to settings..."
|
echo "Syncing version/build to settings..."
|
||||||
echo "Running with PHP binary: $PHP_BIN -r <read composer.json version>"
|
echo "Running with PHP binary: $PHP_BIN -r <read composer.json version>"
|
||||||
VERSION="$("$PHP_BIN" -r '$c=json_decode(file_get_contents("composer.json"), true); echo $c["version"] ?? "";')"
|
VERSION="$("$PHP_BIN" -r '$c=json_decode(file_get_contents("composer.json"), true); echo $c["version"] ?? "";')"
|
||||||
|
|||||||
@@ -130,12 +130,7 @@ export async function fetchVersion() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export async function fetchPing() {
|
export async function fetchPing() {
|
||||||
const response = await fetch('/ping', {
|
return apiFetch('/ping')
|
||||||
headers: {
|
|
||||||
Accept: 'application/json',
|
|
||||||
},
|
|
||||||
})
|
|
||||||
return parseResponse(response)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function fetchVersionCheck() {
|
export async function fetchVersionCheck() {
|
||||||
|
|||||||
@@ -563,6 +563,14 @@ function Acp({ isAdmin }) {
|
|||||||
current: '—',
|
current: '—',
|
||||||
status: systemStatus.storage_writable ? 'ok' : 'bad',
|
status: systemStatus.storage_writable ? 'ok' : 'bad',
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
id: 'storage_link',
|
||||||
|
label: t('system.storage_linked'),
|
||||||
|
path: 'public/storage -> storage/app/public',
|
||||||
|
min: '—',
|
||||||
|
current: '—',
|
||||||
|
status: systemStatus.storage_public_linked ? 'ok' : 'bad',
|
||||||
|
},
|
||||||
{
|
{
|
||||||
id: 'updates',
|
id: 'updates',
|
||||||
label: t('system.updates_writable'),
|
label: t('system.updates_writable'),
|
||||||
@@ -576,9 +584,9 @@ function Acp({ isAdmin }) {
|
|||||||
|
|
||||||
const visibleSystemChecks = useMemo(() => {
|
const visibleSystemChecks = useMemo(() => {
|
||||||
const visibilityBySection = {
|
const visibilityBySection = {
|
||||||
insite: ['php', 'proc', 'storage', 'updates'],
|
insite: ['php', 'proc', 'storage', 'storage_link', 'updates'],
|
||||||
cli: ['php', 'composer', 'node', 'npm', 'proc', 'storage', 'updates'],
|
cli: ['php', 'composer', 'node', 'npm', 'proc', 'storage', 'storage_link'],
|
||||||
ci: ['php', 'composer', 'node', 'npm', 'tar', 'rsync', 'proc', 'storage', 'updates'],
|
ci: ['php', 'composer', 'node', 'npm', 'tar', 'rsync', 'proc', 'storage', 'storage_link', 'updates'],
|
||||||
info: [],
|
info: [],
|
||||||
}
|
}
|
||||||
const allowed = new Set(visibilityBySection[systemSection] || [])
|
const allowed = new Set(visibilityBySection[systemSection] || [])
|
||||||
@@ -3861,14 +3869,15 @@ function Acp({ isAdmin }) {
|
|||||||
<p className="bb-muted mb-0 mt-1">
|
<p className="bb-muted mb-0 mt-1">
|
||||||
CLI default php: {systemStatus?.php_default || '—'} (
|
CLI default php: {systemStatus?.php_default || '—'} (
|
||||||
{systemStatus?.php_default_version || 'unknown'}){' '}
|
{systemStatus?.php_default_version || 'unknown'}){' '}
|
||||||
{cliDefaultPhpIsSufficient ? (
|
{phpSelectedIsSufficient ? (
|
||||||
<i className="bi bi-check-circle-fill text-success" aria-hidden="true" />
|
<i className="bi bi-check-circle-fill text-success" aria-hidden="true" />
|
||||||
) : (
|
) : (
|
||||||
<OverlayTrigger
|
<OverlayTrigger
|
||||||
placement="top"
|
placement="top"
|
||||||
overlay={
|
overlay={
|
||||||
<Tooltip id="cli-default-php-warning" data-bs-theme="light">
|
<Tooltip id="cli-default-php-warning" data-bs-theme="light">
|
||||||
You must select a custom PHP interpreter, as the system default is not sufficient.
|
The selected PHP interpreter is not sufficient for the required
|
||||||
|
composer.json PHP version.
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
@@ -3916,7 +3925,7 @@ function Acp({ isAdmin }) {
|
|||||||
<Form.Text className="bb-muted">
|
<Form.Text className="bb-muted">
|
||||||
Minimum required PHP (from composer.json):{' '}
|
Minimum required PHP (from composer.json):{' '}
|
||||||
{systemStatus?.min_versions?.php || 'unknown'}. Use a custom binary
|
{systemStatus?.min_versions?.php || 'unknown'}. Use a custom binary
|
||||||
on like php84. On KeyHelp setups use e.g. `keyhelp-php84`.
|
like php84. On KeyHelp setups use e.g. `keyhelp-php84`.
|
||||||
</Form.Text>
|
</Form.Text>
|
||||||
</Form.Group>
|
</Form.Group>
|
||||||
<Button type="submit" variant="dark" disabled={systemCliSaving}>
|
<Button type="submit" variant="dark" disabled={systemCliSaving}>
|
||||||
|
|||||||
@@ -204,6 +204,7 @@
|
|||||||
"system.none": "Keine",
|
"system.none": "Keine",
|
||||||
"system.not_found": "Nicht gefunden",
|
"system.not_found": "Nicht gefunden",
|
||||||
"system.storage_writable": "Storage beschreibbar",
|
"system.storage_writable": "Storage beschreibbar",
|
||||||
|
"system.storage_linked": "Storage Public-Link",
|
||||||
"system.updates_writable": "Updates beschreibbar",
|
"system.updates_writable": "Updates beschreibbar",
|
||||||
"system.ok": "OK",
|
"system.ok": "OK",
|
||||||
"system.not_ok": "Nicht OK",
|
"system.not_ok": "Nicht OK",
|
||||||
|
|||||||
@@ -194,6 +194,7 @@
|
|||||||
"system.none": "None",
|
"system.none": "None",
|
||||||
"system.not_found": "Not found",
|
"system.not_found": "Not found",
|
||||||
"system.storage_writable": "Storage writable",
|
"system.storage_writable": "Storage writable",
|
||||||
|
"system.storage_linked": "Storage public link",
|
||||||
"system.updates_writable": "Updates writable",
|
"system.updates_writable": "Updates writable",
|
||||||
"system.ok": "OK",
|
"system.ok": "OK",
|
||||||
"system.not_ok": "Not OK",
|
"system.not_ok": "Not OK",
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ use App\Http\Controllers\AuthController;
|
|||||||
use App\Http\Controllers\ForumController;
|
use App\Http\Controllers\ForumController;
|
||||||
use App\Http\Controllers\I18nController;
|
use App\Http\Controllers\I18nController;
|
||||||
use App\Http\Controllers\PortalController;
|
use App\Http\Controllers\PortalController;
|
||||||
|
use App\Http\Controllers\PingController;
|
||||||
use App\Http\Controllers\PostController;
|
use App\Http\Controllers\PostController;
|
||||||
use App\Http\Controllers\PostThankController;
|
use App\Http\Controllers\PostThankController;
|
||||||
use App\Http\Controllers\PreviewController;
|
use App\Http\Controllers\PreviewController;
|
||||||
@@ -35,6 +36,7 @@ Route::get('/email/verify/{id}/{hash}', [AuthController::class, 'verifyEmail'])
|
|||||||
Route::post('/logout', [AuthController::class, 'logout'])->middleware('auth:sanctum');
|
Route::post('/logout', [AuthController::class, 'logout'])->middleware('auth:sanctum');
|
||||||
Route::post('/user/password', [AuthController::class, 'updatePassword'])->middleware('auth:sanctum');
|
Route::post('/user/password', [AuthController::class, 'updatePassword'])->middleware('auth:sanctum');
|
||||||
|
|
||||||
|
Route::get('/ping', PingController::class);
|
||||||
Route::get('/version', VersionController::class);
|
Route::get('/version', VersionController::class);
|
||||||
Route::get('/version/check', VersionCheckController::class);
|
Route::get('/version/check', VersionCheckController::class);
|
||||||
Route::post('/system/update', SystemUpdateController::class)->middleware('auth:sanctum');
|
Route::post('/system/update', SystemUpdateController::class)->middleware('auth:sanctum');
|
||||||
|
|||||||
@@ -86,6 +86,7 @@ it('returns system status for admins', function (): void {
|
|||||||
'rsync_version',
|
'rsync_version',
|
||||||
'proc_functions',
|
'proc_functions',
|
||||||
'storage_writable',
|
'storage_writable',
|
||||||
|
'storage_public_linked',
|
||||||
'updates_writable',
|
'updates_writable',
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|||||||
Reference in New Issue
Block a user