Initial commit
This commit is contained in:
commit
411038582a
30
.dockerignore
Normal file
30
.dockerignore
Normal file
@ -0,0 +1,30 @@
|
||||
# Python
|
||||
__pycache__
|
||||
*.pyc
|
||||
*.pyo
|
||||
*.pyd
|
||||
.Python
|
||||
venv/
|
||||
env/
|
||||
.venv
|
||||
|
||||
# Git
|
||||
.git
|
||||
.gitignore
|
||||
|
||||
# IDEs
|
||||
.vscode
|
||||
.idea
|
||||
*.swp
|
||||
|
||||
# Docker
|
||||
Dockerfile
|
||||
docker-compose.yml
|
||||
.dockerignore
|
||||
|
||||
# Documentation
|
||||
README.md
|
||||
*.md
|
||||
|
||||
# OS
|
||||
.DS_Store
|
||||
27
.gitignore
vendored
Normal file
27
.gitignore
vendored
Normal file
@ -0,0 +1,27 @@
|
||||
# Python
|
||||
__pycache__/
|
||||
*.py[cod]
|
||||
*$py.class
|
||||
*.so
|
||||
.Python
|
||||
venv/
|
||||
env/
|
||||
ENV/
|
||||
.venv
|
||||
|
||||
# IDEs
|
||||
.vscode/
|
||||
.idea/
|
||||
*.swp
|
||||
*.swo
|
||||
*~
|
||||
|
||||
# OS
|
||||
.DS_Store
|
||||
Thumbs.db
|
||||
|
||||
# Logs
|
||||
*.log
|
||||
|
||||
# Docker
|
||||
docker-compose.override.yml
|
||||
30
Dockerfile
Normal file
30
Dockerfile
Normal file
@ -0,0 +1,30 @@
|
||||
FROM python:3.11-slim
|
||||
|
||||
WORKDIR /app
|
||||
|
||||
# Install system dependencies
|
||||
RUN apt-get update && apt-get install -y \
|
||||
gcc \
|
||||
&& rm -rf /var/lib/apt/lists/*
|
||||
|
||||
# Copy requirements and install Python dependencies
|
||||
COPY requirements.txt .
|
||||
RUN pip install --no-cache-dir -r requirements.txt
|
||||
|
||||
# Copy application code
|
||||
COPY app/ ./app/
|
||||
COPY static/ ./static/
|
||||
COPY templates/ ./templates/
|
||||
|
||||
# Note: Running as root for Docker socket access
|
||||
# In production, consider using rootless Docker or proper socket permissions
|
||||
|
||||
# Expose port
|
||||
EXPOSE 8000
|
||||
|
||||
# Health check
|
||||
HEALTHCHECK --interval=30s --timeout=10s --start-period=5s --retries=3 \
|
||||
CMD python -c "import urllib.request; urllib.request.urlopen('http://localhost:8000/api/health')"
|
||||
|
||||
# Run application
|
||||
CMD ["uvicorn", "app.main:app", "--host", "0.0.0.0", "--port", "8000"]
|
||||
173
README.md
Normal file
173
README.md
Normal file
@ -0,0 +1,173 @@
|
||||
# RedUnits Control Panel
|
||||
|
||||
A beautiful, lightweight service management dashboard for monitoring Docker containers and system resources.
|
||||
|
||||
## Features
|
||||
|
||||
- Real-time service monitoring
|
||||
- Docker container status and metrics
|
||||
- System resource usage (CPU, Memory, Disk)
|
||||
- Auto-refresh every 30 seconds
|
||||
- Responsive dark theme design
|
||||
- Minimal dependencies (FastAPI + Alpine.js)
|
||||
|
||||
## Project Structure
|
||||
|
||||
```
|
||||
~/repo/redunits/www/
|
||||
├── app/
|
||||
│ ├── __init__.py
|
||||
│ ├── main.py # FastAPI application
|
||||
│ └── docker_monitor.py # Docker monitoring module
|
||||
├── static/
|
||||
│ ├── css/
|
||||
│ │ └── style.css # Styles
|
||||
│ └── js/
|
||||
│ └── app.js # Alpine.js frontend logic
|
||||
├── templates/
|
||||
│ └── index.html # Main dashboard template
|
||||
├── Dockerfile
|
||||
├── docker-compose.yml
|
||||
├── requirements.txt
|
||||
└── README.md
|
||||
```
|
||||
|
||||
## Installation
|
||||
|
||||
### Prerequisites
|
||||
|
||||
- Docker and Docker Compose installed
|
||||
- Python 3.11+ (for local development)
|
||||
|
||||
### Quick Start with Docker
|
||||
|
||||
1. Build and run the container:
|
||||
```bash
|
||||
cd ~/repo/redunits/www
|
||||
docker-compose up -d --build
|
||||
```
|
||||
|
||||
2. Access the dashboard:
|
||||
```
|
||||
http://localhost:8000
|
||||
```
|
||||
|
||||
### Local Development
|
||||
|
||||
1. Create virtual environment:
|
||||
```bash
|
||||
python -m venv venv
|
||||
source venv/bin/activate # On Windows: venv\Scripts\activate
|
||||
```
|
||||
|
||||
2. Install dependencies:
|
||||
```bash
|
||||
pip install -r requirements.txt
|
||||
```
|
||||
|
||||
3. Run the application:
|
||||
```bash
|
||||
uvicorn app.main:app --reload --host 0.0.0.0 --port 8000
|
||||
```
|
||||
|
||||
## Deployment on VPS
|
||||
|
||||
### 1. Copy files to VPS:
|
||||
```bash
|
||||
rsync -avz ~/repo/redunits/www/ vps:/opt/redunits/www/
|
||||
```
|
||||
|
||||
### 2. SSH into VPS and deploy:
|
||||
```bash
|
||||
ssh vps
|
||||
cd /opt/redunits/www
|
||||
sudo docker-compose up -d --build
|
||||
```
|
||||
|
||||
### 3. Add to Caddy configuration:
|
||||
|
||||
Edit `/etc/caddy/Caddyfile`:
|
||||
```
|
||||
www.redunits.net, redunits.net {
|
||||
encode zstd gzip
|
||||
reverse_proxy 127.0.0.1:8000
|
||||
}
|
||||
```
|
||||
|
||||
Reload Caddy:
|
||||
```bash
|
||||
sudo systemctl reload caddy
|
||||
```
|
||||
|
||||
## API Endpoints
|
||||
|
||||
- `GET /` - Main dashboard page
|
||||
- `GET /api/services` - Get all services with status
|
||||
- `GET /api/system` - Get system statistics
|
||||
- `GET /api/health` - Health check endpoint
|
||||
|
||||
## Configuration
|
||||
|
||||
### Adding New Services
|
||||
|
||||
Edit `app/main.py` and add to the `SERVICES` list:
|
||||
|
||||
```python
|
||||
{
|
||||
"id": "service-id",
|
||||
"name": "Service Name",
|
||||
"icon": "🔧",
|
||||
"description": "Service description",
|
||||
"url": "https://service.domain.com",
|
||||
"container_name": "docker-container-name",
|
||||
"port": 8080
|
||||
}
|
||||
```
|
||||
|
||||
## Security Notes
|
||||
|
||||
- The application requires read-only access to Docker socket
|
||||
- Run with non-root user inside container
|
||||
- Use reverse proxy (Caddy) for HTTPS in production
|
||||
- Consider adding authentication for production use
|
||||
|
||||
## Monitoring
|
||||
|
||||
The dashboard automatically monitors:
|
||||
|
||||
- Container status (running/stopped)
|
||||
- Container CPU usage
|
||||
- Container memory usage
|
||||
- Container uptime
|
||||
- System CPU usage
|
||||
- System memory usage
|
||||
- System disk usage
|
||||
- Total and running container count
|
||||
|
||||
## Tech Stack
|
||||
|
||||
- **Backend**: FastAPI (Python)
|
||||
- **Frontend**: Alpine.js + Vanilla CSS
|
||||
- **Monitoring**: Docker SDK for Python + psutil
|
||||
- **Container**: Docker with multi-stage build
|
||||
|
||||
## Development
|
||||
|
||||
### Running tests:
|
||||
```bash
|
||||
# TODO: Add tests
|
||||
pytest
|
||||
```
|
||||
|
||||
### Code formatting:
|
||||
```bash
|
||||
black app/
|
||||
```
|
||||
|
||||
## License
|
||||
|
||||
MIT
|
||||
|
||||
## Author
|
||||
|
||||
RedUnits Team
|
||||
3
app/__init__.py
Normal file
3
app/__init__.py
Normal file
@ -0,0 +1,3 @@
|
||||
"""RedUnits Control Panel - Service Management Dashboard"""
|
||||
|
||||
__version__ = "1.0.0"
|
||||
146
app/docker_monitor.py
Normal file
146
app/docker_monitor.py
Normal file
@ -0,0 +1,146 @@
|
||||
import docker
|
||||
from typing import Dict, Optional
|
||||
import logging
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class DockerMonitor:
|
||||
"""Monitor and manage Docker containers"""
|
||||
|
||||
def __init__(self):
|
||||
try:
|
||||
self.client = docker.from_env()
|
||||
self.client.ping()
|
||||
self.available = True
|
||||
logger.info("Docker client initialized successfully")
|
||||
except Exception as e:
|
||||
logger.warning(f"Docker not available: {e}")
|
||||
self.client = None
|
||||
self.available = False
|
||||
|
||||
def get_container_status(self, container_name: str) -> Dict:
|
||||
"""
|
||||
Get status of a specific container.
|
||||
|
||||
Returns:
|
||||
Dict with: online, state, uptime, cpu_percent, memory_mb
|
||||
"""
|
||||
if not self.available:
|
||||
return {"online": False, "state": "unavailable", "message": "Docker not available"}
|
||||
|
||||
try:
|
||||
container = self.client.containers.get(container_name)
|
||||
stats = container.stats(stream=False)
|
||||
|
||||
# Calculate CPU percentage
|
||||
cpu_delta = (
|
||||
stats["cpu_stats"]["cpu_usage"]["total_usage"]
|
||||
- stats["precpu_stats"]["cpu_usage"]["total_usage"]
|
||||
)
|
||||
system_delta = (
|
||||
stats["cpu_stats"]["system_cpu_usage"]
|
||||
- stats["precpu_stats"]["system_cpu_usage"]
|
||||
)
|
||||
cpu_percent = 0.0
|
||||
if system_delta > 0:
|
||||
cpu_percent = (cpu_delta / system_delta) * 100.0
|
||||
|
||||
# Calculate memory usage
|
||||
memory_usage = stats["memory_stats"].get("usage", 0)
|
||||
memory_mb = memory_usage / (1024 * 1024)
|
||||
|
||||
return {
|
||||
"online": container.status == "running",
|
||||
"state": container.status,
|
||||
"uptime": self._format_uptime(container),
|
||||
"cpu_percent": round(cpu_percent, 2),
|
||||
"memory_mb": round(memory_mb, 2),
|
||||
}
|
||||
|
||||
except docker.errors.NotFound:
|
||||
return {
|
||||
"online": False,
|
||||
"state": "not_found",
|
||||
"message": f"Container '{container_name}' not found",
|
||||
}
|
||||
except Exception as e:
|
||||
logger.error(f"Error getting container status: {e}")
|
||||
return {"online": False, "state": "error", "message": str(e)}
|
||||
|
||||
def restart_container(self, container_name: str) -> Dict:
|
||||
"""Restart a specific container."""
|
||||
if not self.available:
|
||||
return {"success": False, "message": "Docker not available"}
|
||||
|
||||
try:
|
||||
container = self.client.containers.get(container_name)
|
||||
container.restart(timeout=10)
|
||||
logger.info(f"Container '{container_name}' restarted")
|
||||
return {"success": True, "message": f"Container '{container_name}' restarted"}
|
||||
except docker.errors.NotFound:
|
||||
return {"success": False, "message": f"Container '{container_name}' not found"}
|
||||
except Exception as e:
|
||||
logger.error(f"Error restarting container '{container_name}': {e}")
|
||||
return {"success": False, "message": str(e)}
|
||||
|
||||
def stop_container(self, container_name: str) -> Dict:
|
||||
"""Stop a specific container."""
|
||||
if not self.available:
|
||||
return {"success": False, "message": "Docker not available"}
|
||||
|
||||
try:
|
||||
container = self.client.containers.get(container_name)
|
||||
container.stop(timeout=10)
|
||||
logger.info(f"Container '{container_name}' stopped")
|
||||
return {"success": True, "message": f"Container '{container_name}' stopped"}
|
||||
except docker.errors.NotFound:
|
||||
return {"success": False, "message": f"Container '{container_name}' not found"}
|
||||
except Exception as e:
|
||||
logger.error(f"Error stopping container '{container_name}': {e}")
|
||||
return {"success": False, "message": str(e)}
|
||||
|
||||
def get_containers_count(self) -> int:
|
||||
"""Get total number of containers."""
|
||||
if not self.available:
|
||||
return 0
|
||||
try:
|
||||
return len(self.client.containers.list(all=True))
|
||||
except Exception as e:
|
||||
logger.error(f"Error counting containers: {e}")
|
||||
return 0
|
||||
|
||||
def get_running_containers_count(self) -> int:
|
||||
"""Get number of running containers."""
|
||||
if not self.available:
|
||||
return 0
|
||||
try:
|
||||
return len(self.client.containers.list())
|
||||
except Exception as e:
|
||||
logger.error(f"Error counting running containers: {e}")
|
||||
return 0
|
||||
|
||||
def _format_uptime(self, container) -> str:
|
||||
"""Format container uptime."""
|
||||
try:
|
||||
from datetime import datetime, timezone
|
||||
|
||||
started_at = container.attrs["State"]["StartedAt"]
|
||||
started = datetime.fromisoformat(started_at.replace("Z", "+00:00"))
|
||||
now = datetime.now(timezone.utc)
|
||||
uptime = now - started
|
||||
|
||||
days = uptime.days
|
||||
hours = uptime.seconds // 3600
|
||||
minutes = (uptime.seconds % 3600) // 60
|
||||
|
||||
if days > 0:
|
||||
return f"{days}d {hours}h"
|
||||
elif hours > 0:
|
||||
return f"{hours}h {minutes}m"
|
||||
else:
|
||||
return f"{minutes}m"
|
||||
|
||||
except Exception as e:
|
||||
logger.error(f"Error formatting uptime: {e}")
|
||||
return "unknown"
|
||||
298
app/main.py
Normal file
298
app/main.py
Normal file
@ -0,0 +1,298 @@
|
||||
import asyncio
|
||||
import json
|
||||
import logging
|
||||
import time
|
||||
from pathlib import Path
|
||||
from typing import List
|
||||
|
||||
import psutil
|
||||
import yaml
|
||||
from fastapi import FastAPI, Request, WebSocket, WebSocketDisconnect, HTTPException
|
||||
from fastapi.responses import HTMLResponse
|
||||
from fastapi.staticfiles import StaticFiles
|
||||
from fastapi.templating import Jinja2Templates
|
||||
|
||||
from .docker_monitor import DockerMonitor
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
app = FastAPI(title="RedUnits Control Panel")
|
||||
|
||||
# Setup paths
|
||||
BASE_DIR = Path(__file__).resolve().parent.parent
|
||||
app.mount("/static", StaticFiles(directory=str(BASE_DIR / "static")), name="static")
|
||||
templates = Jinja2Templates(directory=str(BASE_DIR / "templates"))
|
||||
|
||||
# Initialize Docker monitor
|
||||
docker_monitor = DockerMonitor()
|
||||
|
||||
|
||||
# ──────────────────────────────────────────────
|
||||
# Config loader
|
||||
# ──────────────────────────────────────────────
|
||||
|
||||
def load_services() -> List[dict]:
|
||||
"""Load services from config.yaml. Falls back to empty list on error."""
|
||||
config_path = BASE_DIR / "config.yaml"
|
||||
try:
|
||||
with open(config_path, "r", encoding="utf-8") as f:
|
||||
data = yaml.safe_load(f)
|
||||
return data.get("services", [])
|
||||
except FileNotFoundError:
|
||||
logger.warning(f"config.yaml not found at {config_path}. No services loaded.")
|
||||
return []
|
||||
except Exception as e:
|
||||
logger.error(f"Error loading config.yaml: {e}")
|
||||
return []
|
||||
|
||||
def load_documents() -> List[dict]:
|
||||
"""Load documents from config.yaml."""
|
||||
config_path = BASE_DIR / "config.yaml"
|
||||
try:
|
||||
with open(config_path, "r", encoding="utf-8") as f:
|
||||
data = yaml.safe_load(f)
|
||||
return data.get("documents", [])
|
||||
except Exception:
|
||||
return []
|
||||
|
||||
SERVICES = load_services()
|
||||
|
||||
|
||||
|
||||
# ──────────────────────────────────────────────
|
||||
# WebSocket connection manager
|
||||
# ──────────────────────────────────────────────
|
||||
|
||||
class ConnectionManager:
|
||||
def __init__(self):
|
||||
self.active_connections: List[WebSocket] = []
|
||||
|
||||
async def connect(self, websocket: WebSocket):
|
||||
await websocket.accept()
|
||||
self.active_connections.append(websocket)
|
||||
logger.info(f"WS client connected. Total: {len(self.active_connections)}")
|
||||
|
||||
def disconnect(self, websocket: WebSocket):
|
||||
if websocket in self.active_connections:
|
||||
self.active_connections.remove(websocket)
|
||||
logger.info(f"WS client disconnected. Total: {len(self.active_connections)}")
|
||||
|
||||
async def broadcast(self, data: dict):
|
||||
if not self.active_connections:
|
||||
return
|
||||
message = json.dumps(data)
|
||||
dead = []
|
||||
for ws in self.active_connections:
|
||||
try:
|
||||
await ws.send_text(message)
|
||||
except Exception:
|
||||
dead.append(ws)
|
||||
for ws in dead:
|
||||
self.disconnect(ws)
|
||||
|
||||
|
||||
manager = ConnectionManager()
|
||||
|
||||
|
||||
# ──────────────────────────────────────────────
|
||||
# Background metrics collector
|
||||
# ──────────────────────────────────────────────
|
||||
|
||||
async def collect_and_broadcast():
|
||||
"""Collect system + service metrics and broadcast to all WS clients."""
|
||||
try:
|
||||
services_data = []
|
||||
for service in SERVICES:
|
||||
status = await asyncio.to_thread(
|
||||
docker_monitor.get_container_status, service["container_name"]
|
||||
)
|
||||
services_data.append({**service, "status": status})
|
||||
|
||||
cpu_percent = await asyncio.to_thread(psutil.cpu_percent, 1)
|
||||
memory = psutil.virtual_memory()
|
||||
disk = psutil.disk_usage("/")
|
||||
containers_total = await asyncio.to_thread(docker_monitor.get_containers_count)
|
||||
containers_running = await asyncio.to_thread(docker_monitor.get_running_containers_count)
|
||||
uptime_seconds = time.time() - psutil.boot_time()
|
||||
|
||||
payload = {
|
||||
"type": "update",
|
||||
"services": services_data,
|
||||
"system": {
|
||||
"cpu": {"percent": round(cpu_percent, 1)},
|
||||
"memory": {
|
||||
"percent": round(memory.percent, 1),
|
||||
"used_gb": round(memory.used / (1024 ** 3), 2),
|
||||
"total_gb": round(memory.total / (1024 ** 3), 2),
|
||||
},
|
||||
"disk": {
|
||||
"percent": round(disk.percent, 1),
|
||||
"used_gb": round(disk.used / (1024 ** 3), 2),
|
||||
"total_gb": round(disk.total / (1024 ** 3), 2),
|
||||
},
|
||||
"containers": {
|
||||
"running": containers_running,
|
||||
"total": containers_total,
|
||||
},
|
||||
"uptime": {
|
||||
"days": int(uptime_seconds / 86400),
|
||||
"seconds": int(uptime_seconds),
|
||||
},
|
||||
},
|
||||
}
|
||||
await manager.broadcast(payload)
|
||||
except Exception as e:
|
||||
logger.error(f"Error in collect_and_broadcast: {e}")
|
||||
|
||||
|
||||
async def metrics_loop():
|
||||
"""Background loop: collect and broadcast metrics every 5 seconds."""
|
||||
while True:
|
||||
await collect_and_broadcast()
|
||||
await asyncio.sleep(5)
|
||||
|
||||
|
||||
# ──────────────────────────────────────────────
|
||||
# App lifecycle
|
||||
# ──────────────────────────────────────────────
|
||||
|
||||
@app.on_event("startup")
|
||||
async def startup_event():
|
||||
asyncio.create_task(metrics_loop())
|
||||
logger.info("Metrics background task started")
|
||||
|
||||
|
||||
# ──────────────────────────────────────────────
|
||||
# HTTP routes
|
||||
# ──────────────────────────────────────────────
|
||||
|
||||
@app.get("/", response_class=HTMLResponse)
|
||||
async def root(request: Request):
|
||||
"""Render main dashboard page."""
|
||||
return templates.TemplateResponse("index.html", {"request": request})
|
||||
|
||||
|
||||
@app.get("/api/services")
|
||||
async def get_services():
|
||||
"""Get all services with their current status (one-shot REST fallback)."""
|
||||
services_with_status = []
|
||||
for service in SERVICES:
|
||||
status = await asyncio.to_thread(
|
||||
docker_monitor.get_container_status, service["container_name"]
|
||||
)
|
||||
services_with_status.append({**service, "status": status})
|
||||
return {"services": services_with_status}
|
||||
|
||||
|
||||
@app.get("/api/system")
|
||||
async def get_system_stats():
|
||||
"""Get system statistics (one-shot REST fallback)."""
|
||||
cpu_percent = await asyncio.to_thread(psutil.cpu_percent, 1)
|
||||
memory = psutil.virtual_memory()
|
||||
disk = psutil.disk_usage("/")
|
||||
containers_total = await asyncio.to_thread(docker_monitor.get_containers_count)
|
||||
containers_running = await asyncio.to_thread(docker_monitor.get_running_containers_count)
|
||||
uptime_seconds = time.time() - psutil.boot_time()
|
||||
|
||||
return {
|
||||
"cpu": {"percent": round(cpu_percent, 1)},
|
||||
"memory": {
|
||||
"percent": round(memory.percent, 1),
|
||||
"used_gb": round(memory.used / (1024 ** 3), 2),
|
||||
"total_gb": round(memory.total / (1024 ** 3), 2),
|
||||
},
|
||||
"disk": {
|
||||
"percent": round(disk.percent, 1),
|
||||
"used_gb": round(disk.used / (1024 ** 3), 2),
|
||||
"total_gb": round(disk.total / (1024 ** 3), 2),
|
||||
},
|
||||
"containers": {
|
||||
"running": containers_running,
|
||||
"total": containers_total,
|
||||
},
|
||||
"uptime": {
|
||||
"days": int(uptime_seconds / 86400),
|
||||
"seconds": int(uptime_seconds),
|
||||
},
|
||||
}
|
||||
|
||||
|
||||
@app.get("/api/health")
|
||||
async def health_check():
|
||||
"""Health check endpoint."""
|
||||
return {"status": "ok", "message": "RedUnits Control Panel is running"}
|
||||
|
||||
|
||||
@app.post("/api/services/{service_id}/restart")
|
||||
async def restart_service(service_id: str):
|
||||
"""Restart a service container."""
|
||||
service = next((s for s in SERVICES if s["id"] == service_id), None)
|
||||
if not service:
|
||||
raise HTTPException(status_code=404, detail=f"Service '{service_id}' not found")
|
||||
result = await asyncio.to_thread(
|
||||
docker_monitor.restart_container, service["container_name"]
|
||||
)
|
||||
if not result["success"]:
|
||||
raise HTTPException(status_code=500, detail=result["message"])
|
||||
return result
|
||||
|
||||
|
||||
@app.post("/api/services/{service_id}/stop")
|
||||
async def stop_service(service_id: str):
|
||||
"""Stop a service container."""
|
||||
service = next((s for s in SERVICES if s["id"] == service_id), None)
|
||||
if not service:
|
||||
raise HTTPException(status_code=404, detail=f"Service '{service_id}' not found")
|
||||
result = await asyncio.to_thread(
|
||||
docker_monitor.stop_container, service["container_name"]
|
||||
)
|
||||
if not result["success"]:
|
||||
raise HTTPException(status_code=500, detail=result["message"])
|
||||
return result
|
||||
|
||||
|
||||
@app.get("/api/documents")
|
||||
async def get_documents():
|
||||
"""Get list of available documents."""
|
||||
return {"documents": load_documents()}
|
||||
|
||||
|
||||
@app.get("/api/document/{doc_id}")
|
||||
async def get_document_content(doc_id: str):
|
||||
"""Fetch raw document content (acts as a proxy)."""
|
||||
docs = load_documents()
|
||||
doc = next((d for d in docs if d["id"] == doc_id), None)
|
||||
if not doc:
|
||||
raise HTTPException(status_code=404, detail=f"Document '{doc_id}' not found")
|
||||
|
||||
import requests
|
||||
try:
|
||||
resp = await asyncio.to_thread(requests.get, doc["url"], timeout=10)
|
||||
resp.raise_for_status()
|
||||
return {"content": resp.text}
|
||||
except Exception as e:
|
||||
logger.error(f"Error fetching document '{doc_id}': {e}")
|
||||
raise HTTPException(status_code=500, detail=str(e))
|
||||
|
||||
|
||||
|
||||
# ──────────────────────────────────────────────
|
||||
# WebSocket endpoint
|
||||
# ──────────────────────────────────────────────
|
||||
|
||||
@app.websocket("/ws")
|
||||
async def websocket_endpoint(websocket: WebSocket):
|
||||
await manager.connect(websocket)
|
||||
# Send initial data immediately on connect
|
||||
await collect_and_broadcast()
|
||||
try:
|
||||
while True:
|
||||
# Keep connection alive; client messages are ignored
|
||||
await websocket.receive_text()
|
||||
except WebSocketDisconnect:
|
||||
manager.disconnect(websocket)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
import uvicorn
|
||||
uvicorn.run(app, host="0.0.0.0", port=8000)
|
||||
30
config.yaml
Normal file
30
config.yaml
Normal file
@ -0,0 +1,30 @@
|
||||
services:
|
||||
- id: vault
|
||||
name: Vault
|
||||
icon: "🗄️"
|
||||
description: "Gitea - Self-hosted Git service for repositories and collaboration"
|
||||
url: "https://vault.redunits.net"
|
||||
container_name: gitea
|
||||
port: 3080
|
||||
|
||||
- id: cell
|
||||
name: Cell
|
||||
icon: "🧠"
|
||||
description: "Qdrant - Vector database for AI embeddings and semantic search"
|
||||
url: "https://cell.redunits.net"
|
||||
container_name: qdrant
|
||||
port: 6333
|
||||
|
||||
- id: aevyra
|
||||
name: Aevyra
|
||||
icon: "🌐"
|
||||
description: "Aevyra Web - Main web application and public-facing site"
|
||||
url: "https://www.aevyra.net"
|
||||
container_name: aevyra-web
|
||||
port: 3001
|
||||
|
||||
documents:
|
||||
- id: naming-rules
|
||||
title: "Правила именования устройств"
|
||||
description: "Стандарт формирования имён для хостов и оборудования"
|
||||
url: "https://vault.redunits.net/user/repo/raw/branch/main/naming.md" # Замени на реальную ссылку (Raw format)
|
||||
28
docker-compose.yml
Normal file
28
docker-compose.yml
Normal file
@ -0,0 +1,28 @@
|
||||
version: '3.8'
|
||||
|
||||
services:
|
||||
redunits-panel:
|
||||
build: .
|
||||
container_name: redunits-control-panel
|
||||
ports:
|
||||
- "8000:8000"
|
||||
volumes:
|
||||
# Mount Docker socket for container monitoring
|
||||
- /var/run/docker.sock:/var/run/docker.sock:ro
|
||||
# Mount config file for easy editing without rebuild
|
||||
- ./config.yaml:/app/config.yaml:ro
|
||||
environment:
|
||||
- PYTHONUNBUFFERED=1
|
||||
# GID 988 = docker group on the VPS host — needed for docker.sock access
|
||||
group_add:
|
||||
- "988"
|
||||
restart: unless-stopped
|
||||
networks:
|
||||
- redunits-network
|
||||
labels:
|
||||
- "com.redunits.service=control-panel"
|
||||
- "com.redunits.version=1.0.0"
|
||||
|
||||
networks:
|
||||
redunits-network:
|
||||
driver: bridge
|
||||
9
requirements.txt
Normal file
9
requirements.txt
Normal file
@ -0,0 +1,9 @@
|
||||
fastapi==0.109.0
|
||||
uvicorn[standard]==0.27.0
|
||||
jinja2==3.1.3
|
||||
docker==7.0.0
|
||||
requests==2.31.0
|
||||
urllib3<2
|
||||
psutil==5.9.8
|
||||
python-multipart==0.0.9
|
||||
pyyaml==6.0.1
|
||||
645
static/css/style.css
Normal file
645
static/css/style.css
Normal file
@ -0,0 +1,645 @@
|
||||
@import url('https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600;700&display=swap');
|
||||
|
||||
:root {
|
||||
/* Colors inspired by the mock-up */
|
||||
--bg-main: #fbfbf9;
|
||||
--sys-card-bg: #f6f5ef;
|
||||
--svc-online-bg: #ffffff;
|
||||
--svc-offline-bg: #f6f5ef;
|
||||
|
||||
--text-main: #111827;
|
||||
--text-muted: #6b7280;
|
||||
--text-light: #9ca3af;
|
||||
|
||||
--red-main: #f05252;
|
||||
--red-hover: #e02424;
|
||||
|
||||
--green: #10b981;
|
||||
--green-bg: #ecfdf5;
|
||||
--red: #ef4444;
|
||||
--red-bg: #fef2f2;
|
||||
--yellow: #f59e0b;
|
||||
|
||||
--border-color: #e5e7eb;
|
||||
--border-card: #f3f4f6;
|
||||
}
|
||||
|
||||
* {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
body {
|
||||
font-family: 'Inter', system-ui, -apple-system, sans-serif;
|
||||
background-color: var(--bg-main);
|
||||
color: var(--text-main);
|
||||
padding: 30px 40px;
|
||||
min-height: 100vh;
|
||||
}
|
||||
|
||||
.container {
|
||||
max-width: 1400px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
/* ── HEADER ── */
|
||||
header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
padding-bottom: 24px;
|
||||
border-bottom: 1px solid var(--border-color);
|
||||
margin-bottom: 40px;
|
||||
}
|
||||
|
||||
.header-left {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 16px;
|
||||
}
|
||||
|
||||
.logo-dot {
|
||||
width: 16px;
|
||||
height: 16px;
|
||||
background-color: #ff9fb2;
|
||||
border-radius: 50%;
|
||||
}
|
||||
|
||||
h1 {
|
||||
font-size: 1.5rem;
|
||||
font-weight: 600;
|
||||
color: #8c8c8c;
|
||||
}
|
||||
|
||||
.header-right {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 16px;
|
||||
font-size: 0.9rem;
|
||||
color: var(--text-muted);
|
||||
}
|
||||
|
||||
.indicator-group {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
padding: 4px 12px;
|
||||
border-radius: 99px;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.ws-live {
|
||||
color: var(--green);
|
||||
border: 1px solid var(--green);
|
||||
background: var(--green-bg);
|
||||
}
|
||||
|
||||
.ws-offline {
|
||||
color: var(--yellow);
|
||||
border: 1px solid var(--yellow);
|
||||
}
|
||||
|
||||
.ws-dot {
|
||||
width: 6px;
|
||||
height: 6px;
|
||||
border-radius: 50%;
|
||||
background-color: currentColor;
|
||||
animation: pulse 1.5s infinite;
|
||||
}
|
||||
|
||||
@keyframes pulse {
|
||||
|
||||
0%,
|
||||
100% {
|
||||
opacity: 1;
|
||||
}
|
||||
|
||||
50% {
|
||||
opacity: 0.4;
|
||||
}
|
||||
}
|
||||
|
||||
.refresh-btn {
|
||||
background: transparent;
|
||||
border: 1px solid var(--border-color);
|
||||
color: var(--text-muted);
|
||||
padding: 6px 14px;
|
||||
border-radius: 8px;
|
||||
cursor: pointer;
|
||||
font-weight: 500;
|
||||
transition: all 0.2s;
|
||||
}
|
||||
|
||||
.refresh-btn:hover:not(:disabled) {
|
||||
background: #f3f4f6;
|
||||
color: var(--text-main);
|
||||
}
|
||||
|
||||
/* ── SECTIONS ── */
|
||||
.section {
|
||||
margin-bottom: 48px;
|
||||
}
|
||||
|
||||
.section-title {
|
||||
font-size: 0.85rem;
|
||||
font-weight: 700;
|
||||
color: var(--text-muted);
|
||||
letter-spacing: 0.08em;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
/* ── SYSTEM STATUS ── */
|
||||
.system-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(280px, 1fr));
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.stat-card {
|
||||
background-color: var(--sys-card-bg);
|
||||
padding: 24px;
|
||||
border-radius: 16px;
|
||||
border: 1px solid transparent;
|
||||
}
|
||||
|
||||
.stat-label {
|
||||
font-size: 0.95rem;
|
||||
color: var(--text-main);
|
||||
margin-bottom: 12px;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.stat-value {
|
||||
font-size: 2.2rem;
|
||||
font-weight: 600;
|
||||
color: var(--text-main);
|
||||
margin-bottom: 16px;
|
||||
}
|
||||
|
||||
.progress-bar {
|
||||
width: 100%;
|
||||
height: 4px;
|
||||
background: #e5e5e5;
|
||||
border-radius: 2px;
|
||||
margin-bottom: 12px;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.progress-fill {
|
||||
height: 100%;
|
||||
border-radius: 2px;
|
||||
transition: width 0.5s ease;
|
||||
}
|
||||
|
||||
.progress-fill {
|
||||
background-color: var(--text-main);
|
||||
}
|
||||
|
||||
.progress-fill.green {
|
||||
background-color: var(--green);
|
||||
}
|
||||
|
||||
.progress-fill.red {
|
||||
background-color: var(--red);
|
||||
}
|
||||
|
||||
.stat-detail {
|
||||
font-size: 0.85rem;
|
||||
color: var(--text-muted);
|
||||
}
|
||||
|
||||
.stat-status {
|
||||
font-size: 0.9rem;
|
||||
font-weight: 500;
|
||||
margin-top: 24px;
|
||||
}
|
||||
|
||||
.status-ok {
|
||||
color: #d97706;
|
||||
}
|
||||
|
||||
/* Warning color from mockup for "Some services offline" */
|
||||
.status-warn {
|
||||
color: var(--yellow);
|
||||
}
|
||||
|
||||
/* ── SERVICES ── */
|
||||
.services-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(340px, 1fr));
|
||||
gap: 24px;
|
||||
}
|
||||
|
||||
.service-card {
|
||||
background-color: var(--svc-online-bg);
|
||||
border-radius: 20px;
|
||||
padding: 28px;
|
||||
border: 1px solid var(--border-card);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
box-shadow: 0 4px 20px rgba(0, 0, 0, 0.02);
|
||||
}
|
||||
|
||||
.card-offline {
|
||||
background-color: var(--svc-offline-bg);
|
||||
border: 1px solid var(--border-color);
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
.service-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.service-name {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 12px;
|
||||
font-size: 1.4rem;
|
||||
font-weight: 600;
|
||||
color: var(--text-main);
|
||||
}
|
||||
|
||||
.status-badge {
|
||||
padding: 4px 12px;
|
||||
border-radius: 99px;
|
||||
font-size: 0.8rem;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.status-online {
|
||||
color: var(--green);
|
||||
border: 1px solid var(--green);
|
||||
background-color: var(--green-bg);
|
||||
}
|
||||
|
||||
.status-offline {
|
||||
color: var(--red);
|
||||
border: 1px solid var(--red);
|
||||
background-color: var(--red-bg);
|
||||
}
|
||||
|
||||
.service-description {
|
||||
color: var(--text-muted);
|
||||
font-size: 0.95rem;
|
||||
line-height: 1.5;
|
||||
margin-bottom: 16px;
|
||||
min-height: 44px;
|
||||
/* Align cards if description differs slightly */
|
||||
}
|
||||
|
||||
.service-url {
|
||||
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;
|
||||
font-size: 0.85rem;
|
||||
color: var(--text-light);
|
||||
margin-bottom: 24px;
|
||||
letter-spacing: 0.05em;
|
||||
}
|
||||
|
||||
.service-metrics {
|
||||
background-color: var(--sys-card-bg);
|
||||
border-radius: 12px;
|
||||
padding: 16px 20px;
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
margin-bottom: 24px;
|
||||
margin-top: auto;
|
||||
/* push to bottom */
|
||||
}
|
||||
|
||||
.metric {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 6px;
|
||||
}
|
||||
|
||||
.metric-label {
|
||||
font-size: 0.75rem;
|
||||
color: var(--text-muted);
|
||||
text-transform: uppercase;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.metric-value {
|
||||
font-size: 1.05rem;
|
||||
font-weight: 700;
|
||||
color: var(--text-main);
|
||||
display: flex;
|
||||
flex-direction: row;
|
||||
align-items: baseline;
|
||||
gap: 4px;
|
||||
}
|
||||
|
||||
.metric-unit {
|
||||
display: inline;
|
||||
font-size: 0.8rem;
|
||||
color: var(--text-muted);
|
||||
}
|
||||
|
||||
.offline-message {
|
||||
color: var(--text-muted);
|
||||
font-size: 1.1rem;
|
||||
text-align: center;
|
||||
margin: 40px 0;
|
||||
margin-top: auto;
|
||||
}
|
||||
|
||||
/* ── ACTIONS ── */
|
||||
.service-actions {
|
||||
display: flex;
|
||||
gap: 12px;
|
||||
align-items: stretch;
|
||||
}
|
||||
|
||||
.service-actions-center {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.btn {
|
||||
border-radius: 12px;
|
||||
font-weight: 600;
|
||||
font-size: 0.95rem;
|
||||
cursor: pointer;
|
||||
transition: all 0.2s;
|
||||
text-align: center;
|
||||
text-decoration: none;
|
||||
display: inline-flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
padding: 12px 24px;
|
||||
border: none;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
background-color: var(--red-main);
|
||||
color: #ffffff;
|
||||
padding: 12px 32px;
|
||||
/* Wider open button */
|
||||
font-size: 1.1rem;
|
||||
}
|
||||
|
||||
.btn-primary:hover {
|
||||
background-color: var(--red-hover);
|
||||
}
|
||||
|
||||
.btn-outline {
|
||||
background-color: #ffffff;
|
||||
border: 1px solid var(--border-color);
|
||||
color: var(--text-main);
|
||||
}
|
||||
|
||||
.btn-outline:hover {
|
||||
background-color: #f9fafb;
|
||||
border-color: #d1d5db;
|
||||
}
|
||||
|
||||
.btn-icon {
|
||||
background-color: #ffffff;
|
||||
border: 1px solid var(--border-color);
|
||||
color: var(--text-main);
|
||||
width: 48px;
|
||||
padding: 0;
|
||||
font-size: 1.2rem;
|
||||
}
|
||||
|
||||
.btn-icon:hover {
|
||||
background-color: #f9fafb;
|
||||
}
|
||||
|
||||
.btn-full {
|
||||
padding: 14px 24px;
|
||||
font-size: 1rem;
|
||||
background-color: #ffffff;
|
||||
border: 1px solid var(--border-color);
|
||||
margin-bottom: 15px;
|
||||
}
|
||||
|
||||
/* ── TOAST ── */
|
||||
.toast {
|
||||
position: fixed;
|
||||
top: 24px;
|
||||
right: 24px;
|
||||
z-index: 1000;
|
||||
padding: 16px 24px;
|
||||
border-radius: 12px;
|
||||
font-weight: 600;
|
||||
font-size: 0.95rem;
|
||||
box-shadow: 0 10px 25px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.toast-success {
|
||||
background-color: var(--green-bg);
|
||||
color: var(--green);
|
||||
border: 1px solid var(--green);
|
||||
}
|
||||
|
||||
.toast-error {
|
||||
background-color: var(--red-bg);
|
||||
color: var(--red);
|
||||
border: 1px solid var(--red);
|
||||
}
|
||||
|
||||
/* ── DOCUMENTS ── */
|
||||
.docs-list {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
|
||||
gap: 16px;
|
||||
}
|
||||
|
||||
.doc-item {
|
||||
background-color: var(--svc-online-bg);
|
||||
border: 1px solid var(--border-color);
|
||||
border-radius: 12px;
|
||||
padding: 16px 20px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 14px;
|
||||
cursor: pointer;
|
||||
transition: all 0.2s;
|
||||
}
|
||||
|
||||
.doc-item:hover {
|
||||
border-color: #d1d5db;
|
||||
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.03);
|
||||
transform: translateY(-2px);
|
||||
}
|
||||
|
||||
.doc-icon {
|
||||
font-size: 1.5rem;
|
||||
}
|
||||
|
||||
.doc-title {
|
||||
font-weight: 600;
|
||||
color: var(--text-main);
|
||||
margin-bottom: 4px;
|
||||
font-size: 0.95rem;
|
||||
}
|
||||
|
||||
.doc-desc {
|
||||
font-size: 0.8rem;
|
||||
color: var(--text-muted);
|
||||
}
|
||||
|
||||
/* ── MODAL ── */
|
||||
.modal-overlay {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
background: rgba(17, 24, 39, 0.4);
|
||||
backdrop-filter: blur(4px);
|
||||
z-index: 2000;
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
padding: 20px;
|
||||
}
|
||||
|
||||
.modal-content {
|
||||
background: #fff;
|
||||
border-radius: 16px;
|
||||
width: 100%;
|
||||
max-width: 800px;
|
||||
max-height: 90vh;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
box-shadow: 0 20px 40px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.modal-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
padding: 20px 24px;
|
||||
border-bottom: 1px solid var(--border-color);
|
||||
}
|
||||
|
||||
.modal-header h3 {
|
||||
font-size: 1.2rem;
|
||||
color: var(--text-main);
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.modal-close {
|
||||
background: transparent;
|
||||
border: none;
|
||||
font-size: 1.2rem;
|
||||
color: var(--text-muted);
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.modal-close:hover {
|
||||
color: var(--text-main);
|
||||
}
|
||||
|
||||
.modal-body {
|
||||
padding: 24px;
|
||||
overflow-y: auto;
|
||||
}
|
||||
|
||||
.doc-loading {
|
||||
text-align: center;
|
||||
padding: 40px;
|
||||
color: var(--text-muted);
|
||||
}
|
||||
|
||||
/* ── MARKDOWN BASIC STYLES ── */
|
||||
.markdown-body {
|
||||
font-family: inherit;
|
||||
font-size: 0.95rem;
|
||||
line-height: 1.6;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
.markdown-body h1,
|
||||
.markdown-body h2,
|
||||
.markdown-body h3 {
|
||||
margin-top: 1.5em;
|
||||
margin-bottom: 1em;
|
||||
font-weight: 600;
|
||||
color: #111;
|
||||
}
|
||||
|
||||
.markdown-body h1 {
|
||||
font-size: 1.8em;
|
||||
border-bottom: 1px solid #eee;
|
||||
padding-bottom: 0.3em;
|
||||
}
|
||||
|
||||
.markdown-body h2 {
|
||||
font-size: 1.4em;
|
||||
}
|
||||
|
||||
.markdown-body p,
|
||||
.markdown-body ul,
|
||||
.markdown-body ol {
|
||||
margin-bottom: 1em;
|
||||
}
|
||||
|
||||
.markdown-body code {
|
||||
background: #f6f8fa;
|
||||
padding: 0.2em 0.4em;
|
||||
border-radius: 3px;
|
||||
font-family: monospace;
|
||||
}
|
||||
|
||||
.markdown-body pre {
|
||||
background: #f6f8fa;
|
||||
padding: 16px;
|
||||
border-radius: 8px;
|
||||
overflow-x: auto;
|
||||
margin-bottom: 1em;
|
||||
}
|
||||
|
||||
.markdown-body pre code {
|
||||
background: transparent;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
.markdown-body blockquote {
|
||||
border-left: 4px solid #dfe2e5;
|
||||
padding: 0 1em;
|
||||
color: #6a737d;
|
||||
margin-left: 0;
|
||||
}
|
||||
|
||||
.markdown-body table {
|
||||
border-collapse: collapse;
|
||||
width: 100%;
|
||||
margin-bottom: 1em;
|
||||
}
|
||||
|
||||
.markdown-body th,
|
||||
.markdown-body td {
|
||||
border: 1px solid #dfe2e5;
|
||||
padding: 6px 13px;
|
||||
}
|
||||
|
||||
footer {
|
||||
text-align: center;
|
||||
padding: 40px 0;
|
||||
color: var(--text-light);
|
||||
font-size: 0.85rem;
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.system-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.services-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
header {
|
||||
flex-direction: column;
|
||||
gap: 20px;
|
||||
align-items: flex-start;
|
||||
}
|
||||
}
|
||||
5
static/js/alpine.min.js
vendored
Normal file
5
static/js/alpine.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
193
static/js/app.js
Normal file
193
static/js/app.js
Normal file
@ -0,0 +1,193 @@
|
||||
function dashboard() {
|
||||
return {
|
||||
services: [],
|
||||
system: {
|
||||
cpu: { percent: 0 },
|
||||
memory: { percent: 0, used_gb: 0, total_gb: 0 },
|
||||
disk: { percent: 0, used_gb: 0, total_gb: 0 },
|
||||
containers: { running: 0, total: 0 },
|
||||
uptime: { days: 0, seconds: 0 }
|
||||
},
|
||||
lastUpdate: 'Never',
|
||||
loading: true,
|
||||
wsConnected: false,
|
||||
ws: null,
|
||||
wsReconnectTimeout: null,
|
||||
actionMessage: null,
|
||||
actionError: false,
|
||||
|
||||
// ──────────────────── Documents ────────────────────
|
||||
documents: [],
|
||||
showDocModal: false,
|
||||
docTitle: '',
|
||||
docContent: '',
|
||||
docLoading: false,
|
||||
|
||||
init() {
|
||||
console.log('Initializing RedUnits Control Panel...');
|
||||
this.fetchDocuments();
|
||||
this.connectWebSocket();
|
||||
},
|
||||
|
||||
// ──────────────────── WebSocket ────────────────────
|
||||
|
||||
connectWebSocket() {
|
||||
const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
|
||||
const wsUrl = `${protocol}//${window.location.host}/ws`;
|
||||
|
||||
console.log(`Connecting to WebSocket: ${wsUrl}`);
|
||||
this.ws = new WebSocket(wsUrl);
|
||||
|
||||
this.ws.onopen = () => {
|
||||
console.log('WebSocket connected');
|
||||
this.wsConnected = true;
|
||||
if (this.wsReconnectTimeout) {
|
||||
clearTimeout(this.wsReconnectTimeout);
|
||||
this.wsReconnectTimeout = null;
|
||||
}
|
||||
};
|
||||
|
||||
this.ws.onmessage = (event) => {
|
||||
try {
|
||||
const data = JSON.parse(event.data);
|
||||
if (data.type === 'update') {
|
||||
this.services = data.services;
|
||||
this.system = data.system;
|
||||
this.loading = false;
|
||||
this.updateLastUpdate();
|
||||
}
|
||||
} catch (e) {
|
||||
console.error('Error parsing WS message:', e);
|
||||
}
|
||||
};
|
||||
|
||||
this.ws.onclose = () => {
|
||||
console.warn('WebSocket disconnected. Reconnecting in 5s...');
|
||||
this.wsConnected = false;
|
||||
this.scheduleReconnect();
|
||||
};
|
||||
|
||||
this.ws.onerror = (err) => {
|
||||
console.error('WebSocket error:', err);
|
||||
this.wsConnected = false;
|
||||
this.ws.close();
|
||||
};
|
||||
},
|
||||
|
||||
scheduleReconnect() {
|
||||
if (this.wsReconnectTimeout) return;
|
||||
this.wsReconnectTimeout = setTimeout(() => {
|
||||
this.wsReconnectTimeout = null;
|
||||
this.connectWebSocket();
|
||||
}, 5000);
|
||||
},
|
||||
|
||||
// ──────────────────── Manual refresh (REST fallback) ────────────────────
|
||||
|
||||
async refresh() {
|
||||
this.loading = true;
|
||||
try {
|
||||
const [svcRes, sysRes] = await Promise.all([
|
||||
fetch('/api/services'),
|
||||
fetch('/api/system')
|
||||
]);
|
||||
const svcData = await svcRes.json();
|
||||
const sysData = await sysRes.json();
|
||||
this.services = svcData.services;
|
||||
this.system = sysData;
|
||||
this.updateLastUpdate();
|
||||
} catch (error) {
|
||||
console.error('Error refreshing data:', error);
|
||||
} finally {
|
||||
this.loading = false;
|
||||
}
|
||||
},
|
||||
|
||||
// ──────────────────── Documents ────────────────────
|
||||
|
||||
async fetchDocuments() {
|
||||
try {
|
||||
const response = await fetch('/api/documents');
|
||||
if (response.ok) {
|
||||
const data = await response.json();
|
||||
this.documents = data.documents;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Error fetching documents:', error);
|
||||
}
|
||||
},
|
||||
|
||||
async openDocument(doc) {
|
||||
this.showDocModal = true;
|
||||
this.docTitle = doc.title;
|
||||
this.docContent = '';
|
||||
this.docLoading = true;
|
||||
try {
|
||||
const res = await fetch(`/api/document/${doc.id}`);
|
||||
const data = await res.json();
|
||||
if (res.ok) {
|
||||
this.docContent = marked.parse(data.content);
|
||||
} else {
|
||||
this.docContent = `<div style="color:var(--red)">Failed to load document: ${data.detail || 'Unknown error'}</div>`;
|
||||
}
|
||||
} catch (error) {
|
||||
this.docContent = `<div style="color:var(--red)">Network error while loading document</div>`;
|
||||
} finally {
|
||||
this.docLoading = false;
|
||||
}
|
||||
},
|
||||
|
||||
closeDocument() {
|
||||
this.showDocModal = false;
|
||||
},
|
||||
|
||||
// ──────────────────── Container actions ────────────────────
|
||||
|
||||
async restartService(serviceId) {
|
||||
if (!confirm(`Restart service "${serviceId}"?`)) return;
|
||||
await this._serviceAction(serviceId, 'restart');
|
||||
},
|
||||
|
||||
async stopService(serviceId) {
|
||||
if (!confirm(`Stop service "${serviceId}"? It will go offline.`)) return;
|
||||
await this._serviceAction(serviceId, 'stop');
|
||||
},
|
||||
|
||||
async _serviceAction(serviceId, action) {
|
||||
try {
|
||||
const res = await fetch(`/api/services/${serviceId}/${action}`, { method: 'POST' });
|
||||
const data = await res.json();
|
||||
if (res.ok) {
|
||||
this.showMessage(`✅ ${data.message}`, false);
|
||||
} else {
|
||||
this.showMessage(`❌ ${data.detail || data.message}`, true);
|
||||
}
|
||||
} catch (e) {
|
||||
this.showMessage(`❌ Network error: ${e.message}`, true);
|
||||
}
|
||||
},
|
||||
|
||||
showMessage(msg, isError) {
|
||||
this.actionMessage = msg;
|
||||
this.actionError = isError;
|
||||
setTimeout(() => { this.actionMessage = null; }, 4000);
|
||||
},
|
||||
|
||||
// ──────────────────── Helpers ────────────────────
|
||||
|
||||
updateLastUpdate() {
|
||||
this.lastUpdate = new Date().toLocaleTimeString('en-GB', { timeZone: 'Europe/Riga' });
|
||||
},
|
||||
|
||||
allServicesOnline() {
|
||||
if (this.services.length === 0) return false;
|
||||
return this.services.every(s => s.status.online);
|
||||
},
|
||||
|
||||
getProgressColor(percent) {
|
||||
if (percent < 60) return 'green';
|
||||
if (percent < 80) return 'yellow';
|
||||
return 'red';
|
||||
}
|
||||
};
|
||||
}
|
||||
69
static/js/marked.min.js
vendored
Normal file
69
static/js/marked.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
209
templates/index.html
Normal file
209
templates/index.html
Normal file
@ -0,0 +1,209 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<title>RedUnits Control Panel</title>
|
||||
<link rel="stylesheet" href="/static/css/style.css">
|
||||
<!-- Alpine.js served locally (no CDN dependency) -->
|
||||
<script defer src="/static/js/alpine.min.js"></script>
|
||||
<script src="/static/js/marked.min.js"></script>
|
||||
</head>
|
||||
|
||||
<body>
|
||||
<div class="container" x-data="dashboard()" x-init="init()">
|
||||
|
||||
<!-- Toast notification -->
|
||||
<div class="toast" x-show="actionMessage" x-transition :class="actionError ? 'toast-error' : 'toast-success'"
|
||||
style="display: none;">
|
||||
<span x-text="actionMessage"></span>
|
||||
</div>
|
||||
|
||||
<header>
|
||||
<div class="header-left">
|
||||
<div class="logo-dot"></div>
|
||||
<h1>RedUnits Control Panel</h1>
|
||||
</div>
|
||||
<div class="header-right">
|
||||
<!-- Live WS indicator -->
|
||||
<div class="indicator-group" :class="wsConnected ? 'ws-live' : 'ws-offline'">
|
||||
<span class="ws-dot"></span>
|
||||
<span x-text="wsConnected ? 'live' : 'reconnecting...'"></span>
|
||||
</div>
|
||||
<div class="last-update" x-text="lastUpdate"></div>
|
||||
<button @click="refresh()" class="refresh-btn" :disabled="loading">
|
||||
<span x-show="!loading">↻ refresh</span>
|
||||
<span x-show="loading">⏳...</span>
|
||||
</button>
|
||||
</div>
|
||||
</header>
|
||||
|
||||
<!-- SYSTEM STATUS SECTION -->
|
||||
<section class="section">
|
||||
<h2 class="section-title">SYSTEM STATUS</h2>
|
||||
<div class="system-grid">
|
||||
<div class="stat-card">
|
||||
<div class="stat-label">CPU usage</div>
|
||||
<div class="stat-value" x-text="system.cpu.percent + '%'"></div>
|
||||
<div class="progress-bar">
|
||||
<div class="progress-fill" :style="`width: ${system.cpu.percent}%`"
|
||||
:class="getProgressColor(system.cpu.percent)"></div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="stat-card">
|
||||
<div class="stat-label">Memory usage</div>
|
||||
<div class="stat-value" x-text="system.memory.percent + '%'"></div>
|
||||
<div class="progress-bar">
|
||||
<div class="progress-fill" :style="`width: ${system.memory.percent}%`"
|
||||
:class="getProgressColor(system.memory.percent)"></div>
|
||||
</div>
|
||||
<div class="stat-detail" x-text="`${system.memory.used_gb} GB / ${system.memory.total_gb} GB`">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="stat-card">
|
||||
<div class="stat-label">Docker containers</div>
|
||||
<div class="stat-value" x-text="`${system.containers.running} / ${system.containers.total}`"></div>
|
||||
<div class="progress-bar">
|
||||
<div class="progress-fill"
|
||||
:style="`width: ${system.containers.total > 0 ? (system.containers.running / system.containers.total * 100) : 0}%`"
|
||||
:class="system.containers.running === system.containers.total ? 'green' : 'red'"></div>
|
||||
</div>
|
||||
<div class="stat-detail">
|
||||
<span x-show="system.containers.running < system.containers.total"
|
||||
x-text="`${system.containers.total - system.containers.running} container${(system.containers.total - system.containers.running) > 1 ? 's' : ''} offline`"></span>
|
||||
<span x-show="system.containers.running === system.containers.total">All containers
|
||||
running</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="stat-card">
|
||||
<div class="stat-label">System uptime</div>
|
||||
<div class="stat-value" x-text="system.uptime.days + ' days'"></div>
|
||||
<div class="stat-status" :class="allServicesOnline() ? 'status-ok' : 'status-warn'">
|
||||
<span x-show="allServicesOnline()">✓ All services operational</span>
|
||||
<span x-show="!allServicesOnline()" style="display: flex; align-items: center; gap: 6px;">
|
||||
<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor"
|
||||
stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||
<circle cx="12" cy="12" r="10"></circle>
|
||||
<line x1="12" y1="8" x2="12" y2="12"></line>
|
||||
<line x1="12" y1="16" x2="12.01" y2="16"></line>
|
||||
</svg>
|
||||
Some services offline
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<!-- SERVICES SECTION -->
|
||||
<section class="section">
|
||||
<h2 class="section-title">SERVICES</h2>
|
||||
<div class="services-grid">
|
||||
<template x-for="service in services" :key="service.id">
|
||||
<div class="service-card" :class="!service.status.online ? 'card-offline' : ''">
|
||||
<div class="service-header">
|
||||
<div class="service-name">
|
||||
<span class="service-icon" x-text="service.icon"></span>
|
||||
<span x-text="service.name"></span>
|
||||
</div>
|
||||
<div class="status-badge"
|
||||
:class="service.status.online ? 'status-online' : 'status-offline'">
|
||||
<span x-text="service.status.online ? 'online' : 'offline'"></span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="service-description" x-text="service.description"></div>
|
||||
<div class="service-url" x-text="service.url"></div>
|
||||
|
||||
<!-- Online State Info -->
|
||||
<template x-if="service.status.online">
|
||||
<div class="service-metrics">
|
||||
<div class="metric">
|
||||
<span class="metric-label">CPU</span>
|
||||
<span class="metric-value"><span x-text="service.status.cpu_percent"></span>%</span>
|
||||
</div>
|
||||
<div class="metric">
|
||||
<span class="metric-label">RAM</span>
|
||||
<span class="metric-value">
|
||||
<span x-text="service.status.memory_mb"></span> <span
|
||||
class="metric-unit">MB</span>
|
||||
</span>
|
||||
</div>
|
||||
<div class="metric">
|
||||
<span class="metric-label">Uptime</span>
|
||||
<span class="metric-value" x-text="service.status.uptime"></span>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<!-- Offline State Info -->
|
||||
<template x-if="!service.status.online">
|
||||
<div class="offline-message">
|
||||
Container is not running
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<!-- Actions for Online -->
|
||||
<div class="service-actions" x-show="service.status.online" style="display: none;">
|
||||
<a :href="service.url" target="_blank" class="btn btn-primary">Open</a>
|
||||
<button class="btn btn-icon" @click="restartService(service.id)" title="Restart container">
|
||||
↻
|
||||
</button>
|
||||
<button class="btn btn-outline" @click="stopService(service.id)" title="Stop container">
|
||||
Stop
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<!-- Actions for Offline -->
|
||||
<div class="service-actions-center" x-show="!service.status.online" style="display: none;">
|
||||
<button class="btn btn-outline btn-full" @click="restartService(service.id)">
|
||||
↻ Restart service
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<!-- DOCUMENTS SECTION -->
|
||||
<section class="section" x-show="documents.length > 0" style="display: none;">
|
||||
<h2 class="section-title">QUICK LINKS & DOCS</h2>
|
||||
<div class="docs-list">
|
||||
<template x-for="doc in documents" :key="doc.id">
|
||||
<div class="doc-item" @click="openDocument(doc)">
|
||||
<span class="doc-icon">📄</span>
|
||||
<div class="doc-info">
|
||||
<div class="doc-title" x-text="doc.title"></div>
|
||||
<div class="doc-desc" x-text="doc.description"></div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<!-- DOCUMENT MODAL -->
|
||||
<div class="modal-overlay" x-show="showDocModal" style="display: none;">
|
||||
<div class="modal-content" @click.away="closeDocument()">
|
||||
<div class="modal-header">
|
||||
<h3 x-text="docTitle"></h3>
|
||||
<button class="modal-close" @click="closeDocument()">✕</button>
|
||||
</div>
|
||||
<div class="modal-body">
|
||||
<div x-show="docLoading" class="doc-loading">⏳ Loading document...</div>
|
||||
<div class="markdown-body" x-html="docContent" x-show="!docLoading"></div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<footer>
|
||||
<p>RedUnits Control Panel v1.3</p>
|
||||
</footer>
|
||||
</div>
|
||||
|
||||
<script src="/static/js/app.js"></script>
|
||||
</body>
|
||||
|
||||
</html>
|
||||
Loading…
x
Reference in New Issue
Block a user