SystemD Services Update - Complete: ✅ SERVICE INSTALLATION IMPROVED: Changed from copying to symbolic linking - setup.sh: install_services function now uses ln -sf instead of cp - Benefit: Service files automatically update when originals change - Impact: Improved maintainability and consistency ✅ SYMBOLIC LINK ADVANTAGES: 🔗 Auto-Update: Changes to /opt/aitbc/systemd/*.service automatically reflected in /etc/systemd/system/ 🔄 Synchronization: Installed services always match source files 📝 Maintenance: Single source of truth for service configurations 🎯 Consistency: No divergence between source and installed services ✅ BEFORE vs AFTER: ❌ Before: cp '/opt/aitbc/systemd/' /etc/systemd/system/ - Static copies that don't update - Manual intervention required for updates - Potential divergence between source and installed ✅ After: ln -sf '/opt/aitbc/systemd/' /etc/systemd/system/ - Dynamic symbolic links - Automatic updates when source changes - Always synchronized with source files ✅ TECHNICAL DETAILS: 🔗 ln -sf: Force symbolic link creation (overwrites existing) 📁 Source: /opt/aitbc/systemd/ 📁 Target: /etc/systemd/system/ 🔄 Update: Changes propagate automatically 🎯 Purpose: Maintain service configuration consistency ✅ MAINTENANCE BENEFITS: ✅ Single Source: Update only /opt/aitbc/systemd/ files ✅ Auto-Propagation: Changes automatically apply to installed services ✅ No Manual Sync: No need to manually copy updated files ✅ Consistent State: Installed services always match source ✅ USE CASES IMPROVED: 🔧 Service Updates: Configuration changes apply immediately 🔧 Debugging: Edit source files, changes reflect in running services 🔧 Development: Test service changes without re-copying 🔧 Deployment: Service updates propagate automatically RESULT: Successfully changed systemd service installation to use symbolic links, ensuring automatic updates and eliminating potential configuration divergence between source and installed services.
396 lines
10 KiB
Bash
Executable File
396 lines
10 KiB
Bash
Executable File
#!/bin/bash
|
|
|
|
# AITBC Local Setup Script
|
|
# Sets up AITBC services on a new host with systemd
|
|
|
|
set -e
|
|
|
|
# Colors for output
|
|
RED='\033[0;31m'
|
|
GREEN='\033[0;32m'
|
|
YELLOW='\033[1;33m'
|
|
BLUE='\033[0;34m'
|
|
NC='\033[0m' # No Color
|
|
|
|
# Logging function
|
|
log() {
|
|
echo -e "${BLUE}[$(date +'%Y-%m-%d %H:%M:%S')]${NC} $1"
|
|
}
|
|
|
|
error() {
|
|
echo -e "${RED}[ERROR]${NC} $1"
|
|
exit 1
|
|
}
|
|
|
|
success() {
|
|
echo -e "${GREEN}[SUCCESS]${NC} $1"
|
|
}
|
|
|
|
warning() {
|
|
echo -e "${YELLOW}[WARNING]${NC} $1"
|
|
}
|
|
|
|
# Check if running as root
|
|
check_root() {
|
|
if [ "$EUID" -ne 0 ]; then
|
|
error "This script must be run as root (use sudo)"
|
|
fi
|
|
}
|
|
|
|
# Check prerequisites
|
|
check_prerequisites() {
|
|
log "Checking prerequisites..."
|
|
|
|
# Check if required tools are installed
|
|
command -v python3 >/dev/null 2>&1 || error "Python 3 is not installed"
|
|
command -v pip3 >/dev/null 2>&1 || error "pip3 is not installed"
|
|
command -v git >/dev/null 2>&1 || error "git is not installed"
|
|
command -v systemctl >/dev/null 2>&1 || error "systemctl is not available"
|
|
|
|
# Check Python version
|
|
python_version=$(python3 -c "import sys; print(f'{sys.version_info.major}.{sys.version_info.minor}.{sys.version_info.micro}')")
|
|
if [ "$(printf '%s\n' "3.13.5" "$python_version" | sort -V | head -n1)" != "3.13.5" ]; then
|
|
error "Python 3.13.5+ is required, found $python_version"
|
|
fi
|
|
|
|
success "Prerequisites check passed"
|
|
}
|
|
|
|
# Clone repository
|
|
clone_repo() {
|
|
log "Cloning AITBC repository..."
|
|
|
|
# Remove existing installation if present
|
|
if [ -d "/opt/aitbc" ]; then
|
|
warning "Removing existing /opt/aitbc"
|
|
rm -rf /opt/aitbc
|
|
fi
|
|
|
|
# Clone repository
|
|
cd /opt
|
|
git clone https://github.com/aitbc/aitbc.git aitbc || {
|
|
# Try alternative GitHub URL
|
|
git clone git@github.com:aitbc/aitbc.git aitbc || error "Failed to clone repository"
|
|
}
|
|
|
|
cd /opt/aitbc
|
|
success "Repository cloned successfully"
|
|
}
|
|
|
|
# Setup runtime directories
|
|
setup_runtime_directories() {
|
|
log "Setting up runtime directories..."
|
|
|
|
# Create standard Linux directories
|
|
directories=(
|
|
"/var/lib/aitbc"
|
|
"/var/lib/aitbc/keystore"
|
|
"/var/lib/aitbc/data"
|
|
"/var/lib/aitbc/logs"
|
|
"/etc/aitbc"
|
|
"/var/log/aitbc"
|
|
)
|
|
|
|
for dir in "${directories[@]}"; do
|
|
mkdir -p "$dir"
|
|
log "Created directory: $dir"
|
|
done
|
|
|
|
# Set permissions
|
|
chmod 755 /var/lib/aitbc
|
|
chmod 700 /var/lib/aitbc/keystore # Secure keystore
|
|
chmod 755 /var/lib/aitbc/data
|
|
chmod 755 /var/lib/aitbc/logs
|
|
chmod 755 /etc/aitbc
|
|
chmod 755 /var/log/aitbc
|
|
|
|
# Set ownership
|
|
chown root:root /var/lib/aitbc
|
|
chown root:root /var/lib/aitbc/keystore
|
|
chown root:root /var/lib/aitbc/data
|
|
chown root:root /var/lib/aitbc/logs
|
|
chown root:root /etc/aitbc
|
|
chown root:root /var/log/aitbc
|
|
|
|
# Create README files
|
|
echo "# AITBC Runtime Data Directory" > /var/lib/aitbc/README.md
|
|
echo "# Keystore for blockchain keys (SECURE)" > /var/lib/aitbc/keystore/README.md
|
|
echo "# Application databases" > /var/lib/aitbc/data/README.md
|
|
echo "# Application logs" > /var/lib/aitbc/logs/README.md
|
|
echo "# AITBC Configuration Files" > /etc/aitbc/README.md
|
|
|
|
# Create symlink for standard logging
|
|
ln -sf /var/lib/aitbc/logs /var/log/aitbc
|
|
|
|
success "Runtime directories setup completed"
|
|
}
|
|
|
|
# Setup Python virtual environments
|
|
setup_venvs() {
|
|
log "Setting up Python virtual environments..."
|
|
|
|
# Create central virtual environment if it doesn't exist
|
|
if [ ! -d "/opt/aitbc/venv" ]; then
|
|
log "Creating central virtual environment..."
|
|
cd /opt/aitbc
|
|
python3 -m venv venv
|
|
source venv/bin/activate
|
|
pip install --upgrade pip
|
|
|
|
# Install main requirements
|
|
if [ -f "requirements.txt" ]; then
|
|
pip install -r requirements.txt
|
|
fi
|
|
else
|
|
log "Central virtual environment already exists, activating..."
|
|
source /opt/aitbc/venv/bin/activate
|
|
fi
|
|
|
|
# Install service dependencies in central venv
|
|
log "Installing service dependencies..."
|
|
|
|
# Wallet service dependencies
|
|
log "Installing wallet service dependencies..."
|
|
cd /opt/aitbc/apps/wallet
|
|
if [ -f "requirements.txt" ]; then
|
|
pip install -r requirements.txt
|
|
else
|
|
pip install fastapi uvicorn pydantic httpx python-dotenv websockets
|
|
fi
|
|
|
|
# Coordinator API dependencies
|
|
log "Installing coordinator API dependencies..."
|
|
cd /opt/aitbc/apps/coordinator-api
|
|
if [ -f "requirements.txt" ]; then
|
|
pip install -r requirements.txt
|
|
else
|
|
pip install fastapi uvicorn pydantic httpx python-dotenv
|
|
fi
|
|
|
|
# Exchange API dependencies
|
|
log "Installing exchange API dependencies..."
|
|
cd /opt/aitbc/apps/exchange
|
|
if [ -f "requirements.txt" ]; then
|
|
pip install -r requirements.txt
|
|
else
|
|
pip install fastapi uvicorn pydantic python-multipart
|
|
fi
|
|
|
|
success "Virtual environments setup completed"
|
|
}
|
|
|
|
# Install systemd services
|
|
install_services() {
|
|
log "Installing systemd services..."
|
|
|
|
# Install core services
|
|
services=(
|
|
"aitbc-wallet.service"
|
|
"aitbc-coordinator-api.service"
|
|
"aitbc-exchange-api.service"
|
|
"aitbc-blockchain-rpc.service"
|
|
)
|
|
|
|
for service in "${services[@]}"; do
|
|
if [ -f "/opt/aitbc/systemd/$service" ]; then
|
|
log "Installing $service..."
|
|
ln -sf "/opt/aitbc/systemd/$service" /etc/systemd/system/
|
|
else
|
|
warning "Service file not found: $service"
|
|
fi
|
|
done
|
|
|
|
# Reload systemd
|
|
systemctl daemon-reload
|
|
|
|
success "Systemd services installed"
|
|
}
|
|
|
|
# Create startup script
|
|
create_startup_script() {
|
|
log "Creating startup script..."
|
|
|
|
cat > /opt/aitbc/start-services.sh << 'EOF'
|
|
#!/bin/bash
|
|
|
|
# AITBC Services Startup Script
|
|
|
|
log() {
|
|
echo "[$(date +'%Y-%m-%d %H:%M:%S')] $1"
|
|
}
|
|
|
|
log "Starting AITBC services..."
|
|
|
|
# Start wallet service
|
|
log "Starting wallet service..."
|
|
cd /opt/aitbc/apps/wallet
|
|
source .venv/bin/activate
|
|
nohup python simple_daemon.py > /var/log/aitbc-wallet.log 2>&1 &
|
|
echo $! > /var/run/aitbc-wallet.pid
|
|
|
|
# Start coordinator API
|
|
log "Starting coordinator API..."
|
|
cd /opt/aitbc/apps/coordinator-api/src
|
|
source ../.venv/bin/activate
|
|
nohup python -m uvicorn app.main:app --host 0.0.0.0 --port 8000 > /var/log/aitbc-coordinator.log 2>&1 &
|
|
echo $! > /var/run/aitbc-coordinator.pid
|
|
|
|
# Start exchange API
|
|
log "Starting exchange API..."
|
|
cd /opt/aitbc/apps/exchange
|
|
source .venv/bin/activate
|
|
nohup python simple_exchange_api.py > /var/log/aitbc-exchange.log 2>&1 &
|
|
echo $! > /var/run/aitbc-exchange.pid
|
|
|
|
log "All services started"
|
|
EOF
|
|
|
|
chmod +x /opt/aitbc/start-services.sh
|
|
|
|
success "Startup script created"
|
|
}
|
|
|
|
# Create health check script
|
|
create_health_check() {
|
|
log "Creating health check script..."
|
|
|
|
cat > /opt/aitbc/health-check.sh << 'EOF'
|
|
#!/bin/bash
|
|
|
|
# AITBC Health Check Script
|
|
|
|
RED='\033[0;31m'
|
|
GREEN='\033[0;32m'
|
|
YELLOW='\033[1;33m'
|
|
NC='\033[0m'
|
|
|
|
check_service() {
|
|
local name=$1
|
|
local url=$2
|
|
local expected=${3:-200}
|
|
|
|
if curl -s -o /dev/null -w "%{http_code}" "$url" | grep -q "$expected"; then
|
|
echo -e "${GREEN}✓${NC} $name is healthy"
|
|
return 0
|
|
else
|
|
echo -e "${RED}✗${NC} $name is unhealthy"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
echo "AITBC Service Health Check"
|
|
echo "========================"
|
|
|
|
# Check wallet service
|
|
check_service "Wallet API" "http://localhost:8003/health"
|
|
|
|
# Check exchange API
|
|
check_service "Exchange API" "http://localhost:8001/api/health"
|
|
|
|
# Check coordinator API
|
|
check_service "Coordinator API" "http://localhost:8000/health"
|
|
|
|
# Check process status
|
|
echo ""
|
|
echo "Process Status:"
|
|
ps aux | grep -E "simple_daemon|uvicorn|simple_exchange_api" | grep -v grep | while read line; do
|
|
echo -e "${GREEN}✓${NC} $line"
|
|
done
|
|
EOF
|
|
|
|
chmod +x /opt/aitbc/health-check.sh
|
|
|
|
success "Health check script created"
|
|
}
|
|
|
|
# Start services
|
|
start_services() {
|
|
log "Starting AITBC services..."
|
|
|
|
# Try systemd first
|
|
if systemctl start aitbc-wallet aitbc-coordinator-api aitbc-exchange-api 2>/dev/null; then
|
|
log "Services started via systemd"
|
|
sleep 5
|
|
|
|
# Check if services are running
|
|
if systemctl is-active --quiet aitbc-wallet aitbc-coordinator-api aitbc-exchange-api; then
|
|
success "Services started successfully via systemd"
|
|
else
|
|
warning "Some systemd services failed, falling back to manual startup"
|
|
/opt/aitbc/start-services.sh
|
|
fi
|
|
else
|
|
log "Systemd services not available, using manual startup"
|
|
/opt/aitbc/start-services.sh
|
|
fi
|
|
|
|
# Wait for services to initialize
|
|
sleep 10
|
|
|
|
# Run health check
|
|
/opt/aitbc/health-check.sh
|
|
}
|
|
|
|
# Setup auto-start
|
|
setup_autostart() {
|
|
log "Setting up auto-start..."
|
|
|
|
# Create systemd service for startup script
|
|
cat > /etc/systemd/system/aitbc-startup.service << EOF
|
|
[Unit]
|
|
Description=AITBC Services Startup
|
|
After=network.target
|
|
|
|
[Service]
|
|
Type=oneshot
|
|
ExecStart=/opt/aitbc/start-services.sh
|
|
RemainAfterExit=yes
|
|
User=root
|
|
Group=root
|
|
|
|
[Install]
|
|
WantedBy=multi-user.target
|
|
EOF
|
|
|
|
systemctl enable aitbc-startup.service
|
|
|
|
success "Auto-start configured"
|
|
}
|
|
|
|
# Main function
|
|
main() {
|
|
log "Starting AITBC setup..."
|
|
|
|
check_root
|
|
check_prerequisites
|
|
clone_repo
|
|
setup_runtime_directories
|
|
setup_venvs
|
|
install_services
|
|
create_startup_script
|
|
create_health_check
|
|
start_services
|
|
setup_autostart
|
|
|
|
success "AITBC setup completed!"
|
|
echo ""
|
|
echo "Service Information:"
|
|
echo " Wallet API: http://localhost:8003/health"
|
|
echo " Exchange API: http://localhost:8001/api/health"
|
|
echo " Coordinator API: http://localhost:8000/health"
|
|
echo ""
|
|
echo "Runtime Directories:"
|
|
echo " Keystore: /var/lib/aitbc/keystore/"
|
|
echo " Data: /var/lib/aitbc/data/"
|
|
echo " Logs: /var/lib/aitbc/logs/"
|
|
echo " Config: /etc/aitbc/"
|
|
echo ""
|
|
echo "Management Commands:"
|
|
echo " Health check: /opt/aitbc/health-check.sh"
|
|
echo " Restart services: /opt/aitbc/start-services.sh"
|
|
echo " View logs: tail -f /var/lib/aitbc/logs/aitbc-*.log"
|
|
}
|
|
|
|
# Run main function
|
|
main "$@"
|