open-notebook / frontend /src /components /errors /ConnectionErrorOverlay.tsx
baveshraam's picture
FIX: SurrealDB 2.0 migration syntax and Frontend/CORS link
f871fed
'use client'
import { useState } from 'react'
import { Card } from '@/components/ui/card'
import { Button } from '@/components/ui/button'
import {
Collapsible,
CollapsibleContent,
CollapsibleTrigger,
} from '@/components/ui/collapsible'
import { Database, Server, ChevronDown, ExternalLink } from 'lucide-react'
import { ConnectionError } from '@/lib/types/config'
interface ConnectionErrorOverlayProps {
error: ConnectionError
onRetry: () => void
}
export function ConnectionErrorOverlay({
error,
onRetry,
}: ConnectionErrorOverlayProps) {
const [showDetails, setShowDetails] = useState(false)
const isApiError = error.type === 'api-unreachable'
return (
<div
className="fixed inset-0 bg-background z-50 flex items-center justify-center p-4"
role="alert"
aria-live="assertive"
aria-atomic="true"
>
<Card className="max-w-2xl w-full p-8 space-y-6">
{/* Error icon and title */}
<div className="flex items-center gap-4">
{isApiError ? (
<Server className="w-12 h-12 text-destructive" aria-hidden="true" />
) : (
<Database className="w-12 h-12 text-destructive" aria-hidden="true" />
)}
<div>
<h1 className="text-2xl font-bold" id="error-title">
{isApiError
? 'Unable to Connect to API Server'
: 'Database Connection Failed'}
</h1>
<p className="text-muted-foreground">
{isApiError
? 'The Open Notebook API server could not be reached'
: 'The API server is running, but the database is not accessible'}
</p>
</div>
</div>
{/* Troubleshooting instructions */}
<div className="space-y-4 border-l-4 border-primary pl-4">
<h2 className="font-semibold">This usually means:</h2>
<ul className="list-disc list-inside space-y-2 text-sm">
{isApiError ? (
<>
<li>The API server is not running</li>
<li>The API server is running on a different address</li>
<li>Network connectivity issues</li>
</>
) : (
<>
<li>SurrealDB is not running</li>
<li>Database connection settings are incorrect</li>
<li>Network issues between API and database</li>
</>
)}
</ul>
<h2 className="font-semibold mt-4">Quick fixes:</h2>
{isApiError ? (
<div className="space-y-2 text-sm bg-muted p-4 rounded">
<p className="font-medium">Set the API_URL environment variable:</p>
<code className="block bg-background p-2 rounded text-xs">
# For Docker:
<br />
docker run -e API_URL=http://your-host:5055 ...
<br />
<br />
# For local development (.env file):
<br />
API_URL=http://localhost:5055
</code>
</div>
) : (
<div className="space-y-2 text-sm bg-muted p-4 rounded">
<p className="font-medium">Check if SurrealDB is running:</p>
<code className="block bg-background p-2 rounded text-xs">
# For Docker:
<br />
docker compose ps | grep surrealdb
<br />
docker compose logs surrealdb
</code>
</div>
)}
</div>
{/* Documentation link */}
<div className="text-sm">
<p>For detailed setup instructions, see:</p>
<a
href="https://github.com/lfnovo/open-notebook"
target="_blank"
rel="noopener noreferrer"
className="text-primary hover:underline inline-flex items-center gap-1"
>
Open Notebook Documentation
<ExternalLink className="w-4 h-4" />
</a>
</div>
{/* Collapsible technical details */}
{error.details && (
<Collapsible open={showDetails} onOpenChange={setShowDetails}>
<CollapsibleTrigger asChild>
<Button variant="ghost" size="sm" className="w-full justify-between">
<span>Show Technical Details</span>
<ChevronDown
className={`w-4 h-4 transition-transform ${
showDetails ? 'rotate-180' : ''
}`}
/>
</Button>
</CollapsibleTrigger>
<CollapsibleContent className="pt-4">
<div className="space-y-2 text-sm bg-muted p-4 rounded font-mono">
{error.details.attemptedUrl && (
<div>
<strong>Attempted URL:</strong> {error.details.attemptedUrl}
</div>
)}
{error.details.message && (
<div>
<strong>Message:</strong> {error.details.message}
</div>
)}
{error.details.technicalMessage && (
<div>
<strong>Technical Details:</strong>{' '}
{error.details.technicalMessage}
</div>
)}
{error.details.stack && (
<div>
<strong>Stack Trace:</strong>
<pre className="mt-2 overflow-x-auto text-xs">
{error.details.stack}
</pre>
</div>
)}
</div>
</CollapsibleContent>
</Collapsible>
)}
{/* Retry button */}
<div className="pt-4 border-t">
<Button onClick={onRetry} className="w-full" size="lg">
Retry Connection
</Button>
<p className="text-xs text-muted-foreground text-center mt-2">
Press R or click the button to retry
</p>
</div>
</Card>
</div>
)
}