Initial commit: DashCaddy v1.0

Full codebase including API server (32 modules + routes), dashboard frontend,
DashCA certificate distribution, installer script, and deployment skills.
This commit is contained in:
2026-03-05 02:26:12 -08:00
commit f61e85d9a7
337 changed files with 75282 additions and 0 deletions

242
status/api/README.md Normal file
View File

@@ -0,0 +1,242 @@
# SAMI-CLOUD Status Dashboard API
Cross-platform Node.js API server for managing Caddy reverse proxy and DNS records via REST APIs.
## Features
- **Cross-Platform**: Works on Windows, Linux, and macOS
- **API-Based**: Uses Caddy Admin API and Technitium DNS API (no PowerShell required)
- **App Deployment**: Deploy apps by creating DNS records and Caddy reverse proxy routes
- **App Deletion**: Clean removal of DNS records and Caddy routes
- **Automatic Rollback**: If deployment fails, automatically rolls back changes
## Prerequisites
1. **Node.js** (v14 or higher)
2. **Caddy** with Admin API enabled
3. **Technitium DNS Server** (optional, for DNS management)
## Installation
```bash
cd api
npm install
```
## Configuration
Set the following environment variables (or use defaults):
```bash
# Caddy Admin API endpoint (default: http://localhost:2019)
export CADDY_ADMIN_API=http://localhost:2019
# Technitium DNS Server API endpoint (default: http://192.168.254.204:5380)
export DNS_SERVER_API=http://192.168.254.204:5380
# Technitium DNS API token (required for DNS operations)
export TECHNITIUM_API_TOKEN=your_api_token_here
```
### Windows (PowerShell)
```powershell
$env:CADDY_ADMIN_API="http://localhost:2019"
$env:DNS_SERVER_API="http://192.168.254.204:5380"
$env:TECHNITIUM_API_TOKEN="your_api_token_here"
```
### Windows (Command Prompt)
```cmd
set CADDY_ADMIN_API=http://localhost:2019
set DNS_SERVER_API=http://192.168.254.204:5380
set TECHNITIUM_API_TOKEN=your_api_token_here
```
## Running the Server
```bash
npm start
```
Or directly:
```bash
node caddy-api.js
```
The server will start on port 3001.
## API Endpoints
### Deploy an App
```http
POST /api/apps/deploy
Content-Type: application/json
{
"appId": "myapp",
"config": {
"subdomain": "myapp",
"ip": "192.168.1.100",
"port": "8080",
"createDns": true,
"dnsType": "private",
"sslType": "internal"
}
}
```
**Response:**
```json
{
"success": true,
"message": "App myapp deployed successfully",
"url": "https://myapp.sami",
"domain": "myapp.sami",
"ip": "192.168.1.100",
"port": "8080",
"dnsCreated": true,
"caddyConfigured": true
}
```
### Delete an App
```http
POST /api/apps/delete
Content-Type: application/json
{
"domain": "myapp.sami",
"ip": "192.168.1.100"
}
```
### Get Services List
```http
GET /api/services
```
### Get Caddy Configuration
```http
GET /api/caddy/config
```
### Test API
```http
GET /api/caddy/test
```
### Health Check
```http
GET /health
```
## Caddy Configuration Requirements
Your Caddyfile should have the Admin API enabled:
```caddyfile
{
admin localhost:2019 {
origins localhost localhost:2019
}
}
```
For the status dashboard to proxy API requests, add this to your Caddyfile:
```caddyfile
status.sami {
tls internal
# API proxy to Node.js server
handle /api/* {
reverse_proxy localhost:3001
}
# Static site
root * /path/to/sites/status
file_server
}
```
## Getting Technitium DNS API Token
1. Open Technitium DNS web interface
2. Go to Settings → API
3. Create a new API token or copy existing one
4. Set it as the `TECHNITIUM_API_TOKEN` environment variable
## Deployment Flow
When deploying an app:
1. **Validate** - Checks required fields (appId, subdomain, ip)
2. **DNS Record** - Creates A record in DNS (if `createDns: true` and `dnsType: "private"`)
3. **Caddy Route** - Adds reverse proxy route via Caddy Admin API
4. **Rollback** - If Caddy configuration fails, removes DNS record
## Troubleshooting
### Caddy Admin API not accessible
- Verify Caddy is running
- Check that admin API is enabled in your Caddyfile
- Confirm the CADDY_ADMIN_API URL is correct
### DNS operations failing
- Verify TECHNITIUM_API_TOKEN is set correctly
- Check DNS_SERVER_API URL is accessible
- Ensure the API token has permissions to manage zones
### Routes not appearing in Caddy
- Check Caddy logs: `caddy logs`
- Verify the route was added: `curl http://localhost:2019/config/`
- Ensure the domain resolves correctly in DNS
## Production Deployment
For production use:
1. Set up environment variables persistently
2. Use a process manager (PM2, systemd, etc.)
3. Configure proper logging
4. Set up SSL/TLS for the API if exposed externally
### Using PM2
```bash
npm install -g pm2
pm2 start caddy-api.js --name sami-api
pm2 save
pm2 startup
```
### Using systemd (Linux)
Create `/etc/systemd/system/sami-api.service`:
```ini
[Unit]
Description=SAMI-CLOUD API Server
After=network.target
[Service]
Type=simple
User=caddy
WorkingDirectory=/path/to/sites/status/api
Environment="CADDY_ADMIN_API=http://localhost:2019"
Environment="DNS_SERVER_API=http://192.168.254.204:5380"
Environment="TECHNITIUM_API_TOKEN=your_token"
ExecStart=/usr/bin/node caddy-api.js
Restart=on-failure
[Install]
WantedBy=multi-user.target
```
Then:
```bash
sudo systemctl enable sami-api
sudo systemctl start sami-api
```
## License
MIT

362
status/api/caddy-api.js Normal file
View File

@@ -0,0 +1,362 @@
// Cross-platform Node.js API server for Caddy management
// Uses Caddy Admin API and Technitium DNS API directly
// Run with: node caddy-api.js
const express = require('express');
const path = require('path');
const cors = require('cors');
const fs = require('fs');
const app = express();
const PORT = 3001;
// Middleware
app.use(cors());
app.use(express.json());
app.use(express.static('public'));
// Configuration
const CADDY_ADMIN_API = process.env.CADDY_ADMIN_API || 'http://localhost:2019';
const DNS_SERVER_API = process.env.DNS_SERVER_API || 'http://192.168.254.204:5380';
const DNS_API_TOKEN = process.env.TECHNITIUM_API_TOKEN || '';
// Helper function to make HTTP requests
async function makeRequest(url, options = {}) {
const https = url.startsWith('https:') ? require('https') : require('http');
const urlObj = new URL(url);
return new Promise((resolve, reject) => {
const reqOptions = {
hostname: urlObj.hostname,
port: urlObj.port,
path: urlObj.pathname + urlObj.search,
method: options.method || 'GET',
headers: options.headers || {},
...options
};
const req = https.request(reqOptions, (res) => {
let data = '';
res.on('data', (chunk) => data += chunk);
res.on('end', () => {
try {
const parsed = JSON.parse(data);
resolve({ status: res.statusCode, data: parsed });
} catch (e) {
resolve({ status: res.statusCode, data: data });
}
});
});
req.on('error', reject);
if (options.body) {
req.write(typeof options.body === 'string' ? options.body : JSON.stringify(options.body));
}
req.end();
});
}
// Get current Caddy configuration
app.get('/api/caddy/config', async (req, res) => {
try {
const response = await makeRequest(`${CADDY_ADMIN_API}/config/`);
if (response.status === 200) {
res.json({
status: 'success',
config: response.data
});
} else {
res.status(response.status).json({
status: 'error',
message: 'Failed to get Caddy configuration',
details: response.data
});
}
} catch (error) {
console.error('Error getting Caddy config:', error);
res.status(500).json({
status: 'error',
message: error.message
});
}
});
// Get list of services (from apps.json + custom apps)
app.get('/api/services', async (req, res) => {
try {
const servicesPath = path.join(__dirname, '../apps.json');
if (fs.existsSync(servicesPath)) {
const servicesData = fs.readFileSync(servicesPath, 'utf8');
const services = JSON.parse(servicesData);
res.json({ status: 'success', services });
} else {
res.json({ status: 'success', services: [] });
}
} catch (error) {
console.error('Error reading services:', error);
res.status(500).json({
status: 'error',
message: error.message
});
}
});
// Add DNS record via Technitium API
async function addDnsRecord(domain, ipAddress, ttl = 3600) {
if (!DNS_API_TOKEN) {
throw new Error('DNS API token not configured. Set TECHNITIUM_API_TOKEN environment variable.');
}
const url = `${DNS_SERVER_API}/api/zones/records/add?token=${DNS_API_TOKEN}&domain=${domain}&type=A&ipAddress=${ipAddress}&ttl=${ttl}`;
console.log('Adding DNS record:', { domain, ipAddress, ttl });
const response = await makeRequest(url);
if (response.data.status === 'ok') {
return { success: true, message: 'DNS record added successfully' };
} else {
throw new Error(`DNS API error: ${response.data.message || 'Unknown error'}`);
}
}
// Delete DNS record via Technitium API
async function deleteDnsRecord(domain, ipAddress) {
if (!DNS_API_TOKEN) {
console.warn('DNS API token not configured. Skipping DNS deletion.');
return { success: true, message: 'DNS deletion skipped (no token)' };
}
const url = `${DNS_SERVER_API}/api/zones/records/delete?token=${DNS_API_TOKEN}&domain=${domain}&type=A&ipAddress=${ipAddress}`;
console.log('Deleting DNS record:', { domain, ipAddress });
const response = await makeRequest(url);
if (response.data.status === 'ok') {
return { success: true, message: 'DNS record deleted successfully' };
} else {
throw new Error(`DNS API error: ${response.data.message || 'Unknown error'}`);
}
}
// Add route to Caddy via Admin API
async function addCaddyRoute(domain, upstreamUrl, useTls = true) {
// Build Caddy route configuration
const routeConfig = {
"@id": domain,
"match": [{
"host": [domain]
}],
"handle": [{
"handler": "reverse_proxy",
"upstreams": [{
"dial": upstreamUrl.replace(/^https?:\/\//, '')
}]
}],
"terminal": true
};
// If using internal TLS, we need to add TLS configuration
if (useTls) {
// Caddy handles TLS automatically for matched domains
// Internal CA is configured in the global Caddyfile
}
console.log('Adding Caddy route:', JSON.stringify(routeConfig, null, 2));
// Add the route to the HTTP server
const response = await makeRequest(`${CADDY_ADMIN_API}/config/apps/http/servers/srv0/routes/0`, {
method: 'POST',
headers: {
'Content-Type': 'application/json'
},
body: JSON.stringify(routeConfig)
});
if (response.status === 200 || response.status === 201) {
return { success: true, message: 'Caddy route added successfully' };
} else {
throw new Error(`Caddy API error: ${JSON.stringify(response.data)}`);
}
}
// Deploy app endpoint - handles DNS and Caddy configuration via APIs
app.post('/api/apps/deploy', async (req, res) => {
try {
const { appId, config } = req.body;
const { subdomain, ip, createDns, port, sslType, dnsType } = config;
console.log('Deploying app:', { appId, config });
// Validate required fields
if (!appId || !subdomain || !ip) {
return res.status(400).json({
success: false,
message: 'Missing required fields: appId, subdomain, ip'
});
}
// Build the full domain
const domain = subdomain.includes('.') ? subdomain : `${subdomain}.sami`;
const finalPort = port || '80';
const upstreamUrl = `${ip}:${finalPort}`;
// Step 1: Add DNS record if requested (private DNS)
if (createDns && dnsType === 'private') {
try {
await addDnsRecord(domain, ip);
} catch (dnsError) {
console.error('DNS creation failed:', dnsError);
return res.status(500).json({
success: false,
message: `DNS creation failed: ${dnsError.message}`,
step: 'dns'
});
}
}
// Step 2: Add route to Caddy via Admin API
try {
const useTls = sslType === 'internal';
await addCaddyRoute(domain, upstreamUrl, useTls);
} catch (caddyError) {
console.error('Caddy route addition failed:', caddyError);
// Rollback DNS if it was created
if (createDns && dnsType === 'private') {
try {
await deleteDnsRecord(domain, ip);
} catch (rollbackError) {
console.error('DNS rollback failed:', rollbackError);
}
}
return res.status(500).json({
success: false,
message: `Caddy configuration failed: ${caddyError.message}`,
step: 'caddy'
});
}
// Step 3: Return success response
res.json({
success: true,
message: `App ${appId} deployed successfully`,
url: `https://${domain}`,
domain: domain,
ip: ip,
port: finalPort,
containerId: null,
dnsCreated: createDns && dnsType === 'private',
caddyConfigured: true
});
} catch (error) {
console.error('Deployment error:', error);
res.status(500).json({
success: false,
message: error.message
});
}
});
// Delete app endpoint - removes DNS and Caddy configuration
app.post('/api/apps/delete', async (req, res) => {
try {
const { domain, ip } = req.body;
if (!domain) {
return res.status(400).json({
success: false,
message: 'Domain is required'
});
}
console.log('Deleting app:', { domain, ip });
// Step 1: Remove from Caddy
try {
const response = await makeRequest(`${CADDY_ADMIN_API}/id/${domain}`, {
method: 'DELETE'
});
if (response.status !== 200) {
console.warn('Caddy route deletion warning:', response.data);
}
} catch (caddyError) {
console.error('Caddy route deletion failed:', caddyError);
// Continue anyway to try DNS deletion
}
// Step 2: Remove DNS record if IP provided
if (ip) {
try {
await deleteDnsRecord(domain, ip);
} catch (dnsError) {
console.error('DNS deletion failed:', dnsError);
return res.status(500).json({
success: false,
message: `DNS deletion failed: ${dnsError.message}`,
caddyDeleted: true,
dnsDeleted: false
});
}
}
res.json({
success: true,
message: 'App deleted successfully',
domain: domain,
caddyDeleted: true,
dnsDeleted: !!ip
});
} catch (error) {
console.error('Deletion error:', error);
res.status(500).json({
success: false,
message: error.message
});
}
});
// Test endpoint
app.get('/api/caddy/test', (req, res) => {
res.json({
status: 'success',
message: 'Caddy API is running',
timestamp: new Date().toISOString(),
platform: process.platform,
caddyAdminApi: CADDY_ADMIN_API,
dnsServerApi: DNS_SERVER_API,
dnsTokenConfigured: !!DNS_API_TOKEN
});
});
// Health check
app.get('/health', (req, res) => {
res.json({ status: 'healthy', timestamp: new Date().toISOString() });
});
// Start server
app.listen(PORT, () => {
console.log(`\n====================================`);
console.log(`Caddy API server running on http://localhost:${PORT}`);
console.log(`====================================`);
console.log(`Caddy Admin API: ${CADDY_ADMIN_API}`);
console.log(`DNS Server API: ${DNS_SERVER_API}`);
console.log(`DNS Token: ${DNS_API_TOKEN ? '✓ Configured' : '✗ Not configured'}`);
console.log(`\nEndpoints:`);
console.log(` POST /api/apps/deploy - Deploy an app (DNS + Caddy)`);
console.log(` POST /api/apps/delete - Delete an app (DNS + Caddy)`);
console.log(` GET /api/services - Get list of services`);
console.log(` GET /api/caddy/config - Get current Caddy configuration`);
console.log(` GET /api/caddy/test - Test API connectivity`);
console.log(` GET /health - Health check`);
console.log(`====================================\n`);
});
module.exports = app;

View File

@@ -0,0 +1,206 @@
# Caddy Configuration Manager for Windows
# This script adds new service configurations to your Caddyfile
param(
[string]$Config,
[string]$Subdomain,
[string]$CaddyfilePath = "C:\caddy\Caddyfile",
[bool]$ReloadCaddy = $true
)
# Function to write JSON response
function Write-JsonResponse {
param($Status, $Message, $Data = $null)
$response = @{
status = $Status
message = $Message
timestamp = (Get-Date).ToString("yyyy-MM-dd HH:mm:ss")
}
if ($Data) {
$response.data = $Data
}
return $response | ConvertTo-Json
}
# Function to extract CA names from Caddyfile
function Get-CaddyfileCAs {
param([string]$CaddyfilePath)
try {
Write-Host "DEBUG: Checking file path: $CaddyfilePath"
if (-not (Test-Path $CaddyfilePath)) {
Write-Host "DEBUG: File not found"
return @()
}
$content = Get-Content $CaddyfilePath -Raw
Write-Host "DEBUG: File content length: $($content.Length)"
Write-Host "DEBUG: First 200 chars: $($content.Substring(0, [Math]::Min(200, $content.Length)))"
$caNames = @()
# Pattern 1: PKI block with CA definitions - pki { ca ca_name { name "Friendly Name" } }
$pkiPattern = 'pki\s*\{[^}]*?ca\s+([^\s\{]+)\s*\{([^}]*?)\}'
Write-Host "DEBUG: Searching for PKI pattern: $pkiPattern"
$pkiMatches = [regex]::Matches($content, $pkiPattern, [System.Text.RegularExpressions.RegexOptions]::IgnoreCase -bor [System.Text.RegularExpressions.RegexOptions]::Singleline)
Write-Host "DEBUG: PKI matches found: $($pkiMatches.Count)"
foreach ($match in $pkiMatches) {
$caId = $match.Groups[1].Value
$caBlock = $match.Groups[2].Value
Write-Host "DEBUG: Found CA ID: $caId"
Write-Host "DEBUG: CA Block: $caBlock"
# Try to extract the friendly name from within the CA block
$nameMatch = [regex]::Match($caBlock, 'name\s+"([^"]+)"', [System.Text.RegularExpressions.RegexOptions]::IgnoreCase)
if ($nameMatch.Success) {
$friendlyName = $nameMatch.Groups[1].Value
Write-Host "DEBUG: Found friendly name: $friendlyName"
$caNames += "$caId ($friendlyName)"
} else {
Write-Host "DEBUG: No friendly name found, using ID only"
$caNames += $caId
}
}
# Pattern 2: tls { ca ca_name }
$matches1 = [regex]::Matches($content, 'tls\s*\{\s*ca\s+([^\s\}]+)', [System.Text.RegularExpressions.RegexOptions]::IgnoreCase)
Write-Host "DEBUG: TLS block matches: $($matches1.Count)"
foreach ($match in $matches1) {
$caNames += $match.Groups[1].Value
}
# Pattern 3: tls ca_name (direct)
$matches2 = [regex]::Matches($content, 'tls\s+([^\s\{]+)', [System.Text.RegularExpressions.RegexOptions]::IgnoreCase)
Write-Host "DEBUG: Direct TLS matches: $($matches2.Count)"
foreach ($match in $matches2) {
$caName = $match.Groups[1].Value
if ($caName -ne "internal" -and $caName -ne "off") {
$caNames += $caName
}
}
Write-Host "DEBUG: Total CAs found: $($caNames.Count)"
Write-Host "DEBUG: CA list: $($caNames -join ', ')"
# Remove duplicates and return
return $caNames | Sort-Object | Get-Unique
}
catch {
Write-Host "DEBUG: Exception in Get-CaddyfileCAs: $($_.Exception.Message)"
Write-Host "Error reading CAs from Caddyfile: $($_.Exception.Message)"
return @()
}
}
# Handle get-cas command
if ($args[0] -eq "get-cas") {
$CaddyfilePath = if ($args[1]) { $args[1] } else { "C:\caddy\Caddyfile" }
Write-Host "DEBUG: get-cas command received"
Write-Host "DEBUG: Caddyfile path: $CaddyfilePath"
Write-Host "DEBUG: File exists: $(Test-Path $CaddyfilePath)"
try {
$cas = Get-CaddyfileCAs -CaddyfilePath $CaddyfilePath
Write-Host "DEBUG: CAs found: $($cas -join ', ')"
$response = @{
status = "success"
message = "CAs retrieved successfully"
data = @{
cas = $cas
count = $cas.Count
caddyfilePath = $CaddyfilePath
}
timestamp = (Get-Date).ToString("yyyy-MM-dd HH:mm:ss")
}
Write-Output ($response | ConvertTo-Json)
exit 0
}
catch {
Write-Host "DEBUG: Error occurred: $($_.Exception.Message)"
$response = @{
status = "error"
message = "Failed to retrieve CAs: $($_.Exception.Message)"
timestamp = (Get-Date).ToString("yyyy-MM-dd HH:mm:ss")
}
Write-Output ($response | ConvertTo-Json)
exit 1
}
}
# Main configuration addition logic
try {
# Validate required parameters for config addition
if (-not $Config -or -not $Subdomain) {
Write-Output (Write-JsonResponse "error" "Config and Subdomain parameters are required")
exit 1
}
# Check if Caddyfile exists
if (-not (Test-Path $CaddyfilePath)) {
Write-Output (Write-JsonResponse "error" "Caddyfile not found at: $CaddyfilePath")
exit 1
}
# Read existing Caddyfile
$existingConfig = Get-Content $CaddyfilePath -Raw -ErrorAction Stop
# Check if subdomain already exists
if ($existingConfig -match "$Subdomain\.sami\s*\{") {
Write-Output (Write-JsonResponse "error" "Subdomain '$Subdomain.sami' already exists in Caddyfile")
exit 1
}
# Create backup
$backupPath = "$CaddyfilePath.backup.$(Get-Date -Format 'yyyyMMdd-HHmmss')"
Copy-Item $CaddyfilePath $backupPath -ErrorAction Stop
Write-Host "Backup created: $backupPath"
# Append new configuration
$newContent = $existingConfig.TrimEnd() + "`n`n" + $Config.TrimEnd() + "`n"
Set-Content -Path $CaddyfilePath -Value $newContent -NoNewline -ErrorAction Stop
Write-Host "Configuration added successfully"
# Reload Caddy if requested
if ($ReloadCaddy) {
Write-Host "Reloading Caddy..."
# Try to reload Caddy
$reloadResult = & caddy reload --config $CaddyfilePath 2>&1
if ($LASTEXITCODE -eq 0) {
Write-Host "Caddy reloaded successfully"
Write-Output (Write-JsonResponse "success" "Configuration added and Caddy reloaded successfully" @{
backup = $backupPath
subdomain = "$Subdomain.sami"
})
} else {
Write-Host "Caddy reload failed: $reloadResult"
# Restore backup if reload failed
Copy-Item $backupPath $CaddyfilePath -Force
Write-Output (Write-JsonResponse "error" "Caddy reload failed. Configuration rolled back. Error: $reloadResult")
exit 1
}
} else {
Write-Output (Write-JsonResponse "success" "Configuration added successfully (Caddy not reloaded)" @{
backup = $backupPath
subdomain = "$Subdomain.sami"
})
}
} catch {
Write-Output (Write-JsonResponse "error" "Error: $($_.Exception.Message)")
exit 1
}

View File

@@ -0,0 +1,63 @@
@echo off
title SAMI Caddy API Server
echo ========================================
echo Installing SAMI Caddy API Server...
echo ========================================
REM Check if Node.js is installed
echo Checking for Node.js...
node --version >nul 2>&1
if %errorlevel% neq 0 (
echo.
echo ERROR: Node.js is not installed or not in PATH
echo Please install Node.js from https://nodejs.org/
echo.
pause
exit /b 1
)
echo Node.js found:
node --version
echo.
echo Installing dependencies...
echo.
REM Install npm dependencies
npm install
if %errorlevel% neq 0 (
echo.
echo ERROR: Failed to install dependencies
echo Check the error messages above
echo.
pause
exit /b 1
)
echo.
echo ========================================
echo Dependencies installed successfully!
echo ========================================
echo.
echo Starting Caddy API server...
echo.
echo Server URL: http://localhost:3001
echo Test URL: http://localhost:3001/api/caddy/test
echo.
echo Press Ctrl+C to stop the server
echo ========================================
echo.
REM Start the server and keep window open on error
npm start
if %errorlevel% neq 0 (
echo.
echo ERROR: Server failed to start
echo Check the error messages above
echo.
)
echo.
echo Server stopped.
pause

1227
status/api/package-lock.json generated Normal file

File diff suppressed because it is too large Load Diff

21
status/api/package.json Normal file
View File

@@ -0,0 +1,21 @@
{
"name": "sami-caddy-api",
"version": "2.0.0",
"description": "Cross-platform API server for managing Caddy and DNS via REST APIs",
"main": "caddy-api.js",
"scripts": {
"start": "node caddy-api.js",
"dev": "nodemon caddy-api.js",
"test": "node test-api.js"
},
"dependencies": {
"express": "^4.18.2",
"cors": "^2.8.5"
},
"devDependencies": {
"nodemon": "^3.0.1"
},
"keywords": ["caddy", "api", "dns", "technitium", "reverse-proxy", "cross-platform", "sami-cloud"],
"author": "SAMI-CLOUD",
"license": "MIT"
}

44
status/api/start.bat Normal file
View File

@@ -0,0 +1,44 @@
@echo off
REM Quick start script for SAMI-CLOUD API Server (Windows)
echo ====================================
echo SAMI-CLOUD API Server
echo ====================================
echo.
REM Check if Node.js is installed
where node >nul 2>nul
if %errorlevel% neq 0 (
echo Error: Node.js is not installed or not in PATH
echo Please install Node.js from https://nodejs.org/
pause
exit /b 1
)
REM Check if node_modules exists
if not exist "node_modules" (
echo Installing dependencies...
call npm install
echo.
)
REM Check environment variables
if "%CADDY_ADMIN_API%"=="" (
echo Warning: CADDY_ADMIN_API not set, using default: http://localhost:2019
set CADDY_ADMIN_API=http://localhost:2019
)
if "%DNS_SERVER_API%"=="" (
echo Warning: DNS_SERVER_API not set, using default: http://192.168.254.204:5380
set DNS_SERVER_API=http://192.168.254.204:5380
)
if "%TECHNITIUM_API_TOKEN%"=="" (
echo Warning: TECHNITIUM_API_TOKEN not set - DNS operations will fail
echo Set it with: set TECHNITIUM_API_TOKEN=your_token
echo.
)
echo Starting API server...
echo.
node caddy-api.js

42
status/api/start.sh Normal file
View File

@@ -0,0 +1,42 @@
#!/bin/bash
# Quick start script for SAMI-CLOUD API Server (Linux/macOS)
echo "===================================="
echo "SAMI-CLOUD API Server"
echo "===================================="
echo ""
# Check if Node.js is installed
if ! command -v node &> /dev/null; then
echo "Error: Node.js is not installed or not in PATH"
echo "Please install Node.js from https://nodejs.org/"
exit 1
fi
# Check if node_modules exists
if [ ! -d "node_modules" ]; then
echo "Installing dependencies..."
npm install
echo ""
fi
# Check environment variables
if [ -z "$CADDY_ADMIN_API" ]; then
echo "Warning: CADDY_ADMIN_API not set, using default: http://localhost:2019"
export CADDY_ADMIN_API="http://localhost:2019"
fi
if [ -z "$DNS_SERVER_API" ]; then
echo "Warning: DNS_SERVER_API not set, using default: http://192.168.254.204:5380"
export DNS_SERVER_API="http://192.168.254.204:5380"
fi
if [ -z "$TECHNITIUM_API_TOKEN" ]; then
echo "Warning: TECHNITIUM_API_TOKEN not set - DNS operations will fail"
echo "Set it with: export TECHNITIUM_API_TOKEN=your_token"
echo ""
fi
echo "Starting API server..."
echo ""
node caddy-api.js

72
status/api/test-api.js Normal file
View File

@@ -0,0 +1,72 @@
// Simple test script to verify API connectivity
const http = require('http');
const API_URL = 'http://localhost:3001';
function makeRequest(path) {
return new Promise((resolve, reject) => {
http.get(`${API_URL}${path}`, (res) => {
let data = '';
res.on('data', (chunk) => data += chunk);
res.on('end', () => {
try {
resolve({ status: res.statusCode, data: JSON.parse(data) });
} catch (e) {
resolve({ status: res.statusCode, data: data });
}
});
}).on('error', reject);
});
}
async function runTests() {
console.log('Testing SAMI-CLOUD API...\n');
// Test 1: Health Check
console.log('1. Testing health endpoint...');
try {
const health = await makeRequest('/health');
if (health.status === 200) {
console.log(' ✓ Health check passed');
} else {
console.log(' ✗ Health check failed:', health.status);
}
} catch (error) {
console.log(' ✗ Health check error:', error.message);
}
// Test 2: API Test Endpoint
console.log('\n2. Testing API test endpoint...');
try {
const test = await makeRequest('/api/caddy/test');
if (test.status === 200) {
console.log(' ✓ API test passed');
console.log(' Platform:', test.data.platform);
console.log(' Caddy Admin API:', test.data.caddyAdminApi);
console.log(' DNS Server API:', test.data.dnsServerApi);
console.log(' DNS Token:', test.data.dnsTokenConfigured ? 'Configured' : 'Not configured');
} else {
console.log(' ✗ API test failed:', test.status);
}
} catch (error) {
console.log(' ✗ API test error:', error.message);
}
// Test 3: Services Endpoint
console.log('\n3. Testing services endpoint...');
try {
const services = await makeRequest('/api/services');
if (services.status === 200) {
console.log(' ✓ Services endpoint passed');
console.log(' Found', services.data.services.length, 'services');
} else {
console.log(' ✗ Services endpoint failed:', services.status);
}
} catch (error) {
console.log(' ✗ Services endpoint error:', error.message);
}
console.log('\nTests complete!');
}
runTests().catch(console.error);

View File

@@ -0,0 +1,84 @@
@echo off
title SAMI Caddy API Server - Debug Mode
echo ========================================
echo SAMI Caddy API Server - Debug Mode
echo ========================================
cd /d "%~dp0"
echo Current directory: %CD%
echo.
echo Checking Node.js...
node --version
if %errorlevel% neq 0 (
echo ERROR: Node.js not found
pause
exit /b 1
)
echo.
echo Checking files...
if exist "caddy-api.js" (
echo ✓ caddy-api.js found
) else (
echo ✗ caddy-api.js NOT found
pause
exit /b 1
)
if exist "package.json" (
echo ✓ package.json found
) else (
echo ✗ package.json NOT found
pause
exit /b 1
)
if exist "caddy-manager.ps1" (
echo ✓ caddy-manager.ps1 found
) else (
echo ✗ caddy-manager.ps1 NOT found
pause
exit /b 1
)
echo.
echo Installing dependencies...
npm install
if %errorlevel% neq 0 (
echo ERROR: npm install failed
pause
exit /b 1
)
echo.
echo ========================================
echo Starting server with error capture...
echo ========================================
echo Server will run on: http://localhost:3001
echo Test endpoint: http://localhost:3001/api/caddy/test
echo.
echo If server starts successfully, you'll see "Caddy API server running..."
echo Press Ctrl+C to stop the server
echo ========================================
echo.
REM Capture both stdout and stderr
node caddy-api.js 2>&1
set SERVER_EXIT_CODE=%errorlevel%
echo.
echo ========================================
echo Server exited with code: %SERVER_EXIT_CODE%
echo ========================================
if %SERVER_EXIT_CODE% neq 0 (
echo ERROR: Server failed to start or crashed
echo Check the error messages above
) else (
echo Server stopped normally
)
echo.
echo Press any key to close this window...
pause >nul

62
status/apps.json Normal file
View File

@@ -0,0 +1,62 @@
[
{
"id": "plex",
"name": "Plex",
"logo": "assets/plex.png",
"url": "https://plex.sami"
},
{
"id": "jellyfin",
"name": "Jellyfin",
"logo": "📺",
"url": "https://jellyfin.sami"
},
{
"id": "emby",
"name": "Emby",
"logo": "🎬",
"url": "https://emby.sami"
},
{
"id": "router",
"name": "Router UI",
"logo": "assets/router.png",
"url": "https://router.sami"
},
{
"id": "chat",
"name": "Chat",
"logo": "assets/chat.png",
"url": "https://chat.sami"
},
{
"id": "torrent",
"name": "qBittorrent",
"logo": "assets/qBittorrent.png",
"url": "https://torrent.sami"
},
{
"id": "sync",
"name": "Syncthing",
"logo": "assets/syncthing.png",
"url": "https://sync.sami"
},
{
"id": "radarr",
"name": "Radarr",
"logo": "assets/radarr.png",
"url": "https://radarr.sami"
},
{
"id": "sonarr",
"name": "Sonarr",
"logo": "assets/sonarr.png",
"url": "https://sonarr.sami"
},
{
"id": "prowlarr",
"name": "Prowlarr",
"logo": "assets/prowlarr.png",
"url": "https://prowlarr.sami"
}
]

21
status/assets/.htaccess Normal file
View File

@@ -0,0 +1,21 @@
# Font file headers to prevent sanitizer issues
<FilesMatch "\.(woff2|woff|ttf|eot)$">
Header set Access-Control-Allow-Origin "*"
Header set Access-Control-Allow-Methods "GET, POST, OPTIONS"
Header set Access-Control-Allow-Headers "Content-Type"
Header set Cache-Control "public, max-age=31536000"
# Proper MIME types
<IfModule mod_mime.c>
AddType font/woff2 .woff2
AddType font/woff .woff
AddType font/ttf .ttf
AddType application/vnd.ms-fontobject .eot
</IfModule>
</FilesMatch>
# Prevent direct access to font conversion scripts
<FilesMatch "\.(py|bat)$">
Order allow,deny
Deny from all
</FilesMatch>

Binary file not shown.

After

Width:  |  Height:  |  Size: 356 KiB

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 972 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 12 KiB

BIN
status/assets/chat.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 10 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 53 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 356 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 15 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 25 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 21 KiB

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 972 KiB

BIN
status/assets/emby.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 9.0 KiB

BIN
status/assets/favicon.ico Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 638 B

View File

@@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 64 64"><rect width="64" height="64" rx="12" fill="#0e1116"/><path d="M16 38h20a8 8 0 1 0-1.8-15.7A9.5 9.5 0 0 0 12 30c0 4.4 3.6 8 8 8z" fill="#8FD6FF"/></svg>

After

Width:  |  Height:  |  Size: 211 B

91
status/assets/fonts.css Normal file
View File

@@ -0,0 +1,91 @@
/* Sami Sans Font Family - External CSS */
@font-face {
font-family: 'Sami Sans';
src: url('fonts/SamiSans-Regular.woff2') format('woff2'),
url('fonts/SamiSans-Regular.ttf') format('truetype');
font-weight: 400;
font-style: normal;
font-display: swap;
}
@font-face {
font-family: 'Sami Sans';
src: url('fonts/SamiSans-Regular.woff2') format('woff2'),
url('fonts/SamiSans-Italic.ttf') format('truetype');
font-weight: 400;
font-style: italic;
font-display: swap;
}
@font-face {
font-family: 'Sami Sans';
src: url('fonts/SamiSans-Medium.woff2') format('woff2'),
url('fonts/SamiSans-Medium.ttf') format('truetype');
font-weight: 500;
font-style: normal;
font-display: swap;
}
@font-face {
font-family: 'Sami Sans';
src: url('fonts/SamiSans-SemiBold.woff2') format('woff2'),
url('fonts/SamiSans-SemiBold.ttf') format('truetype');
font-weight: 600;
font-style: normal;
font-display: swap;
}
@font-face {
font-family: 'Sami Sans';
src: url('fonts/SamiSans-Bold.woff2') format('woff2'),
url('fonts/SamiSans-Bold.ttf') format('truetype');
font-weight: 700;
font-style: normal;
font-display: swap;
}
@font-face {
font-family: 'Sami Sans';
src: url('fonts/SamiSans-ExtraBold.woff2') format('woff2'),
url('fonts/SamiSans-ExtraBold.ttf') format('truetype');
font-weight: 800;
font-style: normal;
font-display: swap;
}
@font-face {
font-family: 'Sami Sans';
src: url('fonts/SamiSans-Black.woff2') format('woff2'),
url('fonts/SamiSans-Black.ttf') format('truetype');
font-weight: 900;
font-style: normal;
font-display: swap;
}
@font-face {
font-family: 'Sami Sans';
src: url('fonts/SamiSans-Light.woff2') format('woff2'),
url('fonts/SamiSans-Light.ttf') format('truetype');
font-weight: 300;
font-style: normal;
font-display: swap;
}
@font-face {
font-family: 'Sami Sans';
src: url('fonts/SamiSans-ExtraLight.woff2') format('woff2'),
url('fonts/SamiSans-ExtraLight.ttf') format('truetype');
font-weight: 200;
font-style: normal;
font-display: swap;
}
@font-face {
font-family: 'Sami Sans';
src: url('fonts/SamiSans-Thin.woff2') format('woff2'),
url('fonts/SamiSans-Thin.ttf') format('truetype');
font-weight: 100;
font-style: normal;
font-display: swap;
}

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

BIN
status/assets/icon-192.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 13 KiB

BIN
status/assets/icon-512.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 53 KiB

BIN
status/assets/jellyfin.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 45 KiB

BIN
status/assets/nginx.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 13 KiB

BIN
status/assets/pics.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 122 KiB

BIN
status/assets/plex.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 10 KiB

BIN
status/assets/prowlarr.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 98 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 135 KiB

BIN
status/assets/radarr.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 18 KiB

BIN
status/assets/router.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 41 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.2 MiB

View File

@@ -0,0 +1,20 @@
{
"name": "SAMI-CLOUD Status",
"short_name": "SAMI-CLOUD",
"start_url": "index.html",
"display": "standalone",
"background_color": "#0b0f1a",
"theme_color": "#0e1116",
"icons": [
{
"src": "assets/icon-192.png",
"sizes": "192x192",
"type": "image/png"
},
{
"src": "assets/icon-512.png",
"sizes": "512x512",
"type": "image/png"
}
]
}

BIN
status/assets/sonarr.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 13 KiB

Binary file not shown.

BIN
status/assets/syncthing.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 27 KiB

Some files were not shown because too many files have changed in this diff Show More