Enhance deployment system with retry functionality and improved UX

Major Improvements:
- Added retry deployment buttons in machine list for failed deployments
- Added retry button in SSH console modal footer for enhanced UX
- Enhanced deployment process with comprehensive cleanup of existing services
- Improved binary installation with password-based sudo authentication
- Updated configuration generation to include all required sections (agent, ai, network, security)
- Fixed deployment verification and error handling

Security Enhancements:
- Enhanced verifiedStopExistingServices with thorough cleanup process
- Improved binary copying with proper sudo authentication
- Added comprehensive configuration validation

UX Improvements:
- Users can retry deployments without re-running machine discovery
- Retry buttons available from both machine list and console modal
- Real-time deployment progress with detailed console output
- Clear error states with actionable retry options

Technical Changes:
- Modified ServiceDeployment.tsx with retry button components
- Enhanced api/setup_manager.go with improved deployment functions
- Updated main.go with command line argument support (--config, --setup)
- Added comprehensive zero-trust security validation system

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
anthonyrawlins
2025-08-31 10:23:27 +10:00
parent df4d98bf30
commit be761cfe20
234 changed files with 7508 additions and 38528 deletions

View File

@@ -94,7 +94,7 @@ export default function SystemDetection({
const getStatusColor = (condition: boolean) => {
return condition ? 'text-green-600' : 'text-red-600'
return condition ? 'text-eucalyptus-600' : 'text-red-600'
}
const getStatusIcon = (condition: boolean) => {
@@ -106,7 +106,7 @@ export default function SystemDetection({
<div className="flex items-center justify-center py-12">
<div className="text-center">
<ArrowPathIcon className="h-8 w-8 text-bzzz-primary animate-spin mx-auto mb-4" />
<p className="text-gray-600">Detecting system configuration...</p>
<p className="text-chorus-text-secondary">Detecting system configuration...</p>
</div>
</div>
)
@@ -116,10 +116,10 @@ export default function SystemDetection({
return (
<div className="text-center py-12">
<ExclamationTriangleIcon className="h-12 w-12 text-red-500 mx-auto mb-4" />
<h3 className="text-lg font-medium text-gray-900 mb-2">
<h3 className="heading-subsection mb-2">
System Detection Failed
</h3>
<p className="text-gray-600 mb-4">
<p className="text-chorus-text-secondary mb-4">
Unable to detect system configuration. Please try again.
</p>
<button
@@ -136,9 +136,9 @@ export default function SystemDetection({
return (
<div className="space-y-6">
{/* System Overview */}
<div className="bg-gray-50 rounded-lg p-6">
<div className="card">
<div className="flex items-center justify-between mb-4">
<h3 className="text-lg font-medium text-gray-900">System Overview</h3>
<h3 className="heading-subsection">System Overview</h3>
<button
onClick={refreshSystemInfo}
disabled={refreshing}
@@ -150,12 +150,12 @@ export default function SystemDetection({
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
<div>
<div className="text-sm font-medium text-gray-700">Hostname</div>
<div className="text-lg text-gray-900">{detectedInfo.network.hostname}</div>
<div className="text-sm font-medium text-chorus-text-secondary">Hostname</div>
<div className="text-lg text-chorus-text-primary">{detectedInfo.network.hostname}</div>
</div>
<div>
<div className="text-sm font-medium text-gray-700">Operating System</div>
<div className="text-lg text-gray-900">
<div className="text-sm font-medium text-chorus-text-secondary">Operating System</div>
<div className="text-lg text-chorus-text-primary">
{detectedInfo.os} ({detectedInfo.architecture})
</div>
</div>
@@ -165,22 +165,22 @@ export default function SystemDetection({
{/* Hardware Information */}
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
{/* CPU & Memory */}
<div className="bg-white border border-gray-200 rounded-lg p-6">
<div className="card">
<div className="flex items-center mb-4">
<CpuChipIcon className="h-6 w-6 text-bzzz-primary mr-2" />
<h3 className="text-lg font-medium text-gray-900">CPU & Memory</h3>
<h3 className="heading-subsection">CPU & Memory</h3>
</div>
<div className="space-y-3">
<div>
<div className="text-sm font-medium text-gray-700">CPU</div>
<div className="text-gray-900">
<div className="text-sm font-medium text-chorus-text-secondary">CPU</div>
<div className="text-chorus-text-primary">
{detectedInfo.cpu_cores} cores
</div>
</div>
<div>
<div className="text-sm font-medium text-gray-700">Memory</div>
<div className="text-gray-900">
<div className="text-sm font-medium text-chorus-text-secondary">Memory</div>
<div className="text-chorus-text-primary">
{Math.round(detectedInfo.memory_mb / 1024)} GB total
</div>
</div>
@@ -188,21 +188,21 @@ export default function SystemDetection({
</div>
{/* Storage */}
<div className="bg-white border border-gray-200 rounded-lg p-6">
<div className="card">
<div className="flex items-center mb-4">
<CircleStackIcon className="h-6 w-6 text-bzzz-primary mr-2" />
<h3 className="text-lg font-medium text-gray-900">Storage</h3>
<h3 className="heading-subsection">Storage</h3>
</div>
<div className="space-y-3">
<div>
<div className="text-sm font-medium text-gray-700">Disk Space</div>
<div className="text-gray-900">
<div className="text-sm font-medium text-chorus-text-secondary">Disk Space</div>
<div className="text-chorus-text-primary">
{detectedInfo.storage.total_space_gb} GB total, {' '}
{detectedInfo.storage.free_space_gb} GB available
</div>
</div>
<div className="w-full bg-gray-200 rounded-full h-2">
<div className="w-full bg-chorus-border-invisible rounded-full h-2">
<div
className="bg-bzzz-primary h-2 rounded-full"
style={{
@@ -216,19 +216,19 @@ export default function SystemDetection({
{/* GPU Information */}
{detectedInfo.gpus && detectedInfo.gpus.length > 0 && (
<div className="bg-white border border-gray-200 rounded-lg p-6">
<div className="card">
<div className="flex items-center mb-4">
<ServerIcon className="h-6 w-6 text-bzzz-primary mr-2" />
<h3 className="text-lg font-medium text-gray-900">
<h3 className="heading-subsection">
GPU Configuration ({detectedInfo.gpus.length} GPU{detectedInfo.gpus.length !== 1 ? 's' : ''})
</h3>
</div>
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
{detectedInfo.gpus.map((gpu, index) => (
<div key={index} className="bg-gray-50 rounded-lg p-4">
<div className="font-medium text-gray-900">{gpu.name}</div>
<div className="text-sm text-gray-600">
<div key={index} className="bg-chorus-warm rounded-lg p-4">
<div className="font-medium text-chorus-text-primary">{gpu.name}</div>
<div className="text-sm text-chorus-text-secondary">
{gpu.type.toUpperCase()} {gpu.memory} {gpu.driver}
</div>
</div>
@@ -238,21 +238,21 @@ export default function SystemDetection({
)}
{/* Network Information */}
<div className="bg-white border border-gray-200 rounded-lg p-6">
<div className="card">
<div className="flex items-center mb-4">
<GlobeAltIcon className="h-6 w-6 text-bzzz-primary mr-2" />
<h3 className="text-lg font-medium text-gray-900">Network Configuration</h3>
<h3 className="heading-subsection">Network Configuration</h3>
</div>
<div className="space-y-3">
<div>
<div className="text-sm font-medium text-gray-700">Hostname</div>
<div className="text-gray-900">{detectedInfo.network.hostname}</div>
<div className="text-sm font-medium text-chorus-text-secondary">Hostname</div>
<div className="text-chorus-text-primary">{detectedInfo.network.hostname}</div>
</div>
{detectedInfo.network.private_ips && detectedInfo.network.private_ips.length > 0 && (
<div>
<div className="text-sm font-medium text-gray-700 mb-2">Private IP Addresses</div>
<div className="text-sm font-medium text-chorus-text-secondary mb-2">Private IP Addresses</div>
<div className="space-y-2">
{detectedInfo.network.private_ips.map((ip, index) => (
<div key={index} className="flex justify-between items-center text-sm">
@@ -266,16 +266,16 @@ export default function SystemDetection({
{detectedInfo.network.public_ip && (
<div>
<div className="text-sm font-medium text-gray-700">Public IP</div>
<div className="text-gray-900">{detectedInfo.network.public_ip}</div>
<div className="text-sm font-medium text-chorus-text-secondary">Public IP</div>
<div className="text-chorus-text-primary">{detectedInfo.network.public_ip}</div>
</div>
)}
</div>
</div>
{/* Software Requirements */}
<div className="bg-white border border-gray-200 rounded-lg p-6">
<h3 className="text-lg font-medium text-gray-900 mb-4">Software Requirements</h3>
<div className="card">
<h3 className="heading-subsection mb-4">Software Requirements</h3>
<div className="space-y-4">
{[
@@ -304,9 +304,9 @@ export default function SystemDetection({
<div className="flex items-center">
<StatusIcon className={`h-5 w-5 mr-3 ${getStatusColor(software.installed)}`} />
<div>
<div className="font-medium text-gray-900">{software.name}</div>
<div className="font-medium text-chorus-text-primary">{software.name}</div>
{software.version && (
<div className="text-sm text-gray-600">Version: {software.version}</div>
<div className="text-sm text-chorus-text-secondary">Version: {software.version}</div>
)}
</div>
</div>
@@ -327,8 +327,8 @@ export default function SystemDetection({
</div>
{/* System Validation */}
<div className="bg-blue-50 border border-blue-200 rounded-lg p-6">
<h3 className="text-lg font-medium text-blue-900 mb-4">System Validation</h3>
<div className="panel panel-info">
<h3 className="heading-subsection mb-4 panel-title">System Validation</h3>
<div className="space-y-2">
{[
@@ -351,13 +351,13 @@ export default function SystemDetection({
<div key={index} className="flex items-center">
<StatusIcon className={`h-4 w-4 mr-3 ${
validation.passed
? 'text-green-600'
? 'text-eucalyptus-600'
: 'text-red-600'
}`} />
<span className={`text-sm ${
validation.passed
? 'text-green-800'
: 'text-red-800'
? 'text-eucalyptus-600'
: 'text-red-600'
}`}>
{validation.check}
{validation.warning && validation.passed && (
@@ -371,7 +371,7 @@ export default function SystemDetection({
</div>
{/* Action Buttons */}
<div className="flex justify-between pt-6 border-t border-gray-200">
<div className="flex justify-between pt-6 border-t border-chorus-border-defined">
<div>
{onBack && (
<button onClick={onBack} className="btn-outline">