Add QA test outputs, build scripts, and Dockerfile validation

- Created `qa-test-output-after-fix.txt` and `qa-test-output.txt` to log results of certificate page authentication tests.
- Added `build.sh` for deterministic backend builds in CI, utilizing `go list` for efficiency.
- Introduced `codeql_scan.sh` for CodeQL database creation and analysis for Go and JavaScript/TypeScript.
- Implemented `dockerfile_check.sh` to validate Dockerfiles for base image and package manager mismatches.
- Added `sourcery_precommit_wrapper.sh` to facilitate Sourcery CLI usage in pre-commit hooks.
This commit is contained in:
GitHub Actions
2025-12-11 18:26:24 +00:00
parent 65d837a13f
commit 8294d6ee49
609 changed files with 111623 additions and 0 deletions

View File

@@ -0,0 +1,353 @@
import { useState, useEffect, useMemo } from 'react'
import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query'
import { Card } from '../components/ui/Card'
import { Button } from '../components/ui/Button'
import { Input } from '../components/ui/Input'
import { Switch } from '../components/ui/Switch'
import { toast } from '../utils/toast'
import { getSettings, updateSetting } from '../api/settings'
import { getFeatureFlags, updateFeatureFlags } from '../api/featureFlags'
import client from '../api/client'
// CrowdSec runtime control is now in the Security page
import { Loader2, Server, RefreshCw, Save, Activity } from 'lucide-react'
import { ConfigReloadOverlay } from '../components/LoadingStates'
interface HealthResponse {
status: string
service: string
version: string
git_commit: string
build_time: string
}
interface UpdateInfo {
current_version: string
latest_version: string
update_available: boolean
release_url?: string
}
export default function SystemSettings() {
const queryClient = useQueryClient()
const [caddyAdminAPI, setCaddyAdminAPI] = useState('http://localhost:2019')
const [sslProvider, setSslProvider] = useState('auto')
const [domainLinkBehavior, setDomainLinkBehavior] = useState('new_tab')
// Fetch Settings
const { data: settings } = useQuery({
queryKey: ['settings'],
queryFn: getSettings,
})
// Update local state when settings load
useEffect(() => {
if (settings) {
if (settings['caddy.admin_api']) setCaddyAdminAPI(settings['caddy.admin_api'])
// Default to 'auto' if empty or invalid value
if (settings['caddy.ssl_provider']) {
const validProviders = ['auto', 'letsencrypt-staging', 'letsencrypt-prod', 'zerossl']
const provider = settings['caddy.ssl_provider']
setSslProvider(validProviders.includes(provider) ? provider : 'auto')
}
if (settings['ui.domain_link_behavior']) setDomainLinkBehavior(settings['ui.domain_link_behavior'])
}
}, [settings])
// Fetch Health/System Status
const { data: health, isLoading: isLoadingHealth } = useQuery({
queryKey: ['health'],
queryFn: async (): Promise<HealthResponse> => {
const response = await client.get<HealthResponse>('/health')
return response.data
},
})
// Check for Updates
const {
data: updateInfo,
refetch: checkUpdates,
isFetching: isCheckingUpdates,
} = useQuery({
queryKey: ['updates'],
queryFn: async (): Promise<UpdateInfo> => {
const response = await client.get<UpdateInfo>('/system/updates')
return response.data
},
enabled: false, // Manual trigger
})
const saveSettingsMutation = useMutation({
mutationFn: async () => {
await updateSetting('caddy.admin_api', caddyAdminAPI, 'caddy', 'string')
await updateSetting('caddy.ssl_provider', sslProvider, 'caddy', 'string')
await updateSetting('ui.domain_link_behavior', domainLinkBehavior, 'ui', 'string')
},
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ['settings'] })
toast.success('System settings saved')
},
onError: (error: Error) => {
toast.error(`Failed to save settings: ${error.message}`)
},
})
// Feature Flags
const { data: featureFlags, refetch: refetchFlags } = useQuery({
queryKey: ['feature-flags'],
queryFn: getFeatureFlags,
})
const featureToggles = useMemo(
() => [
{
key: 'feature.cerberus.enabled',
label: 'Cerberus Security Suite',
tooltip: 'Advanced security features including WAF, Access Lists, Rate Limiting, and CrowdSec.',
},
{
key: 'feature.crowdsec.console_enrollment',
label: 'CrowdSec Console Enrollment',
tooltip: 'Allow enrolling this node with CrowdSec Console for centralized fleet management.',
},
{
key: 'feature.uptime.enabled',
label: 'Uptime Monitoring',
tooltip: 'Monitor the availability of your proxy hosts and remote servers.',
},
],
[]
)
const updateFlagMutation = useMutation({
mutationFn: async (payload: Record<string, boolean>) => updateFeatureFlags(payload),
onSuccess: () => {
refetchFlags()
toast.success('Feature flag updated')
},
onError: (err: unknown) => {
const msg = err instanceof Error ? err.message : String(err)
toast.error(`Failed to update flag: ${msg}`)
},
})
// CrowdSec control
// Determine loading message
const { message, submessage } = updateFlagMutation.isPending
? { message: 'Updating features...', submessage: 'Applying configuration changes' }
: { message: 'Loading...', submessage: 'Please wait' }
return (
<>
{updateFlagMutation.isPending && (
<ConfigReloadOverlay
message={message}
submessage={submessage}
type="charon"
/>
)}
<div className="space-y-6">
<h1 className="text-2xl font-bold text-gray-900 dark:text-white flex items-center gap-2">
<Server className="w-8 h-8" />
System Settings
</h1>
{/* Features */}
<Card className="p-6">
<h2 className="text-lg font-semibold mb-4 text-gray-900 dark:text-white">Features</h2>
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
{featureFlags ? (
featureToggles.map(({ key, label, tooltip }) => (
<div
key={key}
className="flex items-center justify-between p-4 bg-gray-50 dark:bg-gray-800/50 rounded-lg border border-gray-100 dark:border-gray-800"
title={tooltip}
>
<p className="text-sm font-medium text-gray-900 dark:text-white cursor-help">{label}</p>
<Switch
aria-label={`${label} toggle`}
checked={!!featureFlags[key]}
disabled={updateFlagMutation.isPending}
onChange={(e) => updateFlagMutation.mutate({ [key]: e.target.checked })}
/>
</div>
))
) : (
<p className="text-sm text-gray-500 col-span-2">Loading features...</p>
)}
</div>
</Card>
{/* General Configuration */}
<Card className="p-6">
<h2 className="text-lg font-semibold mb-4 text-gray-900 dark:text-white">General Configuration</h2>
<div className="space-y-4">
<Input
label="Caddy Admin API Endpoint"
type="text"
value={caddyAdminAPI}
onChange={(e) => setCaddyAdminAPI(e.target.value)}
placeholder="http://localhost:2019"
/>
<p className="text-sm text-gray-500 dark:text-gray-400 -mt-2">
URL to the Caddy admin API (usually on port 2019)
</p>
<div className="w-full">
<label className="block text-sm font-medium text-gray-300 mb-1.5">
SSL Provider
</label>
<select
value={sslProvider}
onChange={(e) => setSslProvider(e.target.value)}
className="w-full bg-gray-900 border border-gray-700 rounded-lg px-4 py-2 text-white focus:outline-none focus:ring-2 focus:ring-blue-500 focus:border-blue-500 transition-colors"
>
<option value="auto">Auto (Recommended)</option>
<option value="letsencrypt-prod">Let's Encrypt (Prod)</option>
<option value="letsencrypt-staging">Let's Encrypt (Staging)</option>
<option value="zerossl">ZeroSSL</option>
</select>
<p className="text-sm text-gray-500 dark:text-gray-400 mt-1">
Choose the Certificate Authority. 'Auto' uses Let's Encrypt with ZeroSSL fallback. Staging is for testing.
</p>
</div>
<div className="w-full">
<label className="block text-sm font-medium text-gray-300 mb-1.5">
Domain Link Behavior
</label>
<select
value={domainLinkBehavior}
onChange={(e) => setDomainLinkBehavior(e.target.value)}
className="w-full bg-gray-900 border border-gray-700 rounded-lg px-4 py-2 text-white focus:outline-none focus:ring-2 focus:ring-blue-500 focus:border-blue-500 transition-colors"
>
<option value="same_tab">Same Tab</option>
<option value="new_tab">New Tab (Default)</option>
<option value="new_window">New Window</option>
</select>
<p className="text-sm text-gray-500 dark:text-gray-400 mt-1">
Control how domain links open in the Proxy Hosts list.
</p>
</div>
<div className="flex justify-end">
<Button
onClick={() => saveSettingsMutation.mutate()}
isLoading={saveSettingsMutation.isPending}
>
<Save className="w-4 h-4 mr-2" />
Save Settings
</Button>
</div>
</div>
</Card>
{/* Optional Features - Removed (Moved to top) */}
{/* System Status */}
<Card className="p-6">
<h2 className="text-lg font-semibold mb-4 text-gray-900 dark:text-white flex items-center gap-2">
<Activity className="w-5 h-5" />
System Status
</h2>
{isLoadingHealth ? (
<div className="flex items-center justify-center py-8">
<Loader2 className="w-6 h-6 animate-spin text-blue-500" />
</div>
) : health ? (
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
<div>
<p className="text-sm text-gray-500 dark:text-gray-400">Service</p>
<p className="text-lg font-medium text-gray-900 dark:text-white">{health.service}</p>
</div>
<div>
<p className="text-sm text-gray-500 dark:text-gray-400">Status</p>
<p className="text-lg font-medium text-green-600 dark:text-green-400 capitalize">
{health.status}
</p>
</div>
<div>
<p className="text-sm text-gray-500 dark:text-gray-400">Version</p>
<p className="text-lg font-medium text-gray-900 dark:text-white">{health.version}</p>
</div>
<div>
<p className="text-sm text-gray-500 dark:text-gray-400">Build Time</p>
<p className="text-lg font-medium text-gray-900 dark:text-white">
{health.build_time || 'N/A'}
</p>
</div>
<div className="md:col-span-2">
<p className="text-sm text-gray-500 dark:text-gray-400">Git Commit</p>
<p className="text-sm font-mono text-gray-900 dark:text-white">
{health.git_commit || 'N/A'}
</p>
</div>
</div>
) : (
<p className="text-red-500">Unable to fetch system status</p>
)}
</Card>
{/* Update Check */}
<Card className="p-6">
<h2 className="text-lg font-semibold mb-4 text-gray-900 dark:text-white">Software Updates</h2>
<div className="space-y-4">
{updateInfo && (
<div className="grid grid-cols-1 md:grid-cols-2 gap-4 mb-4">
<div>
<p className="text-sm text-gray-500 dark:text-gray-400">Current Version</p>
<p className="text-lg font-medium text-gray-900 dark:text-white">
{updateInfo.current_version}
</p>
</div>
<div>
<p className="text-sm text-gray-500 dark:text-gray-400">Latest Version</p>
<p className="text-lg font-medium text-gray-900 dark:text-white">
{updateInfo.latest_version}
</p>
</div>
{updateInfo.update_available && (
<div className="md:col-span-2">
<div className="bg-blue-50 dark:bg-blue-900/20 border border-blue-200 dark:border-blue-800 rounded-lg p-4">
<p className="text-blue-800 dark:text-blue-300 font-medium">
A new version is available!
</p>
{updateInfo.release_url && (
<a
href={updateInfo.release_url}
target="_blank"
rel="noopener noreferrer"
className="text-blue-600 dark:text-blue-400 hover:underline text-sm"
>
View Release Notes
</a>
)}
</div>
</div>
)}
{!updateInfo.update_available && (
<div className="md:col-span-2">
<p className="text-green-600 dark:text-green-400">
You are running the latest version
</p>
</div>
)}
</div>
)}
<Button
onClick={() => checkUpdates()}
isLoading={isCheckingUpdates}
variant="secondary"
>
<RefreshCw className="w-4 h-4 mr-2" />
Check for Updates
</Button>
</div>
</Card>
</div>
</>
)
}